-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathconv.py
142 lines (114 loc) · 3.92 KB
/
conv.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
'''
Conv converts a value from it's on-file represnetion to a more
programmer / python friendly value. This transform should be loss-less,
so it's not intended for pretty printing (which can be lossy).
General strategy I've taken here is to convert numeric types to
their actual numeric value (applying any shifts needed)
but only if that can be done losslessly, and
enum-like types to strings. Bitsets are converted to lists of true/false.
'''
import parser
'''
Conv() on its own is a no-op
'''
class Conv(object):
def from_file_repr(self, file_repr):
return file_repr
def to_file_repr(self, idiomatic):
return idiomatic
'''
Treats the file representation as the index into a list.
Used for all the enum-like fields, more convenient than DictConv for
contiguous values.
'''
class ListConv(Conv):
def __init__(self, items):
self.items = items
def from_file_repr(self, file_repr):
if file_repr >= len(self.items):
raise ValueError('index {} out of range for {}'.format(file_repr, self.items))
return self.items[file_repr]
def to_file_repr(self, idiomatic):
return self.items.index(idiomatic)
'''
Treats the file representation as the key in a dictionary.
Used for all the enum-like fields that aren't contiguous.
'''
class DictConv(Conv):
def __init__(self, d):
self.d = d
self.d_inv = {v: k for k, v in d.iteritems()}
def from_file_repr(self, file_repr):
return self.d[file_repr]
def to_file_repr(self, idiomatic):
return self.d_inv[idiomatic]
'''
Adds `amount` to the file representation.
For numeric types that need to be offset by a certain amount.
A lot of file representations are 0 indexed in the file but 1 indexed in the minilogue xd LED display.
Also some of the file representations are unsigned, offset by 100 or similar to support negative values.
'''
class AddConv(Conv):
def __init__(self, amount):
self.amount = amount
def from_file_repr(self, file_repr):
return file_repr + self.amount
def to_file_repr(self, idiomatic):
return idiomatic - self.amount
'''
Same as AddConv but multiplies by an amount instead of adds.
In order to be lossless should only use integers.
'''
class MulConv(Conv):
def __init__(self, amount):
self.amount = amount
def from_file_repr(self, file_repr):
return file_repr * self.amount
def to_file_repr(self, idiomatic):
return idiomatic / self.amount
'''
0 means False
1 means True
'''
class BoolConv(Conv):
def from_file_repr(self, file_repr):
return bool(file_repr)
def to_file_repr(self, idiomatic):
return int(idiomatic)
'''
For when the file representation is a bitset / bitflag which is annoying to work
with in python. Converts the int bitset to a list of True/False.
TODO: Not sure if this gets the order backwards or not yet.
'''
class BitFlags(Conv):
def from_file_repr(self, file_repr):
# to binary string, then reverse
# TODO: this needs confirming
bits = ("{:016b}".format(file_repr))[::-1]
return [bool(int(x)) for x in bits]
def to_file_repr(self, idiomatic):
# bool to bit to string, reversed again
bits = [str(int(x)) for x in idiomatic][::-1]
bits_str = ''.join(bits)
return int(bits_str, 2) # parse binary string
NOTES = ('C','C#','D','D#','E','F','F#','G','G#','A','A#', 'B')
class NoteConv(Conv):
def from_file_repr(self, file_repr):
note = NOTES[file_repr % 12]
octave_designation = (file_repr / 12) - 2
return (note, octave_designation)
def to_file_repr(self, idiomatic):
(note, octave_designation) = idiomatic
ni = NOTES.index(note)
return ni + ((octave_designation + 2) * 12)
'''
For the parts of the file that are nested / have sub-structures.
Given a schema, will parse the field's bytes using parser.parse
'''
class NestedConv(Conv):
def __init__(self, schema):
self.schema = schema
def from_file_repr(self, file_repr):
return parser.parse(file_repr, self.schema)
def to_file_repr(self, parsed):
return parsed.serialize()