Repository URL to install this package:
|
Version:
0.24.2 ▾
|
# This code is part of Qiskit.
#
# (C) Copyright IBM 2021.
#
# This code is licensed under the Apache License, Version 2.0. You may
# obtain a copy of this license in the LICENSE.txt file in the root directory
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
#
# Any modifications or derivative works of this code must retain this
# copyright notice, and modified files need to carry a notice indicating
# that they have been altered from the originals.
"""QASM3 Exporter"""
import collections
import re
import io
import itertools
import numbers
from os.path import dirname, join, abspath
from typing import Iterable, List, Sequence, Union, Generator
from qiskit.circuit import (
Barrier,
CircuitInstruction,
Clbit,
Gate,
Instruction,
Measure,
Parameter,
ParameterExpression,
QuantumCircuit,
QuantumRegister,
Qubit,
Reset,
Delay,
)
from qiskit.circuit.bit import Bit
from qiskit.circuit.controlflow import (
IfElseOp,
ForLoopOp,
WhileLoopOp,
SwitchCaseOp,
ControlFlowOp,
BreakLoopOp,
ContinueLoopOp,
CASE_DEFAULT,
)
from qiskit.circuit.library import standard_gates
from qiskit.circuit.register import Register
from qiskit.circuit.tools import pi_check
from . import ast
from .experimental import ExperimentalFeatures
from .exceptions import QASM3ExporterError
from .printer import BasicPrinter
# Reserved keywords that gates and variables cannot be named. It is possible that some of these
# _could_ be accepted as variable names by OpenQASM 3 parsers, but it's safer for us to just be very
# conservative.
_RESERVED_KEYWORDS = frozenset(
{
"OPENQASM",
"U",
"angle",
"array",
"barrier",
"bit",
"bool",
"box",
"break",
"cal",
"complex",
"const",
"continue",
"creg",
"ctrl",
"def",
"defcal",
"defcalgrammar",
"delay",
"duration",
"durationof",
"else",
"end",
"extern",
"float",
"for",
"gate",
"gphase",
"if",
"in",
"include",
"input",
"int",
"inv",
"let",
"measure",
"mutable",
"negctrl",
"output",
"pow",
"qreg",
"qubit",
"reset",
"return",
"sizeof",
"stretch",
"uint",
"while",
}
)
# This probably isn't precisely the same as the OQ3 spec, but we'd need an extra dependency to fully
# handle all Unicode character classes, and this should be close enough for users who aren't
# actively _trying_ to break us (fingers crossed).
_VALID_IDENTIFIER = re.compile(r"[\w][\w\d]*", flags=re.U)
def _escape_invalid_identifier(name: str) -> str:
if name in _RESERVED_KEYWORDS or not _VALID_IDENTIFIER.fullmatch(name):
name = "_" + re.sub(r"[^\w\d]", "_", name)
return name
class Exporter:
"""QASM3 exporter main class."""
def __init__(
self,
includes: Sequence[str] = ("stdgates.inc",),
basis_gates: Sequence[str] = ("U",),
disable_constants: bool = False,
alias_classical_registers: bool = False,
indent: str = " ",
experimental: ExperimentalFeatures = ExperimentalFeatures(0),
):
"""
Args:
includes: the filenames that should be emitted as includes. These files will be parsed
for gates, and any objects dumped from this exporter will use those definitions
where possible.
basis_gates: the basic defined gate set of the backend.
disable_constants: if ``True``, always emit floating-point constants for numeric
parameter values. If ``False`` (the default), then values close to multiples of
QASM 3 constants (``pi``, ``euler``, and ``tau``) will be emitted in terms of those
constants instead, potentially improving accuracy in the output.
alias_classical_registers: If ``True``, then classical bit and classical register
declarations will look similar to quantum declarations, where the whole set of bits
will be declared in a flat array, and the registers will just be aliases to
collections of these bits. This is inefficient for running OpenQASM 3 programs,
however, and may not be well supported on backends. Instead, the default behaviour
of ``False`` means that individual classical registers will gain their own
``bit[size] register;`` declarations, and loose :obj:`.Clbit`\\ s will go onto their
own declaration. In this form, each :obj:`.Clbit` must be in either zero or one
:obj:`.ClassicalRegister`\\ s.
indent: the indentation string to use for each level within an indented block. Can be
set to the empty string to disable indentation.
experimental: any experimental features to enable during the export. See
:class:`ExperimentalFeatures` for more details.
"""
self.basis_gates = basis_gates
self.disable_constants = disable_constants
self.alias_classical_registers = alias_classical_registers
self.includes = list(includes)
self.indent = indent
self.experimental = experimental
def dumps(self, circuit):
"""Convert the circuit to QASM 3, returning the result as a string."""
with io.StringIO() as stream:
self.dump(circuit, stream)
return stream.getvalue()
def dump(self, circuit, stream):
"""Convert the circuit to QASM 3, dumping the result to a file or text stream."""
builder = QASM3Builder(
circuit,
includeslist=self.includes,
basis_gates=self.basis_gates,
disable_constants=self.disable_constants,
alias_classical_registers=self.alias_classical_registers,
experimental=self.experimental,
)
BasicPrinter(stream, indent=self.indent, experimental=self.experimental).visit(
builder.build_program()
)
class GlobalNamespace:
"""Global namespace dict-like."""
BASIS_GATE = object()
qiskit_gates = {
"p": standard_gates.PhaseGate,
"x": standard_gates.XGate,
"y": standard_gates.YGate,
"z": standard_gates.ZGate,
"h": standard_gates.HGate,
"s": standard_gates.SGate,
"sdg": standard_gates.SdgGate,
"t": standard_gates.TGate,
"tdg": standard_gates.TdgGate,
"sx": standard_gates.SXGate,
"rx": standard_gates.RXGate,
"ry": standard_gates.RYGate,
"rz": standard_gates.RZGate,
"cx": standard_gates.CXGate,
"cy": standard_gates.CYGate,
"cz": standard_gates.CZGate,
"cp": standard_gates.CPhaseGate,
"crx": standard_gates.CRXGate,
"cry": standard_gates.CRYGate,
"crz": standard_gates.CRZGate,
"ch": standard_gates.CHGate,
"swap": standard_gates.SwapGate,
"ccx": standard_gates.CCXGate,
"cswap": standard_gates.CSwapGate,
"cu": standard_gates.CUGate,
"CX": standard_gates.CXGate,
"phase": standard_gates.PhaseGate,
"cphase": standard_gates.CPhaseGate,
"id": standard_gates.IGate,
"u1": standard_gates.U1Gate,
"u2": standard_gates.U2Gate,
"u3": standard_gates.U3Gate,
}
include_paths = [abspath(join(dirname(__file__), "..", "qasm", "libs"))]
def __init__(self, includelist, basis_gates=()):
self._data = {gate: self.BASIS_GATE for gate in basis_gates}
for includefile in includelist:
if includefile == "stdgates.inc":
self._data.update(self.qiskit_gates)
else:
# TODO What do if an inc file is not standard?
# Should it be parsed?
pass
def __setitem__(self, name_str, instruction):
self._data[name_str] = type(instruction)
self._data[id(instruction)] = name_str
def __getitem__(self, key):
if isinstance(key, Instruction):
try:
# Registered gates.
return self._data[id(key)]
except KeyError:
pass
# Built-in gates.
if key.name not in self._data:
raise KeyError(key)
return key.name
return self._data[key]
def __iter__(self):
return iter(self._data)
def __contains__(self, instruction):
if isinstance(instruction, standard_gates.UGate):
return True
if id(instruction) in self._data:
return True
if self._data.get(instruction.name) is self.BASIS_GATE:
return True
if type(instruction) in [Gate, Instruction]: # user-defined instructions/gate
return self._data.get(instruction.name, None) == instruction
type_ = self._data.get(instruction.name)
if isinstance(type_, type) and isinstance(instruction, type_):
return True
return False
def register(self, instruction):
"""Register an instruction in the namespace"""
# The second part of the condition is a nasty hack to ensure that gates that come with at
# least one parameter always have their id in the name. This is a workaround a bug, where
# gates with parameters do not contain the information required to build the gate definition
# in symbolic form (unless the parameters are all symbolic). The exporter currently
# (2021-12-01) builds gate declarations with parameters in the signature, but then ignores
# those parameters during the body, and just uses the concrete values from the first
# instance of the gate it sees, such as:
# gate rzx(_gate_p_0) _gate_q_0, _gate_q_1 {
# h _gate_q_1;
# cx _gate_q_0, _gate_q_1;
# rz(0.2) _gate_q_1; // <- note the concrete value.
# cx _gate_q_0, _gate_q_1;
# h _gate_q_1;
# }
# This then means that multiple calls to the same gate with different parameters will be
# incorrect. By forcing all gates to be defined including their id, we generate a QASM3
# program that does what was intended, even though the output QASM3 is silly. See gh-7335.
if instruction.name in self._data or (
isinstance(instruction, Gate)
and not all(isinstance(param, Parameter) for param in instruction.params)
):
key = f"{instruction.name}_{id(instruction)}"
else:
key = instruction.name
self[key] = instruction
# A _Scope is the structure used in the builder to store the contexts and re-mappings of bits from
# the top-level scope where the bits were actually defined. In the class, 'circuit' is an instance
# of QuantumCircuit that defines this level, and 'bit_map' is a mapping of 'Bit: Bit', where the
# keys are bits in the circuit in this scope, and the values are the Bit in the top-level scope in
# this context that this bit actually represents. 'symbol_map' is a bidirectional mapping of
# '<Terra object>: Identifier' and 'str: <Terra object>', where the string in the second map is the
# name of the identifier. This is a cheap hack around actually implementing a proper symbol table.
_Scope = collections.namedtuple("_Scope", ("circuit", "bit_map", "symbol_map"))
class QASM3Builder:
"""QASM3 builder constructs an AST from a QuantumCircuit."""
builtins = (Barrier, Measure, Reset, Delay, BreakLoopOp, ContinueLoopOp)
gate_parameter_prefix = "_gate_p"
gate_qubit_prefix = "_gate_q"
def __init__(
self,
quantumcircuit,
includeslist,
basis_gates,
disable_constants,
alias_classical_registers,
experimental=ExperimentalFeatures(0),
):
# This is a stack of stacks; the outer stack is a list of "outer" look-up contexts, and the
# inner stack is for scopes within these. A "outer" look-up context in this sense means
# the main program body or a gate/subroutine definition, whereas the scopes are for things
# like the body of a ``for`` loop construct.
self._circuit_ctx = []
self.push_context(quantumcircuit)
self.includeslist = includeslist
# `_global_io_declarations` and `_global_classical_declarations` are stateful, and any
# operation that needs a parameter can append to them during the build. We make all
# classical declarations global because the IBM QSS stack (our initial consumer of OQ3
# strings) prefers declarations to all be global, and it's valid OQ3, so it's not vendor
# lock-in. It's possibly slightly memory inefficient, but that's not likely to be a problem
# in the near term.
self._global_io_declarations = []
self._global_classical_declarations = []
self._gate_to_declare = {}
self._subroutine_to_declare = {}
self._opaque_to_declare = {}
self._flat_reg = False
self._physical_qubit = False
self._loose_clbit_index_lookup = {}
# An arbitrary counter to help with generation of unique ids for symbol names when there are
# clashes (though we generally prefer to keep user names if possible).
self._counter = itertools.count()
self.disable_constants = disable_constants
self.alias_classical_registers = alias_classical_registers
self.global_namespace = GlobalNamespace(includeslist, basis_gates)
self.experimental = experimental
def _unique_name(self, prefix: str, scope: _Scope) -> str:
table = scope.symbol_map
name = basename = _escape_invalid_identifier(prefix)
while name in table or name in _RESERVED_KEYWORDS:
name = f"{basename}__generated{next(self._counter)}"
return name
def _register_gate(self, gate):
self.global_namespace.register(gate)
self._gate_to_declare[id(gate)] = gate
def _register_subroutine(self, instruction):
self.global_namespace.register(instruction)
self._subroutine_to_declare[id(instruction)] = instruction
def _register_opaque(self, instruction):
if instruction not in self.global_namespace:
self.global_namespace.register(instruction)
self._opaque_to_declare[id(instruction)] = instruction
def _register_variable(self, variable, scope: _Scope, name=None) -> ast.Identifier:
"""Register a variable in the symbol table for the given scope, returning the name that
should be used to refer to the variable. The same name will be returned by subsequent calls
to :meth:`_lookup_variable` within the same scope.
If ``name`` is given explicitly, it must not already be defined in the scope.
"""
# Note that the registration only checks for the existence of a variable that was declared
# in the current scope, not just one that's available. This is a rough implementation of
# the shadowing proposal currently being drafted for OpenQASM 3, though we expect it to be
# expanded and modified in the future (2022-03-07).
table = scope.symbol_map
if name is not None:
if name in _RESERVED_KEYWORDS:
raise QASM3ExporterError(f"cannot reserve the keyword '{name}' as a variable name")
if name in table:
raise QASM3ExporterError(
f"tried to reserve '{name}', but it is already used by '{table[name]}'"
)
else:
name = self._unique_name(variable.name, scope)
identifier = ast.Identifier(name)
table[identifier.string] = variable
table[variable] = identifier
return identifier
def _reserve_variable_name(self, name: ast.Identifier, scope: _Scope) -> ast.Identifier:
"""Reserve a variable name in the given scope, raising a :class:`.QASM3ExporterError` if
the name is already in use.
This is useful for autogenerated names that the exporter itself reserves when dealing with
objects that have no standard Terra object backing them, such as the declaration of all
circuit qubits, so cannot be placed into the symbol table by the normal means.
Returns the same identifier, for convenience in chaining."""
table = scope.symbol_map
if name.string in table:
variable = table[name.string]
raise QASM3ExporterError(
f"tried to reserve '{name.string}', but it is already used by '{variable}'"
)
table[name.string] = "<internal object>"
return name
def _lookup_variable(self, variable) -> ast.Identifier:
"""Lookup a Terra object within the current context, and return the name that should be used
to represent it in OpenQASM 3 programmes."""
for scope in reversed(self.current_context()):
if variable in scope.symbol_map:
return scope.symbol_map[variable]
raise KeyError(f"'{variable}' is not defined in the current context")
def build_header(self):
"""Builds a Header"""
version = ast.Version("3")
includes = self.build_includes()
return ast.Header(version, includes)
def build_program(self):
"""Builds a Program"""
self.hoist_declarations(self.global_scope(assert_=True).circuit.data)
return ast.Program(self.build_header(), self.build_global_statements())
def hoist_declarations(self, instructions):
"""Walks the definitions in gates/instructions to make a list of gates to declare."""
for instruction in instructions:
if isinstance(instruction.operation, ControlFlowOp):
for block in instruction.operation.blocks:
self.hoist_declarations(block.data)
continue
if instruction.operation in self.global_namespace or isinstance(
instruction.operation, self.builtins
):
continue
if instruction.operation.definition is None:
self._register_opaque(instruction.operation)
else:
self.hoist_declarations(instruction.operation.definition.data)
if isinstance(instruction.operation, Gate):
self._register_gate(instruction.operation)
else:
self._register_subroutine(instruction.operation)
def global_scope(self, assert_=False):
"""Return the global circuit scope that is used as the basis of the full program. If
``assert_=True``, then this raises :obj:`.QASM3ExporterError` if the current context is not
the global one."""
if assert_ and len(self._circuit_ctx) != 1 and len(self._circuit_ctx[0]) != 1:
# Defensive code to help catch logic errors.
raise QASM3ExporterError( # pragma: no cover
f"Not currently in the global context. Current contexts are: {self._circuit_ctx}"
)
return self._circuit_ctx[0][0]
def current_outermost_scope(self):
"""Return the outermost scope for this context. If building the main program, then this is
the :obj:`.QuantumCircuit` instance that the full program is being built from. If building
a gate or subroutine definition, this is the body that defines the gate or subroutine."""
return self._circuit_ctx[-1][0]
def current_scope(self):
"""Return the current circuit scope."""
return self._circuit_ctx[-1][-1]
def current_context(self):
"""Return the current context (list of scopes)."""
return self._circuit_ctx[-1]
def push_scope(self, circuit: QuantumCircuit, qubits: Iterable[Qubit], clbits: Iterable[Clbit]):
"""Push a new scope (like a ``for`` or ``while`` loop body) onto the current context
stack."""
current_map = self.current_scope().bit_map
qubits = tuple(current_map[qubit] for qubit in qubits)
clbits = tuple(current_map[clbit] for clbit in clbits)
if circuit.num_qubits != len(qubits):
raise QASM3ExporterError( # pragma: no cover
f"Tried to push a scope whose circuit needs {circuit.num_qubits} qubits, but only"
f" provided {len(qubits)} qubits to create the mapping."
)
if circuit.num_clbits != len(clbits):
raise QASM3ExporterError( # pragma: no cover
f"Tried to push a scope whose circuit needs {circuit.num_clbits} clbits, but only"
f" provided {len(clbits)} clbits to create the mapping."
)
mapping = dict(itertools.chain(zip(circuit.qubits, qubits), zip(circuit.clbits, clbits)))
self._circuit_ctx[-1].append(_Scope(circuit, mapping, {}))
def pop_scope(self) -> _Scope:
"""Pop the current scope (like a ``for`` or ``while`` loop body) off the current context
stack."""
if len(self._circuit_ctx[-1]) <= 1:
raise QASM3ExporterError( # pragma: no cover
"Tried to pop a scope from the current context, but there are no current scopes."
)
return self._circuit_ctx[-1].pop()
def push_context(self, outer_context: QuantumCircuit):
"""Push a new context (like for a ``gate`` or ``def`` body) onto the stack."""
mapping = {bit: bit for bit in itertools.chain(outer_context.qubits, outer_context.clbits)}
self._circuit_ctx.append([_Scope(outer_context, mapping, {})])
def pop_context(self):
"""Pop the current context (like for a ``gate`` or ``def`` body) onto the stack."""
if len(self._circuit_ctx) == 1:
raise QASM3ExporterError( # pragma: no cover
"Tried to pop the current context, but that is the global context."
)
if len(self._circuit_ctx[-1]) != 1:
raise QASM3ExporterError( # pragma: no cover
"Tried to pop the current context while there are still"
f" {len(self._circuit_ctx[-1]) - 1} unclosed scopes."
)
self._circuit_ctx.pop()
def build_includes(self):
"""Builds a list of included files."""
return [ast.Include(filename) for filename in self.includeslist]
def build_global_statements(self) -> List[ast.Statement]:
"""
globalStatement
: subroutineDefinition
| kernelDeclaration
| quantumGateDefinition
| calibration
| quantumDeclarationStatement # build_quantumdeclaration
| pragma
;
statement
: expressionStatement
| assignmentStatement
| classicalDeclarationStatement
| branchingStatement
| loopStatement
| endStatement
| aliasStatement
| quantumStatement # build_quantuminstruction
;
"""
definitions = self.build_definitions()
# These two "declarations" functions populate stateful variables, since the calls to
# `build_quantum_instructions` might also append to those declarations.
self.build_parameter_declarations()
self.build_classical_declarations()
context = self.global_scope(assert_=True).circuit
if getattr(context, "_layout", None) is not None:
self._physical_qubit = True
quantum_declarations = []
else:
quantum_declarations = self.build_quantum_declarations()
quantum_instructions = self.build_quantum_instructions(context.data)
self._physical_qubit = False
return [
statement
for source in (
# In older versions of the reference OQ3 grammar, IO declarations had to come before
# anything else, so we keep doing that as a courtesy.
self._global_io_declarations,
definitions,
self._global_classical_declarations,
quantum_declarations,
quantum_instructions,
)
for statement in source
]
def build_definitions(self):
"""Builds all the definition."""
ret = []
for instruction in self._opaque_to_declare.values():
ret.append(self.build_definition(instruction, self.build_opaque_definition))
for instruction in self._subroutine_to_declare.values():
ret.append(self.build_definition(instruction, self.build_subroutine_definition))
for instruction in self._gate_to_declare.values():
ret.append(self.build_definition(instruction, self.build_gate_definition))
return ret
def build_definition(self, instruction, builder):
"""Using a given definition builder, builds that definition."""
try:
return instruction._define_qasm3()
except AttributeError:
pass
self._flat_reg = True
definition = builder(instruction)
self._flat_reg = False
return definition
def build_opaque_definition(self, instruction):
"""Builds an Opaque gate definition as a CalibrationDefinition"""
# We can't do anything sensible with this yet, so it's better to loudly say that.
raise QASM3ExporterError(
"Exporting opaque instructions with pulse-level calibrations is not yet supported by"
" the OpenQASM 3 exporter. Received this instruction, which appears opaque:"
f"\n{instruction}"
)
def build_subroutine_definition(self, instruction):
"""Builds a SubroutineDefinition"""
if instruction.definition.parameters:
# We don't yet have the type system to store the parameter types in a symbol table, and
# we currently don't have the correct logic in place to handle parameters correctly in
# the definition.
raise QASM3ExporterError(
"Exporting subroutines with parameters is not yet supported by the OpenQASM 3"
" exporter. Received this instruction, which appears parameterized:"
f"\n{instruction}"
)
name = self.global_namespace[instruction]
self.push_context(instruction.definition)
scope = self.current_scope()
quantum_arguments = [
ast.QuantumArgument(
self._reserve_variable_name(
ast.Identifier(f"{self.gate_qubit_prefix}_{n_qubit}"), scope
)
)
for n_qubit in range(len(instruction.definition.qubits))
]
subroutine_body = ast.SubroutineBlock(
self.build_quantum_instructions(instruction.definition.data),
)
self.pop_context()
return ast.SubroutineDefinition(ast.Identifier(name), subroutine_body, quantum_arguments)
def build_gate_definition(self, gate):
"""Builds a QuantumGateDefinition"""
self.push_context(gate.definition)
signature = self.build_gate_signature(gate)
body = ast.QuantumBlock(self.build_quantum_instructions(gate.definition.data))
self.pop_context()
return ast.QuantumGateDefinition(signature, body)
def build_gate_signature(self, gate):
"""Builds a QuantumGateSignature"""
name = self.global_namespace[gate]
params = []
definition = gate.definition
# Dummy parameters
scope = self.current_scope()
for num in range(len(gate.params) - len(definition.parameters)):
param_name = f"{self.gate_parameter_prefix}_{num}"
params.append(self._reserve_variable_name(ast.Identifier(param_name), scope))
params += [self._register_variable(param, scope) for param in definition.parameters]
quantum_arguments = [
self._reserve_variable_name(
ast.Identifier(f"{self.gate_qubit_prefix}_{n_qubit}"), scope
)
for n_qubit in range(len(definition.qubits))
]
return ast.QuantumGateSignature(ast.Identifier(name), quantum_arguments, params or None)
def build_parameter_declarations(self):
"""Builds lists of the input, output and standard variables used in this program."""
global_scope = self.global_scope(assert_=True)
for parameter in global_scope.circuit.parameters:
parameter_name = self._register_variable(parameter, global_scope)
declaration = _infer_variable_declaration(
global_scope.circuit, parameter, parameter_name
)
if declaration is None:
continue
if isinstance(declaration, ast.IODeclaration):
self._global_io_declarations.append(declaration)
else:
self._global_classical_declarations.append(declaration)
@property
def base_classical_register_name(self):
"""The base register name"""
name = "_all_clbits" if self.alias_classical_registers else "_loose_clbits"
if name in self.global_namespace._data:
raise NotImplementedError # TODO choose a different name if there is a name collision
return name
@property
def base_quantum_register_name(self):
"""The base register name"""
name = "_all_qubits"
if name in self.global_namespace._data:
raise NotImplementedError # TODO choose a different name if there is a name collision
return name
def build_classical_declarations(self):
"""Extend the global classical declarations with AST nodes declaring all the classical bits
and registers.
The behaviour of this function depends on the setting ``alias_classical_registers``. If this
is ``True``, then the output will be in the same form as the output of
:meth:`.build_classical_declarations`, with the registers being aliases. If ``False``, it
will instead return a :obj:`.ast.ClassicalDeclaration` for each classical register, and one
for the loose :obj:`.Clbit` instances, and will raise :obj:`QASM3ExporterError` if any
registers overlap.
This function populates the lookup table ``self._loose_clbit_index_lookup``.
"""
global_scope = self.global_scope()
circuit = self.current_scope().circuit
if self.alias_classical_registers:
self._loose_clbit_index_lookup = {
bit: index for index, bit in enumerate(circuit.clbits)
}
self._global_classical_declarations.append(
self.build_clbit_declaration(
len(circuit.clbits),
self._reserve_variable_name(
ast.Identifier(self.base_classical_register_name), global_scope
),
)
)
self._global_classical_declarations.extend(self.build_aliases(circuit.cregs))
return
loose_register_size = 0
for index, bit in enumerate(circuit.clbits):
found_bit = circuit.find_bit(bit)
if len(found_bit.registers) > 1:
raise QASM3ExporterError(
f"Clbit {index} is in multiple registers, but 'alias_classical_registers' is"
f" False. Registers and indices: {found_bit.registers}."
)
if not found_bit.registers:
self._loose_clbit_index_lookup[bit] = loose_register_size
loose_register_size += 1
if loose_register_size > 0:
self._global_classical_declarations.append(
self.build_clbit_declaration(
loose_register_size,
self._reserve_variable_name(
ast.Identifier(self.base_classical_register_name), global_scope
),
)
)
self._global_classical_declarations.extend(
self.build_clbit_declaration(
len(register), self._register_variable(register, global_scope)
)
for register in circuit.cregs
)
def build_clbit_declaration(
self, n_clbits: int, name: ast.Identifier
) -> ast.ClassicalDeclaration:
"""Return a declaration of the :obj:`.Clbit`\\ s as a ``bit[n]``."""
return ast.ClassicalDeclaration(ast.BitArrayType(n_clbits), name)
def build_quantum_declarations(self):
"""Return a list of AST nodes declaring all the qubits in the current scope, and all the
alias declarations for these qubits."""
return [self.build_qubit_declarations()] + self.build_aliases(
self.current_scope().circuit.qregs
)
def build_qubit_declarations(self):
"""Return a declaration of all the :obj:`.Qubit`\\ s in the current scope."""
global_scope = self.global_scope(assert_=True)
# Base register
return ast.QuantumDeclaration(
self._reserve_variable_name(
ast.Identifier(self.base_quantum_register_name), global_scope
),
ast.Designator(self.build_integer(self.current_scope().circuit.num_qubits)),
)
def build_aliases(self, registers: Iterable[Register]) -> List[ast.AliasStatement]:
"""Return a list of alias declarations for the given registers. The registers can be either
classical or quantum."""
out = []
scope = self.current_scope()
for register in registers:
elements = []
# Greedily consolidate runs of bits into ranges. We don't bother trying to handle
# steps; there's no need in generated code. Even single bits are referenced as ranges
# because the concatenation in an alias statement can only concatenate arraylike values.
start_index, prev_index = None, None
register_identifier = (
ast.Identifier(self.base_quantum_register_name)
if isinstance(register, QuantumRegister)
else ast.Identifier(self.base_classical_register_name)
)
for bit in register:
cur_index = self.find_bit(bit).index
if start_index is None:
start_index = cur_index
elif cur_index != prev_index + 1:
elements.append(
ast.SubscriptedIdentifier(
register_identifier,
ast.Range(
start=self.build_integer(start_index),
end=self.build_integer(prev_index),
),
)
)
start_index = prev_index = cur_index
prev_index = cur_index
# After the loop, if there were any bits at all, there's always one unemitted range.
if len(register) != 0:
elements.append(
ast.SubscriptedIdentifier(
register_identifier,
ast.Range(
start=self.build_integer(start_index),
end=self.build_integer(prev_index),
),
)
)
out.append(ast.AliasStatement(self._register_variable(register, scope), elements))
return out
def build_quantum_instructions(self, instructions):
"""Builds a list of call statements"""
ret = []
for instruction in instructions:
if isinstance(instruction.operation, ForLoopOp):
ret.append(self.build_for_loop(instruction))
continue
if isinstance(instruction.operation, WhileLoopOp):
ret.append(self.build_while_loop(instruction))
continue
if isinstance(instruction.operation, IfElseOp):
ret.append(self.build_if_statement(instruction))
continue
if isinstance(instruction.operation, SwitchCaseOp):
ret.extend(self.build_switch_statement(instruction))
continue
# Build the node, ignoring any condition.
if isinstance(instruction.operation, Gate):
nodes = [self.build_gate_call(instruction)]
elif isinstance(instruction.operation, Barrier):
operands = [
self.build_single_bit_reference(operand) for operand in instruction.qubits
]
nodes = [ast.QuantumBarrier(operands)]
elif isinstance(instruction.operation, Measure):
measurement = ast.QuantumMeasurement(
[self.build_single_bit_reference(operand) for operand in instruction.qubits]
)
qubit = self.build_single_bit_reference(instruction.clbits[0])
nodes = [ast.QuantumMeasurementAssignment(qubit, measurement)]
elif isinstance(instruction.operation, Reset):
nodes = [
ast.QuantumReset(self.build_single_bit_reference(operand))
for operand in instruction.qubits
]
elif isinstance(instruction.operation, Delay):
nodes = [self.build_delay(instruction)]
elif isinstance(instruction.operation, BreakLoopOp):
nodes = [ast.BreakStatement()]
elif isinstance(instruction.operation, ContinueLoopOp):
nodes = [ast.ContinueStatement()]
else:
nodes = [self.build_subroutine_call(instruction)]
if instruction.operation.condition is None:
ret.extend(nodes)
else:
eqcondition = self.build_eqcondition(instruction.operation.condition)
body = ast.ProgramBlock(nodes)
ret.append(ast.BranchingStatement(eqcondition, body))
return ret
def build_if_statement(self, instruction: CircuitInstruction) -> ast.BranchingStatement:
"""Build an :obj:`.IfElseOp` into a :obj:`.ast.BranchingStatement`."""
condition = self.build_eqcondition(instruction.operation.condition)
true_circuit = instruction.operation.blocks[0]
self.push_scope(true_circuit, instruction.qubits, instruction.clbits)
true_body = self.build_program_block(true_circuit.data)
self.pop_scope()
if len(instruction.operation.blocks) == 1:
return ast.BranchingStatement(condition, true_body, None)
false_circuit = instruction.operation.blocks[1]
self.push_scope(false_circuit, instruction.qubits, instruction.clbits)
false_body = self.build_program_block(false_circuit.data)
self.pop_scope()
return ast.BranchingStatement(condition, true_body, false_body)
def build_switch_statement(
self, instruction: CircuitInstruction
) -> Generator[ast.Statement, None, None]:
"""Build a :obj:`.SwitchCaseOp` into a :class:`.ast.SwitchStatement`."""
if ExperimentalFeatures.SWITCH_CASE_V1 not in self.experimental:
raise QASM3ExporterError(
"'switch' statements are not stabilized in OpenQASM 3 yet."
" To enable experimental support, set the flag"
" 'ExperimentalFeatures.SWITCH_CASE_V1' in the 'experimental' keyword"
" argument of the exporter."
)
if isinstance(instruction.operation.target, Clbit):
target = self.build_single_bit_reference(instruction.operation.target)
else:
real_target = self._lookup_variable(instruction.operation.target)
global_scope = self.global_scope()
target = self._reserve_variable_name(
ast.Identifier(self._unique_name("switch_dummy", global_scope)), global_scope
)
self._global_classical_declarations.append(
ast.ClassicalDeclaration(ast.IntType(), target, None)
)
yield ast.AssignmentStatement(target, real_target)
def case(values, case_block):
values = [
ast.DefaultCase() if v is CASE_DEFAULT else self.build_integer(v) for v in values
]
self.push_scope(case_block, instruction.qubits, instruction.clbits)
case_body = self.build_program_block(case_block.data)
self.pop_scope()
return values, case_body
yield ast.SwitchStatement(
target,
(case(values, block) for values, block in instruction.operation.cases_specifier()),
)
def build_while_loop(self, instruction: CircuitInstruction) -> ast.WhileLoopStatement:
"""Build a :obj:`.WhileLoopOp` into a :obj:`.ast.WhileLoopStatement`."""
condition = self.build_eqcondition(instruction.operation.condition)
loop_circuit = instruction.operation.blocks[0]
self.push_scope(loop_circuit, instruction.qubits, instruction.clbits)
loop_body = self.build_program_block(loop_circuit.data)
self.pop_scope()
return ast.WhileLoopStatement(condition, loop_body)
def build_for_loop(self, instruction: CircuitInstruction) -> ast.ForLoopStatement:
"""Build a :obj:`.ForLoopOp` into a :obj:`.ast.ForLoopStatement`."""
indexset, loop_parameter, loop_circuit = instruction.operation.params
self.push_scope(loop_circuit, instruction.qubits, instruction.clbits)
scope = self.current_scope()
if loop_parameter is None:
# The loop parameter is implicitly declared by the ``for`` loop (see also
# _infer_parameter_declaration), so it doesn't matter that we haven't declared this.
loop_parameter_ast = self._reserve_variable_name(ast.Identifier("_"), scope)
else:
loop_parameter_ast = self._register_variable(loop_parameter, scope)
if isinstance(indexset, range):
# QASM 3 uses inclusive ranges on both ends, unlike Python.
indexset_ast = ast.Range(
start=self.build_integer(indexset.start),
end=self.build_integer(indexset.stop - 1),
step=self.build_integer(indexset.step) if indexset.step != 1 else None,
)
else:
try:
indexset_ast = ast.IndexSet([self.build_integer(value) for value in indexset])
except QASM3ExporterError:
raise QASM3ExporterError(
"The values in QASM 3 'for' loops must all be integers, but received"
f" '{indexset}'."
) from None
body_ast = self.build_program_block(loop_circuit)
self.pop_scope()
return ast.ForLoopStatement(indexset_ast, loop_parameter_ast, body_ast)
def build_delay(self, instruction: CircuitInstruction) -> ast.QuantumDelay:
"""Build a built-in delay statement."""
if instruction.clbits:
raise QASM3ExporterError(
f"Found a delay instruction acting on classical bits: {instruction}"
)
duration_value, unit = instruction.operation.duration, instruction.operation.unit
if unit == "ps":
duration = ast.DurationLiteral(1000 * duration_value, ast.DurationUnit.NANOSECOND)
else:
unit_map = {
"ns": ast.DurationUnit.NANOSECOND,
"us": ast.DurationUnit.MICROSECOND,
"ms": ast.DurationUnit.MILLISECOND,
"s": ast.DurationUnit.SECOND,
"dt": ast.DurationUnit.SAMPLE,
}
duration = ast.DurationLiteral(duration_value, unit_map[unit])
return ast.QuantumDelay(
duration, [self.build_single_bit_reference(qubit) for qubit in instruction.qubits]
)
def build_integer(self, value) -> ast.Integer:
"""Build an integer literal, raising a :obj:`.QASM3ExporterError` if the input is not
actually an
integer."""
if not isinstance(value, numbers.Integral):
# This is meant to be purely defensive, in case a non-integer slips into the logic
# somewhere, but no valid Terra object should trigger this.
raise QASM3ExporterError(f"'{value}' is not an integer") # pragma: no cover
return ast.Integer(int(value))
def build_program_block(self, instructions):
"""Builds a ProgramBlock"""
return ast.ProgramBlock(self.build_quantum_instructions(instructions))
def build_eqcondition(self, condition):
"""Classical Conditional condition from a instruction.condition"""
if isinstance(condition[0], Clbit):
condition_on = self.build_single_bit_reference(condition[0])
else:
condition_on = self._lookup_variable(condition[0])
return ast.ComparisonExpression(
condition_on, ast.EqualsOperator(), self.build_integer(condition[1])
)
def _rebind_scoped_parameters(self, expression):
"""If the input is a :class:`.ParameterExpression`, rebind any internal
:class:`.Parameter`\\ s so that their names match their names in the scope. Other inputs
are returned unchanged."""
# This is a little hacky, but the entirety of the Expression handling is essentially
# missing, pending a new system in Terra to replace it (2022-03-07).
if not isinstance(expression, ParameterExpression):
return expression
return expression.subs(
{
param: Parameter(self._lookup_variable(param).string)
for param in expression.parameters
}
)
def build_gate_call(self, instruction: CircuitInstruction):
"""Builds a QuantumGateCall"""
if isinstance(instruction.operation, standard_gates.UGate):
gate_name = ast.Identifier("U")
else:
gate_name = ast.Identifier(self.global_namespace[instruction.operation])
qubits = [self.build_single_bit_reference(qubit) for qubit in instruction.qubits]
if self.disable_constants:
parameters = [
ast.Expression(self._rebind_scoped_parameters(param))
for param in instruction.operation.params
]
else:
parameters = [
ast.Expression(pi_check(self._rebind_scoped_parameters(param), output="qasm"))
for param in instruction.operation.params
]
return ast.QuantumGateCall(gate_name, qubits, parameters=parameters)
def build_subroutine_call(self, instruction: CircuitInstruction):
"""Builds a SubroutineCall"""
identifier = ast.Identifier(self.global_namespace[instruction.operation])
expressions = [ast.Expression(param) for param in instruction.operation.params]
# TODO: qubits should go inside the brackets of subroutine calls, but neither Terra nor the
# AST here really support the calls, so there's no sensible way of writing it yet.
bits = [self.build_single_bit_reference(bit) for bit in instruction.qubits]
return ast.SubroutineCall(identifier, bits, expressions)
def build_single_bit_reference(self, bit: Bit) -> ast.Identifier:
"""Get an identifier node that refers to one particular bit."""
found_bit = self.find_bit(bit)
if self._physical_qubit and isinstance(bit, Qubit):
return ast.PhysicalQubitIdentifier(ast.Identifier(str(found_bit.index)))
if self._flat_reg:
return ast.Identifier(f"{self.gate_qubit_prefix}_{found_bit.index}")
if found_bit.registers:
# We preferentially return a reference via a register in the hope that this is what the
# user is used to seeing as well.
register, index = found_bit.registers[0]
return ast.SubscriptedIdentifier(
self._lookup_variable(register), self.build_integer(index)
)
# Otherwise reference via the list of all qubits, or the list of loose clbits.
if isinstance(bit, Qubit):
return ast.SubscriptedIdentifier(
ast.Identifier(self.base_quantum_register_name), self.build_integer(found_bit.index)
)
return ast.SubscriptedIdentifier(
ast.Identifier(self.base_classical_register_name),
self.build_integer(self._loose_clbit_index_lookup[bit]),
)
def find_bit(self, bit: Bit):
"""Look up the bit using :meth:`.QuantumCircuit.find_bit` in the current outermost scope."""
# This is a hacky work-around for now. Really this should be a proper symbol-table lookup,
# but with us expecting to put in a whole new AST for Terra 0.20, this should be sufficient
# for the use-cases we support. (Jake, 2021-11-22.)
if len(self.current_context()) > 1:
ancestor_bit = self.current_scope().bit_map[bit]
return self.current_outermost_scope().circuit.find_bit(ancestor_bit)
return self.current_scope().circuit.find_bit(bit)
def _infer_variable_declaration(
circuit: QuantumCircuit, parameter: Parameter, parameter_name: ast.Identifier
) -> Union[ast.ClassicalDeclaration, None]:
"""Attempt to infer what type a parameter should be declared as to work with a circuit.
This is very simplistic; it assumes all parameters are real numbers that need to be input to the
program, unless one is used as a loop variable, in which case it shouldn't be declared at all,
because the ``for`` loop declares it implicitly (per the Qiskit/QSS reading of the OpenQASM
spec at Qiskit/openqasm@8ee55ec).
.. note::
This is a hack around not having a proper type system implemented in Terra, and really this
whole function should be removed in favour of proper symbol-table building and lookups.
This function is purely to try and hack the parameters for ``for`` loops into the exporter
for now.
Args:
circuit: The global-scope circuit, which is the base of the exported program.
parameter: The parameter to infer the type of.
parameter_name: The name of the parameter to use in the declaration.
Returns:
A suitable :obj:`.ast.ClassicalDeclaration` node, or, if the parameter should *not* be
declared, then ``None``.
"""
def is_loop_variable(circuit, parameter):
"""Recurse into the instructions a parameter is used in, checking at every level if it is
used as the loop variable of a ``for`` loop."""
# This private access is hacky, and shouldn't need to happen; the type of a parameter
# _should_ be an intrinsic part of the parameter, or somewhere publicly accessible, but
# Terra doesn't have those concepts yet. We can only try and guess at the type by looking
# at all the places it's used in the circuit.
for instruction, index in circuit._parameter_table[parameter]:
if isinstance(instruction, ForLoopOp):
# The parameters of ForLoopOp are (indexset, loop_parameter, body).
if index == 1:
return True
if isinstance(instruction, ControlFlowOp):
if is_loop_variable(instruction.params[index], parameter):
return True
return False
if is_loop_variable(circuit, parameter):
return None
# Arbitrary choice of double-precision float for all other parameters, but it's what we actually
# expect people to be binding to their Parameters right now.
return ast.IODeclaration(ast.IOModifier.INPUT, ast.FloatType.DOUBLE, parameter_name)