Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Combine RB and XEB to compute inferred errors #6455

Merged
merged 27 commits into from
Feb 13, 2024
Merged
Show file tree
Hide file tree
Changes from 26 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
f3f6727
Initial commit
NoureldinYosri Feb 5, 2024
cc8aa10
nits
NoureldinYosri Feb 5, 2024
d616efe
Add TwoQubitRandomizedBenchMarkResult and the computation of derived …
NoureldinYosri Feb 5, 2024
d06cc93
nit
NoureldinYosri Feb 5, 2024
592ecfc
nit
NoureldinYosri Feb 6, 2024
9435cff
add target_error arg
NoureldinYosri Feb 6, 2024
76e5a8d
nit
NoureldinYosri Feb 6, 2024
2a2f8b8
nit
NoureldinYosri Feb 6, 2024
a94730c
add tests and docstrings
NoureldinYosri Feb 6, 2024
9aa9d4e
ensure the dictionary index is sorted
NoureldinYosri Feb 7, 2024
e6151f0
add a method to plot histogram
NoureldinYosri Feb 7, 2024
a91e175
Merge branch 'main' into 2q_xeb
NoureldinYosri Feb 7, 2024
8e47acd
Merge branch 'main' into 2q_xeb
NoureldinYosri Feb 7, 2024
64effd8
address comments
NoureldinYosri Feb 8, 2024
5b2af9e
Merge branch 'main' into 2q_xeb
NoureldinYosri Feb 9, 2024
2acc13d
Combine RB and XEB to compute inferred errors
NoureldinYosri Feb 10, 2024
82f8428
cache methods
NoureldinYosri Feb 10, 2024
798f82a
add plotting
NoureldinYosri Feb 10, 2024
596974f
Merge branch 'main' into 2q_xeb
NoureldinYosri Feb 10, 2024
cf1002b
docstring
NoureldinYosri Feb 10, 2024
6994800
fix inferred errors
NoureldinYosri Feb 10, 2024
af59604
address comments
NoureldinYosri Feb 12, 2024
db9657a
Merge branch 'main' into 2q_xeb
NoureldinYosri Feb 12, 2024
c4fd143
Merge branch 'main' into 2q_xeb
NoureldinYosri Feb 13, 2024
6aee73b
address comments
NoureldinYosri Feb 13, 2024
9f7d628
nit
NoureldinYosri Feb 13, 2024
f14bf7e
final nit
NoureldinYosri Feb 13, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 5 additions & 1 deletion cirq-core/cirq/experiments/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -65,4 +65,8 @@

from cirq.experiments.xeb_fitting import XEBPhasedFSimCharacterizationOptions

from cirq.experiments.two_qubit_xeb import TwoQubitXEBResult, parallel_two_qubit_xeb
from cirq.experiments.two_qubit_xeb import (
InferredXEBResult,
TwoQubitXEBResult,
parallel_two_qubit_xeb,
)
190 changes: 177 additions & 13 deletions cirq-core/cirq/experiments/two_qubit_xeb.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,10 @@
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from typing import Sequence, TYPE_CHECKING, Optional, Tuple, Dict
from typing import Sequence, TYPE_CHECKING, Optional, Tuple, Dict, cast, Mapping

from dataclasses import dataclass
from types import MappingProxyType
import itertools
import functools

Expand All @@ -22,25 +23,24 @@
import numpy as np
import pandas as pd

from cirq import ops, devices, value, vis
from cirq import ops, value, vis
from cirq.experiments.xeb_sampling import sample_2q_xeb_circuits
from cirq.experiments.xeb_fitting import benchmark_2q_xeb_fidelities
from cirq.experiments.xeb_fitting import fit_exponential_decays, exponential_decay
from cirq.experiments import random_quantum_circuit_generation as rqcg
from cirq.experiments.qubit_characterizations import ParallelRandomizedBenchmarkingResult
from cirq.qis import noise_utils
from cirq._compat import cached_method

if TYPE_CHECKING:
import cirq


def _grid_qubits_for_sampler(sampler: 'cirq.Sampler'):
def _grid_qubits_for_sampler(sampler: 'cirq.Sampler') -> Optional[Sequence['cirq.GridQubit']]:
if hasattr(sampler, 'processor'):
device = sampler.processor.get_device()
return sorted(device.metadata.qubit_set)
else:
qubits = devices.GridQubit.rect(3, 2, 4, 3)
# Delete one qubit from the rectangular arangement to
# 1) make it irregular 2) simplify simulation.
return qubits[:-1]
return None


