forked from OSchip/llvm-project
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:
parent
a5a06ea11d
commit
8caf4de31f
|
@ -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.
|
@ -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:...
|
Loading…
Reference in New Issue