| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 1 | //===- InputFiles.cpp -----------------------------------------------------===// |
| 2 | // |
| 3 | // The LLVM Linker |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | |
| 10 | #include "InputFiles.h" |
| Rafael Espindola | 192e1fa | 2015-08-06 15:08:23 +0000 | [diff] [blame] | 11 | #include "Error.h" |
| Rafael Espindola | 9d13d04 | 2016-02-11 15:24:48 +0000 | [diff] [blame] | 12 | #include "InputSection.h" |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 13 | #include "Symbols.h" |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 14 | #include "llvm/ADT/STLExtras.h" |
| Rafael Espindola | 9f77ef0 | 2016-02-12 20:54:57 +0000 | [diff] [blame] | 15 | #include "llvm/IR/LLVMContext.h" |
| Rafael Espindola | 4de44b7 | 2016-03-02 15:43:50 +0000 | [diff] [blame^] | 16 | #include "llvm/IR/Module.h" |
| Rafael Espindola | 9f77ef0 | 2016-02-12 20:54:57 +0000 | [diff] [blame] | 17 | #include "llvm/Object/IRObjectFile.h" |
| 18 | #include "llvm/Support/raw_ostream.h" |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 19 | |
| Michael J. Spencer | 1b348a6 | 2015-09-04 22:28:10 +0000 | [diff] [blame] | 20 | using namespace llvm; |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 21 | using namespace llvm::ELF; |
| Rafael Espindola | f98d6d8 | 2015-09-03 20:03:54 +0000 | [diff] [blame] | 22 | using namespace llvm::object; |
| Rui Ueyama | f5c4aca | 2015-09-30 17:06:09 +0000 | [diff] [blame] | 23 | using namespace llvm::sys::fs; |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 24 | |
| 25 | using namespace lld; |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 26 | using namespace lld::elf; |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 27 | |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 28 | namespace { |
| 29 | class ECRAII { |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 30 | std::error_code EC; |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 31 | |
| 32 | public: |
| 33 | std::error_code &getEC() { return EC; } |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 34 | ~ECRAII() { fatal(EC); } |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 35 | }; |
| Rafael Espindola | f98d6d8 | 2015-09-03 20:03:54 +0000 | [diff] [blame] | 36 | } |
| 37 | |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 38 | template <class ELFT> |
| Rafael Espindola | 2a4b271 | 2015-10-13 01:17:02 +0000 | [diff] [blame] | 39 | ELFFileBase<ELFT>::ELFFileBase(Kind K, MemoryBufferRef M) |
| 40 | : InputFile(K, M), ELFObj(MB.getBuffer(), ECRAII().getEC()) {} |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 41 | |
| 42 | template <class ELFT> |
| Rui Ueyama | 2022e81 | 2015-11-20 02:10:52 +0000 | [diff] [blame] | 43 | ELFKind ELFFileBase<ELFT>::getELFKind() { |
| Rui Ueyama | f588ac4 | 2016-01-06 00:09:41 +0000 | [diff] [blame] | 44 | if (ELFT::TargetEndianness == support::little) |
| 45 | return ELFT::Is64Bits ? ELF64LEKind : ELF32LEKind; |
| 46 | return ELFT::Is64Bits ? ELF64BEKind : ELF32BEKind; |
| Rui Ueyama | 2022e81 | 2015-11-20 02:10:52 +0000 | [diff] [blame] | 47 | } |
| 48 | |
| 49 | template <class ELFT> |
| Rafael Espindola | af70764 | 2015-10-12 01:55:32 +0000 | [diff] [blame] | 50 | typename ELFFileBase<ELFT>::Elf_Sym_Range |
| 51 | ELFFileBase<ELFT>::getSymbolsHelper(bool Local) { |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 52 | if (!Symtab) |
| 53 | return Elf_Sym_Range(nullptr, nullptr); |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 54 | Elf_Sym_Range Syms = ELFObj.symbols(Symtab); |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 55 | uint32_t NumSymbols = std::distance(Syms.begin(), Syms.end()); |
| 56 | uint32_t FirstNonLocal = Symtab->sh_info; |
| 57 | if (FirstNonLocal > NumSymbols) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 58 | fatal("Invalid sh_info in symbol table"); |
| Davide Italiano | 6d328d3 | 2015-09-16 20:45:57 +0000 | [diff] [blame] | 59 | if (!Local) |
| Rui Ueyama | 90b3daa | 2015-09-30 02:37:51 +0000 | [diff] [blame] | 60 | return make_range(Syms.begin() + FirstNonLocal, Syms.end()); |
| 61 | // +1 to skip over dummy symbol. |
| 62 | return make_range(Syms.begin() + 1, Syms.begin() + FirstNonLocal); |
| Davide Italiano | 6d328d3 | 2015-09-16 20:45:57 +0000 | [diff] [blame] | 63 | } |
| 64 | |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 65 | template <class ELFT> |
| 66 | uint32_t ELFFileBase<ELFT>::getSectionIndex(const Elf_Sym &Sym) const { |
| Rui Ueyama | dc8d3a2 | 2015-12-24 08:36:56 +0000 | [diff] [blame] | 67 | uint32_t I = Sym.st_shndx; |
| 68 | if (I == ELF::SHN_XINDEX) |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 69 | return ELFObj.getExtendedSymbolTableIndex(&Sym, Symtab, SymtabSHNDX); |
| Rafael Espindola | 02ce26a | 2015-12-24 14:22:24 +0000 | [diff] [blame] | 70 | if (I >= ELF::SHN_LORESERVE || I == ELF::SHN_ABS) |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 71 | return 0; |
| Rui Ueyama | dc8d3a2 | 2015-12-24 08:36:56 +0000 | [diff] [blame] | 72 | return I; |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 73 | } |
| 74 | |
| Rafael Espindola | af70764 | 2015-10-12 01:55:32 +0000 | [diff] [blame] | 75 | template <class ELFT> void ELFFileBase<ELFT>::initStringTable() { |
| Rafael Espindola | 3e60379 | 2015-10-01 20:26:37 +0000 | [diff] [blame] | 76 | if (!Symtab) |
| 77 | return; |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 78 | ErrorOr<StringRef> StringTableOrErr = ELFObj.getStringTableForSymtab(*Symtab); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 79 | fatal(StringTableOrErr); |
| Davide Italiano | 6d328d3 | 2015-09-16 20:45:57 +0000 | [diff] [blame] | 80 | StringTable = *StringTableOrErr; |
| Rafael Espindola | 6a3b5de | 2015-10-01 19:52:48 +0000 | [diff] [blame] | 81 | } |
| 82 | |
| 83 | template <class ELFT> |
| Rafael Espindola | af70764 | 2015-10-12 01:55:32 +0000 | [diff] [blame] | 84 | typename ELFFileBase<ELFT>::Elf_Sym_Range |
| 85 | ELFFileBase<ELFT>::getNonLocalSymbols() { |
| Davide Italiano | 6d328d3 | 2015-09-16 20:45:57 +0000 | [diff] [blame] | 86 | return getSymbolsHelper(false); |
| 87 | } |
| 88 | |
| 89 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 90 | elf::ObjectFile<ELFT>::ObjectFile(MemoryBufferRef M) |
| Rafael Espindola | 2a4b271 | 2015-10-13 01:17:02 +0000 | [diff] [blame] | 91 | : ELFFileBase<ELFT>(Base::ObjectKind, M) {} |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 92 | |
| 93 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 94 | typename elf::ObjectFile<ELFT>::Elf_Sym_Range |
| 95 | elf::ObjectFile<ELFT>::getLocalSymbols() { |
| Davide Italiano | 6d328d3 | 2015-09-16 20:45:57 +0000 | [diff] [blame] | 96 | return this->getSymbolsHelper(true); |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 97 | } |
| 98 | |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 99 | template <class ELFT> uint32_t elf::ObjectFile<ELFT>::getMipsGp0() const { |
| Rui Ueyama | 70eed36 | 2016-01-06 22:42:43 +0000 | [diff] [blame] | 100 | if (MipsReginfo) |
| 101 | return MipsReginfo->Reginfo->ri_gp_value; |
| 102 | return 0; |
| Simon Atanasyan | 57830b6 | 2015-12-25 13:02:13 +0000 | [diff] [blame] | 103 | } |
| 104 | |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 105 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 106 | const typename elf::ObjectFile<ELFT>::Elf_Sym * |
| 107 | elf::ObjectFile<ELFT>::getLocalSymbol(uintX_t SymIndex) { |
| Rui Ueyama | c4aaed9 | 2015-10-22 18:49:53 +0000 | [diff] [blame] | 108 | uint32_t FirstNonLocal = this->Symtab->sh_info; |
| 109 | if (SymIndex >= FirstNonLocal) |
| 110 | return nullptr; |
| 111 | Elf_Sym_Range Syms = this->ELFObj.symbols(this->Symtab); |
| 112 | return Syms.begin() + SymIndex; |
| 113 | } |
| 114 | |
| 115 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 116 | void elf::ObjectFile<ELFT>::parse(DenseSet<StringRef> &ComdatGroups) { |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 117 | // Read section and symbol tables. |
| Rui Ueyama | 52d3b67 | 2016-01-06 02:06:33 +0000 | [diff] [blame] | 118 | initializeSections(ComdatGroups); |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 119 | initializeSymbols(); |
| 120 | } |
| 121 | |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 122 | // Sections with SHT_GROUP and comdat bits define comdat section groups. |
| 123 | // They are identified and deduplicated by group name. This function |
| 124 | // returns a group name. |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 125 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 126 | StringRef elf::ObjectFile<ELFT>::getShtGroupSignature(const Elf_Shdr &Sec) { |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 127 | const ELFFile<ELFT> &Obj = this->ELFObj; |
| 128 | uint32_t SymtabdSectionIndex = Sec.sh_link; |
| 129 | ErrorOr<const Elf_Shdr *> SecOrErr = Obj.getSection(SymtabdSectionIndex); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 130 | fatal(SecOrErr); |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 131 | const Elf_Shdr *SymtabSec = *SecOrErr; |
| 132 | uint32_t SymIndex = Sec.sh_info; |
| 133 | const Elf_Sym *Sym = Obj.getSymbol(SymtabSec, SymIndex); |
| 134 | ErrorOr<StringRef> StringTableOrErr = Obj.getStringTableForSymtab(*SymtabSec); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 135 | fatal(StringTableOrErr); |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 136 | ErrorOr<StringRef> SignatureOrErr = Sym->getName(*StringTableOrErr); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 137 | fatal(SignatureOrErr); |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 138 | return *SignatureOrErr; |
| 139 | } |
| 140 | |
| 141 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 142 | ArrayRef<typename elf::ObjectFile<ELFT>::uint32_X> |
| 143 | elf::ObjectFile<ELFT>::getShtGroupEntries(const Elf_Shdr &Sec) { |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 144 | const ELFFile<ELFT> &Obj = this->ELFObj; |
| Rui Ueyama | 33b3f21 | 2016-01-06 20:30:02 +0000 | [diff] [blame] | 145 | ErrorOr<ArrayRef<uint32_X>> EntriesOrErr = |
| 146 | Obj.template getSectionContentsAsArray<uint32_X>(&Sec); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 147 | fatal(EntriesOrErr); |
| Rui Ueyama | 33b3f21 | 2016-01-06 20:30:02 +0000 | [diff] [blame] | 148 | ArrayRef<uint32_X> Entries = *EntriesOrErr; |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 149 | if (Entries.empty() || Entries[0] != GRP_COMDAT) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 150 | fatal("Unsupported SHT_GROUP format"); |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 151 | return Entries.slice(1); |
| 152 | } |
| 153 | |
| 154 | template <class ELFT> |
| Rafael Espindola | f82ed2a | 2015-10-24 22:51:01 +0000 | [diff] [blame] | 155 | static bool shouldMerge(const typename ELFFile<ELFT>::Elf_Shdr &Sec) { |
| 156 | typedef typename ELFFile<ELFT>::uintX_t uintX_t; |
| 157 | uintX_t Flags = Sec.sh_flags; |
| 158 | if (!(Flags & SHF_MERGE)) |
| 159 | return false; |
| 160 | if (Flags & SHF_WRITE) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 161 | fatal("Writable SHF_MERGE sections are not supported"); |
| Rafael Espindola | f82ed2a | 2015-10-24 22:51:01 +0000 | [diff] [blame] | 162 | uintX_t EntSize = Sec.sh_entsize; |
| George Rimar | 564da7e | 2015-11-09 08:40:44 +0000 | [diff] [blame] | 163 | if (!EntSize || Sec.sh_size % EntSize) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 164 | fatal("SHF_MERGE section size must be a multiple of sh_entsize"); |
| Rafael Espindola | f82ed2a | 2015-10-24 22:51:01 +0000 | [diff] [blame] | 165 | |
| Rafael Espindola | 7efa5be | 2016-02-19 14:17:40 +0000 | [diff] [blame] | 166 | // Don't try to merge if the aligment is larger than the sh_entsize and this |
| 167 | // is not SHF_STRINGS. |
| Rafael Espindola | f82ed2a | 2015-10-24 22:51:01 +0000 | [diff] [blame] | 168 | // |
| Rafael Espindola | 7efa5be | 2016-02-19 14:17:40 +0000 | [diff] [blame] | 169 | // Since this is not a SHF_STRINGS, we would need to pad after every entity. |
| 170 | // It would be equivalent for the producer of the .o to just set a larger |
| Rafael Espindola | f82ed2a | 2015-10-24 22:51:01 +0000 | [diff] [blame] | 171 | // sh_entsize. |
| Rafael Espindola | 7efa5be | 2016-02-19 14:17:40 +0000 | [diff] [blame] | 172 | if (Flags & SHF_STRINGS) |
| 173 | return true; |
| 174 | |
| Rafael Espindola | f82ed2a | 2015-10-24 22:51:01 +0000 | [diff] [blame] | 175 | if (Sec.sh_addralign > EntSize) |
| 176 | return false; |
| 177 | |
| 178 | return true; |
| 179 | } |
| 180 | |
| 181 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 182 | void elf::ObjectFile<ELFT>::initializeSections( |
| Rafael Espindola | f1d598c | 2016-02-12 21:17:10 +0000 | [diff] [blame] | 183 | DenseSet<StringRef> &ComdatGroups) { |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 184 | uint64_t Size = this->ELFObj.getNumSections(); |
| Rafael Espindola | 7167585 | 2015-09-22 00:16:19 +0000 | [diff] [blame] | 185 | Sections.resize(Size); |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 186 | unsigned I = -1; |
| Rafael Espindola | d42f4e5 | 2015-10-08 12:02:38 +0000 | [diff] [blame] | 187 | const ELFFile<ELFT> &Obj = this->ELFObj; |
| 188 | for (const Elf_Shdr &Sec : Obj.sections()) { |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 189 | ++I; |
| Rui Ueyama | 733153d | 2016-02-24 18:33:35 +0000 | [diff] [blame] | 190 | if (Sections[I] == InputSection<ELFT>::Discarded) |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 191 | continue; |
| 192 | |
| Rafael Espindola | cde2513 | 2015-08-13 14:45:44 +0000 | [diff] [blame] | 193 | switch (Sec.sh_type) { |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 194 | case SHT_GROUP: |
| Rui Ueyama | 733153d | 2016-02-24 18:33:35 +0000 | [diff] [blame] | 195 | Sections[I] = InputSection<ELFT>::Discarded; |
| Rui Ueyama | 52d3b67 | 2016-01-06 02:06:33 +0000 | [diff] [blame] | 196 | if (ComdatGroups.insert(getShtGroupSignature(Sec)).second) |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 197 | continue; |
| Rui Ueyama | 33b3f21 | 2016-01-06 20:30:02 +0000 | [diff] [blame] | 198 | for (uint32_t SecIndex : getShtGroupEntries(Sec)) { |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 199 | if (SecIndex >= Size) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 200 | fatal("Invalid section index in group"); |
| Rui Ueyama | 733153d | 2016-02-24 18:33:35 +0000 | [diff] [blame] | 201 | Sections[SecIndex] = InputSection<ELFT>::Discarded; |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 202 | } |
| 203 | break; |
| Rafael Espindola | cde2513 | 2015-08-13 14:45:44 +0000 | [diff] [blame] | 204 | case SHT_SYMTAB: |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 205 | this->Symtab = &Sec; |
| Rafael Espindola | cde2513 | 2015-08-13 14:45:44 +0000 | [diff] [blame] | 206 | break; |
| Rafael Espindola | 2034822 | 2015-08-24 21:43:25 +0000 | [diff] [blame] | 207 | case SHT_SYMTAB_SHNDX: { |
| Rafael Espindola | d42f4e5 | 2015-10-08 12:02:38 +0000 | [diff] [blame] | 208 | ErrorOr<ArrayRef<Elf_Word>> ErrorOrTable = Obj.getSHNDXTable(Sec); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 209 | fatal(ErrorOrTable); |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 210 | this->SymtabSHNDX = *ErrorOrTable; |
| Rafael Espindola | 2034822 | 2015-08-24 21:43:25 +0000 | [diff] [blame] | 211 | break; |
| 212 | } |
| Rafael Espindola | cde2513 | 2015-08-13 14:45:44 +0000 | [diff] [blame] | 213 | case SHT_STRTAB: |
| 214 | case SHT_NULL: |
| Rafael Espindola | cde2513 | 2015-08-13 14:45:44 +0000 | [diff] [blame] | 215 | break; |
| Michael J. Spencer | 67bc8d6 | 2015-08-27 23:15:56 +0000 | [diff] [blame] | 216 | case SHT_RELA: |
| 217 | case SHT_REL: { |
| 218 | uint32_t RelocatedSectionIndex = Sec.sh_info; |
| 219 | if (RelocatedSectionIndex >= Size) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 220 | fatal("Invalid relocated section index"); |
| Rafael Espindola | c159c96 | 2015-10-19 21:00:02 +0000 | [diff] [blame] | 221 | InputSectionBase<ELFT> *RelocatedSection = |
| 222 | Sections[RelocatedSectionIndex]; |
| Sean Silva | 09247f8 | 2016-02-04 21:41:07 +0000 | [diff] [blame] | 223 | // Strictly speaking, a relocation section must be included in the |
| 224 | // group of the section it relocates. However, LLVM 3.3 and earlier |
| 225 | // would fail to do so, so we gracefully handle that case. |
| Rui Ueyama | 733153d | 2016-02-24 18:33:35 +0000 | [diff] [blame] | 226 | if (RelocatedSection == InputSection<ELFT>::Discarded) |
| Sean Silva | 09247f8 | 2016-02-04 21:41:07 +0000 | [diff] [blame] | 227 | continue; |
| Michael J. Spencer | 67bc8d6 | 2015-08-27 23:15:56 +0000 | [diff] [blame] | 228 | if (!RelocatedSection) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 229 | fatal("Unsupported relocation reference"); |
| George Rimar | 58941ee | 2016-02-25 08:23:37 +0000 | [diff] [blame] | 230 | if (Config->Relocatable) { |
| 231 | // For -r, relocation sections are handled as regular input sections. |
| 232 | Sections[I] = new (Alloc) InputSection<ELFT>(this, &Sec); |
| 233 | } else if (auto *S = dyn_cast<InputSection<ELFT>>(RelocatedSection)) { |
| Rafael Espindola | c159c96 | 2015-10-19 21:00:02 +0000 | [diff] [blame] | 234 | S->RelocSections.push_back(&Sec); |
| Rafael Espindola | 0c6a4f1 | 2015-11-11 19:54:14 +0000 | [diff] [blame] | 235 | } else if (auto *S = dyn_cast<EHInputSection<ELFT>>(RelocatedSection)) { |
| 236 | if (S->RelocSection) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 237 | fatal("Multiple relocation sections to .eh_frame are not supported"); |
| Rafael Espindola | 0c6a4f1 | 2015-11-11 19:54:14 +0000 | [diff] [blame] | 238 | S->RelocSection = &Sec; |
| 239 | } else { |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 240 | fatal("Relocations pointing to SHF_MERGE are not supported"); |
| Rafael Espindola | 0c6a4f1 | 2015-11-11 19:54:14 +0000 | [diff] [blame] | 241 | } |
| Michael J. Spencer | 67bc8d6 | 2015-08-27 23:15:56 +0000 | [diff] [blame] | 242 | break; |
| 243 | } |
| Rui Ueyama | e79b09a | 2015-11-21 22:19:32 +0000 | [diff] [blame] | 244 | default: |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 245 | Sections[I] = createInputSection(Sec); |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 246 | } |
| 247 | } |
| 248 | } |
| 249 | |
| Rafael Espindola | f1d598c | 2016-02-12 21:17:10 +0000 | [diff] [blame] | 250 | template <class ELFT> |
| 251 | InputSectionBase<ELFT> * |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 252 | elf::ObjectFile<ELFT>::createInputSection(const Elf_Shdr &Sec) { |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 253 | ErrorOr<StringRef> NameOrErr = this->ELFObj.getSectionName(&Sec); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 254 | fatal(NameOrErr); |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 255 | StringRef Name = *NameOrErr; |
| 256 | |
| 257 | // .note.GNU-stack is a marker section to control the presence of |
| 258 | // PT_GNU_STACK segment in outputs. Since the presence of the segment |
| 259 | // is controlled only by the command line option (-z execstack) in LLD, |
| 260 | // .note.GNU-stack is ignored. |
| 261 | if (Name == ".note.GNU-stack") |
| Rui Ueyama | 733153d | 2016-02-24 18:33:35 +0000 | [diff] [blame] | 262 | return InputSection<ELFT>::Discarded; |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 263 | |
| 264 | // A MIPS object file has a special section that contains register |
| 265 | // usage info, which needs to be handled by the linker specially. |
| Simon Atanasyan | 57830b6 | 2015-12-25 13:02:13 +0000 | [diff] [blame] | 266 | if (Config->EMachine == EM_MIPS && Name == ".reginfo") { |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 267 | MipsReginfo = new (Alloc) MipsReginfoInputSection<ELFT>(this, &Sec); |
| Simon Atanasyan | 57830b6 | 2015-12-25 13:02:13 +0000 | [diff] [blame] | 268 | return MipsReginfo; |
| 269 | } |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 270 | |
| 271 | if (Name == ".eh_frame") |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 272 | return new (EHAlloc.Allocate()) EHInputSection<ELFT>(this, &Sec); |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 273 | if (shouldMerge<ELFT>(Sec)) |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 274 | return new (MAlloc.Allocate()) MergeInputSection<ELFT>(this, &Sec); |
| 275 | return new (Alloc) InputSection<ELFT>(this, &Sec); |
| Rui Ueyama | 3f11c8c | 2015-12-24 08:41:12 +0000 | [diff] [blame] | 276 | } |
| 277 | |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 278 | template <class ELFT> void elf::ObjectFile<ELFT>::initializeSymbols() { |
| Rafael Espindola | 6a3b5de | 2015-10-01 19:52:48 +0000 | [diff] [blame] | 279 | this->initStringTable(); |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 280 | Elf_Sym_Range Syms = this->getNonLocalSymbols(); |
| Reid Kleckner | f7b85e0 | 2015-08-11 20:06:51 +0000 | [diff] [blame] | 281 | uint32_t NumSymbols = std::distance(Syms.begin(), Syms.end()); |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 282 | SymbolBodies.reserve(NumSymbols); |
| Rafael Espindola | 3031851 | 2015-08-04 14:00:56 +0000 | [diff] [blame] | 283 | for (const Elf_Sym &Sym : Syms) |
| Rui Ueyama | c5e372d | 2016-01-21 02:10:12 +0000 | [diff] [blame] | 284 | SymbolBodies.push_back(createSymbolBody(&Sym)); |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 285 | } |
| 286 | |
| 287 | template <class ELFT> |
| Rafael Espindola | c159c96 | 2015-10-19 21:00:02 +0000 | [diff] [blame] | 288 | InputSectionBase<ELFT> * |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 289 | elf::ObjectFile<ELFT>::getSection(const Elf_Sym &Sym) const { |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 290 | uint32_t Index = this->getSectionIndex(Sym); |
| 291 | if (Index == 0) |
| Rafael Espindola | 4cda581 | 2015-10-16 15:29:48 +0000 | [diff] [blame] | 292 | return nullptr; |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 293 | if (Index >= Sections.size() || !Sections[Index]) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 294 | fatal("Invalid section index"); |
| Rui Ueyama | 0b28952 | 2016-02-25 18:43:51 +0000 | [diff] [blame] | 295 | InputSectionBase<ELFT> *S = Sections[Index]; |
| 296 | if (S == InputSectionBase<ELFT>::Discarded) |
| 297 | return S; |
| 298 | return S->Repl; |
| Rafael Espindola | 4cda581 | 2015-10-16 15:29:48 +0000 | [diff] [blame] | 299 | } |
| 300 | |
| 301 | template <class ELFT> |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 302 | SymbolBody *elf::ObjectFile<ELFT>::createSymbolBody(const Elf_Sym *Sym) { |
| Rui Ueyama | c5e372d | 2016-01-21 02:10:12 +0000 | [diff] [blame] | 303 | ErrorOr<StringRef> NameOrErr = Sym->getName(this->StringTable); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 304 | fatal(NameOrErr); |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 305 | StringRef Name = *NameOrErr; |
| Rafael Espindola | 2034822 | 2015-08-24 21:43:25 +0000 | [diff] [blame] | 306 | |
| Rafael Espindola | 4cda581 | 2015-10-16 15:29:48 +0000 | [diff] [blame] | 307 | switch (Sym->st_shndx) { |
| Rafael Espindola | 51d4690 | 2015-08-28 21:26:51 +0000 | [diff] [blame] | 308 | case SHN_UNDEF: |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 309 | return new (Alloc) UndefinedElf<ELFT>(Name, *Sym); |
| Rafael Espindola | 51d4690 | 2015-08-28 21:26:51 +0000 | [diff] [blame] | 310 | case SHN_COMMON: |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 311 | return new (Alloc) DefinedCommon(Name, Sym->st_size, Sym->st_value, |
| 312 | Sym->getBinding() == llvm::ELF::STB_WEAK, |
| 313 | Sym->getVisibility()); |
| Rafael Espindola | 51d4690 | 2015-08-28 21:26:51 +0000 | [diff] [blame] | 314 | } |
| Rafael Espindola | 2034822 | 2015-08-24 21:43:25 +0000 | [diff] [blame] | 315 | |
| Rafael Espindola | b13df65 | 2015-08-11 17:33:02 +0000 | [diff] [blame] | 316 | switch (Sym->getBinding()) { |
| 317 | default: |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 318 | fatal("unexpected binding"); |
| Rafael Espindola | b13df65 | 2015-08-11 17:33:02 +0000 | [diff] [blame] | 319 | case STB_GLOBAL: |
| Rafael Espindola | 3a63f3f | 2015-08-28 20:19:34 +0000 | [diff] [blame] | 320 | case STB_WEAK: |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 321 | case STB_GNU_UNIQUE: { |
| Rafael Espindola | c159c96 | 2015-10-19 21:00:02 +0000 | [diff] [blame] | 322 | InputSectionBase<ELFT> *Sec = getSection(*Sym); |
| Rui Ueyama | 733153d | 2016-02-24 18:33:35 +0000 | [diff] [blame] | 323 | if (Sec == InputSection<ELFT>::Discarded) |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 324 | return new (Alloc) UndefinedElf<ELFT>(Name, *Sym); |
| 325 | return new (Alloc) DefinedRegular<ELFT>(Name, *Sym, Sec); |
| Rafael Espindola | 444576d | 2015-10-09 19:25:07 +0000 | [diff] [blame] | 326 | } |
| Rafael Espindola | b13df65 | 2015-08-11 17:33:02 +0000 | [diff] [blame] | 327 | } |
| Michael J. Spencer | 84487f1 | 2015-07-24 21:03:07 +0000 | [diff] [blame] | 328 | } |
| 329 | |
| Igor Kudrin | 2696bbe | 2015-10-01 18:02:21 +0000 | [diff] [blame] | 330 | void ArchiveFile::parse() { |
| Rui Ueyama | 9b09369 | 2016-01-06 00:51:35 +0000 | [diff] [blame] | 331 | ErrorOr<std::unique_ptr<Archive>> FileOrErr = Archive::create(MB); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 332 | fatal(FileOrErr, "Failed to parse archive"); |
| Rui Ueyama | 9b09369 | 2016-01-06 00:51:35 +0000 | [diff] [blame] | 333 | File = std::move(*FileOrErr); |
| Michael J. Spencer | 1b348a6 | 2015-09-04 22:28:10 +0000 | [diff] [blame] | 334 | |
| 335 | // Allocate a buffer for Lazy objects. |
| 336 | size_t NumSyms = File->getNumberOfSymbols(); |
| 337 | LazySymbols.reserve(NumSyms); |
| 338 | |
| 339 | // Read the symbol table to construct Lazy objects. |
| 340 | for (const Archive::Symbol &Sym : File->symbols()) |
| 341 | LazySymbols.emplace_back(this, Sym); |
| 342 | } |
| 343 | |
| 344 | // Returns a buffer pointing to a member file containing a given symbol. |
| 345 | MemoryBufferRef ArchiveFile::getMember(const Archive::Symbol *Sym) { |
| Rafael Espindola | 8f3a6ae | 2015-11-05 14:40:28 +0000 | [diff] [blame] | 346 | ErrorOr<Archive::Child> COrErr = Sym->getMember(); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 347 | fatal(COrErr, "Could not get the member for symbol " + Sym->getName()); |
| Rafael Espindola | 8f3a6ae | 2015-11-05 14:40:28 +0000 | [diff] [blame] | 348 | const Archive::Child &C = *COrErr; |
| Michael J. Spencer | 1b348a6 | 2015-09-04 22:28:10 +0000 | [diff] [blame] | 349 | |
| Rafael Espindola | 8f3a6ae | 2015-11-05 14:40:28 +0000 | [diff] [blame] | 350 | if (!Seen.insert(C.getChildOffset()).second) |
| Michael J. Spencer | 1b348a6 | 2015-09-04 22:28:10 +0000 | [diff] [blame] | 351 | return MemoryBufferRef(); |
| Michael J. Spencer | 88f0d63 | 2015-09-08 20:36:20 +0000 | [diff] [blame] | 352 | |
| Rui Ueyama | 784b769 | 2015-12-17 01:51:23 +0000 | [diff] [blame] | 353 | ErrorOr<MemoryBufferRef> RefOrErr = C.getMemoryBufferRef(); |
| 354 | if (!RefOrErr) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 355 | fatal(RefOrErr, "Could not get the buffer for the member defining symbol " + |
| 356 | Sym->getName()); |
| Rui Ueyama | 784b769 | 2015-12-17 01:51:23 +0000 | [diff] [blame] | 357 | return *RefOrErr; |
| Michael J. Spencer | 1b348a6 | 2015-09-04 22:28:10 +0000 | [diff] [blame] | 358 | } |
| 359 | |
| Rafael Espindola | e1901cc | 2015-09-24 15:11:50 +0000 | [diff] [blame] | 360 | template <class ELFT> |
| 361 | SharedFile<ELFT>::SharedFile(MemoryBufferRef M) |
| Rui Ueyama | f588ac4 | 2016-01-06 00:09:41 +0000 | [diff] [blame] | 362 | : ELFFileBase<ELFT>(Base::SharedKind, M), AsNeeded(Config->AsNeeded) {} |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 363 | |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 364 | template <class ELFT> |
| 365 | const typename ELFFile<ELFT>::Elf_Shdr * |
| 366 | SharedFile<ELFT>::getSection(const Elf_Sym &Sym) const { |
| 367 | uint32_t Index = this->getSectionIndex(Sym); |
| 368 | if (Index == 0) |
| 369 | return nullptr; |
| 370 | ErrorOr<const Elf_Shdr *> Ret = this->ELFObj.getSection(Index); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 371 | fatal(Ret); |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 372 | return *Ret; |
| 373 | } |
| 374 | |
| Rui Ueyama | 7c71331 | 2016-01-06 01:56:36 +0000 | [diff] [blame] | 375 | // Partially parse the shared object file so that we can call |
| 376 | // getSoName on this object. |
| Rafael Espindola | 6a3b5de | 2015-10-01 19:52:48 +0000 | [diff] [blame] | 377 | template <class ELFT> void SharedFile<ELFT>::parseSoName() { |
| Rafael Espindola | c8b1581 | 2015-10-01 15:47:50 +0000 | [diff] [blame] | 378 | typedef typename ELFFile<ELFT>::Elf_Dyn Elf_Dyn; |
| 379 | typedef typename ELFFile<ELFT>::uintX_t uintX_t; |
| 380 | const Elf_Shdr *DynamicSec = nullptr; |
| 381 | |
| 382 | const ELFFile<ELFT> Obj = this->ELFObj; |
| 383 | for (const Elf_Shdr &Sec : Obj.sections()) { |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 384 | switch (Sec.sh_type) { |
| 385 | default: |
| 386 | continue; |
| 387 | case SHT_DYNSYM: |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 388 | this->Symtab = &Sec; |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 389 | break; |
| 390 | case SHT_DYNAMIC: |
| Rafael Espindola | c8b1581 | 2015-10-01 15:47:50 +0000 | [diff] [blame] | 391 | DynamicSec = &Sec; |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 392 | break; |
| 393 | case SHT_SYMTAB_SHNDX: { |
| 394 | ErrorOr<ArrayRef<Elf_Word>> ErrorOrTable = Obj.getSHNDXTable(Sec); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 395 | fatal(ErrorOrTable); |
| Rafael Espindola | 115f0f3 | 2015-11-03 14:13:40 +0000 | [diff] [blame] | 396 | this->SymtabSHNDX = *ErrorOrTable; |
| 397 | break; |
| 398 | } |
| 399 | } |
| Rafael Espindola | c8b1581 | 2015-10-01 15:47:50 +0000 | [diff] [blame] | 400 | } |
| 401 | |
| Rafael Espindola | 6a3b5de | 2015-10-01 19:52:48 +0000 | [diff] [blame] | 402 | this->initStringTable(); |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 403 | SoName = this->getName(); |
| Rafael Espindola | c8b1581 | 2015-10-01 15:47:50 +0000 | [diff] [blame] | 404 | |
| Rui Ueyama | 361d8b9 | 2015-10-12 15:49:02 +0000 | [diff] [blame] | 405 | if (!DynamicSec) |
| 406 | return; |
| 407 | auto *Begin = |
| 408 | reinterpret_cast<const Elf_Dyn *>(Obj.base() + DynamicSec->sh_offset); |
| 409 | const Elf_Dyn *End = Begin + DynamicSec->sh_size / sizeof(Elf_Dyn); |
| Rafael Espindola | c8b1581 | 2015-10-01 15:47:50 +0000 | [diff] [blame] | 410 | |
| Rui Ueyama | 361d8b9 | 2015-10-12 15:49:02 +0000 | [diff] [blame] | 411 | for (const Elf_Dyn &Dyn : make_range(Begin, End)) { |
| 412 | if (Dyn.d_tag == DT_SONAME) { |
| 413 | uintX_t Val = Dyn.getVal(); |
| 414 | if (Val >= this->StringTable.size()) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 415 | fatal("Invalid DT_SONAME entry"); |
| Rui Ueyama | e69ab10 | 2016-01-06 01:14:11 +0000 | [diff] [blame] | 416 | SoName = StringRef(this->StringTable.data() + Val); |
| Rui Ueyama | 361d8b9 | 2015-10-12 15:49:02 +0000 | [diff] [blame] | 417 | return; |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 418 | } |
| 419 | } |
| Rafael Espindola | 6a3b5de | 2015-10-01 19:52:48 +0000 | [diff] [blame] | 420 | } |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 421 | |
| Rui Ueyama | 7c71331 | 2016-01-06 01:56:36 +0000 | [diff] [blame] | 422 | // Fully parse the shared object file. This must be called after parseSoName(). |
| 423 | template <class ELFT> void SharedFile<ELFT>::parseRest() { |
| Rafael Espindola | 6a3b5de | 2015-10-01 19:52:48 +0000 | [diff] [blame] | 424 | Elf_Sym_Range Syms = this->getNonLocalSymbols(); |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 425 | uint32_t NumSymbols = std::distance(Syms.begin(), Syms.end()); |
| 426 | SymbolBodies.reserve(NumSymbols); |
| 427 | for (const Elf_Sym &Sym : Syms) { |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 428 | ErrorOr<StringRef> NameOrErr = Sym.getName(this->StringTable); |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 429 | fatal(NameOrErr.getError()); |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 430 | StringRef Name = *NameOrErr; |
| 431 | |
| Rui Ueyama | f8432d9 | 2015-10-13 16:34:14 +0000 | [diff] [blame] | 432 | if (Sym.isUndefined()) |
| 433 | Undefs.push_back(Name); |
| 434 | else |
| 435 | SymbolBodies.emplace_back(this, Name, Sym); |
| Rafael Espindola | 18173d4 | 2015-09-08 15:50:05 +0000 | [diff] [blame] | 436 | } |
| 437 | } |
| Rafael Espindola | f98d6d8 | 2015-09-03 20:03:54 +0000 | [diff] [blame] | 438 | |
| Rafael Espindola | 9f77ef0 | 2016-02-12 20:54:57 +0000 | [diff] [blame] | 439 | BitcodeFile::BitcodeFile(MemoryBufferRef M) : InputFile(BitcodeKind, M) {} |
| 440 | |
| 441 | bool BitcodeFile::classof(const InputFile *F) { |
| 442 | return F->kind() == BitcodeKind; |
| 443 | } |
| 444 | |
| Rafael Espindola | 4de44b7 | 2016-03-02 15:43:50 +0000 | [diff] [blame^] | 445 | void BitcodeFile::parse(DenseSet<StringRef> &ComdatGroups) { |
| Rafael Espindola | 9f77ef0 | 2016-02-12 20:54:57 +0000 | [diff] [blame] | 446 | LLVMContext Context; |
| 447 | ErrorOr<std::unique_ptr<IRObjectFile>> ObjOrErr = |
| 448 | IRObjectFile::create(MB, Context); |
| 449 | fatal(ObjOrErr); |
| 450 | IRObjectFile &Obj = **ObjOrErr; |
| Rafael Espindola | 4de44b7 | 2016-03-02 15:43:50 +0000 | [diff] [blame^] | 451 | const Module &M = Obj.getModule(); |
| 452 | |
| 453 | DenseSet<const Comdat *> KeptComdats; |
| 454 | for (const auto &P : M.getComdatSymbolTable()) { |
| 455 | StringRef N = Saver.save(P.first()); |
| 456 | if (ComdatGroups.insert(N).second) |
| 457 | KeptComdats.insert(&P.second); |
| 458 | } |
| 459 | |
| Rafael Espindola | 9f77ef0 | 2016-02-12 20:54:57 +0000 | [diff] [blame] | 460 | for (const BasicSymbolRef &Sym : Obj.symbols()) { |
| Rafael Espindola | 4de44b7 | 2016-03-02 15:43:50 +0000 | [diff] [blame^] | 461 | if (const GlobalValue *GV = Obj.getSymbolGV(Sym.getRawDataRefImpl())) |
| 462 | if (const Comdat *C = GV->getComdat()) |
| 463 | if (!KeptComdats.count(C)) |
| 464 | continue; |
| 465 | |
| Rafael Espindola | 9f77ef0 | 2016-02-12 20:54:57 +0000 | [diff] [blame] | 466 | SmallString<64> Name; |
| 467 | raw_svector_ostream OS(Name); |
| 468 | Sym.printName(OS); |
| 469 | StringRef NameRef = Saver.save(StringRef(Name)); |
| Rafael Espindola | 6feecec | 2016-02-19 22:50:16 +0000 | [diff] [blame] | 470 | SymbolBody *Body; |
| Rafael Espindola | 148445e | 2016-02-25 16:25:41 +0000 | [diff] [blame] | 471 | uint32_t Flags = Sym.getFlags(); |
| Rafael Espindola | 1b62553 | 2016-02-29 14:29:48 +0000 | [diff] [blame] | 472 | bool IsWeak = Flags & BasicSymbolRef::SF_Weak; |
| Rafael Espindola | 148445e | 2016-02-25 16:25:41 +0000 | [diff] [blame] | 473 | if (Flags & BasicSymbolRef::SF_Undefined) |
| Rafael Espindola | 1b62553 | 2016-02-29 14:29:48 +0000 | [diff] [blame] | 474 | Body = new (Alloc) Undefined(NameRef, IsWeak, STV_DEFAULT, false); |
| Rafael Espindola | 6feecec | 2016-02-19 22:50:16 +0000 | [diff] [blame] | 475 | else |
| Rafael Espindola | 1b62553 | 2016-02-29 14:29:48 +0000 | [diff] [blame] | 476 | Body = new (Alloc) DefinedBitcode(NameRef, IsWeak); |
| Rafael Espindola | 9f77ef0 | 2016-02-12 20:54:57 +0000 | [diff] [blame] | 477 | SymbolBodies.push_back(Body); |
| 478 | } |
| 479 | } |
| 480 | |
| Rui Ueyama | c4b6506 | 2015-10-12 15:31:09 +0000 | [diff] [blame] | 481 | template <typename T> |
| 482 | static std::unique_ptr<InputFile> createELFFileAux(MemoryBufferRef MB) { |
| 483 | std::unique_ptr<T> Ret = llvm::make_unique<T>(MB); |
| 484 | |
| 485 | if (!Config->FirstElf) |
| 486 | Config->FirstElf = Ret.get(); |
| 487 | |
| Rui Ueyama | e717a71 | 2015-10-13 16:20:50 +0000 | [diff] [blame] | 488 | if (Config->EKind == ELFNoneKind) { |
| 489 | Config->EKind = Ret->getELFKind(); |
| Rui Ueyama | c4b6506 | 2015-10-12 15:31:09 +0000 | [diff] [blame] | 490 | Config->EMachine = Ret->getEMachine(); |
| 491 | } |
| 492 | |
| 493 | return std::move(Ret); |
| 494 | } |
| 495 | |
| 496 | template <template <class> class T> |
| Rui Ueyama | 533c030 | 2016-01-06 00:09:43 +0000 | [diff] [blame] | 497 | static std::unique_ptr<InputFile> createELFFile(MemoryBufferRef MB) { |
| Rui Ueyama | d94478b | 2015-11-20 02:19:36 +0000 | [diff] [blame] | 498 | std::pair<unsigned char, unsigned char> Type = getElfArchType(MB.getBuffer()); |
| Rui Ueyama | c4b6506 | 2015-10-12 15:31:09 +0000 | [diff] [blame] | 499 | if (Type.second != ELF::ELFDATA2LSB && Type.second != ELF::ELFDATA2MSB) |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 500 | fatal("Invalid data encoding: " + MB.getBufferIdentifier()); |
| Rui Ueyama | c4b6506 | 2015-10-12 15:31:09 +0000 | [diff] [blame] | 501 | |
| 502 | if (Type.first == ELF::ELFCLASS32) { |
| 503 | if (Type.second == ELF::ELFDATA2LSB) |
| Rui Ueyama | d94478b | 2015-11-20 02:19:36 +0000 | [diff] [blame] | 504 | return createELFFileAux<T<ELF32LE>>(MB); |
| 505 | return createELFFileAux<T<ELF32BE>>(MB); |
| Rui Ueyama | c4b6506 | 2015-10-12 15:31:09 +0000 | [diff] [blame] | 506 | } |
| 507 | if (Type.first == ELF::ELFCLASS64) { |
| 508 | if (Type.second == ELF::ELFDATA2LSB) |
| Rui Ueyama | d94478b | 2015-11-20 02:19:36 +0000 | [diff] [blame] | 509 | return createELFFileAux<T<ELF64LE>>(MB); |
| 510 | return createELFFileAux<T<ELF64BE>>(MB); |
| Rui Ueyama | c4b6506 | 2015-10-12 15:31:09 +0000 | [diff] [blame] | 511 | } |
| Rui Ueyama | 64cfffd | 2016-01-28 18:40:06 +0000 | [diff] [blame] | 512 | fatal("Invalid file class: " + MB.getBufferIdentifier()); |
| Rui Ueyama | c4b6506 | 2015-10-12 15:31:09 +0000 | [diff] [blame] | 513 | } |
| 514 | |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 515 | std::unique_ptr<InputFile> elf::createObjectFile(MemoryBufferRef MB, |
| 516 | StringRef ArchiveName) { |
| Rui Ueyama | c89bff2 | 2016-02-23 18:17:11 +0000 | [diff] [blame] | 517 | using namespace sys::fs; |
| 518 | std::unique_ptr<InputFile> F; |
| 519 | if (identify_magic(MB.getBuffer()) == file_magic::bitcode) |
| 520 | F.reset(new BitcodeFile(MB)); |
| 521 | else |
| 522 | F = createELFFile<ObjectFile>(MB); |
| Rui Ueyama | 71c066d | 2016-02-02 08:22:41 +0000 | [diff] [blame] | 523 | F->ArchiveName = ArchiveName; |
| 524 | return F; |
| Rui Ueyama | 533c030 | 2016-01-06 00:09:43 +0000 | [diff] [blame] | 525 | } |
| 526 | |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 527 | std::unique_ptr<InputFile> elf::createSharedFile(MemoryBufferRef MB) { |
| Rui Ueyama | 533c030 | 2016-01-06 00:09:43 +0000 | [diff] [blame] | 528 | return createELFFile<SharedFile>(MB); |
| 529 | } |
| 530 | |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 531 | template class elf::ELFFileBase<ELF32LE>; |
| 532 | template class elf::ELFFileBase<ELF32BE>; |
| 533 | template class elf::ELFFileBase<ELF64LE>; |
| 534 | template class elf::ELFFileBase<ELF64BE>; |
| Davide Italiano | 6d328d3 | 2015-09-16 20:45:57 +0000 | [diff] [blame] | 535 | |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 536 | template class elf::ObjectFile<ELF32LE>; |
| 537 | template class elf::ObjectFile<ELF32BE>; |
| 538 | template class elf::ObjectFile<ELF64LE>; |
| 539 | template class elf::ObjectFile<ELF64BE>; |
| Rafael Espindola | f98d6d8 | 2015-09-03 20:03:54 +0000 | [diff] [blame] | 540 | |
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 541 | template class elf::SharedFile<ELF32LE>; |
| 542 | template class elf::SharedFile<ELF32BE>; |
| 543 | template class elf::SharedFile<ELF64LE>; |
| 544 | template class elf::SharedFile<ELF64BE>; |