Source code for qutip.qip.compiler.gatecompiler

import numpy as np
from .instruction import Instruction
from .scheduler import Scheduler
from ..circuit import QubitCircuit, Gate


__all__ = ['GateCompiler']


[docs]class GateCompiler(object): """ Base class. It compiles a :class:`.QubitCircuit` into the pulse sequence for the processor. The core member function `compile` calls compiling method from the sub-class and concatenate the compiled pulses. Parameters ---------- N: int The number of the component systems. params: dict, optional A Python dictionary contains the name and the value of the parameters, such as laser frequency, detuning etc. It will be saved in the class attributes and can be used to calculate the control pulses. pulse_dict: dict, optional A map between the pulse label and its index in the pulse list. If given, the compiled pulse can be identified with ``(pulse_label, coeff)``, instead of ``(pulse_index, coeff)``. The number of key-value pairs should match the number of pulses in the processor. If it is empty, an integer ``pulse_index`` needs to be used in the compiling routine saved under the attributes ``gate_compiler``. Attributes ---------- gate_compiler: dict The Python dictionary in the form of {gate_name: compiler_function}. It saves the compiling routine for each gate. See sub-classes for implementation. Note that for continuous pulse, the first coeff should always be 0. args: dict Arguments for individual compiling routines. It adds more flexibility in customizing compiler. """ def __init__(self, N, params=None, pulse_dict=None): self.gate_compiler = {} self.N = N self.params = params if params is not None else {} self.pulse_dict = pulse_dict if pulse_dict is not None else {} self.gate_compiler = {"GLOBALPHASE": self.globalphase_compiler} self.args = {"params": self.params} self.global_phase = 0.
[docs] def globalphase_compiler(self, gate, args): """ Compiler for the GLOBALPHASE gate """ pass
[docs] def compile(self, circuit, schedule_mode=None, args=None): """ Compile the the native gates into control pulse sequence. It calls each compiling method and concatenates the compiled pulses. Parameters ---------- circuit: :class:`.QubitCircuit` or list of :class:`.Gate` A list of elementary gates that can be implemented in the corresponding hardware. The gate names have to be in `gate_compiler`. schedule_mode: str, optional ``"ASAP"`` for "as soon as possible" or ``"ALAP"`` for "as late as possible" or ``False`` or ``None`` for no schedule. Default is None. args: dict, optional A dictionary of arguments used in a specific gate compiler function. Returns ------- tlist: array_like A NumPy array specifies the time of each coefficient coeffs: array_like A 2d NumPy array of the shape ``(len(ctrls), len(tlist))``. Each row corresponds to the control pulse sequence for one Hamiltonian. """ if isinstance(circuit, QubitCircuit): gates = circuit.gates else: gates = circuit if args is not None: self.args.update(args) instruction_list = [] # compile gates for gate in gates: if gate.name not in self.gate_compiler: raise ValueError("Unsupported gate %s" % gate.name) instruction = self.gate_compiler[gate.name](gate, self.args) if instruction is None: continue # neglecting global phase gate instruction_list += instruction if not instruction_list: return None, None if self.pulse_dict: num_controls = len(self.pulse_dict) else: # if pulse_dict is not given, compute the number of pulses num_controls = 0 for instruction in instruction_list: for pulse_index, _ in instruction.pulse_info: num_controls = max(num_controls, pulse_index) num_controls += 1 # schedule # scheduled_start_time: # An ordered list of the start_time for each pulse, # corresponding to gates in the instruction_list. # instruction_list reordered according to the scheduled result instruction_list, scheduled_start_time = \ self._schedule(instruction_list, schedule_mode) # An instruction can be composed from several different pulse elements. # We separate them an assign them to each pulse index. pulse_instructions = [[] for tmp in range(num_controls)] for instruction, start_time in \ zip(instruction_list, scheduled_start_time): for pulse_name, coeff in instruction.pulse_info: if self.pulse_dict: try: pulse_ind = self.pulse_dict[pulse_name] except KeyError: raise ValueError( f"Pulse name {pulse_name} not found" " in pulse_dict.") else: pulse_ind = pulse_name pulse_instructions[pulse_ind].append( (start_time, instruction.tlist, coeff)) # concatenate pulses compiled_tlist, compiled_coeffs = \ self._concatenate_pulses( pulse_instructions, scheduled_start_time, num_controls) return compiled_tlist, compiled_coeffs
def _schedule(self, instruction_list, schedule_mode): """ Schedule the instructions if required and reorder instruction_list accordingly """ if schedule_mode: scheduler = Scheduler(schedule_mode) scheduled_start_time = scheduler.schedule(instruction_list) time_ordered_pos = np.argsort(scheduled_start_time) instruction_list = [instruction_list[i] for i in time_ordered_pos] scheduled_start_time.sort() else: # no scheduling scheduled_start_time = [0.] for instruction in instruction_list[:-1]: scheduled_start_time.append( instruction.duration + scheduled_start_time[-1]) return instruction_list, scheduled_start_time def _concatenate_pulses( self, pulse_instructions, scheduled_start_time, num_controls): """ Concatenate compiled pulses coefficients and tlist for each pulse. If there is idling time, add zeros properly to prevent wrong spline. """ # Concatenate tlist and coeffs for each control pulses compiled_tlist = [[] for tmp in range(num_controls)] compiled_coeffs = [[] for tmp in range(num_controls)] for pulse_ind in range(num_controls): last_pulse_time = 0. for start_time, tlist, coeff in pulse_instructions[pulse_ind]: # compute the gate time, step size and coeffs # according to different pulse mode gate_tlist, coeffs, step_size, pulse_mode = \ self._process_gate_pulse(start_time, tlist, coeff) if abs(last_pulse_time) < step_size * 1.0e-6: # if first pulse compiled_tlist[pulse_ind].append([0.]) if pulse_mode == "continuous": compiled_coeffs[pulse_ind].append([0.]) # for discrete pulse len(coeffs) = len(tlist) - 1 # If there is idling time between the last pulse and # the current one, we need to add zeros in between. if np.abs(start_time - last_pulse_time) > step_size * 1.0e-6: idling_tlist = self._process_idling_tlist( pulse_mode, start_time, last_pulse_time, step_size) compiled_tlist[pulse_ind].append(idling_tlist) compiled_coeffs[pulse_ind].append(np.zeros(len(idling_tlist))) # Add the gate time and coeffs to the list. execution_time = gate_tlist + start_time last_pulse_time = execution_time[-1] compiled_tlist[pulse_ind].append(execution_time) compiled_coeffs[pulse_ind].append(coeffs) for i in range(num_controls): if not compiled_coeffs[i]: compiled_tlist[i] = None compiled_coeffs[i] = None else: compiled_tlist[i] = np.concatenate(compiled_tlist[i]) compiled_coeffs[i] = np.concatenate(compiled_coeffs[i]) return compiled_tlist, compiled_coeffs def _process_gate_pulse( self, start_time, tlist, coeff): # compute the gate time, step size and coeffs # according to different pulse mode if np.isscalar(tlist): pulse_mode = "discrete" # a single constant rectanglar pulse, where # tlist and coeff are just float numbers step_size = tlist coeff = np.array([coeff]) gate_tlist = np.array([tlist]) elif len(tlist) - 1 == len(coeff): # discrete pulse pulse_mode = "discrete" step_size = tlist[1] - tlist[0] coeff = np.asarray(coeff) gate_tlist = np.asarray(tlist)[1:] # first t always 0 by def elif len(tlist) == len(coeff): # continuos pulse pulse_mode = "continuous" step_size = tlist[1] - tlist[0] coeff = np.asarray(coeff)[1:] gate_tlist = np.asarray(tlist)[1:] else: raise ValueError( "The shape of the compiled pulse is not correct.") return gate_tlist, coeff, step_size, pulse_mode def _process_idling_tlist( self, pulse_mode, start_time, last_pulse_time, step_size): idling_tlist = [] if pulse_mode == "continuous": # We add sufficient number of zeros at the begining # and the end of the idling to prevent wrong cubic spline. if start_time - last_pulse_time > 3 * step_size: idling_tlist1 = np.linspace( last_pulse_time + step_size/5, last_pulse_time + step_size, 5 ) idling_tlist2 = np.linspace( start_time - step_size, start_time, 5 ) idling_tlist.extend([idling_tlist1, idling_tlist2]) else: idling_tlist.append( np.arange( last_pulse_time + step_size, start_time, step_size ) ) elif pulse_mode == "discrete": # idling until the start time idling_tlist.append([start_time]) return np.concatenate(idling_tlist)