mirror of
https://github.com/RPCS3/rpcs3.git
synced 2025-01-16 07:10:40 +00:00
ad8988afd3
- PS3 games include both PPU and SPU code in their PPU executables, so to make patching games that make use of the same SPU libraries easier, we add a system to find and patch them. - Patches for this system still use SPU LS (Local Storage) addresses despite the fact that we aren't loading anything into SPU LS at this time. The patches are checked against each segment and patched in place.
251 lines
5.0 KiB
C++
251 lines
5.0 KiB
C++
#include "bin_patch.h"
|
|
#include <yaml-cpp/yaml.h>
|
|
#include "File.h"
|
|
#include "Config.h"
|
|
|
|
template <>
|
|
void fmt_class_string<patch_type>::format(std::string& out, u64 arg)
|
|
{
|
|
format_enum(out, arg, [](patch_type value)
|
|
{
|
|
switch (value)
|
|
{
|
|
case patch_type::load: return "load";
|
|
case patch_type::byte: return "byte";
|
|
case patch_type::le16: return "le16";
|
|
case patch_type::le32: return "le32";
|
|
case patch_type::le64: return "le64";
|
|
case patch_type::bef32: return "bef32";
|
|
case patch_type::bef64: return "bef64";
|
|
case patch_type::be16: return "be16";
|
|
case patch_type::be32: return "be32";
|
|
case patch_type::be64: return "be64";
|
|
case patch_type::lef32: return "lef32";
|
|
case patch_type::lef64: return "lef64";
|
|
}
|
|
|
|
return unknown;
|
|
});
|
|
}
|
|
|
|
void patch_engine::append(const std::string& patch)
|
|
{
|
|
if (fs::file f{patch})
|
|
{
|
|
YAML::Node root;
|
|
|
|
try
|
|
{
|
|
root = YAML::Load(f.to_string());
|
|
}
|
|
catch (const std::exception& e)
|
|
{
|
|
LOG_FATAL(GENERAL, "Failed to load patch file %s\n%s thrown: %s", patch, typeid(e).name(), e.what());
|
|
return;
|
|
}
|
|
|
|
for (auto pair : root)
|
|
{
|
|
auto& name = pair.first.Scalar();
|
|
auto& data = m_map[name];
|
|
|
|
for (auto patch : pair.second)
|
|
{
|
|
u64 type64 = 0;
|
|
cfg::try_to_enum_value(&type64, &fmt_class_string<patch_type>::format, patch[0].Scalar());
|
|
|
|
struct patch info{};
|
|
info.type = static_cast<patch_type>(type64);
|
|
info.offset = patch[1].as<u32>(0);
|
|
|
|
switch (info.type)
|
|
{
|
|
case patch_type::load:
|
|
{
|
|
// Special syntax: copy named sequence (must be loaded before)
|
|
const auto found = m_map.find(patch[1].Scalar());
|
|
|
|
if (found != m_map.end())
|
|
{
|
|
// Address modifier (optional)
|
|
const u32 mod = patch[2].as<u32>(0);
|
|
|
|
for (const auto& rd : found->second)
|
|
{
|
|
info = rd;
|
|
info.offset += mod;
|
|
data.emplace_back(info);
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
// TODO: error
|
|
break;
|
|
}
|
|
case patch_type::bef32:
|
|
case patch_type::lef32:
|
|
{
|
|
info.value = std::bit_cast<u32>(patch[2].as<f32>());
|
|
break;
|
|
}
|
|
case patch_type::bef64:
|
|
case patch_type::lef64:
|
|
{
|
|
info.value = std::bit_cast<u64>(patch[2].as<f64>());
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
info.value = patch[2].as<u64>();
|
|
break;
|
|
}
|
|
}
|
|
|
|
data.emplace_back(info);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
std::size_t patch_engine::apply(const std::string& name, u8* dst) const
|
|
{
|
|
const auto found = m_map.find(name);
|
|
|
|
if (found == m_map.cend())
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// Apply modifications sequentially
|
|
for (const auto& p : found->second)
|
|
{
|
|
auto ptr = dst + p.offset;
|
|
|
|
switch (p.type)
|
|
{
|
|
case patch_type::load:
|
|
{
|
|
// Invalid in this context
|
|
break;
|
|
}
|
|
case patch_type::byte:
|
|
{
|
|
*ptr = static_cast<u8>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::le16:
|
|
{
|
|
*reinterpret_cast<le_t<u16, 1>*>(ptr) = static_cast<u16>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::le32:
|
|
case patch_type::lef32:
|
|
{
|
|
*reinterpret_cast<le_t<u32, 1>*>(ptr) = static_cast<u32>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::le64:
|
|
case patch_type::lef64:
|
|
{
|
|
*reinterpret_cast<le_t<u64, 1>*>(ptr) = static_cast<u64>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::be16:
|
|
{
|
|
*reinterpret_cast<be_t<u16, 1>*>(ptr) = static_cast<u16>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::be32:
|
|
case patch_type::bef32:
|
|
{
|
|
*reinterpret_cast<be_t<u32, 1>*>(ptr) = static_cast<u32>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::be64:
|
|
case patch_type::bef64:
|
|
{
|
|
*reinterpret_cast<be_t<u64, 1>*>(ptr) = static_cast<u64>(p.value);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return found->second.size();
|
|
}
|
|
|
|
std::size_t patch_engine::apply_with_ls_check(const std::string& name, u8* dst, u32 filesz, u32 ls_addr) const
|
|
{
|
|
u32 rejected = 0;
|
|
|
|
const auto found = m_map.find(name);
|
|
|
|
if (found == m_map.cend())
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// Apply modifications sequentially
|
|
for (const auto& p : found->second)
|
|
{
|
|
auto ptr = dst + (p.offset - ls_addr);
|
|
|
|
if(p.offset < ls_addr || p.offset >= (ls_addr + filesz))
|
|
{
|
|
// This patch is out of range for this segment
|
|
rejected++;
|
|
continue;
|
|
}
|
|
|
|
switch (p.type)
|
|
{
|
|
case patch_type::load:
|
|
{
|
|
// Invalid in this context
|
|
break;
|
|
}
|
|
case patch_type::byte:
|
|
{
|
|
*ptr = static_cast<u8>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::le16:
|
|
{
|
|
*reinterpret_cast<le_t<u16, 1>*>(ptr) = static_cast<u16>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::le32:
|
|
case patch_type::lef32:
|
|
{
|
|
*reinterpret_cast<le_t<u32, 1>*>(ptr) = static_cast<u32>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::le64:
|
|
case patch_type::lef64:
|
|
{
|
|
*reinterpret_cast<le_t<u64, 1>*>(ptr) = static_cast<u64>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::be16:
|
|
{
|
|
*reinterpret_cast<be_t<u16, 1>*>(ptr) = static_cast<u16>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::be32:
|
|
case patch_type::bef32:
|
|
{
|
|
*reinterpret_cast<be_t<u32, 1>*>(ptr) = static_cast<u32>(p.value);
|
|
break;
|
|
}
|
|
case patch_type::be64:
|
|
case patch_type::bef64:
|
|
{
|
|
*reinterpret_cast<be_t<u64, 1>*>(ptr) = static_cast<u64>(p.value);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return (found->second.size() - rejected);
|
|
}
|