Skip to content

Commit 9deaee6

Browse files
author
O'Keefe, Colin B
committed
Add EXTERN, CALL, and PRAGMA EXTERN; other minior changes
s/EXTERN/STUB - renaming EXTERN keyword to STUB Add zero-qubit frame in a DEFRAME in tests Remove sample-rate from DEFWAVEFORM Added NON_VOLATILE pragma Add EXTERN, numerical expression support, printing Add: CALL instruction Addressing CALL instructions Add PRAMGA EXTERN for declaring extern function signatures Checking extern type for externs in expressions Add tests for extern and extern pragma processing
1 parent c181970 commit 9deaee6

38 files changed

+733
-303
lines changed

cl-quil-tests.asd

+1
Original file line numberDiff line numberDiff line change
@@ -53,4 +53,5 @@
5353
(:file "linear-reversible-circuit-tests")
5454
(:file "permutation-tests")
5555
(:file "sqisw-decomp-tests")
56+
(:file "stub-tests")
5657
(:file "extern-tests")))

src/addresser/logical-schedule.lisp

+9-1
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,13 @@
118118
(address-resources (classical-right-operand inst))
119119
(address-resources (classical-target inst)))))
120120

121+
(defmethod instruction-resources ((inst call))
122+
(loop :with union := (address-resources nil)
123+
:for arg :in (cl-quil.frontend::call-arguments inst)
124+
:when (typep arg 'memory-ref)
125+
:do (setf union (resource-union union (address-resources arg)))
126+
:finally (return union)))
127+
121128
(defmethod instruction-resources ((inst measurement))
122129
(qubit-resources (measurement-qubit inst)))
123130

@@ -152,7 +159,8 @@
152159
(defun local-classical-instruction-p (instr)
153160
(or (typep instr 'unary-classical-instruction)
154161
(typep instr 'binary-classical-instruction)
155-
(typep instr 'trinary-classical-instruction)))
162+
(typep instr 'trinary-classical-instruction)
163+
(typep instr 'call)))
156164

157165
(defun local-classical-quantum-instruction-p (instr)
158166
(or (typep instr 'measure)))

src/addresser/rewiring.lisp

+1-1
Original file line numberDiff line numberDiff line change
@@ -190,7 +190,7 @@ Returns NIL. This mutates the instruction."
190190
:assert-wired t)))
191191
(application-arguments instr))))
192192

193-
(extern-application
193+
(stub-application
194194
(setf (application-arguments instr)
195195
(mapcar (lambda (q) (qubit (apply-rewiring-l2p rewiring (qubit-index q)
196196
:assert-wired t)))

src/analysis/expansion.lisp

+3
Original file line numberDiff line numberDiff line change
@@ -283,6 +283,9 @@ An instruction is unitary if it is of type APPLICATION, whether that be INSTR it
283283
(:method ((instr pragma) param-value arg-value)
284284
instr)
285285

286+
(:method ((instr call) param-value arg-value)
287+
instr)
288+
286289
(:method ((instr unary-classical-instruction) param-value arg-value)
287290
(let ((addr (classical-target instr)))
288291
(if (not (is-formal addr))

src/analysis/resolve-objects.lisp

+4-4
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121
(let* ((operator (application-operator instr))
2222
(addl-qubits (operator-description-additional-qubits operator))
2323
(name (operator-description-root-name operator))
24-
(found-extern (gethash name (parsed-program-extern-operations parsed-program)))
24+
(found-stub (gethash name (parsed-program-stub-operations parsed-program)))
2525
(found-gate-defn (or (find name (parsed-program-gate-definitions parsed-program)
2626
:test #'string=
2727
:key #'gate-definition-name)
@@ -30,9 +30,9 @@
3030
:test #'string=
3131
:key #'circuit-definition-name)))
3232
(cond
33-
;; externs take priority over defined operators
34-
(found-extern
35-
(change-class instr 'extern-application))
33+
;; stubs take priority over defined operators
34+
(found-stub
35+
(change-class instr 'stub-application))
3636

3737
;; Gate application
3838
(found-gate-defn

src/analysis/type-safety.lisp

+6
Original file line numberDiff line numberDiff line change
@@ -233,6 +233,12 @@
233233
(when (typep right-mref 'memory-ref)
234234
(enforce-mref-bounds right-mref (find-descriptor-for-mref right-mref memory-regions)))))
235235

236+
(:method ((instr call) memory-regions)
237+
(loop :for arg :in (call-arguments instr)
238+
:when (typep arg 'memory-ref)
239+
:do (enforce-mref-bounds arg (find-descriptor-for-mref arg memory-regions))))
240+
241+
236242
;; NEG needs to be INT or REAL
237243
(:method ((instr classical-negate) memory-regions)
238244
(check-mref (classical-target instr))

src/ast.lisp

+87-46
Original file line numberDiff line numberDiff line change
@@ -421,20 +421,27 @@ If no exit rewiring is found, return NIL."
421421
(:documentation "A directive to include another file in a Quil file."))
422422

423423

424-
(defclass extern ()
425-
((name :reader extern-name
424+
(defclass stub ()
425+
((name :reader stub-name
426426
:initarg :name
427-
:documentation "The name of the operation being marked as an EXTERN"))
428-
(:documentation "A directive to mark a particular operation as an extern. I.e. an
429-
operation that does not have a definition. Names marked as EXTERN can
427+
:documentation "The name of the operation being marked as a stub"))
428+
(:documentation "A directive to mark a particular operation as a stub. I.e. an
429+
operation that does not have a definition. Names marked as STUB can
430430
be parsed as they appear, and are protected from the optimizing
431431
compiler, similar to the effect of a PRESERVE_BLOCK pragma.
432432
433-
NB: A name marked as an EXTERN will take priority over all other
433+
NB: A name marked as a stub will take priority over all other
434434
names. Meaning if, for example, a DEFCIRCUIT is defined with name
435-
marked as EXTERN, that circuit will be totally ignored by
435+
marked as STUB, that circuit will be totally ignored by
436436
compilation passes."))
437437

438+
(defclass extern ()
439+
((name :reader extern-name
440+
:initarg :name
441+
:documentation "The name of the function."))
442+
(:documentation "A function that operates on classical memory and values, declared to
443+
be available in the underlying system."))
444+
438445
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; Definitions ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
439446

440447
;;; Gate Definitions
@@ -1115,6 +1122,22 @@ Each addressing mode will be a vector of symbols:
11151122
(bit real real)
11161123
(bit real immediate))
11171124

1125+
1126+
(defclass call (classical-instruction)
1127+
((extern
1128+
:initarg :extern
1129+
:reader call-extern)
1130+
(arguments
1131+
:initarg :arguments
1132+
:reader call-arguments)))
1133+
1134+
(defmethod mnemonic ((call call))
1135+
(declare (ignore call))
1136+
(values "CALL" 'call))
1137+
1138+
(defmethod arguments ((call call))
1139+
(map 'vector #'identity (cons (call-extern call) (call-arguments call))))
1140+
11181141
(defclass jump (instruction)
11191142
((label :initarg :label
11201143
:accessor jump-label))
@@ -1316,17 +1339,17 @@ consists of a CONTROLLED-OPERATOR acting on a NAMED-OPERATOR."
13161339
13171340
* Application is a circuit application.
13181341
1319-
* Application is an extern application.
1342+
* Application is a stub application.
13201343
13211344
* Application is an invalid application.
13221345
13231346
Determining this requires the context of the surrounding program."))
13241347

1325-
(defclass extern-application (application)
1348+
(defclass stub-application (application)
13261349
()
1327-
(:documentation "Represents the application of an extern operation. Externs allow the user to bypass the parsing and compilation stages for particular operations that are meant to receive specific definition at the backend compilation stage.
1350+
(:documentation "Represents the application of a stub operation. Stubs allow the user to bypass the parsing and compilation stages for particular operations that are meant to receive specific definition at the backend compilation stage.
13281351
1329-
Externs are similar to instances of UNRESOLVED-APPLICATION. They are semantically empty from the perspective of the quantum abstract virtual machine, and cannot be simulated or executed."))
1352+
Stubs are similar to instances of UNRESOLVED-APPLICATION. They are semantically empty from the perspective of the quantum abstract virtual machine, and cannot be simulated or executed."))
13301353

13311354
(declaim (inline gate-application-p))
13321355
(defun gate-application-p (x)
@@ -1568,15 +1591,17 @@ For example,
15681591
(cond
15691592
((eql 'mref (first expr))
15701593
(format stream "~A[~A]" (second expr) (third expr)))
1571-
((= (length expr) 3)
1594+
((and (= (length expr) 3)
1595+
(lisp-symbol->quil-infix-operator (first expr)))
15721596
(format stream "(~A~A~A)"
15731597
(print-delayed-expression (second expr) nil)
15741598
(lisp-symbol->quil-infix-operator (first expr))
15751599
(print-delayed-expression (third expr) nil)))
1576-
((= (length expr) 2)
1577-
(format stream "~A(~A)"
1600+
(t
1601+
(format stream "~A(~{~A~^,~})"
15781602
(lisp-symbol->quil-function-or-prefix-operator (first expr))
1579-
(print-delayed-expression (second expr) nil)))))
1603+
(loop :for ex :in (rest expr)
1604+
:collect (print-delayed-expression ex nil))))))
15801605
(number
15811606
(format stream "(~/cl-quil:complex-fmt/)" expr))
15821607
(symbol
@@ -1657,9 +1682,17 @@ For example,
16571682
(format stream "MEASURE ~/cl-quil:instruction-fmt/"
16581683
(measurement-qubit instr)))
16591684

1685+
(:method ((instr stub) (stream stream))
1686+
(format stream "STUB ~A" (stub-name instr)))
1687+
16601688
(:method ((instr extern) (stream stream))
16611689
(format stream "EXTERN ~A" (extern-name instr)))
16621690

1691+
(:method ((instr call) (stream stream))
1692+
(format stream "CALL ~A ~{~/cl-quil:instruction-fmt/~^ ~}"
1693+
(extern-name (call-extern instr))
1694+
(call-arguments instr)))
1695+
16631696
(:method ((instr application) (stream stream))
16641697
(print-operator-description (application-operator instr) stream)
16651698
(format stream "~@[(~{~/cl-quil:instruction-fmt/~^, ~})~]~{ ~/cl-quil:instruction-fmt/~}"
@@ -1766,33 +1799,46 @@ For example,
17661799
;;; simply a list of AST objects.
17671800

17681801
(defclass parsed-program (transformable)
1769-
((gate-definitions :initarg :gate-definitions
1770-
:accessor parsed-program-gate-definitions
1771-
:type list
1772-
:documentation "The gate definitions introduced by DEFGATE.")
1773-
(circuit-definitions :initarg :circuit-definitions
1774-
:accessor parsed-program-circuit-definitions
1775-
:type list
1776-
:documentation "The circuit definitions introduced by DEFCIRCUIT.")
1777-
(memory-definitions :initarg :memory-definitions
1778-
:accessor parsed-program-memory-definitions
1779-
:type list
1780-
:documentation "The memory definitions introduced by DECLARE.")
1781-
(executable-program :initarg :executable-code
1782-
:accessor parsed-program-executable-code
1783-
:type (vector instruction)
1784-
:documentation "A vector of executable Quil instructions.")
1785-
(extern-operations :initarg :extern-operations
1786-
:accessor parsed-program-extern-operations
1787-
:type hash-table
1788-
:documentation "A hash table mapping string NAMEs to generalized booleans, indicating that an operation so named is an extern."))
1802+
((gate-definitions
1803+
:initarg :gate-definitions
1804+
:accessor parsed-program-gate-definitions
1805+
:type list
1806+
:documentation "The gate definitions introduced by DEFGATE.")
1807+
(circuit-definitions
1808+
:initarg :circuit-definitions
1809+
:accessor parsed-program-circuit-definitions
1810+
:type list
1811+
:documentation "The circuit definitions introduced by DEFCIRCUIT.")
1812+
(memory-definitions
1813+
:initarg :memory-definitions
1814+
:accessor parsed-program-memory-definitions
1815+
:type list
1816+
:documentation "The memory definitions introduced by DECLARE.")
1817+
(executable-program
1818+
:initarg :executable-code
1819+
:accessor parsed-program-executable-code
1820+
:type (vector instruction)
1821+
:documentation "A vector of executable Quil instructions.")
1822+
(extern-declarations
1823+
:initarg :extern-declarations
1824+
:accessor parsed-program-extern-declarations
1825+
:type hash-table
1826+
:documentation "A hash table mapping string to booleans.")
1827+
(stub-operations
1828+
:initarg :stub-operations
1829+
:accessor parsed-program-stub-operations
1830+
:type hash-table
1831+
:documentation "A hash table mapping string names to booleans."))
17891832
(:default-initargs
17901833
:gate-definitions '()
17911834
:circuit-definitions '()
17921835
:memory-definitions '()
17931836
:executable-code #()
1794-
:extern-operations (make-hash-table :test #'equal))
1795-
(:documentation "A representation of a parsed Quil program, in which instructions have been duly sorted into their various categories (e.g. definitions vs executable code), and internal references have been resolved."))
1837+
:extern-declarations (make-hash-table :test #'equal)
1838+
:stub-operations (make-hash-table :test #'equal))
1839+
(:documentation "A representation of a parsed Quil program, in which instructions have
1840+
been duly sorted into their various categories (e.g. definitions vs
1841+
executable code), and internal references have been resolved."))
17961842

17971843
(defmethod copy-instance ((parsed-program parsed-program))
17981844
(let ((pp (make-instance 'parsed-program)))
@@ -1808,15 +1854,10 @@ For example,
18081854
(setf (parsed-program-executable-code pp)
18091855
(map 'vector #'copy-instance
18101856
(parsed-program-executable-code parsed-program)))
1811-
(setf (parsed-program-extern-operations pp)
1812-
(let ((new-table
1813-
(make-hash-table :test #'equal))
1814-
(old-table
1815-
(parsed-program-extern-operations parsed-program)))
1816-
(loop :for key :being :the :hash-key :of old-table
1817-
:using (:hash-value value)
1818-
:do (setf (gethash key new-table) value))
1819-
new-table))
1857+
(setf (parsed-program-stub-operations pp)
1858+
(a:copy-hash-table (parsed-program-stub-operations parsed-program)))
1859+
(setf (parsed-program-extern-declarations pp)
1860+
(a:copy-hash-table (parsed-program-extern-declarations parsed-program)))
18201861
pp))
18211862

18221863
(defvar *print-parsed-program-text* nil

src/cfg.lisp

+9-9
Original file line numberDiff line numberDiff line change
@@ -175,19 +175,19 @@ Return the following values:
175175
(vector-push-extend instr (basic-block-code blk))
176176
(values blk nil nil))
177177

178-
(defmethod process-instruction (cfg blk (instr extern-application))
178+
(defmethod process-instruction (cfg blk (instr stub-application))
179179
(assert (not (null blk)) (blk))
180-
;; extern applications should be preserved from nativization and optimization
181-
(let ((extern-blk
182-
(find-or-make-block-from-label cfg (label (princ-to-string (gensym "EXTERN-")))))
180+
;; stub applications should be preserved from nativization and optimization
181+
(let ((stub-blk
182+
(find-or-make-block-from-label cfg (label (princ-to-string (gensym "STUB-")))))
183183
(new-blk
184184
(make-instance 'basic-block)))
185-
(change-class extern-blk 'preserved-block)
186-
(vector-push-extend instr (basic-block-code extern-blk))
187-
(link-blocks blk (unconditional-edge extern-blk))
188-
(link-blocks extern-blk (unconditional-edge new-blk))
185+
(change-class stub-blk 'preserved-block)
186+
(vector-push-extend instr (basic-block-code stub-blk))
187+
(link-blocks blk (unconditional-edge stub-blk))
188+
(link-blocks stub-blk (unconditional-edge new-blk))
189189
;; we finish with the old block, return a new empty block, and the
190-
;; extern is isolated in a preserved block linked between these
190+
;; stub is isolated in a preserved block linked between these
191191
;; two.
192192
(values new-blk blk nil)))
193193

src/cl-quil.lisp

+8-2
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,7 @@ This also signals ambiguous definitions, which may be handled as needed."
8989
(circ-defs '())
9090
(memory-defs '())
9191
(exec-code '())
92+
(stubs (make-hash-table :test #'equal))
9293
(externs (make-hash-table :test #'equal))
9394
;; The following maps definition signatures to a list of (filename . defn) pairs
9495
(all-seen-defns (make-hash-table :test 'equalp)))
@@ -112,15 +113,18 @@ This also signals ambiguous definitions, which may be handled as needed."
112113
(gate-definition (push instr gate-defs))
113114
(circuit-definition (push instr circ-defs))
114115
(memory-descriptor (push instr memory-defs))
115-
(extern
116+
(stub
117+
(setf (gethash (stub-name instr) stubs) t))
118+
(extern
116119
(setf (gethash (extern-name instr) externs) t))
117120
(t (push instr exec-code)))))
118121
(mapc #'bin code)
119122
(make-instance 'parsed-program
120123
:gate-definitions (nreverse gate-defs)
121124
:circuit-definitions (nreverse circ-defs)
122125
:memory-definitions (nreverse memory-defs)
123-
:extern-operations externs
126+
:stub-operations stubs
127+
:extern-declarations externs
124128
:executable-code (coerce (nreverse exec-code)
125129
'simple-vector)))))
126130

@@ -219,6 +223,8 @@ In the presence of multiple definitions with a common signature, a signal is rai
219223
"Parse a string STRING into a list of raw Quil syntax objects."
220224
(check-type string string)
221225
(let* ((*memory-region-names* nil)
226+
(*names-declared-extern* +builtin-externs+)
227+
(*expression-externs* +builtin-externs+)
222228
(tok-lines (tokenize string)))
223229
(loop :with parsed-program := nil
224230
:until (endp tok-lines) :do

src/compiler-hook.lisp

+8-3
Original file line numberDiff line numberDiff line change
@@ -324,11 +324,16 @@ Returns a value list: (processed-program, of type parsed-program
324324
(parsed-program-circuit-definitions parsed-program))
325325
(setf (parsed-program-memory-definitions processed-program)
326326
(parsed-program-memory-definitions parsed-program))
327-
;; retain the old extern operations table
328-
(setf (parsed-program-extern-operations processed-program)
329-
(parsed-program-extern-operations parsed-program))
327+
;; retain the old stub operations table
328+
(setf (parsed-program-stub-operations processed-program)
329+
(parsed-program-stub-operations parsed-program))
330330
;; ... and output the results.
331+
(setf (cl-quil.frontend::parsed-program-extern-declarations processed-program)
332+
(cl-quil.frontend::parsed-program-extern-declarations parsed-program))
333+
331334
(values
332335
processed-program
333336
topological-swaps
334337
unpreserved-duration))))))
338+
339+

0 commit comments

Comments
 (0)