tmc/tools/script_disassembler/script_disassembler.py

321 lines
10 KiB
Python

from dataclasses import dataclass
import struct
# Disassembler for tmc scripts
# Input 'macros' to generate the macros for the script commands
# Input the script bytes as hex to disassemble the script
@dataclass
class Context:
ptr: int
data: bytes
def u16_to_hex(value):
return '0x' + (struct.pack('>H', value).hex())
def barray_to_u16_hex(barray):
count = len(barray)//2
integers = struct.unpack('H'*count, barray)
return [u16_to_hex(x) for x in integers]
def u32_to_hex(value):
return '0x' + (struct.pack('>I', value).hex())
def barray_to_u32_hex(barray):
count = len(barray)//4
integers = struct.unpack('I'*count, barray)
return [u32_to_hex(x) for x in integers]
"""
def GetNextScriptCommandWordAfterCommandMetadata(ctx):
return struct.unpack('I', ctx.data[ctx.ptr+2:ctx.ptr+6])[0]
def sub_0807E10C(ctx):
meta = GetNextScriptCommandWordAfterCommandMetadata(ctx)
print('function', hex(meta))
def sub_0807E5F8(ctx):
curPtr = start + ctx.ptr
print('current', hex(curPtr))
# gUnk_02033280.unk_00 |= GetNextScriptCommandWordAfterCommandMetadata(unk2->unk_00.raw);
meta = GetNextScriptCommandWordAfterCommandMetadata(ctx)
print('meta', meta)
newPtr = curPtr | meta
print('Start next script at: ', hex(newPtr))
"""
commands = [
{'fun': 'nullsub_507', 'params': ''},
{'fun': 'sub_0807E004', 'params': '', 'name': 'start executing scripts'},
{'fun': 'sub_0807E014', 'params': '', 'name': 'stop executing scripts'},
{'fun': 'sub_0807E024', 'params': 's', 'name': 'jump by offset'},
{'fun': 'sub_0807E050', 'params': 's', 'name': 'jump if'},
{'fun': 'sub_0807E064', 'params': 's', 'name': 'jump if not'},
{'fun': 'sub_0807E078', 'params': 's'},
{'fun': 'sub_0807E0A0', 'params': 'w','name': 'abs jump' },
{'fun': 'sub_0807E0B8', 'params': 'w', 'name': 'abs jump if'},
{'fun': 'sub_0807E0CC', 'params': 'w', 'name': 'abs jump if not'},
{'fun': 'sub_0807E0E0', 'params': 'w'},
{'fun': 'sub_0807E10C', 'params':'w', 'name': 'Execute function via pointer'},# 'exec': sub_0807E10C},
{'fun': 'sub_0807E124', 'params': 'ww'},
{'fun': 'sub_0807E148', 'params': 'w'},
{'fun': 'sub_0807E158', 'params': 'w'},
{'fun': 'sub_0807E188', 'params': 's'},
{'fun': 'sub_0807E1D8', 'params': 's'},
{'fun': 'sub_0807E4AC', 'params': 's'},
{'fun': 'sub_0807E200'},
{'fun': 'sub_0807E220'},
{'fun': 'sub_0807E240'},
{'fun': 'sub_0807E260', 'params': 's'},
{'fun': 'sub_0807E280'},
{'fun': 'sub_0807E2A8', 'params': 's'},
{'fun': 'sub_0807E2E4', 'params': ''},
{'fun': 'sub_0807E30C', 'params': ''},
{'fun': 'sub_0807E390'},
{'fun': 'sub_0807E3BC'},
{'fun': 'sub_0807E3E8'},
{'fun': 'sub_0807E40C', 'params': 's'},
{'fun': 'sub_0807E42C'},
{'fun': 'sub_0807E48C'},
{'fun': 'sub_0807E4CC'},
{'fun': 'sub_0807E4EC'},
{'fun': 'sub_0807E514'},
{'fun': 'sub_0807E538'},
{'fun': 'sub_0807E564'},
{'fun': 'sub_0807E584'},
{'fun': 'sub_0807E5A0'},
{'fun': 'sub_0807E5CC'},
{'fun': 'sub_0807E5F8', 'params': 'w'},# 'exec': sub_0807E5F8},
{'fun': 'sub_0807E610'},
{'fun': 'sub_0807E628'},
{'fun': 'sub_0807E634'},
{'fun': 'sub_0807E644'},
{'fun': 'sub_0807E650', 'params': 's'},
{'fun': 'sub_0807E65C'},
{'fun': 'sub_0807E668'},
{'fun': 'sub_0807E674'},
{'fun': 'sub_0807E680', 'params': 's'},
{'fun': 'sub_0807E690'},
{'fun': 'sub_0807E6AC', 'params': 'w'},
{'fun': 'sub_0807E6DC', 'params': ''},
{'fun': 'sub_0807E700'},
{'fun': 'sub_0807E72C'},
{'fun': 'sub_0807E75C'},
{'fun': 'sub_0807E778'},
{'fun': 'sub_0807E788'},
{'fun': 'sub_0807E79C'},
{'fun': 'nullsub_508'},
{'fun': 'sub_0807E7B0'},
{'fun': 'sub_0807E7C4'},
{'fun': 'sub_0807E7D8'},
{'fun': 'sub_0807E7EC'},
{'fun': 'sub_0807E800'},
{'fun': 'sub_0807E80C'},
{'fun': 'sub_0807E858'},
{'fun': 'sub_0807E864', 'params': ''},
{'fun': 'sub_0807E878', 'params': ''},
{'fun': 'sub_0807E888'},
{'fun': 'sub_0807E898'},
{'fun': 'sub_0807E8C4', 'params': 'w'},
{'fun': 'sub_0807E8D4'},
{'fun': 'sub_0807E8E4_0', 'params': ''}, # duplicate
{'fun': 'sub_0807E8E4_1', 'params': ''}, # duplicate
{'fun': 'sub_0807E8E4_2', 'params': ''}, # duplicate
{'fun': 'sub_0807E8E4_3', 'params': ''}, # duplicate
{'fun': 'sub_0807E908'},
{'fun': 'sub_0807E914'},
{'fun': 'sub_0807E924'},
{'fun': 'sub_0807E930', 'params': 's'},
{'fun': 'sub_0807E944', 'params': ''},
{'fun': 'sub_0807E974'},
{'fun': 'sub_0807E9D4', 'params': ''},
{'fun': 'sub_0807E9DC', 'params': ''},
{'fun': 'sub_0807E9E4'},
{'fun': 'sub_0807E9F0', 'params': ''},
{'fun': 'sub_0807EA4C', 'params': ''},
{'fun': 'sub_0807EA88'},
{'fun': 'sub_0807EA94', 'params': ''},
{'fun': 'sub_0807EAB4', 'params': 's'},
{'fun': 'sub_0807EAC0'},
{'fun': 'sub_0807EAD0', 'params': 'ss'},
{'fun': 'sub_0807EAF0'},
{'fun': 'sub_0807EB18'},
{'fun': 'sub_0807EB28', 'params': 's'},
{'fun': 'sub_0807EB38'},
{'fun': 'sub_0807EB44', 'params': 's'},
{'fun': 'sub_0807EB4C'},
{'fun': 'sub_0807EB74', 'params': ''},
{'fun': 'sub_0807EB8C', 'params': ''},
{'fun': 'sub_0807EBA8'},
{'fun': 'sub_0807EBB0'},
{'fun': 'sub_0807EBC0', 'params': 's'},
{'fun': 'sub_0807EBC8'},
{'fun': 'sub_0807EBD8'},
{'fun': 'sub_0807EBF4'},
{'fun': 'sub_0807EC08'},
{'fun': 'nullsub_509'},
{'fun': 'sub_0807EC1C', 'params': 's'},
{'fun': 'sub_0807EC64'},
{'fun': 'sub_0807EC94'},
{'fun': 'sub_0807ECC4'},
{'fun': 'sub_0807ECF4', 'params': 's'},
{'fun': 'sub_0807ED24'},
{'fun': 'sub_0807EDD4', 'params': 'ss'},
{'fun': 'sub_0807EE04'},
{'fun': 'sub_0807EE30', 'params': ''},
{'fun': 'sub_0807EEB4'},
{'fun': 'sub_0807EEF4'},
{'fun': 'sub_0807EF3C'},
{'fun': 'sub_0807EF80', 'params': 's'},
{'fun': 'sub_0807EF90'},
{'fun': 'sub_0807EFA0'},
{'fun': 'sub_0807EFAC', 'params': 's'},
{'fun': 'sub_0807EFD4'},
{'fun': 'sub_0807EFE4', 'params': ''},
{'fun': 'sub_0807EFF4'},
{'fun': 'sub_0807F004'},
{'fun': 'sub_0807F010'},
{'fun': 'sub_0807F034', 'params': 's'},
{'fun': 'sub_0807F050'},
{'fun': 'sub_0807F060'},
{'fun': 'sub_0807F078'},
{'fun': 'sub_0807F088'},
{'fun': 'sub_0807F098', 'params': ''},
{'fun': 'sub_0807F0A4'},
{'fun': 'sub_0807F0B4', 'params': 's'},
{'fun': 'sub_0807F0C8'}
]
# definitions for parameters
parameters = {
'': {
'length':0,
'param': '',
'expr': '',
'read': lambda ctx: ''
},
's': {
'length': 1,
'param': 's',
'expr': ' .short \s',
'read': lambda ctx: barray_to_u16_hex(ctx.data[ctx.ptr+2:ctx.ptr+4])[0]
},
'w': {
'length': 2,
'param': 'w',
'expr': ' .word \w',
'read': lambda ctx: barray_to_u32_hex(ctx.data[ctx.ptr+2:ctx.ptr+6])[0]
},
'ss': {
'length': 2,
'param': 'a,b',
'expr': ' .short \\a\n .short \\b',
'read': lambda ctx: ', '.join(barray_to_u16_hex(ctx.data[ctx.ptr+2:ctx.ptr+6]))
},
'ww': {
'length': 4,
'param': 'a,b',
'expr': ' .word \\a\n .word \\b',
'read': lambda ctx: ', '.join(barray_to_u32_hex(ctx.data[ctx.ptr+2:ctx.ptr+10]))
},
}
def ExecuteScriptCommandSet(ctx: Context):
cmd = struct.unpack('H', ctx.data[ctx.ptr:ctx.ptr+2])[0]
if cmd == 0 or cmd ==0xffff:
return 0
unk_06 = cmd >> 0xA
operationId = cmd & 0x3FF
#print(cmd)
#print('advance by', unk_06)
#print('command id', operationId)
command = commands[operationId]
#print(command)
param_length = unk_06 - 1
if unk_06 > 1:
meta = struct.unpack(
'H'*(unk_06-1), ctx.data[ctx.ptr+2:ctx.ptr+2*unk_06])
#print('meta', meta)
# Handle parameters
if not 'params' in command:
raise Exception('Parameters not defined for ' + command['fun'] + ' Should be of length ' + str(param_length))
if not command['params'] in parameters:
raise Exception('Parameter configuration ' + command['params'] + ' not defined')
params = parameters[command['params']]
if unk_06-1 != params['length']:
raise Exception(f'Call {command["fun"]} with ' + str(unk_06-1) +' length, while length of ' + str(params['length'])+' defined')
print(command['fun'] + ' ' + params['read'](ctx))
# Execute script
ctx.ptr += unk_06*2
return 1
def disassemble_script(input_data):
input_bytes = bytearray.fromhex(input_data)
ctx = Context(0, input_bytes)
while True:
if not ExecuteScriptCommandSet(ctx):
break
# Print rest (did not manage to get there)
print('\n'.join(['.short ' + x for x in barray_to_u16_hex(ctx.data[ctx.ptr:])]))
def generate_macros():
print('@ All the macro functions for scripts')
print('@ Generated by disassemble_script.py')
print('.macro SCRIPT_START name')
print(' .globl \\name')
print(' .section .text')
print('\\name:')
print('.endm')
print('.macro SCRIPT_END')
print(' .short 0xffff')
print('.endm')
print('')
for num, command in enumerate(commands):
if not 'params' in command:
#raise Exception('Parameters not defined for ' + command['fun'] + ' Should be of length ' + str(param_length))
continue # TODO
if not command['params'] in parameters:
raise Exception('Parameter configuration ' + command['params'] + ' not defined')
params = parameters[command['params']]
id = ((params['length']+1) << 0xA) + num
print(f'.macro {command["fun"]} {params["param"]}')
print(f' .short {u16_to_hex(id)}')
if params['expr'] != '':
print(params['expr'])
print('.endm')
print('')
#print('#define ' + command['fun'] + '(' + params['param'] + ') asm(".short '+u16_to_hex(id)+'");' + params['expr'])
print('')
def main():
# Read input
input_data = input()
if input_data.strip() == 'macros':
generate_macros()
return
disassemble_script(input_data)
main()