786 lines
28 KiB
C++
786 lines
28 KiB
C++
//===- lib/ReaderWriter/ELF/ELFFile.cpp -----------------------------------===//
|
|
//
|
|
// 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()) {
|
|
if (isIgnoredSection(§ion))
|
|
continue;
|
|
|
|
if (isMergeableStringSection(§ion)) {
|
|
_mergeStringSections.push_back(§ion);
|
|
continue;
|
|
}
|
|
|
|
if (section.sh_type == llvm::ELF::SHT_RELA) {
|
|
auto sHdr = _objFile->getSection(section.sh_info);
|
|
|
|
auto sectionName = _objFile->getSectionName(sHdr);
|
|
if (std::error_code ec = sectionName.getError())
|
|
return ec;
|
|
|
|
auto rai(_objFile->begin_rela(§ion));
|
|
auto rae(_objFile->end_rela(§ion));
|
|
|
|
_relocationAddendReferences[sHdr] = make_range(rai, rae);
|
|
totalRelocs += std::distance(rai, rae);
|
|
} else if (section.sh_type == llvm::ELF::SHT_REL) {
|
|
auto sHdr = _objFile->getSection(section.sh_info);
|
|
|
|
auto sectionName = _objFile->getSectionName(sHdr);
|
|
if (std::error_code ec = sectionName.getError())
|
|
return ec;
|
|
|
|
auto ri(_objFile->begin_rel(§ion));
|
|
auto re(_objFile->end_rel(§ion));
|
|
|
|
_relocationReferences[sHdr] = make_range(ri, re);
|
|
totalRelocs += std::distance(ri, re);
|
|
} else {
|
|
_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.
|
|
auto SymI = _objFile->begin_symbols(), SymE = _objFile->end_symbols();
|
|
|
|
// Skip over dummy sym.
|
|
if (SymI != SymE)
|
|
++SymI;
|
|
|
|
for (; SymI != SymE; ++SymI) {
|
|
const Elf_Shdr *section = _objFile->getSection(&*SymI);
|
|
|
|
auto symbolName = _objFile->getSymbolName(SymI);
|
|
if (std::error_code ec = symbolName.getError())
|
|
return ec;
|
|
|
|
if (isAbsoluteSymbol(&*SymI)) {
|
|
ELFAbsoluteAtom<ELFT> *absAtom = createAbsoluteAtom(
|
|
*symbolName, &*SymI, (int64_t)getSymbolValue(&*SymI));
|
|
addAtom(*absAtom);
|
|
_symbolToAtomMapping.insert(std::make_pair(&*SymI, absAtom));
|
|
} else if (isUndefinedSymbol(&*SymI)) {
|
|
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 (isDefinedSymbol(&*SymI)) {
|
|
_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<Elf_Sym_Iter> &symbols = i.second;
|
|
|
|
// Sort symbols by position.
|
|
std::stable_sort(symbols.begin(), symbols.end(),
|
|
[this](Elf_Sym_Iter a, Elf_Sym_Iter 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;
|
|
|
|
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 = _objFile->getSymbolName(symbol);
|
|
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);
|
|
ELFReference<ELFT> *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++) {
|
|
const Elf_Shdr *shdr = _objFile->getSection(groupMembers[i]);
|
|
ErrorOr<StringRef> sectionName = _objFile->getSectionName(shdr);
|
|
if (std::error_code ec = sectionName.getError())
|
|
return ec;
|
|
sectionNames.push_back(*sectionName);
|
|
}
|
|
const Elf_Sym *symbol = _objFile->getSymbol(section->sh_info);
|
|
const Elf_Shdr *symtab = _objFile->getSection(section->sh_link);
|
|
ErrorOr<StringRef> symbolName = _objFile->getSymbolName(symtab, symbol);
|
|
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);
|
|
ELFReference<ELFT> *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<Elf_Rela_Iter> 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,
|
|
range<Elf_Rel_Iter> rels) {
|
|
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(ri->targetSymbolIndex());
|
|
const Elf_Shdr *shdr = _objFile->getSection(symbol);
|
|
|
|
// 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) {
|
|
Elf_Sym *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");
|
|
Elf_Sym *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");
|
|
Elf_Sym *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
|