1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-13 17:00:59 +00:00

pb2py generator, build_pb2.sh

This commit is contained in:
slush0 2016-04-27 01:32:57 +02:00 committed by Pavol Rusnak
parent 32403b618e
commit e6df94cbd5
No known key found for this signature in database
GPG Key ID: 91F3B339B9A02A3D
8 changed files with 211 additions and 40 deletions

0
src/lib/__init__.py Normal file
View File

View File

@ -76,7 +76,6 @@ FLAG_REQUIRED = 1
FLAG_REQUIRED_MASK = 1 FLAG_REQUIRED_MASK = 1
FLAG_SINGLE = 0 FLAG_SINGLE = 0
FLAG_REPEATED = 2 FLAG_REPEATED = 2
FLAG_PACKED_REPEATED = 6
FLAG_REPEATED_MASK = 6 FLAG_REPEATED_MASK = 6
FLAG_PRIMITIVE = 0 FLAG_PRIMITIVE = 0
FLAG_EMBEDDED = 8 FLAG_EMBEDDED = 8
@ -110,12 +109,15 @@ class MessageType:
# Creates a new message type. # Creates a new message type.
self.__tags_to_types = dict() # Maps a tag to a type instance. self.__tags_to_types = dict() # Maps a tag to a type instance.
self.__tags_to_names = dict() # Maps a tag to a given field name. self.__tags_to_names = dict() # Maps a tag to a given field name.
self.__defaults = dict() # Maps a tag to its default value.
self.__flags = dict() # Maps a tag to FLAG_ self.__flags = dict() # Maps a tag to FLAG_
def add_field(self, tag, name, field_type, flags=FLAG_SIMPLE): def add_field(self, tag, name, field_type, flags=FLAG_SIMPLE, default=None):
# Adds a field to the message type. # Adds a field to the message type.
if tag in self.__tags_to_names or tag in self.__tags_to_types: if tag in self.__tags_to_names or tag in self.__tags_to_types:
raise ValueError('The tag %s is already used.' % tag) raise ValueError('The tag %s is already used.' % tag)
if default != None:
self.__defaults[tag] = default
self.__tags_to_names[tag] = name self.__tags_to_names[tag] = name
self.__tags_to_types[tag] = field_type self.__tags_to_types[tag] = field_type
self.__flags[tag] = flags self.__flags[tag] = flags
@ -137,19 +139,12 @@ class MessageType:
if self.__has_flag(tag, FLAG_SINGLE, FLAG_REPEATED_MASK): if self.__has_flag(tag, FLAG_SINGLE, FLAG_REPEATED_MASK):
# Single value. # Single value.
UVarintType.dump(fp, _pack_key(tag, field_type.WIRE_TYPE)) UVarintType.dump(fp, _pack_key(tag, field_type.WIRE_TYPE))
field_type.dump(fp, value.__dict__[self.__tags_to_names[tag]]) field_type.dump(fp, getattr(value, self.__tags_to_names[tag]))
elif self.__has_flag(tag, FLAG_PACKED_REPEATED, FLAG_REPEATED_MASK):
# Repeated packed value.
UVarintType.dump(fp, _pack_key(tag, BytesType.WIRE_TYPE))
internal_fp = BytesIO()
for single_value in value[self.__tags_to_names[tag]]:
field_type.dump(internal_fp, single_value)
BytesType.dump(fp, internal_fp.getvalue())
elif self.__has_flag(tag, FLAG_REPEATED, FLAG_REPEATED_MASK): elif self.__has_flag(tag, FLAG_REPEATED, FLAG_REPEATED_MASK):
# Repeated value. # Repeated value.
key = _pack_key(tag, field_type.WIRE_TYPE) key = _pack_key(tag, field_type.WIRE_TYPE)
# Put it together sequently. # Put it together sequently.
for single_value in value[self.__tags_to_names[tag]]: for single_value in getattr(value, self.__tags_to_names[tag]):
UVarintType.dump(fp, key) UVarintType.dump(fp, key)
field_type.dump(fp, single_value) field_type.dump(fp, single_value)
elif self.__has_flag(tag, FLAG_REQUIRED, FLAG_REQUIRED_MASK): elif self.__has_flag(tag, FLAG_REQUIRED, FLAG_REQUIRED_MASK):
@ -163,30 +158,18 @@ class MessageType:
if tag in self.__tags_to_types: if tag in self.__tags_to_types:
field_type = self.__tags_to_types[tag] field_type = self.__tags_to_types[tag]
if not self.__has_flag(tag, FLAG_PACKED_REPEATED, FLAG_REPEATED_MASK): if wire_type != field_type.WIRE_TYPE:
if wire_type != field_type.WIRE_TYPE: raise TypeError(
raise TypeError( 'Value of tag %s has incorrect wiretype %s, %s expected.' % \
'Value of tag %s has incorrect wiretype %s, %s expected.' % \ (tag, wire_type, field_type.WIRE_TYPE))
(tag, wire_type, field_type.WIRE_TYPE))
elif wire_type != BytesType.WIRE_TYPE:
raise TypeError('Tag %s has wiretype %s while the field is packed repeated.' % (tag, wire_type))
if self.__has_flag(tag, FLAG_SINGLE, FLAG_REPEATED_MASK): if self.__has_flag(tag, FLAG_SINGLE, FLAG_REPEATED_MASK):
# Single value. # Single value.
setattr(message, self.__tags_to_names[tag], field_type.load(fp)) setattr(message, self.__tags_to_names[tag], field_type.load(fp))
elif self.__has_flag(tag, FLAG_PACKED_REPEATED, FLAG_REPEATED_MASK):
# Repeated packed value.
repeated_value = message[self.__tags_to_names[tag]] = list()
internal_fp = EofWrapper(fp, UVarintType.load(fp)) # Limit with value length.
while True:
try:
repeated_value.append(field_type.load(internal_fp))
except EOFError:
break
elif self.__has_flag(tag, FLAG_REPEATED, FLAG_REPEATED_MASK): elif self.__has_flag(tag, FLAG_REPEATED, FLAG_REPEATED_MASK):
# Repeated value. # Repeated value.
if not self.__tags_to_names[tag] in message: if not self.__tags_to_names[tag] in message.__dict__:
repeated_value = message[self.__tags_to_names[tag]] = list() setattr(message, self.__tags_to_names[tag], list())
repeated_value.append(field_type.load(fp)) getattr(message, self.__tags_to_names[tag]).append(field_type.load(fp))
else: else:
# Skip this field. # Skip this field.
@ -194,11 +177,15 @@ class MessageType:
{0: UVarintType, 2: BytesType}[wire_type].load(fp) {0: UVarintType, 2: BytesType}[wire_type].load(fp)
except EOFError: except EOFError:
# Check if all required fields are present.
for tag, name in iter(self.__tags_to_names.items()): for tag, name in iter(self.__tags_to_names.items()):
if self.__has_flag(tag, FLAG_REQUIRED, FLAG_REQUIRED_MASK) and not name in message: # Fill in default value if value not set
if name not in message.__dict__ and tag in self.__defaults:
setattr(message, name, self.__defaults[tag])
# Check if all required fields are present.
if self.__has_flag(tag, FLAG_REQUIRED, FLAG_REQUIRED_MASK) and not name in message.__dict__:
if self.__has_flag(tag, FLAG_REPEATED, FLAG_REPEATED_MASK): if self.__has_flag(tag, FLAG_REPEATED, FLAG_REPEATED_MASK):
message[name] = list() # Empty list (no values was in input stream). But required field. setattr(message, name, list()) # Empty list (no values was in input stream). But required field.
else: else:
raise ValueError('The field %s (\'%s\') is required but missing.' % (tag, name)) raise ValueError('The field %s (\'%s\') is required but missing.' % (tag, name))
return message return message

