Source code for scqubits.core.zeropi_full

# zeropi_full.py
#
# This file is part of scqubits.
#
#    Copyright (c) 2019, Jens Koch and Peter Groszkowski
#    All rights reserved.
#
#    This source code is licensed under the BSD-style license found in the
#    LICENSE file in the root directory of this source tree.
############################################################################

import os

import numpy as np
from scipy import sparse

import scqubits
import scqubits.core.central_dispatch as dispatch
import scqubits.core.descriptors as descriptors
import scqubits.core.discretization as discretization
from scqubits.core.noise import NoisySystem, NOISE_PARAMS
import scqubits.core.operators as op
import scqubits.core.qubit_base as base
import scqubits.io_utils.fileio_serializers as serializers
import scqubits.ui.qubit_widget as ui
import scqubits.utils.spectrum_utils as spec_utils


# - ZeroPi noise class

class NoisyFullZeroPi(NoisySystem):
    pass


[docs]class FullZeroPi(base.QubitBaseClass, serializers.Serializable, NoisyFullZeroPi): r"""Zero-Pi qubit [Brooks2013]_ [Dempster2014]_ including coupling to the zeta mode. The circuit is described by the Hamiltonian :math:`H = H_{0-\pi} + H_\text{int} + H_\zeta`, where .. math:: &H_{0-\pi} = -2E_\text{CJ}\partial_\phi^2+2E_{\text{C}\Sigma}(i\partial_\theta-n_g)^2 +2E_{C\Sigma}dC_J\,\partial_\phi\partial_\theta\\ &\qquad\qquad\qquad+2E_{C\Sigma}(\delta C_J/C_J)\partial_\phi\partial_\theta +2\,\delta E_J \sin\theta\sin(\phi-\varphi_\text{ext}/2)\\ &H_\text{int} = 2E_{C\Sigma}dC\,\partial_\theta\partial_\zeta + E_L dE_L \phi\,\zeta\\ &H_\zeta = E_{\zeta} a^\dagger a expressed in phase basis. The definition of the relevant charging energies :math:`E_\text{CJ}`, :math:`E_{\text{C}\Sigma}`, Josephson energies :math:`E_\text{J}`, inductive energies :math:`E_\text{L}`, and relative amounts of disorder :math:`dC_\text{J}`, :math:`dE_\text{J}`, :math:`dC`, :math:`dE_\text{L}` follows [Groszkowski2018]_. Internally, the ``FullZeroPi`` class formulates the Hamiltonian matrix via the product basis of the decoupled Zero-Pi qubit (see ``ZeroPi``) on one hand, and the zeta LC oscillator on the other hand. Parameters ---------- EJ: float mean Josephson energy of the two junctions EL: float inductive energy of the two (super-)inductors ECJ: float charging energy associated with the two junctions EC: float or None charging energy of the large shunting capacitances; set to `None` if `ECS` is provided instead dEJ: float relative disorder in EJ, i.e., (EJ1-EJ2)/EJavg dEL: float relative disorder in EL, i.e., (EL1-EL2)/ELavg dCJ: float relative disorder of the junction capacitances, i.e., (CJ1-CJ2)/CJavg dC: float relative disorder in large capacitances, i.e., (C1-C2)/Cavg ng: float offset charge associated with theta zeropi_cutoff: int cutoff in the number of states of the disordered zero-pi qubit zeta_cutoff: int cutoff in the zeta oscillator basis (Fock state basis) flux: float magnetic flux through the circuit loop, measured in units of flux quanta (h/2e) grid: Grid1d object specifies the range and spacing of the discretization lattice ncut: int charge number cutoff for `n_theta`, `n_theta = -ncut, ..., ncut` ECS: float, optional total charging energy including large shunting capacitances and junction capacitances; may be provided instead of EC truncated_dim: int, optional desired dimension of the truncated quantum system; expected: truncated_dim > 1 """ EJ = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') EL = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') ECJ = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') EC = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') ECS = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') dEJ = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') dCJ = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') ng = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') flux = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') grid = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') ncut = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi') zeropi_cutoff = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE', inner_object_name='_zeropi', attr_name='truncated_dim') dC = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE') dEL = descriptors.WatchedProperty('QUANTUMSYSTEM_UPDATE') def __init__(self, EJ, EL, ECJ, EC, dEJ, dCJ, dC, dEL, flux, ng, zeropi_cutoff, zeta_cutoff, grid, ncut, ECS=None, truncated_dim=None): self._zeropi = scqubits.ZeroPi( EJ=EJ, EL=EL, ECJ=ECJ, EC=EC, ng=ng, flux=flux, grid=grid, ncut=ncut, dEJ=dEJ, dCJ=dCJ, ECS=ECS, # the zeropi_cutoff defines the truncated_dim of the "base" zeropi object truncated_dim=zeropi_cutoff ) self.dC = dC self.dEL = dEL self.zeta_cutoff = zeta_cutoff self._sys_type = type(self).__name__ self.truncated_dim = truncated_dim self._evec_dtype = np.complex_ self._init_params.remove('ECS') # used for file IO Serializable purposes; remove ECS as init parameter self._image_filename = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'qubit_img/fullzeropi.jpg') dispatch.CENTRAL_DISPATCH.register('GRID_UPDATE', self)
[docs] @staticmethod def default_params(): return { 'EJ': 10.0, 'EL': 0.04, 'ECJ': 20.0, 'EC': 0.04, 'dEJ': 0.05, 'dCJ': 0.05, 'dC': 0.08, 'dEL': 0.05, 'ng': 0.1, 'flux': 0.23, 'ncut': 30, 'zeropi_cutoff': 10, 'zeta_cutoff': 40, 'truncated_dim': 10 }
[docs] @classmethod def create(cls): phi_grid = discretization.Grid1d(-25.0, 25.0, 360) init_params = cls.default_params() zeropi = cls(**init_params, grid=phi_grid) zeropi.widget() return zeropi
[docs] def supported_noise_channels(self): """Return a list of supported noise channels""" return ['tphi_1_over_f_cc', 'tphi_1_over_f_flux' 't1_bias_flux_line' # 't1_capacitive', 't1_inductive', ]
[docs] def widget(self, params=None): init_params = params or self.get_initdata() del init_params['grid'] init_params['grid_max_val'] = self.grid.max_val init_params['grid_min_val'] = self.grid.min_val init_params['grid_pt_count'] = self.grid.pt_count ui.create_widget(self.set_params, init_params, image_filename=self._image_filename)
[docs] def set_params(self, **kwargs): phi_grid = discretization.Grid1d(kwargs.pop('grid_min_val'), kwargs.pop('grid_max_val'), kwargs.pop('grid_pt_count')) self.grid = phi_grid for param_name, param_val in kwargs.items(): setattr(self, param_name, param_val)
[docs] def receive(self, event, sender, **kwargs): if sender is self._zeropi.grid: self.broadcast('QUANTUMSYSTEM_UPDATE')
def __str__(self): output_str = super().__str__() + '\n\n' output_str += 'INTERNAL 0-Pi object: ' + self._zeropi.__str__() return output_str
[docs] def set_EC_via_ECS(self, ECS): """Helper function to set `EC` by providing `ECS`, keeping `ECJ` constant.""" self._zeropi.set_EC_via_ECS(ECS)
@property def E_zeta(self): """Returns energy quantum of the zeta mode""" return (8.0 * self.EL * self.EC) ** 0.5 def set_E_zeta(self, value): raise ValueError("It's not possible to directly set `E_zeta`. Instead one can set its value through `EL` or `EC`.")
[docs] def hamiltonian(self, return_parts=False): """Returns Hamiltonian in basis obtained by discretizing phi, employing charge basis for theta, and Fock basis for zeta. Parameters ---------- return_parts: bool, optional If set to true, `hamiltonian` returns [hamiltonian, evals, evecs, g_coupling_matrix] Returns ------- scipy.sparse.csc_matrix or list """ zeropi_dim = self.zeropi_cutoff zeropi_evals, zeropi_evecs = self._zeropi.eigensys(evals_count=zeropi_dim) zeropi_diag_hamiltonian = sparse.dia_matrix((zeropi_dim, zeropi_dim), dtype=np.complex_) zeropi_diag_hamiltonian.setdiag(zeropi_evals) zeta_dim = self.zeta_cutoff prefactor = self.E_zeta zeta_diag_hamiltonian = op.number_sparse(zeta_dim, prefactor) hamiltonian_mat = sparse.kron(zeropi_diag_hamiltonian, sparse.identity(zeta_dim, format='dia', dtype=np.complex_)) hamiltonian_mat += sparse.kron(sparse.identity(zeropi_dim, format='dia', dtype=np.complex_), zeta_diag_hamiltonian) gmat = self.g_coupling_matrix(zeropi_evecs) zeropi_coupling = sparse.dia_matrix((zeropi_dim, zeropi_dim), dtype=np.complex_) for l1 in range(zeropi_dim): for l2 in range(zeropi_dim): zeropi_coupling += gmat[l1, l2] * op.hubbard_sparse(l1, l2, zeropi_dim) hamiltonian_mat += sparse.kron(zeropi_coupling, op.annihilation_sparse(zeta_dim)) + sparse.kron(zeropi_coupling.conjugate().T, op.creation_sparse(zeta_dim)) if return_parts: return [hamiltonian_mat.tocsc(), zeropi_evals, zeropi_evecs, gmat] return hamiltonian_mat.tocsc()
[docs] def d_hamiltonian_d_flux(self, zeropi_evecs=None): r"""Calculates a derivative of the Hamiltonian w.r.t flux, at the current value of flux, as stored in the object. The returned operator is in the product basis The flux is assumed to be given in the units of the ratio \Phi_{ext}/\Phi_0. So if \frac{\partial H}{ \partial \Phi_{\rm ext}}, is needed, the expression returned by this function, needs to be multiplied by 1/\Phi_0. Returns ------- scipy.sparse.csc_matrix matrix representing the derivative of the Hamiltonian """ return self._zeropi_operator_in_product_basis(self._zeropi.d_hamiltonian_d_flux(), zeropi_evecs=zeropi_evecs)
[docs] def d_hamiltonian_d_EJ(self, zeropi_evecs=None): r"""Calculates a derivative of the Hamiltonian w.r.t EJ. Returns ------- scipy.sparse.csc_matrix matrix representing the derivative of the Hamiltonian """ return self._zeropi_operator_in_product_basis(self._zeropi.d_hamiltonian_d_EJ(), zeropi_evecs=zeropi_evecs)
[docs] def d_hamiltonian_d_ng(self): r"""Calculates a derivative of the Hamiltonian w.r.t ng. as stored in the object. Returns ------- scipy.sparse.csc_matrix matrix representing the derivative of the Hamiltonian """ return -8 * self.EC * self.n_theta_operator()
def _zeropi_operator_in_product_basis(self, zeropi_operator, zeropi_evecs=None): """Helper method that converts a zeropi operator into one in the product basis. Returns ------- scipy.sparse.csc_matrix operator written in the product basis """ zeropi_dim = self.zeropi_cutoff zeta_dim = self.zeta_cutoff if zeropi_evecs is None: _, zeropi_evecs = self._zeropi.eigensys(evals_count=zeropi_dim) op_eigen_basis = sparse.dia_matrix((zeropi_dim, zeropi_dim), dtype=np.complex_) # is this guaranteed to be zero? op_zeropi = spec_utils.get_matrixelement_table(zeropi_operator, zeropi_evecs) for n in range(zeropi_dim): for m in range(zeropi_dim): op_eigen_basis += op_zeropi[n, m] * op.hubbard_sparse(n, m, zeropi_dim) return sparse.kron(op_eigen_basis, sparse.identity(zeta_dim, format='csc', dtype=np.complex_), format='csc')
[docs] def i_d_dphi_operator(self, zeropi_evecs=None): r""" Operator :math:`i d/d\phi`. Returns ------- scipy.sparse.csc_matrix """ return self._zeropi_operator_in_product_basis(self._zeropi.i_d_dphi_operator(), zeropi_evecs=zeropi_evecs)
[docs] def n_theta_operator(self, zeropi_evecs=None): r""" Operator :math:`n_\theta`. Returns ------- scipy.sparse.csc_matrix """ return self._zeropi_operator_in_product_basis(self._zeropi.n_theta_operator(), zeropi_evecs=zeropi_evecs)
[docs] def phi_operator(self, zeropi_evecs=None): r""" Operator :math:`\phi`. Returns ------- scipy.sparse.csc_matrix """ return self._zeropi_operator_in_product_basis(self._zeropi.phi_operator(), zeropi_evecs=zeropi_evecs)
[docs] def hilbertdim(self): """Returns Hilbert space dimension Returns ------- int """ return self.zeropi_cutoff * self.zeta_cutoff
def _evals_calc(self, evals_count, hamiltonian_mat=None): if hamiltonian_mat is None: hamiltonian_mat = self.hamiltonian() evals = sparse.linalg.eigsh(hamiltonian_mat, k=evals_count, sigma=0.0, which='LM', return_eigenvectors=False) return np.sort(evals) def _esys_calc(self, evals_count, hamiltonian_mat=None): if hamiltonian_mat is None: hamiltonian_mat = self.hamiltonian() evals, evecs = sparse.linalg.eigsh(hamiltonian_mat, k=evals_count, sigma=0.0, which='LM', return_eigenvectors=True) evals, evecs = spec_utils.order_eigensystem(evals, evecs) return evals, evecs
[docs] def g_phi_coupling_matrix(self, zeropi_states): """Returns a matrix of coupling strengths g^\\phi_{ll'} [cmp. Dempster et al., Eq. (18)], using the states from the list `zeropi_states`. Most commonly, `zeropi_states` will contain eigenvectors of the `DisorderedZeroPi` type. """ prefactor = self.EL * (self.dEL / 2.0) * (8.0 * self.EC / self.EL) ** 0.25 return prefactor * spec_utils.get_matrixelement_table(self._zeropi.phi_operator(), zeropi_states)
[docs] def g_theta_coupling_matrix(self, zeropi_states): """Returns a matrix of coupling strengths i*g^\\theta_{ll'} [cmp. Dempster et al., Eq. (17)], using the states from the list 'zeropi_states'. """ prefactor = 1j * self.ECS * (self.dC / 2.0) * (32.0 * self.EL / self.EC) ** 0.25 return prefactor * spec_utils.get_matrixelement_table(self._zeropi.n_theta_operator(), zeropi_states)
[docs] def g_coupling_matrix(self, zeropi_states=None, evals_count=None): """Returns a matrix of coupling strengths g_{ll'} [cmp. Dempster et al., text above Eq. (17)], using the states from 'zeropi_states'. If `zeropi_states==None`, then a set of `self.zeropi` eigenstates is calculated. Only in that case is `which` used for the eigenstate number (and hence the coupling matrix size). """ if evals_count is None: evals_count = self._zeropi.truncated_dim if zeropi_states is None: _, zeropi_states = self._zeropi.eigensys(evals_count=evals_count) return self.g_phi_coupling_matrix(zeropi_states) + self.g_theta_coupling_matrix(zeropi_states)