flycast/core/reios/reios_elf.cpp

67 lines
1.5 KiB
C++

#include "reios.h"
extern "C" {
#include <elf/elf.h>
}
#include "hw/sh4/sh4_mem.h"
bool reios_loadElf(const std::string& elf) {
FILE* f = nowide::fopen(elf.c_str(), "rb");
if (!f)
return false;
std::fseek(f, 0, SEEK_END);
size_t size = std::ftell(f);
if (size == 0 || size > 16_MB) {
std::fclose(f);
return false;
}
void* elfF = malloc(size);
std::fseek(f, 0, SEEK_SET);
size_t nread = std::fread(elfF, 1, size, f);
std::fclose(f);
elf_t elfFile;
if (nread != size || elf_newFile(elfF, nread, &elfFile) != 0)
{
free(elfF);
return false;
}
bool phys = false;
for (size_t i = 0; i < elf_getNumProgramHeaders(&elfFile); i++)
{
uint32_t type = elf_getProgramHeaderType(&elfFile, i);
if (type != PT_LOAD) {
DEBUG_LOG(REIOS, "Ignoring section %d type %d", (int)i, type);
continue;
}
// Load that section
uint64_t dest;
if (phys)
dest = elf_getProgramHeaderPaddr(&elfFile, i);
else
dest = elf_getProgramHeaderVaddr(&elfFile, i);
size_t len = elf_getProgramHeaderFileSize(&elfFile, i);
void *src = (u8 *)(elfFile.elfFile) + elf_getProgramHeaderOffset(&elfFile, i);
u8* ptr = GetMemPtr(dest, len);
if (ptr == NULL)
{
WARN_LOG(REIOS, "Invalid load address for section %d: %08lx", (int)i, (long)dest);
continue;
}
DEBUG_LOG(REIOS, "Loading section %d to %08lx - %08lx", (int)i, (long)dest, (long)(dest + len - 1));
memcpy(ptr, src, len);
ptr += len;
memset(ptr, 0, elf_getProgramHeaderMemorySize(&elfFile, i) - len);
}
free(elfF);
return true;
}