Skip to content

Commit

Permalink
Followup of #11095 (#11404)
Browse files Browse the repository at this point in the history
* Followup
- Add exception handling for the edge case in which a basis gate property is not reported
- Cleanup docs
- Replace logging with RuntimeWarning
- Add more inline comments
- Fix wrong typehints
- Update handling of faulty qubits with set operation

* bugfix + more warning message

* Update reno

* Add more check for filter option
  • Loading branch information
nkanazawa1989 authored Jan 18, 2024
1 parent 68ac142 commit 828ad4d
Show file tree
Hide file tree
Showing 6 changed files with 174 additions and 86 deletions.
167 changes: 94 additions & 73 deletions qiskit/providers/backend_compat.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,8 @@

from __future__ import annotations
import logging
from typing import List, Iterable, Any, Dict, Optional, Tuple
import warnings
from typing import List, Iterable, Any, Dict, Optional

from qiskit.providers.backend import BackendV1, BackendV2
from qiskit.providers.backend import QubitProperties
Expand All @@ -38,20 +39,17 @@ def convert_to_target(
):
"""Decode transpiler target from backend data set.
This function generates ``Target`` instance from intermediate
legacy objects such as ``BackendProperties`` and ``PulseDefaults``.
.. note::
Passing in legacy objects like BackendProperties as properties and PulseDefaults
as defaults will be deprecated in the future.
This function generates :class:`.Target`` instance from intermediate
legacy objects such as :class:`.BackendProperties` and :class:`.PulseDefaults`.
These objects are usually components of the legacy :class:`.BackendV1` model.
Args:
configuration: Backend configuration as ``BackendConfiguration``
properties: Backend property dictionary or ``BackendProperties``
defaults: Backend pulse defaults dictionary or ``PulseDefaults``
custom_name_mapping: A name mapping must be supplied for the operation
not included in Qiskit Standard Gate name mapping, otherwise the operation
will be dropped in the resulting ``Target`` object.
not included in Qiskit Standard Gate name mapping, otherwise the operation
will be dropped in the resulting ``Target`` object.
add_delay: If True, adds delay to the instruction set.
filter_faulty: If True, this filters the non-operational qubits.
Expand Down Expand Up @@ -95,12 +93,11 @@ def convert_to_target(
# Create instruction property placeholder from backend configuration
basis_gates = set(getattr(configuration, "basis_gates", []))
gate_configs = {gate.name: gate for gate in configuration.gates}
inst_name_map = {} # type: Dict[str, Instruction]
prop_name_map = {} # type: Dict[str, Dict[Tuple[int, ...], InstructionProperties]]
all_instructions = set.union(basis_gates, set(required))
inst_name_map = {} # type: Dict[str, Instruction]

faulty_ops = set()
faulty_qubits = []
faulty_qubits = set()
unsupported_instructions = []

# Create name to Qiskit instruction object repr mapping
Expand All @@ -110,6 +107,10 @@ def convert_to_target(
if name in qiskit_inst_mapping:
inst_name_map[name] = qiskit_inst_mapping[name]
elif name in gate_configs:
# GateConfig model is a translator of QASM opcode.
# This doesn't have quantum definition, so Qiskit transpiler doesn't perform
# any optimization in quantum domain.
# Usually GateConfig counterpart should exist in Qiskit namespace so this is rarely called.
this_config = gate_configs[name]
params = list(map(Parameter, getattr(this_config, "parameters", [])))
coupling_map = getattr(this_config, "coupling_map", [])
Expand All @@ -119,29 +120,44 @@ def convert_to_target(
params=params,
)
else:
logger.warning(
"Definition of instruction %s is not found in the Qiskit namespace and "
"GateConfig is not provided by the BackendConfiguration payload. "
"Qiskit Gate model cannot be instantiated for this instruction and "
"this instruction is silently excluded from the Target. "
"Please add new gate class to Qiskit or provide GateConfig for this name.",
name,
warnings.warn(
f"No gate definition for {name} can be found and is being excluded "
"from the generated target. You can use `custom_name_mapping` to provide "
"a definition for this operation.",
RuntimeWarning,
)
unsupported_instructions.append(name)

for name in unsupported_instructions:
all_instructions.remove(name)

# Create empty inst properties from gate configs
for name, spec in gate_configs.items():
if hasattr(spec, "coupling_map"):
coupling_map = spec.coupling_map
prop_name_map[name] = dict.fromkeys(map(tuple, coupling_map))
else:
prop_name_map[name] = None
# Create inst properties placeholder
# Without any assignment, properties value is None,
# which defines a global instruction that can be applied to any qubit(s).
# The None value behaves differently from an empty dictionary.
# See API doc of Target.add_instruction for details.
prop_name_map = dict.fromkeys(all_instructions)
for name in all_instructions:
if name in gate_configs:
if coupling_map := getattr(gate_configs[name], "coupling_map", None):
# Respect operational qubits that gate configuration defines
# This ties instruction to particular qubits even without properties information.
# Note that each instruction is considered to be ideal unless
# its spec (e.g. error, duration) is bound by the properties object.
prop_name_map[name] = dict.fromkeys(map(tuple, coupling_map))

# Populate instruction properties
if properties:

def _get_value(prop_dict, prop_name):
if ndval := prop_dict.get(prop_name, None):
return ndval[0]
return None

# is_qubit_operational is a bit of expensive operation so precache the value
faulty_qubits = {
q for q in range(configuration.num_qubits) if not properties.is_qubit_operational(q)
}
qubit_properties = [
QubitProperties(
t1=properties.qubit_property(qubit_idx)["T1"][0],
Expand All @@ -153,52 +169,64 @@ def convert_to_target(

in_data["qubit_properties"] = qubit_properties

if filter_faulty:
faulty_qubits = properties.faulty_qubits()

for name in prop_name_map.keys():
for qubits, params in properties.gate_property(name).items():
in_param = {
"error": params["gate_error"][0] if "gate_error" in params else None,
"duration": params["gate_length"][0] if "gate_length" in params else None,
}
inst_prop = InstructionProperties(**in_param)

if filter_faulty and (
(not properties.is_gate_operational(name, qubits))
or any(not properties.is_qubit_operational(qubit) for qubit in qubits)
for name in all_instructions:
try:
for qubits, params in properties.gate_property(name).items():
if filter_faulty and (
set.intersection(faulty_qubits, qubits)
or not properties.is_gate_operational(name, qubits)
):
try:
# Qubits might be pre-defined by the gate config
# However properties objects says the qubits is non-operational
del prop_name_map[name][qubits]
except KeyError:
pass
faulty_ops.add((name, qubits))
continue
if prop_name_map[name] is None:
# This instruction is tied to particular qubits
# i.e. gate config is not provided, and instruction has been globally defined.
prop_name_map[name] = {}
prop_name_map[name][qubits] = InstructionProperties(
error=_get_value(params, "gate_error"),
duration=_get_value(params, "gate_length"),
)
if isinstance(prop_name_map[name], dict) and any(
v is None for v in prop_name_map[name].values()
):
faulty_ops.add((name, qubits))
try:
del prop_name_map[name][qubits]
except KeyError:
pass
continue

if prop_name_map[name] is None:
prop_name_map[name] = {}

prop_name_map[name][qubits] = inst_prop
# Properties provides gate properties only for subset of qubits
# Associated qubit set might be defined by the gate config here
logger.info(
"Gate properties of instruction %s are not provided for every qubits. "
"This gate is ideal for some qubits and the rest is with finite error. "
"Created backend target may confuse error-aware circuit optimization.",
name,
)
except BackendPropertyError:
# This gate doesn't report any property
continue

# Measure instruction property is stored in qubit property
prop_name_map["measure"] = {}

for qubit_idx in range(configuration.num_qubits):
if qubit_idx in faulty_qubits:
if filter_faulty and (qubit_idx in faulty_qubits):
continue
qubit_prop = properties.qubit_property(qubit_idx)
in_prop = {
"duration": qubit_prop["readout_length"][0]
if "readout_length" in qubit_prop
else None,
"error": qubit_prop["readout_error"][0] if "readout_error" in qubit_prop else None,
}
prop_name_map["measure"][(qubit_idx,)] = InstructionProperties(**in_prop)
prop_name_map["measure"][(qubit_idx,)] = InstructionProperties(
error=_get_value(qubit_prop, "readout_error"),
duration=_get_value(qubit_prop, "readout_length"),
)

if add_delay and "delay" not in prop_name_map:
prop_name_map["delay"] = {
(q,): None for q in range(configuration.num_qubits) if q not in faulty_qubits
}
for op in required:
# Map required ops to each operational qubit
if prop_name_map[op] is None:
prop_name_map[op] = {
(q,): None
for q in range(configuration.num_qubits)
if not filter_faulty or (q not in faulty_qubits)
}

if defaults:
inst_sched_map = defaults.instruction_schedule_map
Expand Down Expand Up @@ -238,29 +266,22 @@ def convert_to_target(
qubits,
)

# Remove 'delay' if add_delay is set to False.
if not add_delay:
if "delay" in all_instructions:
all_instructions.remove("delay")

# Add parsed properties to target
target = Target(**in_data)
for inst_name in all_instructions:
if inst_name == "delay" and not add_delay:
continue
if inst_name in qiskit_control_flow_mapping:
# Control flow operator doesn't have gate property.
target.add_instruction(
instruction=qiskit_control_flow_mapping[inst_name],
name=inst_name,
)
elif properties is None:
target.add_instruction(
instruction=inst_name_map[inst_name],
name=inst_name,
)
else:
target.add_instruction(
instruction=inst_name_map[inst_name],
properties=prop_name_map.get(inst_name, None),
name=inst_name,
)

return target
Expand Down
17 changes: 13 additions & 4 deletions qiskit/providers/models/backendproperties.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,12 +14,14 @@

import copy
import datetime
from typing import Any, Iterable, Tuple, Union
from typing import Any, Iterable, Tuple, Union, Dict
import dateutil.parser

from qiskit.providers.exceptions import BackendPropertyError
from qiskit.utils.units import apply_prefix

PropertyT = Tuple[Any, datetime.datetime]


class Nduv:
"""Class representing name-date-unit-value
Expand Down Expand Up @@ -279,8 +281,11 @@ def __eq__(self, other):
return False

