mirror of
https://github.com/zeldaret/mm.git
synced 2024-11-26 22:30:58 +00:00
132 lines
3.8 KiB
Python
Executable File
132 lines
3.8 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
import argparse
|
|
import os
|
|
import sys
|
|
import subprocess
|
|
import tempfile
|
|
from pathlib import Path
|
|
|
|
script_dir = Path(os.path.dirname(os.path.realpath(__file__)))
|
|
root_dir = script_dir / ".."
|
|
src_dir = root_dir / "src"
|
|
|
|
# Project-specific
|
|
CPP_FLAGS = [
|
|
"-Iinclude",
|
|
"-Iinclude/libc",
|
|
"-Isrc",
|
|
"-Ibuild/n64-us",
|
|
"-I.",
|
|
|
|
"-DF3DEX_GBI_2",
|
|
"-DF3DEX_GBI_PL",
|
|
"-DGBI_DOWHILE",
|
|
"-D__sgi",
|
|
"-D_LANGUAGE_C",
|
|
"-DNON_MATCHING",
|
|
"-D_Static_assert(x, y)=",
|
|
"-D__attribute__(x)="
|
|
"-D_MIPS_SZLONG=32",
|
|
"-ffreestanding",
|
|
"-DM2CTX",
|
|
|
|
"-std=gnu89",
|
|
]
|
|
|
|
# Read through the processes context and replace whatever
|
|
def custom_replacements(output):
|
|
actorList = []
|
|
output = output.splitlines()
|
|
|
|
i = 0
|
|
while i < len(output):
|
|
line = output[i]
|
|
|
|
############### actorLists[2].first -> Player* ###############
|
|
if "typedef struct ActorListEntry " in line:
|
|
actorListText = ""
|
|
i += 1
|
|
while not output[i].startswith("}"):
|
|
actorListText += output[i]
|
|
i += 1
|
|
actorCats = [
|
|
"actorSwitch",
|
|
"bg",
|
|
"player",
|
|
"explosive",
|
|
"npc",
|
|
"enemy",
|
|
"prop",
|
|
"itemAction",
|
|
"misc",
|
|
"boss",
|
|
"door",
|
|
"chest",
|
|
]
|
|
for x in range(12):
|
|
actorList.append(actorListText.replace("first;", f"{actorCats[x]};") + "\n")
|
|
if x == 2:
|
|
actorList[x] = actorList[x].replace("Actor*", "struct Player*")
|
|
elif "ActorListEntry actorLists[ACTORCAT_MAX];" in line:
|
|
output[i] = "struct {\n" + "".join(actorList) + "};"
|
|
########################################################
|
|
|
|
i += 1
|
|
return "\n".join(output)
|
|
|
|
def import_c_file(in_file) -> str:
|
|
in_file = os.path.relpath(in_file, root_dir)
|
|
|
|
cpp_command = ["gcc", "-E", "-P", "-undef", "-dM", *CPP_FLAGS, in_file]
|
|
cpp_command2 = ["gcc", "-E", "-P", "-undef", *CPP_FLAGS, in_file]
|
|
|
|
with tempfile.NamedTemporaryFile(suffix=".c") as tmp:
|
|
stock_macros = subprocess.check_output(["gcc", "-E", "-P", "-undef", "-dM", tmp.name], cwd=root_dir, encoding="utf-8")
|
|
|
|
out_text = ""
|
|
try:
|
|
out_text += subprocess.check_output(cpp_command, cwd=root_dir, encoding="utf-8")
|
|
out_text += subprocess.check_output(cpp_command2, cwd=root_dir, encoding="utf-8")
|
|
except subprocess.CalledProcessError:
|
|
print(
|
|
"Failed to preprocess input file, when running command:\n"
|
|
+ " ".join(cpp_command),
|
|
file=sys.stderr,
|
|
)
|
|
sys.exit(1)
|
|
|
|
if not out_text:
|
|
print("Output is empty - aborting")
|
|
sys.exit(1)
|
|
|
|
for line in stock_macros.strip().splitlines():
|
|
out_text = out_text.replace(line + "\n", "")
|
|
return out_text
|
|
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser(usage="./m2ctx.py path/to/file.c or ./m2ctx.py (from an actor or gamestate's asm dir)",
|
|
description="Creates a ctx.c file for m2c or decomp.me. "
|
|
"Output will be saved as ctx.c")
|
|
parser.add_argument('filepath', help="path of c file to be processed")
|
|
parser.add_argument("--custom", "-c", dest="custom", action="store_true", default=False,
|
|
help="Apply custom replacements to the output to help aid m2c output")
|
|
args = parser.parse_args()
|
|
|
|
c_file_path = args.filepath
|
|
print("Using file: {}".format(c_file_path))
|
|
|
|
output = import_c_file(c_file_path)
|
|
|
|
if args.custom:
|
|
output = custom_replacements(output)
|
|
|
|
ctxPath = root_dir / "ctx.c"
|
|
with ctxPath.open("w", encoding="UTF-8") as f:
|
|
f.write(output)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|