""" Contains different types of devices belonging to the AD2USB family. """ import usb.core import usb.util import time import threading import serial import serial.tools.list_ports import socket from pyftdi.pyftdi.ftdi import * from pyftdi.pyftdi.usbtools import * from . import util from .event import event class Device(object): """ Generic parent device to all AD2USB products. """ # Generic device events on_open = event.Event('Called when the device has been opened') on_close = event.Event('Called when the device has been closed') on_read = event.Event('Called when a line has been read from the device') on_write = event.Event('Called when data has been written to the device') def __init__(self): """ Constructor """ self._id = '' self._buffer = '' self._interface = None self._device = None self._running = False self._read_thread = Device.ReadThread(self) @property def id(self): """ Retrieve the device ID. """ return self._id @id.setter def id(self, value): """ Sets the device ID. """ self._id = value def is_reader_alive(self): """ Indicates whether or not the reader thread is alive. """ return self._read_thread.is_alive() def stop_reader(self): """ Stops the reader thread. """ self._read_thread.stop() class ReadThread(threading.Thread): """ Reader thread which processes messages from the device. """ READ_TIMEOUT = 10 def __init__(self, device): """ Constructor """ threading.Thread.__init__(self) self._device = device self._running = False def stop(self): """ Stops the running thread. """ self._running = False def run(self): """ The actual read process. """ self._running = True while self._running: try: self._device.read_line(timeout=self.READ_TIMEOUT) except util.TimeoutError, err: pass time.sleep(0.01) class USBDevice(Device): """ AD2USB device exposed with PyFTDI's interface. """ # Constants FTDI_VENDOR_ID = 0x0403 FTDI_PRODUCT_ID = 0x6001 BAUDRATE = 115200 @staticmethod def find_all(): """ Returns all FTDI devices matching our vendor and product IDs. """ devices = [] try: devices = Ftdi.find_all([(USBDevice.FTDI_VENDOR_ID, USBDevice.FTDI_PRODUCT_ID)], nocache=True) except (usb.core.USBError, FtdiError), err: raise util.CommError('Error enumerating AD2USB devices: {0}'.format(str(err))) return devices def __init__(self, vid=FTDI_VENDOR_ID, pid=FTDI_PRODUCT_ID, serial=None, description=None, interface=0): """ Constructor """ Device.__init__(self) self._device = Ftdi() self._interface = interface self._vendor_id = vid self._product_id = pid self._serial_number = serial self._description = description def open(self, baudrate=BAUDRATE, interface=None, index=0, no_reader_thread=False): """ Opens the device. """ # Set up defaults if baudrate is None: baudrate = USBDevice.BAUDRATE if self._interface is None and interface is None: self._interface = 0 if interface is not None: self._interface = interface if index is None: index = 0 # Open the device and start up the thread. try: self._device.open(self._vendor_id, self._product_id, self._interface, index, self._serial_number, self._description) self._device.set_baudrate(baudrate) self._id = 'USB {0}:{1}'.format(self._device.usb_dev.bus, self._device.usb_dev.address) except (usb.core.USBError, FtdiError), err: raise util.NoDeviceError('Error opening device: {0}'.format(str(err))) else: self._running = True if not no_reader_thread: self._read_thread.start() self.on_open((self._serial_number, self._description)) def close(self): """ Closes the device. """ try: self._running = False self._read_thread.stop() self._device.close() # HACK: Probably should fork pyftdi and make this call in .close(). self._device.usb_dev.attach_kernel_driver(self._interface) except: pass self.on_close() def write(self, data): """ Writes data to the device. """ try: self._device.write_data(data) self.on_write(data) except FtdiError, err: raise util.CommError('Error writing to device: {0}'.format(str(err))) def read(self): """ Reads a single character from the device. """ ret = None try: ret = self._device.read_data(1) except (usb.core.USBError, FtdiError), err: raise util.CommError('Error reading from device: {0}'.format(str(err))) return ret def read_line(self, timeout=0.0, purge_buffer=False): """ Reads a line from the device. """ if purge_buffer: self._buffer = '' def timeout_event(): timeout_event.reading = False timeout_event.reading = True got_line = False ret = None timer = None if timeout > 0: timer = threading.Timer(timeout, timeout_event) timer.start() try: while timeout_event.reading: buf = self._device.read_data(1) if buf != '': self._buffer += buf if buf == "\n": if len(self._buffer) > 1: if self._buffer[-2] == "\r": self._buffer = self._buffer[:-2] # ignore if we just got \r\n with nothing else in the buffer. if len(self._buffer) != 0: got_line = True break else: self._buffer = self._buffer[:-1] except (usb.core.USBError, FtdiError), err: timer.cancel() raise util.CommError('Error reading from device: {0}'.format(str(err))) else: if got_line: ret = self._buffer self._buffer = '' self.on_read(ret) if timer: if timer.is_alive(): timer.cancel() else: raise util.TimeoutError('Timeout while waiting for line terminator.') return ret class SerialDevice(Device): """ AD2USB or AD2SERIAL device exposed with the pyserial interface. """ # Constants BAUDRATE = 19200 @staticmethod def find_all(pattern=None): """ Returns all serial ports present. """ devices = [] try: if pattern: devices = serial.tools.list_ports.grep(pattern) else: devices = serial.tools.list_ports.comports() except SerialException, err: raise util.CommError('Error enumerating serial devices: {0}'.format(str(err))) return devices def __init__(self, interface=None): """ Constructor """ Device.__init__(self) self._interface = interface self._id = interface self._device = serial.Serial(timeout=0, writeTimeout=0) # Timeout = non-blocking to match pyftdi. def open(self, baudrate=BAUDRATE, interface=None, index=None, no_reader_thread=False): """ Opens the device. """ # Set up the defaults if baudrate is None: baudrate = SerialDevice.BAUDRATE if self._interface is None and interface is None: raise util.NoDeviceError('No device interface specified.') if interface is not None: self._interface = interface self._device.port = self._interface # Open the device and start up the reader thread. try: self._device.open() self._device.baudrate = baudrate # NOTE: Setting the baudrate before opening the # port caused issues with Moschip 7840/7820 # USB Serial Driver converter. (mos7840) # # Moving it to this point seems to resolve # all issues with it. except (serial.SerialException, ValueError), err: raise util.NoDeviceError('Error opening device on port {0}.'.format(interface)) else: self._running = True self.on_open(('N/A', "AD2SERIAL")) if not no_reader_thread: self._read_thread.start() def close(self): """ Closes the device. """ try: self._running = False self._read_thread.stop() self._device.close() except: pass self.on_close() def write(self, data): """ Writes data to the device. """ try: self._device.write(data) except serial.SerialTimeoutException, err: pass except serial.SerialException, err: raise util.CommError('Error writing to device.') else: self.on_write(data) def read(self): """ Reads a single character from the device. """ ret = None try: ret = self._device.read(1) except serial.SerialException, err: raise util.CommError('Error reading from device: {0}'.format(str(err))) return ret def read_line(self, timeout=0.0, purge_buffer=False): """ Reads a line from the device. """ def timeout_event(): timeout_event.reading = False timeout_event.reading = True got_line = False ret = None timer = None if timeout > 0: timer = threading.Timer(timeout, timeout_event) timer.start() try: while timeout_event.reading: buf = self._device.read(1) if buf != '' and buf != "\xff": # AD2SERIAL specifically apparently sends down \xFF on boot. self._buffer += buf if buf == "\n": if len(self._buffer) > 1: if self._buffer[-2] == "\r": self._buffer = self._buffer[:-2] # ignore if we just got \r\n with nothing else in the buffer. if len(self._buffer) != 0: got_line = True break else: self._buffer = self._buffer[:-1] except (OSError, serial.SerialException), err: timer.cancel() raise util.CommError('Error reading from device: {0}'.format(str(err))) else: if got_line: ret = self._buffer self._buffer = '' self.on_read(ret) if timer: if timer.is_alive(): timer.cancel() else: raise util.TimeoutError('Timeout while waiting for line terminator.') return ret class SocketDevice(Device): """ Device that supports communication with an AD2USB that is exposed via ser2sock or another Serial to IP interface. """ def __init__(self, interface=("localhost", 10000)): """ Constructor """ Device.__init__(self) self._interface = interface self._host, self._port = interface def open(self, baudrate=None, interface=None, index=0, no_reader_thread=False): """ Opens the device. """ if interface is not None: self._interface = interface self._host, self._port = interface try: self._device = socket.socket(socket.AF_INET, socket.SOCK_STREAM) self._device.connect((self._host, self._port)) self._id = '{0}:{1}'.format(self._host, self._port) except socket.error, err: raise util.NoDeviceError('Error opening device at {0}:{1}'.format(self._host, self._port)) else: self._running = True self.on_open(('N/A', "AD2SOCKET")) if not no_reader_thread: self._read_thread.start() def close(self): """ Closes the device. """ self._running = False try: self._read_thread.stop() self._device.shutdown(socket.SHUT_RDWR) # Make sure that it closes immediately. self._device.close() except: pass self.on_close() def write(self, data): """ Writes data to the device. """ data_sent = None try: data_sent = self._device.send(data) if data_sent == 0: raise util.CommError('Error writing to device.') self.on_write(data) except socket.error, err: raise util.CommError('Error writing to device: {0}'.format(str(err))) return data_sent def read(self): """ Reads a single character from the device. """ data = None try: data = self._device.recv(1) except socket.error, err: raise util.CommError('Error while reading from device: {0}'.format(str(err))) return data def read_line(self, timeout=0.0, purge_buffer=False): """ Reads a line from the device. """ if purge_buffer: self._buffer = '' def timeout_event(): timeout_event.reading = False timeout_event.reading = True got_line = False ret = None timer = None if timeout > 0: timer = threading.Timer(timeout, timeout_event) timer.start() try: while timeout_event.reading: buf = self._device.recv(1) if buf != '': self._buffer += buf if buf == "\n": if len(self._buffer) > 1: if self._buffer[-2] == "\r": self._buffer = self._buffer[:-2] # ignore if we just got \r\n with nothing else in the buffer. if len(self._buffer) != 0: got_line = True break else: self._buffer = self._buffer[:-1] except socket.error, err: timer.cancel() raise util.CommError('Error reading from device: {0}'.format(str(err))) else: if got_line: ret = self._buffer self._buffer = '' self.on_read(ret) if timer: if timer.is_alive(): timer.cancel() else: raise util.TimeoutError('Timeout while waiting for line terminator.') return ret