-
Notifications
You must be signed in to change notification settings - Fork 613
/
Copy pathsym_info.py
executable file
·335 lines (275 loc) · 9.33 KB
/
sym_info.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
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
#!/usr/bin/env python3
# SPDX-FileCopyrightText: © 2025 ZeldaRET
# SPDX-License-Identifier: CC0-1.0
import argparse
import bisect
from dataclasses import dataclass
from pathlib import Path
from typing import Optional
import struct
import sys
import elftools.elf.elffile
import mapfile_parser
@dataclass
class MdebugSymbolicHeader:
magic: int
vstamp: int
ilineMax: int
cbLine: int
cbLineOffset: int
idnMax: int
cbDnOffset: int
ipdMax: int
cbPdOffset: int
isymMax: int
cbSymOffset: int
ioptMax: int
cbOptOffset: int
iauxMax: int
cbAuxOffset: int
issMax: int
cbSsOffset: int
issExtMax: int
cbSsExtOffset: int
ifdMax: int
cbFdOffset: int
crfd: int
cbRfdOffset: int
iextMax: int
cbExtOffset: int
@dataclass
class MdebugFileDescriptor:
adr: int
rss: int
issBase: int
cbSs: int
isymBase: int
csym: int
ilineBase: int
cline: int
ioptBase: int
copt: int
ipdFirst: int
cpd: int
iauxBase: int
caux: int
rfdBase: int
crfd: int
bitfield: int
cbLineOffset: int
cbLine: int
@dataclass
class MdebugSymbol:
iss: int
value: int
st: int
sc: int
index: int
@dataclass
class LocalSymbol:
name: str
address: int
def read_mdebug_symbolic_header(f, offset: int) -> MdebugSymbolicHeader:
f.seek(offset)
data = f.read(96)
return MdebugSymbolicHeader(*struct.unpack(">2H23I", data))
def read_mdebug_file_descriptor(f, offset: int) -> MdebugFileDescriptor:
f.seek(offset)
data = f.read(72)
return MdebugFileDescriptor(*struct.unpack(">I2iI6iHh4iI2I", data))
def read_mdebug_symbol(f, offset: int) -> MdebugSymbol:
f.seek(offset)
data = f.read(12)
word0, word1, word2 = struct.unpack(">III", data)
return MdebugSymbol(
word0, word1, (word2 >> 26) & 0x3F, (word2 >> 21) & 0x1F, word2 & 0xFFFFF
)
def read_mdebug_string(f, offset: int) -> str:
f.seek(offset)
data = bytearray()
while True:
char = f.read(1)[0]
if char == 0:
break
data.append(char)
return data.decode("ascii")
def read_local_symbols_from_mdebug(elf_path: Path) -> list[LocalSymbol]:
local_symbols = []
with open(elf_path, "r+b") as f:
elf = elftools.elf.elffile.ELFFile(f)
mdebug_offset = 0
for section in elf.iter_sections():
if section.name == ".mdebug":
mdebug_offset = section["sh_offset"]
break
if mdebug_offset == 0:
print(f"No .mdebug section found in '{elf_path}'")
return []
symbolic_header = read_mdebug_symbolic_header(f, mdebug_offset)
for fd_num in range(symbolic_header.ifdMax):
fd = read_mdebug_file_descriptor(
f, symbolic_header.cbFdOffset + fd_num * 72
)
for sym_num in range(fd.isymBase, fd.isymBase + fd.csym):
sym = read_mdebug_symbol(f, symbolic_header.cbSymOffset + sym_num * 12)
if sym.st == 2: # stStatic
if not (
sym.sc == 2 or sym.sc == 3 or sym.sc == 15
): # scData, scBss, scRData
continue
sym_name = read_mdebug_string(
f, symbolic_header.cbSsOffset + fd.issBase + sym.iss
)
# EGCS mangles names of internal variables, and seemingly ":V" is for in-function static variables
if "." in sym_name:
continue
if ":" in sym_name:
sym_name, rest = sym_name.split(":", 1)
if not rest.startswith("V"):
continue
local_symbols.append(LocalSymbol(sym_name, sym.value))
elif sym.st == 14: # stStaticProc
sym_name = read_mdebug_string(
f, symbolic_header.cbSsOffset + fd.issBase + sym.iss
)
local_symbols.append(LocalSymbol(sym_name, sym.value))
return local_symbols
def merge_local_symbols(
map_file: mapfile_parser.mapfile.MapFile, local_symbols: list[LocalSymbol]
):
local_symbols.sort(key=lambda s: s.address)
for segment in map_file:
for file in segment:
# TODO: handle segmented addresses?
if file.vram < 0x80000000:
continue
start_address = file.vram
end_address = file.vram + file.size
start_index = bisect.bisect_left(
local_symbols, start_address, key=lambda s: s.address
)
end_index = bisect.bisect_left(
local_symbols, end_address, key=lambda s: s.address
)
if start_index == end_index:
continue
symbols = file.copySymbolList()
for sym in local_symbols[start_index:end_index]:
if file.vrom is None:
vrom = None
else:
vrom = sym.address - start_address + file.vrom
symbols.append(
mapfile_parser.mapfile.Symbol(
sym.name, sym.address, None, vrom, None
)
)
symbols.sort(key=lambda s: s.vram)
# Recompute symbol sizes
for i in range(len(symbols)):
if i == len(symbols) - 1:
symbols[i].size = end_address - symbols[i].vram
else:
symbols[i].size = symbols[i + 1].vram - symbols[i].vram
file.setSymbolList(symbols)
def find_symbols_by_name(
map_file: mapfile_parser.mapfile.MapFile, sym_name: str
) -> list[mapfile_parser.mapfile.FoundSymbolInfo]:
infos = []
for segment in map_file:
for file in segment:
for sym in file:
if sym.name == sym_name:
infos.append(mapfile_parser.mapfile.FoundSymbolInfo(file, sym))
return infos
def print_map_file(map_file: mapfile_parser.mapfile.MapFile):
for segment in map_file:
print(f"{segment.name}")
for file in segment:
# Ignore debug sections
if (
file.sectionType in (".pdr", ".line", ".gnu.attributes")
or file.sectionType.startswith(".debug")
or file.sectionType.startswith(".mdebug")
):
continue
print(f" {file.asStr()}")
for sym in file:
vram_str = f"{sym.vram:08X}"
if sym.vrom is None:
vrom_str = " "
else:
vrom_str = f"{sym.vrom:06X}"
print(f" {vram_str} {vrom_str} {sym.name}")
def sym_info_main():
parser = argparse.ArgumentParser(
description="Display various information about symbol or addresses."
)
parser.add_argument(
"symname",
nargs="?",
help="symbol name or VROM/VRAM address to lookup. If not given, all symbols will be printed.",
)
parser.add_argument(
"-e",
"--expected",
dest="use_expected",
action="store_true",
help="use the map file and elf in expected/build/ instead of build/",
)
parser.add_argument(
"-v",
"--version",
dest="oot_version",
help="which version should be processed (default: gc-eu-mq-dbg)",
default="gc-eu-mq-dbg",
)
args = parser.parse_args()
BUILTMAP = Path("build") / args.oot_version / f"oot-{args.oot_version}.map"
BUILTELF = Path("build") / args.oot_version / f"oot-{args.oot_version}.elf"
map_path = BUILTMAP
elf_path = BUILTELF
if args.use_expected:
map_path = "expected" / BUILTMAP
elf_path = "expected" / BUILTELF
if not map_path.exists():
print(f"Could not find map_file at '{map_path}'")
sys.exit(1)
map_file = mapfile_parser.mapfile.MapFile()
map_file.readMapFile(map_path)
if elf_path.exists():
local_symbols = read_local_symbols_from_mdebug(elf_path)
merge_local_symbols(map_file, local_symbols)
else:
print(
f"Could not find ELF file at '{elf_path}', local symbols will not be available"
)
sym_name = args.symname
if sym_name is None:
print_map_file(map_file)
sys.exit(0)
infos: list[mapfile_parser.mapfile.FoundSymbolInfo] = []
possible_files: list[mapfile_parser.mapfile.File] = []
try:
address = int(sym_name, 0)
if address >= 0x01000000:
info, possible_files = map_file.findSymbolByVram(address)
if info is not None:
infos = [info]
else:
info, possible_files = map_file.findSymbolByVrom(address)
if info is not None:
infos = [info]
except ValueError:
infos = find_symbols_by_name(map_file, sym_name)
if not infos:
print(f"'{sym_name}' not found in map file '{map_path}'")
if len(possible_files) > 0:
print("But it may be a local symbol of either of the following files:")
for f in possible_files:
print(f" {f.asStr()})")
sys.exit(1)
for info in infos:
print(info.getAsStrPlusOffset(sym_name))
if __name__ == "__main__":
sym_info_main()