forked from M-Labs/artiq
Marius Weber
2538840756
* Input validation and masking of SI -> mu conversions (close #1446) Signed-off-by: Marius Weber <marius.weber@physics.ox.ac.uk> * Update RELEASE_NOTES Signed-off-by: Marius Weber <marius.weber@physics.ox.ac.uk> Co-authored-by: Robert Jördens <rj@quartiq.de>
343 lines
14 KiB
Python
343 lines
14 KiB
Python
"""
|
|
Driver for the AD9914 DDS (with parallel bus) on RTIO.
|
|
"""
|
|
|
|
|
|
from artiq.language.core import *
|
|
from artiq.language.types import *
|
|
from artiq.language.units import *
|
|
from artiq.coredevice.rtio import rtio_output
|
|
|
|
from numpy import int32, int64
|
|
|
|
|
|
__all__ = [
|
|
"AD9914",
|
|
"PHASE_MODE_CONTINUOUS", "PHASE_MODE_ABSOLUTE", "PHASE_MODE_TRACKING"
|
|
]
|
|
|
|
|
|
_PHASE_MODE_DEFAULT = -1
|
|
PHASE_MODE_CONTINUOUS = 0
|
|
PHASE_MODE_ABSOLUTE = 1
|
|
PHASE_MODE_TRACKING = 2
|
|
|
|
AD9914_REG_CFR1L = 0x01
|
|
AD9914_REG_CFR1H = 0x03
|
|
AD9914_REG_CFR2L = 0x05
|
|
AD9914_REG_CFR2H = 0x07
|
|
AD9914_REG_CFR3L = 0x09
|
|
AD9914_REG_CFR3H = 0x0b
|
|
AD9914_REG_CFR4L = 0x0d
|
|
AD9914_REG_CFR4H = 0x0f
|
|
AD9914_REG_DRGFL = 0x11
|
|
AD9914_REG_DRGFH = 0x13
|
|
AD9914_REG_DRGBL = 0x15
|
|
AD9914_REG_DRGBH = 0x17
|
|
AD9914_REG_DRGAL = 0x19
|
|
AD9914_REG_DRGAH = 0x1b
|
|
AD9914_REG_POW = 0x31
|
|
AD9914_REG_ASF = 0x33
|
|
AD9914_REG_USR0 = 0x6d
|
|
AD9914_FUD = 0x80
|
|
AD9914_GPIO = 0x81
|
|
|
|
|
|
class AD9914:
|
|
"""Driver for one AD9914 DDS channel.
|
|
|
|
The time cursor is not modified by any function in this class.
|
|
|
|
Output event replacement is not supported and issuing commands at the same
|
|
time is an error.
|
|
|
|
:param sysclk: DDS system frequency. The DDS system clock must be a
|
|
phase-locked multiple of the RTIO clock.
|
|
:param bus_channel: RTIO channel number of the DDS bus.
|
|
:param channel: channel number (on the bus) of the DDS device to control.
|
|
"""
|
|
|
|
kernel_invariants = {"core", "sysclk", "bus_channel", "channel",
|
|
"rtio_period_mu", "sysclk_per_mu", "write_duration_mu",
|
|
"dac_cal_duration_mu", "init_duration_mu", "init_sync_duration_mu",
|
|
"set_duration_mu", "set_x_duration_mu", "exit_x_duration_mu"}
|
|
|
|
def __init__(self, dmgr, sysclk, bus_channel, channel, core_device="core"):
|
|
self.core = dmgr.get(core_device)
|
|
self.sysclk = sysclk
|
|
self.bus_channel = bus_channel
|
|
self.channel = channel
|
|
self.phase_mode = PHASE_MODE_CONTINUOUS
|
|
|
|
self.rtio_period_mu = int64(8)
|
|
self.sysclk_per_mu = int32(self.sysclk * self.core.ref_period)
|
|
|
|
self.write_duration_mu = 5 * self.rtio_period_mu
|
|
self.dac_cal_duration_mu = 147000 * self.rtio_period_mu
|
|
self.init_duration_mu = 13 * self.write_duration_mu + self.dac_cal_duration_mu
|
|
self.init_sync_duration_mu = 21 * self.write_duration_mu + 2 * self.dac_cal_duration_mu
|
|
self.set_duration_mu = 7 * self.write_duration_mu
|
|
self.set_x_duration_mu = 7 * self.write_duration_mu
|
|
self.exit_x_duration_mu = 3 * self.write_duration_mu
|
|
|
|
@kernel
|
|
def write(self, addr, data):
|
|
rtio_output((self.bus_channel << 8) | addr, data)
|
|
delay_mu(self.write_duration_mu)
|
|
|
|
@kernel
|
|
def init(self):
|
|
"""Resets and initializes the DDS channel.
|
|
|
|
This needs to be done for each DDS channel before it can be used, and
|
|
it is recommended to use the startup kernel for this purpose.
|
|
"""
|
|
delay_mu(-self.init_duration_mu)
|
|
self.write(AD9914_GPIO, (1 << self.channel) << 1);
|
|
|
|
# Note another undocumented "feature" of the AD9914:
|
|
# Programmable modulus breaks if the digital ramp enable bit is
|
|
# not set at the same time.
|
|
self.write(AD9914_REG_CFR1H, 0x0000) # Enable cosine output
|
|
self.write(AD9914_REG_CFR2L, 0x8900) # Enable matched latency
|
|
self.write(AD9914_REG_CFR2H, 0x0089) # Enable profile mode + programmable modulus + DRG
|
|
self.write(AD9914_REG_DRGAL, 0) # Programmable modulus A = 0
|
|
self.write(AD9914_REG_DRGAH, 0)
|
|
self.write(AD9914_REG_DRGBH, 0x8000) # Programmable modulus B == 2**31
|
|
self.write(AD9914_REG_DRGBL, 0x0000)
|
|
self.write(AD9914_REG_ASF, 0x0fff) # Set amplitude to maximum
|
|
self.write(AD9914_REG_CFR4H, 0x0105) # Enable DAC calibration
|
|
self.write(AD9914_FUD, 0)
|
|
delay_mu(self.dac_cal_duration_mu)
|
|
self.write(AD9914_REG_CFR4H, 0x0005) # Disable DAC calibration
|
|
self.write(AD9914_FUD, 0)
|
|
|
|
@kernel
|
|
def init_sync(self, sync_delay):
|
|
"""Resets and initializes the DDS channel as well as configures
|
|
the AD9914 DDS for synchronisation. The synchronisation procedure
|
|
follows the steps outlined in the AN-1254 application note.
|
|
|
|
This needs to be done for each DDS channel before it can be used, and
|
|
it is recommended to use the startup kernel for this.
|
|
|
|
This function cannot be used in a batch; the correct way of
|
|
initializing multiple DDS channels is to call this function
|
|
sequentially with a delay between the calls. 10ms provides a good
|
|
timing margin.
|
|
|
|
:param sync_delay: integer from 0 to 0x3f that sets the value of
|
|
SYNC_OUT (bits 3-5) and SYNC_IN (bits 0-2) delay ADJ bits.
|
|
"""
|
|
delay_mu(-self.init_sync_duration_mu)
|
|
self.write(AD9914_GPIO, (1 << self.channel) << 1)
|
|
|
|
self.write(AD9914_REG_CFR4H, 0x0105) # Enable DAC calibration
|
|
self.write(AD9914_FUD, 0)
|
|
delay_mu(self.dac_cal_duration_mu)
|
|
self.write(AD9914_REG_CFR4H, 0x0005) # Disable DAC calibration
|
|
self.write(AD9914_FUD, 0)
|
|
self.write(AD9914_REG_CFR2L, 0x8b00) # Enable matched latency and sync_out
|
|
self.write(AD9914_FUD, 0)
|
|
# Set cal with sync and set sync_out and sync_in delay
|
|
self.write(AD9914_REG_USR0, 0x0840 | (sync_delay & 0x3f))
|
|
self.write(AD9914_FUD, 0)
|
|
self.write(AD9914_REG_CFR4H, 0x0105) # Enable DAC calibration
|
|
self.write(AD9914_FUD, 0)
|
|
delay_mu(self.dac_cal_duration_mu)
|
|
self.write(AD9914_REG_CFR4H, 0x0005) # Disable DAC calibration
|
|
self.write(AD9914_FUD, 0)
|
|
self.write(AD9914_REG_CFR1H, 0x0000) # Enable cosine output
|
|
self.write(AD9914_REG_CFR2H, 0x0089) # Enable profile mode + programmable modulus + DRG
|
|
self.write(AD9914_REG_DRGAL, 0) # Programmable modulus A = 0
|
|
self.write(AD9914_REG_DRGAH, 0)
|
|
self.write(AD9914_REG_DRGBH, 0x8000) # Programmable modulus B == 2**31
|
|
self.write(AD9914_REG_DRGBL, 0x0000)
|
|
self.write(AD9914_REG_ASF, 0x0fff) # Set amplitude to maximum
|
|
self.write(AD9914_FUD, 0)
|
|
|
|
@kernel
|
|
def set_phase_mode(self, phase_mode):
|
|
"""Sets the phase mode of the DDS channel. Supported phase modes are:
|
|
|
|
* :const:`PHASE_MODE_CONTINUOUS`: the phase accumulator is unchanged when
|
|
switching frequencies. The DDS phase is the sum of the phase
|
|
accumulator and the phase offset. The only discrete jumps in the
|
|
DDS output phase come from changes to the phase offset.
|
|
|
|
* :const:`PHASE_MODE_ABSOLUTE`: the phase accumulator is reset when
|
|
switching frequencies. Thus, the phase of the DDS at the time of
|
|
the frequency change is equal to the phase offset.
|
|
|
|
* :const:`PHASE_MODE_TRACKING`: when switching frequencies, the phase
|
|
accumulator is set to the value it would have if the DDS had been
|
|
running at the specified frequency since the start of the
|
|
experiment.
|
|
|
|
.. warning:: This setting may become inconsistent when used as part of
|
|
a DMA recording. When using DMA, it is recommended to specify the
|
|
phase mode explicitly when calling :meth:`set` or :meth:`set_mu`.
|
|
"""
|
|
self.phase_mode = phase_mode
|
|
|
|
@kernel
|
|
def set_mu(self, ftw, pow=0, phase_mode=_PHASE_MODE_DEFAULT,
|
|
asf=0x0fff, ref_time_mu=-1):
|
|
"""Sets the DDS channel to the specified frequency and phase.
|
|
|
|
This uses machine units (FTW and POW). The frequency tuning word width
|
|
is 32, the phase offset word width is 16, and the amplitude scale factor
|
|
width is 12.
|
|
|
|
The "frequency update" pulse is sent to the DDS with a fixed latency
|
|
with respect to the current position of the time cursor.
|
|
|
|
:param ftw: frequency to generate.
|
|
:param pow: adds an offset to the phase.
|
|
:param phase_mode: if specified, overrides the default phase mode set
|
|
by :meth:`set_phase_mode` for this call.
|
|
:param ref_time_mu: reference time used to compute phase. Specifying this
|
|
makes it easier to have a well-defined phase relationship between
|
|
DDSes on the same bus that are updated at a similar time.
|
|
:return: Resulting phase offset word after application of phase
|
|
tracking offset. When using :const:`PHASE_MODE_CONTINUOUS` in
|
|
subsequent calls, use this value as the "current" phase.
|
|
"""
|
|
if phase_mode == _PHASE_MODE_DEFAULT:
|
|
phase_mode = self.phase_mode
|
|
if ref_time_mu < 0:
|
|
ref_time_mu = now_mu()
|
|
delay_mu(-self.set_duration_mu)
|
|
|
|
self.write(AD9914_GPIO, (1 << self.channel) << 1)
|
|
|
|
self.write(AD9914_REG_DRGFL, ftw & 0xffff)
|
|
self.write(AD9914_REG_DRGFH, (ftw >> 16) & 0xffff)
|
|
|
|
# We need the RTIO fine timestamp clock to be phase-locked
|
|
# to DDS SYSCLK, and divided by an integer self.sysclk_per_mu.
|
|
if phase_mode == PHASE_MODE_CONTINUOUS:
|
|
# Do not clear phase accumulator on FUD
|
|
# Disable autoclear phase accumulator and enables OSK.
|
|
self.write(AD9914_REG_CFR1L, 0x0108)
|
|
else:
|
|
# Clear phase accumulator on FUD
|
|
# Enable autoclear phase accumulator and enables OSK.
|
|
self.write(AD9914_REG_CFR1L, 0x2108)
|
|
fud_time = now_mu() + 2 * self.write_duration_mu
|
|
pow -= int32((ref_time_mu - fud_time) * self.sysclk_per_mu * ftw >> (32 - 16))
|
|
if phase_mode == PHASE_MODE_TRACKING:
|
|
pow += int32(ref_time_mu * self.sysclk_per_mu * ftw >> (32 - 16))
|
|
|
|
self.write(AD9914_REG_POW, pow)
|
|
self.write(AD9914_REG_ASF, asf)
|
|
self.write(AD9914_FUD, 0)
|
|
return pow
|
|
|
|
@portable(flags={"fast-math"})
|
|
def frequency_to_ftw(self, frequency):
|
|
"""Returns the 32-bit frequency tuning word corresponding to the given
|
|
frequency.
|
|
"""
|
|
return int32(round(float(int64(2)**32*frequency/self.sysclk)))
|
|
|
|
@portable(flags={"fast-math"})
|
|
def ftw_to_frequency(self, ftw):
|
|
"""Returns the frequency corresponding to the given frequency tuning
|
|
word.
|
|
"""
|
|
return ftw*self.sysclk/int64(2)**32
|
|
|
|
@portable(flags={"fast-math"})
|
|
def turns_to_pow(self, turns):
|
|
"""Returns the 16-bit phase offset word corresponding to the given
|
|
phase in turns."""
|
|
return round(float(turns*2**16)) & 0xffff
|
|
|
|
@portable(flags={"fast-math"})
|
|
def pow_to_turns(self, pow):
|
|
"""Returns the phase in turns corresponding to the given phase offset
|
|
word."""
|
|
return pow/2**16
|
|
|
|
@portable(flags={"fast-math"})
|
|
def amplitude_to_asf(self, amplitude):
|
|
"""Returns 12-bit amplitude scale factor corresponding to given
|
|
amplitude."""
|
|
code = round(float(amplitude * 0x0fff))
|
|
if code < 0 or code > 0xfff:
|
|
raise ValueError("Invalid AD9914 amplitude!")
|
|
return code
|
|
|
|
@portable(flags={"fast-math"})
|
|
def asf_to_amplitude(self, asf):
|
|
"""Returns the amplitude corresponding to the given amplitude scale
|
|
factor."""
|
|
return asf/0x0fff
|
|
|
|
@kernel
|
|
def set(self, frequency, phase=0.0, phase_mode=_PHASE_MODE_DEFAULT,
|
|
amplitude=1.0):
|
|
"""Like :meth:`set_mu`, but uses Hz and turns."""
|
|
return self.pow_to_turns(
|
|
self.set_mu(self.frequency_to_ftw(frequency),
|
|
self.turns_to_pow(phase), phase_mode,
|
|
self.amplitude_to_asf(amplitude)))
|
|
|
|
# Extended-resolution functions
|
|
@kernel
|
|
def set_x_mu(self, xftw, amplitude=0x0fff):
|
|
"""Set the DDS frequency and amplitude with an extended-resolution
|
|
(63-bit) frequency tuning word.
|
|
|
|
Phase control is not implemented in this mode; the phase offset
|
|
can assume any value.
|
|
|
|
After this function has been called, exit extended-resolution mode
|
|
before calling functions that use standard-resolution mode.
|
|
"""
|
|
delay_mu(-self.set_x_duration_mu)
|
|
|
|
self.write(AD9914_GPIO, (1 << self.channel) << 1)
|
|
|
|
self.write(AD9914_REG_DRGAL, xftw & 0xffff)
|
|
self.write(AD9914_REG_DRGAH, (xftw >> 16) & 0x7fff)
|
|
self.write(AD9914_REG_DRGFL, (xftw >> 31) & 0xffff)
|
|
self.write(AD9914_REG_DRGFH, (xftw >> 47) & 0xffff)
|
|
self.write(AD9914_REG_ASF, amplitude)
|
|
|
|
self.write(AD9914_FUD, 0)
|
|
|
|
@kernel
|
|
def exit_x(self):
|
|
"""Exits extended-resolution mode."""
|
|
delay_mu(-self.exit_x_duration_mu)
|
|
self.write(AD9914_GPIO, (1 << self.channel) << 1)
|
|
self.write(AD9914_REG_DRGAL, 0)
|
|
self.write(AD9914_REG_DRGAH, 0)
|
|
|
|
@portable(flags={"fast-math"})
|
|
def frequency_to_xftw(self, frequency):
|
|
"""Returns the 63-bit frequency tuning word corresponding to the given
|
|
frequency (extended resolution mode).
|
|
"""
|
|
return int64(round(2.0*float(int64(2)**62)*frequency/self.sysclk)) & (
|
|
(int64(1) << 63) - 1)
|
|
|
|
@portable(flags={"fast-math"})
|
|
def xftw_to_frequency(self, xftw):
|
|
"""Returns the frequency corresponding to the given frequency tuning
|
|
word (extended resolution mode).
|
|
"""
|
|
return xftw*self.sysclk/(2.0*float(int64(2)**62))
|
|
|
|
@kernel
|
|
def set_x(self, frequency, amplitude=1.0):
|
|
"""Like :meth:`set_x_mu`, but uses Hz and turns.
|
|
|
|
Note that the precision of ``float`` is less than the precision
|
|
of the extended frequency tuning word.
|
|
"""
|
|
self.set_x_mu(self.frequency_to_xftw(frequency),
|
|
self.amplitude_to_asf(amplitude))
|