| //===- lib/ReaderWriter/ELF/ELFFile.cpp -------------------------*- C++ -*-===// |
| // |
| // The LLVM Linker |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "ELFFile.h" |
| #include "FileCommon.h" |
| #include "llvm/ADT/STLExtras.h" |
| |
| namespace lld { |
| namespace elf { |
| |
| template <typename ELFT> |
| ELFFile<ELFT>::ELFFile(StringRef name, ELFLinkingContext &ctx) |
| : SimpleFile(name), _ordinal(0), _doStringsMerge(ctx.mergeCommonStrings()), |
| _useWrap(false), _ctx(ctx) { |
| setLastError(std::error_code()); |
| } |
| |
| template <typename ELFT> |
| ELFFile<ELFT>::ELFFile(std::unique_ptr<MemoryBuffer> mb, ELFLinkingContext &ctx) |
| : SimpleFile(mb->getBufferIdentifier()), _mb(std::move(mb)), _ordinal(0), |
| _doStringsMerge(ctx.mergeCommonStrings()), |
| _useWrap(ctx.wrapCalls().size()), _ctx(ctx) {} |
| |
| template <typename ELFT> |
| std::error_code ELFFile<ELFT>::isCompatible(MemoryBufferRef mb, |
| ELFLinkingContext &ctx) { |
| return elf::isCompatible<ELFT>(mb, ctx); |
| } |
| |
| template <typename ELFT> |
| Atom *ELFFile<ELFT>::findAtom(const Elf_Sym *sourceSym, |
| const Elf_Sym *targetSym) { |
| // Return the atom for targetSym if we can do so. |
| Atom *target = _symbolToAtomMapping.lookup(targetSym); |
| if (!target) |
| // Some realocations (R_ARM_V4BX) do not have a defined |
| // target. For this cases make it points to itself. |
| target = _symbolToAtomMapping.lookup(sourceSym); |
| |
| if (target->definition() != Atom::definitionRegular) |
| return target; |
| Atom::Scope scope = llvm::cast<DefinedAtom>(target)->scope(); |
| if (scope == DefinedAtom::scopeTranslationUnit) |
| return target; |
| if (!redirectReferenceUsingUndefAtom(sourceSym, targetSym)) |
| return target; |
| |
| // Otherwise, create a new undefined symbol and returns it. |
| StringRef targetName = target->name(); |
| auto it = _undefAtomsForGroupChild.find(targetName); |
| if (it != _undefAtomsForGroupChild.end()) |
| return it->getValue(); |
| auto atom = new (_readerStorage) SimpleUndefinedAtom(*this, targetName); |
| _undefAtomsForGroupChild[targetName] = atom; |
| addAtom(*atom); |
| return atom; |
| } |
| |
| template <typename ELFT> |
| ErrorOr<StringRef> ELFFile<ELFT>::getSectionName(const Elf_Shdr *shdr) const { |
| if (!shdr) |
| return StringRef(); |
| return _objFile->getSectionName(shdr); |
| } |
| |
| template <class ELFT> std::error_code ELFFile<ELFT>::doParse() { |
| std::error_code ec; |
| _objFile.reset(new llvm::object::ELFFile<ELFT>(_mb->getBuffer(), ec)); |
| if (ec) |
| return ec; |
| |
| if ((ec = createAtomsFromContext())) |
| return ec; |
| |
| // Read input sections from the input file that need to be converted to |
| // atoms |
| if ((ec = createAtomizableSections())) |
| return ec; |
| |
| // For mergeable strings, we would need to split the section into various |
| // atoms |
| if ((ec = createMergeableAtoms())) |
| return ec; |
| |
| // Create the necessary symbols that are part of the section that we |
| // created in createAtomizableSections function |
| if ((ec = createSymbolsFromAtomizableSections())) |
| return ec; |
| |
| // Create the appropriate atoms from the file |
| if ((ec = createAtoms())) |
| return ec; |
| return std::error_code(); |
| } |
| |
| template <class ELFT> Reference::KindArch ELFFile<ELFT>::kindArch() { |
| switch (_objFile->getHeader()->e_machine) { |
| case llvm::ELF::EM_X86_64: |
| return Reference::KindArch::x86_64; |
| case llvm::ELF::EM_386: |
| return Reference::KindArch::x86; |
| case llvm::ELF::EM_ARM: |
| return Reference::KindArch::ARM; |
| case llvm::ELF::EM_HEXAGON: |
| return Reference::KindArch::Hexagon; |
| case llvm::ELF::EM_MIPS: |
| return Reference::KindArch::Mips; |
| case llvm::ELF::EM_AARCH64: |
| return Reference::KindArch::AArch64; |
| } |
| llvm_unreachable("unsupported e_machine value"); |
| } |
| |
| template <class ELFT> |
| std::error_code ELFFile<ELFT>::createAtomizableSections() { |
| // Handle: SHT_REL and SHT_RELA sections: |
| // Increment over the sections, when REL/RELA section types are found add |
| // the contents to the RelocationReferences map. |
| // Record the number of relocs to guess at preallocating the buffer. |
| uint64_t totalRelocs = 0; |
| for (const Elf_Shdr §ion : _objFile->sections()) { |
| switch (section.sh_type) { |
| case llvm::ELF::SHT_SYMTAB: |
| _symtab = §ion; |
| continue; |
| case llvm::ELF::SHT_SYMTAB_SHNDX: { |
| ErrorOr<ArrayRef<Elf_Word>> tableOrErr = _objFile->getSHNDXTable(section); |
| if (std::error_code ec = tableOrErr.getError()) |
| return ec; |
| _shndxTable = *tableOrErr; |
| continue; |
| } |
| } |
| |
| if (isIgnoredSection(§ion)) |
| continue; |
| |
| if (isMergeableStringSection(§ion)) { |
| _mergeStringSections.push_back(§ion); |
| continue; |
| } |
| |
| if (section.sh_type == llvm::ELF::SHT_RELA) { |
| auto sHdrOrErr = _objFile->getSection(section.sh_info); |
| if (std::error_code ec = sHdrOrErr.getError()) |
| return ec; |
| auto sHdr = *sHdrOrErr; |
| auto rai = _objFile->rela_begin(§ion); |
| auto rae = _objFile->rela_end(§ion); |
| _relocationAddendReferences[sHdr] = make_range(rai, rae); |
| totalRelocs += std::distance(rai, rae); |
| } else if (section.sh_type == llvm::ELF::SHT_REL) { |
| auto sHdrOrErr = _objFile->getSection(section.sh_info); |
| if (std::error_code ec = sHdrOrErr.getError()) |
| return ec; |
| auto sHdr = *sHdrOrErr; |
| auto ri = _objFile->rel_begin(§ion); |
| auto re = _objFile->rel_end(§ion); |
| _relocationReferences[sHdr] = §ion; |
| totalRelocs += std::distance(ri, re); |
| } else { |
| auto sectionName = _objFile->getSectionName(§ion); |
| if (std::error_code ec = sectionName.getError()) |
| return ec; |
| _ctx.notifyInputSectionName(*sectionName); |
| _sectionSymbols[§ion]; |
| } |
| } |
| _references.reserve(totalRelocs); |
| return std::error_code(); |
| } |
| |
| template <class ELFT> std::error_code ELFFile<ELFT>::createMergeableAtoms() { |
| // Divide the section that contains mergeable strings into tokens |
| // TODO |
| // a) add resolver support to recognize multibyte chars |
| // b) Create a separate section chunk to write mergeable atoms |
| std::vector<MergeString *> tokens; |
| for (const Elf_Shdr *msi : _mergeStringSections) { |
| auto sectionName = getSectionName(msi); |
| if (std::error_code ec = sectionName.getError()) |
| return ec; |
| |
| auto sectionContents = getSectionContents(msi); |
| if (std::error_code ec = sectionContents.getError()) |
| return ec; |
| |
| StringRef secCont(reinterpret_cast<const char *>(sectionContents->begin()), |
| sectionContents->size()); |
| |
| unsigned int prev = 0; |
| for (std::size_t i = 0, e = sectionContents->size(); i != e; ++i) { |
| if ((*sectionContents)[i] == '\0') { |
| tokens.push_back(new (_readerStorage) MergeString( |
| prev, secCont.slice(prev, i + 1), msi, *sectionName)); |
| prev = i + 1; |
| } |
| } |
| } |
| |
| // Create Mergeable atoms |
| for (const MergeString *tai : tokens) { |
| ArrayRef<uint8_t> content((const uint8_t *)tai->_string.data(), |
| tai->_string.size()); |
| ELFMergeAtom<ELFT> *atom = createMergedString(tai->_sectionName, tai->_shdr, |
| content, tai->_offset); |
| atom->setOrdinal(++_ordinal); |
| addAtom(*atom); |
| _mergeAtoms.push_back(atom); |
| } |
| return std::error_code(); |
| } |
| |
| template <class ELFT> |
| std::error_code ELFFile<ELFT>::createSymbolsFromAtomizableSections() { |
| // Increment over all the symbols collecting atoms and symbol names for |
| // later use. |
| if (!_symtab) |
| return std::error_code(); |
| |
| ErrorOr<StringRef> strTableOrErr = |
| _objFile->getStringTableForSymtab(*_symtab); |
| if (std::error_code ec = strTableOrErr.getError()) |
| return ec; |
| StringRef strTable = *strTableOrErr; |
| |
| auto SymI = _objFile->symbol_begin(_symtab), |
| SymE = _objFile->symbol_end(_symtab); |
| // Skip over dummy sym. |
| ++SymI; |
| |
| for (; SymI != SymE; ++SymI) { |
| ErrorOr<const Elf_Shdr *> section = |
| _objFile->getSection(SymI, _symtab, _shndxTable); |
| if (std::error_code ec = section.getError()) |
| return ec; |
| |
| auto symbolName = SymI->getName(strTable); |
| if (std::error_code ec = symbolName.getError()) |
| return ec; |
| |
| if (SymI->isAbsolute()) { |
| ELFAbsoluteAtom<ELFT> *absAtom = createAbsoluteAtom( |
| *symbolName, &*SymI, (int64_t)getSymbolValue(&*SymI)); |
| addAtom(*absAtom); |
| _symbolToAtomMapping.insert(std::make_pair(&*SymI, absAtom)); |
| } else if (SymI->isUndefined()) { |
| if (_useWrap && |
| (_wrapSymbolMap.find(*symbolName) != _wrapSymbolMap.end())) { |
| auto wrapAtom = _wrapSymbolMap.find(*symbolName); |
| _symbolToAtomMapping.insert( |
| std::make_pair(&*SymI, wrapAtom->getValue())); |
| continue; |
| } |
| ELFUndefinedAtom<ELFT> *undefAtom = |
| createUndefinedAtom(*symbolName, &*SymI); |
| addAtom(*undefAtom); |
| _symbolToAtomMapping.insert(std::make_pair(&*SymI, undefAtom)); |
| } else if (isCommonSymbol(&*SymI)) { |
| ELFCommonAtom<ELFT> *commonAtom = createCommonAtom(*symbolName, &*SymI); |
| commonAtom->setOrdinal(++_ordinal); |
| addAtom(*commonAtom); |
| _symbolToAtomMapping.insert(std::make_pair(&*SymI, commonAtom)); |
| } else if (SymI->isDefined()) { |
| _sectionSymbols[*section].push_back(SymI); |
| } else { |
| llvm::errs() << "Unable to create atom for: " << *symbolName << "\n"; |
| return llvm::object::object_error::parse_failed; |
| } |
| } |
| |
| return std::error_code(); |
| } |
| |
| template <class ELFT> std::error_code ELFFile<ELFT>::createAtoms() { |
| // Holds all the atoms that are part of the section. They are the targets of |
| // the kindGroupChild reference. |
| llvm::StringMap<std::vector<ELFDefinedAtom<ELFT> *>> atomsForSection; |
| |
| // Contains a list of comdat sections for a group. |
| for (auto &i : _sectionSymbols) { |
| const Elf_Shdr *section = i.first; |
| std::vector<const Elf_Sym *> &symbols = i.second; |
| |
| // Sort symbols by position. |
| std::stable_sort(symbols.begin(), symbols.end(), |
| [this](const Elf_Sym *a, const Elf_Sym *b) { |
| return getSymbolValue(&*a) < getSymbolValue(&*b); |
| }); |
| |
| ErrorOr<StringRef> sectionName = this->getSectionName(section); |
| if (std::error_code ec = sectionName.getError()) |
| return ec; |
| |
| auto sectionContents = getSectionContents(section); |
| if (std::error_code ec = sectionContents.getError()) |
| return ec; |
| |
| // SHT_GROUP sections are handled in the following loop. |
| if (isGroupSection(section)) |
| continue; |
| |
| bool addAtoms = (!isGnuLinkOnceSection(*sectionName) && |
| !isSectionMemberOfGroup(section)); |
| |
| if (handleSectionWithNoSymbols(section, symbols)) { |
| ELFDefinedAtom<ELFT> *newAtom = |
| createSectionAtom(section, *sectionName, *sectionContents); |
| newAtom->setOrdinal(++_ordinal); |
| if (addAtoms) |
| addAtom(*newAtom); |
| else |
| atomsForSection[*sectionName].push_back(newAtom); |
| continue; |
| } |
| |
| ELFDefinedAtom<ELFT> *previousAtom = nullptr; |
| ELFReference<ELFT> *anonFollowedBy = nullptr; |
| |
| if (!_symtab) |
| continue; |
| ErrorOr<StringRef> strTableOrErr = |
| _objFile->getStringTableForSymtab(*_symtab); |
| if (std::error_code ec = strTableOrErr.getError()) |
| return ec; |
| StringRef strTable = *strTableOrErr; |
| for (auto si = symbols.begin(), se = symbols.end(); si != se; ++si) { |
| auto symbol = *si; |
| StringRef symbolName = ""; |
| if (symbol->getType() != llvm::ELF::STT_SECTION) { |
| auto symName = symbol->getName(strTable); |
| if (std::error_code ec = symName.getError()) |
| return ec; |
| symbolName = *symName; |
| } |
| |
| uint64_t contentSize = symbolContentSize( |
| section, &*symbol, (si + 1 == se) ? nullptr : &**(si + 1)); |
| |
| // Check to see if we need to add the FollowOn Reference |
| ELFReference<ELFT> *followOn = nullptr; |
| if (previousAtom) { |
| // Replace the followon atom with the anonymous atom that we created, |
| // so that the next symbol that we create is a followon from the |
| // anonymous atom. |
| if (anonFollowedBy) { |
| followOn = anonFollowedBy; |
| } else { |
| followOn = new (_readerStorage) |
| ELFReference<ELFT>(Reference::kindLayoutAfter); |
| previousAtom->addReference(followOn); |
| } |
| } |
| |
| ArrayRef<uint8_t> symbolData((const uint8_t *)sectionContents->data() + |
| getSymbolValue(&*symbol), |
| contentSize); |
| |
| // If the linker finds that a section has global atoms that are in a |
| // mergeable section, treat them as defined atoms as they shouldn't be |
| // merged away as well as these symbols have to be part of symbol |
| // resolution |
| if (isMergeableStringSection(section)) { |
| if (symbol->getBinding() != llvm::ELF::STB_GLOBAL) |
| continue; |
| ELFDefinedAtom<ELFT> *atom = createDefinedAtom( |
| symbolName, *sectionName, &**si, section, symbolData, |
| _references.size(), _references.size(), _references); |
| atom->setOrdinal(++_ordinal); |
| if (addAtoms) |
| addAtom(*atom); |
| else |
| atomsForSection[*sectionName].push_back(atom); |
| continue; |
| } |
| |
| // Don't allocate content to a weak symbol, as they may be merged away. |
| // Create an anonymous atom to hold the data. |
| ELFDefinedAtom<ELFT> *anonAtom = nullptr; |
| anonFollowedBy = nullptr; |
| if (symbol->getBinding() == llvm::ELF::STB_WEAK) { |
| // Create anonymous new non-weak ELF symbol that holds the symbol |
| // data. |
| auto sym = new (_readerStorage) Elf_Sym(*symbol); |
| sym->setBinding(llvm::ELF::STB_GLOBAL); |
| anonAtom = createDefinedAtomAndAssignRelocations( |
| "", *sectionName, sym, section, symbolData, *sectionContents); |
| symbolData = ArrayRef<uint8_t>(); |
| |
| // If this is the last atom, let's not create a followon reference. |
| if (anonAtom && (si + 1) != se) { |
| anonFollowedBy = new (_readerStorage) |
| ELFReference<ELFT>(Reference::kindLayoutAfter); |
| anonAtom->addReference(anonFollowedBy); |
| } |
| } |
| |
| ELFDefinedAtom<ELFT> *newAtom = createDefinedAtomAndAssignRelocations( |
| symbolName, *sectionName, &*symbol, section, symbolData, |
| *sectionContents); |
| newAtom->setOrdinal(++_ordinal); |
| |
| // If the atom was a weak symbol, let's create a followon reference to |
| // the anonymous atom that we created. |
| if (anonAtom) |
| createEdge(newAtom, anonAtom, Reference::kindLayoutAfter); |
| |
| if (previousAtom) { |
| // Set the followon atom to the weak atom that we have created, so |
| // that they would alias when the file gets written. |
| followOn->setTarget(anonAtom ? anonAtom : newAtom); |
| } |
| |
| // The previous atom is always the atom created before unless the atom |
| // is a weak atom. |
| previousAtom = anonAtom ? anonAtom : newAtom; |
| |
| if (addAtoms) |
| addAtom(*newAtom); |
| else |
| atomsForSection[*sectionName].push_back(newAtom); |
| |
| _symbolToAtomMapping.insert(std::make_pair(&*symbol, newAtom)); |
| if (anonAtom) { |
| anonAtom->setOrdinal(++_ordinal); |
| if (addAtoms) |
| addAtom(*anonAtom); |
| else |
| atomsForSection[*sectionName].push_back(anonAtom); |
| } |
| } |
| } |
| |
| for (auto &i : _sectionSymbols) |
| if (std::error_code ec = handleSectionGroup(i.first, atomsForSection)) |
| return ec; |
| for (auto &i : _sectionSymbols) |
| if (std::error_code ec = handleGnuLinkOnceSection(i.first, atomsForSection)) |
| return ec; |
| |
| updateReferences(); |
| return std::error_code(); |
| } |
| |
| template <class ELFT> |
| std::error_code ELFFile<ELFT>::handleGnuLinkOnceSection( |
| const Elf_Shdr *section, |
| llvm::StringMap<std::vector<ELFDefinedAtom<ELFT> *>> &atomsForSection) { |
| ErrorOr<StringRef> sectionName = this->getSectionName(section); |
| if (std::error_code ec = sectionName.getError()) |
| return ec; |
| if (!isGnuLinkOnceSection(*sectionName)) |
| return std::error_code(); |
| |
| unsigned int referenceStart = _references.size(); |
| std::vector<ELFReference<ELFT> *> refs; |
| for (auto ha : atomsForSection[*sectionName]) { |
| _groupChild[ha->symbol()] = std::make_pair(*sectionName, section); |
| auto *ref = |
| new (_readerStorage) ELFReference<ELFT>(Reference::kindGroupChild); |
| ref->setTarget(ha); |
| refs.push_back(ref); |
| } |
| atomsForSection[*sectionName].clear(); |
| // Create a gnu linkonce atom. |
| ELFDefinedAtom<ELFT> *atom = createDefinedAtom( |
| *sectionName, *sectionName, nullptr, section, ArrayRef<uint8_t>(), |
| referenceStart, _references.size(), _references); |
| atom->setOrdinal(++_ordinal); |
| addAtom(*atom); |
| for (auto reference : refs) |
| atom->addReference(reference); |
| return std::error_code(); |
| } |
| |
| template <class ELFT> |
| std::error_code ELFFile<ELFT>::handleSectionGroup( |
| const Elf_Shdr *section, |
| llvm::StringMap<std::vector<ELFDefinedAtom<ELFT> *>> &atomsForSection) { |
| ErrorOr<StringRef> sectionName = this->getSectionName(section); |
| if (std::error_code ec = sectionName.getError()) |
| return ec; |
| if (!isGroupSection(section)) |
| return std::error_code(); |
| |
| auto sectionContents = getSectionContents(section); |
| if (std::error_code ec = sectionContents.getError()) |
| return ec; |
| |
| // A section of type SHT_GROUP defines a grouping of sections. The |
| // name of a symbol from one of the containing object's symbol tables |
| // provides a signature for the section group. The section header of |
| // the SHT_GROUP section specifies the identifying symbol entry, as |
| // described: the sh_link member contains the section header index of |
| // the symbol table section that contains the entry. The sh_info |
| // member contains the symbol table index of the identifying entry. |
| // The sh_flags member of the section header contains 0. The name of |
| // the section (sh_name) is not specified. |
| std::vector<StringRef> sectionNames; |
| const Elf_Word *groupMembers = |
| reinterpret_cast<const Elf_Word *>(sectionContents->data()); |
| const size_t count = section->sh_size / sizeof(Elf_Word); |
| for (size_t i = 1; i < count; i++) { |
| ErrorOr<const Elf_Shdr *> shdr = _objFile->getSection(groupMembers[i]); |
| if (std::error_code ec = shdr.getError()) |
| return ec; |
| ErrorOr<StringRef> sectionName = _objFile->getSectionName(*shdr); |
| if (std::error_code ec = sectionName.getError()) |
| return ec; |
| sectionNames.push_back(*sectionName); |
| } |
| ErrorOr<const Elf_Shdr *> symtab = _objFile->getSection(section->sh_link); |
| if (std::error_code ec = symtab.getError()) |
| return ec; |
| const Elf_Sym *symbol = _objFile->getSymbol(*symtab, section->sh_info); |
| ErrorOr<const Elf_Shdr *> strtab_sec = |
| _objFile->getSection((*symtab)->sh_link); |
| if (std::error_code ec = strtab_sec.getError()) |
| return ec; |
| ErrorOr<StringRef> strtab_or_err = _objFile->getStringTable(*strtab_sec); |
| if (std::error_code ec = strtab_or_err.getError()) |
| return ec; |
| StringRef strtab = *strtab_or_err; |
| ErrorOr<StringRef> symbolName = symbol->getName(strtab); |
| if (std::error_code ec = symbolName.getError()) |
| return ec; |
| |
| unsigned int referenceStart = _references.size(); |
| std::vector<ELFReference<ELFT> *> refs; |
| for (auto name : sectionNames) { |
| for (auto ha : atomsForSection[name]) { |
| _groupChild[ha->symbol()] = std::make_pair(*symbolName, section); |
| auto *ref = |
| new (_readerStorage) ELFReference<ELFT>(Reference::kindGroupChild); |
| ref->setTarget(ha); |
| refs.push_back(ref); |
| } |
| atomsForSection[name].clear(); |
| } |
| |
| // Create an atom for comdat signature. |
| ELFDefinedAtom<ELFT> *atom = createDefinedAtom( |
| *symbolName, *sectionName, nullptr, section, ArrayRef<uint8_t>(), |
| referenceStart, _references.size(), _references); |
| atom->setOrdinal(++_ordinal); |
| addAtom(*atom); |
| for (auto reference : refs) |
| atom->addReference(reference); |
| return std::error_code(); |
| } |
| |
| template <class ELFT> std::error_code ELFFile<ELFT>::createAtomsFromContext() { |
| if (!_useWrap) |
| return std::error_code(); |
| // Steps: |
| // a) Create an undefined atom for the symbol specified by the --wrap option, |
| // as that may be needed to be pulled from an archive. |
| // b) Create an undefined atom for __wrap_<symbolname>. |
| // c) All references to the symbol specified by wrap should point to |
| // __wrap_<symbolname> |
| // d) All references to __real_symbol should point to the <symbol> |
| for (auto &wrapsym : _ctx.wrapCalls()) { |
| StringRef wrapStr = wrapsym.getKey(); |
| // Create a undefined symbol fror the wrap symbol. |
| UndefinedAtom *wrapSymAtom = |
| new (_readerStorage) SimpleUndefinedAtom(*this, wrapStr); |
| StringRef wrapCallSym = |
| _ctx.allocateString((llvm::Twine("__wrap_") + wrapStr).str()); |
| StringRef realCallSym = |
| _ctx.allocateString((llvm::Twine("__real_") + wrapStr).str()); |
| UndefinedAtom *wrapCallAtom = |
| new (_readerStorage) SimpleUndefinedAtom(*this, wrapCallSym); |
| // Create maps, when there is call to sym, it should point to wrapCallSym. |
| _wrapSymbolMap.insert(std::make_pair(wrapStr, wrapCallAtom)); |
| // Whenever there is a reference to realCall it should point to the symbol |
| // created for each wrap usage. |
| _wrapSymbolMap.insert(std::make_pair(realCallSym, wrapSymAtom)); |
| addAtom(*wrapSymAtom); |
| addAtom(*wrapCallAtom); |
| } |
| return std::error_code(); |
| } |
| |
| template <class ELFT> |
| ELFDefinedAtom<ELFT> *ELFFile<ELFT>::createDefinedAtomAndAssignRelocations( |
| StringRef symbolName, StringRef sectionName, const Elf_Sym *symbol, |
| const Elf_Shdr *section, ArrayRef<uint8_t> symContent, |
| ArrayRef<uint8_t> secContent) { |
| unsigned int referenceStart = _references.size(); |
| |
| // Add Rela (those with r_addend) references: |
| auto rari = _relocationAddendReferences.find(section); |
| if (rari != _relocationAddendReferences.end()) |
| createRelocationReferences(symbol, symContent, rari->second); |
| |
| // Add Rel references. |
| auto rri = _relocationReferences.find(section); |
| if (rri != _relocationReferences.end()) |
| createRelocationReferences(symbol, symContent, secContent, rri->second); |
| |
| // Create the DefinedAtom and add it to the list of DefinedAtoms. |
| return createDefinedAtom(symbolName, sectionName, symbol, section, symContent, |
| referenceStart, _references.size(), _references); |
| } |
| |
| template <class ELFT> |
| void ELFFile<ELFT>::createRelocationReferences(const Elf_Sym *symbol, |
| ArrayRef<uint8_t> content, |
| range<const Elf_Rela *> rels) { |
| bool isMips64EL = _objFile->isMips64EL(); |
| const auto symValue = getSymbolValue(symbol); |
| for (const auto &rel : rels) { |
| if (rel.r_offset < symValue || symValue + content.size() <= rel.r_offset) |
| continue; |
| auto elfRelocation = new (_readerStorage) |
| ELFReference<ELFT>(&rel, rel.r_offset - symValue, kindArch(), |
| rel.getType(isMips64EL), rel.getSymbol(isMips64EL)); |
| addReferenceToSymbol(elfRelocation, symbol); |
| _references.push_back(elfRelocation); |
| } |
| } |
| |
| template <class ELFT> |
| void ELFFile<ELFT>::createRelocationReferences(const Elf_Sym *symbol, |
| ArrayRef<uint8_t> symContent, |
| ArrayRef<uint8_t> secContent, |
| const Elf_Shdr *relSec) { |
| auto rels = _objFile->rels(relSec); |
| bool isMips64EL = _objFile->isMips64EL(); |
| const auto symValue = getSymbolValue(symbol); |
| for (const auto &rel : rels) { |
| if (rel.r_offset < symValue || symValue + symContent.size() <= rel.r_offset) |
| continue; |
| auto elfRelocation = new (_readerStorage) |
| ELFReference<ELFT>(rel.r_offset - symValue, kindArch(), |
| rel.getType(isMips64EL), rel.getSymbol(isMips64EL)); |
| Reference::Addend addend = getInitialAddend(symContent, symValue, rel); |
| elfRelocation->setAddend(addend); |
| addReferenceToSymbol(elfRelocation, symbol); |
| _references.push_back(elfRelocation); |
| } |
| } |
| |
| template <class ELFT> |
| void ELFFile<ELFT>::updateReferenceForMergeStringAccess(ELFReference<ELFT> *ref, |
| const Elf_Sym *symbol, |
| const Elf_Shdr *shdr) { |
| // If the target atom is mergeable strefng atom, the atom might have been |
| // merged with other atom having the same contents. Try to find the |
| // merged one if that's the case. |
| int64_t addend = ref->addend(); |
| if (addend < 0) |
| addend = 0; |
| |
| const MergeSectionKey ms = {shdr, addend}; |
| auto msec = _mergedSectionMap.find(ms); |
| if (msec != _mergedSectionMap.end()) { |
| ref->setTarget(msec->second); |
| return; |
| } |
| |
| // The target atom was not merged. Mergeable atoms are not in |
| // _symbolToAtomMapping, so we cannot find it by calling findAtom(). We |
| // instead call findMergeAtom(). |
| if (symbol->getType() != llvm::ELF::STT_SECTION) |
| addend = getSymbolValue(symbol) + addend; |
| ELFMergeAtom<ELFT> *mergedAtom = findMergeAtom(shdr, addend); |
| ref->setOffset(addend - mergedAtom->offset()); |
| ref->setAddend(0); |
| ref->setTarget(mergedAtom); |
| } |
| |
| template <class ELFT> void ELFFile<ELFT>::updateReferences() { |
| for (auto &ri : _references) { |
| if (ri->kindNamespace() != Reference::KindNamespace::ELF) |
| continue; |
| const Elf_Sym *symbol = |
| _objFile->getSymbol(_symtab, ri->targetSymbolIndex()); |
| ErrorOr<const Elf_Shdr *> shdr = |
| _objFile->getSection(symbol, _symtab, _shndxTable); |
| |
| // If the atom is not in mergeable string section, the target atom is |
| // simply that atom. |
| if (isMergeableStringSection(*shdr)) |
| updateReferenceForMergeStringAccess(ri, symbol, *shdr); |
| else |
| ri->setTarget(findAtom(findSymbolForReference(ri), symbol)); |
| } |
| } |
| |
| template <class ELFT> |
| bool ELFFile<ELFT>::isIgnoredSection(const Elf_Shdr *section) { |
| switch (section->sh_type) { |
| case llvm::ELF::SHT_NULL: |
| case llvm::ELF::SHT_STRTAB: |
| case llvm::ELF::SHT_SYMTAB: |
| case llvm::ELF::SHT_SYMTAB_SHNDX: |
| return true; |
| default: |
| break; |
| } |
| return false; |
| } |
| |
| template <class ELFT> |
| bool ELFFile<ELFT>::isMergeableStringSection(const Elf_Shdr *section) { |
| if (_doStringsMerge && section) { |
| int64_t sectionFlags = section->sh_flags; |
| sectionFlags &= ~llvm::ELF::SHF_ALLOC; |
| // Mergeable string sections have both SHF_MERGE and SHF_STRINGS flags |
| // set. sh_entsize is the size of each character which is normally 1. |
| if ((section->sh_entsize < 2) && |
| (sectionFlags == (llvm::ELF::SHF_MERGE | llvm::ELF::SHF_STRINGS))) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| template <class ELFT> |
| ELFDefinedAtom<ELFT> * |
| ELFFile<ELFT>::createSectionAtom(const Elf_Shdr *section, StringRef sectionName, |
| ArrayRef<uint8_t> content) { |
| auto *sym = new (_readerStorage) Elf_Sym; |
| sym->st_name = 0; |
| sym->setBindingAndType(llvm::ELF::STB_LOCAL, llvm::ELF::STT_SECTION); |
| sym->st_other = 0; |
| sym->st_shndx = 0; |
| sym->st_value = 0; |
| sym->st_size = 0; |
| auto *newAtom = createDefinedAtomAndAssignRelocations( |
| "", sectionName, sym, section, content, content); |
| newAtom->setOrdinal(++_ordinal); |
| return newAtom; |
| } |
| |
| template <class ELFT> |
| uint64_t ELFFile<ELFT>::symbolContentSize(const Elf_Shdr *section, |
| const Elf_Sym *symbol, |
| const Elf_Sym *nextSymbol) { |
| const auto symValue = getSymbolValue(symbol); |
| // if this is the last symbol, take up the remaining data. |
| return nextSymbol ? getSymbolValue(nextSymbol) - symValue |
| : section->sh_size - symValue; |
| } |
| |
| template <class ELFT> |
| void ELFFile<ELFT>::createEdge(ELFDefinedAtom<ELFT> *from, |
| ELFDefinedAtom<ELFT> *to, uint32_t edgeKind) { |
| auto reference = new (_readerStorage) ELFReference<ELFT>(edgeKind); |
| reference->setTarget(to); |
| from->addReference(reference); |
| } |
| |
| /// Does the atom need to be redirected using a separate undefined atom? |
| template <class ELFT> |
| bool ELFFile<ELFT>::redirectReferenceUsingUndefAtom( |
| const Elf_Sym *sourceSymbol, const Elf_Sym *targetSymbol) const { |
| auto groupChildTarget = _groupChild.find(targetSymbol); |
| |
| // If the reference is not to a group child atom, there is no need to redirect |
| // using a undefined atom. Its also not needed if the source and target are |
| // from the same section. |
| if ((groupChildTarget == _groupChild.end()) || |
| (sourceSymbol->st_shndx == targetSymbol->st_shndx)) |
| return false; |
| |
| auto groupChildSource = _groupChild.find(sourceSymbol); |
| |
| // If the source symbol is not in a group, use a undefined symbol too. |
| if (groupChildSource == _groupChild.end()) |
| return true; |
| |
| // If the source and child are from the same group, we dont need the |
| // relocation to go through a undefined symbol. |
| if (groupChildSource->second.second == groupChildTarget->second.second) |
| return false; |
| return true; |
| } |
| |
| template <class ELFT> |
| void RuntimeFile<ELFT>::addAbsoluteAtom(StringRef symbolName, bool isHidden) { |
| assert(!symbolName.empty() && "AbsoluteAtoms must have a name"); |
| auto *sym = new (this->_readerStorage) Elf_Sym; |
| sym->st_name = 0; |
| sym->st_value = 0; |
| sym->st_shndx = llvm::ELF::SHN_ABS; |
| sym->setBindingAndType(llvm::ELF::STB_GLOBAL, llvm::ELF::STT_OBJECT); |
| if (isHidden) |
| sym->setVisibility(llvm::ELF::STV_HIDDEN); |
| else |
| sym->setVisibility(llvm::ELF::STV_DEFAULT); |
| sym->st_size = 0; |
| ELFAbsoluteAtom<ELFT> *atom = this->createAbsoluteAtom(symbolName, sym, -1); |
| this->addAtom(*atom); |
| } |
| |
| template <class ELFT> |
| void RuntimeFile<ELFT>::addUndefinedAtom(StringRef symbolName) { |
| assert(!symbolName.empty() && "UndefinedAtoms must have a name"); |
| auto *sym = new (this->_readerStorage) Elf_Sym; |
| sym->st_name = 0; |
| sym->st_value = 0; |
| sym->st_shndx = llvm::ELF::SHN_UNDEF; |
| sym->setBindingAndType(llvm::ELF::STB_GLOBAL, llvm::ELF::STT_NOTYPE); |
| sym->setVisibility(llvm::ELF::STV_DEFAULT); |
| sym->st_size = 0; |
| ELFUndefinedAtom<ELFT> *atom = this->createUndefinedAtom(symbolName, sym); |
| this->addAtom(*atom); |
| } |
| |
| template class ELFFile<ELF32LE>; |
| template class ELFFile<ELF32BE>; |
| template class ELFFile<ELF64LE>; |
| template class ELFFile<ELF64BE>; |
| |
| template class RuntimeFile<ELF32LE>; |
| template class RuntimeFile<ELF32BE>; |
| template class RuntimeFile<ELF64LE>; |
| template class RuntimeFile<ELF64BE>; |
| |
| } // end namespace elf |
| } // end namespace lld |