mirror of
https://github.com/pret/pokemon-reverse-engineering-tools.git
synced 2026-04-25 15:57:07 -05:00
199 lines
5.9 KiB
Python
199 lines
5.9 KiB
Python
# coding: utf-8
|
|
"""
|
|
RGBDS BSS section and constant parsing.
|
|
"""
|
|
|
|
import os
|
|
|
|
def make_wram_labels(wram_sections):
|
|
wram_labels = {}
|
|
for section in wram_sections:
|
|
for label in section['labels']:
|
|
if label['address'] not in wram_labels.keys():
|
|
wram_labels[label['address']] = []
|
|
wram_labels[label['address']] += [label['label']]
|
|
return wram_labels
|
|
|
|
def bracket_value(string, i=0):
|
|
return string.split('[')[1 + i*2].split(']')[0]
|
|
|
|
def read_bss_sections(bss):
|
|
sections = []
|
|
section = {
|
|
'name': None,
|
|
'type': None,
|
|
'bank': None,
|
|
'start': None,
|
|
'labels': [],
|
|
}
|
|
address = None
|
|
if type(bss) is not list: bss = bss.split('\n')
|
|
for line in bss:
|
|
|
|
comment_index = line.find(';')
|
|
line, comment = line[:comment_index].lstrip(), line[comment_index:]
|
|
|
|
if 'SECTION' == line[:7]:
|
|
if section: # previous
|
|
sections += [section]
|
|
|
|
section_def = line.split(',')
|
|
name = section_def[0].split('"')[1]
|
|
type_ = section_def[1].strip()
|
|
if len(section_def) > 2:
|
|
bank = bracket_value(section_def[2])
|
|
else:
|
|
bank = None
|
|
|
|
if '[' in type_:
|
|
address = int(bracket_value(type_).replace('$','0x'), 16)
|
|
else:
|
|
if address == None or bank != section['bank']:
|
|
for type__, addr in [
|
|
('VRAM', 0x8000),
|
|
('SRAM', 0xa000),
|
|
('WRAM0', 0xc000),
|
|
('WRAMX', 0xd000),
|
|
('HRAM', 0xff80),
|
|
]:
|
|
if type__ == type_ and section['type'] == type__:
|
|
address = addr
|
|
# else: keep going from this address
|
|
|
|
section = {
|
|
'name': name,
|
|
'type': type_,
|
|
'bank': bank,
|
|
'start': address,
|
|
'labels': [],
|
|
}
|
|
|
|
elif ':' in line:
|
|
# rgbds allows labels without :, but prefer convention
|
|
label = line[:line.find(':')]
|
|
if ';' not in label:
|
|
section['labels'] += [{
|
|
'label': label,
|
|
'address': address,
|
|
'length': 0,
|
|
}]
|
|
|
|
elif line[:3] == 'ds ':
|
|
length = eval(line[3:].replace('$','0x'))
|
|
address += length
|
|
# adjacent labels use the same space
|
|
for label in section['labels'][::-1]:
|
|
if label['length'] == 0:
|
|
label['length'] = length
|
|
else:
|
|
break
|
|
|
|
elif 'EQU' in line:
|
|
# some space is defined using constants
|
|
name, value = line.split('EQU')
|
|
name, value = name.strip(), value.strip().replace('$','0x').replace('%','0b')
|
|
globals()[name] = eval(value)
|
|
|
|
sections.append(section)
|
|
return sections
|
|
|
|
def constants_to_dict(constants):
|
|
return dict((eval(constant[constant.find('EQU')+3:constant.find(';')].replace('$','0x')), constant[:constant.find('EQU')].strip()) for constant in constants)
|
|
|
|
def scrape_constants(text):
|
|
if type(text) is not list:
|
|
text = text.split('\n')
|
|
return constants_to_dict([line for line in text if 'EQU' in line[:line.find(';')]])
|
|
|
|
def read_constants(filepath):
|
|
"""
|
|
Load lines from a file and call scrape_constants.
|
|
"""
|
|
lines = None
|
|
|
|
with open(filepath, "r") as file_handler:
|
|
lines = file_handler.readlines()
|
|
|
|
constants = scrape_constants(lines)
|
|
return constants
|
|
|
|
class WRAMProcessor(object):
|
|
"""
|
|
RGBDS BSS section and constant parsing.
|
|
"""
|
|
|
|
def __init__(self, config):
|
|
"""
|
|
Setup for WRAM parsing.
|
|
"""
|
|
self.config = config
|
|
|
|
self.paths = {}
|
|
self.paths["wram"] = os.path.join(self.config.path, "wram.asm")
|
|
self.paths["hram"] = os.path.join(self.config.path, "hram.asm")
|
|
self.paths["gbhw"] = os.path.join(self.config.path, "gbhw.asm")
|
|
|
|
def initialize(self):
|
|
"""
|
|
Read constants.
|
|
"""
|
|
self.setup_wram_sections()
|
|
self.setup_wram_labels()
|
|
self.setup_hram_constants()
|
|
self.setup_gbhw_constants()
|
|
|
|
def read_wram_sections(self):
|
|
"""
|
|
Opens the wram file and calls read_bss_sections.
|
|
"""
|
|
wram_content = None
|
|
wram_file_path = self.paths["wram"]
|
|
|
|
with open(wram_file_path, "r") as wram:
|
|
wram_content = wram.readlines()
|
|
|
|
wram_sections = read_bss_sections(wram_content)
|
|
return wram_sections
|
|
|
|
def setup_wram_sections(self):
|
|
"""
|
|
Call read_wram_sections and set a variable.
|
|
"""
|
|
self.wram_sections = self.read_wram_sections()
|
|
return self.wram_sections
|
|
|
|
def setup_wram_labels(self):
|
|
"""
|
|
Make wram labels based on self.wram_sections as input.
|
|
"""
|
|
self.wram_labels = make_wram_labels(self.wram_sections)
|
|
return self.wram_labels
|
|
|
|
def read_hram_constants(self):
|
|
"""
|
|
Read constants from hram.asm using read_constants.
|
|
"""
|
|
hram_constants = read_constants(self.paths["hram"])
|
|
return hram_constants
|
|
|
|
def setup_hram_constants(self):
|
|
"""
|
|
Call read_hram_constants and set a variable.
|
|
"""
|
|
self.hram_constants = self.read_hram_constants()
|
|
return self.hram_constants
|
|
|
|
def read_gbhw_constants(self):
|
|
"""
|
|
Read constants from gbhw.asm using read_constants.
|
|
"""
|
|
gbhw_constants = read_constants(self.paths["gbhw"])
|
|
return gbhw_constants
|
|
|
|
def setup_gbhw_constants(self):
|
|
"""
|
|
Call read_gbhw_constants and set a variable.
|
|
"""
|
|
self.gbhw_constants = self.read_gbhw_constants()
|
|
return self.gbhw_constants
|