mirror of
https://github.com/pmret/papermario.git
synced 2024-11-09 12:32:38 +01:00
ecb65c720a
* Change splat subrepo to use main branch * git subrepo pull --force tools/splat subrepo: subdir: "tools/splat" merged: "fbf3fa68a7" upstream: origin: "https://github.com/ethteck/splat.git" branch: "main" commit: "fbf3fa68a7" git-subrepo: version: "0.4.5" origin: "https://github.com/ingydotnet/git-subrepo" commit: "aa416e4" * building now * vram classes in the us yaml * more progrezz * more * prepare to deal with the shiftable build * revert a couple things * bring the yaml more in line with the old one while debugging * got statis * vacuum * git subrepo pull --force tools/splat subrepo: subdir: "tools/splat" merged: "d6490bb877" upstream: origin: "https://github.com/ethteck/splat.git" branch: "main" commit: "d6490bb877" git-subrepo: version: "0.4.5" origin: "https://github.com/ingydotnet/git-subrepo" commit: "aa416e4" * clean
321 lines
7.7 KiB
Python
Executable File
321 lines
7.7 KiB
Python
Executable File
#! /usr/bin/env python3
|
|
|
|
import argparse
|
|
import sys
|
|
from pathlib import Path
|
|
|
|
from util.gc import gcinfo
|
|
from util.n64 import find_code_length, rominfo
|
|
from util.psx import psxexeinfo
|
|
|
|
parser = argparse.ArgumentParser(
|
|
description="Create a splat config from an N64 ROM, PSX executable, or a GameCube disc image."
|
|
)
|
|
parser.add_argument(
|
|
"file", help="Path to a .z64/.n64 ROM, PSX executable, or .iso/.gcm GameCube image"
|
|
)
|
|
|
|
|
|
def main(file_path: Path):
|
|
if not file_path.exists():
|
|
sys.exit(f"File {file_path} does not exist ({file_path.absolute()})")
|
|
if file_path.is_dir():
|
|
sys.exit(f"Path {file_path} is a directory ({file_path.absolute()})")
|
|
|
|
# Check for N64 ROM
|
|
if file_path.suffix.lower() == ".n64" or file_path.suffix.lower() == ".z64":
|
|
create_n64_config(file_path)
|
|
return
|
|
|
|
file_bytes = file_path.read_bytes()
|
|
|
|
# Check for GC disc image
|
|
if int.from_bytes(file_bytes[0x1C:0x20], byteorder="big") == 0xC2339F3D:
|
|
create_gc_config(file_path, file_bytes)
|
|
return
|
|
|
|
# Check for PSX executable
|
|
if file_bytes[0:8] == b"PS-X EXE":
|
|
create_psx_config(file_path, file_bytes)
|
|
return
|
|
|
|
|
|
def create_n64_config(rom_path: Path):
|
|
rom_bytes = rominfo.read_rom(rom_path)
|
|
|
|
rom = rominfo.get_info(rom_path, rom_bytes)
|
|
basename = rom.name.replace(" ", "").lower()
|
|
|
|
header = f"""\
|
|
name: {rom.name.title()} ({rom.get_country_name()})
|
|
sha1: {rom.sha1}
|
|
options:
|
|
basename: {basename}
|
|
target_path: {rom_path.with_suffix(".z64")}
|
|
elf_path: build/{basename}.elf
|
|
base_path: .
|
|
platform: n64
|
|
compiler: {rom.compiler}
|
|
|
|
# asm_path: asm
|
|
# src_path: src
|
|
# build_path: build
|
|
# create_asm_dependencies: True
|
|
|
|
ld_script_path: {basename}.ld
|
|
ld_dependencies: True
|
|
|
|
find_file_boundaries: True
|
|
header_encoding: {rom.header_encoding}
|
|
|
|
o_as_suffix: True
|
|
use_legacy_include_asm: False
|
|
mips_abi_float_regs: o32
|
|
|
|
asm_function_macro: glabel
|
|
asm_jtbl_label_macro: jlabel
|
|
asm_data_macro: dlabel
|
|
|
|
# section_order: [".text", ".data", ".rodata", ".bss"]
|
|
# auto_all_sections: [".data", ".rodata", ".bss"]
|
|
|
|
symbol_addrs_path:
|
|
- symbol_addrs.txt
|
|
reloc_addrs_path:
|
|
- reloc_addrs.txt
|
|
|
|
# undefined_funcs_auto_path: undefined_funcs_auto.txt
|
|
# undefined_syms_auto_path: undefined_syms_auto.txt
|
|
|
|
extensions_path: tools/splat_ext
|
|
|
|
# string_encoding: ASCII
|
|
# data_string_encoding: ASCII
|
|
rodata_string_guesser_level: 2
|
|
data_string_guesser_level: 2
|
|
# libultra_symbols: True
|
|
# hardware_regs: True
|
|
# gfx_ucode: # one of [f3d, f3db, f3dex, f3dexb, f3dex2]
|
|
"""
|
|
|
|
first_section_end = find_code_length.run(rom_bytes, 0x1000, rom.entry_point)
|
|
|
|
segments = f"""\
|
|
segments:
|
|
- name: header
|
|
type: header
|
|
start: 0x0
|
|
|
|
- name: boot
|
|
type: bin
|
|
start: 0x40
|
|
|
|
- name: entry
|
|
type: code
|
|
start: 0x1000
|
|
vram: 0x{rom.entry_point:X}
|
|
subsegments:
|
|
- [0x1000, hasm]
|
|
|
|
- name: main
|
|
type: code
|
|
start: 0x{0x1000 + rom.entrypoint_info.entry_size:X}
|
|
vram: 0x{rom.entry_point + rom.entrypoint_info.entry_size:X}
|
|
follows_vram: entry
|
|
"""
|
|
|
|
if rom.entrypoint_info.bss_size is not None:
|
|
segments += f"""\
|
|
bss_size: 0x{rom.entrypoint_info.bss_size:X}
|
|
"""
|
|
|
|
segments += f"""\
|
|
subsegments:
|
|
- [0x{0x1000 + rom.entrypoint_info.entry_size:X}, asm]
|
|
"""
|
|
|
|
if (
|
|
rom.entrypoint_info.bss_size is not None
|
|
and rom.entrypoint_info.bss_start_address is not None
|
|
):
|
|
bss_start = rom.entrypoint_info.bss_start_address - rom.entry_point + 0x1000
|
|
# first_section_end points to the start of data
|
|
segments += f"""\
|
|
- [0x{first_section_end:X}, data]
|
|
- {{ start: 0x{bss_start:X}, type: bss, vram: 0x{rom.entrypoint_info.bss_start_address:08X} }}
|
|
"""
|
|
# Point next segment to the detected end of the main one
|
|
first_section_end = bss_start
|
|
|
|
segments += f"""\
|
|
|
|
- type: bin
|
|
start: 0x{first_section_end:X}
|
|
follows_vram: main
|
|
- [0x{rom.size:X}]
|
|
"""
|
|
|
|
out_file = f"{basename}.yaml"
|
|
with open(out_file, "w", newline="\n") as f:
|
|
print(f"Writing config to {out_file}")
|
|
f.write(header)
|
|
f.write(segments)
|
|
|
|
|
|
def create_gc_config(iso_path: Path, iso_bytes: bytes):
|
|
gc = gcinfo.get_info(iso_path, iso_bytes)
|
|
basename = gc.system_code + gc.game_code + gc.region_code + gc.publisher_code
|
|
|
|
header = f"""\
|
|
name: \"{gc.name.title()} ({gc.get_region_name()})\"
|
|
system_code: {gc.system_code}
|
|
game_code: {gc.game_code}
|
|
region_code: {gc.region_code}
|
|
publisher_code: {gc.publisher_code}
|
|
sha1: {gc.sha1}
|
|
options:
|
|
filesystem_path: filesystem
|
|
basename: {basename}
|
|
target_path: {iso_path.with_suffix(".iso")}
|
|
base_path: .
|
|
compiler: {gc.compiler}
|
|
platform: gc
|
|
# undefined_funcs_auto: True
|
|
# undefined_funcs_auto_path: undefined_funcs_auto.txt
|
|
# undefined_syms_auto: True
|
|
# undefined_syms_auto_path: undefined_syms_auto.txt
|
|
# symbol_addrs_path: symbol_addrs.txt
|
|
# asm_path: asm
|
|
# src_path: src
|
|
# build_path: build
|
|
# extensions_path: tools/splat_ext
|
|
# section_order: [".text", ".data", ".rodata", ".bss"]
|
|
# auto_all_sections: [".data", ".rodata", ".bss"]
|
|
"""
|
|
|
|
segments = f"""\
|
|
segments:
|
|
- name: filesystem
|
|
type: fst
|
|
path: filesystem/sys/fst.bin
|
|
- name: bootinfo
|
|
type: bootinfo
|
|
path: filesystem/sys/boot.bin
|
|
- name: bi2
|
|
type: bi2
|
|
path: filesystem/sys/bi2.bin
|
|
- name: apploader
|
|
type: apploader
|
|
path: filesystem/sys/apploader.img
|
|
- name: main
|
|
type: dol
|
|
path: filesystem/sys/main.dol
|
|
"""
|
|
|
|
out_file = f"{basename}.yaml"
|
|
with open(out_file, "w", newline="\n") as f:
|
|
print(f"Writing config to {out_file}")
|
|
f.write(header)
|
|
f.write(segments)
|
|
|
|
|
|
def create_psx_config(exe_path: Path, exe_bytes: bytes):
|
|
exe = psxexeinfo.PsxExe.get_info(exe_path, exe_bytes)
|
|
basename = exe_path.name.replace(" ", "").lower()
|
|
|
|
header = f"""\
|
|
name: {exe_path.name}
|
|
sha1: {exe.sha1}
|
|
options:
|
|
basename: {basename}
|
|
target_path: {exe_path}
|
|
base_path: .
|
|
platform: psx
|
|
compiler: GCC
|
|
|
|
# asm_path: asm
|
|
# src_path: src
|
|
# build_path: build
|
|
# create_asm_dependencies: True
|
|
|
|
ld_script_path: {basename}.ld
|
|
|
|
find_file_boundaries: False
|
|
gp_value: 0x{exe.initial_gp:08X}
|
|
|
|
o_as_suffix: True
|
|
use_legacy_include_asm: False
|
|
|
|
asm_function_macro: glabel
|
|
asm_jtbl_label_macro: jlabel
|
|
asm_data_macro: dlabel
|
|
|
|
section_order: [".rodata", ".text", ".data", ".bss"]
|
|
# auto_all_sections: [".data", ".rodata", ".bss"]
|
|
|
|
symbol_addrs_path:
|
|
- symbol_addrs.txt
|
|
reloc_addrs_path:
|
|
- reloc_addrs.txt
|
|
|
|
# undefined_funcs_auto_path: undefined_funcs_auto.txt
|
|
# undefined_syms_auto_path: undefined_syms_auto.txt
|
|
|
|
extensions_path: tools/splat_ext
|
|
|
|
subalign: 2
|
|
|
|
string_encoding: ASCII
|
|
data_string_encoding: ASCII
|
|
rodata_string_guesser_level: 2
|
|
data_string_guesser_level: 2
|
|
"""
|
|
|
|
segments = f"""\
|
|
segments:
|
|
- name: header
|
|
type: header
|
|
start: 0x0
|
|
|
|
- name: main
|
|
type: code
|
|
start: 0x800
|
|
vram: 0x{exe.destination_vram:X}
|
|
bss_size: 0x{exe.bss_size:X}
|
|
subsegments:
|
|
"""
|
|
text_offset = exe.text_offset
|
|
if text_offset != 0x800:
|
|
segments += f"""\
|
|
- [0x800, rodata, 800]
|
|
"""
|
|
segments += f"""\
|
|
- [0x{text_offset:X}, asm, {text_offset:X}]
|
|
"""
|
|
|
|
if exe.data_vram != 0 and exe.data_size != 0:
|
|
data_offset = exe.data_offset
|
|
segments += f"""\
|
|
- [0x{data_offset:X}, data, {data_offset:X}]
|
|
"""
|
|
|
|
if exe.bss_size != 0:
|
|
segments += f"""\
|
|
- {{ start: 0x{exe.size:X}, type: bss, name: {exe.bss_vram:X}, vram: 0x{exe.bss_vram:X} }}
|
|
"""
|
|
|
|
segments += f"""\
|
|
- [0x{exe.size:X}]
|
|
"""
|
|
|
|
out_file = f"{basename}.yaml"
|
|
with open(out_file, "w", newline="\n") as f:
|
|
print(f"Writing config to {out_file}")
|
|
f.write(header)
|
|
f.write(segments)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
args = parser.parse_args()
|
|
main(Path(args.file))
|