Add ELFReference class - parse relocations needed by that class and update DefinedAtom to use the class. Tweak coding style. Add testcase to check relocations. Change was signed-off-by: Michael Spencer

llvm-svn: 164027
This commit is contained in:
Sid Manning 2012-09-17 12:49:38 +00:00
parent a5a06ea11d
commit 8caf4de31f
3 changed files with 469 additions and 164 deletions

View File

@ -14,8 +14,10 @@
#include "lld/ReaderWriter/ReaderELF.h" #include "lld/ReaderWriter/ReaderELF.h"
#include "lld/Core/File.h" #include "lld/Core/File.h"
#include "lld/Core/Reference.h"
#include "llvm/ADT/ArrayRef.h" #include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/SmallString.h"
#include "llvm/ADT/StringRef.h" #include "llvm/ADT/StringRef.h"
#include "llvm/Object/ELF.h" #include "llvm/Object/ELF.h"
#include "llvm/Object/ObjectFile.h" #include "llvm/Object/ObjectFile.h"
@ -39,58 +41,133 @@ using namespace lld;
namespace { // anonymous namespace { // anonymous
// This atom class corresponds to absolute symbol
class ELFAbsoluteAtom: public AbsoluteAtom {
/// \brief Relocation References: Defined Atoms may contain
/// references that will need to be patched before
/// the executable is written.
template <llvm::support::endianness target_endianness, bool is64Bits>
class ELFReference final : public Reference {
typedef llvm::object::Elf_Rel_Impl
<target_endianness, is64Bits, false> Elf_Rel;
typedef llvm::object::Elf_Rel_Impl
<target_endianness, is64Bits, true> Elf_Rela;
public: public:
ELFAbsoluteAtom(const File &F,
llvm::StringRef N,
uint64_t V)
: OwningFile(F)
, Name(N)
, Value(V)
{}
virtual const class File &file() const { ELFReference(const Elf_Rela *rela, uint64_t offset, const Atom *target)
return OwningFile; : _target(target)
, _targetSymbolIndex(rela->getSymbol())
, _offsetInAtom(offset)
, _addend(rela->r_addend)
, _kind(rela->getType()) {}
ELFReference(const Elf_Rel *rel, uint64_t offset, const Atom *target)
: _target(target)
, _targetSymbolIndex(rel->getSymbol())
, _offsetInAtom(offset)
, _addend(0)
, _kind(rel->getType()) {}
virtual uint64_t offsetInAtom() const {
return _offsetInAtom;
} }
virtual llvm::StringRef name() const { virtual Kind kind() const {
return Name; return _kind;
} }
virtual uint64_t value() const { virtual void setKind(Kind kind) {
return Value; _kind = kind;
} }
virtual const Atom *target() const {
return _target;
}
/// \brief targetSymbolIndex: This is the symbol table index that contains
/// the target reference.
uint64_t targetSymbolIndex() const {
return _targetSymbolIndex;
}
virtual Addend addend() const {
return _addend;
}
virtual void setAddend(Addend A) {
_addend = A;
}
virtual void setTarget(const Atom *newAtom) {
_target = newAtom;
}
private: private:
const File &OwningFile; const Atom *_target;
llvm::StringRef Name; uint64_t _targetSymbolIndex;
uint64_t Value; uint64_t _offsetInAtom;
Addend _addend;
Kind _kind;
}; };
// This atom corresponds to undefined symbols. /// \brief ELFAbsoluteAtom: These atoms store symbols that are fixed to a
/// particular address. This atom has no content its address will be used by
/// the writer to fixup references that point to it.
template<llvm::support::endianness target_endianness, bool is64Bits> template<llvm::support::endianness target_endianness, bool is64Bits>
class ELFUndefinedAtom: public UndefinedAtom { class ELFAbsoluteAtom final: public AbsoluteAtom {
public:
ELFAbsoluteAtom(const File &file,
llvm::StringRef name,
uint64_t value)
: _owningFile(file)
, _name(name)
, _value(value)
{}
virtual const class File &file() const {
return _owningFile;
}
virtual llvm::StringRef name() const {
return _name;
}
virtual uint64_t value() const {
return _value;
}
private:
const File &_owningFile;
llvm::StringRef _name;
uint64_t _value;
};
/// \brief ELFUndefinedAtom: These atoms store undefined symbols and are
/// place holders that will be replaced by defined atoms later in the
/// linking process.
template<llvm::support::endianness target_endianness, bool is64Bits>
class ELFUndefinedAtom final: public UndefinedAtom {
typedef llvm::object::Elf_Sym_Impl<target_endianness, is64Bits> Elf_Sym; typedef llvm::object::Elf_Sym_Impl<target_endianness, is64Bits> Elf_Sym;
public: public:
ELFUndefinedAtom(const File &F, ELFUndefinedAtom(const File &file,
llvm::StringRef N, llvm::StringRef name,
const Elf_Sym *E) const Elf_Sym *symbol)
: OwningFile(F) : _owningFile(file)
, Name(N) , _name(name)
, Symbol(E) , _symbol(symbol)
{} {}
virtual const class File &file() const { virtual const class File &file() const {
return OwningFile; return _owningFile;
} }
virtual llvm::StringRef name() const { virtual llvm::StringRef name() const {
return Name; return _name;
} }
// FIXME What distinguishes a symbol in ELF that can help // FIXME What distinguishes a symbol in ELF that can help
@ -100,49 +177,58 @@ public:
// //
virtual CanBeNull canBeNull() const { virtual CanBeNull canBeNull() const {
if (Symbol->getBinding() == llvm::ELF::STB_WEAK) if (_symbol->getBinding() == llvm::ELF::STB_WEAK)
return CanBeNull::canBeNullAtBuildtime; return CanBeNull::canBeNullAtBuildtime;
else else
return CanBeNull::canBeNullNever; return CanBeNull::canBeNullNever;
} }
private: private:
const File &OwningFile; const File &_owningFile;
llvm::StringRef Name; llvm::StringRef _name;
const Elf_Sym *Symbol; const Elf_Sym *_symbol;
}; };
// This atom corresponds to defined symbols. /// \brief ELFDefinedAtom: This atom stores defined symbols and will contain
/// either data or code.
template<llvm::support::endianness target_endianness, bool is64Bits> template<llvm::support::endianness target_endianness, bool is64Bits>
class ELFDefinedAtom: public DefinedAtom { class ELFDefinedAtom final: public DefinedAtom {
typedef llvm::object::Elf_Sym_Impl<target_endianness, is64Bits> Elf_Sym; typedef llvm::object::Elf_Sym_Impl<target_endianness, is64Bits> Elf_Sym;
typedef llvm::object::Elf_Shdr_Impl<target_endianness, is64Bits> Elf_Shdr; typedef llvm::object::Elf_Shdr_Impl<target_endianness, is64Bits> Elf_Shdr;
public: public:
ELFDefinedAtom(const File &F, ELFDefinedAtom(const File &file,
llvm::StringRef N, llvm::StringRef symbolName,
llvm::StringRef SN, llvm::StringRef sectionName,
const Elf_Sym *E, const Elf_Sym *symbol,
const Elf_Shdr *S, const Elf_Shdr *section,
llvm::ArrayRef<uint8_t> D) llvm::ArrayRef<uint8_t> contentData,
: OwningFile(F) unsigned int referenceStart,
, SymbolName(N) unsigned int referenceEnd,
, SectionName(SN) std::vector<ELFReference
, Symbol(E) <target_endianness, is64Bits> *> &referenceList)
, Section(S)
, ContentData(D) { : _owningFile(file)
static uint64_t ordernumber = 0; , _symbolName(symbolName)
_ordinal = ++ordernumber; , _sectionName(sectionName)
, _symbol(symbol)
, _section(section)
, _contentData(contentData)
, _referenceStartIndex(referenceStart)
, _referenceEndIndex(referenceEnd)
, _referenceList(referenceList) {
static uint64_t orderNumber = 0;
_ordinal = ++orderNumber;
} }
virtual const class File &file() const { virtual const class File &file() const {
return OwningFile; return _owningFile;
} }
virtual llvm::StringRef name() const { virtual llvm::StringRef name() const {
return SymbolName; return _symbolName;
} }
virtual uint64_t ordinal() const { virtual uint64_t ordinal() const {
@ -153,18 +239,18 @@ public:
// Common symbols are not allocated in object files so // Common symbols are not allocated in object files so
// their size is zero. // their size is zero.
if ((Symbol->getType() == llvm::ELF::STT_COMMON) if ((_symbol->getType() == llvm::ELF::STT_COMMON)
|| Symbol->st_shndx == llvm::ELF::SHN_COMMON) || _symbol->st_shndx == llvm::ELF::SHN_COMMON)
return (uint64_t)0; return (uint64_t)0;
return ContentData.size(); return _contentData.size();
} }
virtual Scope scope() const { virtual Scope scope() const {
if (Symbol->st_other == llvm::ELF::STV_HIDDEN) if (_symbol->st_other == llvm::ELF::STV_HIDDEN)
return scopeLinkageUnit; return scopeLinkageUnit;
else if (Symbol->getBinding() != llvm::ELF::STB_LOCAL) else if (_symbol->getBinding() != llvm::ELF::STB_LOCAL)
return scopeGlobal; return scopeGlobal;
else else
return scopeTranslationUnit; return scopeTranslationUnit;
@ -180,11 +266,11 @@ public:
virtual Merge merge() const { virtual Merge merge() const {
if (Symbol->getBinding() == llvm::ELF::STB_WEAK) if (_symbol->getBinding() == llvm::ELF::STB_WEAK)
return mergeAsWeak; return mergeAsWeak;
if ((Symbol->getType() == llvm::ELF::STT_COMMON) if ((_symbol->getType() == llvm::ELF::STT_COMMON)
|| Symbol->st_shndx == llvm::ELF::SHN_COMMON) || _symbol->st_shndx == llvm::ELF::SHN_COMMON)
return mergeAsTentative; return mergeAsTentative;
return mergeNo; return mergeNo;
@ -192,14 +278,14 @@ public:
virtual ContentType contentType() const { virtual ContentType contentType() const {
if (Symbol->getType() == llvm::ELF::STT_FUNC) if (_symbol->getType() == llvm::ELF::STT_FUNC)
return typeCode; return typeCode;
if ((Symbol->getType() == llvm::ELF::STT_COMMON) if ((_symbol->getType() == llvm::ELF::STT_COMMON)
|| Symbol->st_shndx == llvm::ELF::SHN_COMMON) || _symbol->st_shndx == llvm::ELF::SHN_COMMON)
return typeZeroFill; return typeZeroFill;
if (Symbol->getType() == llvm::ELF::STT_OBJECT) if (_symbol->getType() == llvm::ELF::STT_OBJECT)
return typeData; return typeData;
return typeUnknown; return typeUnknown;
@ -209,25 +295,25 @@ public:
// Unallocated common symbols specify their alignment // Unallocated common symbols specify their alignment
// constraints in st_value. // constraints in st_value.
if ((Symbol->getType() == llvm::ELF::STT_COMMON) if ((_symbol->getType() == llvm::ELF::STT_COMMON)
|| Symbol->st_shndx == llvm::ELF::SHN_COMMON) { || _symbol->st_shndx == llvm::ELF::SHN_COMMON) {
return (Alignment(Symbol->st_value)); return (Alignment(_symbol->st_value));
} }
return Alignment(llvm::Log2_64(Section->sh_addralign)); return Alignment(llvm::Log2_64(_section->sh_addralign));
} }
// Do we have a choice for ELF? All symbols // Do we have a choice for ELF? All symbols
// live in explicit sections. // live in explicit sections.
virtual SectionChoice sectionChoice() const { virtual SectionChoice sectionChoice() const {
if (Symbol->st_shndx > llvm::ELF::SHN_LORESERVE) if (_symbol->st_shndx > llvm::ELF::SHN_LORESERVE)
return sectionBasedOnContent; return sectionBasedOnContent;
return sectionCustomRequired; return sectionCustomRequired;
} }
virtual llvm::StringRef customSectionName() const { virtual llvm::StringRef customSectionName() const {
return SectionName; return _sectionName;
} }
// It isn't clear that __attribute__((used)) is transmitted to // It isn't clear that __attribute__((used)) is transmitted to
@ -238,7 +324,7 @@ public:
virtual ContentPermissions permissions() const { virtual ContentPermissions permissions() const {
switch (Section->sh_type) { switch (_section->sh_type) {
// permRW_L is for sections modified by the runtime // permRW_L is for sections modified by the runtime
// loader. // loader.
case llvm::ELF::SHT_REL: case llvm::ELF::SHT_REL:
@ -247,7 +333,7 @@ public:
case llvm::ELF::SHT_DYNAMIC: case llvm::ELF::SHT_DYNAMIC:
case llvm::ELF::SHT_PROGBITS: case llvm::ELF::SHT_PROGBITS:
switch (Section->sh_flags) { switch (_section->sh_flags) {
case (llvm::ELF::SHF_ALLOC | llvm::ELF::SHF_EXECINSTR): case (llvm::ELF::SHF_ALLOC | llvm::ELF::SHF_EXECINSTR):
return permR_X; return permR_X;
@ -281,34 +367,49 @@ public:
} }
virtual llvm::ArrayRef<uint8_t> rawContent() const { virtual llvm::ArrayRef<uint8_t> rawContent() const {
return ContentData; return _contentData;
} }
virtual reference_iterator begin() const { DefinedAtom::reference_iterator begin() const {
return reference_iterator(*this, nullptr); uintptr_t index = _referenceStartIndex;
const void *it = reinterpret_cast<const void*>(index);
return reference_iterator(*this, it);
} }
virtual reference_iterator end() const { DefinedAtom::reference_iterator end() const {
return reference_iterator(*this, nullptr); uintptr_t index = _referenceEndIndex;
const void *it = reinterpret_cast<const void*>(index);
return reference_iterator(*this, it);
}
const Reference *derefIterator(const void *It) const {
uintptr_t index = reinterpret_cast<uintptr_t>(It);
assert(index >= _referenceStartIndex);
assert(index < _referenceEndIndex);
return ((_referenceList)[index]);
}
void incrementIterator(const void*& It) const {
uintptr_t index = reinterpret_cast<uintptr_t>(It);
++index;
It = reinterpret_cast<const void*>(index);
} }
private: private:
virtual const Reference *derefIterator(const void *iter) const {
return nullptr;
}
virtual void incrementIterator(const void *&iter) const {
}
const File &OwningFile; const File &_owningFile;
llvm::StringRef SymbolName; llvm::StringRef _symbolName;
llvm::StringRef SectionName; llvm::StringRef _sectionName;
const Elf_Sym *Symbol; const Elf_Sym *_symbol;
const Elf_Shdr *Section; const Elf_Shdr *_section;
// ContentData will hold the bits that make up the atom. // _contentData will hold the bits that make up the atom.
llvm::ArrayRef<uint8_t> ContentData; llvm::ArrayRef<uint8_t> _contentData;
uint64_t _ordinal; uint64_t _ordinal;
unsigned int _referenceStartIndex;
unsigned int _referenceEndIndex;
std::vector<ELFReference<target_endianness, is64Bits> *> &_referenceList;
}; };
@ -318,91 +419,152 @@ private:
template<llvm::support::endianness target_endianness, bool is64Bits> template<llvm::support::endianness target_endianness, bool is64Bits>
class FileELF: public File { class FileELF: public File {
typedef llvm::object::Elf_Sym_Impl<target_endianness, is64Bits> Elf_Sym; typedef llvm::object::Elf_Sym_Impl
typedef llvm::object::Elf_Shdr_Impl<target_endianness, is64Bits> Elf_Shdr; <target_endianness, is64Bits> Elf_Sym;
typedef llvm::object::Elf_Shdr_Impl
<target_endianness, is64Bits> Elf_Shdr;
typedef llvm::object::Elf_Rel_Impl
<target_endianness, is64Bits, false> Elf_Rel;
typedef llvm::object::Elf_Rel_Impl
<target_endianness, is64Bits, true> Elf_Rela;
public: public:
FileELF(std::unique_ptr<llvm::MemoryBuffer> MB, llvm::error_code &EC) : FileELF(std::unique_ptr<llvm::MemoryBuffer> MB, llvm::error_code &EC) :
File(MB->getBufferIdentifier()) { File(MB->getBufferIdentifier()) {
llvm::OwningPtr<llvm::object::Binary> Bin; llvm::OwningPtr<llvm::object::Binary> binaryFile;
EC = llvm::object::createBinary(MB.release(), Bin); EC = llvm::object::createBinary(MB.release(), binaryFile);
if (EC) if (EC)
return; return;
// Point Obj to correct class and bitwidth ELF object // Point Obj to correct class and bitwidth ELF object
Obj.reset(llvm::dyn_cast<llvm::object::ELFObjectFile<target_endianness, _objFile.reset(llvm::dyn_cast<llvm::object::ELFObjectFile<target_endianness,
is64Bits> >(Bin.get())); is64Bits> >(binaryFile.get()));
if (!Obj) { if (!_objFile) {
EC = make_error_code(llvm::object::object_error::invalid_file_type); EC = make_error_code(llvm::object::object_error::invalid_file_type);
return; return;
} }
Bin.take(); binaryFile.take();
std::map< const Elf_Shdr *, std::vector<const Elf_Sym *>> SectionSymbols; std::map< const Elf_Shdr *, std::vector<const Elf_Sym *>> sectionSymbols;
llvm::object::symbol_iterator it(Obj->begin_symbols()); // Handle: SHT_REL and SHT_RELA sections:
llvm::object::symbol_iterator ie(Obj->end_symbols()); // Increment over the sections, when REL/RELA section types are
// found add the contents to the RelocationReferences map.
llvm::object::section_iterator sit(_objFile->begin_sections());
llvm::object::section_iterator sie(_objFile->end_sections());
for (; sit != sie; sit.increment(EC)) {
if (EC)
return;
const Elf_Shdr *section = _objFile->getElfSection(sit);
if (section->sh_type == llvm::ELF::SHT_RELA) {
llvm::StringRef sectionName;
if ((EC = _objFile->getSectionName(section, sectionName)))
return;
// Get rid of the leading .rela so Atoms can use their own section
// name to find the relocs.
sectionName = sectionName.drop_front(5);
auto rai(_objFile->beginELFRela(section));
auto rae(_objFile->endELFRela(section));
auto &Ref = _relocationAddendRefences[sectionName];
for (; rai != rae; rai++) {
Ref.push_back(&*rai);
}
}
if (section->sh_type == llvm::ELF::SHT_REL) {
llvm::StringRef sectionName;
if ((EC = _objFile->getSectionName(section, sectionName)))
return;
// Get rid of the leading .rel so Atoms can use their own section
// name to find the relocs.
sectionName = sectionName.drop_front(4);
auto ri(_objFile->beginELFRel(section));
auto re(_objFile->endELFRel(section));
auto &Ref = _relocationReferences[sectionName];
for (; ri != re; ri++) {
Ref.push_back(&*ri);
}
}
}
// Increment over all the symbols collecting atoms and symbol
// names for later use.
llvm::object::symbol_iterator it(_objFile->begin_symbols());
llvm::object::symbol_iterator ie(_objFile->end_symbols());
for (; it != ie; it.increment(EC)) { for (; it != ie; it.increment(EC)) {
if (EC) if (EC)
return; return;
llvm::object::SectionRef SR;
llvm::object::section_iterator section(SR);
if ((EC = it->getSection(section))) if ((EC = it->getSection(sit)))
return; return;
const Elf_Shdr *Section = Obj->getElfSection(section); const Elf_Shdr *section = _objFile->getElfSection(sit);
const Elf_Sym *Symbol = Obj->getElfSymbol(it); const Elf_Sym *symbol = _objFile->getElfSymbol(it);
llvm::StringRef SymbolName; llvm::StringRef symbolName;
if ((EC = Obj->getSymbolName(Section, Symbol, SymbolName))) if ((EC = _objFile->getSymbolName(section, symbol, symbolName)))
return; return;
if (Symbol->st_shndx == llvm::ELF::SHN_ABS) { if (symbol->st_shndx == llvm::ELF::SHN_ABS) {
// Create an absolute atom. // Create an absolute atom.
AbsoluteAtoms._atoms.push_back( auto *newAtom = new (_readerStorage.Allocate
new (AtomStorage.Allocate<ELFAbsoluteAtom> ()) <ELFAbsoluteAtom<target_endianness, is64Bits> > ())
ELFAbsoluteAtom(*this, SymbolName, ELFAbsoluteAtom<target_endianness, is64Bits>
Symbol->st_value)); (*this, symbolName, symbol->st_value);
} else if (Symbol->st_shndx == llvm::ELF::SHN_UNDEF) { _absoluteAtoms._atoms.push_back(newAtom);
_symbolToAtomMapping.insert(std::make_pair(symbol, newAtom));
} else if (symbol->st_shndx == llvm::ELF::SHN_UNDEF) {
// Create an undefined atom. // Create an undefined atom.
UndefinedAtoms._atoms.push_back( auto *newAtom = new (_readerStorage.Allocate
new (AtomStorage.Allocate<ELFUndefinedAtom< <ELFUndefinedAtom<target_endianness, is64Bits> > ())
target_endianness, is64Bits>>()) ELFUndefinedAtom<target_endianness, is64Bits>
ELFUndefinedAtom<target_endianness, is64Bits> ( (*this, symbolName, symbol);
*this, SymbolName, Symbol));
_undefinedAtoms._atoms.push_back(newAtom);
_symbolToAtomMapping.insert(std::make_pair(symbol, newAtom));
} else { } else {
// This is actually a defined symbol. Add it to its section's list of // This is actually a defined symbol. Add it to its section's list of
// symbols. // symbols.
if (Symbol->getType() == llvm::ELF::STT_NOTYPE if (symbol->getType() == llvm::ELF::STT_NOTYPE
|| Symbol->getType() == llvm::ELF::STT_OBJECT || symbol->getType() == llvm::ELF::STT_OBJECT
|| Symbol->getType() == llvm::ELF::STT_FUNC || symbol->getType() == llvm::ELF::STT_FUNC
|| Symbol->getType() == llvm::ELF::STT_SECTION || symbol->getType() == llvm::ELF::STT_SECTION
|| Symbol->getType() == llvm::ELF::STT_FILE || symbol->getType() == llvm::ELF::STT_FILE
|| Symbol->getType() == llvm::ELF::STT_TLS || symbol->getType() == llvm::ELF::STT_TLS
|| Symbol->getType() == llvm::ELF::STT_COMMON || symbol->getType() == llvm::ELF::STT_COMMON
|| Symbol->st_shndx == llvm::ELF::SHN_COMMON) { || symbol->st_shndx == llvm::ELF::SHN_COMMON) {
SectionSymbols[Section].push_back(Symbol); sectionSymbols[section].push_back(symbol);
} }
else { else {
llvm::errs() << "Unable to create atom for: " << SymbolName << "\n"; llvm::errs() << "Unable to create atom for: " << symbolName << "\n";
EC = llvm::object::object_error::parse_failed; EC = llvm::object::object_error::parse_failed;
return; return;
} }
} }
} }
for (auto &i : SectionSymbols) { for (auto &i : sectionSymbols) {
auto &Symbs = i.second; auto &symbols = i.second;
llvm::StringRef SymbolName; llvm::StringRef symbolName;
llvm::StringRef SectionName; llvm::StringRef sectionName;
// Sort symbols by position. // Sort symbols by position.
std::stable_sort(Symbs.begin(), Symbs.end(), std::stable_sort(symbols.begin(), symbols.end(),
// From ReaderCOFF.cpp: // From ReaderCOFF.cpp:
// For some reason MSVC fails to allow the lambda in this context with // For some reason MSVC fails to allow the lambda in this context with
// a "illegal use of local type in type instantiation". MSVC is clearly // a "illegal use of local type in type instantiation". MSVC is clearly
@ -413,47 +575,93 @@ public:
})); }));
// i.first is the section the symbol lives in // i.first is the section the symbol lives in
for (auto si = Symbs.begin(), se = Symbs.end(); si != se; ++si) { for (auto si = symbols.begin(), se = symbols.end(); si != se; ++si) {
StringRef symbolContents; StringRef symbolContents;
if ((EC = Obj->getSectionContents(i.first, symbolContents))) if ((EC = _objFile->getSectionContents(i.first, symbolContents)))
return; return;
if ((EC = Obj->getSymbolName(i.first, *si, SymbolName))) if ((EC = _objFile->getSymbolName(i.first, *si, symbolName)))
return; return;
if ((EC = Obj->getSectionName(i.first, SectionName))) if ((EC = _objFile->getSectionName(i.first, sectionName)))
return; return;
bool IsCommon = false; bool isCommon = false;
if (((*si)->getType() == llvm::ELF::STT_COMMON) if (((*si)->getType() == llvm::ELF::STT_COMMON)
|| (*si)->st_shndx == llvm::ELF::SHN_COMMON) || (*si)->st_shndx == llvm::ELF::SHN_COMMON)
IsCommon = true; isCommon = true;
// Get the symbol's content: // Get the symbol's content:
llvm::ArrayRef<uint8_t> SymbolData; llvm::ArrayRef<uint8_t> symbolData;
uint64_t contentSize;
if (si + 1 == se) { if (si + 1 == se) {
// if this is the last symbol, take up the remaining data. // if this is the last symbol, take up the remaining data.
SymbolData = llvm::ArrayRef<uint8_t>((uint8_t *)symbolContents.data() contentSize = (isCommon) ? 0
+ (*si)->st_value, : ((i.first)->sh_size - (*si)->st_value);
(IsCommon) ? 0 :
((i.first)->sh_size - (*si)->st_value));
} }
else { else {
SymbolData = llvm::ArrayRef<uint8_t>((uint8_t *)symbolContents.data() contentSize = (isCommon) ? 0
+ (*si)->st_value, : (*(si + 1))->st_value - (*si)->st_value;
(IsCommon) ? 0 :
(*(si + 1))->st_value - (*si)->st_value);
} }
DefinedAtoms._atoms.push_back( symbolData = llvm::ArrayRef<uint8_t>((uint8_t *)symbolContents.data()
new (AtomStorage.Allocate<ELFDefinedAtom< + (*si)->st_value, contentSize);
target_endianness, is64Bits> > ())
ELFDefinedAtom<target_endianness, is64Bits> (*this,
SymbolName, SectionName, unsigned int referenceStart = _references.size();
*si, i.first, SymbolData));
// Only relocations that are inside the domain of the atom are
// added.
// Add Rela (those with r_addend) references:
for (auto &rai : _relocationAddendRefences[sectionName]) {
if ((rai->r_offset >= (*si)->st_value) &&
(rai->r_offset < (*si)->st_value+contentSize)) {
auto *ERef = new (_readerStorage.Allocate
<ELFReference<target_endianness, is64Bits> > ())
ELFReference<target_endianness, is64Bits> (
rai, rai->r_offset-(*si)->st_value, nullptr);
_references.push_back(ERef);
}
}
// Add Rel references:
for (auto &ri : _relocationReferences[sectionName]) {
if (((ri)->r_offset >= (*si)->st_value) &&
((ri)->r_offset < (*si)->st_value+contentSize)) {
auto *ERef = new (_readerStorage.Allocate
<ELFReference<target_endianness, is64Bits> > ())
ELFReference<target_endianness, is64Bits> (
(ri), (ri)->r_offset-(*si)->st_value, nullptr);
_references.push_back(ERef);
}
}
// Create the DefinedAtom and add it to the list of DefinedAtoms.
auto *newAtom = new (_readerStorage.Allocate
<ELFDefinedAtom<target_endianness, is64Bits> > ())
ELFDefinedAtom<target_endianness, is64Bits>
(*this, symbolName, sectionName,
*si, i.first, symbolData,
referenceStart, _references.size(), _references);
_definedAtoms._atoms.push_back(newAtom);
_symbolToAtomMapping.insert(std::make_pair((*si), newAtom));
} }
} }
// All the Atoms and References are created. Now update each Reference's
// target with the Atom pointer it refers to.
for (auto &ri : _references) {
const Elf_Sym *Symbol = _objFile->getElfSymbol(ri->targetSymbolIndex());
ri->setTarget(findAtom (Symbol));
}
} }
virtual void addAtom(const Atom&) { virtual void addAtom(const Atom&) {
@ -461,30 +669,49 @@ public:
} }
virtual const atom_collection<DefinedAtom> &defined() const { virtual const atom_collection<DefinedAtom> &defined() const {
return DefinedAtoms; return _definedAtoms;
} }
virtual const atom_collection<UndefinedAtom> &undefined() const { virtual const atom_collection<UndefinedAtom> &undefined() const {
return UndefinedAtoms; return _undefinedAtoms;
} }
virtual const atom_collection<SharedLibraryAtom> &sharedLibrary() const { virtual const atom_collection<SharedLibraryAtom> &sharedLibrary() const {
return SharedLibraryAtoms; return _sharedLibraryAtoms;
} }
virtual const atom_collection<AbsoluteAtom> &absolute() const { virtual const atom_collection<AbsoluteAtom> &absolute() const {
return AbsoluteAtoms; return _absoluteAtoms;
} }
Atom *findAtom(const Elf_Sym *symbol) {
return (_symbolToAtomMapping.lookup(symbol));
}
private: private:
std::unique_ptr<llvm::object::ELFObjectFile<target_endianness, is64Bits> > std::unique_ptr<llvm::object::ELFObjectFile<target_endianness, is64Bits> >
Obj; _objFile;
atom_collection_vector<DefinedAtom> DefinedAtoms; atom_collection_vector<DefinedAtom> _definedAtoms;
atom_collection_vector<UndefinedAtom> UndefinedAtoms; atom_collection_vector<UndefinedAtom> _undefinedAtoms;
atom_collection_vector<SharedLibraryAtom> SharedLibraryAtoms; atom_collection_vector<SharedLibraryAtom> _sharedLibraryAtoms;
atom_collection_vector<AbsoluteAtom> AbsoluteAtoms; atom_collection_vector<AbsoluteAtom> _absoluteAtoms;
llvm::BumpPtrAllocator AtomStorage;
/// \brief _relocationAddendRefences and _relocationReferences contain the list
/// of relocations references. In ELF, if a section named, ".text" has
/// relocations will also have a section named ".rel.text" or ".rela.text"
/// which will hold the entries. -- .rel or .rela is prepended to create
/// the SHT_REL(A) section name.
///
std::map<llvm::StringRef, std::vector<const Elf_Rela *> >
_relocationAddendRefences;
std::map<llvm::StringRef, std::vector<const Elf_Rel *> >
_relocationReferences;
std::vector<ELFReference<target_endianness, is64Bits> *> _references;
llvm::DenseMap<const Elf_Sym *, Atom *> _symbolToAtomMapping;
llvm::BumpPtrAllocator _readerStorage;
}; };
// ReaderELF is reader object that will instantiate correct FileELF // ReaderELF is reader object that will instantiate correct FileELF
@ -541,7 +768,6 @@ ReaderOptionsELF::ReaderOptionsELF() {
ReaderOptionsELF::~ReaderOptionsELF() { ReaderOptionsELF::~ReaderOptionsELF() {
} }
Reader *createReaderELF(const ReaderOptionsELF &options) { Reader *createReaderELF(const ReaderOptionsELF &options) {
return new ReaderELF(options); return new ReaderELF(options);
} }

Binary file not shown.

79
lld/test/elf/reloc.objtxt Normal file
View File

@ -0,0 +1,79 @@
RUN: lld-core -reader ELF %p/Inputs/reloc-test.elf-i386 | FileCheck %s -check-prefix ELF-i386
ELF-i386:---
ELF-i386:atoms:
ELF-i386: - name: .text
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .text
ELF-i386: - name: .data
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .data
ELF-i386: - name: .bss
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .bss
ELF-i386: - name: .rodata.str1.1
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .rodata.str1.1
ELF-i386: content: [ 68, 65, 6C, 6C, 6F, 20, 77, 6F, 72, 6C, 64, 00 ]
ELF-i386: - name: .text.startup
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .text.startup
ELF-i386: - name: main
ELF-i386: scope: global
ELF-i386: type: code
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .text.startup
ELF-i386: content: [ 55, 89, E5, 83, E4, F0, 83, EC, 10, C7, 04, 24,
ELF-i386: 00, 00, 00, 00, E8, FC, FF, FF, FF, 31, C0, C9,
ELF-i386: C3 ]
ELF-i386: fixups:
ELF-i386: - offset: 12
ELF-i386: kind: ???
ELF-i386: target: .rodata.str1.1
ELF-i386: - offset: 17
ELF-i386: kind: call32
ELF-i386: target: puts
ELF-i386: - name: .comment
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .comment
ELF-i386: content: [ 00, 47, 43, 43, 3A, 20, 28, 47, 4E, 55, 29, 20,
ELF-i386: 34, 2E, 37, 2E, 30, 00 ]
ELF-i386: - name: .note.GNU-stack
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .note.GNU-stack
ELF-i386: - name: .eh_frame
ELF-i386: type: unknown
ELF-i386: section-choice: custom-required
ELF-i386: section-name: .eh_frame
ELF-i386: content: [ 14, 00, 00, 00, 00, 00, 00, 00, 01, 7A, 52, 00,
ELF-i386: 01, 7C, 08, 01, 1B, 0C, 04, 04, 88, 01, 00, 00,
ELF-i386: 1C, 00, 00, 00, 1C, 00, 00, 00, 00, 00, 00, 00,
ELF-i386: 19, 00, 00, 00, 00, 41, 0E, 08, 85, 02, 42, 0D,
ELF-i386: 05, 55, C5, 0C, 04, 04, 00, 00 ]
ELF-i386: fixups:
ELF-i386: - offset: 32
ELF-i386: kind: call32
ELF-i386: target: .text.startup
ELF-i386: - name: puts
ELF-i386: definition: undefined
ELF-i386: - name: test.c
ELF-i386: definition: absolute
ELF-i386: value: 0x0
ELF-i386:...