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
|
# build-aux/measurestack/util.py - Analyze stack sizes for compiled objects
#
# Copyright (C) 2024-2025 Luke T. Shumaker <lukeshu@lukeshu.com>
# SPDX-License-Identifier: AGPL-3.0-or-later
import re
import typing
from . import analyze, vcg
from .analyze import BaseName, Node, QName
# pylint: disable=unused-variable
__all__ = [
"synthetic_node",
"read_source",
"get_zero_or_one",
"re_call_other",
"Plugin",
"PluginApplication",
]
def synthetic_node(
name: str, nstatic: int, calls: typing.Collection[str] = frozenset()
) -> Node:
n = Node()
n.funcname = QName(name)
n.location = "<synthetic>"
n.usage_kind = "static"
n.nstatic = nstatic
n.ndynamic = 0
n.calls = dict((QName(c), False) for c in calls)
return n
re_location = re.compile(r"(?P<filename>.+):(?P<row>[0-9]+):(?P<col>[0-9]+)")
def read_source(location: str) -> str:
m = re_location.fullmatch(location)
if not m:
raise ValueError(f"unexpected label value {location!r}")
filename = m.group("filename")
row = int(m.group("row")) - 1
col = int(m.group("col")) - 1
with open(filename, "r", encoding="utf-8") as fh:
return fh.readlines()[row][col:].rstrip()
def get_zero_or_one(
pred: typing.Callable[[str], bool], fnames: typing.Collection[str]
) -> str | None:
count = sum(1 for fname in fnames if pred(fname))
assert count < 2
if count:
return next(fname for fname in fnames if pred(fname))
return None
re_call_other = re.compile(r"(?P<func>[^(]+)\(.*")
class Plugin(typing.Protocol):
def is_intrhandler(self, name: QName) -> bool: ...
# init_array returns a list of functions that are placed in the
# `.init_array.*` section; AKA functions marked with
# `__attribute__((constructor))`.
def init_array(self) -> typing.Collection[QName]: ...
# extra_includes returns a list of functions that are never
# called, but are included in the binary anyway. This may because
# it is an unused method in a used vtable. This may be because it
# is an atexit() callback (we never exit).
def extra_includes(self) -> typing.Collection[BaseName]: ...
def extra_nodes(self) -> typing.Collection[Node]: ...
def indirect_callees(
self, loc: str, line: str
) -> tuple[typing.Collection[QName], bool] | None: ...
def skipmodels(self) -> dict[BaseName, analyze.SkipModel]: ...
class PluginApplication:
_location_xform: typing.Callable[[str], str]
_plugins: list[Plugin]
def __init__(
self, location_xform: typing.Callable[[str], str], plugins: list[Plugin]
) -> None:
self._location_xform = location_xform
self._plugins = plugins
def extra_nodes(self) -> typing.Collection[Node]:
ret: list[Node] = []
for plugin in self._plugins:
ret.extend(plugin.extra_nodes())
return ret
def indirect_callees(
self, elem: vcg.VCGElem
) -> tuple[typing.Collection[QName], bool]:
loc = elem.attrs.get("label", "")
line = read_source(loc)
for plugin in self._plugins:
ret = plugin.indirect_callees(loc, line)
if ret is not None:
return ret
placeholder = "__indirect_call"
if m := re_call_other.fullmatch(line):
placeholder += ":" + m.group("func")
placeholder += " at " + self._location_xform(elem.attrs.get("label", ""))
return [QName(placeholder)], False
def skipmodels(self) -> dict[BaseName, analyze.SkipModel]:
ret: dict[BaseName, analyze.SkipModel] = {}
for plugin in self._plugins:
ret.update(plugin.skipmodels())
return ret
|