Skip to content

Commit

Permalink
Cleanups having to do with fidelity of sets of instructions
Browse files Browse the repository at this point in the history
Move defs near to use sites; refactor instrs fidelity

Fidelity calculations used by the compressor and by the fidelity
addresser were needlessly constructing logical schedules.

Peeking at the logical schedule fidelity calculations, they too have
been cleaned up and refatored for clarity and performance.

docstring typo

Float literal woes
  • Loading branch information
macrologist committed Feb 7, 2024
1 parent 967c9ff commit 75ea5ab
Show file tree
Hide file tree
Showing 5 changed files with 85 additions and 74 deletions.
8 changes: 4 additions & 4 deletions app/tests/rpcq-tests.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -54,8 +54,8 @@
"0-1" (make-hash-table))))
(specs (plist-isa-subtable
"1Q" (plist-isa-subtable
"0" (plist-isa-subtable "f1QRB" 0.98)
"1" (plist-isa-subtable "f1QRB" 0.98))
"0" (plist-isa-subtable "f1QRB" 0.98d0)
"1" (plist-isa-subtable "f1QRB" 0.98d0))
"2Q" (plist-isa-subtable
"0-1" (make-hash-table))))
(target-device (make-instance 'rpcq::|TargetDevice|
Expand Down Expand Up @@ -90,8 +90,8 @@
"0-1" (make-hash-table))))
(specs (plist-isa-subtable
"1Q" (plist-isa-subtable
"0" (plist-isa-subtable "f1QRB" 0.98)
"1" (plist-isa-subtable "f1QRB" 0.98))
"0" (plist-isa-subtable "f1QRB" 0.98d0)
"1" (plist-isa-subtable "f1QRB" 0.98d0))
"2Q" (plist-isa-subtable
"0-1" (make-hash-table))))
(target-device (make-instance 'rpcq::|TargetDevice|
Expand Down
10 changes: 7 additions & 3 deletions src/addresser/fidelity-addresser.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,13 @@
(defmethod cost-flatten ((cost fidelity-cost))
(fidelity-cost-value cost))

(defun calculate-instructions-log-fidelity (instructions chip-specification)
"Calculates the fidelity of a sequence of native INSTRUCTIONS on a chip with architecture governed by CHIP-SPECIFICATION (and with assumed perfect parallelization across resources)."
(reduce #'+ instructions
:key (lambda (instr)
(expt (log (get-instruction-fidelity instr chip-specification))
2))))

(defun application-fidelity-cost (state instr)
"Compute the fidelity cost of INSTR, with respect to the provided addresser state."
;; calculate log-infidelity coming from INSTR, using recombination:
Expand All @@ -40,9 +47,6 @@
(when (rewiring-assigned-for-instruction-qubits-p l2p instr)
(rewire-l2p-instruction l2p instr-copy))
(expand-to-native-instructions (list instr-copy) chip-spec))))
;; (when (= 1 (length (application-arguments instr)))
;; (setf instruction-expansion (append instruction-expansion (list (make-instance 'measure-discard :qubit (qubit (apply-rewiring-l2p l2p (qubit-index (first (application-arguments instr))))))))))

;; compute the naive cost
(let ((instr-cost (calculate-instructions-log-fidelity instruction-expansion chip-spec)))
;; then, see if there's a non-naive cost available
Expand Down
93 changes: 34 additions & 59 deletions src/addresser/logical-schedule.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -29,8 +29,8 @@
;;;
;;; where the arrow A ---> B means that A logically follows B.
;;;
;;; The logical scheduler (defined below) holds a set of instructions and their
;;; resource dependencies. In particular, it maintain a list of 'first' or 'top'
;;; The logical schedule (defined below) holds a set of instructions and their
;;; resource dependencies. In particular, it maintains a list of 'first' or 'top'
;;; instructions (corresponding to the right fringe of the above diagram, i.e. X
;;; 0 and H 3), a set of 'last' or 'bottom' instructions (corresponding to the
;;; left fringe, i.e. CNOT 1 3), as well as hash tables storing information on
Expand Down Expand Up @@ -169,22 +169,26 @@
(instruction-resources instr2)))

;;;
;;; the core logical scheduler class and some of its utilities
;;; the core logical schedule class and some of its utilities
;;;

(defclass logical-schedule ()
((first-instrs :initform nil
:accessor lschedule-first-instrs
:documentation "List of the instructions appearing at the \"top\" of a logical scheduler. Excepting COMMUTING_BLOCKS threads, these instructions are guaranteed to occupy disjoint collections of resources.")
(last-instrs :initform nil
:accessor lschedule-last-instrs
:documentation "List of the instructions appearing at the \"bottom\" of a logical scheduler. These are sorted topologically ascending: earlier items in the list come logically after deeper items in the list.")
(later-instrs :initform (make-instr-hash-table)
:accessor lschedule-later-instrs
:documentation "Hash table mapping instruction to a list of instructions after it.")
(earlier-instrs :initform (make-instr-hash-table)
:accessor lschedule-earlier-instrs
:documentation "Hash table mapping instruction to a list of instructions before it."))
((first-instrs
:initform nil
:accessor lschedule-first-instrs
:documentation "List of the instructions appearing at the \"top\" of a logical schedule. Excepting COMMUTING_BLOCKS threads, these instructions are guaranteed to occupy disjoint collections of resources.")
(last-instrs
:initform nil
:accessor lschedule-last-instrs
:documentation "List of the instructions appearing at the \"bottom\" of a logical schedule. These are sorted topologically ascending: earlier items in the list come logically after deeper items in the list.")
(later-instrs
:initform (make-instr-hash-table)
:accessor lschedule-later-instrs
:documentation "Hash table mapping instruction to a list of instructions after it.")
(earlier-instrs
:initform (make-instr-hash-table)
:accessor lschedule-earlier-instrs
:documentation "Hash table mapping instruction to a list of instructions before it."))
(:documentation "Data structure used to track the logical precedence of instructions in a straight-line Quil program."))

(defun make-instr-hash-table (&optional size)
Expand Down Expand Up @@ -658,55 +662,26 @@ Returns the reduction of all bumped values by COMBINE-VALUES, and a hash table m
(+ (length (lschedule-topmost-instructions lschedule))
(hash-table-count (lschedule-earlier-instrs lschedule))))


(defun lschedule-calculate-log-fidelity (lschedule chip-spec)
(labels
((get-fidelity (instr)
(labels ((warn-and-skip (instr)
(format-noise "Unknown fidelity for ~/cl-quil::instruction-fmt/. Skipping." instr)
(return-from get-fidelity 0d0)))
(let (fidelity)
(typecase instr
(measurement
(let* ((qubit-index (qubit-index (measurement-qubit instr)))
(qubit-obj (chip-spec-nth-qubit chip-spec qubit-index))
(specs-obj (gethash (make-measure-binding :qubit qubit-index :target '_)
(hardware-object-gate-information qubit-obj)))
(measure-fidelity (and specs-obj (gate-record-fidelity specs-obj))))
(unless specs-obj
(warn-and-skip instr))
(setf fidelity measure-fidelity)))
(application
(let ((obj (lookup-hardware-object chip-spec instr)))
(unless obj
(warn-and-skip instr))
(let ((specs-hash (hardware-object-gate-information obj)))
(unless specs-hash (warn-and-skip instr))
(when (> (hash-table-count specs-hash) 0)
(let ((binding (binding-from-instr instr)))
(dohash ((key val) specs-hash)
(when (binding-subsumes-p key binding)
(setf fidelity (gate-record-fidelity val))))))
(unless fidelity (warn-and-skip instr)))))
(otherwise
(warn-and-skip instr)))
(expt (log fidelity) 2)))))
"Computes sqrt( ∑ ln(f_i)^2 ) where f_i is a fidelity associated with
an instruction i, i ranging over all instructions in the LSCHEDULE"
(flet ((log-squared (x) (expt (log x) 2)))
(let ((running-fidelity 0d0))
(dolist (instr (lschedule-first-instrs lschedule))
(unless (gethash instr (lschedule-earlier-instrs lschedule))
(incf running-fidelity (get-fidelity instr))))
(maphash (lambda (instr val)
(declare (ignore val))
(incf running-fidelity (get-fidelity instr)))
(lschedule-earlier-instrs lschedule))
(map-lschedule-in-topological-order
lschedule
(lambda (instr)
(incf running-fidelity (log-squared (get-instruction-fidelity instr chip-spec)))))
(sqrt running-fidelity))))

(defun lschedule-calculate-fidelity (lschedule chip-spec)
"Calculate fidelity as the minimum fidelity of the individual instructions.
This relies on the fact that the function $\exp\{-\sqrt{\log(x)^2 + \log(y)^2}\}$ is approximately equal to $\min\{x, y\}$ for $x, y \in (0, 1]$."
(multiple-value-bind (max-value value-hash)
(lschedule-calculate-log-fidelity lschedule chip-spec)
(values (exp (- max-value)) value-hash)))
"Calculate fidelity as the minimum fidelity of the individual instructions."
(let ((min-fidelity 1.0d0))
(map-lschedule-in-topological-order
lschedule
(lambda (instr)
(setf min-fidelity (min min-fidelity (get-instruction-fidelity instr chip-spec)))))
min-fidelity))

(defun lschedule-all-instructions (lschedule)
"Return a list of the instructions of LSCHEDULE."
Expand Down
37 changes: 37 additions & 0 deletions src/chip/chip-specification.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -809,3 +809,40 @@ Compilers are listed in descending precedence.")
(adjoin-hardware-object (build-qubit j :type '(:RZ :X/2 :MEASURE)) chip-spec))
(warm-hardware-objects chip-spec)
chip-spec))

(declaim (inline warn-and-return-perfect-fidelity))
(defun warn-and-return-perfect-fidelity (instr)
(format-noise "Unknown fidelity for ~/cl-quil::instruction-fmt/. Skipping." instr)
1.0d0)

(defun get-instruction-fidelity (instr chip-spec)
"Return the double-float fidelity value associated with object INSTR
on CHIP-SPEC. If no case for on INSTR is matched, the default is
perfect fidelity (i.e. 1.0)."
(declare (values double-float))
(typecase instr
(measurement
(let* ((qubit-index (qubit-index (measurement-qubit instr)))
(qubit-obj (chip-spec-nth-qubit chip-spec qubit-index))
(specs-obj (gethash (make-measure-binding :qubit qubit-index :target '_)
(hardware-object-gate-information qubit-obj))))
(if specs-obj
(gate-record-fidelity specs-obj)
(warn-and-return-perfect-fidelity instr))))

(application
(let (fidelity)
(a:when-let* ((obj (lookup-hardware-object chip-spec instr))
(specs-hash (hardware-object-gate-information obj))
(binding (and (< 0 (hash-table-count specs-hash))
(binding-from-instr instr))))
(dohash ((key val) specs-hash)
(when (binding-subsumes-p key binding)
(setf fidelity (gate-record-fidelity val))
(setf binding key))))
(or fidelity (warn-and-return-perfect-fidelity instr))))

(t
(warn-and-return-perfect-fidelity instr))))


11 changes: 3 additions & 8 deletions src/compressor/compressor.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -87,17 +87,12 @@
;; sift through it for durations
(lschedule-calculate-duration lschedule chip-specification)))

(defun calculate-instructions-log-fidelity (instructions chip-specification)
"Calculates the fidelity of a sequence of native INSTRUCTIONS on a chip with architecture governed by CHIP-SPECIFICATION (and with assumed perfect parallelization across resources)."
(let ((lschedule (make-lschedule)))
;; load up the logical schedule
(append-instructions-to-lschedule lschedule instructions)
;; sift through it for fidelities
(lschedule-calculate-log-fidelity lschedule chip-specification)))

(defun calculate-instructions-fidelity (instructions chip-specification)
"Calculates the fidelity of a sequence of native INSTRUCTIONS on a chip with architecture governed by CHIP-SPECIFICATION (and with assumed perfect parallelization across resources)."
(exp (- (calculate-instructions-log-fidelity instructions chip-specification))))
(reduce #'min instructions
:key (lambda (instr) (get-instruction-fidelity instr chip-specification))
:initial-value 1.0d0))

(defun find-noncommuting-instructions (node)
"Return at most *REWRITING-PEEPHOLE-SIZE* of the earliest instructions below NODE,
Expand Down

0 comments on commit 75ea5ab

Please sign in to comment.