2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-17 22:43:11 -04:00
|
|
|
// Refer to the license.txt file included.
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
// PatchEngine
|
|
|
|
// Supports simple memory patches, and has a partial Action Replay implementation
|
|
|
|
// in ActionReplay.cpp/h.
|
|
|
|
|
2017-06-08 00:53:59 -04:00
|
|
|
#include "Core/PatchEngine.h"
|
|
|
|
|
2014-02-19 13:09:14 -05:00
|
|
|
#include <algorithm>
|
|
|
|
#include <map>
|
2015-05-10 20:16:05 -05:00
|
|
|
#include <set>
|
2008-12-08 05:30:24 +00:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2010-11-10 04:12:31 +00:00
|
|
|
|
2016-09-30 15:29:35 +00:00
|
|
|
#include "Common/Assert.h"
|
2014-07-27 13:37:09 -04:00
|
|
|
#include "Common/IniFile.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Common/StringUtil.h"
|
|
|
|
|
|
|
|
#include "Core/ActionReplay.h"
|
|
|
|
#include "Core/ConfigManager.h"
|
|
|
|
#include "Core/GeckoCode.h"
|
|
|
|
#include "Core/GeckoCodeConfig.h"
|
2015-01-17 13:17:36 -08:00
|
|
|
#include "Core/PowerPC/PowerPC.h"
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
namespace PatchEngine
|
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
const char* PatchTypeStrings[] = {
|
2018-04-12 14:18:04 +02:00
|
|
|
"byte",
|
|
|
|
"word",
|
|
|
|
"dword",
|
2008-12-10 22:36:26 +00:00
|
|
|
};
|
|
|
|
|
2014-07-08 15:58:25 +02:00
|
|
|
static std::vector<Patch> onFrame;
|
|
|
|
static std::map<u32, int> speedHacks;
|
2013-04-16 23:14:36 -04:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
void LoadPatchSection(const std::string& section, std::vector<Patch>& patches, IniFile& globalIni,
|
|
|
|
IniFile& localIni)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
// Load the name of all enabled patches
|
|
|
|
std::string enabledSectionName = section + "_Enabled";
|
|
|
|
std::vector<std::string> enabledLines;
|
|
|
|
std::set<std::string> enabledNames;
|
|
|
|
localIni.GetLines(enabledSectionName, &enabledLines);
|
|
|
|
for (const std::string& line : enabledLines)
|
|
|
|
{
|
|
|
|
if (line.size() != 0 && line[0] == '$')
|
|
|
|
{
|
|
|
|
std::string name = line.substr(1, line.size() - 1);
|
|
|
|
enabledNames.insert(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const IniFile* inis[2] = {&globalIni, &localIni};
|
|
|
|
|
|
|
|
for (const IniFile* ini : inis)
|
|
|
|
{
|
|
|
|
std::vector<std::string> lines;
|
|
|
|
Patch currentPatch;
|
|
|
|
ini->GetLines(section, &lines);
|
|
|
|
|
|
|
|
for (std::string& line : lines)
|
|
|
|
{
|
|
|
|
if (line.size() == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (line[0] == '$')
|
|
|
|
{
|
|
|
|
// Take care of the previous code
|
|
|
|
if (currentPatch.name.size())
|
|
|
|
{
|
|
|
|
patches.push_back(currentPatch);
|
|
|
|
}
|
|
|
|
currentPatch.entries.clear();
|
|
|
|
|
|
|
|
// Set active and name
|
|
|
|
currentPatch.name = line.substr(1, line.size() - 1);
|
|
|
|
currentPatch.active = enabledNames.find(currentPatch.name) != enabledNames.end();
|
|
|
|
currentPatch.user_defined = (ini == &localIni);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string::size_type loc = line.find('=');
|
|
|
|
|
|
|
|
if (loc != std::string::npos)
|
|
|
|
{
|
|
|
|
line[loc] = ':';
|
|
|
|
}
|
|
|
|
|
2017-06-11 16:33:10 +02:00
|
|
|
const std::vector<std::string> items = SplitString(line, ':');
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
if (items.size() >= 3)
|
|
|
|
{
|
|
|
|
PatchEntry pE;
|
|
|
|
bool success = true;
|
|
|
|
success &= TryParse(items[0], &pE.address);
|
|
|
|
success &= TryParse(items[2], &pE.value);
|
|
|
|
|
|
|
|
pE.type = PatchType(std::find(PatchTypeStrings, PatchTypeStrings + 3, items[1]) -
|
|
|
|
PatchTypeStrings);
|
|
|
|
success &= (pE.type != (PatchType)3);
|
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
currentPatch.entries.push_back(pE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentPatch.name.size() && currentPatch.entries.size())
|
|
|
|
{
|
|
|
|
patches.push_back(currentPatch);
|
|
|
|
}
|
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
2013-04-16 23:14:36 -04:00
|
|
|
|
2014-03-15 03:29:53 +01:00
|
|
|
static void LoadSpeedhacks(const std::string& section, IniFile& ini)
|
2013-04-16 23:14:36 -04:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
std::vector<std::string> keys;
|
|
|
|
ini.GetKeys(section, &keys);
|
|
|
|
for (const std::string& key : keys)
|
|
|
|
{
|
|
|
|
std::string value;
|
|
|
|
ini.GetOrCreateSection(section)->Get(key, &value, "BOGUS");
|
|
|
|
if (value != "BOGUS")
|
|
|
|
{
|
|
|
|
u32 address;
|
|
|
|
u32 cycles;
|
|
|
|
bool success = true;
|
|
|
|
success &= TryParse(key, &address);
|
|
|
|
success &= TryParse(value, &cycles);
|
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
speedHacks[address] = (int)cycles;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2010-10-02 02:04:44 +00:00
|
|
|
int GetSpeedhackCycles(const u32 addr)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
std::map<u32, int>::const_iterator iter = speedHacks.find(addr);
|
|
|
|
if (iter == speedHacks.end())
|
|
|
|
return 0;
|
|
|
|
else
|
|
|
|
return iter->second;
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2013-09-23 02:39:14 -04:00
|
|
|
void LoadPatches()
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
IniFile merged = SConfig::GetInstance().LoadGameIni();
|
|
|
|
IniFile globalIni = SConfig::GetInstance().LoadDefaultGameIni();
|
|
|
|
IniFile localIni = SConfig::GetInstance().LoadLocalGameIni();
|
2013-09-07 23:02:49 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
LoadPatchSection("OnFrame", onFrame, globalIni, localIni);
|
|
|
|
ActionReplay::LoadAndApplyCodes(globalIni, localIni);
|
2013-09-07 23:02:49 +02:00
|
|
|
|
2017-03-21 16:24:16 -04:00
|
|
|
Gecko::SetActiveCodes(Gecko::LoadCodes(globalIni, localIni));
|
2013-09-07 23:02:49 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
LoadSpeedhacks("Speedhacks", merged);
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
static void ApplyPatches(const std::vector<Patch>& patches)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
for (const Patch& patch : patches)
|
|
|
|
{
|
|
|
|
if (patch.active)
|
|
|
|
{
|
|
|
|
for (const PatchEntry& entry : patch.entries)
|
|
|
|
{
|
|
|
|
u32 addr = entry.address;
|
|
|
|
u32 value = entry.value;
|
|
|
|
switch (entry.type)
|
|
|
|
{
|
|
|
|
case PATCH_8BIT:
|
|
|
|
PowerPC::HostWrite_U8((u8)value, addr);
|
|
|
|
break;
|
|
|
|
case PATCH_16BIT:
|
|
|
|
PowerPC::HostWrite_U16((u16)value, addr);
|
|
|
|
break;
|
|
|
|
case PATCH_32BIT:
|
|
|
|
PowerPC::HostWrite_U32(value, addr);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// unknown patchtype
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 15:29:35 +00:00
|
|
|
// Requires MSR.DR, MSR.IR
|
|
|
|
// There's no perfect way to do this, it's just a heuristic.
|
|
|
|
// We require at least 2 stack frames, if the stack is shallower than that then it won't work.
|
|
|
|
static bool IsStackSane()
|
|
|
|
{
|
2018-05-05 17:02:58 -04:00
|
|
|
DEBUG_ASSERT(MSR.DR && MSR.IR);
|
2016-09-30 15:29:35 +00:00
|
|
|
|
|
|
|
// Check the stack pointer
|
|
|
|
u32 SP = GPR(1);
|
|
|
|
if (!PowerPC::HostIsRAMAddress(SP))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Read the frame pointer from the stack (find 2nd frame from top), assert that it makes sense
|
|
|
|
u32 next_SP = PowerPC::HostRead_U32(SP);
|
|
|
|
if (next_SP <= SP || !PowerPC::HostIsRAMAddress(next_SP) ||
|
|
|
|
!PowerPC::HostIsRAMAddress(next_SP + 4))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check the link register makes sense (that it points to a valid IBAT address)
|
2017-02-22 20:12:19 -08:00
|
|
|
const u32 address = PowerPC::HostRead_U32(next_SP + 4);
|
|
|
|
return PowerPC::HostIsInstructionRAMAddress(address) && 0 != PowerPC::HostRead_U32(address);
|
2016-09-30 15:29:35 +00:00
|
|
|
}
|
|
|
|
|
2016-09-24 15:14:51 +00:00
|
|
|
bool ApplyFramePatches()
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2016-09-24 15:14:51 +00:00
|
|
|
// Because we're using the VI Interrupt to time this instead of patching the game with a
|
|
|
|
// callback hook we can end up catching the game in an exception vector.
|
|
|
|
// We deal with this by returning false so that SystemTimers will reschedule us in a few cycles
|
|
|
|
// where we can try again after the CPU hopefully returns back to the normal instruction flow.
|
2018-05-05 17:02:58 -04:00
|
|
|
if (!MSR.DR || !MSR.IR || !IsStackSane())
|
2016-09-24 15:14:51 +00:00
|
|
|
{
|
2016-11-23 00:58:07 -05:00
|
|
|
DEBUG_LOG(
|
2016-09-24 15:14:51 +00:00
|
|
|
ACTIONREPLAY,
|
|
|
|
"Need to retry later. CPU configuration is currently incorrect. PC = 0x%08X, MSR = 0x%08X",
|
2018-05-05 17:02:58 -04:00
|
|
|
PC, MSR.Hex);
|
2016-09-24 15:14:51 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
ApplyPatches(onFrame);
|
|
|
|
|
|
|
|
// Run the Gecko code handler
|
2016-09-24 15:14:51 +00:00
|
|
|
Gecko::RunCodeHandler();
|
2016-06-24 10:43:46 +02:00
|
|
|
ActionReplay::RunAllActive();
|
2016-09-24 15:14:51 +00:00
|
|
|
|
|
|
|
return true;
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
2009-02-20 22:04:52 +00:00
|
|
|
|
2013-07-25 16:43:00 -04:00
|
|
|
void Shutdown()
|
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
onFrame.clear();
|
2016-07-01 13:07:57 +02:00
|
|
|
speedHacks.clear();
|
|
|
|
ActionReplay::ApplyCodes({});
|
2016-09-30 16:19:47 +00:00
|
|
|
Gecko::Shutdown();
|
2013-07-25 16:43:00 -04:00
|
|
|
}
|
|
|
|
|
2017-04-06 15:34:40 +01:00
|
|
|
void Reload()
|
|
|
|
{
|
|
|
|
Shutdown();
|
|
|
|
LoadPatches();
|
|
|
|
}
|
|
|
|
|
2008-12-08 05:30:24 +00:00
|
|
|
} // namespace
|