You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
183 lines
5.1 KiB
183 lines
5.1 KiB
"""
|
|
Minimalistic CBOR implementation, supports only what we need in cardano.
|
|
"""
|
|
|
|
import ustruct as struct
|
|
from micropython import const
|
|
|
|
from trezor import log
|
|
|
|
_CBOR_TYPE_MASK = const(0xE0)
|
|
_CBOR_INFO_BITS = const(0x1F)
|
|
|
|
_CBOR_UNSIGNED_INT = const(0b000 << 5)
|
|
_CBOR_BYTE_STRING = const(0b010 << 5)
|
|
_CBOR_ARRAY = const(0b100 << 5)
|
|
_CBOR_MAP = const(0b101 << 5)
|
|
_CBOR_TAG = const(0b110 << 5)
|
|
_CBOR_PRIMITIVE = const(0b111 << 5)
|
|
|
|
_CBOR_UINT8_FOLLOWS = const(0x18)
|
|
_CBOR_UINT16_FOLLOWS = const(0x19)
|
|
_CBOR_UINT32_FOLLOWS = const(0x1a)
|
|
_CBOR_UINT64_FOLLOWS = const(0x1b)
|
|
_CBOR_VAR_FOLLOWS = const(0x1f)
|
|
|
|
_CBOR_BREAK = const(0x1f)
|
|
_CBOR_RAW_TAG = const(0x18)
|
|
|
|
|
|
def _header(typ, l: int):
|
|
if l < 24:
|
|
return struct.pack(">B", typ + l)
|
|
elif l < 2 ** 8:
|
|
return struct.pack(">BB", typ + 24, l)
|
|
elif l < 2 ** 16:
|
|
return struct.pack(">BH", typ + 25, l)
|
|
elif l < 2 ** 32:
|
|
return struct.pack(">BI", typ + 26, l)
|
|
elif l < 2 ** 64:
|
|
return struct.pack(">BQ", typ + 27, l)
|
|
else:
|
|
raise NotImplementedError("Length %d not suppported" % l)
|
|
|
|
|
|
def _cbor_encode(value):
|
|
if isinstance(value, int):
|
|
yield _header(_CBOR_UNSIGNED_INT, value)
|
|
elif isinstance(value, bytes):
|
|
yield _header(_CBOR_BYTE_STRING, len(value))
|
|
yield value
|
|
elif isinstance(value, bytearray):
|
|
yield _header(_CBOR_BYTE_STRING, len(value))
|
|
yield bytes(value)
|
|
elif isinstance(value, list):
|
|
# definite-length valued list
|
|
yield _header(_CBOR_ARRAY, len(value))
|
|
for x in value:
|
|
yield from _cbor_encode(x)
|
|
elif isinstance(value, dict):
|
|
yield _header(_CBOR_MAP, len(value))
|
|
for k, v in value.items():
|
|
yield from _cbor_encode(k)
|
|
yield from _cbor_encode(v)
|
|
elif isinstance(value, Tagged):
|
|
yield _header(_CBOR_TAG, value.tag)
|
|
yield from _cbor_encode(value.value)
|
|
elif isinstance(value, IndefiniteLengthArray):
|
|
yield bytes([_CBOR_ARRAY + 31])
|
|
for x in value.array:
|
|
yield from _cbor_encode(x)
|
|
yield bytes([_CBOR_PRIMITIVE + 31])
|
|
elif isinstance(value, Raw):
|
|
yield value.value
|
|
else:
|
|
if __debug__:
|
|
log.debug(__name__, "not implemented (encode): %s", type(value))
|
|
raise NotImplementedError()
|
|
|
|
|
|
def _read_length(cbor, aux):
|
|
if aux == _CBOR_UINT8_FOLLOWS:
|
|
return (cbor[0], cbor[1:])
|
|
elif aux == _CBOR_UINT16_FOLLOWS:
|
|
res = cbor[1]
|
|
res += cbor[0] << 8
|
|
return (res, cbor[2:])
|
|
elif aux == _CBOR_UINT32_FOLLOWS:
|
|
res = cbor[3]
|
|
res += cbor[2] << 8
|
|
res += cbor[1] << 16
|
|
res += cbor[0] << 24
|
|
return (res, cbor[4:])
|
|
elif aux == _CBOR_UINT64_FOLLOWS:
|
|
res = cbor[7]
|
|
res += cbor[6] << 8
|
|
res += cbor[5] << 16
|
|
res += cbor[4] << 24
|
|
res += cbor[3] << 32
|
|
res += cbor[2] << 40
|
|
res += cbor[1] << 48
|
|
res += cbor[0] << 56
|
|
return (res, cbor[8:])
|
|
else:
|
|
raise NotImplementedError("Length %d not suppported" % aux)
|
|
|
|
|
|
def _cbor_decode(cbor):
|
|
fb = cbor[0]
|
|
data = b""
|
|
fb_type = fb & _CBOR_TYPE_MASK
|
|
fb_aux = fb & _CBOR_INFO_BITS
|
|
if fb_type == _CBOR_UNSIGNED_INT:
|
|
if fb_aux < 0x18:
|
|
return (fb_aux, cbor[1:])
|
|
else:
|
|
val, data = _read_length(cbor[1:], fb_aux)
|
|
return (int(val), data)
|
|
elif fb_type == _CBOR_BYTE_STRING:
|
|
ln, data = _read_length(cbor[1:], fb_aux)
|
|
return (data[0:ln], data[ln:])
|
|
elif fb_type == _CBOR_ARRAY:
|
|
if fb_aux == _CBOR_VAR_FOLLOWS:
|
|
res = []
|
|
data = cbor[1:]
|
|
while True:
|
|
item, data = _cbor_decode(data)
|
|
if item == _CBOR_PRIMITIVE + _CBOR_BREAK:
|
|
break
|
|
res.append(item)
|
|
return (res, data)
|
|
else:
|
|
if fb_aux < _CBOR_UINT8_FOLLOWS:
|
|
ln = fb_aux
|
|
data = cbor[1:]
|
|
else:
|
|
ln, data = _read_length(cbor[1:], fb_aux)
|
|
res = []
|
|
for i in range(ln):
|
|
item, data = _cbor_decode(data)
|
|
res.append(item)
|
|
return (res, data)
|
|
elif fb_type == _CBOR_MAP:
|
|
return ({}, cbor[1:])
|
|
elif fb_type == _CBOR_TAG:
|
|
if cbor[1] == _CBOR_RAW_TAG: # only tag 24 (0x18) is supported
|
|
return _cbor_decode(cbor[2:])
|
|
else:
|
|
raise NotImplementedError()
|
|
elif fb_type == _CBOR_PRIMITIVE: # only break code is supported
|
|
return (cbor[0], cbor[1:])
|
|
else:
|
|
if __debug__:
|
|
log.debug(__name__, "not implemented (decode): %s", cbor[0])
|
|
raise NotImplementedError()
|
|
|
|
|
|
class Tagged:
|
|
def __init__(self, tag, value):
|
|
self.tag = tag
|
|
self.value = value
|
|
|
|
|
|
class Raw:
|
|
def __init__(self, value):
|
|
self.value = value
|
|
|
|
|
|
class IndefiniteLengthArray:
|
|
def __init__(self, array):
|
|
assert isinstance(array, list)
|
|
self.array = array
|
|
|
|
|
|
def encode(value):
|
|
return b"".join(_cbor_encode(value))
|
|
|
|
|
|
def decode(cbor: bytes):
|
|
res, check = _cbor_decode(cbor)
|
|
if not (check == b""):
|
|
raise ValueError()
|
|
return res
|