Source code for qiskit.pulse.instruction_schedule_map
# This code is part of Qiskit.
#
# (C) Copyright IBM 2019.
#
# 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.
"""
A convenient way to track reusable subschedules by name and qubit.
This can be used for scheduling circuits with custom definitions, for instance::
inst_map = InstructionScheduleMap()
inst_map.add('new_inst', 0, qubit_0_new_inst_schedule)
sched = schedule(quantum_circuit, backend, inst_map)
An instance of this class is instantiated by Pulse-enabled backends and populated with defaults
(if available)::
inst_map = backend.defaults().instruction_schedule_map
"""
import inspect
import warnings
from collections import defaultdict
from typing import Callable, Iterable, List, Tuple, Union, Optional, NamedTuple
from qiskit.circuit.instruction import Instruction
from qiskit.circuit.parameterexpression import ParameterExpression, ParameterValueType
from qiskit.pulse.exceptions import PulseError
from qiskit.pulse.schedule import Schedule, ScheduleBlock, ParameterizedSchedule
Generator = NamedTuple(
"Generator",
[("function", Union[Callable, Schedule, ScheduleBlock]), ("signature", inspect.Signature)],
)
[docs]class InstructionScheduleMap:
"""Mapping from :py:class:`~qiskit.circuit.QuantumCircuit`
:py:class:`qiskit.circuit.Instruction` names and qubits to
:py:class:`~qiskit.pulse.Schedule` s. In particular, the mapping is formatted as type::
Dict[str, Dict[Tuple[int], Schedule]]
where the first key is the name of a circuit instruction (e.g. ``'u1'``, ``'measure'``), the
second key is a tuple of qubit indices, and the final value is a Schedule implementing the
requested instruction.
These can usually be seen as gate calibrations.
"""
def __init__(self):
"""Initialize a circuit instruction to schedule mapper instance."""
# The processed and reformatted circuit instruction definitions
self._map = defaultdict(lambda: defaultdict(Generator))
# A backwards mapping from qubit to supported instructions
self._qubit_instructions = defaultdict(set)
@property
def instructions(self) -> List[str]:
"""Return all instructions which have definitions.
By default, these are typically the basis gates along with other instructions such as
measure and reset.
Returns:
The names of all the circuit instructions which have Schedule definitions in this.
"""
return list(self._map.keys())
[docs] def qubits_with_instruction(
self, instruction: Union[str, Instruction]
) -> List[Union[int, Tuple[int]]]:
"""Return a list of the qubits for which the given instruction is defined. Single qubit
instructions return a flat list, and multiqubit instructions return a list of ordered
tuples.
Args:
instruction: The name of the circuit instruction.
Returns:
Qubit indices which have the given instruction defined. This is a list of tuples if the
instruction has an arity greater than 1, or a flat list of ints otherwise.
Raises:
PulseError: If the instruction is not found.
"""
instruction = _get_instruction_string(instruction)
if instruction not in self._map:
return []
return [
qubits[0] if len(qubits) == 1 else qubits
for qubits in sorted(self._map[instruction].keys())
]
[docs] def qubit_instructions(self, qubits: Union[int, Iterable[int]]) -> List[str]:
"""Return a list of the instruction names that are defined by the backend for the given
qubit or qubits.
Args:
qubits: A qubit index, or a list or tuple of indices.
Returns:
All the instructions which are defined on the qubits.
For 1 qubit, all the 1Q instructions defined. For multiple qubits, all the instructions
which apply to that whole set of qubits (e.g. ``qubits=[0, 1]`` may return ``['cx']``).
"""
if _to_tuple(qubits) in self._qubit_instructions:
return list(self._qubit_instructions[_to_tuple(qubits)])
return []
[docs] def has(self, instruction: Union[str, Instruction], qubits: Union[int, Iterable[int]]) -> bool:
"""Is the instruction defined for the given qubits?
Args:
instruction: The instruction for which to look.
qubits: The specific qubits for the instruction.
Returns:
True iff the instruction is defined.
"""
instruction = _get_instruction_string(instruction)
return instruction in self._map and _to_tuple(qubits) in self._map[instruction]
[docs] def assert_has(
self, instruction: Union[str, Instruction], qubits: Union[int, Iterable[int]]
) -> None:
"""Error if the given instruction is not defined.
Args:
instruction: The instruction for which to look.
qubits: The specific qubits for the instruction.
Raises:
PulseError: If the instruction is not defined on the qubits.
"""
instruction = _get_instruction_string(instruction)
if not self.has(instruction, _to_tuple(qubits)):
if instruction in self._map:
raise PulseError(
"Operation '{inst}' exists, but is only defined for qubits "
"{qubits}.".format(
inst=instruction, qubits=self.qubits_with_instruction(instruction)
)
)
raise PulseError(
"Operation '{inst}' is not defined for this " "system.".format(inst=instruction)
)
[docs] def get(
self,
instruction: Union[str, Instruction],
qubits: Union[int, Iterable[int]],
*params: Union[complex, ParameterExpression],
**kwparams: Union[complex, ParameterExpression],
) -> Union[Schedule, ScheduleBlock]:
"""Return the defined :py:class:`~qiskit.pulse.Schedule` or
:py:class:`~qiskit.pulse.ScheduleBlock` for the given instruction on the given qubits.
If all keys are not specified this method returns schedule with unbound parameters.
Args:
instruction: Name of the instruction or the instruction itself.
qubits: The qubits for the instruction.
*params: Command parameters for generating the output schedule.
**kwparams: Keyworded command parameters for generating the schedule.
Returns:
The Schedule defined for the input.
Raises:
PulseError: When invalid parameters are specified.
"""
instruction = _get_instruction_string(instruction)
self.assert_has(instruction, qubits)
generator = self._map[instruction][_to_tuple(qubits)]
_error_message = (
f"*params={params}, **kwparams={kwparams} do not match with "
f"the schedule generator signature {generator.signature}."
)
function = generator.function
if callable(function):
try:
# callables require full binding, but default values can exist.
binds = generator.signature.bind(*params, **kwparams)
binds.apply_defaults()
except TypeError as ex:
raise PulseError(_error_message) from ex
return function(**binds.arguments)
try:
# schedules allow partial binding
binds = generator.signature.bind_partial(*params, **kwparams)
except TypeError as ex:
raise PulseError(_error_message) from ex
if len(binds.arguments) > 0:
if isinstance(function, ParameterizedSchedule):
return function.bind_parameters(**binds.arguments)
value_dict = dict()
for param in function.parameters:
try:
value_dict[param] = binds.arguments[param.name]
except KeyError:
pass
return function.assign_parameters(value_dict, inplace=False)
else:
return function
[docs] def add(
self,
instruction: Union[str, Instruction],
qubits: Union[int, Iterable[int]],
schedule: Union[Schedule, ScheduleBlock, Callable[..., Union[Schedule, ScheduleBlock]]],
arguments: Optional[List[str]] = None,
) -> None:
"""Add a new known instruction for the given qubits and its mapping to a pulse schedule.
Args:
instruction: The name of the instruction to add.
qubits: The qubits which the instruction applies to.
schedule: The Schedule that implements the given instruction.
arguments: List of parameter names to create a parameter-bound schedule from the
associated gate instruction. If :py:meth:`get` is called with arguments rather
than keyword arguments, this parameter list is used to map the input arguments to
parameter objects stored in the target schedule.
Raises:
PulseError: If the qubits are provided as an empty iterable.
"""
instruction = _get_instruction_string(instruction)
# validation of target qubit
qubits = _to_tuple(qubits)
if qubits == ():
raise PulseError(f"Cannot add definition {instruction} with no target qubits.")
# generate signature
if isinstance(schedule, (Schedule, ScheduleBlock)):
ordered_names = sorted(list({par.name for par in schedule.parameters}))
if arguments:
if set(arguments) != set(ordered_names):
raise PulseError(
"Arguments does not match with schedule parameters. "
f"{set(arguments)} != {schedule.parameters}."
)
ordered_names = arguments
parameters = list()
for argname in ordered_names:
param_signature = inspect.Parameter(
name=argname,
annotation=ParameterValueType,
kind=inspect.Parameter.POSITIONAL_OR_KEYWORD,
)
parameters.append(param_signature)
signature = inspect.Signature(parameters=parameters, return_annotation=type(schedule))
elif isinstance(schedule, ParameterizedSchedule):
# TODO remove this
warnings.warn(
"ParameterizedSchedule has been deprecated. "
"Define Schedule with Parameter objects.",
DeprecationWarning,
)
parameters = list()
for argname in schedule.parameters:
param_signature = inspect.Parameter(
name=argname,
annotation=ParameterValueType,
kind=inspect.Parameter.POSITIONAL_OR_KEYWORD,
)
parameters.append(param_signature)
signature = inspect.Signature(parameters=parameters, return_annotation=Schedule)
elif callable(schedule):
if arguments:
warnings.warn(
"Arguments are overridden by the callback function signature. "
"Input `arguments` are ignored.",
UserWarning,
)
signature = inspect.signature(schedule)
else:
raise PulseError(
"Supplied schedule must be one of the Schedule, ScheduleBlock or a "
"callable that outputs a schedule."
)
self._map[instruction][qubits] = Generator(schedule, signature)
self._qubit_instructions[qubits].add(instruction)
[docs] def remove(
self, instruction: Union[str, Instruction], qubits: Union[int, Iterable[int]]
) -> None:
"""Remove the given instruction from the listing of instructions defined in self.
Args:
instruction: The name of the instruction to add.
qubits: The qubits which the instruction applies to.
"""
instruction = _get_instruction_string(instruction)
qubits = _to_tuple(qubits)
self.assert_has(instruction, qubits)
self._map[instruction].pop(qubits)
self._qubit_instructions[qubits].remove(instruction)
if not self._map[instruction]:
self._map.pop(instruction)
if not self._qubit_instructions[qubits]:
self._qubit_instructions.pop(qubits)
[docs] def pop(
self,
instruction: Union[str, Instruction],
qubits: Union[int, Iterable[int]],
*params: Union[complex, ParameterExpression],
**kwparams: Union[complex, ParameterExpression],
) -> Union[Schedule, ScheduleBlock]:
"""Remove and return the defined schedule for the given instruction on the given
qubits.
Args:
instruction: Name of the instruction.
qubits: The qubits for the instruction.
*params: Command parameters for generating the output schedule.
**kwparams: Keyworded command parameters for generating the schedule.
Returns:
The Schedule defined for the input.
"""
instruction = _get_instruction_string(instruction)
schedule = self.get(instruction, qubits, *params, **kwparams)
self.remove(instruction, qubits)
return schedule
[docs] def get_parameters(
self, instruction: Union[str, Instruction], qubits: Union[int, Iterable[int]]
) -> Tuple[str]:
"""Return the list of parameters taken by the given instruction on the given qubits.
Args:
instruction: Name of the instruction.
qubits: The qubits for the instruction.
Returns:
The names of the parameters required by the instruction.
"""
instruction = _get_instruction_string(instruction)
self.assert_has(instruction, qubits)
signature = self._map[instruction][_to_tuple(qubits)].signature
return tuple(signature.parameters.keys())
def __str__(self):
single_q_insts = "1Q instructions:\n"
multi_q_insts = "Multi qubit instructions:\n"
for qubits, insts in self._qubit_instructions.items():
if len(qubits) == 1:
single_q_insts += f" q{qubits[0]}: {insts}\n"
else:
multi_q_insts += f" {qubits}: {insts}\n"
instructions = single_q_insts + multi_q_insts
return "<{name}({insts})>" "".format(name=self.__class__.__name__, insts=instructions)
def _to_tuple(values: Union[int, Iterable[int]]) -> Tuple[int, ...]:
"""Return the input as a tuple.
Args:
values: An integer, or iterable of integers.
Returns:
The input values as a sorted tuple.
"""
try:
return tuple(values)
except TypeError:
return (values,)
def _get_instruction_string(inst: Union[str, Instruction]):
if isinstance(inst, str):
return inst
else:
try:
return inst.name
except AttributeError as ex:
raise PulseError(
'Input "inst" has no attribute "name".' 'This should be a circuit "Instruction".'
) from ex