2022-11-15 19:55:48 -05:00
|
|
|
#ifndef __RECOMP_PORT__
|
|
|
|
#define __RECOMP_PORT__
|
|
|
|
|
|
|
|
#include <span>
|
|
|
|
#include <string_view>
|
|
|
|
#include <cstdint>
|
2022-11-15 19:55:48 -05:00
|
|
|
#include <vector>
|
|
|
|
#include <unordered_map>
|
2023-01-12 23:39:49 -05:00
|
|
|
#include <span>
|
2023-02-04 00:14:03 -05:00
|
|
|
#include <unordered_set>
|
2023-03-24 17:11:17 -04:00
|
|
|
#include <filesystem>
|
|
|
|
#include "rabbitizer.hpp"
|
2023-01-12 23:39:49 -05:00
|
|
|
#include "elfio/elfio.hpp"
|
2022-11-15 19:55:48 -05:00
|
|
|
|
|
|
|
#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 {
|
|
|
|
|
2023-03-24 17:11:17 -04:00
|
|
|
// Potential argument types for function declarations
|
|
|
|
enum class FunctionArgType {
|
|
|
|
u32,
|
|
|
|
s32,
|
|
|
|
};
|
|
|
|
|
|
|
|
// Mapping of function name to argument types
|
|
|
|
using DeclaredFunctionMap = std::unordered_map<std::string, std::vector<FunctionArgType>>;
|
|
|
|
|
2023-03-24 19:22:30 -04:00
|
|
|
struct InstructionPatch {
|
|
|
|
std::string func_name;
|
|
|
|
int32_t vram;
|
|
|
|
uint32_t value;
|
|
|
|
};
|
|
|
|
|
2023-03-24 17:11:17 -04:00
|
|
|
struct Config {
|
|
|
|
int32_t entrypoint;
|
|
|
|
std::filesystem::path elf_path;
|
|
|
|
std::filesystem::path output_func_path;
|
|
|
|
std::filesystem::path relocatable_sections_path;
|
|
|
|
std::vector<std::string> stubbed_funcs;
|
|
|
|
DeclaredFunctionMap declared_funcs;
|
2023-03-24 19:22:30 -04:00
|
|
|
std::vector<InstructionPatch> instruction_patches;
|
2023-03-24 17:11:17 -04:00
|
|
|
|
|
|
|
Config(const char* path);
|
|
|
|
bool good() { return !bad; }
|
|
|
|
private:
|
|
|
|
bool bad;
|
|
|
|
};
|
|
|
|
|
2022-11-15 19:55:48 -05:00
|
|
|
struct JumpTable {
|
|
|
|
uint32_t vram;
|
|
|
|
uint32_t addend_reg;
|
|
|
|
uint32_t rom;
|
|
|
|
uint32_t lw_vram;
|
2022-11-15 19:55:48 -05:00
|
|
|
uint32_t addu_vram;
|
2022-11-15 19:55:48 -05:00
|
|
|
uint32_t jr_vram;
|
|
|
|
std::vector<uint32_t> entries;
|
|
|
|
};
|
|
|
|
|
2023-01-12 23:39:49 -05:00
|
|
|
struct AbsoluteJump {
|
|
|
|
uint32_t jump_target;
|
|
|
|
uint32_t instruction_vram;
|
|
|
|
};
|
|
|
|
|
2022-11-15 19:55:48 -05:00
|
|
|
struct Function {
|
|
|
|
uint32_t vram;
|
|
|
|
uint32_t rom;
|
|
|
|
const std::span<const uint32_t> words;
|
|
|
|
std::string name;
|
2023-01-12 23:39:49 -05:00
|
|
|
ELFIO::Elf_Half section_index;
|
2022-11-15 19:55:48 -05:00
|
|
|
bool ignored;
|
2023-01-12 23:39:49 -05:00
|
|
|
bool reimplemented;
|
2023-03-24 18:04:21 -04:00
|
|
|
bool stubbed;
|
2023-01-12 23:39:49 -05:00
|
|
|
};
|
|
|
|
|
2023-02-04 00:14:03 -05:00
|
|
|
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 target_address;
|
|
|
|
uint32_t symbol_index;
|
|
|
|
uint32_t target_section;
|
|
|
|
RelocType type;
|
|
|
|
bool needs_relocation;
|
|
|
|
};
|
|
|
|
|
2023-01-12 23:39:49 -05:00
|
|
|
struct Section {
|
2023-03-24 17:11:17 -04:00
|
|
|
ELFIO::Elf_Xword rom_addr = 0;
|
|
|
|
ELFIO::Elf64_Addr ram_addr = 0;
|
|
|
|
ELFIO::Elf_Xword size = 0;
|
2023-01-12 23:39:49 -05:00
|
|
|
std::vector<uint32_t> function_addrs;
|
2023-02-04 00:14:03 -05:00
|
|
|
std::vector<Reloc> relocs;
|
2023-01-12 23:39:49 -05:00
|
|
|
std::string name;
|
2023-03-24 17:11:17 -04:00
|
|
|
bool executable = false;
|
|
|
|
bool relocatable = false;
|
2022-11-15 19:55:48 -05:00
|
|
|
};
|
|
|
|
|
2022-11-15 19:55:48 -05:00
|
|
|
struct FunctionStats {
|
|
|
|
std::vector<JumpTable> jump_tables;
|
2023-01-12 23:39:49 -05:00
|
|
|
std::vector<AbsoluteJump> absolute_jumps;
|
2022-11-15 19:55:48 -05:00
|
|
|
};
|
|
|
|
|
2022-11-15 19:55:48 -05:00
|
|
|
struct Context {
|
2023-01-12 23:39:49 -05:00
|
|
|
// ROM address of each section
|
|
|
|
std::vector<Section> sections;
|
|
|
|
std::vector<Function> functions;
|
2022-11-15 19:55:48 -05:00
|
|
|
std::unordered_map<uint32_t, std::vector<size_t>> functions_by_vram;
|
2023-03-24 18:04:21 -04:00
|
|
|
// A mapping of function name to index in the functions vector
|
|
|
|
std::unordered_map<std::string, size_t> functions_by_name;
|
2022-11-15 19:55:48 -05:00
|
|
|
std::vector<uint8_t> rom;
|
2023-01-12 23:39:49 -05:00
|
|
|
// A list of the list of each function (by index in `functions`) in a given section
|
|
|
|
std::vector<std::vector<size_t>> section_functions;
|
2023-02-04 00:14:03 -05:00
|
|
|
// The section names that were specified as relocatable
|
|
|
|
std::unordered_set<std::string> relocatable_sections;
|
2023-01-12 23:39:49 -05:00
|
|
|
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);
|
2023-03-24 18:04:21 -04:00
|
|
|
functions_by_vram.reserve(functions.capacity());
|
|
|
|
functions_by_name.reserve(functions.capacity());
|
2023-01-12 23:39:49 -05:00
|
|
|
rom.reserve(8 * 1024 * 1024);
|
|
|
|
executable_section_count = 0;
|
|
|
|
}
|
2022-11-15 19:55:48 -05:00
|
|
|
};
|
2022-11-15 19:55:48 -05:00
|
|
|
|
2022-11-15 19:55:48 -05:00
|
|
|
bool analyze_function(const Context& context, const Function& function, const std::vector<rabbitizer::InstructionCpu>& instructions, FunctionStats& stats);
|
2023-03-24 17:11:17 -04:00
|
|
|
bool recompile_function(const Context& context, const Function& func, const std::filesystem::path& output_path, std::span<std::vector<uint32_t>> static_funcs);
|
2022-11-15 19:55:48 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|