2016-11-25 21:53:55 +00:00
|
|
|
# This file is part of the TREZOR project.
|
|
|
|
#
|
|
|
|
# Copyright (C) 2012-2016 Marek Palatinus <slush@satoshilabs.com>
|
|
|
|
# Copyright (C) 2012-2016 Pavol Rusnak <stick@satoshilabs.com>
|
|
|
|
#
|
|
|
|
# This library is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This library is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU Lesser General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Lesser General Public License
|
|
|
|
# along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2016-09-27 20:49:51 +00:00
|
|
|
from __future__ import absolute_import
|
|
|
|
|
2012-11-13 14:09:39 +00:00
|
|
|
import struct
|
2016-09-21 14:51:17 +00:00
|
|
|
import binascii
|
2017-06-23 19:31:42 +00:00
|
|
|
from . import mapping
|
|
|
|
|
2012-11-13 14:09:39 +00:00
|
|
|
|
2013-01-05 14:42:09 +00:00
|
|
|
class NotImplementedException(Exception):
|
|
|
|
pass
|
|
|
|
|
2017-06-23 19:31:42 +00:00
|
|
|
|
2014-07-09 22:44:46 +00:00
|
|
|
class ConnectionError(Exception):
|
|
|
|
pass
|
|
|
|
|
2017-06-23 19:31:42 +00:00
|
|
|
|
2012-11-13 14:09:39 +00:00
|
|
|
class Transport(object):
|
|
|
|
def __init__(self, device, *args, **kwargs):
|
|
|
|
self.device = device
|
2016-06-26 19:29:29 +00:00
|
|
|
self.session_id = 0
|
2013-09-09 13:36:17 +00:00
|
|
|
self.session_depth = 0
|
2012-11-13 14:09:39 +00:00
|
|
|
self._open()
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2013-09-09 13:36:17 +00:00
|
|
|
def session_begin(self):
|
2014-08-26 13:54:06 +00:00
|
|
|
"""
|
|
|
|
Apply a lock to the device in order to preform synchronous multistep "conversations" with the device. For example, before entering the transaction signing workflow, one begins a session. After the transaction is complete, the session may be ended.
|
|
|
|
"""
|
2013-09-09 13:36:17 +00:00
|
|
|
if self.session_depth == 0:
|
|
|
|
self._session_begin()
|
|
|
|
self.session_depth += 1
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2013-09-09 13:36:17 +00:00
|
|
|
def session_end(self):
|
2014-08-26 13:54:06 +00:00
|
|
|
"""
|
|
|
|
End a session. Se session_begin for an in depth description of TREZOR sessions.
|
|
|
|
"""
|
2013-09-09 13:36:17 +00:00
|
|
|
self.session_depth -= 1
|
|
|
|
self.session_depth = max(0, self.session_depth)
|
|
|
|
if self.session_depth == 0:
|
|
|
|
self._session_end()
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2012-11-13 14:09:39 +00:00
|
|
|
def close(self):
|
2014-08-26 13:54:06 +00:00
|
|
|
"""
|
|
|
|
Close the connection to the physical device or file descriptor represented by the Transport.
|
|
|
|
"""
|
2012-11-13 14:09:39 +00:00
|
|
|
self._close()
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2012-11-13 14:09:39 +00:00
|
|
|
def write(self, msg):
|
2014-08-26 13:54:06 +00:00
|
|
|
"""
|
|
|
|
Write mesage to tansport. msg should be a member of a valid `protobuf class <https://developers.google.com/protocol-buffers/docs/pythontutorial>`_ with a SerializeToString() method.
|
|
|
|
"""
|
2016-06-26 19:29:29 +00:00
|
|
|
raise NotImplementedException("Not implemented")
|
2012-11-15 20:08:02 +00:00
|
|
|
|
2012-11-13 14:09:39 +00:00
|
|
|
def read(self):
|
2014-08-26 13:54:06 +00:00
|
|
|
"""
|
|
|
|
If there is data available to be read from the transport, reads the data and tries to parse it as a protobuf message. If the parsing succeeds, return a protobuf object.
|
|
|
|
Otherwise, returns None.
|
|
|
|
"""
|
2016-06-26 19:29:29 +00:00
|
|
|
if not self._ready_to_read():
|
2012-12-09 13:53:09 +00:00
|
|
|
return None
|
2012-12-13 18:48:24 +00:00
|
|
|
|
2012-12-09 13:53:09 +00:00
|
|
|
data = self._read()
|
2016-05-26 15:20:44 +00:00
|
|
|
if data is None:
|
2012-12-09 13:53:09 +00:00
|
|
|
return None
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2012-12-13 18:48:24 +00:00
|
|
|
return self._parse_message(data)
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2012-12-13 18:48:24 +00:00
|
|
|
def read_blocking(self):
|
2014-08-26 13:54:06 +00:00
|
|
|
"""
|
2016-06-26 19:29:29 +00:00
|
|
|
Same as read, except blocks until data is available to be read.
|
2014-08-26 13:54:06 +00:00
|
|
|
"""
|
2012-12-13 18:48:24 +00:00
|
|
|
while True:
|
|
|
|
data = self._read()
|
2017-06-23 19:31:42 +00:00
|
|
|
if data is not None:
|
2012-12-13 18:48:24 +00:00
|
|
|
break
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2012-12-13 18:48:24 +00:00
|
|
|
return self._parse_message(data)
|
|
|
|
|
|
|
|
def _parse_message(self, data):
|
2016-06-26 19:29:29 +00:00
|
|
|
(session_id, msg_type, data) = data
|
|
|
|
|
2016-09-13 10:25:06 +00:00
|
|
|
# Raise exception if we get the response with unexpected session ID
|
|
|
|
if session_id != self.session_id:
|
|
|
|
raise Exception("Session ID mismatch. Have %d, got %d" %
|
|
|
|
(self.session_id, session_id))
|
2016-06-26 19:29:29 +00:00
|
|
|
|
2014-07-26 14:27:28 +00:00
|
|
|
if msg_type == 'protobuf':
|
|
|
|
return data
|
|
|
|
else:
|
|
|
|
inst = mapping.get_class(msg_type)()
|
2016-06-26 19:29:29 +00:00
|
|
|
inst.ParseFromString(bytes(data))
|
2014-07-26 14:27:28 +00:00
|
|
|
return inst
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2016-06-26 19:29:29 +00:00
|
|
|
# Functions to be implemented in specific transports:
|
|
|
|
def _open(self):
|
|
|
|
raise NotImplementedException("Not implemented")
|
|
|
|
|
|
|
|
def _close(self):
|
|
|
|
raise NotImplementedException("Not implemented")
|
|
|
|
|
|
|
|
def _write_chunk(self, chunk):
|
|
|
|
raise NotImplementedException("Not implemented")
|
|
|
|
|
|
|
|
def _read_chunk(self):
|
|
|
|
raise NotImplementedException("Not implemented")
|
|
|
|
|
|
|
|
def _ready_to_read(self):
|
|
|
|
"""
|
|
|
|
Returns True if there is data to be read from the transport. Otherwise, False.
|
|
|
|
"""
|
|
|
|
raise NotImplementedException("Not implemented")
|
|
|
|
|
|
|
|
def _session_begin(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def _session_end(self):
|
|
|
|
pass
|
|
|
|
|
2017-06-23 19:31:42 +00:00
|
|
|
|
2016-06-26 19:29:29 +00:00
|
|
|
class TransportV1(Transport):
|
|
|
|
def write(self, msg):
|
|
|
|
ser = msg.SerializeToString()
|
|
|
|
header = struct.pack(">HL", mapping.get_type(msg), len(ser))
|
|
|
|
data = bytearray(b"##" + header + ser)
|
|
|
|
|
|
|
|
while len(data):
|
|
|
|
# Report ID, data padded to 63 bytes
|
|
|
|
chunk = b'?' + data[:63] + b'\0' * (63 - len(data[:63]))
|
|
|
|
self._write_chunk(chunk)
|
|
|
|
data = data[63:]
|
|
|
|
|
|
|
|
def _read(self):
|
|
|
|
chunk = self._read_chunk()
|
|
|
|
(msg_type, datalen, data) = self.parse_first(chunk)
|
|
|
|
|
|
|
|
while len(data) < datalen:
|
|
|
|
chunk = self._read_chunk()
|
|
|
|
data.extend(self.parse_next(chunk))
|
|
|
|
|
|
|
|
# Strip padding zeros
|
|
|
|
data = data[:datalen]
|
|
|
|
return (0, msg_type, data)
|
|
|
|
|
|
|
|
def parse_first(self, chunk):
|
|
|
|
if chunk[:3] != b"?##":
|
|
|
|
raise Exception("Unexpected magic characters")
|
|
|
|
|
|
|
|
try:
|
|
|
|
headerlen = struct.calcsize(">HL")
|
2016-07-19 19:37:54 +00:00
|
|
|
(msg_type, datalen) = struct.unpack(">HL", bytes(chunk[3:3 + headerlen]))
|
2016-06-26 19:29:29 +00:00
|
|
|
except:
|
2016-06-26 22:14:19 +00:00
|
|
|
raise Exception("Cannot parse header")
|
2016-06-26 19:29:29 +00:00
|
|
|
|
|
|
|
data = chunk[3 + headerlen:]
|
|
|
|
return (msg_type, datalen, data)
|
|
|
|
|
|
|
|
def parse_next(self, chunk):
|
|
|
|
if chunk[0:1] != b"?":
|
|
|
|
raise Exception("Unexpected magic characters")
|
|
|
|
|
|
|
|
return chunk[1:]
|
|
|
|
|
2017-06-23 19:31:42 +00:00
|
|
|
|
2016-06-26 19:29:29 +00:00
|
|
|
class TransportV2(Transport):
|
|
|
|
def write(self, msg):
|
2016-11-15 12:46:00 +00:00
|
|
|
if not self.session_id:
|
|
|
|
raise Exception('Missing session_id for v2 transport')
|
|
|
|
|
2016-06-26 22:14:19 +00:00
|
|
|
data = bytearray(msg.SerializeToString())
|
|
|
|
|
2017-07-24 15:24:06 +00:00
|
|
|
dataheader = struct.pack(">LL", mapping.get_type(msg), len(data))
|
|
|
|
data = dataheader + data
|
|
|
|
seq = -1
|
2016-08-05 11:11:24 +00:00
|
|
|
|
2016-06-26 22:14:19 +00:00
|
|
|
while len(data):
|
2017-07-24 15:24:06 +00:00
|
|
|
if seq < 0:
|
|
|
|
repheader = struct.pack(">BL", 0x01, self.session_id)
|
2016-06-26 22:14:19 +00:00
|
|
|
else:
|
2017-07-24 15:24:06 +00:00
|
|
|
repheader = struct.pack(">BLL", 0x02, self.session_id, seq)
|
|
|
|
datalen = 64 - len(repheader)
|
|
|
|
chunk = repheader + data[:datalen] + b'\0' * (datalen - len(data[:datalen]))
|
2016-06-26 22:14:19 +00:00
|
|
|
self._write_chunk(chunk)
|
|
|
|
data = data[datalen:]
|
2017-07-24 15:24:06 +00:00
|
|
|
seq += 1
|
2016-06-26 19:29:29 +00:00
|
|
|
|
|
|
|
def _read(self):
|
2016-11-15 12:46:00 +00:00
|
|
|
if not self.session_id:
|
|
|
|
raise Exception('Missing session_id for v2 transport')
|
|
|
|
|
2016-06-26 22:14:19 +00:00
|
|
|
chunk = self._read_chunk()
|
|
|
|
(session_id, msg_type, datalen, data) = self.parse_first(chunk)
|
|
|
|
|
2017-07-24 15:24:06 +00:00
|
|
|
while len(data) < datalen:
|
2016-06-26 22:14:19 +00:00
|
|
|
chunk = self._read_chunk()
|
2016-07-13 15:41:08 +00:00
|
|
|
(next_session_id, next_data) = self.parse_next(chunk)
|
2016-06-26 22:14:19 +00:00
|
|
|
|
2016-07-13 15:41:08 +00:00
|
|
|
if next_session_id != session_id:
|
2016-06-26 22:14:19 +00:00
|
|
|
raise Exception("Session id mismatch")
|
|
|
|
|
2016-07-13 15:41:08 +00:00
|
|
|
data.extend(next_data)
|
2016-06-26 22:14:19 +00:00
|
|
|
|
2017-07-24 15:24:06 +00:00
|
|
|
data = data[:datalen] # Strip padding
|
2016-06-26 22:14:19 +00:00
|
|
|
return (session_id, msg_type, data)
|
|
|
|
|
|
|
|
def parse_first(self, chunk):
|
|
|
|
try:
|
2017-07-24 15:24:06 +00:00
|
|
|
headerlen = struct.calcsize(">BLLL")
|
|
|
|
(magic, session_id, msg_type, datalen) = struct.unpack(">BLLL", bytes(chunk[:headerlen]))
|
2016-06-26 22:14:19 +00:00
|
|
|
except:
|
|
|
|
raise Exception("Cannot parse header")
|
2017-07-24 15:24:06 +00:00
|
|
|
if magic != 0x01:
|
|
|
|
raise Exception("Unexpected magic character")
|
|
|
|
return (session_id, msg_type, datalen, chunk[headerlen:])
|
2016-06-26 22:14:19 +00:00
|
|
|
|
|
|
|
def parse_next(self, chunk):
|
|
|
|
try:
|
2017-07-24 15:24:06 +00:00
|
|
|
headerlen = struct.calcsize(">BLL")
|
|
|
|
(magic, session_id, sequence) = struct.unpack(">BLL", bytes(chunk[:headerlen]))
|
2016-06-26 22:14:19 +00:00
|
|
|
except:
|
|
|
|
raise Exception("Cannot parse header")
|
2017-07-24 15:24:06 +00:00
|
|
|
if magic != 0x02:
|
|
|
|
raise Exception("Unexpected magic characters")
|
|
|
|
return (session_id, chunk[headerlen:])
|
2016-06-26 19:29:29 +00:00
|
|
|
|
2016-09-21 14:51:17 +00:00
|
|
|
def parse_session_open(self, chunk):
|
2016-09-13 10:25:06 +00:00
|
|
|
try:
|
2017-07-24 15:24:06 +00:00
|
|
|
headerlen = struct.calcsize(">BL")
|
|
|
|
(magic, session_id) = struct.unpack(">BL", bytes(chunk[:headerlen]))
|
2016-09-13 10:25:06 +00:00
|
|
|
except:
|
|
|
|
raise Exception("Cannot parse header")
|
2017-07-24 15:24:06 +00:00
|
|
|
if magic != 0x03:
|
|
|
|
raise Exception("Unexpected magic character")
|
2016-09-21 14:51:17 +00:00
|
|
|
return session_id
|
2016-09-13 10:25:06 +00:00
|
|
|
|
|
|
|
def _session_begin(self):
|
2017-07-24 15:24:06 +00:00
|
|
|
self._write_chunk(bytearray(b'\x03' + b'\0' * 63))
|
2016-09-21 14:51:17 +00:00
|
|
|
self.session_id = self.parse_session_open(self._read_chunk())
|
2016-09-13 10:25:06 +00:00
|
|
|
|
|
|
|
def _session_end(self):
|
2016-09-21 14:51:17 +00:00
|
|
|
header = struct.pack(">L", self.session_id)
|
2017-07-24 15:24:06 +00:00
|
|
|
self._write_chunk(bytearray(b'\x04' + header + b'\0' * (63 - len(header))))
|
|
|
|
if self._read_chunk()[0] != 0x04:
|
2016-09-21 14:51:17 +00:00
|
|
|
raise Exception("Expected session close")
|
|
|
|
self.session_id = None
|
2016-09-13 10:25:06 +00:00
|
|
|
|
2016-06-26 22:14:19 +00:00
|
|
|
'''
|
2016-06-26 19:29:29 +00:00
|
|
|
def read_headers(self, read_f):
|
|
|
|
c = read_f.read(2)
|
|
|
|
if c != b"?!":
|
|
|
|
raise Exception("Unexpected magic characters")
|
|
|
|
|
|
|
|
try:
|
|
|
|
headerlen = struct.calcsize(">HL")
|
|
|
|
(session_id, msg_type, datalen) = struct.unpack(">LLL", read_f.read(headerlen))
|
|
|
|
except:
|
|
|
|
raise Exception("Cannot parse header length")
|
2016-02-10 15:46:58 +00:00
|
|
|
|
2016-06-26 19:29:29 +00:00
|
|
|
return (0, msg_type, datalen)
|
2016-06-26 22:14:19 +00:00
|
|
|
'''
|