eden-miror/src/common/src/xml.cpp
2013-08-29 23:35:09 -04:00

487 lines
21 KiB
C++

/**
* Copyright (C) 2005-2012 Gekko Emulator
*
* @file xml.h
* @author ShizZy <shizzy247@gmail.com>
* @date 2012-02-12
* @brief Used for parsing XML configurations
*
* @section LICENSE
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of
* the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details at
* http://www.gnu.org/copyleft/gpl.html
*
* Official project repository can be found at:
* http://code.google.com/p/gekko-gc-emu/
*/
#include <iostream>
#include <fstream>
#include <rapidxml.hpp>
#include "common.h"
#include "misc_utils.h"
#include "config.h"
#include "log.h"
/// Gets a RapidXML boolean element value
static bool GetXMLElementAsBool(rapidxml::xml_node<> *node, const char* element_name) {
rapidxml::xml_node<> *sub_node = node->first_node(element_name);
if (sub_node) {
return (E_OK == _stricmp(sub_node->value(), "true")) ? true : false;
}
return false;
}
/// Gets a RapidXML string element value
static char* GetXMLElementAsString(rapidxml::xml_node<> *node, const char* element_name,
char* element_value) {
rapidxml::xml_node<> *sub_node = node->first_node(element_name);
if (sub_node) {
strcpy(element_value, sub_node->value());
return element_value;
}
return NULL;
}
/// Gets a RapidXML integer element value
static int GetXMLElementAsInt(rapidxml::xml_node<> *node, const char* element_name) {
rapidxml::xml_node<> *sub_node = node->first_node(element_name);
if (sub_node) {
return atoi(sub_node->value());
}
return 0;
}
namespace common {
/**
* @brief Parse the "General" XML group
* @param node RapidXML node for the "General" XML group
* @param config Config class object to parse data into
*/
void ParseGeneralNode(rapidxml::xml_node<> *node, Config& config) {
// Don't parse the node if it doesn't exist!
if (!node) {
return;
}
char temp_str[MAX_PATH];
config.set_enable_multicore(GetXMLElementAsBool(node, "EnableMultiCore"));
config.set_enable_idle_skipping(GetXMLElementAsBool(node, "EnableIdleSkipping"));
config.set_enable_hle(GetXMLElementAsBool(node, "EnableHLE"));
config.set_enable_auto_boot(GetXMLElementAsBool(node, "EnableAutoBoot"));
config.set_enable_cheats(GetXMLElementAsBool(node, "EnableCheats"));
config.set_default_boot_file(GetXMLElementAsString(node, "DefaultBootFile", temp_str), MAX_PATH);
// Parse all search paths in the DVDImagePaths node
rapidxml::xml_node<> *sub_node = node->first_node("DVDImagePaths");
if (sub_node) {
int i = 0;
for (rapidxml::xml_node<> *elem = sub_node->first_node("Path"); elem;
elem = elem->next_sibling()) {
config.set_dvd_image_path(i, elem->value(), MAX_PATH);
LOG_NOTICE(TCONFIG, "Adding %s to DVD image search paths...\n",
config.dvd_image_path(i));
i++;
// Stop if we have parsed the maximum paths
if (MAX_SEARCH_PATHS < i) {
LOG_WARNING(TCONFIG, "Maximum number of DVDImagePath search paths is %d, not parsing"
" any more!", MAX_SEARCH_PATHS);
break;
}
}
}
}
/**
* @brief Parse the "Debug" XML group
* @param node RapidXML node for the "Debug" XML group
* @param config Config class object to parse data into
*/
void ParseDebugNode(rapidxml::xml_node<> *node, Config& config) {
// Don't parse the node if it doesn't exist!
if (!node) {
return;
}
config.set_enable_show_fps(GetXMLElementAsBool(node, "EnableShowFPS"));
config.set_enable_dump_opcode0(GetXMLElementAsBool(node, "EnableDumpOpcode0"));
config.set_enable_pause_on_unknown_opcode(GetXMLElementAsBool(node,
"EnablePauseOnUnknownOpcode"));
config.set_enable_dump_gcm_reads(GetXMLElementAsBool(node, "EnableDumpGCMReads"));
}
/**
* @brief Parse the "Patches" and "Cheats" XML group
* @param node RapidXML node for the "Patches" or "Cheats" XML group
* @param config Config class object to parse data into
*/
void ParsePatchesNode(rapidxml::xml_node<> *node, Config& config, const char* node_name) {
int i = 0;
char node_name_str[8];
// Get lowercase section name
strcpy(node_name_str, node_name);
// TODO: not available on Unix
common::LowerStr(node_name_str);
// Parse all search patches in the Patches node
rapidxml::xml_node<> *sub_node = node->first_node(node_name);
if (sub_node) {
for (rapidxml::xml_node<> *elem = sub_node->first_node("Patch"); elem;
elem = elem->next_sibling()) {
// Get enable attribute (note: defaults to true)
rapidxml::xml_attribute<> *attr = elem->first_attribute("enable");
if (attr) {
if (E_OK == _stricmp(attr->value(), "false")) {
continue; // Patch is disabled, skip it
}
}
// Get address attribute
attr = elem->first_attribute("address");
if (!attr) {
LOG_ERROR(TCONFIG, "Patch without 'address' attribute illegal!");
continue;
} else {
u32 data = 0;
u32 address = 0;
{
// Convert address hexstring to unsigned int
std::stringstream ss;
ss << std::hex << attr->value();
ss >> address;
}
attr = elem->first_attribute("instr");
// Get "data" attribute if no "instr" attribute
if (!attr) {
attr = elem->first_attribute("data");
// Neither found - error
if (!attr) {
LOG_ERROR(TCONFIG, "Patch without 'instr' or 'data' attributes "
"illegal!");
continue;
} else {
// Found data, convert hexstring to unsigned int
std::stringstream ss;
ss << std::hex << attr->value();
ss >> data;
}
} else {
// Found instr
char instr_str[4];
// Convert to lowercase
strcpy(instr_str, attr->value());
// TODO: not available on Unix
common::LowerStr(instr_str);
// Convert instruction to equivalent PPC bytecode
// TODO(ShizZy): Pull this out to the PowerPC modules at some point
if (E_OK == _stricmp(instr_str, "blr")) {
data = 0x4E800020; // PowerPC BLR instruction bytecode
} else if (E_OK == _stricmp(instr_str, "nop")) {
data = 0x60000000; // PowerPC NOP instruction bytecode
} else {
LOG_ERROR(TCONFIG, "Patch with invalid 'instr' attribute illegal!");
continue;
}
}
Config::Patch patch = { address, data };
if (E_OK == _stricmp(node_name_str, "patches")) {
LOG_NOTICE(TCONFIG, "Adding patch addr=0x%08x data=0x%08x to patches...\n",
address, data, node_name_str);
config.set_patches(i, patch);
} else if (E_OK == _stricmp(node_name_str, "cheats")) {
LOG_NOTICE(TCONFIG, "Adding cheat addr=0x%08x data=0x%08x to cheats...\n",
address, data, node_name_str);
config.set_cheats(i, patch);
} else {
LOG_ERROR(TCONFIG, "Unexpected patch type %s, ignoring...", node_name_str);
}
// Stop if we have parsed the maximum patches
if (MAX_PATCHES_PER_GAME < ++i) {
LOG_WARNING(TCONFIG, "Maximum number of patches search paths is %d, not parsing"
" any more!", MAX_PATCHES_PER_GAME);
break;
}
}
}
}
}
/**
* @brief Parse the "Boot" XML group
* @param node RapidXML node for the "Boot" XML group
* @param config Config class object to parse data into
*/
void ParseBootNode(rapidxml::xml_node<> *node, Config& config) {
// Don't parse the node if it doesn't exist!
if (!node) {
return;
}
config.set_enable_ipl(GetXMLElementAsBool(node, "EnableIPL"));
ParsePatchesNode(node, config, "Patches");
ParsePatchesNode(node, config, "Cheats");
}
/**
* @brief Parse the "Video" XML group
* @param node RapidXML node for the "Video" XML group
* @param config Config class object to parse data into
*/
void ParsePowerPCNode(rapidxml::xml_node<> *node, Config& config) {
// Don't parse the node if it doesn't exist!
if (!node) {
return;
}
rapidxml::xml_attribute<> *attr = node->first_attribute("core");
// Attribute not found - error
if (!attr) {
LOG_ERROR(TCONFIG, "PowerPC without 'core' attribute illegal!");
} else {
char core_str[12] = "null";
// Convert to lowercase
strcpy(core_str, attr->value());
// TODO: not available on Unix
common::LowerStr(core_str);
// Use interpreter core
if (E_OK == _stricmp(core_str, "interpreter")) {
config.set_powerpc_core(Config::CPU_INTERPRETER); // Interpreter selected
// Use dynarec core
} else if (E_OK == _stricmp(core_str, "dynarec")) {
config.set_powerpc_core(Config::CPU_DYNAREC); // Dynarec selected
// Unsupported type
} else {
LOG_ERROR(TCONFIG, "Invalid PowerPC type %s for attribute 'core' selected!",
core_str);
}
// Set frequency
attr = node->first_attribute("freq");
if (attr) {
config.set_powerpc_frequency(atoi(attr->value()));
}
LOG_NOTICE(TCONFIG, "Configured core=%s freq=%d", core_str, config.powerpc_frequency());
}
}
/**
* @brief Parse the "Video" XML group
* @param node RapidXML node for the "Video" XML group
* @param config Config class object to parse data into
*/
void ParseVideoNode(rapidxml::xml_node<> *node, Config& config) {
char res_str[512];
Config::ResolutionType res;
// Don't parse the node if it doesn't exist!
if (!node) {
return;
}
config.set_enable_fullscreen(GetXMLElementAsBool(node, "EnableFullscreen"));
// Set resolutions
GetXMLElementAsString(node, "WindowResolution", res_str);
sscanf(res_str, "%d_%d", &res.width, &res.height);
config.set_window_resolution(res);
GetXMLElementAsString(node, "FullscreenResolution", res_str);
sscanf(res_str, "%d_%d", &res.width, &res.height);
config.set_fullscreen_resolution(res);
// Parse all search renderer nodes
for (rapidxml::xml_node<> *elem = node->first_node("Renderer"); 1; ) {
Config::RendererConfig renderer_config;
rapidxml::xml_attribute<> *attr = elem->first_attribute("name");
Config::RendererType type = Config::StringToRenderType(attr->value());
renderer_config.enable_wireframe = GetXMLElementAsBool(elem, "EnableWireframe");
renderer_config.enable_shaders = GetXMLElementAsBool(elem, "EnableShaders");
renderer_config.enable_textures = GetXMLElementAsBool(elem, "EnableTextures");
renderer_config.enable_texture_dumping = GetXMLElementAsBool(elem, "EnableTextureDumping");
renderer_config.anti_aliasing_mode = GetXMLElementAsInt(elem, "AntiAliasingMode");
renderer_config.anistropic_filtering_mode = GetXMLElementAsInt(elem, "AnistropicFilteringMode");
config.set_renderer_config(type, renderer_config);
LOG_NOTICE(TCONFIG, "Renderer %s configured", attr->value());
break;
}
}
/**
* @brief Parse the "Devices" XML group
* @param node RapidXML node for the "Devices" XML group
* @param config Config class object to parse data into
*/
void ParseDevicesNode(rapidxml::xml_node<> *node, Config& config) {
// Don't parse the node if it doesn't exist!c
if (!node) {
return;
}
// Parse GameCube section
rapidxml::xml_node<> *gamecube_node = node->first_node("GameCube");
if (!gamecube_node) {
return;
}
// Parse all MemSlot nodes
for (rapidxml::xml_node<> *elem = gamecube_node->first_node("MemSlot"); elem;
elem = elem->next_sibling("MemSlot")) {
Config::MemSlot slot_config;
// Select MemSlot a or b
rapidxml::xml_attribute<> *attr = elem->first_attribute("slot");
int slot = (E_OK == _stricmp(attr->value(), "a")) ? 0 : 1;
// Enable
attr = elem->first_attribute("enable");
slot_config.enable = (E_OK == _stricmp(attr->value(), "true")) ? true : false;
// Select device
attr = elem->first_attribute("device");
slot_config.device = 0; // Only support memcards right now
LOG_NOTICE(TCONFIG, "Configured MemSlot[%d]=%s enabled=%s", slot, attr->value(),
slot_config.enable ? "true" : "false");
config.set_mem_slots(slot, slot_config);
}
// Parse all ControlerPort nodes
for (rapidxml::xml_node<> *elem = gamecube_node->first_node("ControllerPort"); elem;
elem = elem->next_sibling("ControllerPort")) {
Config::ControllerPort port_config;
// Select MemSlot a or b
rapidxml::xml_attribute<> *attr = elem->first_attribute("port");
int port = atoi(attr->value());
// Enable
attr = elem->first_attribute("enable");
port_config.enable = (E_OK == _stricmp(attr->value(), "true")) ? true : false;
// Select device
attr = elem->first_attribute("device");
port_config.device = 0; // Only support memcards right now
LOG_NOTICE(TCONFIG, "Configured ControllerPort[%d]=%s enabled=%s", port, attr->value(),
port_config.enable ? "true" : "false");
// Parse keyboard configuration - TODO: Move to EmuWindow (?)
rapidxml::xml_node<> *keyboard_node = elem->first_node("KeyboardController");
if (keyboard_node) {
attr = keyboard_node->first_attribute("enable");
port_config.keys.enable = (E_OK == _stricmp(attr->value(), "true")) ? true : false;
port_config.keys.key_code[Config::BUTTON_A] = GetXMLElementAsInt(keyboard_node, "AKey");
port_config.keys.key_code[Config::BUTTON_B] = GetXMLElementAsInt(keyboard_node, "BKey");
port_config.keys.key_code[Config::BUTTON_X] = GetXMLElementAsInt(keyboard_node, "XKey");
port_config.keys.key_code[Config::BUTTON_Y] = GetXMLElementAsInt(keyboard_node, "YKey");
port_config.keys.key_code[Config::TRIGGER_L] = GetXMLElementAsInt(keyboard_node, "LKey");
port_config.keys.key_code[Config::TRIGGER_R] = GetXMLElementAsInt(keyboard_node, "RKey");
port_config.keys.key_code[Config::BUTTON_Z] = GetXMLElementAsInt(keyboard_node, "ZKey");
port_config.keys.key_code[Config::BUTTON_START] = GetXMLElementAsInt(keyboard_node, "StartKey");
port_config.keys.key_code[Config::ANALOG_UP] = GetXMLElementAsInt(keyboard_node, "AnalogUpKey");
port_config.keys.key_code[Config::ANALOG_DOWN] = GetXMLElementAsInt(keyboard_node, "AnalogDownKey");
port_config.keys.key_code[Config::ANALOG_LEFT] = GetXMLElementAsInt(keyboard_node, "AnalogLeftKey");
port_config.keys.key_code[Config::ANALOG_RIGHT] = GetXMLElementAsInt(keyboard_node, "AnalogRightKey");
port_config.keys.key_code[Config::C_UP] = GetXMLElementAsInt(keyboard_node, "CUpKey");
port_config.keys.key_code[Config::C_DOWN] = GetXMLElementAsInt(keyboard_node, "CDownKey");
port_config.keys.key_code[Config::C_LEFT] = GetXMLElementAsInt(keyboard_node, "CLeftKey");
port_config.keys.key_code[Config::C_RIGHT] = GetXMLElementAsInt(keyboard_node, "CRightKey");
port_config.keys.key_code[Config::DPAD_UP] = GetXMLElementAsInt(keyboard_node, "DPadUpKey");
port_config.keys.key_code[Config::DPAD_DOWN] = GetXMLElementAsInt(keyboard_node, "DPadDownKey");
port_config.keys.key_code[Config::DPAD_LEFT] = GetXMLElementAsInt(keyboard_node, "DPadLeftKey");
port_config.keys.key_code[Config::DPAD_RIGHT] = GetXMLElementAsInt(keyboard_node, "DPadRightKey");
}
// Parse joypad configuration
rapidxml::xml_node<> *joypad_node = elem->first_node("JoypadController");
if (joypad_node) {
attr = joypad_node->first_attribute("enable");
port_config.pads.enable = (E_OK == _stricmp(attr->value(), "true")) ? true : false;
port_config.pads.key_code[Config::BUTTON_A] = GetXMLElementAsInt(joypad_node, "AKey");
port_config.pads.key_code[Config::BUTTON_B] = GetXMLElementAsInt(joypad_node, "BKey");
port_config.pads.key_code[Config::BUTTON_X] = GetXMLElementAsInt(joypad_node, "XKey");
port_config.pads.key_code[Config::BUTTON_Y] = GetXMLElementAsInt(joypad_node, "YKey");
port_config.pads.key_code[Config::TRIGGER_L] = GetXMLElementAsInt(joypad_node, "LKey");
port_config.pads.key_code[Config::TRIGGER_R] = GetXMLElementAsInt(joypad_node, "RKey");
port_config.pads.key_code[Config::BUTTON_Z] = GetXMLElementAsInt(joypad_node, "ZKey");
port_config.pads.key_code[Config::BUTTON_START] = GetXMLElementAsInt(joypad_node, "StartKey");
port_config.pads.key_code[Config::ANALOG_UP] = GetXMLElementAsInt(joypad_node, "AnalogUpKey");
port_config.pads.key_code[Config::ANALOG_DOWN] = GetXMLElementAsInt(joypad_node, "AnalogDownKey");
port_config.pads.key_code[Config::ANALOG_LEFT] = GetXMLElementAsInt(joypad_node, "AnalogLeftKey");
port_config.pads.key_code[Config::ANALOG_RIGHT] = GetXMLElementAsInt(joypad_node, "AnalogRightKey");
port_config.pads.key_code[Config::C_UP] = GetXMLElementAsInt(joypad_node, "CUpKey");
port_config.pads.key_code[Config::C_DOWN] = GetXMLElementAsInt(joypad_node, "CDownKey");
port_config.pads.key_code[Config::C_LEFT] = GetXMLElementAsInt(joypad_node, "CLeftKey");
port_config.pads.key_code[Config::C_RIGHT] = GetXMLElementAsInt(joypad_node, "CRightKey");
port_config.pads.key_code[Config::DPAD_UP] = GetXMLElementAsInt(joypad_node, "DPadUpKey");
port_config.pads.key_code[Config::DPAD_DOWN] = GetXMLElementAsInt(joypad_node, "DPadDownKey");
port_config.pads.key_code[Config::DPAD_LEFT] = GetXMLElementAsInt(joypad_node, "DPadLeftKey");
port_config.pads.key_code[Config::DPAD_RIGHT] = GetXMLElementAsInt(joypad_node, "DPadRightKey");
}
config.set_controller_ports(port, port_config);
}
}
/// Loads/parses an XML configuration file
void LoadXMLConfig(Config& config, const char* filename) {
// Open the XML file
char full_filename[MAX_PATH];
strcpy(full_filename, config.program_dir());
strcat(full_filename, filename);
std::ifstream ifs(full_filename);
// Check that the file is valid
if (ifs.fail()) {
LOG_ERROR(TCONFIG, "XML configuration file %s failed to open!", filename);
return;
}
// Read and parse XML string
std::string xml_str((std::istreambuf_iterator<char>(ifs)), std::istreambuf_iterator<char>());
rapidxml::xml_document<> doc;
doc.parse<0>(const_cast<char *>(xml_str.c_str()));
// Try to load a system configuration
rapidxml::xml_node<> *node = doc.first_node("SysConfig");
// Try to load a game configuation
if (!node) {
node = doc.first_node("GameConfig");
}
// Try to load a user configuation
if (!node) {
node = doc.first_node("UserConfig");
}
// Not proper XML format
if (!node) {
LOG_ERROR(TCONFIG, "XML configuration file incorrect format %s!", filename)
return;
}
// Parse all sub nodes into the config
ParseGeneralNode(node->first_node("General"), config);
ParseDebugNode(node->first_node("Debug"), config);
ParseBootNode(node->first_node("Boot"), config);
ParsePowerPCNode(node->first_node("PowerPC"), config);
ParseVideoNode(node->first_node("Video"), config);
ParseDevicesNode(node->first_node("Devices"), config);
}
} // namespace