# Repository: https://gitlab.com/quantify-os/quantify-core
# Licensed according to the LICENCE file on the main branch
"""Models and fit functions to be used with the lmfit fitting framework."""
from __future__ import annotations
from typing import Tuple
import lmfit
import numpy as np
from numpy.typing import NDArray
[docs]
def get_model_common_doc() -> str:
"""
Returns a common docstring to be used with custom fitting
:class:`~lmfit.model.Model` s.
.. admonition:: Usage example for a custom fitting model
:class: dropdown, tip
See the usage example at the end of the :class:`~ResonatorModel` source-code:
.. literalinclude:: ../../../quantify_core/analysis/fitting_models.py
:pyobject: ResonatorModel
"""
return (
lmfit.models.COMMON_INIT_DOC.replace(
":class:`Model`", ":class:`~lmfit.model.Model`"
)
.replace("\n ", "\n")
.replace(" : str", " : :obj:`str`")
.replace("['x']", ":code:`['x']`")
.replace(", optional", "")
.replace(" optional", "")
.replace("{'raise', 'propagate', 'omit'}", "")
)
[docs]
def get_guess_common_doc() -> str:
"""
Returns a common docstring to be used for the :meth:`~lmfit.model.Model.guess`
method of custom fitting :class:`~lmfit.model.Model` s.
.. admonition:: Usage example for a custom fitting model
:class: dropdown, tip
See the usage example at the end of the :class:`~ResonatorModel` source-code:
.. literalinclude:: ../../../quantify_core/analysis/fitting_models.py
:pyobject: ResonatorModel
"""
return (
lmfit.models.COMMON_GUESS_DOC.replace(
" : Parameters", " : :class:`~lmfit.parameter.Parameters`"
)
.replace("\n ", "\n")
.replace(" optional", "")
.replace(" : array_like", " : :class:`~numpy.ndarray`")
)
[docs]
def mk_seealso(
function_name: str,
role: str = "func",
prefix: str = "\n\n",
module_location: str = ".",
) -> str:
"""
Returns a sphinx `seealso` pointing to a function.
Intended to be used for building custom fitting model docstrings.
.. admonition:: Usage example for a custom fitting model
:class: dropdown, tip
See the usage example at the end of the :class:`~ResonatorModel` source-code:
.. literalinclude:: ../../../quantify_core/analysis/fitting_models.py
:pyobject: ResonatorModel
Parameters
----------
function_name
name of the function to point to
role
a sphinx role, e.g. :code:`"func"`
prefix
string preceding the `seealso`
module_location
can be used to indicate a function outside this module, e.g.,
:code:`my_module.submodule` which contains the function.
Returns
-------
:
resulting string
"""
return f"{prefix}.. seealso:: :{role}:`~{module_location}{function_name}`\n"
# pylint: disable=too-many-arguments
# pylint: disable=invalid-name
[docs]
def hanger_func_complex_SI(
f: float,
fr: float,
Ql: float,
Qe: float,
A: float,
theta: float,
phi_v: float,
phi_0: float,
alpha: float = 1,
) -> complex:
r"""
This is the complex function for a hanger (lambda/4 resonator).
Parameters
----------
f:
frequency
fr:
resonance frequency
A:
background transmission amplitude
Ql:
loaded quality factor of the resonator
Qe:
magnitude of extrinsic quality factor :code:`Qe = |Q_extrinsic|`
theta:
phase of extrinsic quality factor (in rad)
phi_v:
phase to account for propagation delay to sample
phi_0:
phase to account for propagation delay from sample
alpha:
slope of signal around the resonance
Returns
-------
:
complex valued transmission
See eq. S4 from Bruno et al. (2015)
`ArXiv:1502.04082 <https://arxiv.org/abs/1502.04082>`_.
.. math::
S_{21} = A \left(1+\alpha \frac{f-f_r}{f_r} \right)
\left(1- \frac{\frac{Q_l}{|Q_e|}e^{i\theta} }{1+2iQ_l \frac{f-f_r}{f_r}} \right)
e^{i (\phi_v f + \phi_0)}
The loaded and extrinsic quality factors are related to the internal and coupled Q
according to:
.. math::
\frac{1}{Q_l} = \frac{1}{Q_c}+\frac{1}{Q_i}
and
.. math::
\frac{1}{Q_c} = \mathrm{Re}\left(\frac{1}{|Q_e|e^{-i\theta}}\right)
"""
slope_corr = 1 + alpha * (f - fr) / fr
hanger_contribution = 1 - Ql / Qe * np.exp(1j * theta) / (
1 + 2.0j * Ql * (f - fr) / fr
)
propagation_delay_corr = np.exp(1j * (phi_v * f + phi_0))
S21 = A * slope_corr * hanger_contribution * propagation_delay_corr
return S21
[docs]
def cos_func(
x: float,
frequency: float,
amplitude: float,
offset: float,
phase: float = 0,
) -> float:
r"""
An oscillating cosine function:
:math:`y = \mathrm{amplitude} \times \cos(2 \pi \times \mathrm{frequency} \times x + \mathrm{phase}) + \mathrm{offset}`
Parameters
----------
x:
The independent variable (time, for example)
frequency:
A generalized frequency (in units of inverse x)
amplitude:
Amplitude of the oscillation
offset:
Output signal vertical offset
phase:
Phase offset / rad
Returns
-------
:
Output signal magnitude
""" # pylint: disable=line-too-long
return amplitude * np.cos(2 * np.pi * frequency * x + phase) + offset
[docs]
def lorentzian_func(
x: float,
x0: float,
width: float,
a: float,
c: float,
) -> float:
r"""
A Lorentzian function.
.. math::
y = \frac{a*\mathrm{width}}{\pi(\mathrm{width}^2 + (x - x_0)^2)} + c
Parameters
----------
x:
independent variable
x0:
horizontal offset
width:
Lorenztian linewidth
a:
amplitude
c:
vertical offset
Returns
-------
:
Lorentzian function
"""
return a * width / (np.pi * ((x - x0) ** 2) + width**2) + c
[docs]
def exp_decay_func(
t: float,
tau: float,
amplitude: float,
offset: float,
n_factor: float,
) -> float:
r"""
This is a general exponential decay function:
:math:`y = \mathrm{amplitude} \times \exp\left(-(t/\tau)^\mathrm{n\_factor}\right) + \mathrm{offset}`
Parameters
----------
t:
time
tau:
decay time
amplitude:
amplitude of the exponential decay
offset:
asymptote of the exponential decay, the value at t=infinity
n_factor:
exponential decay factor
Returns
-------
:
Output of exponential function as a float
""" # pylint: disable=line-too-long
return amplitude * np.exp(-((t / tau) ** n_factor)) + offset
[docs]
def exp_damp_osc_func(
t: float,
tau: float,
n_factor: float,
frequency: float,
phase: float,
amplitude: float,
offset: float,
):
r"""
A sinusoidal oscillation with an exponentially decaying envelope function:
:math:`y = \mathrm{amplitude} \times \exp\left(-(t/\tau)^\mathrm{n\_factor}\right)(\cos(2\pi\mathrm{frequency}\times t + \mathrm{phase}) + \mathrm{oscillation_offset}) + \mathrm{exponential_offset}`
Parameters
----------
t:
time
tau:
decay time
n_factor:
exponential decay factor
frequency:
frequency of the oscillation
phase:
phase of the oscillation
amplitude:
initial amplitude of the oscillation
oscillation_offset:
vertical offset of cosine oscillation relative to exponential asymptote
exponential_offset:
offset of exponential asymptote
Returns
-------
:
Output of decaying cosine function as a float
""" # pylint: disable=line-too-long
oscillation = amplitude * (np.cos(2 * np.pi * frequency * t + phase))
exp_decay = np.exp(-((t / tau) ** n_factor))
osc_decay = oscillation * exp_decay + offset
return osc_decay
[docs]
class LorentzianModel(lmfit.model.Model):
"""
Model for data which follows a Lorentzian function.
Uses the function :func:`~lorentzian_func` as the
defining equation.
"""
# pylint: disable=empty-docstring
# pylint: disable=abstract-method
# pylint: disable=too-few-public-methods
def __init__(self, *args, **kwargs) -> None:
# pass in the defining equation so the user doesn't have to later.
super().__init__(lorentzian_func, *args, **kwargs)
self.set_param_hint("x0", vary=True)
self.set_param_hint("a", vary=True)
self.set_param_hint("c", vary=True)
self.set_param_hint("width", vary=True)
# pylint: disable=missing-function-docstring
[docs]
def guess(self, data: NDArray, **kws) -> lmfit.parameter.Parameters:
"""Guess some initial values for the model based on the data."""
x = kws.get("x")
if x is None:
return None # type: ignore
# Guess that the resonance is where the function takes its maximal
# value
x0_guess = x[np.argmax(data)]
self.set_param_hint("x0", value=x0_guess)
# assume the user isn't trying to fit just a small part of a resonance curve.
xmin = x.min()
xmax = x.max()
width_max = xmax - xmin
delta_x = np.diff(x) # assume f is sorted
min_delta_x = delta_x[delta_x > 0].min()
# assume data actually samples the resonance reasonably
width_min = min_delta_x
width_guess = np.sqrt(width_min * width_max) # geometric mean, why not?
self.set_param_hint("width", value=width_guess)
# The guess for the vertical offset is the mean absolute value of the data
c_guess = np.mean(data)
self.set_param_hint("c", value=c_guess)
# Calculate A_guess from difference between the peak and the backround level
a_guess = np.pi * width_guess * (np.max(data) - c_guess)
self.set_param_hint("a", value=a_guess)
params = self.make_params()
return lmfit.models.update_param_vals(params, self.prefix, **kws)
# This class is used a literal include in the docs so the pylint options are here
# pylint: disable=empty-docstring
# pylint: disable=abstract-method
# pylint: disable=too-few-public-methods
[docs]
class CosineModel(lmfit.model.Model):
"""
Exemplary lmfit model with a guess for a cosine.
.. note::
The :mod:`lmfit.models` module provides several fitting models that might fit
your needs out of the box.
"""
[docs]
def __init__(self, *args, **kwargs):
# pass in the model's equation
super().__init__(cos_func, *args, **kwargs)
# configure constraints that are independent from the data to be fitted
self.set_param_hint("frequency", min=0, vary=True) # enforce positive frequency
self.set_param_hint("amplitude", min=0, vary=True) # enforce positive amplitude
self.set_param_hint("offset", vary=True)
self.set_param_hint(
"phase", vary=True, min=-np.pi, max=np.pi
) # enforce phase range
# pylint: disable=missing-function-docstring
[docs]
def guess(self, data, x, **kws) -> lmfit.parameter.Parameters:
"""
Guess parameters based on the data
Parameters
----------
data: np.ndarray
Data to fit to
x: np.ndarray
Independet variable
"""
self.set_param_hint("offset", value=np.average(data))
self.set_param_hint("amplitude", value=(np.max(data) - np.min(data)) / 2)
# Guess frequency and phase using Fourier Transform
freq_guess, phase_guess = fft_freq_phase_guess(data, x)
phase_wrap = (phase_guess + np.pi) % (2 * np.pi) - np.pi
self.set_param_hint("frequency", value=freq_guess)
self.set_param_hint("phase", value=phase_wrap)
params = self.make_params()
return lmfit.models.update_param_vals(params, self.prefix, **kws)
# Same design patter is used in lmfit.models to inherit common docstrings.
# We adjust these common docstrings to our docs build pipeline
__init__.__doc__ = get_model_common_doc() + mk_seealso("cos_func")
guess.__doc__ = get_guess_common_doc()
[docs]
class ResonatorModel(lmfit.model.Model):
"""
Resonator model
Implementation and design patterns inspired by the
`complex resonator model example <https://lmfit.github.io/lmfit-py/examples/example_complex_resonator_model.html>`_
(`lmfit` documentation).
""" # pylint: disable=line-too-long
# pylint: disable=empty-docstring
# pylint: disable=abstract-method
[docs]
def __init__(self, *args, **kwargs):
# pass in the defining equation so the user doesn't have to later.
super().__init__(hanger_func_complex_SI, *args, **kwargs)
self.set_param_hint("Ql", min=0) # Enforce Q is positive
self.set_param_hint("Qe", min=0) # Enforce Q is positive
# Internal and coupled quality factor can be derived from fitted params
self.set_param_hint("Qi", expr="1./(1./Ql-1./Qe*cos(theta))", vary=False)
self.set_param_hint("Qc", expr="Qe/cos(theta)", vary=False)
# pylint: disable=too-many-locals
# pylint: disable=missing-function-docstring
[docs]
def guess(self, data, **kws) -> lmfit.parameter.Parameters:
f = kws.get("f")
if f is None:
return None
argmin_s21 = np.abs(data).argmin()
fmin = f.min()
fmax = f.max()
# guess that the resonance is the lowest point
fr_guess = f[argmin_s21]
# assume the user isn't trying to fit just a small part of a resonance curve.
Q_min = 0.1 * (fr_guess / (fmax - fmin))
delta_f = np.diff(f) # assume f is sorted
min_delta_f = delta_f[delta_f > 0].min()
Q_max = (
fr_guess / min_delta_f
) # assume data actually samples the resonance reasonably
Q_guess = np.sqrt(Q_min * Q_max) # geometric mean, why not?
(phi_0_guess, phi_v_guess) = resonator_phase_guess(
data, f
) # Come up with a guess for phase velocity
self.set_param_hint("fr", value=fr_guess, min=fmin, max=fmax)
self.set_param_hint("Ql", value=Q_guess * 1.01, min=Q_min, max=Q_max)
self.set_param_hint("Qe", value=Q_guess * 0.99, min=0)
self.set_param_hint("A", value=np.mean(abs(data)), min=0)
# The parameters below need a proper guess.
self.set_param_hint("theta", value=0, min=-np.pi / 2, max=np.pi / 2)
self.set_param_hint("phi_0", value=phi_0_guess)
self.set_param_hint("phi_v", value=phi_v_guess)
self.set_param_hint("alpha", value=0, min=-1, max=1)
params = self.make_params()
return lmfit.models.update_param_vals(params, self.prefix, **kws)
# Same design patter is used in lmfit.models
__init__.__doc__ = get_model_common_doc() + mk_seealso("hanger_func_complex_SI")
guess.__doc__ = get_guess_common_doc()
[docs]
class ExpDecayModel(lmfit.model.Model):
"""
Model for an exponential decay, such as a qubit T1 measurement.
"""
# pylint: disable=empty-docstring
# pylint: disable=abstract-method
# pylint: disable=too-few-public-methods
[docs]
def __init__(self, *args, **kwargs):
# pass in the defining equation so the user doesn't have to later.
super().__init__(exp_decay_func, *args, **kwargs)
self.set_param_hint("tau", min=0) # Enforce T1 is positive
self.set_param_hint("amplitude", vary=True)
self.set_param_hint("offset", vary=True)
self.set_param_hint("n_factor", expr="1", vary=False)
# pylint: disable=missing-function-docstring
[docs]
def guess(self, data, **kws) -> lmfit.parameter.Parameters:
delay = kws.get("delay")
if delay is None:
return None
# To guess the upper amplitude and offset,
# use the first and last values of the data
self.set_param_hint("offset", value=data[-1])
self.set_param_hint("amplitude", value=data[0] - data[-1])
# The guess for tau is somewhere in the middle of the time range
tau = np.median(delay)
self.set_param_hint("tau", value=tau, min=0)
params = self.make_params()
return lmfit.models.update_param_vals(params, self.prefix, **kws)
# Same design patter is used in lmfit.models
__init__.__doc__ = get_model_common_doc() + mk_seealso("exp_decay_func")
guess.__doc__ = get_guess_common_doc()
[docs]
class RabiModel(lmfit.model.Model):
r"""
Model for a Rabi oscillation as a function of the microwave drive amplitude.
Phase of oscillation is fixed at :math:`\pi` in order to ensure that the oscillation
is at a minimum when the drive amplitude is 0.
"""
# pylint: disable=empty-docstring
# pylint: disable=abstract-method
# pylint: disable=too-few-public-methods
[docs]
def __init__(self, *args, **kwargs):
# pass in the defining equation so the user doesn't have to later.
super().__init__(cos_func, *args, **kwargs)
# Enforce oscillation frequency is positive
self.set_param_hint("frequency", min=0)
# Fix the phase at pi so that the ouput is at a minimum when x=0
self.set_param_hint("phase", expr="3.141592653589793", vary=False)
# Pi-pulse amplitude can be derived from the oscillation frequency
self.set_param_hint("amp180", expr="1/(2*frequency)", vary=False)
# pylint: disable=missing-function-docstring
[docs]
def guess(self, data, **kws) -> lmfit.parameter.Parameters:
drive_amp = kws.get("drive_amp")
if drive_amp is None:
return None
amp_guess = abs(max(data) - min(data)) / 2 # amp is positive by convention
offs_guess = np.mean(data)
(freq_guess, _) = fft_freq_phase_guess(data, drive_amp)
self.set_param_hint("frequency", value=freq_guess, min=0)
self.set_param_hint("amplitude", value=amp_guess, min=0)
self.set_param_hint("offset", value=offs_guess)
params = self.make_params()
return lmfit.models.update_param_vals(params, self.prefix, **kws)
# Same design patter is used in lmfit.models
__init__.__doc__ = get_model_common_doc() + mk_seealso("cos_func")
guess.__doc__ = get_guess_common_doc()
[docs]
class DecayOscillationModel(lmfit.model.Model):
r"""
Model for a decaying oscillation which decays to a point with 0 offset from
the centre of the of the oscillation (as in a Ramsey experiment, for example).
"""
# pylint: disable=empty-docstring
# pylint: disable=abstract-method
# pylint: disable=too-few-public-methods
[docs]
def __init__(self, *args, **kwargs):
# pass in the defining equation so the user doesn't have to later.
super().__init__(exp_damp_osc_func, *args, **kwargs)
# Enforce oscillation frequency is positive
self.set_param_hint("frequency", min=0)
# Enforce amplitude is positive
self.set_param_hint("amplitude", min=0)
# Enforce decay time is positive
self.set_param_hint("tau", min=0)
# Fix the n_factor at 1
self.set_param_hint("n_factor", expr="1", vary=False)
# pylint: disable=missing-function-docstring
[docs]
def guess(self, data, **kws) -> lmfit.parameter.Parameters:
t = kws.get("t")
if t is None:
raise ValueError(
'Time variable "t" must be specified in order to guess parameters'
)
return None
amp_guess = abs(max(data) - min(data)) / 2 # amp is positive by convention
exp_offs_guess = np.mean(data)
tau_guess = 2 / 3 * np.max(t)
(freq_guess, phase_guess) = fft_freq_phase_guess(data, t)
self.set_param_hint("frequency", value=freq_guess, min=0)
self.set_param_hint("amplitude", value=amp_guess, min=0)
self.set_param_hint("offset", value=exp_offs_guess)
self.set_param_hint("phase", value=phase_guess)
self.set_param_hint("tau", value=tau_guess, min=0)
params = self.make_params()
return lmfit.models.update_param_vals(params, self.prefix, **kws)
# Same design patter is used in lmfit.models
__init__.__doc__ = get_model_common_doc() + mk_seealso("exp_damp_osc_func")
guess.__doc__ = get_guess_common_doc()
[docs]
def resonator_phase_guess(s21: np.ndarray, freq: np.ndarray) -> Tuple[float, float]:
"""
Guesses the phase velocity in resonator spectroscopy,
based on the median of all the differences between consecutive phases.
Parameters
----------
s21:
Resonator S21 data
freq:
Frequency of the spectroscopy pulse
Returns
-------
phi_0:
Guess for the phase offset
phi_v:
Guess for the phase velocity
"""
phase = np.angle(s21)
med_diff = np.median(np.diff(phase))
freq_step = np.median(np.diff(freq))
phi_v = med_diff / freq_step
phi_0 = phase[0] - phi_v * freq[0]
return phi_0, phi_v
[docs]
def fft_freq_phase_guess(data: np.ndarray, t: np.ndarray) -> Tuple[float, float]:
"""
Guess for a cosine fit using FFT, only works for evenly spaced points.
Parameters
----------
data:
Input data to FFT
t:
Independent variable (e.g. time)
Returns
-------
freq_guess:
Guess for the frequency of the cosine function
ph_guess:
Guess for the phase of the cosine function
"""
# Only first half of array is used, because the second half contains the
# negative frequecy components, and we want a positive frequency.
power = np.fft.fft(data)[: len(data) // 2]
freq = np.fft.fftfreq(len(data), t[1] - t[0])[: len(power)]
power[0] = 0 # Removes DC component from fourier transform
# Use absolute value of complex valued spectrum
abs_power = np.abs(power)
freq_guess = abs(freq[abs_power == max(abs_power)][0])
# the condition data == max(data) can have several solutions
# (for example when discretization is visible)
# to prevent errors we pick the first solution
ph_guess = 2 * np.pi - (2 * np.pi * t[data == max(data)] * freq_guess)[0]
return freq_guess, ph_guess