1
0
forked from M-Labs/artiq

lda: fixes

This commit is contained in:
Yann Sionneau 2015-02-18 11:23:20 -07:00 committed by Sebastien Bourdeauducq
parent 3c50610f3d
commit dc132bd248
2 changed files with 47 additions and 18 deletions

View File

@ -1,8 +1,10 @@
import logging import logging
import ctypes import ctypes
import struct import struct
from artiq.language.units import dB, check_unit, Quantity from artiq.language.units import dB, check_unit, Quantity
logger = logging.getLogger("lda") logger = logging.getLogger("lda")
@ -19,6 +21,12 @@ class Ldasim:
self._att_max = 63*dB self._att_max = 63*dB
self._att_step_size = 0.25*dB self._att_step_size = 0.25*dB
def get_att_max(self):
return self._att_max
def get_att_step_size(self):
return self._att_step_size
def get_attenuation(self): def get_attenuation(self):
"""Reads last attenuation value set to the simulated device. """Reads last attenuation value set to the simulated device.
@ -36,19 +44,21 @@ class Ldasim:
:type attenuation: int, float or Fraction :type attenuation: int, float or Fraction
""" """
step = self.get_att_step_size()
if isinstance(attenuation, Quantity): if isinstance(attenuation, Quantity):
check_unit(attenuation, 'dB') check_unit(attenuation, 'dB')
att = attenuation
else: else:
att = attenuation*dB att = attenuation*dB
if att > self._att_max: att = round(att/step)*step
if att > self.get_att_max():
raise ValueError('Cannot set attenuation {} > {}' raise ValueError('Cannot set attenuation {} > {}'
.format(att, self._att_max)) .format(att, self.get_att_max()))
elif att < 0*dB: elif att < 0*dB:
raise ValueError('Cannot set attenuation {} < 0'.format(att)) raise ValueError('Cannot set attenuation {} < 0'.format(att))
elif att % self._att_step_size != 0*dB:
raise ValueError('Cannot set attenuation {} with step size {}'
.format(att, self._att_step_size))
else: else:
att = round(att.amount*4)/4. * dB att = round(att.amount*4)/4. * dB
print("[LDA-sim] setting attenuation to {}".format(att)) print("[LDA-sim] setting attenuation to {}".format(att))
@ -65,8 +75,8 @@ class Lda:
/lib, /usr/local/lib). This can be done either from hidapi sources /lib, /usr/local/lib). This can be done either from hidapi sources
or by installing the libhidapi-libusb0 binary package on Debian-like OS. or by installing the libhidapi-libusb0 binary package on Debian-like OS.
On Windows you should put hidapi.dll shared library in the same directory On Windows you should put hidapi.dll shared library in the
as the controller. artiq\devices\lda folder.
""" """
_vendor_id = 0x041f _vendor_id = 0x041f
@ -75,7 +85,7 @@ class Lda:
"LDA-602": 0x1208, "LDA-602": 0x1208,
"LDA-302P-1": 0x120E, "LDA-302P-1": 0x120E,
} }
_max_att = { _att_max = {
"LDA-102": 63*dB, "LDA-102": 63*dB,
"LDA-602": 63*dB, "LDA-602": 63*dB,
"LDA-302P-1": 63*dB "LDA-302P-1": 63*dB
@ -95,11 +105,27 @@ class Lda:
from artiq.devices.lda.hidapi import hidapi from artiq.devices.lda.hidapi import hidapi
self.hidapi = hidapi self.hidapi = hidapi
self.product = product self.product = product
if serial is None: self.serial = serial
serial = next(self.enumerate(product))
if self.serial is None:
self.serial = next(self.enumerate(self.product))
self._dev = self.hidapi.hid_open(self._vendor_id, self._dev = self.hidapi.hid_open(self._vendor_id,
self._product_ids[product], serial) self._product_ids[self.product],
assert self._dev self.serial)
if not self._dev:
raise IOError
def close(self):
"""Close the device.
"""
self.hidapi.hid_close(self._dev)
def get_att_step_size(self):
return self._att_step_size[self.product]
def get_att_max(self):
return self._att_max[self.product]
@classmethod @classmethod
def enumerate(cls, product): def enumerate(cls, product):
@ -184,18 +210,20 @@ class Lda:
:type attenuation: int, float or Fraction :type attenuation: int, float or Fraction
""" """
step = self.get_att_step_size()
if isinstance(attenuation, Quantity): if isinstance(attenuation, Quantity):
check_unit(attenuation, 'dB') check_unit(attenuation, 'dB')
att = attenuation
else: else:
att = attenuation*dB att = attenuation*dB
if att > self._max_att[self.product]: att = round(att/step)*step
if att > self.get_att_max():
raise ValueError('Cannot set attenuation {} > {}' raise ValueError('Cannot set attenuation {} > {}'
.format(att, self._max_att[self.product])) .format(att, self.get_att_max()))
elif att < 0: elif att < 0*dB:
raise ValueError('Cannot set attenuation {} < 0'.format(att)) raise ValueError('Cannot set attenuation {} < 0'.format(att))
elif att % self._att_step_size[self.product] != 0:
raise ValueError('Cannot set attenuation {} with {} step size'
.format(att, self._att_step_size[self.product]))
else: else:
self.set(0x8d, bytes([int(round(att.amount*4))])) self.set(0x8d, bytes([int(round(att.amount*4))]))

View File

@ -43,6 +43,7 @@ hidapi.hid_free_enumeration.argtypes = [ctypes.POINTER(HidDeviceInfo)]
hidapi.hid_open.argtypes = [ctypes.c_ushort, ctypes.c_ushort, hidapi.hid_open.argtypes = [ctypes.c_ushort, ctypes.c_ushort,
ctypes.c_wchar_p] ctypes.c_wchar_p]
hidapi.hid_open.restype = ctypes.c_void_p hidapi.hid_open.restype = ctypes.c_void_p
hidapi.hid_close.argtypes = [ctypes.c_void_p]
hidapi.hid_read_timeout.argtypes = [ctypes.c_void_p, ctypes.c_char_p, hidapi.hid_read_timeout.argtypes = [ctypes.c_void_p, ctypes.c_char_p,
ctypes.c_size_t, ctypes.c_int] ctypes.c_size_t, ctypes.c_int]
hidapi.hid_read.argtypes = [ctypes.c_void_p, ctypes.c_char_p, ctypes.c_size_t] hidapi.hid_read.argtypes = [ctypes.c_void_p, ctypes.c_char_p, ctypes.c_size_t]