diff options
author | Luke T. Shumaker <lukeshu@lukeshu.com> | 2025-03-22 19:17:02 -0600 |
---|---|---|
committer | Luke T. Shumaker <lukeshu@lukeshu.com> | 2025-03-22 19:17:02 -0600 |
commit | 185c3329145959433b8b805de5f114b66b8fcaee (patch) | |
tree | 048ec8b781f1d3b45d29a1d862d724908d0d9785 /build-aux/stack.c.gen | |
parent | 2774e918b6ced670f80036532052189d568e5c5c (diff) | |
parent | 2300ddae5f98180311419413d98fbc8384470665 (diff) |
Merge branch 'lukeshu/qa'
Diffstat (limited to 'build-aux/stack.c.gen')
-rwxr-xr-x | build-aux/stack.c.gen | 98 |
1 files changed, 53 insertions, 45 deletions
diff --git a/build-aux/stack.c.gen b/build-aux/stack.c.gen index a8a971e..5a983cb 100755 --- a/build-aux/stack.c.gen +++ b/build-aux/stack.c.gen @@ -89,6 +89,10 @@ def parse_vcg(reader: typing.TextIO) -> typing.Iterator[VCGElem]: elem.attrs[k] = v pos = m.end() + del _raise + del pos + del line + del lineno yield elem @@ -130,7 +134,9 @@ class QName: return self._content def __eq__(self, other: typing.Any) -> bool: - assert isinstance(other, QName) + assert isinstance( + other, QName + ), f"comparing QName with {other.__class__.__name__}" return self._content == other._content def __lt__(self, other: "QName") -> bool: @@ -160,7 +166,7 @@ class Node: def synthetic_node( - name: str, nstatic: int, calls: typing.Collection[str] = set() + name: str, nstatic: int, calls: typing.Collection[str] = frozenset() ) -> Node: n = Node() @@ -198,7 +204,7 @@ class Application(typing.Protocol): def indirect_callees( self, elem: VCGElem ) -> tuple[typing.Collection[QName], bool]: ... - def skip_call(self, chain: list[QName], funcname: QName) -> bool: ... + def skip_call(self, chain: typing.Sequence[QName], funcname: QName) -> bool: ... def analyze( @@ -217,8 +223,8 @@ def analyze( flags=re.MULTILINE, ) - graph: dict[QName, Node] = dict() - qualified: dict[BaseName, set[QName]] = dict() + graph: dict[QName, Node] = {} + qualified: dict[BaseName, set[QName]] = {} def handle_elem(elem: VCGElem) -> None: match elem.typ: @@ -286,7 +292,7 @@ def analyze( raise ValueError(f"unknown elem type {repr(elem.typ)}") for ci_fname in ci_fnames: - with open(ci_fname, "r") as fh: + with open(ci_fname, "r", encoding="utf-8") as fh: for elem in parse_vcg(fh): handle_elem(elem) @@ -327,7 +333,9 @@ def analyze( track_inclusion: bool = True def nstatic( - orig_funcname: QName, chain: list[QName] = [], missing_ok: bool = False + orig_funcname: QName, + chain: typing.Sequence[QName] = (), + missing_ok: bool = False, ) -> int: nonlocal dbg nonlocal track_inclusion @@ -348,7 +356,7 @@ def analyze( return 0 if len(chain) == cfg_max_call_depth: - raise ValueError(f"max call depth exceeded: {chain+[funcname]}") + raise ValueError(f"max call depth exceeded: {[*chain, funcname]}") node = graph[funcname] if dbg: @@ -361,13 +369,13 @@ def analyze( [ 0, *[ - nstatic(call, chain + [funcname], missing_ok) + nstatic(call, [*chain, funcname], missing_ok) for call, missing_ok in node.calls.items() ], ] ) - groups: dict[str, AnalyzeResultGroup] = dict() + groups: dict[str, AnalyzeResultGroup] = {} for grp_name, grp_filter in app_func_filters.items(): rows: dict[QName, AnalyzeResultVal] = {} for funcname in graph: @@ -393,7 +401,7 @@ def read_source(location: str) -> str: filename = m.group("filename") row = int(m.group("row")) - 1 col = int(m.group("col")) - 1 - with open(m.group("filename"), "r") as fh: + with open(filename, "r", encoding="utf-8") as fh: return fh.readlines()[row][col:].rstrip() @@ -428,7 +436,7 @@ class Plugin(typing.Protocol): def indirect_callees( self, loc: str, line: str ) -> tuple[typing.Collection[QName], bool] | None: ... - def skip_call(self, chain: list[QName], call: QName) -> bool: ... + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: ... class PluginApplication: @@ -462,7 +470,7 @@ class PluginApplication: placeholder += " at " + self._location_xform(elem.attrs.get("label", "")) return [QName(placeholder)], False - def skip_call(self, chain: list[QName], funcname: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], funcname: QName) -> bool: for plugin in self._plugins: if plugin.skip_call(chain, funcname): return True @@ -497,7 +505,7 @@ class CmdPlugin: return [QName("get_root")], False return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -511,7 +519,7 @@ class LibObjPlugin: re_lo_iface = re.compile(r"^\s*#\s*define\s+(?P<name>\S+)_LO_IFACE") re_lo_func = re.compile(r"LO_FUNC *\([^,]*, *(?P<name>[^,) ]+) *[,)]") for fname in arg_c_fnames: - with open(fname, "r") as fh: + with open(fname, "r", encoding="utf-8") as fh: while line := fh.readline(): if m := re_lo_iface.match(line): iface_name = m.group("name") @@ -532,16 +540,16 @@ class LibObjPlugin: r"^LO_IMPLEMENTATION_[HC]\s*\(\s*(?P<iface>[^, ]+)\s*,\s*(?P<impl_typ>[^,]+)\s*,\s*(?P<impl_name>[^, ]+)\s*[,)].*" ) for fname in arg_c_fnames: - with open(fname, "r") as fh: + with open(fname, "r", encoding="utf-8") as fh: for line in fh: line = line.strip() if m := re_lo_implementation.match(line): implementations[m.group("iface")].add(m.group("impl_name")) objcalls: dict[str, set[QName]] = {} # method_name => {method_impls} - for iface_name in ifaces: - for method_name in ifaces[iface_name]: - if QName(method_name) not in objcalls: + for iface_name, iface in ifaces.items(): + for method_name in iface: + if method_name not in objcalls: objcalls[method_name] = set() for impl_name in implementations[iface_name]: objcalls[method_name].add(QName(impl_name + "_" + method_name)) @@ -574,7 +582,7 @@ class LibObjPlugin: ], False return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -643,7 +651,7 @@ class LibHWPlugin: ], False return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -665,7 +673,7 @@ class LibCRPlugin: ) -> tuple[typing.Collection[QName], bool] | None: return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -694,7 +702,7 @@ class LibCRIPCPlugin: ], False return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -730,7 +738,7 @@ class Lib9PPlugin: def config_h_get(varname: str) -> int | None: if config_h_fname: - with open(config_h_fname, "r") as fh: + with open(config_h_fname, "r", encoding="utf-8") as fh: for line in fh: line = line.rstrip() if line.startswith("#define"): @@ -751,7 +759,7 @@ class Lib9PPlugin: r"^\s*\[LIB9P_TYP_T[^]]+\]\s*=\s*\(tmessage_handler\)\s*(?P<handler>\S+),\s*$" ) tmessage_handlers = set() - with open(lib9p_srv_c_fname, "r") as fh: + with open(lib9p_srv_c_fname, "r", encoding="utf-8") as fh: for line in fh: line = line.rstrip() if m := re_tmessage_handler.fullmatch(line): @@ -761,7 +769,7 @@ class Lib9PPlugin: lib9p_msgs: set[str] = set() if lib9p_generated_c_fname: re_lib9p_msg_entry = re.compile(r"^\s*_MSG_(?:[A-Z]+)\((?P<typ>\S+)\),$") - with open(lib9p_generated_c_fname, "r") as fh: + with open(lib9p_generated_c_fname, "r", encoding="utf-8") as fh: for line in fh: line = line.rstrip() if m := re_lib9p_msg_entry.fullmatch(line): @@ -774,7 +782,7 @@ class Lib9PPlugin: assert self.CONFIG_9P_SRV_MAX_REQS if "read" in str(name.base()): return self._CONFIG_9P_NUM_SOCKS - elif "write" in str(name.base()): + if "write" in str(name.base()): return self._CONFIG_9P_NUM_SOCKS * self.CONFIG_9P_SRV_MAX_REQS return 1 @@ -809,7 +817,7 @@ class Lib9PPlugin: return [QName(f"{meth}_{msg}") for msg in self.lib9p_msgs], True return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: if "lib9p/srv.c:srv_util_pathfree" in str(call): assert isinstance(self.CONFIG_9P_SRV_MAX_DEPTH, int) if len(chain) >= self.CONFIG_9P_SRV_MAX_DEPTH and all( @@ -848,7 +856,7 @@ class LibMiscPlugin: ) -> tuple[typing.Collection[QName], bool] | None: return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: if ( len(chain) > 1 and str(chain[-1].base()) == "__assert_msg_fail" @@ -902,7 +910,7 @@ class PicoFmtPlugin: return [QName(x) for x in self.known_fct.values()], False return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: if str(call.base()) in self.known_out.values(): out = "" for pcall in chain: @@ -1028,7 +1036,7 @@ class PicoSDKPlugin: return self.app_preinit_array, False return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False def extra_nodes(self) -> typing.Collection[Node]: @@ -1182,7 +1190,7 @@ class TinyUSBDevicePlugin: r"^\s*#\s*define\s+(?P<k>CFG_TUD_(?:\S{3}|AUDIO|VIDEO|MIDI|VENDOR|USBTMC|DFU_RUNTIME|ECM_RNDIS))\s+(?P<v>\S+).*" ) tusb_config: dict[str, bool] = {} - with open(tusb_config_h_fname, "r") as fh: + with open(tusb_config_h_fname, "r", encoding="utf-8") as fh: in_table = False for line in fh: line = line.rstrip() @@ -1198,7 +1206,7 @@ class TinyUSBDevicePlugin: re_tud_if1 = re.compile(r"^\s*#\s*if (\S+)\s*") re_tud_if2 = re.compile(r"^\s*#\s*if (\S+)\s*\|\|\s*(\S+)\s*") re_tud_endif = re.compile(r"^\s*#\s*endif\s*") - with open(usbd_c_fname, "r") as fh: + with open(usbd_c_fname, "r", encoding="utf-8") as fh: in_table = False enabled = True for line in fh: @@ -1251,15 +1259,15 @@ class TinyUSBDevicePlugin: QName("tud_vendor_control_xfer_cb"), *sorted(self.tud_drivers["control_xfer_cb"]), ], False - elif call.startswith("driver->"): + if call.startswith("driver->"): return sorted(self.tud_drivers[call[len("driver->") :]]), False - elif call == "event.func_call.func": + if call == "event.func_call.func": # callback from usb_defer_func() return [], False return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -1326,7 +1334,7 @@ class NewlibPlugin: ) -> tuple[typing.Collection[QName], bool] | None: return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -1356,7 +1364,7 @@ class LibGCCPlugin: ) -> tuple[typing.Collection[QName], bool] | None: return None - def skip_call(self, chain: list[QName], call: QName) -> bool: + def skip_call(self, chain: typing.Sequence[QName], call: QName) -> bool: return False @@ -1514,20 +1522,20 @@ def main( size: int rows: list[CrRow] = [] - main: CrRow | None = None + mainrow: CrRow | None = None for funcname, val in result.groups["Threads"].rows.items(): name = str(funcname.base()) base = val.nstatic size = base + intrstack if name in ("main", "_entry_point"): - main = CrRow(name=name, cnt=1, base=base, size=size) + mainrow = CrRow(name=name, cnt=1, base=base, size=size) else: size = next_power_of_2(size + stack_guard_size) - stack_guard_size rows.append(CrRow(name=name, cnt=val.cnt, base=base, size=size)) namelen = max(len(r.name) for r in rows) baselen = max(len(str(r.base)) for r in rows) sizesum = sum(r.cnt * (r.size + stack_guard_size) for r in rows) - sizelen = len(str(max(sizesum, main.size if main else 0))) + sizelen = len(str(max(sizesum, mainrow.size if mainrow else 0))) def print_row(comment: bool, name: str, size: int, eqn: str | None = None) -> None: prefix = "const size_t CONFIG_COROUTINE_STACK_SIZE_" @@ -1552,12 +1560,12 @@ def main( f"LM_NEXT_POWER_OF_2({str(row.base).rjust(baselen)}+{intrstack}+{stack_guard_size})-{stack_guard_size}", ) print_row(True, "TOTAL (inc. stack guard)", sizesum) - if main: + if mainrow: print_row( True, "MAIN/KERNEL", - main.size, - f" {str(main.base).rjust(baselen)}+{intrstack}", + mainrow.size, + f" {str(mainrow.base).rjust(baselen)}+{intrstack}", ) print() print("/*") @@ -1592,7 +1600,7 @@ if __name__ == "__main__": for obj_fname in obj_fnames: if re_c_obj_suffix.search(obj_fname): ci_fnames.add(re_c_obj_suffix.sub(".c.ci", obj_fname)) - with open(obj_fname + ".d", "r") as fh: + with open(obj_fname + ".d", "r", encoding="utf-8") as fh: c_fnames.update( fh.read().replace("\\\n", " ").split(":")[-1].split() ) |