WUMSLoader/source/ElfUtils.cpp

254 lines
11 KiB
C++
Raw Normal View History

2022-02-04 21:44:03 +01:00
#include "utils/logger.h"
2020-04-28 14:43:07 +02:00
#include <coreinit/cache.h>
2022-02-04 21:44:03 +01:00
#include <coreinit/debug.h>
2020-04-28 14:43:07 +02:00
#include <coreinit/memdefaultheap.h>
#include <whb/file.h>
#include <whb/log.h>
2022-02-04 21:44:03 +01:00
#include <whb/sdcard.h>
2020-04-28 14:43:07 +02:00
#include "ElfUtils.h"
2022-02-04 21:44:03 +01:00
#include "elfio/elfio.hpp"
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
int32_t LoadFileToMem(const char *relativefilepath, char **fileOut, uint32_t *sizeOut) {
2020-04-28 14:43:07 +02:00
char path[256];
2022-02-04 21:44:03 +01:00
int result = 0;
2021-09-18 11:55:01 +02:00
char *sdRootPath = nullptr;
2020-04-28 14:43:07 +02:00
if (!WHBMountSdCard()) {
2021-04-17 13:25:54 +02:00
DEBUG_FUNCTION_LINE("Failed to mount SD Card...");
2020-04-28 14:43:07 +02:00
result = -1;
goto exit;
}
sdRootPath = WHBGetSdCardMountPath();
2020-05-17 19:05:51 +02:00
sprintf(path, "%s/%s", sdRootPath, relativefilepath);
2020-04-28 14:43:07 +02:00
2021-04-17 13:25:54 +02:00
DEBUG_FUNCTION_LINE("Loading file %s.", path);
2020-04-28 14:43:07 +02:00
*fileOut = WHBReadWholeFile(path, sizeOut);
if (!(*fileOut)) {
result = -2;
2021-04-17 13:25:54 +02:00
DEBUG_FUNCTION_LINE("WHBReadWholeFile(%s) returned NULL", path);
2020-04-28 14:43:07 +02:00
goto exit;
}
2022-02-04 21:44:03 +01:00
exit:
2020-04-28 14:43:07 +02:00
WHBUnmountSdCard();
return result;
}
2020-05-17 19:05:51 +02:00
uint32_t load_loader_elf_from_sd(unsigned char *baseAddress, const char *relativePath) {
2022-02-04 21:44:03 +01:00
char *elf_data = nullptr;
2020-04-28 14:43:07 +02:00
uint32_t fileSize = 0;
2020-05-17 19:05:51 +02:00
if (LoadFileToMem(relativePath, &elf_data, &fileSize) != 0) {
2020-04-28 14:43:07 +02:00
OSFatal("Failed to load hook_payload.elf from the SD Card.");
}
uint32_t result = load_loader_elf(baseAddress, elf_data, fileSize);
2020-05-17 19:05:51 +02:00
MEMFreeToDefaultHeap((void *) elf_data);
2020-04-28 14:43:07 +02:00
return result;
}
2020-05-17 19:05:51 +02:00
uint32_t load_loader_elf(unsigned char *baseAddress, char *elf_data, uint32_t fileSize) {
2020-04-28 14:43:07 +02:00
ELFIO::Elf32_Ehdr *ehdr;
ELFIO::Elf32_Phdr *phdrs;
uint8_t *image;
int32_t i;
ehdr = (ELFIO::Elf32_Ehdr *) elf_data;
2020-05-17 19:05:51 +02:00
if (ehdr->e_phoff == 0 || ehdr->e_phnum == 0) {
2020-04-28 14:43:07 +02:00
return 0;
}
2020-05-17 19:05:51 +02:00
if (ehdr->e_phentsize != sizeof(ELFIO::Elf32_Phdr)) {
2020-04-28 14:43:07 +02:00
return 0;
}
2020-05-17 19:05:51 +02:00
phdrs = (ELFIO::Elf32_Phdr *) (elf_data + ehdr->e_phoff);
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
for (i = 0; i < ehdr->e_phnum; i++) {
if (phdrs[i].p_type != PT_LOAD) {
2020-04-28 14:43:07 +02:00
continue;
}
2020-05-17 19:05:51 +02:00
if (phdrs[i].p_filesz > phdrs[i].p_memsz) {
2020-04-28 14:43:07 +02:00
continue;
}
2020-05-17 19:05:51 +02:00
if (!phdrs[i].p_filesz) {
2020-04-28 14:43:07 +02:00
continue;
}
2020-05-17 19:05:51 +02:00
uint32_t p_paddr = phdrs[i].p_paddr + (uint32_t) baseAddress;
2022-02-04 21:44:03 +01:00
image = (uint8_t *) (elf_data + phdrs[i].p_offset);
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
memcpy((void *) p_paddr, image, phdrs[i].p_filesz);
DCFlushRange((void *) p_paddr, phdrs[i].p_filesz);
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
if (phdrs[i].p_flags & PF_X) {
ICInvalidateRange((void *) p_paddr, phdrs[i].p_memsz);
2020-04-28 14:43:07 +02:00
}
}
//! clear BSS
2021-09-18 11:55:01 +02:00
auto *shdr = (ELFIO::Elf32_Shdr *) (elf_data + ehdr->e_shoff);
2020-05-17 19:05:51 +02:00
for (i = 0; i < ehdr->e_shnum; i++) {
const char *section_name = ((const char *) elf_data) + shdr[ehdr->e_shstrndx].sh_offset + shdr[i].sh_name;
if (section_name[0] == '.' && section_name[1] == 'b' && section_name[2] == 's' && section_name[3] == 's') {
memset((void *) (shdr[i].sh_addr + baseAddress), 0, shdr[i].sh_size);
DCFlushRange((void *) (shdr[i].sh_addr + baseAddress), shdr[i].sh_size);
} else if (section_name[0] == '.' && section_name[1] == 's' && section_name[2] == 'b' && section_name[3] == 's' && section_name[4] == 's') {
memset((void *) (shdr[i].sh_addr + baseAddress), 0, shdr[i].sh_size);
DCFlushRange((void *) (shdr[i].sh_addr + baseAddress), shdr[i].sh_size);
2020-04-28 14:43:07 +02:00
}
}
return ehdr->e_entry;
}
// See https://github.com/decaf-emu/decaf-emu/blob/43366a34e7b55ab9d19b2444aeb0ccd46ac77dea/src/libdecaf/src/cafe/loader/cafe_loader_reloc.cpp#L144
2022-01-27 12:52:45 +01:00
bool ElfUtils::elfLinkOne(char type, size_t offset, int32_t addend, uint32_t destination, uint32_t symbol_addr, relocation_trampoline_entry_t *trampoline_data, uint32_t trampoline_data_length,
2021-09-23 18:38:29 +02:00
RelocationType reloc_type) {
2020-05-17 19:05:51 +02:00
if (type == R_PPC_NONE) {
2020-04-28 14:43:07 +02:00
return true;
}
2020-05-17 19:05:51 +02:00
auto target = destination + offset;
2022-02-04 21:44:03 +01:00
auto value = symbol_addr + addend;
2020-04-28 14:43:07 +02:00
auto relValue = value - static_cast<uint32_t>(target);
switch (type) {
2020-05-17 19:05:51 +02:00
case R_PPC_NONE:
break;
case R_PPC_ADDR32:
*((uint32_t *) (target)) = value;
break;
case R_PPC_ADDR16_LO:
*((uint16_t *) (target)) = static_cast<uint16_t>(value & 0xFFFF);
break;
case R_PPC_ADDR16_HI:
*((uint16_t *) (target)) = static_cast<uint16_t>(value >> 16);
break;
case R_PPC_ADDR16_HA:
*((uint16_t *) (target)) = static_cast<uint16_t>((value + 0x8000) >> 16);
break;
case R_PPC_DTPMOD32:
DEBUG_FUNCTION_LINE("################IMPLEMENT ME\n");
//*((int32_t *)(target)) = tlsModuleIndex;
break;
case R_PPC_DTPREL32:
*((uint32_t *) (target)) = value;
break;
case R_PPC_GHS_REL16_HA:
*((uint16_t *) (target)) = static_cast<uint16_t>((relValue + 0x8000) >> 16);
break;
case R_PPC_GHS_REL16_HI:
*((uint16_t *) (target)) = static_cast<uint16_t>(relValue >> 16);
break;
case R_PPC_GHS_REL16_LO:
*((uint16_t *) (target)) = static_cast<uint16_t>(relValue & 0xFFFF);
break;
case R_PPC_REL14: {
auto distance = static_cast<int32_t>(value) - static_cast<int32_t>(target);
if (distance > 0x7FFC || distance < -0x7FFC) {
DEBUG_FUNCTION_LINE("***14-bit relative branch cannot hit target.");
return false;
}
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
if (distance & 3) {
DEBUG_FUNCTION_LINE("***RELOC ERROR %d: lower 2 bits must be zero before shifting.", -470040);
return false;
}
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
if ((distance >= 0 && (distance & 0xFFFF8000)) ||
2020-04-28 14:43:07 +02:00
(distance < 0 && ((distance & 0xFFFF8000) != 0xFFFF8000))) {
2020-05-17 19:05:51 +02:00
DEBUG_FUNCTION_LINE("***RELOC ERROR %d: upper 17 bits before shift must all be the same.", -470040);
2020-04-28 14:43:07 +02:00
return false;
2020-05-17 19:05:51 +02:00
}
*(int32_t *) target = (*(int32_t *) target & 0xFFBF0003) | (distance & 0x0000fffc);
break;
}
case R_PPC_REL24: {
// if (isWeakSymbol && !symbolValue) {
// symbolValue = static_cast<uint32_t>(target);
// value = symbolValue + addend;
// }
auto distance = static_cast<int32_t>(value) - static_cast<int32_t>(target);
if (distance > 0x1FFFFFC || distance < -0x1FFFFFC) {
2022-01-27 12:52:45 +01:00
if (trampoline_data == nullptr) {
DEBUG_FUNCTION_LINE("***24-bit relative branch cannot hit target. Trampoline isn't provided\n");
2020-04-28 14:43:07 +02:00
DEBUG_FUNCTION_LINE("***value %08X - target %08X = distance %08X\n", value, target, distance);
return false;
2020-05-17 19:05:51 +02:00
} else {
2022-01-27 12:52:45 +01:00
relocation_trampoline_entry_t *freeSlot = nullptr;
for (uint32_t i = 0; i < trampoline_data_length; i++) {
2020-05-17 19:05:51 +02:00
// We want to override "old" relocations of imports
// Pending relocations have the status RELOC_TRAMP_IMPORT_IN_PROGRESS.
// When all relocations are done successfully, they will be turned into RELOC_TRAMP_IMPORT_DONE
// so they can be overridden/updated/reused on the next application launch.
//
// Relocations that won't change will have the status RELOC_TRAMP_FIXED and are set to free when the module is unloaded.
2022-01-27 12:52:45 +01:00
if (trampoline_data[i].status == RELOC_TRAMP_FREE ||
trampoline_data[i].status == RELOC_TRAMP_IMPORT_DONE) {
freeSlot = &(trampoline_data[i]);
2020-05-17 19:05:51 +02:00
break;
}
}
2021-09-18 12:01:04 +02:00
if (freeSlot == nullptr) {
2022-01-27 12:52:45 +01:00
DEBUG_FUNCTION_LINE("***24-bit relative branch cannot hit target. Trampoline data list is full\n");
2022-02-04 21:44:03 +01:00
DEBUG_FUNCTION_LINE("***value %08X - target %08X = distance %08X\n", value, target, (target - (uint32_t) & (freeSlot->trampoline[0])));
2020-05-17 19:05:51 +02:00
return false;
}
2022-02-04 21:44:03 +01:00
if (target - (uint32_t) & (freeSlot->trampoline[0]) > 0x1FFFFFC) {
2020-05-17 19:05:51 +02:00
DEBUG_FUNCTION_LINE("**Cannot link 24-bit jump (too far to tramp buffer).");
2022-02-04 21:44:03 +01:00
DEBUG_FUNCTION_LINE("***value %08X - target %08X = distance %08X\n", value, target, (target - (uint32_t) & (freeSlot->trampoline[0])));
2020-05-17 19:05:51 +02:00
return false;
}
2022-01-27 12:52:45 +01:00
freeSlot->trampoline[0] = 0x3D600000 | ((((uint32_t) value) >> 16) & 0x0000FFFF); // lis r11, real_addr@h
2022-02-04 21:44:03 +01:00
freeSlot->trampoline[1] = 0x616B0000 | (((uint32_t) value) & 0x0000ffff); // ori r11, r11, real_addr@l
freeSlot->trampoline[2] = 0x7D6903A6; // mtctr r11
freeSlot->trampoline[3] = 0x4E800420; // bctr
2022-01-27 12:52:45 +01:00
DCFlushRange((void *) freeSlot->trampoline, sizeof(freeSlot->trampoline));
ICInvalidateRange((unsigned char *) freeSlot->trampoline, sizeof(freeSlot->trampoline));
2020-05-17 19:05:51 +02:00
if (reloc_type == RELOC_TYPE_FIXED) {
freeSlot->status = RELOC_TRAMP_FIXED;
} else {
// Relocations for the imports may be overridden
2021-09-18 12:01:04 +02:00
freeSlot->status = RELOC_TRAMP_IMPORT_DONE;
2020-05-17 19:05:51 +02:00
}
2022-02-04 21:44:03 +01:00
auto symbolValue = (uint32_t) & (freeSlot->trampoline[0]);
value = symbolValue + addend;
distance = static_cast<int32_t>(value) - static_cast<int32_t>(target);
2020-04-28 14:43:07 +02:00
}
2020-05-17 19:05:51 +02:00
}
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
if (distance & 3) {
DEBUG_FUNCTION_LINE("***RELOC ERROR %d: lower 2 bits must be zero before shifting.", -470022);
return false;
}
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
if (distance < 0 && (distance & 0xFE000000) != 0xFE000000) {
DEBUG_FUNCTION_LINE("***RELOC ERROR %d: upper 7 bits before shift must all be the same (1).", -470040);
return false;
2020-04-28 14:43:07 +02:00
}
2020-05-17 19:05:51 +02:00
if (distance >= 0 && (distance & 0xFE000000)) {
DEBUG_FUNCTION_LINE("***RELOC ERROR %d: upper 7 bits before shift must all be the same (0).", -470040);
return false;
}
2020-04-28 14:43:07 +02:00
2020-05-17 19:05:51 +02:00
*(int32_t *) target = (*(int32_t *) target & 0xfc000003) | (distance & 0x03fffffc);
break;
2020-04-28 14:43:07 +02:00
}
2020-05-17 19:05:51 +02:00
default:
DEBUG_FUNCTION_LINE("***ERROR: Unsupported Relocation_Add Type (%08X):", type);
2020-04-28 14:43:07 +02:00
return false;
}
return true;
}