1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-05 13:01:12 +00:00
trezor-firmware/trezorlib/transport_hid.py

112 lines
3.5 KiB
Python
Raw Normal View History

2013-03-10 15:55:59 +00:00
'''USB HID implementation of Transport.'''
import hid
2013-09-24 23:14:54 +00:00
import time
import platform
2013-03-10 15:55:59 +00:00
from transport import Transport, NotImplementedException
DEVICE_IDS = [
2013-11-15 00:43:05 +00:00
(0x10c4, 0xea80), # Shield
(0x534c, 0x0001), # Trezor
2013-03-10 15:55:59 +00:00
]
2013-03-10 16:52:04 +00:00
class FakeRead(object):
# Let's pretend we have a file-like interface
def __init__(self, func):
self.func = func
def read(self, size):
return self.func(size)
2013-03-10 15:55:59 +00:00
class HidTransport(Transport):
def __init__(self, device, *args, **kwargs):
self.hid = None
self.buffer = ''
device = device[int(bool(kwargs.get('debug_link')))]
2013-03-10 15:55:59 +00:00
super(HidTransport, self).__init__(device, *args, **kwargs)
2013-09-09 13:37:39 +00:00
@classmethod
def _detect_debuglink(cls, path):
# Takes platform-specific path of USB and
# decide if the HID interface is normal transport
# or debuglink
if platform.system() in ('Linux', 'Darwin'):
# Sample: 0003:0017:00
if path.endswith(':00'):
return False
return True
elif platform.system() == 'Windows':
# Sample: \\\\?\\hid#vid_534c&pid_0001&mi_01#7&1d71791f&0&0000#{4d1e55b2-f16f-11cf-88cb-001111000030}
# Note: 'mi' parameter is optional and might be unset
if '&mi_01#' in path: # ,,,<o.O>,,,~
return True
return False
else:
raise Exception("USB interface detection not implemented for %s" % platform.system())
2013-03-10 15:55:59 +00:00
@classmethod
def enumerate(cls):
devices = {}
2013-03-10 15:55:59 +00:00
for d in hid.enumerate(0, 0):
vendor_id = d['vendor_id']
product_id = d['product_id']
serial_number = d['serial_number']
path = d['path']
2013-11-15 00:43:05 +00:00
if (vendor_id, product_id) in DEVICE_IDS:
devices.setdefault(serial_number, [None, None])
devices[serial_number][int(bool(cls._detect_debuglink(path)))] = path
2013-03-10 15:55:59 +00:00
# List of two-tuples (path_normal, path_debuglink)
return devices.values()
2013-09-09 13:37:39 +00:00
2013-03-10 15:55:59 +00:00
def _open(self):
self.buffer = ''
2013-10-19 12:19:09 +00:00
self.hid = hid.device()
2013-11-15 00:43:05 +00:00
self.hid.open_path(self.device)
2013-09-24 23:14:54 +00:00
self.hid.set_nonblocking(True)
2013-04-01 14:59:16 +00:00
self.hid.send_feature_report([0x41, 0x01]) # enable UART
self.hid.send_feature_report([0x43, 0x03]) # purge TX/RX FIFOs
2013-03-10 15:55:59 +00:00
def _close(self):
self.hid.close()
self.buffer = ''
self.hid = None
def ready_to_read(self):
return False
def _write(self, msg):
msg = bytearray(msg)
2013-09-09 13:37:39 +00:00
while len(msg):
2013-09-24 23:14:54 +00:00
# Report ID, data padded to 63 bytes
self.hid.write([63, ] + list(msg[:63]) + [0] * (63 - len(msg[:63])))
2013-09-09 13:37:39 +00:00
msg = msg[63:]
2013-03-10 15:55:59 +00:00
def _read(self):
2013-03-10 16:52:04 +00:00
(msg_type, datalen) = self._read_headers(FakeRead(self._raw_read))
2013-03-10 15:55:59 +00:00
return (msg_type, self._raw_read(datalen))
def _raw_read(self, length):
while len(self.buffer) < length:
data = self.hid.read(64)
2013-09-24 23:14:54 +00:00
if not len(data):
time.sleep(0.05)
continue
2013-09-09 13:37:39 +00:00
2013-03-10 15:55:59 +00:00
report_id = data[0]
if report_id > 63:
# Command report
raise Exception("Not implemented")
2013-09-09 13:37:39 +00:00
# Payload received, skip the report ID
2013-09-24 23:14:54 +00:00
self.buffer += str(bytearray(data[1:]))
2013-03-10 15:55:59 +00:00
ret = self.buffer[:length]
self.buffer = self.buffer[length:]
2013-04-01 14:59:16 +00:00
return ret