def _manhattan_distance(qubit1: 'cirq.GridQubit', qubit2: 'cirq.GridQubit') -> int:
Expand All @@ -65,7 +65,7 @@ def all_qubit_pairs(self) -> Tuple[Tuple['cirq.GridQubit', 'cirq.GridQubit'], ..
return tuple(sorted(self._qubit_pair_map.keys()))

def plot_heatmap(self, ax: Optional[plt.Axes] = None, **plot_kwargs) -> plt.Axes:
"""plot the heatmap for xeb error.
"""plot the heatmap of XEB errors.

Args:
ax: the plt.Axes to plot on. If not given, a new figure is created,
Expand All @@ -75,7 +75,6 @@ def plot_heatmap(self, ax: Optional[plt.Axes] = None, **plot_kwargs) -> plt.Axes
show_plot = not ax
if not isinstance(ax, plt.Axes):
fig, ax = plt.subplots(1, 1, figsize=(8, 8))

heatmap_data: Dict[Tuple['cirq.GridQubit', ...], float] = {
pair: self.xeb_error(*pair) for pair in self.all_qubit_pairs
}
Expand Down Expand Up @@ -131,10 +130,13 @@ def _record(self, q0, q1) -> pd.Series:
q0, q1 = q1, q0
return self.fidelities.iloc[self._qubit_pair_map[(q0, q1)]]

def xeb_fidelity(self, q0: 'cirq.GridQubit', q1: 'cirq.GridQubit') -> float:
"""Return the XEB fidelity of a qubit pair."""
return self._record(q0, q1).layer_fid

def xeb_error(self, q0: 'cirq.GridQubit', q1: 'cirq.GridQubit') -> float:
"""Return the XEB error of a qubit pair."""
p = self._record(q0, q1).layer_fid
return 1 - p
return 1 - self.xeb_fidelity(q0, q1)

def all_errors(self) -> Dict[Tuple['cirq.GridQubit', 'cirq.GridQubit'], float]:
"""Return the XEB error of all qubit pairs."""
Expand All @@ -156,9 +158,163 @@ def plot_histogram(self, ax: Optional[plt.Axes] = None, **plot_kwargs) -> plt.Ax
fig.show(**plot_kwargs)
return ax

@cached_method
def pauli_error(self) -> Dict[Tuple['cirq.GridQubit', 'cirq.GridQubit'], float]:
"""Return the Pauli error of all qubit pairs."""
return {
NoureldinYosri marked this conversation as resolved.
Show resolved Hide resolved
pair: noise_utils.decay_constant_to_pauli_error(
noise_utils.xeb_fidelity_to_decay_constant(self.xeb_fidelity(*pair), num_qubits=2),
num_qubits=2,
)
for pair in self.all_qubit_pairs
}


@dataclass(frozen=True)
class InferredXEBResult:
"""Uses the results from XEB and RB to compute inferred two-qubit Pauli errors."""

rb_result: ParallelRandomizedBenchmarkingResult
xeb_result: TwoQubitXEBResult

@property
def all_qubit_pairs(self) -> Sequence[Tuple['cirq.GridQubit', 'cirq.GridQubit']]:
return self.xeb_result.all_qubit_pairs

@cached_method
def single_qubit_pauli_error(self) -> Mapping['cirq.Qid', float]:
"""Return the single-qubit Pauli error for all qubits (RB results)."""
return self.rb_result.pauli_error()

@cached_method
def two_qubit_pauli_error(self) -> Mapping[Tuple['cirq.GridQubit', 'cirq.GridQubit'], float]:
"""Return the two-qubit Pauli error for all pairs."""
return MappingProxyType(self.xeb_result.pauli_error())

@cached_method
def inferred_pauli_error(self) -> Mapping[Tuple['cirq.GridQubit', 'cirq.GridQubit'], float]:
"""Return the inferred Pauli error for all pairs."""
single_q_paulis = self.rb_result.pauli_error()
xeb = self.xeb_result.pauli_error()

def _pauli_error(q0: 'cirq.GridQubit', q1: 'cirq.GridQubit') -> float:
q0, q1 = sorted([q0, q1])
return xeb[(q0, q1)] - single_q_paulis[q0] - single_q_paulis[q1]

return MappingProxyType({pair: _pauli_error(*pair) for pair in self.all_qubit_pairs})

@cached_method
def inferred_decay_constant(self) -> Mapping[Tuple['cirq.GridQubit', 'cirq.GridQubit'], float]:
"""Return the inferred decay constant for all pairs."""
return MappingProxyType(
{
pair: noise_utils.pauli_error_to_decay_constant(pauli, 2)
for pair, pauli in self.inferred_pauli_error().items()
}
)

@cached_method
def inferred_xeb_error(self) -> Mapping[Tuple['cirq.GridQubit', 'cirq.GridQubit'], float]:
"""Return the inferred XEB error for all pairs."""
return MappingProxyType(
{
pair: 1 - noise_utils.decay_constant_to_xeb_fidelity(decay, 2)
for pair, decay in self.inferred_decay_constant().items()
}
)

def _target_errors(
self, target_error: str
) -> Mapping[Tuple['cirq.GridQubit', 'cirq.GridQubit'], float]:
error_funcs = {
'pauli': self.inferred_pauli_error,
'decay_constant': self.inferred_decay_constant,
'xeb': self.inferred_xeb_error,
NoureldinYosri marked this conversation as resolved.
Show resolved Hide resolved
}
return error_funcs[target_error]()

def plot_heatmap(
self, target_error: str = 'pauli', ax: Optional[plt.Axes] = None, **plot_kwargs
) -> plt.Axes:
"""plot the heatmap of the target errors.

Args:
target_error: The error to draw. Must be one of 'xeb', 'pauli', or 'decay_constant'
ax: the plt.Axes to plot on. If not given, a new figure is created,
plotted on, and shown.
**plot_kwargs: Arguments to be passed to 'plt.Axes.plot'.
"""
show_plot = not ax
if not isinstance(ax, plt.Axes):
fig, ax = plt.subplots(1, 1, figsize=(8, 8))
heatmap_data = cast(
Mapping[Tuple['cirq.GridQubit', ...], float], self._target_errors(target_error)
)

name = f'{target_error} error' if target_error != 'decay_constant' else 'decay constant'
ax.set_title(f'device {name} heatmap')

vis.TwoQubitInteractionHeatmap(heatmap_data).plot(ax=ax, **plot_kwargs)
if show_plot:
fig.show()
return ax

def plot_histogram(
self,
target_error: str = 'pauli',
ax: Optional[plt.Axes] = None,
kind: str = 'two_qubit',
**plot_kwargs,
) -> plt.Axes:
"""plot a histogram of target error.

Args:
target_error: The error to draw. Must be one of 'xeb', 'pauli', or 'decay_constant'
kind: Whether to plot the single-qubit RB errors ('single_qubit') or the
two-qubit inferred errors ('two_qubit') or both ('both').
ax: the plt.Axes to plot on. If not given, a new figure is created,
plotted on, and shown.
**plot_kwargs: Arguments to be passed to 'plt.Axes.plot'.

Raises:
ValueError: If
- `kind` is not one of 'single_qubit', 'two_qubit', or 'both'.
- `target_error` is not one of 'pauli', 'xeb', or 'decay_constant'
- single qubit error is requested and `target_error` is not 'pauli'.
"""
if kind not in ('single_qubit', 'two_qubit', 'both'):
raise ValueError(
f"kind must be one of 'single_qubit', 'two_qubit', or 'both', not {kind}"
)
if kind != 'two_qubit' and target_error != 'pauli':
raise ValueError(f'{target_error} is not supported for single qubits')
fig = None
if ax is None:
fig, ax = plt.subplots(1, 1, figsize=(8, 8))

alpha = 1 / (1 + int(kind == 'both'))
NoureldinYosri marked this conversation as resolved.
Show resolved Hide resolved
if kind == 'single_qubit' or kind == 'both':
self.rb_result.plot_integrated_histogram(
ax=ax, alpha=alpha, label='single qubit', color='green', **plot_kwargs
)
if kind == 'two_qubit' or kind == 'both':
vis.integrated_histogram(
data=self._target_errors(target_error),
ax=ax,
alpha=alpha,
label='two qubit',
color='blue',
**plot_kwargs,
)

if fig is not None:
fig.show(**plot_kwargs)
return ax

NoureldinYosri marked this conversation as resolved.
Show resolved Hide resolved

def parallel_two_qubit_xeb(
sampler: 'cirq.Sampler',
qubits: Optional[Sequence['cirq.GridQubit']] = None,
entangling_gate: 'cirq.Gate' = ops.CZ,
n_repetitions: int = 10**4,
n_combinations: int = 10,
Expand All @@ -172,6 +328,7 @@ def parallel_two_qubit_xeb(

Args:
sampler: The quantum engine or simulator to run the circuits.
qubits: Qubits under test. If none, uses all qubits on the sampler's device.
entangling_gate: The entangling gate to use.
n_repetitions: The number of repetitions to use.
n_combinations: The number of combinations to generate.
Expand All @@ -184,10 +341,17 @@ def parallel_two_qubit_xeb(

Returns:
A TwoQubitXEBResult object representing the results of the experiment.

Raises:
ValueError: If qubits are not specified and the sampler has no device.
"""
rs = value.parse_random_state(random_state)

qubits = _grid_qubits_for_sampler(sampler)
if qubits is None:
qubits = _grid_qubits_for_sampler(sampler)
if qubits is None:
raise ValueError("Couldn't determine qubits from sampler. Please specify them.")

graph = nx.Graph(
pair for pair in itertools.combinations(qubits, 2) if _manhattan_distance(*pair) == 1
)
Expand Down
Loading
Loading