532 lines
19 KiB
C++
532 lines
19 KiB
C++
#include "FunctionCompiler.hh"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include <phosg/Filesystem.hh>
|
|
#include <phosg/Hash.hh>
|
|
#include <phosg/Time.hh>
|
|
#include <stdexcept>
|
|
|
|
#ifdef HAVE_RESOURCE_FILE
|
|
#include <resource_file/Emulators/PPC32Emulator.hh>
|
|
#include <resource_file/Emulators/SH4Emulator.hh>
|
|
#include <resource_file/Emulators/X86Emulator.hh>
|
|
#endif
|
|
|
|
#include "CommandFormats.hh"
|
|
#include "CommonFileFormats.hh"
|
|
#include "Compression.hh"
|
|
#include "Loggers.hh"
|
|
|
|
using namespace std;
|
|
|
|
static bool is_function_compiler_available = true;
|
|
|
|
bool function_compiler_available() {
|
|
#ifndef HAVE_RESOURCE_FILE
|
|
return false;
|
|
#else
|
|
return is_function_compiler_available;
|
|
#endif
|
|
}
|
|
|
|
void set_function_compiler_available(bool is_available) {
|
|
is_function_compiler_available = is_available;
|
|
}
|
|
|
|
const char* name_for_architecture(CompiledFunctionCode::Architecture arch) {
|
|
switch (arch) {
|
|
case CompiledFunctionCode::Architecture::POWERPC:
|
|
return "PowerPC";
|
|
case CompiledFunctionCode::Architecture::X86:
|
|
return "x86";
|
|
case CompiledFunctionCode::Architecture::SH4:
|
|
return "SH-4";
|
|
default:
|
|
throw logic_error("invalid architecture");
|
|
}
|
|
}
|
|
|
|
template <bool BE>
|
|
string CompiledFunctionCode::generate_client_command_t(
|
|
const unordered_map<string, uint32_t>& label_writes,
|
|
const void* suffix_data,
|
|
size_t suffix_size,
|
|
uint32_t override_relocations_offset) const {
|
|
using FooterT = RELFileFooterT<BE>;
|
|
|
|
FooterT footer;
|
|
footer.num_relocations = this->relocation_deltas.size();
|
|
footer.unused1.clear(0);
|
|
footer.root_offset = this->entrypoint_offset_offset;
|
|
footer.unused2.clear(0);
|
|
|
|
phosg::StringWriter w;
|
|
if (!label_writes.empty()) {
|
|
string modified_code = this->code;
|
|
for (const auto& it : label_writes) {
|
|
size_t offset = this->label_offsets.at(it.first);
|
|
if (offset > modified_code.size() - 4) {
|
|
throw runtime_error("label out of range");
|
|
}
|
|
*reinterpret_cast<U32T<FooterT::IsBE>*>(modified_code.data() + offset) = it.second;
|
|
}
|
|
w.write(modified_code);
|
|
} else {
|
|
w.write(this->code);
|
|
}
|
|
if (suffix_size) {
|
|
w.write(suffix_data, suffix_size);
|
|
}
|
|
while (w.size() & 3) {
|
|
w.put_u8(0);
|
|
}
|
|
|
|
footer.relocations_offset = w.size();
|
|
|
|
// Always write at least 4 bytes even if there are no relocations
|
|
if (this->relocation_deltas.empty()) {
|
|
w.put_u32(0);
|
|
}
|
|
|
|
if (override_relocations_offset) {
|
|
footer.relocations_offset = override_relocations_offset;
|
|
} else {
|
|
for (uint16_t delta : this->relocation_deltas) {
|
|
w.put<U16T<FooterT::IsBE>>(delta);
|
|
}
|
|
if (this->relocation_deltas.size() & 1) {
|
|
w.put_u16(0);
|
|
}
|
|
}
|
|
|
|
w.put(footer);
|
|
return std::move(w.str());
|
|
}
|
|
|
|
string CompiledFunctionCode::generate_client_command(
|
|
const unordered_map<string, uint32_t>& label_writes,
|
|
const void* suffix_data,
|
|
size_t suffix_size,
|
|
uint32_t override_relocations_offset) const {
|
|
if (this->arch == Architecture::POWERPC) {
|
|
return this->generate_client_command_t<true>(
|
|
label_writes, suffix_data, suffix_size, override_relocations_offset);
|
|
} else if ((this->arch == Architecture::X86) || (this->arch == Architecture::SH4)) {
|
|
return this->generate_client_command_t<false>(
|
|
label_writes, suffix_data, suffix_size, override_relocations_offset);
|
|
} else {
|
|
throw logic_error("invalid architecture");
|
|
}
|
|
}
|
|
|
|
bool CompiledFunctionCode::is_big_endian() const {
|
|
return this->arch == Architecture::POWERPC;
|
|
}
|
|
|
|
shared_ptr<CompiledFunctionCode> compile_function_code(
|
|
CompiledFunctionCode::Architecture arch,
|
|
const string& function_directory,
|
|
const string& system_directory,
|
|
const string& name,
|
|
const string& text) {
|
|
#ifndef HAVE_RESOURCE_FILE
|
|
(void)arch;
|
|
(void)function_directory;
|
|
(void)system_directory;
|
|
(void)name;
|
|
(void)text;
|
|
throw runtime_error("function compiler is not available");
|
|
|
|
#else
|
|
auto ret = make_shared<CompiledFunctionCode>();
|
|
ret->arch = arch;
|
|
ret->short_name = name;
|
|
ret->index = 0;
|
|
ret->hide_from_patches_menu = false;
|
|
|
|
unordered_set<string> get_include_stack;
|
|
function<string(const string&)> get_include = [&](const string& name) -> string {
|
|
const char* arch_name_token;
|
|
switch (arch) {
|
|
case CompiledFunctionCode::Architecture::POWERPC:
|
|
arch_name_token = "ppc";
|
|
break;
|
|
case CompiledFunctionCode::Architecture::X86:
|
|
arch_name_token = "x86";
|
|
break;
|
|
case CompiledFunctionCode::Architecture::SH4:
|
|
arch_name_token = "sh4";
|
|
break;
|
|
default:
|
|
throw runtime_error("unknown architecture");
|
|
}
|
|
|
|
// Look in the function directory first, then the system directory
|
|
string asm_filename = phosg::string_printf("%s/%s.%s.inc.s", function_directory.c_str(), name.c_str(), arch_name_token);
|
|
if (!phosg::isfile(asm_filename)) {
|
|
asm_filename = phosg::string_printf("%s/%s.%s.inc.s", system_directory.c_str(), name.c_str(), arch_name_token);
|
|
}
|
|
if (phosg::isfile(asm_filename)) {
|
|
if (!get_include_stack.emplace(name).second) {
|
|
throw runtime_error("mutual recursion between includes: " + name);
|
|
}
|
|
ResourceDASM::EmulatorBase::AssembleResult ret;
|
|
switch (arch) {
|
|
case CompiledFunctionCode::Architecture::POWERPC:
|
|
ret = ResourceDASM::PPC32Emulator::assemble(phosg::load_file(asm_filename), get_include);
|
|
break;
|
|
case CompiledFunctionCode::Architecture::X86:
|
|
ret = ResourceDASM::X86Emulator::assemble(phosg::load_file(asm_filename), get_include);
|
|
break;
|
|
case CompiledFunctionCode::Architecture::SH4:
|
|
ret = ResourceDASM::SH4Emulator::assemble(phosg::load_file(asm_filename), get_include);
|
|
break;
|
|
default:
|
|
throw runtime_error("unknown architecture");
|
|
}
|
|
get_include_stack.erase(name);
|
|
return ret.code;
|
|
}
|
|
|
|
string bin_filename = function_directory + "/" + name + ".inc.bin";
|
|
if (phosg::isfile(bin_filename)) {
|
|
return phosg::load_file(bin_filename);
|
|
}
|
|
bin_filename = system_directory + "/" + name + ".inc.bin";
|
|
if (phosg::isfile(bin_filename)) {
|
|
return phosg::load_file(bin_filename);
|
|
}
|
|
throw runtime_error("data not found for include: " + name + " (from " + asm_filename + " or " + bin_filename + ")");
|
|
};
|
|
|
|
ResourceDASM::EmulatorBase::AssembleResult assembled;
|
|
if (arch == CompiledFunctionCode::Architecture::POWERPC) {
|
|
assembled = ResourceDASM::PPC32Emulator::assemble(text, get_include);
|
|
} else if (arch == CompiledFunctionCode::Architecture::X86) {
|
|
assembled = ResourceDASM::X86Emulator::assemble(text, get_include);
|
|
} else if (arch == CompiledFunctionCode::Architecture::SH4) {
|
|
assembled = ResourceDASM::SH4Emulator::assemble(text, get_include);
|
|
} else {
|
|
throw runtime_error("invalid architecture");
|
|
}
|
|
ret->code = std::move(assembled.code);
|
|
ret->label_offsets = std::move(assembled.label_offsets);
|
|
for (const auto& it : assembled.metadata_keys) {
|
|
if (it.first == "hide_from_patches_menu") {
|
|
ret->hide_from_patches_menu = true;
|
|
} else if (it.first == "index") {
|
|
if (it.second.size() != 1) {
|
|
throw runtime_error("invalid index value in .meta directive");
|
|
}
|
|
ret->index = it.second[0];
|
|
} else if (it.first == "name") {
|
|
ret->long_name = it.second;
|
|
} else if (it.first == "description") {
|
|
ret->description = it.second;
|
|
} else {
|
|
throw runtime_error("unknown metadata key: " + it.first);
|
|
}
|
|
}
|
|
|
|
set<uint32_t> reloc_indexes;
|
|
for (const auto& it : ret->label_offsets) {
|
|
if (phosg::starts_with(it.first, "reloc")) {
|
|
reloc_indexes.emplace(it.second / 4);
|
|
}
|
|
}
|
|
|
|
try {
|
|
ret->entrypoint_offset_offset = ret->label_offsets.at("entry_ptr");
|
|
} catch (const out_of_range&) {
|
|
throw runtime_error("code does not contain entry_ptr label");
|
|
}
|
|
|
|
uint32_t prev_index = 0;
|
|
for (const auto& it : reloc_indexes) {
|
|
uint32_t delta = it - prev_index;
|
|
if (delta > 0xFFFF) {
|
|
throw runtime_error("relocation delta too far away");
|
|
}
|
|
ret->relocation_deltas.emplace_back(delta);
|
|
prev_index = it;
|
|
}
|
|
|
|
return ret;
|
|
#endif
|
|
}
|
|
|
|
FunctionCodeIndex::FunctionCodeIndex(const string& directory) {
|
|
if (!function_compiler_available()) {
|
|
function_compiler_log.info("Function compiler is not available");
|
|
return;
|
|
}
|
|
|
|
string system_dir_path = phosg::ends_with(directory, "/") ? (directory + "System") : (directory + "/System");
|
|
|
|
uint32_t next_menu_item_id = 1;
|
|
for (const auto& subdir_name : phosg::list_directory_sorted(directory)) {
|
|
string subdir_path = phosg::ends_with(directory, "/") ? (directory + subdir_name) : (directory + "/" + subdir_name);
|
|
if (!phosg::isdir(subdir_path)) {
|
|
function_compiler_log.warning("Skipping %s (not a directory)", subdir_name.c_str());
|
|
continue;
|
|
}
|
|
|
|
for (const auto& filename : phosg::list_directory_sorted(subdir_path)) {
|
|
try {
|
|
if (!phosg::ends_with(filename, ".s")) {
|
|
continue;
|
|
}
|
|
|
|
string name = filename.substr(0, filename.size() - 2);
|
|
if (phosg::ends_with(name, ".inc")) {
|
|
continue;
|
|
}
|
|
|
|
bool is_patch = phosg::ends_with(name, ".patch");
|
|
if (is_patch) {
|
|
name.resize(name.size() - 6);
|
|
}
|
|
|
|
// Figure out the version or specific_version
|
|
CompiledFunctionCode::Architecture arch = CompiledFunctionCode::Architecture::UNKNOWN;
|
|
uint32_t specific_version = 0;
|
|
string short_name = name;
|
|
if (phosg::ends_with(name, ".ppc")) {
|
|
arch = CompiledFunctionCode::Architecture::POWERPC;
|
|
name.resize(name.size() - 4);
|
|
short_name = name;
|
|
} else if (phosg::ends_with(name, ".x86")) {
|
|
arch = CompiledFunctionCode::Architecture::X86;
|
|
name.resize(name.size() - 4);
|
|
short_name = name;
|
|
} else if (phosg::ends_with(name, ".sh4")) {
|
|
arch = CompiledFunctionCode::Architecture::SH4;
|
|
name.resize(name.size() - 4);
|
|
short_name = name;
|
|
} else if (is_patch && (name.size() >= 5) && (name[name.size() - 5] == '.')) {
|
|
specific_version = (name[name.size() - 4] << 24) | (name[name.size() - 3] << 16) | (name[name.size() - 2] << 8) | name[name.size() - 1];
|
|
if (specific_version_is_dc(specific_version)) {
|
|
arch = CompiledFunctionCode::Architecture::SH4;
|
|
} else if (specific_version_is_gc(specific_version)) {
|
|
arch = CompiledFunctionCode::Architecture::POWERPC;
|
|
} else if (specific_version_is_xb(specific_version) || specific_version_is_bb(specific_version)) {
|
|
arch = CompiledFunctionCode::Architecture::X86;
|
|
} else {
|
|
throw runtime_error("unable to determine architecture from specific_version");
|
|
}
|
|
short_name = name.substr(0, name.size() - 5);
|
|
}
|
|
|
|
if (arch == CompiledFunctionCode::Architecture::UNKNOWN) {
|
|
throw runtime_error("unable to determine architecture");
|
|
}
|
|
|
|
string path = subdir_path + "/" + filename;
|
|
string text = phosg::load_file(path);
|
|
auto code = compile_function_code(arch, subdir_path, system_dir_path, name, text);
|
|
if (code->index != 0) {
|
|
if (!this->index_to_function.emplace(code->index, code).second) {
|
|
throw runtime_error(phosg::string_printf(
|
|
"duplicate function index: %08" PRIX32, code->index));
|
|
}
|
|
}
|
|
code->specific_version = specific_version;
|
|
code->source_path = path;
|
|
code->short_name = short_name;
|
|
this->name_to_function.emplace(name, code);
|
|
if (is_patch) {
|
|
code->menu_item_id = next_menu_item_id++;
|
|
this->menu_item_id_and_specific_version_to_patch_function.emplace(
|
|
static_cast<uint64_t>(code->menu_item_id) << 32 | specific_version, code);
|
|
this->name_and_specific_version_to_patch_function.emplace(
|
|
phosg::string_printf("%s-%08" PRIX32, short_name.c_str(), specific_version), code);
|
|
}
|
|
|
|
string index_prefix = code->index ? phosg::string_printf("%02X => ", code->index) : "";
|
|
string patch_prefix = is_patch ? phosg::string_printf("[%08" PRIX32 "/%08" PRIX32 "] ", code->menu_item_id, code->specific_version) : "";
|
|
function_compiler_log.info("Compiled function %s%s%s (%s)",
|
|
index_prefix.c_str(), patch_prefix.c_str(), name.c_str(), name_for_architecture(code->arch));
|
|
|
|
} catch (const exception& e) {
|
|
function_compiler_log.warning("Failed to compile function %s: %s", filename.c_str(), e.what());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
shared_ptr<const Menu> FunctionCodeIndex::patch_menu(uint32_t specific_version) const {
|
|
auto suffix = phosg::string_printf("-%08" PRIX32, specific_version);
|
|
|
|
auto ret = make_shared<Menu>(MenuID::PATCHES, "Patches");
|
|
ret->items.emplace_back(PatchesMenuItemID::GO_BACK, "Go back", "Return to the\nmain menu", 0);
|
|
for (const auto& it : this->name_and_specific_version_to_patch_function) {
|
|
const auto& fn = it.second;
|
|
if (fn->hide_from_patches_menu || !phosg::ends_with(it.first, suffix)) {
|
|
continue;
|
|
}
|
|
ret->items.emplace_back(
|
|
fn->menu_item_id,
|
|
fn->long_name.empty() ? fn->short_name : fn->long_name,
|
|
fn->description,
|
|
MenuItem::Flag::REQUIRES_SEND_FUNCTION_CALL);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
shared_ptr<const Menu> FunctionCodeIndex::patch_switches_menu(
|
|
uint32_t specific_version, const std::unordered_set<std::string>& auto_patches_enabled) const {
|
|
auto suffix = phosg::string_printf("-%08" PRIX32, specific_version);
|
|
|
|
auto ret = make_shared<Menu>(MenuID::PATCH_SWITCHES, "Patch switches");
|
|
ret->items.emplace_back(PatchesMenuItemID::GO_BACK, "Go back", "Return to the\nmain menu", 0);
|
|
for (const auto& it : this->name_and_specific_version_to_patch_function) {
|
|
const auto& fn = it.second;
|
|
if (fn->hide_from_patches_menu || !phosg::ends_with(it.first, suffix)) {
|
|
continue;
|
|
}
|
|
string name;
|
|
name.push_back(auto_patches_enabled.count(fn->short_name) ? '*' : '-');
|
|
name += fn->long_name.empty() ? fn->short_name : fn->long_name;
|
|
ret->items.emplace_back(fn->menu_item_id, name, fn->description, MenuItem::Flag::REQUIRES_SEND_FUNCTION_CALL);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
bool FunctionCodeIndex::patch_menu_empty(uint32_t specific_version) const {
|
|
uint32_t mask = specific_version_is_indeterminate(specific_version) ? 0xFF000000 : 0xFFFFFFFF;
|
|
for (const auto& it : this->menu_item_id_and_specific_version_to_patch_function) {
|
|
if ((it.first & mask) == (specific_version & mask)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
std::shared_ptr<const CompiledFunctionCode> FunctionCodeIndex::get_patch(
|
|
const std::string& name, uint32_t specific_version) const {
|
|
return this->name_and_specific_version_to_patch_function.at(
|
|
phosg::string_printf("%s-%08" PRIX32, name.c_str(), specific_version));
|
|
}
|
|
|
|
DOLFileIndex::DOLFileIndex(const string& directory) {
|
|
if (!function_compiler_available()) {
|
|
function_compiler_log.info("Function compiler is not available");
|
|
return;
|
|
}
|
|
if (!phosg::isdir(directory)) {
|
|
function_compiler_log.info("DOL file directory is missing");
|
|
return;
|
|
}
|
|
|
|
auto menu = make_shared<Menu>(MenuID::PROGRAMS, "Programs");
|
|
this->menu = menu;
|
|
menu->items.emplace_back(ProgramsMenuItemID::GO_BACK, "Go back", "Return to the\nmain menu", 0);
|
|
|
|
uint32_t next_menu_item_id = 0;
|
|
for (const auto& filename : phosg::list_directory_sorted(directory)) {
|
|
bool is_dol = phosg::ends_with(filename, ".dol");
|
|
bool is_compressed_dol = phosg::ends_with(filename, ".dol.prs");
|
|
if (!is_dol && !is_compressed_dol) {
|
|
continue;
|
|
}
|
|
string name = filename.substr(0, filename.size() - (is_compressed_dol ? 8 : 4));
|
|
|
|
try {
|
|
auto dol = make_shared<File>();
|
|
dol->menu_item_id = next_menu_item_id++;
|
|
dol->name = name;
|
|
|
|
string path = directory + "/" + filename;
|
|
string file_data = phosg::load_file(path);
|
|
|
|
string description;
|
|
if (is_compressed_dol) {
|
|
size_t decompressed_size = prs_decompress_size(file_data);
|
|
|
|
phosg::StringWriter w;
|
|
w.put_u32b(file_data.size());
|
|
w.put_u32b(decompressed_size);
|
|
w.write(file_data);
|
|
while (w.size() & 3) {
|
|
w.put_u8(0);
|
|
}
|
|
dol->data = std::move(w.str());
|
|
|
|
string compressed_size_str = phosg::format_size(file_data.size());
|
|
string decompressed_size_str = phosg::format_size(decompressed_size);
|
|
function_compiler_log.info("Loaded compressed DOL file %s (%s -> %s)",
|
|
dol->name.c_str(), compressed_size_str.c_str(), decompressed_size_str.c_str());
|
|
description = phosg::string_printf("$C6%s$C7\n%s\n%s (orig)",
|
|
dol->name.c_str(), compressed_size_str.c_str(), decompressed_size_str.c_str());
|
|
|
|
} else {
|
|
phosg::StringWriter w;
|
|
w.put_u32b(0);
|
|
w.put_u32b(file_data.size());
|
|
w.write(file_data);
|
|
while (w.size() & 3) {
|
|
w.put_u8(0);
|
|
}
|
|
dol->data = std::move(w.str());
|
|
|
|
string size_str = phosg::format_size(dol->data.size());
|
|
function_compiler_log.info("Loaded DOL file %s (%s)", filename.c_str(), size_str.c_str());
|
|
description = phosg::string_printf("$C6%s$C7\n%s", dol->name.c_str(), size_str.c_str());
|
|
}
|
|
|
|
this->name_to_file.emplace(dol->name, dol);
|
|
this->item_id_to_file.emplace_back(dol);
|
|
|
|
menu->items.emplace_back(dol->menu_item_id, dol->name, description, MenuItem::Flag::REQUIRES_SEND_FUNCTION_CALL);
|
|
|
|
} catch (const exception& e) {
|
|
function_compiler_log.warning("Failed to load DOL file %s: %s", filename.c_str(), e.what());
|
|
}
|
|
}
|
|
}
|
|
|
|
uint32_t specific_version_for_gc_header_checksum(uint32_t header_checksum) {
|
|
static unordered_map<uint32_t, uint32_t> checksum_to_specific_version;
|
|
if (checksum_to_specific_version.empty()) {
|
|
struct {
|
|
char system_code = 'G';
|
|
char game_code1 = 'P';
|
|
char game_code2;
|
|
char region_code;
|
|
char developer_code1 = '8';
|
|
char developer_code2 = 'P';
|
|
uint8_t disc_number = 0;
|
|
uint8_t version_code;
|
|
} __packed__ data;
|
|
for (const char* game_code2 = "OS"; *game_code2; game_code2++) {
|
|
data.game_code2 = *game_code2;
|
|
for (const char* region_code = "JEP"; *region_code; region_code++) {
|
|
data.region_code = *region_code;
|
|
for (uint8_t version_code = 0; version_code < 8; version_code++) {
|
|
data.version_code = version_code;
|
|
uint32_t checksum = phosg::crc32(&data, sizeof(data));
|
|
uint32_t specific_version = 0x33000030 | (*game_code2 << 16) | (*region_code << 8) | version_code;
|
|
if (!checksum_to_specific_version.emplace(checksum, specific_version).second) {
|
|
throw logic_error("multiple specific_versions have same header checksum");
|
|
}
|
|
}
|
|
}
|
|
{
|
|
// Generate entries for Trial Editions
|
|
data.region_code = 'J';
|
|
data.system_code = 'D';
|
|
data.version_code = 0;
|
|
uint32_t checksum = phosg::crc32(&data, sizeof(data));
|
|
uint32_t specific_version = 0x33004A54 | (*game_code2 << 16);
|
|
if (!checksum_to_specific_version.emplace(checksum, specific_version).second) {
|
|
throw logic_error("multiple specific_versions have same header checksum");
|
|
}
|
|
data.system_code = 'G';
|
|
}
|
|
}
|
|
}
|
|
return checksum_to_specific_version.at(header_checksum);
|
|
}
|