blob: f9c1d4db30b63bd9ab9fcdd757e4a7c072ab5630 [file] [log] [blame]
Eli Bendersky058d6472012-01-22 07:05:02 +00001//===-- RuntimeDyldELF.h - Run-time dynamic linker for MC-JIT ---*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// ELF support for MC-JIT runtime dynamic linker.
11//
12//===----------------------------------------------------------------------===//
13
Benjamin Kramera7c40ef2014-08-13 16:26:38 +000014#ifndef LLVM_LIB_EXECUTIONENGINE_RUNTIMEDYLD_RUNTIMEDYLDELF_H
15#define LLVM_LIB_EXECUTIONENGINE_RUNTIMEDYLD_RUNTIMEDYLDELF_H
Eli Bendersky058d6472012-01-22 07:05:02 +000016
17#include "RuntimeDyldImpl.h"
Andrew Kaylor480dcb32013-10-05 01:52:09 +000018#include "llvm/ADT/DenseMap.h"
Eli Bendersky058d6472012-01-22 07:05:02 +000019
20using namespace llvm;
21
Eli Bendersky058d6472012-01-22 07:05:02 +000022namespace llvm {
Tim Northover94bc73d2012-10-29 10:47:04 +000023namespace {
Juergen Ributzka7608dc02014-03-21 20:28:42 +000024// Helper for extensive error checking in debug builds.
Rafael Espindolabff5d0d2014-06-13 01:25:41 +000025std::error_code Check(std::error_code Err) {
Juergen Ributzka7608dc02014-03-21 20:28:42 +000026 if (Err) {
27 report_fatal_error(Err.message());
Tim Northover94bc73d2012-10-29 10:47:04 +000028 }
Juergen Ributzka7608dc02014-03-21 20:28:42 +000029 return Err;
30}
Lang Hamesb5c7b1f2014-11-26 16:54:40 +000031
Tim Northover94bc73d2012-10-29 10:47:04 +000032} // end anonymous namespace
33
Eli Bendersky058d6472012-01-22 07:05:02 +000034class RuntimeDyldELF : public RuntimeDyldImpl {
Lang Hamesb5c7b1f2014-11-26 16:54:40 +000035
Juergen Ributzka7608dc02014-03-21 20:28:42 +000036 void resolveRelocation(const SectionEntry &Section, uint64_t Offset,
37 uint64_t Value, uint32_t Type, int64_t Addend,
38 uint64_t SymOffset = 0);
Rafael Espindolaf1f1c622013-04-29 17:24:34 +000039
Juergen Ributzka7608dc02014-03-21 20:28:42 +000040 void resolveX86_64Relocation(const SectionEntry &Section, uint64_t Offset,
41 uint64_t Value, uint32_t Type, int64_t Addend,
Andrew Kaylor4612fed2013-08-19 23:27:43 +000042 uint64_t SymOffset);
Eli Bendersky058d6472012-01-22 07:05:02 +000043
Juergen Ributzka7608dc02014-03-21 20:28:42 +000044 void resolveX86Relocation(const SectionEntry &Section, uint64_t Offset,
45 uint32_t Value, uint32_t Type, int32_t Addend);
Eli Bendersky058d6472012-01-22 07:05:02 +000046
Juergen Ributzka7608dc02014-03-21 20:28:42 +000047 void resolveAArch64Relocation(const SectionEntry &Section, uint64_t Offset,
48 uint64_t Value, uint32_t Type, int64_t Addend);
Tim Northoverfa1b2f82013-05-04 20:13:59 +000049
Juergen Ributzka7608dc02014-03-21 20:28:42 +000050 void resolveARMRelocation(const SectionEntry &Section, uint64_t Offset,
51 uint32_t Value, uint32_t Type, int32_t Addend);
Eli Bendersky058d6472012-01-22 07:05:02 +000052
Juergen Ributzka7608dc02014-03-21 20:28:42 +000053 void resolveMIPSRelocation(const SectionEntry &Section, uint64_t Offset,
54 uint32_t Value, uint32_t Type, int32_t Addend);
Akira Hatanaka111174b2012-08-17 21:28:04 +000055
Juergen Ributzka7608dc02014-03-21 20:28:42 +000056 void resolvePPC64Relocation(const SectionEntry &Section, uint64_t Offset,
57 uint64_t Value, uint32_t Type, int64_t Addend);
Adhemerval Zanella5fc11b32012-10-25 13:13:48 +000058
Juergen Ributzka7608dc02014-03-21 20:28:42 +000059 void resolveSystemZRelocation(const SectionEntry &Section, uint64_t Offset,
60 uint64_t Value, uint32_t Type, int64_t Addend);
Eli Bendersky058d6472012-01-22 07:05:02 +000061
Craig Topperb51ff602014-03-08 07:51:20 +000062 unsigned getMaxStubSize() override {
Tim Northovere19bed72014-07-23 12:32:47 +000063 if (Arch == Triple::aarch64 || Arch == Triple::aarch64_be)
Andrew Kaylor2ba21c52013-10-15 21:32:56 +000064 return 20; // movz; movk; movk; movk; br
65 if (Arch == Triple::arm || Arch == Triple::thumb)
66 return 8; // 32-bit instruction and 32-bit address
67 else if (Arch == Triple::mipsel || Arch == Triple::mips)
68 return 16;
69 else if (Arch == Triple::ppc64 || Arch == Triple::ppc64le)
70 return 44;
71 else if (Arch == Triple::x86_64)
72 return 6; // 2-byte jmp instruction + 32-bit relative address
73 else if (Arch == Triple::systemz)
74 return 16;
75 else
76 return 0;
77 }
78
Craig Topperb51ff602014-03-08 07:51:20 +000079 unsigned getStubAlignment() override {
Andrew Kaylor2ba21c52013-10-15 21:32:56 +000080 if (Arch == Triple::systemz)
81 return 8;
82 else
83 return 1;
84 }
85
Lang Hamesb5c7b1f2014-11-26 16:54:40 +000086 void findPPC64TOCSection(const ObjectFile &Obj,
87 ObjSectionToIDMap &LocalSections,
Ulrich Weigand8f1f87c2014-06-27 10:32:14 +000088 RelocationValueRef &Rel);
Lang Hamesb5c7b1f2014-11-26 16:54:40 +000089 void findOPDEntrySection(const ObjectFile &Obj,
90 ObjSectionToIDMap &LocalSections,
Adhemerval Zanella5fc11b32012-10-25 13:13:48 +000091 RelocationValueRef &Rel);
92
Andrew Kaylor4612fed2013-08-19 23:27:43 +000093 uint64_t findGOTEntry(uint64_t LoadAddr, uint64_t Offset);
94 size_t getGOTEntrySize();
95
Craig Topperb51ff602014-03-08 07:51:20 +000096 void updateGOTEntries(StringRef Name, uint64_t Addr) override;
Andrew Kaylor4612fed2013-08-19 23:27:43 +000097
Alp Tokercb402912014-01-24 17:20:08 +000098 // Relocation entries for symbols whose position-independent offset is
Andrew Kaylor480dcb32013-10-05 01:52:09 +000099 // updated in a global offset table.
Andrew Kaylor480dcb32013-10-05 01:52:09 +0000100 typedef SmallVector<RelocationValueRef, 2> GOTRelocations;
101 GOTRelocations GOTEntries; // List of entries requiring finalization.
102 SmallVector<std::pair<SID, GOTRelocations>, 8> GOTs; // Allocated tables.
Andrew Kaylor4612fed2013-08-19 23:27:43 +0000103
Andrew Kaylor7bb13442013-10-11 21:25:48 +0000104 // When a module is loaded we save the SectionID of the EH frame section
105 // in a table until we receive a request to register all unregistered
106 // EH frame sections with the memory manager.
107 SmallVector<SID, 2> UnregisteredEHFrameSections;
Andrew Kaylorc442a762013-10-16 00:14:21 +0000108 SmallVector<SID, 2> RegisteredEHFrameSections;
Andrew Kaylor7bb13442013-10-11 21:25:48 +0000109
Eli Bendersky058d6472012-01-22 07:05:02 +0000110public:
Lang Hamesb5c7b1f2014-11-26 16:54:40 +0000111 RuntimeDyldELF(RTDyldMemoryManager *mm);
112 virtual ~RuntimeDyldELF();
113
114 std::unique_ptr<RuntimeDyld::LoadedObjectInfo>
115 loadObject(const object::ObjectFile &O) override;
Preston Gurdcc31af92012-04-16 22:12:58 +0000116
Craig Topperb51ff602014-03-08 07:51:20 +0000117 void resolveRelocation(const RelocationEntry &RE, uint64_t Value) override;
Juergen Ributzka046709f2014-03-21 07:26:41 +0000118 relocation_iterator
119 processRelocationRef(unsigned SectionID, relocation_iterator RelI,
Lang Hamesb5c7b1f2014-11-26 16:54:40 +0000120 const ObjectFile &Obj,
121 ObjSectionToIDMap &ObjSectionToID,
Juergen Ributzka046709f2014-03-21 07:26:41 +0000122 const SymbolTableMap &Symbols, StubMap &Stubs) override;
Lang Hamesb5c7b1f2014-11-26 16:54:40 +0000123 bool isCompatibleFile(const object::ObjectFile &Obj) const override;
Craig Topperb51ff602014-03-08 07:51:20 +0000124 void registerEHFrames() override;
125 void deregisterEHFrames() override;
Lang Hamesb5c7b1f2014-11-26 16:54:40 +0000126 void finalizeLoad(const ObjectFile &Obj,
Lang Hames36072da2014-05-12 21:39:59 +0000127 ObjSectionToIDMap &SectionMap) override;
Eli Bendersky058d6472012-01-22 07:05:02 +0000128};
129
130} // end namespace llvm
131
Danil Malyshev70d22cc2012-03-30 16:45:19 +0000132#endif