mkdd/common.py
2024-06-03 18:33:17 +02:00

482 lines
11 KiB
Python

"""
Common functions & definitions
"""
from dataclasses import dataclass
from enum import Enum
from hashlib import sha1
import json
import os
from shutil import which
from subprocess import PIPE, run
from sys import executable as PYTHON, platform
from typing import List, Tuple, Union
import yaml
try:
from yaml import CLoader as Loader
except ImportError:
from yaml import Loader
#############
# Functions #
#############
def get_file_sha1(path: str) -> bytes:
"""Gets the SHA1 hash of a file"""
with open(path, 'rb') as f:
return sha1(f.read()).digest()
def get_cmd_stdout(cmd: str, text=True) -> str:
"""Run a command and get the stdout output as a string"""
ret = run(cmd.split(), stdout=PIPE, text=text)
assert ret.returncode == 0, f"Command '{cmd}' returned {ret.returncode}"
return ret.stdout
class Binary(Enum):
DOL = 1
REL = 2
# ppcdis source output
SourceDesc = Union[str, Tuple[str, int, int]]
def get_containing_slice(addr: int) -> Tuple[Binary, SourceDesc]:
"""Finds the binary containing an address and its source file
Source file is empty string if not decompiled"""
dol_raw = get_cmd_stdout(
f"{SLICES} {DOL_YML} {DOL_SLICES} --containing {addr:x}")
containing = json.loads(dol_raw)
return (Binary.DOL, containing)
def lookup_sym(sym: str, dol: bool = False, rel: bool = False, source_name: str = None) -> int:
"""Takes a symbol as a name or address and returns the address"""
# Get binary
if dol:
binary_name = DOL_YML
else:
binary_name = None
# Determine type
try:
return int(sym, 16)
except ValueError:
return get_address(sym, binary_name, source_name)
def lookup_sym_full(sym: str, dol: bool = False, rel: bool = False, source_name: str = None
) -> int:
"""Takes a symbol as a name or address and returns both the name and address"""
# Get binary
if dol:
binary_name = DOL_YML
else:
binary_name = None
# Determine type
try:
return int(sym, 16), get_name(sym)
except ValueError:
return get_address(sym, binary_name, source_name), sym
def get_address(name: str, binary: bool = None, source_name: bool = None) -> int:
"""Finds the address of a symbol"""
args = [name]
if binary is not None:
args.append(f"-b {binary}")
if source_name is not None:
args.append(f"-n {source_name}")
raw = get_cmd_stdout(
f"{SYMBOLSCRIPT} {SYMBOLS} --get-addr {' '.join(args)}")
return json.loads(raw)
def get_name(addr: int, binary: bool = None, source_name: bool = None) -> int:
"""Finds the name of a symbol"""
args = [addr]
if binary is not None:
args.append(f"-b {binary}")
if source_name is not None:
args.append(f"-n {source_name}")
raw = get_cmd_stdout(
f"{SYMBOLSCRIPT} {SYMBOLS} --get-name {' '.join(args)}")
return json.loads(raw)
def find_headers(dirname: str, base=None) -> List[str]:
"""Returns a list of all headers in a folder recursively"""
if base is None:
base = dirname
ret = []
for name in os.listdir(dirname):
path = dirname + '/' + name
if os.path.isdir(path):
ret.extend(find_headers(path, base))
elif name.endswith('.h'):
ret.append(path[len(base)+1:])
return ret
def load_from_yaml(path: str, default=None):
"""Loads an object from a yaml file"""
if default is None:
default = {}
with open(path) as f:
ret = yaml.load(f.read(), Loader)
if ret is None:
ret = default
return ret
##################
# Target options #
##################
BUILD_YML = load_from_yaml("config/build_opts.yml")
REGION = BUILD_YML['region']
VERSION = BUILD_YML['version']
################
# Project dirs #
################
# Directory for decompiled dol code
DOL_SRCDIR = "src"
# Include directory
INCDIR = "include"
# Tools directory
TOOLS = "tools"
# Main config directory
MAIN_CONFIG = "config"
# subdir
VERSION_DIR = f"{VERSION}_{REGION}"
# Build artifacts directory
BUILDDIR = f"build/{VERSION_DIR}"
# Build include directory
BUILD_INCDIR = f"{BUILDDIR}/include"
# Output binaries directory
OUTDIR = f"out/{VERSION_DIR}"
# Original binaries directory
ORIG = f"orig/{VERSION_DIR}"
# Version specififc config directory
CONFIG = f"config/{VERSION_DIR}"
#########
# Tools #
#########
# ppcdis
PPCDIS = "tools/ppcdis"
PPCDIS_INCDIR = f"{PPCDIS}/include"
ANALYSER = f"{PYTHON} {PPCDIS}/analyser.py"
DISASSEMBLER = f"{PYTHON} {PPCDIS}/disassembler.py"
ORDERSTRINGS = f"{PYTHON} {PPCDIS}/orderstrings.py"
ORDERFLOATS = f"{PYTHON} {PPCDIS}/orderfloats.py"
ASSETRIP = f"{PYTHON} {PPCDIS}/assetrip.py"
ASSETINC = f"{PYTHON} {PPCDIS}/assetinc.py"
FORCEFILESGEN = f"{PYTHON} {PPCDIS}/forcefilesgen.py"
ELF2DOL = f"{PYTHON} {PPCDIS}/elf2dol.py"
FORCEACTIVEGEN = f"{PYTHON} {PPCDIS}/forceactivegen.py"
SLICES = f"{PYTHON} {PPCDIS}/slices.py"
PROGRESS = f"{PYTHON} {PPCDIS}/progress.py"
SYMBOLSCRIPT = f"{PYTHON} {PPCDIS}/symbols.py"
# Codewarrior
SDK_CW = os.path.join(TOOLS, "1.2.5")
SDK_CC = os.path.join(SDK_CW, "mwcceppc.exe")
SDK_PACTHED_CW = os.path.join(TOOLS, "1.2.5n")
SDK_PACTHED_CC = os.path.join(SDK_PACTHED_CW, "mwcceppc.exe")
JSYSTEM_O0_MW = os.path.join(TOOLS, "3.0a5.2")
JSYSTEM_O0_CC = os.path.join(JSYSTEM_O0_MW, "mwcceppc.exe")
CODEWARRIOR = os.path.join(TOOLS, "2.6")
CC = os.path.join(CODEWARRIOR, "mwcceppc.exe")
LD = os.path.join(CODEWARRIOR, "mwldeppc.exe")
if platform != "win32":
if(which("wibo") is not None):
WIN32_WRAPPER = "wibo"
elif(which("wine") is not None):
WIN32_WRAPPER = "wine"
assert WIN32_WRAPPER != "" "Wine or Wibo not found!"
SDK_CC = f"{WIN32_WRAPPER} {SDK_CC}"
SDK_PACTHED_CC = f"{WIN32_WRAPPER} {SDK_PACTHED_CC}"
CC_1_3_2 = f"{WIN32_WRAPPER} {CC_1_3_2}"
JSYSTEM_O0_CC = f"{WIN32_WRAPPER} {JSYSTEM_O0_CC}"
CC = f"{WIN32_WRAPPER} {CC}"
LD = f"{WIN32_WRAPPER} {LD}"
# DevkitPPC
DEVKITPPC = os.environ.get("DEVKITPPC")
# in case you know what you're doing you can get a version of devkitPPC and put it inside the tools folder
if DEVKITPPC is None:
DEVKITPPC = os.path.join(TOOLS, "devkitppc")
assert(os.path.isdir(DEVKITPPC))
# not tested but workaround for incorrect devkitppc path on windows
if platform == "win32" and DEVKITPPC.startswith("/opt/"):
DEVKITPPC = DEVKITPPC.replace("/opt/", "C:/")
AS = os.path.join(DEVKITPPC, "bin", "powerpc-eabi-as")
OBJDUMP = os.path.join(DEVKITPPC, "bin", "powerpc-eabi-objdump")
CPP = os.path.join(DEVKITPPC, "bin", "powerpc-eabi-cpp")
ICONV = f"{PYTHON} tools/sjis.py" # TODO: get actual iconv working(?)
#########
# Files #
#########
# Slices
DOL_SLICES = f"{CONFIG}/dol_slices.yml"
# Overrides (TODO: do these need to be separate for rel?)
ANALYSIS_OVERRIDES = f"{CONFIG}/analysis_overrides.yml"
DISASM_OVERRIDES = f"{MAIN_CONFIG}/disasm_overrides.yml"
# Binaries
DOL = f"{ORIG}/main.dol" # read in python code
DOL_YML = f"{CONFIG}/dol.yml"
DOL_SHA = f"{ORIG}/main.dol.sha1"
DOL_OK = f"{BUILDDIR}/main.dol.ok"
DOL_ASM_LIST = f"{BUILDDIR}/main.dol.asml"
# Symbols
SYMBOLS = f"{CONFIG}/symbols.yml"
# Assets
ASSETS_YML = f"{CONFIG}/assets.yml"
# Analysis outputs
EXTERNS = f"{BUILDDIR}/externs.pickle"
DOL_LABELS = f"{BUILDDIR}/labels.pickle"
DOL_RELOCS = f"{BUILDDIR}/relocs.pickle"
# Linker
DOL_LCF_TEMPLATE = f"{MAIN_CONFIG}/dol.lcf"
DOL_LCF = f"{BUILDDIR}/dol.lcf"
# Outputs
DOL_ELF = f"{BUILDDIR}/main.elf"
DOL_OUT = f"{OUTDIR}/main.dol"
DOL_MAP = f"{OUTDIR}/debugInfo{VERSION[0]}.MAP"
# Optional full disassembly
DOL_FULL = f"{OUTDIR}/dol.s"
DOL_SDATA2_SIZE = 4
# dol SHA1 Hash
DOL_SHA1_HASH = "f3bf225dd81cd9eb094fa9f8415f95f6bbcb9d10" # PAL SHA1 Hash
if (VERSION == "MarioClub"):
DOL_SHA1_HASH = "db87a9ec1a34275efc45d965dcdcb1a9eb131885" # NTSC-U Debug SHA1 Hash
##############
# Tool Flags #
##############
ASFLAGS = ' '.join([
"-m gekko",
f"-I {INCDIR}",
f"-I {PPCDIS_INCDIR}",
f"-I orig"
])
INCDIRS = [
PPCDIS_INCDIR,
BUILD_INCDIR,
"include",
"libs/PowerPC_EABI_Support/include"
]
MWCC_INCLUDES = ' '.join(f"-i {d}" for d in INCDIRS)
GCC_INCLUDES = ' '.join(f"-I {d}" for d in INCDIRS)
#rework
DEFINES = [
"DEBUG",
"HIO_SCREENSHOT",
"REGION_US",
"MATCHING"
]
if REGION == "eu":
DEFINES = [
"VIDEO_PAL",
"REGION_EU",
"MATCHING"
]
MWCC_DEFINES = ' '.join(f"-d {d}" for d in DEFINES)
GCC_DEFINES = ' '.join(f"-D {d}" for d in DEFINES)
CPPFLAGS = ' '.join([
"-nostdinc",
GCC_DEFINES,
GCC_INCLUDES
])
CFLAGS = [
"-lang=c++",
"-fp fmadd",
"-fp_contract on",
"-Cpp_exceptions off",
"-rostr",
"-RTTI off",
"-char signed",
"-enum int",
"-use_lmw_stmw on",
"-common on",
"-inline auto",
MWCC_DEFINES
]
JSYSTEM_O0 = [ # used for when trying to match something from TP Debug
"-lang=c++",
"-inline off",
"-fp hard",
"-O0",
"-d JGADGET_DEBUG",
"-enum int",
"-str reuse",
"-Cpp_exceptions off",
MWCC_DEFINES
]
JSYSTEM_SPEED = CFLAGS + [ "-O4,p" ]
JSYSTEM_RELEASE = CFLAGS + ["-opt level=4, schedule"]
JAUDIO_RELEASE = CFLAGS + ["-opt level=4, schedule, speed"]
JAUDIO_DSP = CFLAGS + [
"-O4,s",
"-inline noauto",
"-use_lmw_stmw off",
"-func_align 32"
]
# confusion
MSL_C_DEBUG = [
"-opt level=0, peephole, schedule, nospace",
"-inline off, deferred",
"-sym on",
"-enum int",
"-rostr",
"-str pool",
"-fp hard",
"-fp_contract on",
"-use_lmw_stmw on",
"-common off",
"-Cpp_exceptions off",
"-RTTI off"
]
MSL_C = [
"-O4,p",
"-inline auto, deferred",
"-common off",
"-enum int",
"-rostr",
"-str pool",
"-fp hard",
"-fp_contract on",
"-use_lmw_stmw on",
"-common off",
"-Cpp_exceptions off",
"-RTTI off"
]
SDK = [
"-lang=c",
"-O4,p",
"-inline auto",
"-fp_contract off",
"-enum int",
"-str reuse",
"-fp hard",
"-use_lmw_stmw on",
"-Cpp_exceptions off",
"-RTTI off"
]
BASE_GAME_CFLAGS = CFLAGS + [ "-O4,s" ]
KANESHIGE = BASE_GAME_CFLAGS + [ "-inline off" ]
LOCAL_CFLAGS = [
"-nostdinc",
"-w off",
"-proc gekko",
"-maxerrors 1",
"-I-",
MWCC_INCLUDES
]
DOL_CFLAGS = ' '.join(BASE_GAME_CFLAGS + LOCAL_CFLAGS)
MSL_C_DEBUG_CFLAGS = ' '.join(MSL_C_DEBUG + LOCAL_CFLAGS)
MSL_C_CFLAGS = ' '.join(MSL_C + LOCAL_CFLAGS)
SDK_CFLAGS = ' '.join(SDK + LOCAL_CFLAGS)
JSYSTEM_SPEED_CFLAGS = ' '.join(JSYSTEM_SPEED + LOCAL_CFLAGS)
JSYSTEM_RELEASE_CFLAGS = ' '.join(JSYSTEM_RELEASE + LOCAL_CFLAGS)
JSYSTEM_O0_CFLAGS = ' '.join(JSYSTEM_O0 + LOCAL_CFLAGS)
JAUDIO_RELEASE_CFLAGS = ' '.join(JAUDIO_RELEASE + LOCAL_CFLAGS)
JAUDIO_DSP_CFLAGS = ' '.join(JAUDIO_DSP + LOCAL_CFLAGS)
KANESHIGE_CFLAGS = ' '.join(BASE_GAME_CFLAGS + LOCAL_CFLAGS)
if (VERSION == "MarioClub"):
KANESHIGE_CFLAGS = ' '.join(KANESHIGE + LOCAL_CFLAGS)
EXTERNAL_DOL_CFLAGS = ' '.join(BASE_GAME_CFLAGS)
LDFLAGS = ' '.join([
"-fp hard",
"-w off",
"-maxerrors 1"
])
PPCDIS_ANALYSIS_FLAGS = ' '.join([
f"-o {ANALYSIS_OVERRIDES}"
])
PPCDIS_DISASM_FLAGS = ' '.join([
f"-m {SYMBOLS}",
f"-o {DISASM_OVERRIDES}"
])
@dataclass
class SourceContext:
cflags: str
binary: str
labels: str
relocs: str
slices: str
sdata2_threshold: int
DOL_CTX = SourceContext(DOL_CFLAGS, DOL_YML, DOL_LABELS, DOL_RELOCS, DOL_SLICES, DOL_SDATA2_SIZE)
####################
# diff.py Expected #
####################
EXPECTED = f"expected/{VERSION_DIR}"
DOL_EXPECTED = f"{EXPECTED}/build/main.elf"