Source code for qibocal.protocols.coherence.t2_signal

from dataclasses import dataclass
from typing import Union

import numpy as np
import plotly.graph_objects as go
from qibolab import AcquisitionType, AveragingMode, ExecutionParameters
from qibolab.platform import Platform
from qibolab.pulses import PulseSequence
from qibolab.qubits import QubitId
from qibolab.sweeper import Parameter, Sweeper, SweeperType

from qibocal import update
from qibocal.auto.operation import Parameters, Results, Routine

from ..utils import table_dict, table_html
from . import t1_signal, t2, utils


[docs]@dataclass class T2SignalParameters(Parameters): """T2Signal runcard inputs.""" delay_between_pulses_start: int """Initial delay between RX(pi/2) pulses in ns.""" delay_between_pulses_end: int """Final delay between RX(pi/2) pulses in ns.""" delay_between_pulses_step: int """Step delay between RX(pi/2) pulses in ns.""" single_shot: bool = False """If ``True`` save single shot signal data."""
[docs]@dataclass class T2SignalResults(Results): """T2Signal outputs.""" t2: dict[QubitId, Union[float, list[float]]] """T2 for each qubit [ns].""" fitted_parameters: dict[QubitId, dict[str, float]] """Raw fitting output.""" pcov: dict[QubitId, list[float]] """Approximate covariance of fitted parameters."""
[docs]class T2SignalData(t1_signal.T1SignalData): """T2Signal acquisition outputs.""" t2: dict[QubitId, float] """T2 for each qubit [ns].""" fitted_parameters: dict[QubitId, dict[str, float]] """Raw fitting output."""
[docs]def _acquisition( params: T2SignalParameters, platform: Platform, targets: list[QubitId], ) -> T2SignalData: """Data acquisition for Ramsey Experiment (detuned).""" # create a sequence of pulses for the experiment # RX90 - t - RX90 - MZ ro_pulses = {} RX90_pulses1 = {} RX90_pulses2 = {} sequence = PulseSequence() for qubit in targets: RX90_pulses1[qubit] = platform.create_RX90_pulse(qubit, start=0) RX90_pulses2[qubit] = platform.create_RX90_pulse( qubit, start=RX90_pulses1[qubit].finish, ) ro_pulses[qubit] = platform.create_qubit_readout_pulse( qubit, start=RX90_pulses2[qubit].finish ) sequence.add(RX90_pulses1[qubit]) sequence.add(RX90_pulses2[qubit]) sequence.add(ro_pulses[qubit]) # define the parameter to sweep and its range: waits = np.arange( # wait time between RX90 pulses params.delay_between_pulses_start, params.delay_between_pulses_end, params.delay_between_pulses_step, ) sweeper = Sweeper( Parameter.start, waits, [RX90_pulses2[qubit] for qubit in targets], type=SweeperType.ABSOLUTE, ) # execute the sweep results = platform.sweep( sequence, ExecutionParameters( nshots=params.nshots, relaxation_time=params.relaxation_time, acquisition_type=AcquisitionType.INTEGRATION, averaging_mode=( AveragingMode.SINGLESHOT if params.single_shot else AveragingMode.CYCLIC ), ), sweeper, ) data = T2SignalData() for qubit in targets: result = results[ro_pulses[qubit].serial] if params.single_shot: _waits = np.array(len(result.magnitude) * [waits]) else: _waits = waits data.register_qubit( utils.CoherenceType, (qubit), dict(wait=_waits, signal=result.magnitude, phase=result.phase), ) return data
[docs]def _fit(data: T2SignalData) -> T2SignalResults: """The used model is .. math:: y = p_0 - p_1 e^{-x p_2}. """ data = data.average t2s, fitted_parameters, pcovs = utils.exponential_fit(data) return T2SignalResults(t2s, fitted_parameters, pcovs)
[docs]def _plot(data: T2SignalData, target: QubitId, fit: T2SignalResults = None): """Plotting function for Ramsey Experiment.""" data = data.average figures = [] fig = go.Figure() fitting_report = None qubit_data = data[target] fig.add_trace( go.Scatter( x=qubit_data.wait, y=qubit_data.signal, opacity=1, name="Signal", showlegend=True, legendgroup="Signal", ) ) if fit is not None: # add fitting trace waitrange = np.linspace( min(qubit_data.wait), max(qubit_data.wait), 2 * len(qubit_data), ) params = fit.fitted_parameters[target] fig.add_trace( go.Scatter( x=waitrange, y=utils.exp_decay( waitrange, *params, ), name="Fit", line=go.scatter.Line(dash="dot"), ) ) fitting_report = table_html( table_dict( target, ["T2 [ns]"], [np.round(fit.t2[target])], display_error=True ) ) fig.update_layout( showlegend=True, xaxis_title="Time [ns]", yaxis_title="Signal [a.u.]", ) figures.append(fig) return figures, fitting_report
[docs]def _update(results: T2SignalResults, platform: Platform, target: QubitId): update.t2(results.t2[target], platform, target)
t2_signal = Routine(_acquisition, _fit, _plot, _update) """T2Signal Routine object."""