mirror of
https://github.com/elliotnunn/ToolboxToolbox.git
synced 2025-01-17 20:30:50 +00:00
Flattener: faster and better
This commit is contained in:
parent
2ee3131a4e
commit
e434e16c75
@ -7,6 +7,13 @@ import struct
|
||||
import string
|
||||
|
||||
|
||||
OKCHARS = string.ascii_letters + string.digits + "_"
|
||||
|
||||
|
||||
def addr(seg):
|
||||
return 0x100000 * seg
|
||||
|
||||
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument("src", help="rdump file")
|
||||
parser.add_argument("dest", help="binary dest (may also create .txt file)")
|
||||
@ -28,40 +35,65 @@ if not resources or resources[0].id != 0:
|
||||
jt_resource, *other_resources = resources
|
||||
|
||||
bigboy = bytearray()
|
||||
for i, r in enumerate(resources):
|
||||
if i == 0:
|
||||
continue
|
||||
|
||||
while len(bigboy) < i * 0x100000:
|
||||
bigboy.append(0)
|
||||
for r in other_resources:
|
||||
bigboy.extend(bytes(addr(r.id) - len(bigboy)))
|
||||
bigboy.extend(r)
|
||||
|
||||
with open(args.dest, "wb") as f:
|
||||
f.write(bigboy)
|
||||
|
||||
jump_table = [] # (a5_ofs, segnum, seg_ofs)
|
||||
(jt_size, a5_offset_of_jt) = struct.unpack_from(">LL", jt_resource, 8)
|
||||
for jt_ofs in range(16, 16 + jt_size, 8):
|
||||
ofs, be_3f3c, segnum, be_a9f0 = struct.unpack_from(">HHHH", jt_resource, jt_ofs)
|
||||
if be_3f3c != 0x3F3C or be_a9f0 != 0xA9F0:
|
||||
break
|
||||
jump_table.append((jt_ofs - 16 + a5_offset_of_jt + 2, segnum, ofs + 4))
|
||||
|
||||
|
||||
with open(args.dest + ".py", "w") as idascript:
|
||||
# Find MacsBug symbols
|
||||
namedict = {}
|
||||
for b in range(0, len(bigboy), 2):
|
||||
if bigboy[b : b + 2] == b"NV": # link a6, starting a compiled function
|
||||
for c in range(b + 2, len(bigboy), 2):
|
||||
if bigboy[c : c + 2] == b"NV":
|
||||
break
|
||||
if 0x81 <= bigboy[c] < 0xB0:
|
||||
strlen = bigboy[c] & 0x0F
|
||||
if strlen < 2:
|
||||
break
|
||||
namestr = bigboy[c + 1 : c + 1 + strlen]
|
||||
if len(namestr) < strlen:
|
||||
break
|
||||
namestr = namestr.decode("latin-1")
|
||||
if not all(c in (string.ascii_letters + string.digits + "_") for c in namestr):
|
||||
break
|
||||
if strlen % 2 == 0 and bigboy[c + 1 + strlen : c + 1 + strlen + 1] not in b"\0":
|
||||
break
|
||||
for r in other_resources:
|
||||
targets = set(ofs for _, seg, ofs in jump_table if seg == r.id)
|
||||
|
||||
namedict[b] = namestr
|
||||
break
|
||||
bugnames = []
|
||||
lastfound = 0
|
||||
for i in range(0, len(r) - 2, 2):
|
||||
namelen = r[i + 2]
|
||||
if r[i : i + 2] not in (b"\x4e\x75", b"\x4e\xd0"):
|
||||
continue
|
||||
if not (0x81 <= namelen < 0xB0):
|
||||
continue
|
||||
namelen &= 0x3F
|
||||
if i + 3 + namelen > len(r):
|
||||
continue
|
||||
name = r[i + 3 : i + 3 + namelen].decode("latin-1")
|
||||
if not all(c in OKCHARS for c in name):
|
||||
continue
|
||||
|
||||
possibles = []
|
||||
for j in reversed(range(lastfound, i, 2)):
|
||||
if r[j : j + 2] == b"Nu":
|
||||
break # stop looking after an RTS
|
||||
if r[j : j + 2] == b"NV" or j in targets:
|
||||
possibles.append(j)
|
||||
|
||||
lastfound = i
|
||||
|
||||
if len(possibles) > 3:
|
||||
continue # don't bother with this name, too ambiguous
|
||||
|
||||
for j, p in enumerate(possibles, 1):
|
||||
namedict[addr(r.id) + p] = name if len(possibles) == 1 else f"{name}?{j}"
|
||||
|
||||
interseg_calls = {}
|
||||
for r in other_resources:
|
||||
for i in range(0, len(r) - 3, 2):
|
||||
if r[i : i + 2] in (b"\x4e\xad", b"\x48\x6d"):
|
||||
(targ,) = struct.unpack_from(">h", r, i + 2)
|
||||
if targ > 0:
|
||||
interseg_calls.setdefault(targ, []).append(addr(r.id) + i)
|
||||
|
||||
# Make some neat names for the segments...
|
||||
segnames = {}
|
||||
@ -71,18 +103,10 @@ with open(args.dest + ".py", "w") as idascript:
|
||||
else:
|
||||
segnames[r.id] = f"seg_{r.id:X}"
|
||||
|
||||
jt_size, a5_offset_of_jt = struct.unpack_from(">LL", jt_resource, 8)
|
||||
for a5_ofs, segnum, ofs in jump_table:
|
||||
bigboy_ofs = addr(segnum) + ofs
|
||||
|
||||
for jt_ofs in range(16, 16 + jt_size, 8):
|
||||
ofs, be_3f3c, segnum, be_a9f0 = struct.unpack_from(">HHHH", jt_resource, jt_ofs)
|
||||
if be_3f3c != 0x3F3C or be_a9f0 != 0xA9F0:
|
||||
break
|
||||
ofs += 4 # not sure what the leading stuff is?
|
||||
|
||||
bigboy_ofs = ((segnum) * 0x100000) + ofs
|
||||
a5_ofs = jt_ofs - 16 + a5_offset_of_jt + 2
|
||||
|
||||
cool_name = f"{segnames[segnum]}_"
|
||||
cool_name = f"{segnames[segnum]}$"
|
||||
if bigboy_ofs in namedict:
|
||||
cool_name += namedict[bigboy_ofs]
|
||||
del namedict[bigboy_ofs]
|
||||
@ -91,20 +115,9 @@ with open(args.dest + ".py", "w") as idascript:
|
||||
|
||||
print(f'MakeFunction(0x{bigboy_ofs:X}); MakeName(0x{bigboy_ofs:X}, "{cool_name}")', file=idascript)
|
||||
|
||||
call_to_me = struct.pack(">H", a5_ofs)
|
||||
bb_i = -1
|
||||
while 1:
|
||||
bb_i = bigboy.find(call_to_me, bb_i + 1)
|
||||
if bb_i == -1:
|
||||
break
|
||||
if bb_i % 2:
|
||||
continue
|
||||
if bigboy[bb_i - 2 : bb_i] not in (b"\x4e\xad", b"\x48\x6d"):
|
||||
continue # jsr/pea
|
||||
|
||||
# Okay, found one
|
||||
print(f'MakeCode(0x{bb_i-2:X}); op_man(0x{bb_i-2:X}, 0, "{cool_name}")', file=idascript)
|
||||
for caller in interseg_calls.get(a5_ofs, []):
|
||||
print(f'MakeCode(0x{caller:X}); op_man(0x{caller:X}, 0, "{cool_name}")', file=idascript)
|
||||
|
||||
for bigboy_ofs, name in sorted(namedict.items()):
|
||||
cool_name = f"{segnames[bigboy_ofs >> 20]}_{name}"
|
||||
cool_name = f"{segnames[bigboy_ofs >> 20]}${name}"
|
||||
print(f'MakeFunction(0x{bigboy_ofs:X}); MakeName(0x{bigboy_ofs:X}, "{cool_name}")', file=idascript)
|
||||
|
Loading…
x
Reference in New Issue
Block a user