View File

@ -15,15 +15,16 @@ from uasyncio import core
from trezor import ui from trezor import ui
from trezor import msg from trezor import msg
from trezor.utils import unimport
logging.basicConfig(level=logging.INFO) logging.basicConfig(level=logging.INFO)
loop = core.get_event_loop() loop = core.get_event_loop()
def perf_info(): def perf_info():
mem_free = gc.mem_free() mem_alloc = gc.mem_alloc()
gc.collect() gc.collect()
print("free_mem: %s/%s, last_sleep: %.06f" % \ print("mem_alloc: %s/%s, last_sleep: %.06f" % \
(mem_free, gc.mem_free(), loop.last_sleep)) (mem_alloc, gc.mem_alloc(), loop.last_sleep))
loop.call_later(1, perf_info) loop.call_later(1, perf_info)
def animate(): def animate():
@ -100,10 +101,31 @@ def on_read():
print("READY TO READ") print("READY TO READ")
print(msg.read()) print(msg.read())
@unimport
def zprava():
from _io import BytesIO
from trezor.messages.GetAddress import GetAddress
m = GetAddress()
m.address_n = [1, 2, 3]
m.show_display = True
print(m.__dict__)
f = BytesIO()
m.dump(f)
data = f.getvalue()
f.close()
print(data)
# m2 = GetAddress.load(BytesIO(data))
# print(m2.__dict__)
def run(): def run():
# pipe.init('../pipe', on_read) # pipe.init('../pipe', on_read)
# msg.set_notify(on_read) # msg.set_notify(on_read)
zprava()
loop.call_soon(perf_info) loop.call_soon(perf_info)
loop.call_soon(tap_to_confirm()) loop.call_soon(tap_to_confirm())
# loop.call_soon(animate()) # loop.call_soon(animate())

View File

View File

