mirror of
https://github.com/zeldaret/oot.git
synced 2024-12-02 07:46:01 +00:00
451b24f79b
* Silence a number of GCC warnings * Remove * Suggested changes * Format * Fix comment in en_go2
121 lines
5.3 KiB
C
121 lines
5.3 KiB
C
#include "global.h"
|
|
|
|
void Overlay_Relocate(void* allocatedVRamAddress, OverlayRelocationSection* overlayInfo, void* vRamStart) {
|
|
u32 sections[4];
|
|
u32 relocatedValue;
|
|
u32 dbg;
|
|
u32 relocOffset;
|
|
u32 relocData;
|
|
uintptr_t unrelocatedAddress;
|
|
u32 i;
|
|
u32* relocDataP;
|
|
u32* luiRefs[32];
|
|
u32 luiVals[32];
|
|
uintptr_t relocatedAddress;
|
|
u32 reloc;
|
|
u32* luiInstRef;
|
|
uintptr_t allocu32 = (uintptr_t)allocatedVRamAddress;
|
|
u32* regValP;
|
|
u32 isLoNeg;
|
|
s32 pad;
|
|
|
|
relocOffset = 0;
|
|
relocatedValue = 0;
|
|
unrelocatedAddress = 0;
|
|
relocatedAddress = 0;
|
|
|
|
if (gOverlayLogSeverity >= 3) {
|
|
osSyncPrintf("DoRelocation(%08x, %08x, %08x)\n", allocatedVRamAddress, overlayInfo, vRamStart);
|
|
osSyncPrintf("text=%08x, data=%08x, rodata=%08x, bss=%08x\n", overlayInfo->textSize, overlayInfo->dataSize,
|
|
overlayInfo->rodataSize, overlayInfo->bssSize);
|
|
}
|
|
|
|
sections[0] = 0;
|
|
sections[1] = allocu32;
|
|
sections[2] = allocu32 + overlayInfo->textSize;
|
|
sections[3] = sections[2] + overlayInfo->dataSize;
|
|
|
|
for (i = 0; i < overlayInfo->nRelocations; i++) {
|
|
reloc = overlayInfo->relocations[i];
|
|
relocDataP = (u32*)(sections[RELOC_SECTION(reloc)] + RELOC_OFFSET(reloc));
|
|
relocData = *relocDataP;
|
|
|
|
switch (RELOC_TYPE_MASK(reloc)) {
|
|
case R_MIPS_32 << RELOC_TYPE_SHIFT:
|
|
// Handles 32-bit address relocation, used for things such as jump tables and pointers in data.
|
|
// Just relocate the full address.
|
|
|
|
// Check address is valid for relocation
|
|
if ((*relocDataP & 0x0F000000) == 0) {
|
|
relocOffset = *relocDataP - (uintptr_t)vRamStart;
|
|
relocatedValue = relocOffset + allocu32;
|
|
relocatedAddress = relocatedValue;
|
|
unrelocatedAddress = relocData;
|
|
*relocDataP = relocatedAddress;
|
|
}
|
|
break;
|
|
|
|
case R_MIPS_26 << RELOC_TYPE_SHIFT:
|
|
// Handles 26-bit address relocation, used for jumps and jals.
|
|
// Extract the address from the target field of the J-type MIPS instruction.
|
|
// Relocate the address and update the instruction.
|
|
|
|
unrelocatedAddress = PHYS_TO_K0((*relocDataP & 0x03FFFFFF) << 2);
|
|
relocOffset = unrelocatedAddress - (uintptr_t)vRamStart;
|
|
relocatedValue = (*relocDataP & 0xFC000000) | (((allocu32 + relocOffset) & 0x0FFFFFFF) >> 2);
|
|
relocatedAddress = PHYS_TO_K0((relocatedValue & 0x03FFFFFF) << 2);
|
|
*relocDataP = relocatedValue;
|
|
break;
|
|
|
|
case R_MIPS_HI16 << RELOC_TYPE_SHIFT:
|
|
// Handles relocation for a hi/lo pair, part 1.
|
|
// Store the reference to the LUI instruction (hi) using the `rt` register of the instruction.
|
|
// This will be updated later in the `R_MIPS_LO16` section.
|
|
|
|
luiRefs[(*relocDataP >> 0x10) & 0x1F] = relocDataP;
|
|
luiVals[(*relocDataP >> 0x10) & 0x1F] = *relocDataP;
|
|
break;
|
|
|
|
case R_MIPS_LO16 << RELOC_TYPE_SHIFT:
|
|
// Handles relocation for a hi/lo pair, part 2.
|
|
// Grab the stored LUI (hi) from the `R_MIPS_HI16` section using the `rs` register of the instruction.
|
|
// The full address is calculated, relocated, and then used to update both the LUI and lo instructions.
|
|
// If the lo part is negative, add 1 to the LUI value.
|
|
// Note: The lo instruction is assumed to have a signed immediate.
|
|
|
|
luiInstRef = luiRefs[(*relocDataP >> 0x15) & 0x1F];
|
|
regValP = &luiVals[(*relocDataP >> 0x15) & 0x1F];
|
|
|
|
// Check address is valid for relocation
|
|
if ((((*regValP << 0x10) + (s16)*relocDataP) & 0x0F000000) == 0) {
|
|
relocOffset = ((*regValP << 0x10) + (s16)*relocDataP) - (uintptr_t)vRamStart;
|
|
isLoNeg = (((relocOffset + allocu32) & 0x8000) ? 1 : 0);
|
|
unrelocatedAddress = (*luiInstRef << 0x10) + (s16)relocData;
|
|
*luiInstRef =
|
|
(*luiInstRef & 0xFFFF0000) | ((((relocOffset + allocu32) >> 0x10) & 0xFFFF) + isLoNeg);
|
|
relocatedValue = (*relocDataP & 0xFFFF0000) | ((relocOffset + allocu32) & 0xFFFF);
|
|
|
|
relocatedAddress = (*luiInstRef << 0x10) + (s16)relocatedValue;
|
|
*relocDataP = relocatedValue;
|
|
}
|
|
break;
|
|
}
|
|
|
|
dbg = 0x10;
|
|
switch (RELOC_TYPE_MASK(reloc)) {
|
|
case R_MIPS_32 << RELOC_TYPE_SHIFT:
|
|
dbg = 0x16;
|
|
FALLTHROUGH;
|
|
case R_MIPS_26 << RELOC_TYPE_SHIFT:
|
|
dbg += 0xA;
|
|
FALLTHROUGH;
|
|
case R_MIPS_LO16 << RELOC_TYPE_SHIFT:
|
|
if (gOverlayLogSeverity >= 3) {
|
|
osSyncPrintf("%02d %08x %08x %08x ", dbg, relocDataP, relocatedValue, relocatedAddress);
|
|
osSyncPrintf(" %08x %08x %08x %08x\n", (uintptr_t)relocDataP + (uintptr_t)vRamStart - allocu32,
|
|
relocData, unrelocatedAddress, relocOffset);
|
|
}
|
|
// Adding a break prevents matching
|
|
}
|
|
}
|
|
}
|