Browse Source

Rearranged things to allow for initial serial support.

pyserial_fix
Scott Petersen 11 years ago
parent
commit
015bdf40d8
4 changed files with 172 additions and 146 deletions
  1. +7
    -144
      pyad2usb/ad2usb.py
  2. +158
    -0
      pyad2usb/devices.py
  3. +5
    -0
      pyad2usb/util.py
  4. +2
    -2
      test.py

+ 7
- 144
pyad2usb/ad2usb.py View File

@@ -1,16 +1,8 @@
from pyftdi.pyftdi.ftdi import *
from pyftdi.pyftdi.usbtools import *
import time import time
import usb.core
import usb.util
from .event import event
import threading import threading

class NoDeviceError(Exception):
pass

class CommError(Exception):
pass
from .event import event
from . import devices
from . import util


class Overseer(object): class Overseer(object):
on_attached = event.Event('Called when an AD2USB device has been detected.') on_attached = event.Event('Called when an AD2USB device has been detected.')
@@ -20,7 +12,7 @@ class Overseer(object):


@classmethod @classmethod
def find_all(cls): def find_all(cls):
cls.__devices = Device.find_all()
cls.__devices = devices.USBDevice.find_all()


return cls.__devices return cls.__devices


@@ -31,13 +23,13 @@ class Overseer(object):
@classmethod @classmethod
def create(cls, device=None): def create(cls, device=None):
if len(cls.__devices) == 0: if len(cls.__devices) == 0:
raise NoDeviceError('No AD2USB devices present.')
raise util.NoDeviceError('No AD2USB devices present.')


if device is None: if device is None:
device = cls.__devices[0] device = cls.__devices[0]


vendor, product, sernum, ifcount, description = device vendor, product, sernum, ifcount, description = device
device = Device(serial=sernum, description=description)
device = devices.USBDevice(serial=sernum, description=description)


return AD2USB(device) return AD2USB(device)


@@ -99,7 +91,7 @@ class Overseer(object):


for d in removed_devices: for d in removed_devices:
self._overseer.on_detached(d) self._overseer.on_detached(d)
except CommError, err:
except util.CommError, err:
pass pass


time.sleep(0.25) time.sleep(0.25)
@@ -130,132 +122,3 @@ class AD2USB(object):
self._device.on_close += self.on_close self._device.on_close += self.on_close
self._device.on_read += self.on_read self._device.on_read += self.on_read
self._device.on_write += self.on_write self._device.on_write += self.on_write


class Device(object):
FTDI_VENDOR_ID = 0x0403
FTDI_PRODUCT_ID = 0x6001
BAUDRATE = 115200

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')

@staticmethod
def find_all():
devices = []

try:
devices = Ftdi.find_all([(Device.FTDI_VENDOR_ID, Device.FTDI_PRODUCT_ID)], nocache=True)
except (usb.core.USBError, FtdiError), err:
raise 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):
self._vendor_id = vid
self._product_id = pid
self._serial_number = serial
self._description = description
self._buffer = ''
self._device = Ftdi()
self._running = False

self._read_thread = Device.ReadThread(self)

def open(self, baudrate=BAUDRATE, interface=0, index=0):
self._running = True

try:
self._device.open(self._vendor_id,
self._product_id,
interface,
index,
self._serial_number,
self._description)

self._device.set_baudrate(baudrate)
except (usb.core.USBError, FtdiError), err:
self.on_close()

raise CommError('Error opening AD2USB device: {0}'.format(str(err)))
else:
self._read_thread.start()

self.on_open((self._serial_number, self._description))

def close(self):
try:
self._running = False
self._read_thread.stop()

self._device.close()
except (FtdiError, usb.core.USBError):
pass

self.on_close()

def write(self, data):
self._device.write_data(data)

self.on_write(data)

def read_line(self, timeout=0.0):
start_time = time.time()
got_line = False
ret = None

try:
while self._running:
buf = self._device.read_data(1)
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]

if timeout > 0 and time.time() - start_time > timeout:
break

time.sleep(0.01)
except (usb.core.USBError, FtdiError), err:
self.close()

raise CommError('Error reading from AD2USB device: {0}'.format(str(err)))
else:
if got_line:
ret = self._buffer
self._buffer = ''

self.on_read(ret)

return ret

class ReadThread(threading.Thread):
def __init__(self, device):
threading.Thread.__init__(self)
self._device = device
self._running = False

def stop(self):
self._running = False

def run(self):
self._running = True

while self._running:
try:
self._device.read_line()
except CommError, err:
self.stop()

time.sleep(0.25)

+ 158
- 0
pyad2usb/devices.py View File

@@ -0,0 +1,158 @@
import usb.core
import usb.util
import time
import threading
from pyftdi.pyftdi.ftdi import *
from pyftdi.pyftdi.usbtools import *
from . import util
from .event import event

class USBDevice(object):
FTDI_VENDOR_ID = 0x0403
FTDI_PRODUCT_ID = 0x6001
BAUDRATE = 115200

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')

@staticmethod
def find_all():
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):
self._vendor_id = vid
self._product_id = pid
self._serial_number = serial
self._description = description
self._buffer = ''
self._device = Ftdi()
self._running = False

self._read_thread = USBDevice.ReadThread(self)

def open(self, baudrate=BAUDRATE, interface=0, index=0):
self._running = True

try:
self._device.open(self._vendor_id,
self._product_id,
interface,
index,
self._serial_number,
self._description)

self._device.set_baudrate(baudrate)
except (usb.core.USBError, FtdiError), err:
self.on_close()

raise util.CommError('Error opening AD2USB device: {0}'.format(str(err)))
else:
self._read_thread.start()

self.on_open((self._serial_number, self._description))

def close(self):
try:
self._running = False
self._read_thread.stop()

self._device.close()
except (FtdiError, usb.core.USBError):
pass

self.on_close()

def write(self, data):
self._device.write_data(data)

self.on_write(data)

def read_line(self, timeout=0.0):
start_time = time.time()
got_line = False
ret = None

try:
while self._running:
buf = self._device.read_data(1)
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]

if timeout > 0 and time.time() - start_time > timeout:
break

time.sleep(0.01)
except (usb.core.USBError, FtdiError), err:
self.close()

raise util.CommError('Error reading from AD2USB device: {0}'.format(str(err)))
else:
if got_line:
ret = self._buffer
self._buffer = ''

self.on_read(ret)

return ret

class ReadThread(threading.Thread):
def __init__(self, device):
threading.Thread.__init__(self)
self._device = device
self._running = False

def stop(self):
self._running = False

def run(self):
self._running = True

while self._running:
try:
self._device.read_line()
except util.CommError, err:
self.stop()

time.sleep(0.25)


class SerialDevice(object):
BAUDRATE = 9600

def __init__(self):
pass

def __del__(self):
pass

def open(self, port, baudrate=BAUDRATE):
pass

def close(self):
pass

def write(self, data):
pass

def read_line(self, data):
pass

+ 5
- 0
pyad2usb/util.py View File

@@ -0,0 +1,5 @@
class NoDeviceError(Exception):
pass

class CommError(Exception):
pass

+ 2
- 2
test.py View File

@@ -55,5 +55,5 @@ try:


#wut.close() #wut.close()
except Exception, err: except Exception, err:
print 'Error: {0}'.format(str(err))
#traceback.print_exc(err)
#print 'Error: {0}'.format(str(err))
traceback.print_exc(err)

Loading…
Cancel
Save