@ -1,5 +1,12 @@
def hexlify(data: bytes) -> str: import sys
return ''.join(['%02x' % b for b in data]) import gc
def unhexlify(data: str) -> bytes: def unimport(func):
return bytes([int(data[i:i+2], 16) for i in range(0, len(data), 2)]) def inner(*args, **kwargs):
mods = set(sys.modules)
ret = func(*args, **kwargs)
for to_remove in set(sys.modules) - mods:
print(to_remove)
del sys.modules[to_remove]
return ret
return inner

15
tools/build_pb.sh Executable file
View File

@ -0,0 +1,15 @@
#!/bin/bash
CURDIR=$(pwd)
for i in messages types storage ; do
# Compile .proto files to python2 modules using google protobuf library
cd $CURDIR/../../trezor-common/protob
protoc --python_out=$CURDIR/pb2/ -I/usr/include -I. $i.proto
# Convert google protobuf library to trezor's internal format
cd $CURDIR
./pb2py $i ../src/trezor/messages/
done

0
tools/pb2/__init__.py Normal file
View File

140
tools/pb2py Executable file
View File

@ -0,0 +1,140 @@
#!/usr/bin/env python2
import sys
import os
from google.protobuf.internal.enum_type_wrapper import EnumTypeWrapper
def process_type(t, cls):
imports = ["from protobuf import protobuf as p",]
out = ["t = p.MessageType()", ]
print("Processing type %s" % t)
TYPE_STRING = 9
TYPE_BYTES = 12
TYPE_MESSAGE = 11
for k, v in cls.DESCRIPTOR.fields_by_name.items():
#print k
number = v.number
fieldname = k
type = None
repeated = v.label == 3
required = v.label == 2
#print v.has_default_value, v.default_value
if v.type in (4, 13, 14):
# TYPE_UINT64 = 4
# TYPE_UINT32 = 13
# TYPE_ENUM = 14
type = 'p.UVarintType'
elif v.type == 9:
# TYPE_STRING = 9
type = 'p.UnicodeType'
elif v.type == 8:
# TYPE_BOOL = 8
type = 'p.BoolType'
elif v.type == 12:
# TYPE_BYTES = 12
type = 'p.BytesType'
elif v.type == 11:
# TYPE_MESSAGE = 1
type = "p.EmbeddedMessage(%s)" % v.message_type.name
imports.append("from .%s import %s" % (v.message_type.name, v.message_type.name))
else:
raise Exception("Unknown field type %s for field %s" % (v.type, k))
if repeated:
flags = ', flags=p.FLAG_REPEATED'
elif required:
flags = ', flags=p.FLAG_REQUIRED'
else:
flags = ''
if v.has_default_value:
default = ', default=%s' % repr(v.default_value)
else:
default = ''
out.append("t.add_field(%d, '%s', %s%s%s)" % \
(number, fieldname, type, flags, default))
#print fieldname, number, type, repeated, default
#print v.__dict__
#print v.CPPTYPE_STRING
#print v.LABEL_REPEATED
#print v.enum_type
# v.has_default_value, v.default_value
# v.label == 3 # repeated
#print v.number
out.append("%s = t" % t)
return imports + out
def process_enum(t, cls):
out = []
print("Processing enum %s" % t)
for k, v in cls.items():
# Remove type name from the beginning of the constant
# For example "PinMatrixRequestType_Current" -> "Current"
if k.startswith("%s_" % t):
k = k.replace("%s_" % t, '')
# If type ends with *Type, but constant use type name without *Type, remove it too :)
# For example "ButtonRequestType & ButtonRequest_Other" => "Other"
if t.endswith("Type") and k.startswith("%s_" % t.replace("Type", '')):
k = k.replace("%s_" % t.replace("Type", ''), '')
out.append("%s = %s" % (k, v))
return out
def process_module(mod, genpath):
types = dict([(name, cls) for name, cls in mod.__dict__.items() if isinstance(cls, type)])
for t, cls in types.iteritems():
out = process_type(t, cls)
write_to_file(genpath, t, out)
enums = dict([(name, cls) for name, cls in mod.__dict__.items() if isinstance(cls, EnumTypeWrapper)])
for t, cls in enums.iteritems():
out = process_enum(t, cls)
write_to_file(genpath, t, out)
def write_to_file(genpath, t, out):
# Write generated sourcecode to given file
f = open(os.path.join(genpath, "%s.py" % t), 'w')
out = ["# Automatically generated by ./pb2py"] + out
data = "\n".join(out)
f.write(data)
f.close()
if __name__ == '__main__':
if len(sys.argv) < 2:
print("Usage: ./pb2py modulename genpath")
sys.exit()
modulename = sys.argv[1]
genpath = sys.argv[2]
# Dynamically load module from argv[1]
tmp = __import__('pb2', globals(), locals(), ['%s_pb2' % modulename])
mod = getattr(tmp, "%s_pb2" % modulename)
process_module(mod, genpath)