Keysight M3202A AWG example with digitizer¶
The Keysight M3202A is a 1 GSa/a arbitrary waveform generator.
An instance of M3202A represents a module with 4 output channels. The channels of one module share the uploaded waveforms and trigers.
This example loads loads waveforms in AWGs in slots 2 and 3 of chassis 0.
This example uses a digitizer from the keysightSD1 module to read the generated signals. The digitizer is in slot 5. The output channels 1 and 2 of the AWG in slot 2 should be connected to inputs 1 and 2 of the digitizer and output channels 3 and 4 of the AWG slot 3 should be conencted to inputs 3 and 4 of the digitizer.
[ ]:
import logging
import numpy as np
import matplotlib.pyplot as plt
try:
import keysightSD1
except:
# add the path where the keysight library probably resides and try again
import sys
sys.path.append(r'C:\Program Files (x86)\Keysight\SD1\Libraries\Python')
import keysightSD1
import qcodes
from qcodes_contrib_drivers.drivers.Keysight.M3202A import M3202A
import qcodes.logger as logger
from qcodes.logger import start_all_logging
start_all_logging()
# logger.get_file_handler().setLevel(logging.DEBUG)
[ ]:
# try to close station from previous run.
try:
station.close_all_registered_instruments()
except: pass
[ ]:
# Demonstration of some errors
try:
wrong_chassis = M3202A("wrong chassis", chassis = 5, slot = 16)
except Exception as e:
print('Expected failure:', e)
try:
empty_slot = M3202A("empty_slot", chassis = 0, slot = 16)
except Exception as e:
print('Expected failure:', e)
try:
not_M3202A = M3202A("Digitizer", chassis = 0, slot = 5)
except Exception as e:
print('Expected failure:', e)
[ ]:
awg1 = M3202A("AWG1", chassis = 0, slot = 2)
awg2 = M3202A("AWG2", chassis = 0, slot = 3)
station = qcodes.Station()
station.add_component(awg1)
station.add_component(awg2)
[ ]:
# utility functions to create waveforms
def get_divider(prescaler):
if prescaler == 0:
divider = 1
elif prescaler == 1:
divider = 5
else:
divider = prescaler * 10
return divider
def create_sawtooth(period, repetition, prescaler):
divider = get_divider(prescaler)
n_pts = period // divider
w = np.linspace(-1, 1, n_pts)
w = np.tile(w, repetition)
if len(w) < 2000:
raise Exception('not enough data')
return w
def create_sine(period, repetition, prescaler):
divider = get_divider(prescaler)
n_pts = repetition * period // divider
phi = np.linspace(0, np.pi*2*repetition, n_pts)
w = np.sin(phi)
if len(w) < 2000:
raise Exception('not enough data')
return w
[ ]:
# setup output channels
pxi1 = keysightSD1.SD_TriggerExternalSources.TRIGGER_PXI1
trigger_mode = keysightSD1.SD_TriggerBehaviors.TRIGGER_FALL
for awg in [awg1, awg2]:
for ch in range(1,5):
awg.set_channel_offset(0.0, ch)
awg.set_channel_amplitude(1.0, ch)
awg.set_channel_wave_shape(keysightSD1.SD_Waveshapes.AOU_AWG, ch)
awg.awg_queue_config(ch, keysightSD1.SD_QueueMode.CYCLIC)
awg.awg_config_external_trigger(ch, pxi1, trigger_mode)
awg2.amplitude_channel_1.set(0.5)
awg2.offset_channel_2.set(0.5)
Generate and enqueue waveforms¶
Signal duration 20 us:
“AWG1.1”: 20x sawtooth (followed by zeros)
“AWG1.2”: 20 steps (followed by zeros)
“AWG2.3”: 2 us zero, 4 us sine, 2 us zero, 12 us sine
“AWG2.4”: 4 us sine, 4 us zero, 8 us sine, 4 us zero
[ ]:
prescaler_1GSa = 0
prescaler_200MSa = 1
prescaler_50MSa = 2
# 2000 zero's for end
zeros = np.zeros(2000)
# sawtooth period = 1 us: 2 period in 1 sample
sawtooth_2us = 0.5 * create_sawtooth(1000, 2, prescaler_1GSa)
# 20 steps use prescaler to reduce total number of samples. 200MSa/s => 4000 pts; 200 per step
steps = np.concatenate([0.02*i*np.ones(200) for i in range(20)])
# sine wave 2 MHz, 4 periods
sine = 0.6 * create_sine(500, 4, prescaler_1GSa)
# start all uploads
zeros_awg1 = awg1.upload_waveform(zeros)
sawtooth_2us_awg1 = awg1.upload_waveform(sawtooth_2us)
steps_awg1 = awg1.upload_waveform(steps)
zeros_awg2 = awg2.upload_waveform(zeros)
sine_awg2 = awg2.upload_waveform(sine)
# enqueue wave references
delay = 0
ext_trigger = keysightSD1.SD_TriggerModes.EXTTRIG
auto_trigger = keysightSD1.SD_TriggerModes.AUTOTRIG
awg1.awg_queue_waveform(1, sawtooth_2us_awg1, ext_trigger, delay, 10, prescaler_1GSa)
awg1.awg_queue_waveform(1, zeros_awg1, auto_trigger, delay, 1, prescaler_1GSa)
awg1.awg_queue_waveform(2, steps_awg1, ext_trigger, delay, 1, prescaler_200MSa)
awg1.awg_queue_waveform(2, zeros_awg1, auto_trigger, delay, 1, prescaler_1GSa)
awg2.awg_queue_waveform(3, zeros_awg2, ext_trigger, delay, 1, prescaler_1GSa)
awg2.awg_queue_waveform(3, sine_awg2, auto_trigger, delay, 2, prescaler_1GSa)
awg2.awg_queue_waveform(3, zeros_awg2, auto_trigger, delay, 1, prescaler_1GSa)
awg2.awg_queue_waveform(3, sine_awg2, auto_trigger, delay, 6, prescaler_1GSa)
awg2.awg_queue_waveform(4, sine_awg2, ext_trigger, delay, 2, prescaler_1GSa)
awg2.awg_queue_waveform(4, zeros_awg2, auto_trigger, delay, 2, prescaler_1GSa)
awg2.awg_queue_waveform(4, sine_awg2, auto_trigger, delay, 4, prescaler_1GSa)
awg2.awg_queue_waveform(4, zeros_awg2, auto_trigger, delay, 2, prescaler_1GSa)
# start AWGs. They will wait for external trigger.
awg1.awg_start_multiple(0b0011)
awg2.awg_start_multiple(0b1100)
configure digitizer to capture signals¶
[ ]:
def check_error(error, s=''):
if (type(error) is int and error < 0):
print(keysightSD1.SD_Error.getErrorMessage(error), error, s)
PRODUCT = ""
CHASSIS = 0
# change slot numbers to your values
SLOT_IN = 5
NUM_CHANNELS = 4
# CREATE AND OPEN MODULE IN
moduleIn = keysightSD1.SD_AIN()
moduleInID = moduleIn.openWithSlot(PRODUCT, CHASSIS, SLOT_IN)
NUM_CYCLES = 1
DIG_PRESCALER = 0
# sample 21 us
IN_DURATION = 21
TOT_POINTS_IN = 500 * IN_DURATION
FULL_SCALE = 2.5
DELAY_IN = 200 // (DIG_PRESCALER + 1)
mask = 0
for ch in range(1,5):
moduleIn.DAQstop(ch)
moduleIn.DAQflush(ch)
for c in range(0, NUM_CHANNELS):
mask |= 1 << c
ch = c + 1
# MODULE IN
moduleIn.channelInputConfig(ch, FULL_SCALE, keysightSD1.AIN_Impedance.AIN_IMPEDANCE_HZ, keysightSD1.AIN_Coupling.AIN_COUPLING_DC)
moduleIn.channelPrescalerConfig(ch, DIG_PRESCALER)
moduleIn.DAQdigitalTriggerConfig(ch, pxi1, trigger_mode)
moduleIn.DAQconfig(ch, TOT_POINTS_IN, 1, DELAY_IN, ext_trigger)
moduleIn.DAQstartMultiple(mask)
Trigger AWGs and digitizer¶
[ ]:
# start AWGs and digitizers via PXI
# Note: PXI trigger can be set via any module
awg1.set_pxi_trigger(0, pxi1)
awg1.set_pxi_trigger(1, pxi1)
Retrieve digitizer data and plot result¶
[ ]:
POINTS_PER_READ = 20000
READ_TIMEOUT = 1000
numReadPoints = []
dataPoints = []
for c in range(0, NUM_CHANNELS):
numReadPoints.append(0)
dataPoints.append(np.empty(0, dtype=np.short))
readDone = False
print()
print("Reading data...")
cnt = 0
while not readDone:
if (cnt > 10):
print('no more data')
break
readDone = True
for c in range(0, NUM_CHANNELS):
npts = moduleIn.DAQcounterRead(c + 1)
if npts <= 0:
print(f'channel {c} counter {npts} {numReadPoints[c]}')
cnt += 1
else:
readPoints = moduleIn.DAQread(c + 1, min(POINTS_PER_READ, npts), READ_TIMEOUT)
check_error(readPoints)
if type(readPoints) is int or readPoints.size == 0:
cnt += 1
else:
cnt = 0
dataPoints[c] = np.append(dataPoints[c], readPoints)
numReadPoints[c] += readPoints.size
print(f'channel {c} counter {npts} read {readPoints.size} ({numReadPoints[c]})')
readDone = readDone and (numReadPoints[c] >= TOT_POINTS_IN)
for i in range(1,5):
fig = plt.figure(i)
fig.clear()
plt.plot(dataPoints[i-1])
plt.show()