#include "librecomp/mods.hpp" #include "librecomp/helpers.hpp" #include "librecomp/addresses.hpp" void recomp_get_config_u32(uint8_t* rdram, recomp_context* ctx, size_t mod_index) { recomp::config::ConfigValueVariant val = recomp::mods::get_mod_config_value(mod_index, _arg_string<0>(rdram, ctx)); if (uint32_t* as_u32 = std::get_if(&val)) { _return(ctx, *as_u32); } else if (double* as_double = std::get_if(&val)) { _return(ctx, uint32_t(int32_t(*as_double))); } else if (bool* as_bool = std::get_if(&val)) { _return(ctx, uint32_t(*as_bool)); } else { _return(ctx, uint32_t{0}); } } void recomp_get_config_double(uint8_t* rdram, recomp_context* ctx, size_t mod_index) { recomp::config::ConfigValueVariant val = recomp::mods::get_mod_config_value(mod_index, _arg_string<0>(rdram, ctx)); if (uint32_t* as_u32 = std::get_if(&val)) { ctx->f0.d = double(*as_u32); } else if (double* as_double = std::get_if(&val)) { ctx->f0.d = *as_double; } else if (bool* as_bool = std::get_if(&val)) { ctx->f0.d = double(*as_bool); } else { ctx->f0.d = 0.0; } } template void return_string(uint8_t* rdram, recomp_context* ctx, const StringType& str) { // Allocate space in the recomp heap to hold the string, including the null terminator. size_t alloc_size = (str.size() + 1 + 15) & ~15; gpr offset = reinterpret_cast(recomp::alloc(rdram, alloc_size)) - rdram; gpr addr = offset + 0xFFFFFFFF80000000ULL; // Copy the string's data into the allocated memory and null terminate it. for (size_t i = 0; i < str.size(); i++) { MEM_B(i, addr) = str[i]; } MEM_B(str.size(), addr) = 0; // Return the allocated memory. ctx->r2 = addr; } void recomp_get_config_string(uint8_t* rdram, recomp_context* ctx, size_t mod_index) { recomp::config::ConfigValueVariant val = recomp::mods::get_mod_config_value(mod_index, _arg_string<0>(rdram, ctx)); if (std::string* as_string = std::get_if(&val)) { return_string(rdram, ctx, *as_string); } else { _return(ctx, NULLPTR); } } void recomp_free_config_string(uint8_t* rdram, recomp_context* ctx) { gpr str_rdram = (gpr)_arg<0, PTR(char)>(rdram, ctx); gpr offset = str_rdram - 0xFFFFFFFF80000000ULL; recomp::free(rdram, rdram + offset); } void recomp_get_mod_version(uint8_t* rdram, recomp_context* ctx, size_t mod_index) { uint32_t* major_out = _arg<0, uint32_t*>(rdram, ctx); uint32_t* minor_out = _arg<1, uint32_t*>(rdram, ctx); uint32_t* patch_out = _arg<2, uint32_t*>(rdram, ctx); recomp::Version version = recomp::mods::get_mod_version(mod_index); *major_out = version.major; *minor_out = version.minor; *patch_out = version.patch; } void recomp_change_save_file(uint8_t* rdram, recomp_context* ctx, size_t mod_index) { std::string name = _arg_string<0>(rdram, ctx); std::u8string name_u8 = std::u8string{reinterpret_cast(name.data()), name.size()}; std::string mod_id = recomp::mods::get_mod_id(mod_index); std::u8string mod_id_u8 = std::u8string{reinterpret_cast(mod_id.data()), mod_id.size()}; ultramodern::change_save_file(mod_id_u8, name_u8); } void recomp_get_save_file_path(uint8_t* rdram, recomp_context* ctx) { std::filesystem::path save_file_path = ultramodern::get_save_file_path(); return_string(rdram, ctx, std::filesystem::absolute(save_file_path).u8string()); } void recomp_get_mod_folder_path(uint8_t* rdram, recomp_context* ctx) { std::filesystem::path mod_folder_path = recomp::mods::get_mods_directory(); return_string(rdram, ctx, std::filesystem::absolute(mod_folder_path).u8string()); } void recomp_get_mod_file_path(uint8_t* rdram, recomp_context* ctx, size_t mod_index) { std::filesystem::path mod_file_path = recomp::mods::get_mod_path(mod_index); return_string(rdram, ctx, std::filesystem::absolute(mod_file_path).u8string()); } void recomp_is_dependency_met(uint8_t* rdram, recomp_context* ctx, size_t mod_index) { std::string dependency_id = _arg_string<0>(rdram, ctx); recomp::mods::DependencyStatus status = recomp::mods::is_dependency_met(mod_index, dependency_id); _return(ctx, static_cast(status)); } void recomp::mods::register_config_exports() { recomp::overlays::register_ext_base_export("recomp_get_config_u32", recomp_get_config_u32); recomp::overlays::register_ext_base_export("recomp_get_config_double", recomp_get_config_double); recomp::overlays::register_ext_base_export("recomp_get_config_string", recomp_get_config_string); recomp::overlays::register_base_export("recomp_free_config_string", recomp_free_config_string); recomp::overlays::register_ext_base_export("recomp_get_mod_version", recomp_get_mod_version); recomp::overlays::register_ext_base_export("recomp_change_save_file", recomp_change_save_file); recomp::overlays::register_base_export("recomp_get_save_file_path", recomp_get_save_file_path); recomp::overlays::register_base_export("recomp_get_mod_folder_path", recomp_get_mod_folder_path); recomp::overlays::register_ext_base_export("recomp_get_mod_file_path", recomp_get_mod_file_path); recomp::overlays::register_ext_base_export("recomp_is_dependency_met", recomp_is_dependency_met); }