forked from M-Labs/artiq
Basic simulation of Al spectroscopy
This commit is contained in:
commit
c3c83f86b6
104
artiq/sim.py
Normal file
104
artiq/sim.py
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
from random import Random
|
||||||
|
from operator import itemgetter
|
||||||
|
|
||||||
|
from artiq import units
|
||||||
|
|
||||||
|
class SequentialTimeContext:
|
||||||
|
def __init__(self, current_time):
|
||||||
|
self.current_time = current_time
|
||||||
|
self.block_duration = 0
|
||||||
|
|
||||||
|
def take_time(self, amount):
|
||||||
|
self.current_time += amount
|
||||||
|
self.block_duration += amount
|
||||||
|
|
||||||
|
class ParallelTimeContext:
|
||||||
|
def __init__(self, current_time):
|
||||||
|
self.current_time = current_time
|
||||||
|
self.block_duration = 0
|
||||||
|
|
||||||
|
def take_time(self, amount):
|
||||||
|
if amount > self.block_duration:
|
||||||
|
self.block_duration = amount
|
||||||
|
|
||||||
|
class TimeManager:
|
||||||
|
def __init__(self):
|
||||||
|
self.stack = [SequentialTimeContext(0)]
|
||||||
|
self.timeline = []
|
||||||
|
|
||||||
|
def enter_sequential(self):
|
||||||
|
new_context = SequentialTimeContext(self.stack[-1].current_time)
|
||||||
|
self.stack.append(new_context)
|
||||||
|
|
||||||
|
def enter_parallel(self):
|
||||||
|
new_context = ParallelTimeContext(self.stack[-1].current_time)
|
||||||
|
self.stack.append(new_context)
|
||||||
|
|
||||||
|
def exit(self):
|
||||||
|
old_context = self.stack.pop()
|
||||||
|
self.take_time(old_context.block_duration)
|
||||||
|
|
||||||
|
def take_time(self, amount):
|
||||||
|
self.stack[-1].take_time(amount)
|
||||||
|
|
||||||
|
def event(self, description):
|
||||||
|
self.timeline.append((self.stack[-1].current_time, description))
|
||||||
|
|
||||||
|
def format_timeline(self):
|
||||||
|
r = ""
|
||||||
|
prev_time = 0
|
||||||
|
for time, description in sorted(self.timeline, key=itemgetter(0)):
|
||||||
|
t = units.Quantity(time, units.base_s_unit)
|
||||||
|
dt = units.Quantity(time-prev_time, units.base_s_unit)
|
||||||
|
r += "@{:10} (+{:10}) ".format(str(t), str(dt))
|
||||||
|
for item in description:
|
||||||
|
r += "{:16}".format(str(item))
|
||||||
|
r += "\n"
|
||||||
|
prev_time = time
|
||||||
|
return r
|
||||||
|
|
||||||
|
# global namespace for interpreted kernels
|
||||||
|
|
||||||
|
time_manager = TimeManager()
|
||||||
|
prng = Random(42)
|
||||||
|
|
||||||
|
class _Sequential:
|
||||||
|
def __enter__(self):
|
||||||
|
time_manager.enter_sequential()
|
||||||
|
|
||||||
|
def __exit__(self, type, value, traceback):
|
||||||
|
time_manager.exit()
|
||||||
|
sequential = _Sequential()
|
||||||
|
|
||||||
|
class _Parallel:
|
||||||
|
def __enter__(self):
|
||||||
|
time_manager.enter_parallel()
|
||||||
|
|
||||||
|
def __exit__(self, type, value, traceback):
|
||||||
|
time_manager.exit()
|
||||||
|
parallel = _Parallel()
|
||||||
|
|
||||||
|
def delay(duration):
|
||||||
|
units.check_unit(duration, units.base_s_unit)
|
||||||
|
time_manager.take_time(duration.amount)
|
||||||
|
|
||||||
|
def wait_edge(input):
|
||||||
|
duration = prng.randrange(17)*units.ms
|
||||||
|
time_manager.event(("wait_edge", input, duration))
|
||||||
|
time_manager.take_time(duration.amount)
|
||||||
|
|
||||||
|
def pulse(output, frequency, duration):
|
||||||
|
units.check_unit(frequency, units.base_Hz_unit)
|
||||||
|
units.check_unit(duration, units.base_s_unit)
|
||||||
|
time_manager.event(("pulse", output, frequency, duration))
|
||||||
|
time_manager.take_time(duration.amount)
|
||||||
|
|
||||||
|
def count_gate(input, duration):
|
||||||
|
result = prng.randrange(100)
|
||||||
|
units.check_unit(duration, units.base_s_unit)
|
||||||
|
time_manager.event(("count_gate", input, duration, result))
|
||||||
|
time_manager.take_time(duration.amount)
|
||||||
|
return result
|
||||||
|
|
||||||
|
def set_dac_voltage(output):
|
||||||
|
time_manager.event(("set_dac_voltage", output))
|
52
artiq/units.py
Normal file
52
artiq/units.py
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
from collections import namedtuple
|
||||||
|
|
||||||
|
_prefixes_str = "pnum_kM"
|
||||||
|
|
||||||
|
Unit = namedtuple("Unit", "base_prefix name")
|
||||||
|
|
||||||
|
class DimensionError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class Quantity:
|
||||||
|
def __init__(self, amount, unit):
|
||||||
|
self.amount = int(amount)
|
||||||
|
self.unit = unit
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
r_amount = self.amount
|
||||||
|
r_prefix = self.unit.base_prefix
|
||||||
|
if r_amount:
|
||||||
|
while not r_amount % 1000 and r_prefix < len(_prefixes_str):
|
||||||
|
r_amount //= 1000
|
||||||
|
r_prefix += 1
|
||||||
|
return str(r_amount) + " " + _prefixes_str[r_prefix] + self.unit.name
|
||||||
|
|
||||||
|
def __add__(self, other):
|
||||||
|
if self.unit != other.unit:
|
||||||
|
raise DimensionError
|
||||||
|
return Quantity(self.amount + other.amount, self.unit)
|
||||||
|
__radd__ = __add__
|
||||||
|
|
||||||
|
def __rmul__(self, other):
|
||||||
|
if isinstance(other, Quantity):
|
||||||
|
return NotImplemented
|
||||||
|
return Quantity(self.amount*other, self.unit)
|
||||||
|
|
||||||
|
def check_unit(value, unit):
|
||||||
|
if not isinstance(value, Quantity) or value.unit != unit:
|
||||||
|
raise DimensionError
|
||||||
|
|
||||||
|
def _register_unit(base_prefix, name, prefixes):
|
||||||
|
base_prefix_exp = _prefixes_str.index(base_prefix)
|
||||||
|
unit = Unit(base_prefix_exp, name)
|
||||||
|
globals()["base_"+name+"_unit"] = unit
|
||||||
|
for prefix in prefixes:
|
||||||
|
prefix_exp = _prefixes_str.index(prefix)
|
||||||
|
exp_d = prefix_exp - base_prefix_exp
|
||||||
|
assert(exp_d >= 0)
|
||||||
|
quantity = Quantity(1000**exp_d, unit)
|
||||||
|
full_name = prefix + name if prefix != "_" else name
|
||||||
|
globals()[full_name] = quantity
|
||||||
|
|
||||||
|
_register_unit("p", "s", "pnum_")
|
||||||
|
_register_unit("_", "Hz", "_kM")
|
29
examples/al_spectroscopy.py
Normal file
29
examples/al_spectroscopy.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
from artiq.sim import *
|
||||||
|
from artiq.units import *
|
||||||
|
|
||||||
|
def al_clock_probe(spectroscopy_freq, A1, A2):
|
||||||
|
state_0_count = 0
|
||||||
|
for count in range(100):
|
||||||
|
wait_edge("mains_sync")
|
||||||
|
pulse("laser_cooling", 100*MHz, 100*us)
|
||||||
|
delay(5*us)
|
||||||
|
with parallel:
|
||||||
|
pulse("spectroscopy", spectroscopy_freq, 100*us)
|
||||||
|
with sequential:
|
||||||
|
delay(50*us)
|
||||||
|
set_dac_voltage("spectroscopy_b")
|
||||||
|
delay(5*us)
|
||||||
|
while True:
|
||||||
|
delay(5*us)
|
||||||
|
with parallel:
|
||||||
|
pulse("state_detection", 100*MHz, 10*us)
|
||||||
|
photon_count = count_gate("pmt", 10*us)
|
||||||
|
if photon_count < A1 or photon_count > A2:
|
||||||
|
break
|
||||||
|
if photon_count < A1:
|
||||||
|
state_0_count += 1
|
||||||
|
return state_0_count
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
al_clock_probe(30*MHz, 3, 30)
|
||||||
|
print(time_manager.format_timeline())
|
Loading…
Reference in New Issue
Block a user