#include #include #include #include #include #include #include #include "options-internal.h" #include // Fix Initial Option Button Rendering // The calling function doesn't exist in MCPE v0.6.1, so its name is unknown. static OptionButton *OptionsPane_unknown_toggle_creating_function_OptionButton_injection(OptionButton *option_button, Options_Option *option) { // Call Original Method OptionButton *ret = (*OptionButton_constructor)(option_button, option); // Setup Image (*OptionButton_updateImage)(option_button, stored_options); // Return return ret; } // Actually Save options.txt // Hook Last Options::addOptionToSaveOutput Call static void Options_save_Options_addOptionToSaveOutput_injection(Options *options, std::vector *data, std::string option, int32_t value) { // Call Original Method (*Options_addOptionToSaveOutput)(options, data, option, value); // Save Fancy Graphics (*Options_addOptionToSaveOutput)(options, data, "gfx_fancygraphics", options->fancy_graphics); // Save 3D Anaglyph (*Options_addOptionToSaveOutput)(options, data, "gfx_anaglyph", options->anaglyph_3d); // Save File OptionsFile *options_file = &options->options_file; (*OptionsFile_save)(options_file, data); } // MCPI's OptionsFile::getOptionStrings is broken, this is the version in v0.7.0 static std::vector OptionsFile_getOptionStrings_injection(OptionsFile *options_file) { // Get options.txt Path std::string path = options_file->options_txt_path; // Parse std::vector ret; FILE *stream = fopen(path.c_str(), "r"); char line[128]; if (stream != NULL) { while (fgets(line, 0x80, stream) != NULL) { size_t sVar1 = strlen(line); if (2 < sVar1) { std::stringstream string_stream(line); while (true) { std::string data; std::getline(string_stream, data, ':'); int iVar2 = data.find_last_not_of(" \n\r\t"); data.erase(iVar2 + 1); if (data.length() == 0) { break; } ret.push_back(data); } } } fclose(stream); } return ret; } // Get New options.txt Path #ifndef MCPI_SERVER_MODE static char *get_new_options_txt_path() { static char *path = NULL; // Path if (path == NULL) { safe_asprintf(&path, "%s/options.txt", home_get()); } // Return return path; } // Free __attribute__((destructor)) static void _free_new_options_txt_path() { free(get_new_options_txt_path()); } #else static char *get_new_options_txt_path() { // Block options.txt On Servers return (char *) "/dev/null"; } #endif // Modify Option Toggles static void OptionsPane_unknown_toggle_creating_function_injection(OptionsPane *options_pane, uint32_t group_id, std::string *name_ptr, Options_Option *option) { // Modify std::string name = *name_ptr; std::string new_name = name; if (name == "Fancy Graphics") { option = &Options_Option_GRAPHICS; } else if (name == "Soft shadows") { option = &Options_Option_AMBIENT_OCCLUSION; } else if (name == "Fancy Skies" || name == "Animated water") { // These have no corresponding option, so disable the toggle. return; } else if (name == "Third person camera") { // This isn't saved/loaded, so disable the toggle. return; } else if (name == "Lefty" || name == "Use touch screen" || name == "Split touch controls") { // These toggles require touch support, so disable them. return; } else if (name == "Vibrate on destroy") { // This toggle requires vibration support, so disable it. return; } else if (name == "Invert X-axis") { // Fix Incorrect Name new_name = "Invert Y-axis"; } // Call Original Method (*OptionsPane_unknown_toggle_creating_function)(options_pane, group_id, &new_name, option); // Add 3D Anaglyph if (option == &Options_Option_GRAPHICS) { std::string cpp_string = "3D Anaglyph"; (*OptionsPane_unknown_toggle_creating_function)(options_pane, group_id, &cpp_string, &Options_Option_ANAGLYPH); } } // Add Missing Options To Options::getBooleanValue static bool Options_getBooleanValue_injection(Options *options, Options_Option *option) { // Check if (option == &Options_Option_GRAPHICS) { return options->fancy_graphics; } else { // Call Original Method return (*Options_getBooleanValue)(options, option); } } // Init C++ void _init_options_cpp() { // NOP unsigned char nop_patch[4] = {0x00, 0xf0, 0x20, 0xe3}; // "nop" // Fix Options Screen if (feature_has("Fix Options Screen", server_disabled)) { // Fix Initial Option Button Rendering overwrite_call((void *) 0x24510, (void *) OptionsPane_unknown_toggle_creating_function_OptionButton_injection); // "Gui Scale" slider is broken, so disable it. patch((void *) 0x35a10, nop_patch); // "Vibrate on destroy" is disabled, so "Feedback" is empty, so disable it. patch((void *) 0x35960, nop_patch); // Disconnect "This works?" Slider From Difficulty unsigned char this_works_slider_patch[4] = {0x00, 0x30, 0xa0, 0xe3}; // "mov r3, #0x0" patch((void *) 0x3577c, this_works_slider_patch); // Modify Option Toggles overwrite_calls((void *) OptionsPane_unknown_toggle_creating_function, (void *) OptionsPane_unknown_toggle_creating_function_injection); // Add Missing Options To Options::getBooleanValue overwrite_calls((void *) Options_getBooleanValue, (void *) Options_getBooleanValue_injection); } // Actually Save options.txt overwrite_call((void *) 0x197fc, (void *) Options_save_Options_addOptionToSaveOutput_injection); // Fix options.txt Path patch_address((void *) Strings_options_txt_path_pointer, (void *) get_new_options_txt_path()); // When Loading, options.txt Should Be Opened In Read Mode patch_address((void *) Strings_options_txt_fopen_mode_when_loading_pointer, (void *) "r"); // Fix OptionsFile::getOptionStrings overwrite_calls((void *) OptionsFile_getOptionStrings, (void *) OptionsFile_getOptionStrings_injection); // Sensitivity Loading/Saving Is Broken, Disable It patch((void *) 0x1931c, nop_patch); patch((void *) 0x1973c, nop_patch); // Unsplit Touch Controls Breaks Things, Never Load/Save It unsigned char cmp_r0_r0_patch[4] = {0x00, 0x00, 0x50, 0xe1}; // "cmp r0, r0" patch((void *) 0x19378, cmp_r0_r0_patch); patch((void *) 0x197cc, nop_patch); // Custom Username Is Loaded Manually, Disable Loading From options.txt patch((void *) 0x192ac, nop_patch); // Replace "feedback_vibration" Loading/Saving With "gfx_ao" { // Replace String static const char *new_feedback_vibration_options_txt_name = "gfx_ao"; patch_address((void *) Strings_feedback_vibration_options_txt_name_1_pointer, (void *) &new_feedback_vibration_options_txt_name); patch_address((void *) Strings_feedback_vibration_options_txt_name_2_pointer, (void *) &new_feedback_vibration_options_txt_name); // Loading unsigned char offset = (unsigned char) offsetof(Options, ambient_occlusion); unsigned char gfx_ao_loading_patch[4] = {offset, 0x10, 0x84, 0xe2}; // "add r1, r4, #OFFSET" patch((void *) 0x193b8, gfx_ao_loading_patch); // Saving unsigned char gfx_ao_saving_patch[4] = {offset, 0x30, 0xd4, 0xe5}; // "ldrb r3, [r4, #OFFSET]" patch((void *) 0x197f8, gfx_ao_saving_patch); } // Replace "gfx_lowquality" Loading With "gfx_anaglyph" { // Replace String static const char *new_gfx_lowquality_options_txt_name = "gfx_anaglyph"; patch_address((void *) Strings_gfx_lowquality_options_txt_name_pointer, (void *) &new_gfx_lowquality_options_txt_name); // Loading unsigned char offset = (unsigned char) offsetof(Options, anaglyph_3d); unsigned char gfx_anaglyph_loading_patch[4] = {offset, 0x10, 0x84, 0xe2}; // "add r1, r4, #OFFSET" patch((void *) 0x19400, gfx_anaglyph_loading_patch); // Disable Loading Side Effects patch((void *) 0x19414, nop_patch); patch((void *) 0x1941c, nop_patch); } }