#include "helpers/patches.hpp" #include #include #include #include namespace hex { static void pushBytesBack(std::vector &buffer, const char* bytes) { std::string_view string(bytes); buffer.resize(buffer.size() + string.length()); std::memcpy((&buffer.back() - string.length()) + 1, string.begin(), string.length()); } template static void pushBytesBack(std::vector &buffer, T bytes) { buffer.resize(buffer.size() + sizeof(T)); std::memcpy((&buffer.back() - sizeof(T)) + 1, &bytes, sizeof(T)); } std::vector generateIPSPatch(const Patches &patches) { std::vector result; pushBytesBack(result, "PATCH"); std::vector addresses; std::vector values; for (const auto &[address, value] : patches) { addresses.push_back(address); values.push_back(value); } std::optional startAddress; std::vector bytes; for (u32 i = 0; i < addresses.size(); i++) { if (!startAddress.has_value()) startAddress = addresses[i]; if (i != addresses.size() - 1 && addresses[i] == (addresses[i + 1] - 1)) { bytes.push_back(values[i]); } else { bytes.push_back(values[i]); if (bytes.size() > 0xFFFF || startAddress > 0xFF'FFFF) return { }; u32 address = startAddress.value(); auto addressBytes = reinterpret_cast(&address); result.push_back(addressBytes[2]); result.push_back(addressBytes[1]); result.push_back(addressBytes[0]); pushBytesBack(result, changeEndianess(bytes.size(), std::endian::big)); for (auto byte : bytes) result.push_back(byte); bytes.clear(); startAddress = { }; } } pushBytesBack(result, "EOF"); return result; } std::vector generateIPS32Patch(const Patches &patches) { std::vector result; pushBytesBack(result, "IPS32"); std::vector addresses; std::vector values; for (const auto &[address, value] : patches) { addresses.push_back(address); values.push_back(value); } std::optional startAddress; std::vector bytes; for (u32 i = 0; i < addresses.size(); i++) { if (!startAddress.has_value()) startAddress = addresses[i]; if (i != addresses.size() - 1 && addresses[i] == (addresses[i + 1] - 1)) { bytes.push_back(values[i]); } else { bytes.push_back(values[i]); if (bytes.size() > 0xFFFF || startAddress > 0xFFFF'FFFF) return { }; u32 address = startAddress.value(); auto addressBytes = reinterpret_cast(&address); result.push_back(addressBytes[3]); result.push_back(addressBytes[2]); result.push_back(addressBytes[1]); result.push_back(addressBytes[0]); pushBytesBack(result, changeEndianess(bytes.size(), std::endian::big)); for (auto byte : bytes) result.push_back(byte); bytes.clear(); startAddress = { }; } } pushBytesBack(result, "EEOF"); return result; } Patches loadIPSPatch(const std::vector &ipsPatch) { if (ipsPatch.size() < (5 + 3)) return { }; if (std::memcmp(ipsPatch.data(), "PATCH", 5) != 0) return { }; Patches result; bool foundEOF = false; u32 ipsOffset = 5; while (ipsOffset < ipsPatch.size() - (5 + 3)) { u32 offset = ipsPatch[ipsOffset + 2] | (ipsPatch[ipsOffset + 1] << 8) | (ipsPatch[ipsOffset + 0] << 16); u16 size = ipsPatch[ipsOffset + 4] | (ipsPatch[ipsOffset + 3] << 8); ipsOffset += 5; // Handle normal record if (size > 0x0000) { if (ipsOffset + size > ipsPatch.size() - 3) return { }; for (u16 i = 0; i < size; i++) result[offset + i] = ipsPatch[ipsOffset + i]; ipsOffset += size; } // Handle RLE record else { if (ipsOffset + 3 > ipsPatch.size() - 3) return { }; u16 rleSize = ipsPatch[ipsOffset + 0] | (ipsPatch[ipsOffset + 1] << 8); ipsOffset += 2; for (u16 i = 0; i < rleSize; i++) result[offset + i] = ipsPatch[ipsOffset + 0]; ipsOffset += 1; } if (std::memcmp(ipsPatch.data(), "EOF", 3)) foundEOF = true; } if (foundEOF) return result; else return { }; } Patches loadIPS32Patch(const std::vector &ipsPatch) { if (ipsPatch.size() < (5 + 4)) return { }; if (std::memcmp(ipsPatch.data(), "IPS32", 5) != 0) return { }; Patches result; bool foundEEOF = false; u32 ipsOffset = 5; while (ipsOffset < ipsPatch.size() - (5 + 4)) { u32 offset = ipsPatch[ipsOffset + 3] | (ipsPatch[ipsOffset + 2] << 8) | (ipsPatch[ipsOffset + 1] << 16) | (ipsPatch[ipsOffset + 0] << 24); u16 size = ipsPatch[ipsOffset + 5] | (ipsPatch[ipsOffset + 4] << 8); ipsOffset += 6; // Handle normal record if (size > 0x0000) { if (ipsOffset + size > ipsPatch.size() - 3) return { }; for (u16 i = 0; i < size; i++) result[offset + i] = ipsPatch[ipsOffset + i]; ipsOffset += size; } // Handle RLE record else { if (ipsOffset + 3 > ipsPatch.size() - 3) return { }; u16 rleSize = ipsPatch[ipsOffset + 0] | (ipsPatch[ipsOffset + 1] << 8); ipsOffset += 2; for (u16 i = 0; i < rleSize; i++) result[offset + i] = ipsPatch[ipsOffset + 0]; ipsOffset += 1; } if (std::memcmp(ipsPatch.data(), "EEOF", 4)) foundEEOF = true; } if (foundEEOF) return result; else return { }; } }