Skip to content

Commit 5b4eea4

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 Fix: nits; Catch EXTERN PRAGMA parse error and rethrow warning Remove: malformed extern pragmas no longer signal errors Fix: indentation Add: comment about why we check pragma signatures at parse time Add: Comment about why we're parsing EXTERN pragmas diffrently Fix: nits
1 parent c181970 commit 5b4eea4

36 files changed

+1076
-623
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

+91-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 procedure 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,26 @@ 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+
:type extern
1131+
:documentation "The extern to be called.")
1132+
(arguments
1133+
:initarg :arguments
1134+
:reader call-arguments
1135+
:documentation "A list of memory refs and constant values"))
1136+
(:documentation "A call to an EXTERN function."))
1137+
1138+
(defmethod mnemonic ((call call))
1139+
(declare (ignore call))
1140+
(values "CALL" 'call))
1141+
1142+
(defmethod arguments ((call call))
1143+
(coerce (cons (call-extern call) (call-arguments call)) 'vector))
1144+
11181145
(defclass jump (instruction)
11191146
((label :initarg :label
11201147
:accessor jump-label))
@@ -1316,17 +1343,17 @@ consists of a CONTROLLED-OPERATOR acting on a NAMED-OPERATOR."
13161343
13171344
* Application is a circuit application.
13181345
1319-
* Application is an extern application.
1346+
* Application is a stub application.
13201347
13211348
* Application is an invalid application.
13221349
13231350
Determining this requires the context of the surrounding program."))
13241351

1325-
(defclass extern-application (application)
1352+
(defclass stub-application (application)
13261353
()
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.
1354+
(: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.
13281355
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."))
1356+
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."))
13301357

13311358
(declaim (inline gate-application-p))
13321359
(defun gate-application-p (x)
@@ -1568,15 +1595,17 @@ For example,
15681595
(cond
15691596
((eql 'mref (first expr))
15701597
(format stream "~A[~A]" (second expr) (third expr)))
1571-
((= (length expr) 3)
1598+
((and (= (length expr) 3)
1599+
(lisp-symbol->quil-infix-operator (first expr)))
15721600
(format stream "(~A~A~A)"
15731601
(print-delayed-expression (second expr) nil)
15741602
(lisp-symbol->quil-infix-operator (first expr))
15751603
(print-delayed-expression (third expr) nil)))
1576-
((= (length expr) 2)
1577-
(format stream "~A(~A)"
1604+
(t
1605+
(format stream "~A(~{~A~^,~})"
15781606
(lisp-symbol->quil-function-or-prefix-operator (first expr))
1579-
(print-delayed-expression (second expr) nil)))))
1607+
(loop :for ex :in (rest expr)
1608+
:collect (print-delayed-expression ex nil))))))
15801609
(number
15811610
(format stream "(~/cl-quil:complex-fmt/)" expr))
15821611
(symbol
@@ -1657,9 +1686,17 @@ For example,
16571686
(format stream "MEASURE ~/cl-quil:instruction-fmt/"
16581687
(measurement-qubit instr)))
16591688

1689+
(:method ((instr stub) (stream stream))
1690+
(format stream "STUB ~A" (stub-name instr)))
1691+
16601692
(:method ((instr extern) (stream stream))
16611693
(format stream "EXTERN ~A" (extern-name instr)))
16621694

1695+
(:method ((instr call) (stream stream))
1696+
(format stream "CALL ~A ~{~/cl-quil:instruction-fmt/~^ ~}"
1697+
(extern-name (call-extern instr))
1698+
(call-arguments instr)))
1699+
16631700
(:method ((instr application) (stream stream))
16641701
(print-operator-description (application-operator instr) stream)
16651702
(format stream "~@[(~{~/cl-quil:instruction-fmt/~^, ~})~]~{ ~/cl-quil:instruction-fmt/~}"
@@ -1766,33 +1803,46 @@ For example,
17661803
;;; simply a list of AST objects.
17671804

17681805
(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."))
1806+
((gate-definitions
1807+
:initarg :gate-definitions
1808+
:accessor parsed-program-gate-definitions
1809+
:type list
1810+
:documentation "The gate definitions introduced by DEFGATE.")
1811+
(circuit-definitions
1812+
:initarg :circuit-definitions
1813+
:accessor parsed-program-circuit-definitions
1814+
:type list
1815+
:documentation "The circuit definitions introduced by DEFCIRCUIT.")
1816+
(memory-definitions
1817+
:initarg :memory-definitions
1818+
:accessor parsed-program-memory-definitions
1819+
:type list
1820+
:documentation "The memory definitions introduced by DECLARE.")
1821+
(executable-program
1822+
:initarg :executable-code
1823+
:accessor parsed-program-executable-code
1824+
:type (vector instruction)
1825+
:documentation "A vector of executable Quil instructions.")
1826+
(extern-declarations
1827+
:initarg :extern-declarations
1828+
:accessor parsed-program-extern-declarations
1829+
:type hash-table
1830+
:documentation "A hash table mapping string to booleans.")
1831+
(stub-operations
1832+
:initarg :stub-operations
1833+
:accessor parsed-program-stub-operations
1834+
:type hash-table
1835+
:documentation "A hash table mapping string names to booleans."))
17891836
(:default-initargs
17901837
:gate-definitions '()
17911838
:circuit-definitions '()
17921839
:memory-definitions '()
17931840
: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."))
1841+
:extern-declarations (make-hash-table :test #'equal)
1842+
:stub-operations (make-hash-table :test #'equal))
1843+
(:documentation "A representation of a parsed Quil program, in which instructions have
1844+
been duly sorted into their various categories (e.g. definitions vs
1845+
executable code), and internal references have been resolved."))
17961846

17971847
(defmethod copy-instance ((parsed-program parsed-program))
17981848
(let ((pp (make-instance 'parsed-program)))
@@ -1808,15 +1858,10 @@ For example,
18081858
(setf (parsed-program-executable-code pp)
18091859
(map 'vector #'copy-instance
18101860
(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))
1861+
(setf (parsed-program-stub-operations pp)
1862+
(a:copy-hash-table (parsed-program-stub-operations parsed-program)))
1863+
(setf (parsed-program-extern-declarations pp)
1864+
(a:copy-hash-table (parsed-program-extern-declarations parsed-program)))
18201865
pp))
18211866

18221867
(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)