mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-13 04:30:23 +00:00
3e19a9ee9f
Current PPC64 RuntimeDyld code to handle TOC relocations has two problems: - With recent linkers, in addition to the relocations that implicitly refer to the TOC base (R_PPC64_TOC*), you can now also use the .TOC. magic symbol with any other relocation to refer to the TOC base explicitly. This isn't currently used much in ELFv1 code (although it could be), but it is essential in ELFv2 code. - In a complex JIT environment with multiple modules, each module may have its own .toc section, and TOC relocations in one module must refer to *its own* TOC section. The current findPPC64TOC implementation does not correctly implement this; in fact, it will always return the address of the first TOC section it finds anywhere. (Note that at the time findPPC64TOC is called, we don't even *know* which module the relocation originally resided in, so it is not even possible to fix this routine as-is.) This commit fixes both problems by handling TOC relocations earlier, in processRelocationRef. To do this, I've removed the findPPC64TOC routine and replaced it by a new routine findPPC64TOCSection, which works analogously to findOPDEntrySection in scanning the sections of the ObjImage provided by its caller, processRelocationRef. This solves the issue of finding the correct TOC section associated with the current module. This makes it straightforward to implement both R_PPC64_TOC relocations, and relocations explicitly refering to the .TOC. symbol, directly in processRelocationRef. There is now a new problem in implementing the R_PPC64_TOC16* relocations, because those can now in theory involve *three* different sections: the relocation may be applied in section A, refer explicitly to a symbol in section B, and refer implicitly to the TOC section C. The final processing of the relocation thus may only happen after all three of these sections have been assigned final addresses. There is currently no obvious means to implement this in its general form with the common-code RuntimeDyld infrastructure. Fortunately, ppc64 code usually makes no use of this most general form; in fact, TOC16 relocations are only ever generated by LLVM for symbols residing themselves in the TOC, which means "section B" == "section C" in the above terminology. This special case can easily be handled with the current infrastructure, and that is what this patch does. [ Unhandled cases result in an explicit error, unlike the current code which silently returns the wrong TOC base address ... ] This patch makes the JIT work on both BE and LE (ELFv2 requires additional patches, of course), and allowed me to successfully run complex JIT scenarios (via mesa/llvmpipe). Reviewed by Hal Finkel. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@211885 91177308-0d34-0410-b5e6-96231b3b80d8
130 lines
4.9 KiB
C++
130 lines
4.9 KiB
C++
//===-- RuntimeDyldELF.h - Run-time dynamic linker for MC-JIT ---*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// ELF support for MC-JIT runtime dynamic linker.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_RUNTIME_DYLD_ELF_H
|
|
#define LLVM_RUNTIME_DYLD_ELF_H
|
|
|
|
#include "RuntimeDyldImpl.h"
|
|
#include "llvm/ADT/DenseMap.h"
|
|
|
|
using namespace llvm;
|
|
|
|
namespace llvm {
|
|
namespace {
|
|
// Helper for extensive error checking in debug builds.
|
|
std::error_code Check(std::error_code Err) {
|
|
if (Err) {
|
|
report_fatal_error(Err.message());
|
|
}
|
|
return Err;
|
|
}
|
|
} // end anonymous namespace
|
|
|
|
class RuntimeDyldELF : public RuntimeDyldImpl {
|
|
void resolveRelocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint64_t Value, uint32_t Type, int64_t Addend,
|
|
uint64_t SymOffset = 0);
|
|
|
|
void resolveX86_64Relocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint64_t Value, uint32_t Type, int64_t Addend,
|
|
uint64_t SymOffset);
|
|
|
|
void resolveX86Relocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint32_t Value, uint32_t Type, int32_t Addend);
|
|
|
|
void resolveAArch64Relocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint64_t Value, uint32_t Type, int64_t Addend);
|
|
|
|
void resolveARMRelocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint32_t Value, uint32_t Type, int32_t Addend);
|
|
|
|
void resolveMIPSRelocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint32_t Value, uint32_t Type, int32_t Addend);
|
|
|
|
void resolvePPC64Relocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint64_t Value, uint32_t Type, int64_t Addend);
|
|
|
|
void resolveSystemZRelocation(const SectionEntry &Section, uint64_t Offset,
|
|
uint64_t Value, uint32_t Type, int64_t Addend);
|
|
|
|
unsigned getMaxStubSize() override {
|
|
if (Arch == Triple::aarch64 || Arch == Triple::arm64 ||
|
|
Arch == Triple::aarch64_be || Arch == Triple::arm64_be)
|
|
return 20; // movz; movk; movk; movk; br
|
|
if (Arch == Triple::arm || Arch == Triple::thumb)
|
|
return 8; // 32-bit instruction and 32-bit address
|
|
else if (Arch == Triple::mipsel || Arch == Triple::mips)
|
|
return 16;
|
|
else if (Arch == Triple::ppc64 || Arch == Triple::ppc64le)
|
|
return 44;
|
|
else if (Arch == Triple::x86_64)
|
|
return 6; // 2-byte jmp instruction + 32-bit relative address
|
|
else if (Arch == Triple::systemz)
|
|
return 16;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
unsigned getStubAlignment() override {
|
|
if (Arch == Triple::systemz)
|
|
return 8;
|
|
else
|
|
return 1;
|
|
}
|
|
|
|
void findPPC64TOCSection(ObjectImage &Obj, ObjSectionToIDMap &LocalSections,
|
|
RelocationValueRef &Rel);
|
|
void findOPDEntrySection(ObjectImage &Obj, ObjSectionToIDMap &LocalSections,
|
|
RelocationValueRef &Rel);
|
|
|
|
uint64_t findGOTEntry(uint64_t LoadAddr, uint64_t Offset);
|
|
size_t getGOTEntrySize();
|
|
|
|
void updateGOTEntries(StringRef Name, uint64_t Addr) override;
|
|
|
|
// Relocation entries for symbols whose position-independent offset is
|
|
// updated in a global offset table.
|
|
typedef SmallVector<RelocationValueRef, 2> GOTRelocations;
|
|
GOTRelocations GOTEntries; // List of entries requiring finalization.
|
|
SmallVector<std::pair<SID, GOTRelocations>, 8> GOTs; // Allocated tables.
|
|
|
|
// When a module is loaded we save the SectionID of the EH frame section
|
|
// in a table until we receive a request to register all unregistered
|
|
// EH frame sections with the memory manager.
|
|
SmallVector<SID, 2> UnregisteredEHFrameSections;
|
|
SmallVector<SID, 2> RegisteredEHFrameSections;
|
|
|
|
public:
|
|
RuntimeDyldELF(RTDyldMemoryManager *mm) : RuntimeDyldImpl(mm) {}
|
|
|
|
void resolveRelocation(const RelocationEntry &RE, uint64_t Value) override;
|
|
relocation_iterator
|
|
processRelocationRef(unsigned SectionID, relocation_iterator RelI,
|
|
ObjectImage &Obj, ObjSectionToIDMap &ObjSectionToID,
|
|
const SymbolTableMap &Symbols, StubMap &Stubs) override;
|
|
bool isCompatibleFormat(const ObjectBuffer *Buffer) const override;
|
|
bool isCompatibleFile(const object::ObjectFile *Buffer) const override;
|
|
void registerEHFrames() override;
|
|
void deregisterEHFrames() override;
|
|
void finalizeLoad(ObjectImage &ObjImg,
|
|
ObjSectionToIDMap &SectionMap) override;
|
|
virtual ~RuntimeDyldELF();
|
|
|
|
static ObjectImage *createObjectImage(ObjectBuffer *InputBuffer);
|
|
static ObjectImage *createObjectImageFromFile(std::unique_ptr<object::ObjectFile> Obj);
|
|
};
|
|
|
|
} // end namespace llvm
|
|
|
|
#endif
|