2021-04-14 12:12:38 +00:00
|
|
|
# SPDX-License-Identifier: MIT
|
2021-05-03 12:05:09 +00:00
|
|
|
import itertools
|
2021-04-14 12:12:38 +00:00
|
|
|
from construct import *
|
|
|
|
|
|
|
|
ADTPropertyStruct = Struct(
|
|
|
|
"name" / PaddedString(32, "ascii"),
|
|
|
|
"size" / Int32ul,
|
|
|
|
"value" / Bytes(this.size & 0x7fffffff)
|
|
|
|
)
|
|
|
|
|
|
|
|
ADTNodeStruct = Struct(
|
|
|
|
"property_count" / Int32ul,
|
|
|
|
"child_count" / Int32ul,
|
|
|
|
"properties" / Array(this.property_count, Aligned(4, ADTPropertyStruct)),
|
|
|
|
"children" / Array(this.child_count, LazyBound(lambda: ADTNodeStruct))
|
|
|
|
)
|
|
|
|
|
|
|
|
ADTStringList = GreedyRange(CString("ascii"))
|
|
|
|
|
2021-05-12 12:16:06 +00:00
|
|
|
ADT2Tuple = Array(2, Hex(Int64ul))
|
|
|
|
ADT3Tuple = Array(3, Hex(Int64ul))
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
STD_PROPERTIES = {
|
2021-05-13 13:44:50 +00:00
|
|
|
"cpu-impl-reg": ADT2Tuple,
|
2021-05-03 12:05:09 +00:00
|
|
|
"name": CString("ascii"),
|
2021-04-14 12:12:38 +00:00
|
|
|
"compatible": ADTStringList,
|
|
|
|
"model": CString("ascii"),
|
|
|
|
"#size-cells": Int32ul,
|
|
|
|
"#address-cells": Int32ul,
|
|
|
|
}
|
|
|
|
|
2021-05-12 12:16:06 +00:00
|
|
|
def parse_prop(node, path, name, v):
|
2021-05-03 12:05:09 +00:00
|
|
|
t = None
|
|
|
|
|
2021-05-13 13:44:50 +00:00
|
|
|
if v == b'' or v is None:
|
|
|
|
return None, None
|
|
|
|
|
|
|
|
if name == "reg" and path != "/device-tree/memory":
|
2021-05-12 12:16:06 +00:00
|
|
|
n = node._parent
|
|
|
|
while n is not None and n._parent is not None:
|
|
|
|
if "ranges" not in n._properties:
|
|
|
|
break
|
|
|
|
n = n._parent
|
|
|
|
else:
|
2021-05-13 13:44:50 +00:00
|
|
|
ac, sc = node._parent.address_cells, node._parent.size_cells
|
|
|
|
at = Hex(Int64ul) if ac == 2 else Array(ac, Hex(Int32ul))
|
|
|
|
st = Hex(Int64ul) if sc == 2 else Array(sc, Hex(Int32ul))
|
|
|
|
t = GreedyRange(Struct("addr" / at, "size" / st))
|
2021-05-12 12:16:06 +00:00
|
|
|
|
|
|
|
elif name == "ranges":
|
|
|
|
try:
|
2021-05-13 13:44:50 +00:00
|
|
|
ac, sc = node.address_cells, node.size_cells
|
|
|
|
except KeyError:
|
2021-05-12 12:16:06 +00:00
|
|
|
return None, v
|
2021-05-13 13:44:50 +00:00
|
|
|
pac, _ = node._parent.address_cells, node._parent.size_cells
|
|
|
|
at = Hex(Int64ul) if ac == 2 else Array(ac, Hex(Int32ul))
|
|
|
|
pat = Hex(Int64ul) if pac == 2 else Array(pac, Hex(Int32ul))
|
|
|
|
st = Hex(Int64ul) if sc == 2 else Array(sc, Hex(Int32ul))
|
|
|
|
t = GreedyRange(Struct("bus_addr" / pat, "parent_addr" / at, "size" / st))
|
2021-05-12 12:16:06 +00:00
|
|
|
|
|
|
|
if t is not None:
|
2021-05-13 13:44:50 +00:00
|
|
|
v = Sequence(t, Terminated).parse(v)[0]
|
2021-05-12 12:16:06 +00:00
|
|
|
return t, v
|
|
|
|
|
2021-04-14 12:12:38 +00:00
|
|
|
if name in STD_PROPERTIES:
|
2021-05-03 12:05:09 +00:00
|
|
|
t = STD_PROPERTIES[name]
|
|
|
|
elif v and v[-1] == 0 and all(0x20 <= i <= 0x7e for i in v[:-1]):
|
|
|
|
t = CString("ascii")
|
|
|
|
elif len(v) == 4:
|
|
|
|
t = Int32ul
|
|
|
|
elif len(v) == 8:
|
|
|
|
t = Int64ul
|
|
|
|
elif len(v) == 16 and all(v[i] == 0 for i in (6, 7, 14, 15)):
|
|
|
|
t = ADT2Tuple
|
|
|
|
|
|
|
|
if t is not None:
|
2021-05-13 13:44:50 +00:00
|
|
|
v = Sequence(t, Terminated).parse(v)[0]
|
2021-04-14 12:12:38 +00:00
|
|
|
|
2021-05-03 12:05:09 +00:00
|
|
|
return t, v
|
2021-04-14 12:12:38 +00:00
|
|
|
|
2021-05-03 12:05:09 +00:00
|
|
|
def build_prop(path, name, v, t=None):
|
2021-05-13 13:44:50 +00:00
|
|
|
if v is None:
|
|
|
|
return b''
|
2021-05-03 12:05:09 +00:00
|
|
|
if t is not None:
|
|
|
|
return t.build(v)
|
2021-04-14 12:12:38 +00:00
|
|
|
|
2021-05-03 12:05:09 +00:00
|
|
|
if isinstance(v, bytes):
|
|
|
|
return v
|
2021-04-14 12:12:38 +00:00
|
|
|
|
2021-05-03 12:05:09 +00:00
|
|
|
if name in STD_PROPERTIES:
|
|
|
|
t = STD_PROPERTIES[name]
|
|
|
|
elif isinstance(v, str):
|
|
|
|
t = CString("ascii")
|
|
|
|
elif isinstance(v, int):
|
|
|
|
t = Int32ul
|
|
|
|
elif isinstance(v, tuple) and all(isinstance(i, int) for i in v):
|
|
|
|
t = Array(len(v), Int32ul)
|
2021-04-14 12:12:38 +00:00
|
|
|
|
2021-05-03 12:05:09 +00:00
|
|
|
return t.build(v)
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
class ADTNode:
|
2021-05-12 12:16:06 +00:00
|
|
|
def __init__(self, val=None, path="/", parent=None):
|
2021-05-03 12:05:09 +00:00
|
|
|
self._children = []
|
|
|
|
self._properties = {}
|
|
|
|
self._types = {}
|
|
|
|
self._parent_path = path
|
2021-05-12 12:16:06 +00:00
|
|
|
self._parent = parent
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
if val is not None:
|
|
|
|
for p in val.properties:
|
|
|
|
if p.name == "name":
|
2021-05-03 12:05:09 +00:00
|
|
|
_name = p.value.decode("ascii").rstrip("\0")
|
2021-04-14 12:12:38 +00:00
|
|
|
break
|
|
|
|
else:
|
|
|
|
raise ValueError(f"Node in {path} has no name!")
|
|
|
|
|
2021-05-03 12:05:09 +00:00
|
|
|
path = self._parent_path + _name
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
for p in val.properties:
|
2021-05-13 13:44:50 +00:00
|
|
|
try:
|
|
|
|
self._types[p.name], self._properties[p.name] = parse_prop(self, path, p.name, p.value)
|
|
|
|
except Exception as e:
|
|
|
|
print(f"Exception parsing {path}.{p.name} value {p.value.hex()}:")
|
|
|
|
raise
|
2021-05-12 12:16:06 +00:00
|
|
|
|
|
|
|
# Second pass
|
|
|
|
for k, v in self._types.items():
|
|
|
|
if v is None:
|
|
|
|
self._types[k], self._properties[k] = parse_prop(self, path, k, self._properties[k])
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
for c in val.children:
|
2021-05-12 12:16:06 +00:00
|
|
|
node = ADTNode(c, f"{self._path}/", parent=self)
|
|
|
|
self._children.append(node)
|
2021-05-03 12:05:09 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def _path(self):
|
|
|
|
return self._parent_path + self.name
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
def __getitem__(self, item):
|
|
|
|
if isinstance(item, str):
|
2021-05-21 19:27:46 +00:00
|
|
|
while item.startswith("/"):
|
|
|
|
item = item[1:]
|
|
|
|
if "/" in item:
|
|
|
|
a, b = item.split("/", 1)
|
|
|
|
return self[a][b]
|
2021-05-03 12:05:09 +00:00
|
|
|
for i in self._children:
|
2021-04-14 12:12:38 +00:00
|
|
|
if i.name == item:
|
|
|
|
return i
|
|
|
|
raise KeyError(f"Child node '{item}' not found")
|
2021-05-03 12:05:09 +00:00
|
|
|
return self._children[item]
|
|
|
|
|
|
|
|
def __setitem__(self, item, value):
|
|
|
|
if isinstance(item, str):
|
2021-05-21 19:27:46 +00:00
|
|
|
while item.startswith("/"):
|
|
|
|
item = item[1:]
|
|
|
|
if "/" in item:
|
|
|
|
a, b = item.split("/", 1)
|
|
|
|
self[a][b] = value
|
|
|
|
return
|
2021-05-06 14:36:29 +00:00
|
|
|
for i, c in enumerate(self._children):
|
2021-05-03 12:05:09 +00:00
|
|
|
if c.name == item:
|
|
|
|
self._children[i] = value
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
self._children.append(value)
|
|
|
|
else:
|
|
|
|
self._children[item] = value
|
|
|
|
|
|
|
|
def __delitem__(self, item):
|
|
|
|
if isinstance(item, str):
|
2021-05-21 19:27:46 +00:00
|
|
|
while item.startswith("/"):
|
|
|
|
item = item[1:]
|
|
|
|
if "/" in item:
|
|
|
|
a, b = item.split("/", 1)
|
|
|
|
del self[a][b]
|
|
|
|
return
|
2021-05-03 12:05:09 +00:00
|
|
|
for i, c in enumerate(self._children):
|
|
|
|
if c.name == item:
|
|
|
|
del self._children[i]
|
|
|
|
return
|
|
|
|
raise KeyError(f"Child node '{item}' not found")
|
|
|
|
|
|
|
|
del self._children[item]
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
def __getattr__(self, attr):
|
|
|
|
attr = attr.replace("_", "-")
|
2021-05-03 12:05:09 +00:00
|
|
|
return self._properties[attr]
|
|
|
|
|
|
|
|
def __setattr__(self, attr, value):
|
|
|
|
if attr[0] == "_":
|
|
|
|
self.__dict__[attr] = value
|
|
|
|
return
|
|
|
|
attr = attr.replace("_", "-")
|
|
|
|
self._properties[attr] = value
|
|
|
|
|
|
|
|
def __delattr__(self, attr):
|
|
|
|
if attr[0] == "_":
|
|
|
|
del self.__dict__[attr]
|
|
|
|
return
|
|
|
|
del self._properties[attr]
|
2021-04-14 12:12:38 +00:00
|
|
|
|
2021-05-12 12:16:06 +00:00
|
|
|
@property
|
2021-05-13 13:44:50 +00:00
|
|
|
def address_cells(self):
|
|
|
|
return self._properties["#address-cells"]
|
2021-05-12 12:16:06 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def size_cells(self):
|
|
|
|
return self._properties["#size-cells"]
|
|
|
|
|
|
|
|
def _fmt_prop(self, v):
|
|
|
|
if isinstance(v, ListContainer):
|
|
|
|
return f"[{', '.join(self._fmt_prop(i) for i in v)}]"
|
|
|
|
else:
|
|
|
|
return str(v)
|
|
|
|
|
2021-04-14 12:12:38 +00:00
|
|
|
def __str__(self, t=""):
|
|
|
|
return "\n".join([
|
|
|
|
t + f"{self.name} {{",
|
2021-05-12 12:16:06 +00:00
|
|
|
*(t + f" {k} = {self._fmt_prop(v)}" for k, v in self._properties.items() if k != "name"),
|
2021-04-14 12:12:38 +00:00
|
|
|
"",
|
2021-05-03 12:05:09 +00:00
|
|
|
*(i.__str__(t + " ") for i in self._children),
|
|
|
|
t + "}"
|
2021-04-14 12:12:38 +00:00
|
|
|
])
|
|
|
|
|
|
|
|
def __iter__(self):
|
2021-05-03 12:05:09 +00:00
|
|
|
return iter(self._children)
|
|
|
|
|
2021-05-12 12:16:06 +00:00
|
|
|
def get_reg(self, idx):
|
|
|
|
reg = self.reg[idx]
|
|
|
|
addr = reg.addr
|
|
|
|
size = reg.size
|
|
|
|
|
|
|
|
node = self._parent
|
|
|
|
while node is not None:
|
|
|
|
if "ranges" not in node._properties:
|
|
|
|
break
|
|
|
|
for r in node.ranges:
|
|
|
|
if r.bus_addr <= addr < (r.bus_addr + r.size):
|
|
|
|
addr = addr - r.bus_addr + r.parent_addr
|
|
|
|
break
|
|
|
|
node = node._parent
|
|
|
|
|
|
|
|
return addr, size
|
|
|
|
|
2021-05-03 12:05:09 +00:00
|
|
|
def tostruct(self):
|
|
|
|
properties = []
|
|
|
|
for k,v in itertools.chain(self._properties.items()):
|
|
|
|
value = build_prop(self._path, k, v, t=self._types.get(k, None))
|
|
|
|
properties.append({
|
|
|
|
"name": k,
|
|
|
|
"size": len(value),
|
|
|
|
"value": value
|
|
|
|
})
|
|
|
|
|
|
|
|
data = {
|
|
|
|
"property_count": len(self._properties),
|
|
|
|
"child_count": len(self._children),
|
|
|
|
"properties": properties,
|
|
|
|
"children": [c.tostruct() for c in self._children]
|
|
|
|
}
|
|
|
|
return data
|
|
|
|
|
|
|
|
def build(self):
|
|
|
|
return ADTNodeStruct.build(self.tostruct())
|
2021-04-14 12:12:38 +00:00
|
|
|
|
|
|
|
def load_adt(data):
|
|
|
|
return ADTNode(ADTNodeStruct.parse(data))
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
import sys
|
|
|
|
adt_data = open(sys.argv[1], "rb").read()
|
|
|
|
adt = load_adt(adt_data)
|
|
|
|
print(adt)
|
2021-05-03 12:05:09 +00:00
|
|
|
new_data = adt.build()
|
|
|
|
if len(sys.argv) > 2:
|
|
|
|
with open(sys.argv[2], "wb") as fd:
|
|
|
|
fd.write(new_data)
|
|
|
|
assert new_data == adt_data[:len(new_data)]
|
|
|
|
assert adt_data[len(new_data):] == bytes(len(adt_data) - len(new_data))
|