tp/tools/libdol2asm/data/function/asm.py

136 lines
4.0 KiB
Python

import struct
from dataclasses import dataclass, field
from typing import List, Set, Dict
from pathlib import Path
from ...builder import AsyncBuilder
from ...disassemble import AccessCollector
from ... import util
from .. import static_analyze
from ..base import *
from ..symbol import *
from .base import *
@dataclass
class Block():
identifier: Identifier
addr: int
size: int
@property
def start(self):
return self.addr
@property
def end(self):
return self.addr + self.size
def asm_reference(self, addr):
if addr != self.addr:
return None
return self.identifier.label
@dataclass(eq=False)
class ASMFunction(Function):
blocks: List[Block] = field(default_factory=list, repr=False)
include_path: Path = None
asm: bool = True
data: bytearray = None
def gather_references(self, context, valid_range):
addrs = static_analyze.function(self.data, self.addr, self.size)
function_range = AddressRange(self.start, self.end)
self.implicit_references = set([
addr
for addr in addrs.values()
if addr in valid_range and not addr in function_range
])
collector = AccessCollector([])
for i, addr in collector.execute_generator(self.addr, self.data, self.size):
pass
function_range = AddressRange(self.start, self.end)
self.references = set([
access.addr
for access in collector.accesses.values()
if access.addr in valid_range and not access.addr in function_range
])
"""
self.test_references = [
(access.at, access.addr)
for access in collector.accesses.values()
if access.addr in valid_range and not access.addr in function_range
]
"""
async def export_function_body(self, exporter, builder: AsyncBuilder):
await builder.write(f" {{")
await builder.write(f"\tnofralloc")
await builder.write(f"#include \"{self.include_path}\"")
await builder.write(f"}}")
async def export_declaration(self, exporter, builder: AsyncBuilder):
assert self.padding == 0
if False:
for k,v in self.test_references:
symbol_name = "???"
symbol = exporter.gst[-1, v]
if symbol:
symbol_name = symbol.label
await builder.write(f"//\t{k:08X}: {v:08X} ({symbol_name})")
await builder.write("#pragma push")
await builder.write("#pragma optimization_level 0")
await builder.write("#pragma optimizewithasm off")
if self.alignment:
await builder.write(f"#pragma function_align {self.alignment}")
await self.export_function_header(exporter, builder, forward=False, c_export=False, full_qualified_name=True)
await self.export_function_body(exporter, builder)
await builder.write("#pragma pop")
await builder.write("")
@staticmethod
def create(section, group):
first = group[0]
last = group[-1]
start = first.start
end = last.end
blocks = []
for symbol in group:
block = Block(
Identifier("lbl", symbol.addr, None),
symbol.addr, symbol.size,
)
blocks.append(block)
# Calculate additional padding from zeros at the end of the function
data = section.get_data(start, end)
end_padding = 0
last_data = list(util.chunks(data, 4))
for x in last_data[::-1]:
if struct.unpack('>I', x)[0] != 0:
break
end_padding += 4
if end_padding > 0:
data = data[:-end_padding]
end -= end_padding
return ASMFunction(
Identifier("func", start, first.name),
addr=start,
size=end - start,
padding=last.padding + end_padding,
alignment=0,
blocks=blocks,
source=first.source,
data=data)