wut/tools/elf2rpl/main.cpp

954 lines
27 KiB
C++
Raw Normal View History

2018-05-22 23:08:13 +01:00
#include "elf.h"
#include "utils.h"
2016-01-04 13:17:43 +00:00
#include <algorithm>
2018-05-22 23:08:13 +01:00
#include <fmt/format.h>
2016-01-04 13:17:43 +00:00
#include <fstream>
2018-05-22 23:08:13 +01:00
#include <memory>
#include <set>
2016-01-04 13:17:43 +00:00
#include <string>
#include <vector>
2016-01-05 15:44:55 -08:00
#include <zlib.h>
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
constexpr auto CodeBaseAddress = 0x02000000;
constexpr auto DataBaseAddress = 0x10000000;
constexpr auto LoadBaseAddress = 0xC0000000;
2016-01-04 13:17:43 +00:00
struct ElfFile
{
2018-05-22 23:08:13 +01:00
struct Section
2016-01-04 13:17:43 +00:00
{
2018-05-22 23:08:13 +01:00
elf::SectionHeader header;
2016-01-04 13:17:43 +00:00
std::string name;
2018-05-22 23:08:13 +01:00
std::vector<char> data;
2016-01-04 13:17:43 +00:00
};
2018-05-22 23:08:13 +01:00
elf::Header header;
std::vector<std::unique_ptr<Section>> sections;
};
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
static int
getSectionIndex(ElfFile &file, const char *name)
2016-01-04 13:17:43 +00:00
{
2018-05-22 23:08:13 +01:00
int index = 0;
for (const auto &section : file.sections) {
if (section->name == name) {
return index;
}
2018-05-22 23:08:13 +01:00
++index;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
return -1;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
static ElfFile::Section *
getSectionByType(ElfFile &file, elf::SectionType type)
2016-01-04 13:17:43 +00:00
{
2018-05-22 23:08:13 +01:00
for (const auto &section : file.sections) {
if (section->header.type == type) {
return section.get();
2016-01-04 13:17:43 +00:00
}
}
return nullptr;
}
2018-05-22 23:08:13 +01:00
static ElfFile::Section *
getSectionByName(ElfFile &file, const char *name)
{
auto index = getSectionIndex(file, name);
if (index == -1) {
return nullptr;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
return file.sections[index].get();
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
// https://stackoverflow.com/a/16569749
template<class TContainer>
bool begins_with(const TContainer& input, const TContainer& match)
2016-01-04 13:17:43 +00:00
{
2018-05-22 23:08:13 +01:00
return input.size() >= match.size()
&& std::equal(match.begin(), match.end(), input.begin());
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Read the .elf file generated by compiler.
*/
2016-01-04 13:17:43 +00:00
static bool
2018-05-22 23:08:13 +01:00
readElf(ElfFile &file, const std::string &filename)
2016-01-04 13:17:43 +00:00
{
std::ifstream in { filename, std::ifstream::binary };
if (!in.is_open()) {
2018-05-22 23:08:13 +01:00
fmt::print("Could not open {} for reading", filename);
2016-01-04 13:17:43 +00:00
return false;
}
// Read header
2018-05-22 23:08:13 +01:00
in.read(reinterpret_cast<char *>(&file.header), sizeof(elf::Header));
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (file.header.magic != elf::HeaderMagic) {
fmt::print("Invalid ELF magic header {:08X}", elf::HeaderMagic);
2016-01-04 13:17:43 +00:00
return false;
}
2018-05-22 23:08:13 +01:00
if (file.header.fileClass != elf::ELFCLASS32) {
fmt::print("Unexpected ELF file class {}, expected {}", file.header.fileClass, elf::ELFCLASS32);
2016-01-04 13:17:43 +00:00
return false;
}
2018-05-22 23:08:13 +01:00
if (file.header.encoding != elf::ELFDATA2MSB) {
fmt::print("Unexpected ELF encoding {}, expected {}", file.header.encoding, elf::ELFDATA2MSB);
2016-01-04 13:17:43 +00:00
return false;
}
2018-05-22 23:08:13 +01:00
if (file.header.machine != elf::EM_PPC) {
fmt::print("Unexpected ELF machine type {}, expected {}", file.header.machine, elf::EM_PPC);
2016-01-04 13:17:43 +00:00
return false;
}
2018-05-22 23:08:13 +01:00
if (file.header.elfVersion != elf::EV_CURRENT) {
fmt::print("Unexpected ELF version {}, expected {}", file.header.elfVersion, elf::EV_CURRENT);
2016-01-04 13:17:43 +00:00
return false;
}
// Read section headers and data
2018-05-22 23:08:13 +01:00
in.seekg(static_cast<size_t>(file.header.shoff));
for (auto i = 0u; i < file.header.shnum; ++i) {
file.sections.emplace_back(std::make_unique<ElfFile::Section>());
auto &section = *file.sections.back();
2016-01-04 13:17:43 +00:00
in.read(reinterpret_cast<char *>(&section.header), sizeof(elf::SectionHeader));
if (!section.header.size || section.header.type == elf::SHT_NOBITS) {
continue;
}
auto pos = in.tellg();
in.seekg(static_cast<size_t>(section.header.offset));
section.data.resize(section.header.size);
in.read(section.data.data(), section.data.size());
in.seekg(pos);
}
2018-05-22 23:08:13 +01:00
// Set section header names
auto shStrTab = file.sections[file.header.shstrndx]->data.data();
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
for (auto &section : file.sections) {
section->name = shStrTab + section->header.name;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Our linker script sometimes converts .bss from NOBITS to PROGBITS.
*/
static bool
fixBssNoBits(ElfFile &file)
{
auto section = getSectionByName(file, ".bss");
if (!section) {
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Ensure there is actually all 0 in the .bss section
for (const auto c : section->data) {
if (c) {
return false;
}
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Set type back to NOBITS
section->header.type = elf::SHT_NOBITS;
2018-05-23 11:25:49 +01:00
section->header.offset = 0u;
2018-05-22 23:08:13 +01:00
section->data.clear();
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Reorder sections index.
*
* Expected order:
* NULL section
* >.syscall >.text
* > .fexports
* > .rodata > .data > .module_id > .bss
* > .rela.fexports > .rela.text > .rela.rodata > .rela.data
* > {.fimport, .dimport }
* > .symtab > .strtab > .shstrtab
*/
static bool
reorderSectionIndex(ElfFile &file)
{
// Create a map of new index -> old index
std::vector<std::size_t> sectionMap;
sectionMap.push_back(0);
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".syscall"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".text"); index != -1) {
sectionMap.push_back(index);
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".fexports"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".rodata"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".data"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".module_id"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".bss"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".rela.fexports"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".rela.text"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".rela.rodata"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".rela.data"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// All .fimport_ and .dimport_
for (auto index = 0u; index < file.sections.size(); ++index) {
const auto &section = file.sections[index];
if (begins_with<std::string>(section->name, ".fimport_") ||
begins_with<std::string>(section->name, ".dimport_")) {
sectionMap.push_back(index);
2016-01-04 13:17:43 +00:00
}
}
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".symtab"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".strtab"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto index = getSectionIndex(file, ".shstrtab"); index != -1) {
sectionMap.push_back(index);
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (sectionMap.size() != file.sections.size()) {
fmt::print("Invalid section in elf file");
return false;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Apply the new ordering
std::vector<std::unique_ptr<ElfFile::Section>> newSections;
for (auto idx : sectionMap) {
newSections.emplace_back(std::move(file.sections[idx]));
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
file.sections = std::move(newSections);
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Now generate a reverse map, old index -> new index
2018-05-23 11:25:49 +01:00
std::vector<uint16_t> mapOldToNew;
2018-05-22 23:08:13 +01:00
mapOldToNew.resize(file.sections.size());
for (auto i = 0u; i < sectionMap.size(); ++i) {
2018-05-23 11:25:49 +01:00
mapOldToNew[sectionMap[i]] = static_cast<uint16_t>(i);
2018-05-22 23:08:13 +01:00
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Map file header.shstrndx
file.header.shstrndx = mapOldToNew[file.header.shstrndx];
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Map section header.link
for (auto &section : file.sections) {
section->header.link = mapOldToNew[section->header.link];
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
// Map relocation sections header.info
for (auto &section : file.sections) {
if (section->header.type != elf::SHT_RELA) {
2016-01-04 13:17:43 +00:00
continue;
}
2018-05-22 23:08:13 +01:00
section->header.info = mapOldToNew[section->header.info];
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Map symbol.shndx
for (auto &section : file.sections) {
if (section->header.type != elf::SHT_SYMTAB) {
2016-01-04 13:17:43 +00:00
continue;
}
2018-05-22 23:08:13 +01:00
auto symbols = reinterpret_cast<elf::Symbol *>(section->data.data());
auto numSymbols = section->data.size() / sizeof(elf::Symbol);
for (auto i = 0u; i < numSymbols; ++i) {
2018-05-23 11:10:23 +01:00
auto shndx = symbols[i].shndx;
2018-05-22 23:08:13 +01:00
if (shndx < elf::SHN_LORESERVE) {
2018-05-23 11:10:23 +01:00
symbols[i].shndx = mapOldToNew[shndx];
2016-01-04 13:17:43 +00:00
}
}
}
2018-05-22 23:08:13 +01:00
return true;
}
2018-05-22 23:08:13 +01:00
/**
* Generate SHT_RPL_FILEINFO section.
*/
static bool
generateFileInfoSection(ElfFile &file)
{
elf::RplFileInfo info;
2018-05-23 11:25:49 +01:00
info.version = 0xCAFE0402u;
info.textSize = 0u;
info.textAlign = 32u;
info.dataSize = 0u;
info.dataAlign = 4096u;
info.loadSize = 0u;
info.loadAlign = 4u;
info.tempSize = 0u;
info.trampAdjust = 0u;
info.trampAddition = 0u;
info.sdaBase = 0u;
info.sda2Base = 0u;
info.stackSize = 0x10000u;
info.heapSize = 0x8000u;
info.filename = 0u;
2018-05-22 23:08:13 +01:00
info.flags = elf::RPL_IS_RPX; // TODO: Add .rpl support
2018-05-23 11:25:49 +01:00
info.minVersion = 0x5078u;
2018-05-22 23:08:13 +01:00
info.compressionLevel = -1;
2018-05-23 11:25:49 +01:00
info.fileInfoPad = 0u;
info.cafeSdkVersion = 0x51BAu;
info.cafeSdkRevision = 0xCCD1u;
info.tlsAlignShift = uint16_t { 0u };
info.tlsModuleIndex = uint16_t { 0u };
info.runtimeFileInfoSize = 0u;
info.tagOffset = 0u;
2018-05-22 23:08:13 +01:00
// Count file info textSize, dataSize, loadSize
for (auto &section : file.sections) {
2018-05-23 11:25:49 +01:00
auto size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
if (section->header.type == elf::SHT_NOBITS) {
size = section->header.size;
}
2018-05-22 23:08:13 +01:00
if (section->header.addr >= CodeBaseAddress &&
section->header.addr < DataBaseAddress) {
auto val = section->header.addr + section->header.size - CodeBaseAddress;
if (val > info.textSize) {
info.textSize = val;
}
2018-05-22 23:08:13 +01:00
} else if (section->header.addr >= DataBaseAddress &&
section->header.addr < LoadBaseAddress) {
auto val = section->header.addr + section->header.size - DataBaseAddress;
if (val > info.dataSize) {
info.dataSize = val;
}
2018-05-22 23:08:13 +01:00
} else if (section->header.addr >= LoadBaseAddress) {
auto val = section->header.addr + section->header.size - LoadBaseAddress;
if (val > info.loadSize) {
info.loadSize = val;
}
2018-05-22 23:08:13 +01:00
} else if (section->header.addr == 0 &&
section->header.type != elf::SHT_RPL_CRCS &&
section->header.type != elf::SHT_RPL_FILEINFO) {
info.tempSize += (size + 128);
}
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
auto section = std::make_unique<ElfFile::Section>();
2018-05-23 11:25:49 +01:00
section->header.name = 0u;
2018-05-22 23:08:13 +01:00
section->header.type = elf::SHT_RPL_FILEINFO;
2018-05-23 11:25:49 +01:00
section->header.flags = 0u;
section->header.addr = 0u;
section->header.offset = 0u;
section->header.size = 0u;
section->header.link = 0u;
section->header.info = 0u;
section->header.addralign = 4u;
section->header.entsize = 0u;
2018-05-22 23:08:13 +01:00
section->data.insert(section->data.end(),
reinterpret_cast<char *>(&info),
reinterpret_cast<char *>(&info + 1));
file.sections.emplace_back(std::move(section));
2016-01-04 13:17:43 +00:00
return true;
}
2018-05-22 23:08:13 +01:00
/**
* Generate SHT_RPL_CRCS section.
*/
static bool
generateCrcSection(ElfFile &file)
2016-01-04 13:17:43 +00:00
{
2018-05-23 15:36:07 +01:00
std::vector<be_val<uint32_t>> crcs;
2018-05-22 23:08:13 +01:00
for (auto &section : file.sections) {
auto crc = uint32_t { 0u };
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (section->data.size()) {
crc = crc32(0, Z_NULL, 0);
crc = crc32(crc, reinterpret_cast<Bytef *>(section->data.data()), section->data.size());
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
2018-05-23 11:10:23 +01:00
crcs.push_back(crc);
2016-01-04 13:17:43 +00:00
}
2018-05-23 15:36:07 +01:00
// Insert a 0 crc for this section
crcs.insert(crcs.end() - 1, 0);
2018-05-22 23:08:13 +01:00
auto section = std::make_unique<ElfFile::Section>();
2018-05-23 11:25:49 +01:00
section->header.name = 0u;
2018-05-22 23:08:13 +01:00
section->header.type = elf::SHT_RPL_CRCS;
2018-05-23 11:25:49 +01:00
section->header.flags = 0u;
section->header.addr = 0u;
section->header.offset = 0u;
section->header.size = 0u;
section->header.link = 0u;
section->header.info = 0u;
section->header.addralign = 4u;
section->header.entsize = 4u;
2018-05-22 23:08:13 +01:00
section->data.insert(section->data.end(),
reinterpret_cast<char *>(crcs.data()),
reinterpret_cast<char *>(crcs.data() + crcs.size()));
// Insert before FILEINFO
file.sections.insert(file.sections.end() - 1, std::move(section));
return true;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
static bool
getSymbol(ElfFile::Section &section, size_t index, elf::Symbol &symbol)
{
auto symbols = reinterpret_cast<elf::Symbol *>(section.data.data());
auto numSymbols = section.data.size() / sizeof(elf::Symbol);
if (index >= numSymbols) {
return false;
2016-01-04 13:17:43 +00:00
}
2018-05-23 11:10:23 +01:00
symbol = symbols[index];
2018-05-22 23:08:13 +01:00
return true;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
/**
* Fix relocations.
*
* The Wii U does not support every type of relocation.
*/
2016-01-04 13:17:43 +00:00
static bool
2018-05-22 23:08:13 +01:00
fixRelocations(ElfFile &file)
2016-01-04 13:17:43 +00:00
{
2018-05-22 23:08:13 +01:00
std::set<unsigned int> unsupportedTypes;
auto result = true;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
for (auto &section : file.sections) {
std::vector<elf::Rela> newRelocations;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (section->header.type != elf::SHT_RELA) {
continue;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
// Clear flags
2018-05-23 11:25:49 +01:00
section->header.flags = 0u;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
auto &symbolSection = file.sections[section->header.link];
auto &targetSection = file.sections[section->header.info];
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
auto rels = reinterpret_cast<elf::Rela *>(section->data.data());
auto numRels = section->data.size() / sizeof(elf::Rela);
for (auto i = 0u; i < numRels; ++i) {
2018-05-23 11:10:23 +01:00
auto info = rels[i].info;
auto addend = rels[i].addend;
auto offset = rels[i].offset;
2018-05-22 23:08:13 +01:00
auto index = info >> 8;
auto type = info & 0xFF;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
switch (type) {
case elf::R_PPC_NONE:
case elf::R_PPC_ADDR32:
case elf::R_PPC_ADDR16_LO:
case elf::R_PPC_ADDR16_HI:
case elf::R_PPC_ADDR16_HA:
case elf::R_PPC_REL24:
case elf::R_PPC_REL14:
case elf::R_PPC_DTPMOD32:
case elf::R_PPC_DTPREL32:
case elf::R_PPC_EMB_SDA21:
case elf::R_PPC_EMB_RELSDA:
case elf::R_PPC_DIAB_SDA21_LO:
case elf::R_PPC_DIAB_SDA21_HI:
case elf::R_PPC_DIAB_SDA21_HA:
case elf::R_PPC_DIAB_RELSDA_LO:
case elf::R_PPC_DIAB_RELSDA_HI:
case elf::R_PPC_DIAB_RELSDA_HA:
// All valid relocations on Wii U, do nothing
break;
/*
* Convert a R_PPC_REL32 into two GHS_REL16
*/
case elf::R_PPC_REL32:
{
elf::Symbol symbol;
if (!getSymbol(*symbolSection, index, symbol)) {
fmt::print("ERROR: Could not find symbol {} for fixing a R_PPC_REL32 relocation", index);
result = false;
} else {
newRelocations.emplace_back();
auto &newRel = newRelocations.back();
// Modify current relocation to R_PPC_GHS_REL16_LO
2018-05-23 11:10:23 +01:00
rels[i].info = (index << 8) | elf::R_PPC_GHS_REL16_LO;
rels[i].addend = addend;
rels[i].offset = offset;
2018-05-22 23:08:13 +01:00
// Create a R_PPC_GHS_REL16_HI
2018-05-23 11:10:23 +01:00
newRel.info = (index << 8) | elf::R_PPC_GHS_REL16_HI;
newRel.addend = addend + 2;
newRel.offset = offset + 2;
2018-05-22 23:08:13 +01:00
}
2016-01-04 13:17:43 +00:00
break;
}
2018-05-22 23:08:13 +01:00
default:
// Only print error once per type
if (!unsupportedTypes.count(type)) {
fmt::print("ERROR: Unsupported relocation type {}", type);
unsupportedTypes.insert(type);
}
}
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
section->data.insert(section->data.end(),
reinterpret_cast<char *>(newRelocations.data()),
reinterpret_cast<char *>(newRelocations.data() + newRelocations.size()));
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
return result && unsupportedTypes.size() == 0;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Fix file header to look like an RPL file!
*/
static bool
fixFileHeader(ElfFile &file)
{
file.header.magic = elf::HeaderMagic;
2018-05-23 11:25:49 +01:00
file.header.fileClass = uint8_t { 1 };
2018-05-22 23:08:13 +01:00
file.header.encoding = elf::ELFDATA2MSB;
file.header.elfVersion = elf::EV_CURRENT;
file.header.abi = elf::EABI_CAFE;
memset(&file.header.pad, 0, 7);
2018-05-23 11:25:49 +01:00
file.header.type = uint16_t { 0xFE01 };
2018-05-22 23:08:13 +01:00
file.header.machine = elf::EM_PPC;
2018-05-23 11:25:49 +01:00
file.header.version = 1u;
file.header.flags = 0u;
file.header.phoff = 0u;
file.header.phentsize = uint16_t { 0 };
file.header.phnum = uint16_t { 0 };
file.header.shoff = align_up(static_cast<uint32_t>(sizeof(elf::Header)), 64);
file.header.shnum = static_cast<uint16_t>(file.sections.size());
file.header.shentsize = static_cast<uint16_t>(sizeof(elf::SectionHeader));
file.header.ehsize = static_cast<uint16_t>(sizeof(elf::Header));
file.header.shstrndx = static_cast<uint16_t>(getSectionIndex(file, ".shstrtab"));
2018-05-22 23:08:13 +01:00
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Fix the .addralign field for sections.
*/
static bool
fixSectionAlign(ElfFile &file)
{
for (auto &section : file.sections) {
if (section->header.type == elf::SHT_PROGBITS) {
2018-05-23 11:25:49 +01:00
section->header.addralign = 32u;
2018-05-22 23:08:13 +01:00
} else if (section->header.type == elf::SHT_NOBITS) {
2018-05-23 11:25:49 +01:00
section->header.addralign = 64u;
2018-05-22 23:08:13 +01:00
} else if (section->header.type == elf::SHT_RPL_IMPORTS) {
2018-05-23 11:25:49 +01:00
section->header.addralign = 4u;
}
}
2018-05-22 23:08:13 +01:00
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
static bool
relocateSection(ElfFile &file,
ElfFile::Section &section,
uint32_t newSectionAddress)
{
2018-05-23 11:25:49 +01:00
auto sectionSize = section.data.size() ? section.data.size() : static_cast<size_t>(section.header.size);
2018-05-22 23:08:13 +01:00
auto oldSectionAddress = section.header.addr;
auto oldSectionAddressEnd = section.header.addr + sectionSize;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Relocate symbols pointing into this section
for (auto &symSection : file.sections) {
if (symSection->header.type != elf::SectionType::SHT_SYMTAB) {
continue;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
auto symbols = reinterpret_cast<elf::Symbol *>(symSection->data.data());
auto numSymbols = symSection->data.size() / sizeof(elf::Symbol);
for (auto i = 0u; i < numSymbols; ++i) {
2018-05-23 11:10:23 +01:00
auto type = symbols[i].info & 0xf;
auto value = symbols[i].value;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// Only relocate data, func, section symbols
if (type != elf::STT_OBJECT &&
type != elf::STT_FUNC &&
type != elf::STT_SECTION) {
continue;
}
2018-05-22 23:08:13 +01:00
if (value >= oldSectionAddress && value <= oldSectionAddressEnd) {
2018-05-23 11:10:23 +01:00
symbols[i].value = (value - oldSectionAddress) + newSectionAddress;
2018-05-22 23:08:13 +01:00
}
}
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
// Relocate relocations pointing into this section
for (auto &relaSection : file.sections) {
if (relaSection->header.type != elf::SectionType::SHT_RELA) {
continue;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
auto rela = reinterpret_cast<elf::Rela *>(relaSection->data.data());
auto numRelas = relaSection->data.size() / sizeof(elf::Rela);
for (auto i = 0u; i < numRelas; ++i) {
2018-05-23 11:10:23 +01:00
auto offset = rela[i].offset;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (offset >= oldSectionAddress && offset <= oldSectionAddressEnd) {
2018-05-23 11:10:23 +01:00
rela[i].offset = (offset - oldSectionAddress) + newSectionAddress;
2018-05-22 23:08:13 +01:00
}
2016-01-04 13:17:43 +00:00
}
}
2018-05-22 23:08:13 +01:00
section.header.addr = newSectionAddress;
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Fix the loader virtual addresses.
*
* Linker script won't put symtab & strtab sections in our loader address, so
* we must fix that.
*
* Expected order:
* .fexports > .dexports > .symtab > .strtab > .shstrtab > {.fimport, .dimport}
*/
static bool
fixLoaderVirtualAddresses(ElfFile &file)
{
auto addr = LoadBaseAddress;
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
// All symbols pointing to this section require fixing
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".fexports")) {
relocateSection(file, *section,
align_up(addr, section->header.addralign));
addr += section->data.size();
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".dexports")) {
relocateSection(file, *section,
align_up(addr, section->header.addralign));
addr += section->data.size();
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".symtab")) {
relocateSection(file, *section,
align_up(addr, section->header.addralign));
section->header.flags |= elf::SHF_ALLOC;
addr += section->data.size();
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".strtab")) {
relocateSection(file, *section,
align_up(addr, section->header.addralign));
section->header.flags |= elf::SHF_ALLOC;
addr += section->data.size();
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".shstrtab")) {
relocateSection(file, *section,
align_up(addr, section->header.addralign));
section->header.flags |= elf::SHF_ALLOC;
addr += section->data.size();
}
2018-05-22 23:08:13 +01:00
for (auto &section : file.sections) {
if (section->header.type == elf::SHT_RPL_IMPORTS) {
relocateSection(file, *section,
align_up(addr, section->header.addralign));
addr += section->data.size();
}
}
2018-05-22 23:08:13 +01:00
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Calculate section file offsets.
*
* Expected order:
* RPL_CRCS > RPL_FILEINFO >
* .rodata > .data > .module_id >
* .fexports > .dexports >
* .fimports > .dimports >
* .symtab > .strtab > .shstrtab >
* .syscall > .text >
* .rela.fexports > .rela.text > .rela.rodata > .rela.data
*/
static bool
calculateSectionOffsets(ElfFile &file)
{
auto offset = file.header.shoff;
2018-05-23 11:25:49 +01:00
offset += align_up(static_cast<uint32_t>(file.sections.size() * sizeof(elf::SectionHeader)), 64);
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByType(file, elf::SHT_RPL_CRCS)) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByType(file, elf::SHT_RPL_FILEINFO)) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".rodata")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".data")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
offset += section->header.size;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".module_id")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".fexports")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".dexports")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2018-05-22 23:08:13 +01:00
for (auto &section : file.sections) {
if (section->header.type == elf::SHT_RPL_IMPORTS) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".symtab")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".strtab")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".shstrtab")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".syscall")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".text")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".rela.fexports")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".rela.text")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".rela.rodata")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
if (auto section = getSectionByName(file, ".rela.data")) {
section->header.offset = offset;
2018-05-23 11:25:49 +01:00
section->header.size = static_cast<uint32_t>(section->data.size());
2018-05-22 23:08:13 +01:00
offset += section->header.size;
}
2018-05-22 23:08:13 +01:00
return true;
}
2016-01-04 13:17:43 +00:00
2018-05-22 23:08:13 +01:00
/**
* Write out the final RPL.
*/
static bool
writeRpl(ElfFile &file, const std::string &filename)
{
auto shoff = file.header.shoff;
// Write the file out
std::ofstream out { filename, std::ofstream::binary };
2016-01-04 13:17:43 +00:00
if (!out.is_open()) {
2018-05-22 23:08:13 +01:00
fmt::print("Could not open {} for writing", filename);
2016-01-04 13:17:43 +00:00
return false;
}
2018-05-22 23:08:13 +01:00
// Write file header
out.seekp(0, std::ios::beg);
out.write(reinterpret_cast<const char *>(&file.header), sizeof(elf::Header));
2016-01-04 13:17:43 +00:00
// Write section headers
2018-05-22 23:08:13 +01:00
out.seekp(shoff, std::ios::beg);
for (const auto &section : file.sections) {
out.write(reinterpret_cast<const char *>(&section->header), sizeof(elf::SectionHeader));
2016-01-04 13:17:43 +00:00
}
2018-05-22 23:08:13 +01:00
// Write sections
for (const auto &section : file.sections) {
if (section->data.size()) {
2018-05-23 11:10:23 +01:00
out.seekp(section->header.offset, std::ios::beg);
2016-01-04 13:17:43 +00:00
out.write(section->data.data(), section->data.size());
}
}
return true;
}
2018-05-22 23:08:13 +01:00
int main(int argc, const char **argv)
2016-01-04 13:17:43 +00:00
{
if (argc < 3) {
2018-05-22 23:08:13 +01:00
fmt::print("Usage: {} <src elf> <dst rpl>", argv[0]);
2016-01-04 13:17:43 +00:00
return -1;
}
2018-05-22 23:08:13 +01:00
auto src = std::string { argv[1] };
auto dst = std::string { argv[2] };
// Read elf into memory object!
2016-01-04 13:17:43 +00:00
ElfFile elf;
2018-05-22 23:08:13 +01:00
if (!readElf(elf, src)) {
fmt::print("ERROR: readElf failed");
return -1;
}
if (!fixBssNoBits(elf)) {
fmt::print("ERROR: fixBssNoBits failed");
return -1;
}
if (!reorderSectionIndex(elf)) {
fmt::print("ERROR: reorderSectionIndex failed");
return -1;
}
if (!fixRelocations(elf)) {
fmt::print("ERROR: fixRelocations failed");
return -1;
}
if (!fixSectionAlign(elf)) {
fmt::print("ERROR: fixSectionAlign failed");
return -1;
}
if (!fixLoaderVirtualAddresses(elf)) {
fmt::print("ERROR: fixLoaderVirtualAddresses failed");
return -1;
}
if (!generateFileInfoSection(elf)) {
fmt::print("ERROR: generateFileInfoSection failed");
return -1;
}
if (!generateCrcSection(elf)) {
fmt::print("ERROR: generateCrcSection failed");
return -1;
}
if (!fixFileHeader(elf)) {
fmt::print("ERROR: fixFileHeader failed");
return -1;
}
if (!calculateSectionOffsets(elf)) {
fmt::print("ERROR: calculateSectionOffsets failed");
2016-01-04 13:17:43 +00:00
return -1;
}
2018-05-22 23:08:13 +01:00
if (!writeRpl(elf, dst)) {
fmt::print("ERROR: writeRpl failed");
2016-01-04 13:17:43 +00:00
return -1;
}
return 0;
}