#include "function_replacements.h" #include "memory_mapping.h" #include DECL_FUNCTION(uint32_t, KiEffectiveToPhysical, uint32_t addressSpace, uint32_t virtualAddress) { uint32_t result = real_KiEffectiveToPhysical(addressSpace, virtualAddress); if (result == 0) { return MemoryMapping_EffectiveToPhysical(virtualAddress); } return result; } DECL_FUNCTION(int32_t, sCheckDataRange, uint32_t address, uint32_t maxDataSize) { if ((address >= 0x00800000 && address < 0x01000000) || (address & 0xF0000000) == (MEMORY_START_BASE & 0xF0000000)) { return 1; } return real_sCheckDataRange(address, maxDataSize); } DECL_FUNCTION(uint32_t, KiPhysicalToEffectiveCached, uint32_t addressSpace, uint32_t virtualAddress) { uint32_t result = real_KiPhysicalToEffectiveCached(addressSpace, virtualAddress); if (result == 0) { return MemoryMapping_PhysicalToEffective(virtualAddress); } return result; } DECL_FUNCTION(uint32_t, KiPhysicalToEffectiveUncached, uint32_t addressSpace, uint32_t virtualAddress) { uint32_t result = real_KiPhysicalToEffectiveUncached(addressSpace, virtualAddress); if (result == 0) { return MemoryMapping_PhysicalToEffective(virtualAddress); } return result; } DECL_FUNCTION(uint32_t, IPCKDriver_ValidatePhysicalAddress, uint32_t u1, uint32_t physStart, uint32_t physEnd) { uint32_t result = MemoryMapping_PhysicalToEffective(physStart) > 0; if (result) { return result; } return real_IPCKDriver_ValidatePhysicalAddress(u1, physStart, physEnd); } DECL_FUNCTION(uint32_t, KiIsEffectiveRangeValid, uint32_t addressSpace, uint32_t virtualAddress, uint32_t size) { uint32_t result = real_KiIsEffectiveRangeValid(addressSpace, virtualAddress, size); if (result == 0) { result = MemoryMapping_EffectiveToPhysical(virtualAddress) > 0; } return result; } // clang-format off #define k_memcpy ((void(*)(void *, void *, uint32_t))(0xfff09e44)) // clang-format on DECL_FUNCTION(uint32_t, KiGetOrPutUserData, void *src, uint32_t size, void *dst, bool isRead) { // if (isRead && MemoryMapping_EffectiveToPhysical((uint32_t) src) > 0) { k_memcpy(dst, src, size); return 1; } else if (!isRead && MemoryMapping_EffectiveToPhysical((uint32_t) dst) > 0) { // src and dst are swapped here k_memcpy(src, dst, size); return 1; } return real_KiGetOrPutUserData(src, size, dst, isRead); } DECL_FUNCTION(MEMHeapHandle, MEMFindContainHeap, void *block) { auto result = MemoryMapping_MEMFindContainHeap(block); if (result == nullptr) { return real_MEMFindContainHeap(block); } return result; } DECL_FUNCTION(int32_t, LiValidateAddress, void *ptr, uint32_t size, uint32_t alignMask, int32_t errorCode, void *minAddr, void *maxAddr, const char *name) { if (((uint32_t) ptr >= 0x00800000 && (uint32_t) ptr < 0x01000000) || ((uint32_t) ptr & 0xF0000000) == (MEMORY_START_BASE & 0xF0000000)) { return 0; } return real_LiValidateAddress(ptr, size, alignMask, errorCode, minAddr, maxAddr, name); } // clang-format off function_replacement_data_t function_replacements[] __attribute__((section(".data"))) = { REPLACE_FUNCTION_VIA_ADDRESS(LiValidateAddress, 0x32008904, 0x01008904), REPLACE_FUNCTION_VIA_ADDRESS(sCheckDataRange, 0x3200cf60, 0x0100cf60), REPLACE_FUNCTION_VIA_ADDRESS(KiEffectiveToPhysical, 0xffee0aac, 0xffee0aac), REPLACE_FUNCTION_VIA_ADDRESS(KiPhysicalToEffectiveCached, 0xffee0a3c, 0xffee0a3c), REPLACE_FUNCTION_VIA_ADDRESS(KiPhysicalToEffectiveUncached, 0xffee0a80, 0xffee0a80), REPLACE_FUNCTION_VIA_ADDRESS(KiIsEffectiveRangeValid, 0xffee0d6c, 0xffee0d6c), REPLACE_FUNCTION_VIA_ADDRESS(IPCKDriver_ValidatePhysicalAddress, 0xfff0cb5c, 0xfff0cb5c), REPLACE_FUNCTION_VIA_ADDRESS(KiGetOrPutUserData, 0xffee0794, 0xffee0794), REPLACE_FUNCTION(MEMFindContainHeap, LIBRARY_COREINIT, MEMFindContainHeap), }; // clang-format on uint32_t function_replacements_size __attribute__((section(".data"))) = sizeof(function_replacements) / sizeof(function_replacement_data_t);