def gate_property(
self, gate: str, qubits: Union[int, Iterable[int]] = None, name: str = None
) -> Tuple[Any, datetime.datetime]:
self,
gate: str,
qubits: Union[int, Iterable[int]] = None,
name: str = None,
) -> Union[Dict[Tuple[int, ...], Dict[str, PropertyT]], Dict[str, PropertyT], PropertyT,]:
"""
Return the property of the given gate.
Expand Down Expand Up @@ -369,7 +374,11 @@ def gate_length(self, gate: str, qubits: Union[int, Iterable[int]]) -> float:
"""
return self.gate_property(gate, qubits, "gate_length")[0] # Throw away datetime at index 1

def qubit_property(self, qubit: int, name: str = None) -> Tuple[Any, datetime.datetime]:
def qubit_property(
self,
qubit: int,
name: str = None,
) -> Union[Dict[str, PropertyT], PropertyT,]:
"""
Return the property of the given qubit.
Expand Down
5 changes: 3 additions & 2 deletions qiskit/transpiler/target.py
Original file line number Diff line number Diff line change
Expand Up @@ -361,8 +361,9 @@ def add_instruction(self, instruction, properties=None, name=None):
supports.
Args:
instruction (qiskit.circuit.Instruction): The operation object to add to the map. If it's
parameterized any value of the parameter can be set. Optionally for variable width
instruction (Union[qiskit.circuit.Instruction, Type[qiskit.circuit.Instruction]]):
The operation object to add to the map. If it's parameterized any value
of the parameter can be set. Optionally for variable width
instructions (such as control flow operations such as :class:`~.ForLoop` or
:class:`~MCXGate`) you can specify the class. If the class is specified than the
``name`` argument must be specified. When a class is used the gate is treated as global
Expand Down
Original file line number Diff line number Diff line change
@@ -1,19 +1,20 @@
---
upgrade:
- |
The new logic provides better backend model up-conversion mechanism, and better handling of control flow instructions.
Changed default value of two arguments :code:`add_delay` and :code:`filter_faulty` in
the :func:`.qiskit.providers.backend_compat.convert_to_target`.
Now this conversion function adds delay instruction and removes faulty instructions by default.
fixes:
- |
Fixes return of improper Schedule by Backend.instruction_schedule_map.get('measure', [0])
Fixes return of improper measurement schedule that may occur in the following program
.. code-block:: python
#import a fake backend which is a sub-class of BackendV2.
# import a fake backend which is a sub-class of BackendV2.
from qiskit.providers.fake_provider import FakePerth
backend = FakePerth()
sched = backend.instruction_schedule_map.get('measure', [0])
The issue was that the :code:`sched` contained Schedule for measure operation on
all qubits of the backend instead of having the Schedule for measure operation
on just qubit_0.
This unexpectedly returned a measure schedule including all device qubits,
which was fixed in this release.
Now this returns a schedule for qubit 0 as intended.
18 changes: 18 additions & 0 deletions test/python/providers/test_fake_backends.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
from qiskit.providers.backend_compat import BackendV2Converter
from qiskit.providers.models.backendproperties import BackendProperties
from qiskit.providers.backend import BackendV2
from qiskit.providers.models import GateConfig
from qiskit.utils import optionals
from qiskit.circuit.library import (
SXGate,
Expand Down Expand Up @@ -558,6 +559,23 @@ def test_backend_v2_converter_without_delay(self):

self.assertEqual(backend.target.qargs, expected)

def test_backend_v2_converter_with_meaningless_gate_config(self):
"""Test backend with broken gate config can be converted only with properties data."""
backend_v1 = FakeYorktown()
backend_v1.configuration().gates = [
GateConfig(name="NotValidGate", parameters=[], qasm_def="not_valid_gate")
]
backend_v2 = BackendV2Converter(
backend=backend_v1,
filter_faulty=True,
add_delay=False,
)
ops_with_measure = backend_v2.target.operation_names
self.assertCountEqual(
ops_with_measure,
backend_v1.configuration().basis_gates + ["measure"],
)

def test_filter_faulty_qubits_and_gates_backend_v2_converter(self):
"""Test faulty gates and qubits."""
backend = FakeWashington()
Expand Down
Loading

0 comments on commit 828ad4d

Please sign in to comment.