#ifndef __RECOMP_PORT__ #define __RECOMP_PORT__ #include #include #include #include #include #include #include #include #include #include "rabbitizer.hpp" #include "elfio/elfio.hpp" #ifdef _MSC_VER inline uint32_t byteswap(uint32_t val) { return _byteswap_ulong(val); } #else constexpr uint32_t byteswap(uint32_t val) { return __builtin_bswap32(val); } #endif namespace RecompPort { // Potential argument types for function declarations enum class FunctionArgType { u32, s32, }; // Mapping of function name to argument types using DeclaredFunctionMap = std::unordered_map>; struct InstructionPatch { std::string func_name; int32_t vram; uint32_t value; }; struct FunctionHook { std::string func_name; int32_t before_vram; std::string text; }; struct FunctionSize { std::string func_name; uint32_t size_bytes; FunctionSize(const std::string& func_name, uint32_t size_bytes) : func_name(std::move(func_name)), size_bytes(size_bytes) {} }; struct ManualFunction { std::string func_name; std::string section_name; uint32_t vram; uint32_t size; ManualFunction(const std::string& func_name, std::string section_name, uint32_t vram, uint32_t size) : func_name(std::move(func_name)), section_name(std::move(section_name)), vram(vram), size(size) {} }; struct Config { int32_t entrypoint; int32_t functions_per_output_file; bool has_entrypoint; bool uses_mips3_float_mode; bool single_file_output; bool use_absolute_symbols; bool unpaired_lo16_warnings; std::filesystem::path elf_path; std::filesystem::path symbols_file_path; std::filesystem::path func_reference_syms_file_path; std::vector data_reference_syms_file_paths; std::filesystem::path rom_file_path; std::filesystem::path output_func_path; std::filesystem::path relocatable_sections_path; std::filesystem::path output_binary_path; std::vector stubbed_funcs; std::vector ignored_funcs; DeclaredFunctionMap declared_funcs; std::vector instruction_patches; std::vector function_hooks; std::vector manual_func_sizes; std::vector manual_functions; std::string bss_section_suffix; std::string recomp_include; Config(const char* path); bool good() { return !bad; } private: bool bad; }; struct JumpTable { uint32_t vram; uint32_t addend_reg; uint32_t rom; uint32_t lw_vram; uint32_t addu_vram; uint32_t jr_vram; std::vector entries; JumpTable(uint32_t vram, uint32_t addend_reg, uint32_t rom, uint32_t lw_vram, uint32_t addu_vram, uint32_t jr_vram, std::vector&& entries) : vram(vram), addend_reg(addend_reg), rom(rom), lw_vram(lw_vram), addu_vram(addu_vram), jr_vram(jr_vram), entries(std::move(entries)) {} }; struct AbsoluteJump { uint32_t jump_target; uint32_t instruction_vram; AbsoluteJump(uint32_t jump_target, uint32_t instruction_vram) : jump_target(jump_target), instruction_vram(instruction_vram) {} }; struct Function { uint32_t vram; uint32_t rom; std::vector words; std::string name; ELFIO::Elf_Half section_index; bool ignored; bool reimplemented; bool stubbed; std::unordered_map function_hooks; Function(uint32_t vram, uint32_t rom, std::vector words, std::string name, ELFIO::Elf_Half section_index, bool ignored = false, bool reimplemented = false, bool stubbed = false) : vram(vram), rom(rom), words(std::move(words)), name(std::move(name)), section_index(section_index), ignored(ignored), reimplemented(reimplemented), stubbed(stubbed) {} Function() = default; }; enum class RelocType : uint8_t { R_MIPS_NONE = 0, R_MIPS_16, R_MIPS_32, R_MIPS_REL32, R_MIPS_26, R_MIPS_HI16, R_MIPS_LO16, R_MIPS_GPREL16, }; struct Reloc { uint32_t address; uint32_t section_offset; uint32_t symbol_index; uint32_t target_section; RelocType type; bool reference_symbol; }; constexpr uint16_t SectionSelf = (uint16_t)-1; constexpr uint16_t SectionAbsolute = (uint16_t)-2; struct Section { ELFIO::Elf_Xword rom_addr = 0; ELFIO::Elf64_Addr ram_addr = 0; ELFIO::Elf_Xword size = 0; std::vector function_addrs; std::vector relocs; std::string name; ELFIO::Elf_Half bss_section_index = (ELFIO::Elf_Half)-1; bool executable = false; bool relocatable = false; bool has_mips32_relocs = false; }; struct FunctionStats { std::vector jump_tables; std::vector absolute_jumps; }; struct ReferenceSection { uint32_t rom_addr; uint32_t ram_addr; uint32_t size; bool relocatable; }; struct ReferenceSymbol { uint16_t section_index; uint32_t section_offset; bool is_function; }; struct Context { // ROM address of each section std::vector
sections; std::vector functions; std::unordered_map> functions_by_vram; // A mapping of function name to index in the functions vector std::unordered_map functions_by_name; std::vector rom; // A list of the list of each function (by index in `functions`) in a given section std::vector> section_functions; // The section names that were specified as relocatable std::unordered_set relocatable_sections; // Functions with manual size overrides std::unordered_map manually_sized_funcs; //// Reference symbols (used for populating relocations for patches) // A list of the sections that contain the reference symbols. std::vector reference_sections; // A list of the reference symbols. std::vector reference_symbols; // Name of every reference symbol in the same order as `reference_symbols`. std::vector reference_symbol_names; // Mapping of symbol name to reference symbol index. std::unordered_map reference_symbols_by_name; int executable_section_count; Context(const ELFIO::elfio& elf_file) { sections.resize(elf_file.sections.size()); section_functions.resize(elf_file.sections.size()); functions.reserve(1024); functions_by_vram.reserve(functions.capacity()); functions_by_name.reserve(functions.capacity()); rom.reserve(8 * 1024 * 1024); executable_section_count = 0; } // Imports sections and function symbols from a provided context into this context's reference sections and reference functions. void import_reference_context(const Context& reference_context); // Reads a data symbol file and adds its contents into this context's reference data symbols. bool read_data_reference_syms(const std::filesystem::path& data_syms_file_path); static bool from_symbol_file(const std::filesystem::path& symbol_file_path, std::vector&& rom, Context& out, bool with_relocs); Context() = default; }; bool analyze_function(const Context& context, const Function& function, const std::vector& instructions, FunctionStats& stats); bool recompile_function(const Context& context, const Config& config, const Function& func, std::ofstream& output_file, std::span> static_funcs, bool write_header); } #endif