artiq/artiq/frontend/pdq2_client.py

118 lines
4.9 KiB
Python
Raw Normal View History

2015-03-20 10:34:29 +08:00
#!/usr/bin/python
# Copyright (C) 2012-2015 Robert Jordens <jordens@gmail.com>
2014-10-14 16:38:02 +08:00
import argparse
import time
from scipy import interpolate
import numpy as np
from artiq.protocols.pc_rpc import Client
2015-01-28 21:44:15 +08:00
from artiq.tools import verbosity_args, init_logger
2014-10-14 16:38:02 +08:00
2015-01-23 00:52:13 +08:00
def get_argparser():
2014-10-25 11:38:42 +08:00
parser = argparse.ArgumentParser(description="""PDQ2 client.
Evaluates times and voltages, interpolates and uploads
them to the controller.""")
parser.add_argument("-s", "--server", default="::1",
2014-10-27 20:34:33 +08:00
help="hostname or IP of the controller to connect to")
parser.add_argument("--port", default=3252, type=int,
2014-10-27 20:34:33 +08:00
help="TCP port to use to connect to the controller")
2014-10-14 16:38:02 +08:00
parser.add_argument("-c", "--channel", default=0, type=int,
2014-10-14 17:02:25 +08:00
help="channel: 3*board_num+dac_num [%(default)s]")
2014-10-14 16:38:02 +08:00
parser.add_argument("-f", "--frame", default=0, type=int,
2014-10-14 17:02:25 +08:00
help="frame [%(default)s]")
2015-03-20 10:34:29 +08:00
parser.add_argument("-t", "--times", default="np.arange(5)*1e-6",
2014-10-14 17:02:25 +08:00
help="sample times (s) [%(default)s]")
2015-03-20 12:26:12 +08:00
parser.add_argument("-u", "--voltages",
2014-10-14 17:02:25 +08:00
default="(1-np.cos(t/t[-1]*2*np.pi))/2",
help="sample voltages (V) [%(default)s]")
2015-03-20 10:34:29 +08:00
parser.add_argument("-a", "--aux", default=False, action="store_true",
help="axiliary digital output [%(default)%s]")
2014-10-14 16:38:02 +08:00
parser.add_argument("-o", "--order", default=3, type=int,
2015-03-20 10:34:29 +08:00
help="interpolation (0: const, 1: lin, 2: quad,"
" 3: cubic) [%(default)s]")
2014-10-14 16:38:02 +08:00
parser.add_argument("-p", "--plot", help="plot to file [%(default)s]")
parser.add_argument("-r", "--reset", default=False,
2014-10-14 17:02:25 +08:00
action="store_true", help="do reset before")
2015-03-20 10:34:29 +08:00
parser.add_argument("-m", "--dcm", default=False, action="store_true",
help="100MHz clock [%(default)s]")
parser.add_argument("-n", "--disarm", default=False, action="store_true",
help="disarm group [%(default)s]")
parser.add_argument("-e", "--free", default=False, action="store_true",
help="software trigger [%(default)s]")
parser.add_argument("-x", "--demo", default=False, action="store_true",
help="demo mode: pulse and chirp, 1V*ch+0.1V*frame"
" [%(default)s]")
2014-10-14 16:38:02 +08:00
parser.add_argument("-b", "--bit", default=False,
2014-10-14 17:02:25 +08:00
action="store_true", help="do bit test")
2015-01-28 21:44:15 +08:00
verbosity_args(parser)
2015-01-23 00:52:13 +08:00
return parser
2014-10-14 16:38:02 +08:00
2015-01-22 10:38:47 +08:00
def main():
2015-01-23 00:52:13 +08:00
args = get_argparser().parse_args()
2015-01-28 21:44:15 +08:00
init_logger(args)
2014-10-28 15:45:56 +08:00
dev = Client(args.server, args.port, "pdq2")
2014-10-14 16:38:02 +08:00
if args.reset:
2015-03-20 10:34:29 +08:00
dev.write(b"\x00\x00") # flush any escape
dev.cmd("RESET", True)
2014-10-14 16:38:02 +08:00
time.sleep(.1)
2015-03-20 10:34:29 +08:00
dev.cmd("START", False)
dev.cmd("ARM", False)
dev.cmd("DCM", args.dcm)
freq = 100e6 if args.dcm else 50e6
dev.set_freq(freq)
2014-10-25 11:38:42 +08:00
times = eval(args.times, globals(), {})
voltages = eval(args.voltages, globals(), dict(t=times))
2014-10-14 16:38:02 +08:00
if args.demo:
2015-03-20 10:34:29 +08:00
for ch, channel in enumerate(dev.channels):
entry = []
for fr in range(dev.channels[0].num_frames):
vi = .1*fr + ch + voltages
entry.append(channel.segment(times, vi, order=args.order,
end=False, aux=args.aux))
pi = 2*np.pi*(-.5 + .01*fr + .1*ch + 0*voltages)
2014-10-14 16:38:02 +08:00
fi = 10e6*times/times[-1]
2015-03-20 10:34:29 +08:00
channel.segment(2*times, voltages, pi, fi, trigger=False,
silence=True, aux=args.aux)
dev.write_channel(channel, entry)
2014-10-14 16:38:02 +08:00
elif args.bit:
v = [-1, 0, -1]
2015-03-20 10:34:29 +08:00
# for i in range(15):
# v.extend([(1 << i) - 1, 1 << i])
v = np.array(v)*dev.channels[0].max_out/dev.channels[0].max_val
t = np.arange(len(v))
for channel in dev.channels:
s = channel.segment(t, v, order=0, shift=15, stop=False,
trigger=False)
dev.write_channel(channel, [s for i in range(channel.num_frames)])
2014-10-14 16:38:02 +08:00
else:
2015-03-20 10:34:29 +08:00
c = dev.channels[args.channel]
map = [None] * c.num_frames
map[args.frame] = c.segment(times, voltages, order=args.order,
aux=args.aux)
dev.write_channel(c, map)
2014-10-14 16:38:02 +08:00
2015-03-20 10:34:29 +08:00
dev.cmd("START", True)
dev.cmd("ARM", not args.disarm)
dev.cmd("TRIGGER", args.free)
2014-10-14 16:38:02 +08:00
if args.plot:
from matplotlib import pyplot as plt
2015-03-20 10:34:29 +08:00
fig, ax = plt.subplots()
ax.plot(times, voltages, "xk", label="points")
if args.order > 0:
2014-10-14 16:38:02 +08:00
spline = interpolate.splrep(times, voltages, k=args.order)
2015-03-20 10:34:29 +08:00
ttimes = np.arange(0, times[-1], 1/freq)
2014-10-14 16:38:02 +08:00
vvoltages = interpolate.splev(ttimes, spline)
2015-03-20 10:34:29 +08:00
ax.plot(ttimes, vvoltages, ",b", label="interpolation")
2014-10-14 16:38:02 +08:00
fig.savefig(args.plot)
2015-03-20 10:34:29 +08:00
2014-10-14 16:38:02 +08:00
if __name__ == "__main__":
2015-01-22 10:38:47 +08:00
main()