2013-02-12 07:03:35 +08:00
|
|
|
//===- lib/ReaderWriter/ELF/DynamicFile.h ---------------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Linker
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLD_READER_WRITER_ELF_DYNAMIC_FILE_H
|
|
|
|
#define LLD_READER_WRITER_ELF_DYNAMIC_FILE_H
|
|
|
|
|
2013-04-11 10:56:30 +08:00
|
|
|
#include "Atoms.h"
|
|
|
|
|
2013-02-12 07:03:35 +08:00
|
|
|
#include "lld/Core/SharedLibraryFile.h"
|
2013-08-07 06:31:59 +08:00
|
|
|
#include "lld/ReaderWriter/ELFLinkingContext.h"
|
2013-02-12 07:03:35 +08:00
|
|
|
|
|
|
|
#include "llvm/Object/ELF.h"
|
|
|
|
#include "llvm/Support/Path.h"
|
|
|
|
|
|
|
|
#include <unordered_map>
|
|
|
|
|
|
|
|
namespace lld {
|
|
|
|
namespace elf {
|
2014-01-27 09:02:03 +08:00
|
|
|
template <class ELFT> class DynamicFile : public SharedLibraryFile {
|
2013-02-12 07:03:35 +08:00
|
|
|
public:
|
2013-12-20 15:48:29 +08:00
|
|
|
static ErrorOr<std::unique_ptr<DynamicFile>>
|
2014-01-27 07:57:20 +08:00
|
|
|
create(std::unique_ptr<llvm::MemoryBuffer> mb, bool useShlibUndefines);
|
2013-02-12 07:03:35 +08:00
|
|
|
|
|
|
|
virtual const atom_collection<DefinedAtom> &defined() const {
|
|
|
|
return _definedAtoms;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const atom_collection<UndefinedAtom> &undefined() const {
|
|
|
|
return _undefinedAtoms;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const atom_collection<SharedLibraryAtom> &sharedLibrary() const {
|
|
|
|
return _sharedLibraryAtoms;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const atom_collection<AbsoluteAtom> &absolute() const {
|
|
|
|
return _absoluteAtoms;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const SharedLibraryAtom *exports(StringRef name,
|
|
|
|
bool dataSymbolOnly) const {
|
|
|
|
assert(!dataSymbolOnly && "Invalid option for ELF exports!");
|
|
|
|
// See if we have the symbol.
|
|
|
|
auto sym = _nameToSym.find(name);
|
|
|
|
if (sym == _nameToSym.end())
|
|
|
|
return nullptr;
|
|
|
|
// Have we already created a SharedLibraryAtom for it?
|
|
|
|
if (sym->second._atom)
|
|
|
|
return sym->second._atom;
|
|
|
|
// Create a SharedLibraryAtom for this symbol.
|
|
|
|
return sym->second._atom = new (_alloc) ELFDynamicAtom<ELFT>(
|
|
|
|
*this, name, _soname, sym->second._symbol);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2013-12-20 15:48:29 +08:00
|
|
|
DynamicFile(StringRef name) : SharedLibraryFile(name) {}
|
2013-02-12 07:03:35 +08:00
|
|
|
|
2013-02-20 05:04:30 +08:00
|
|
|
mutable llvm::BumpPtrAllocator _alloc;
|
2013-08-09 06:26:50 +08:00
|
|
|
std::unique_ptr<llvm::object::ELFFile<ELFT>> _objFile;
|
2013-02-12 07:03:35 +08:00
|
|
|
atom_collection_vector<DefinedAtom> _definedAtoms;
|
|
|
|
atom_collection_vector<UndefinedAtom> _undefinedAtoms;
|
|
|
|
atom_collection_vector<SharedLibraryAtom> _sharedLibraryAtoms;
|
|
|
|
atom_collection_vector<AbsoluteAtom> _absoluteAtoms;
|
|
|
|
/// \brief DT_SONAME
|
|
|
|
StringRef _soname;
|
|
|
|
|
|
|
|
struct SymAtomPair {
|
2013-02-20 05:04:30 +08:00
|
|
|
SymAtomPair() : _symbol(nullptr), _atom(nullptr) {}
|
2013-08-09 06:26:50 +08:00
|
|
|
const typename llvm::object::ELFFile<ELFT>::Elf_Sym *_symbol;
|
2013-02-20 05:04:30 +08:00
|
|
|
const SharedLibraryAtom *_atom;
|
2013-02-12 07:03:35 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
mutable std::unordered_map<StringRef, SymAtomPair> _nameToSym;
|
|
|
|
};
|
2014-01-27 07:57:20 +08:00
|
|
|
|
|
|
|
template <class ELFT>
|
|
|
|
ErrorOr<std::unique_ptr<DynamicFile<ELFT>>>
|
|
|
|
DynamicFile<ELFT>::create(std::unique_ptr<llvm::MemoryBuffer> mb,
|
|
|
|
bool useShlibUndefines) {
|
|
|
|
std::unique_ptr<DynamicFile> file(new DynamicFile(mb->getBufferIdentifier()));
|
|
|
|
|
|
|
|
error_code ec;
|
|
|
|
file->_objFile.reset(new llvm::object::ELFFile<ELFT>(mb.release(), ec));
|
|
|
|
|
|
|
|
if (ec)
|
|
|
|
return ec;
|
|
|
|
|
|
|
|
llvm::object::ELFFile<ELFT> &obj = *file->_objFile;
|
|
|
|
|
|
|
|
file->_soname = obj.getLoadName();
|
|
|
|
if (file->_soname.empty())
|
|
|
|
file->_soname = llvm::sys::path::filename(file->path());
|
|
|
|
|
|
|
|
// Create a map from names to dynamic symbol table entries.
|
|
|
|
// TODO: This should use the object file's build in hash table instead if
|
|
|
|
// it exists.
|
|
|
|
for (auto i = obj.begin_dynamic_symbols(), e = obj.end_dynamic_symbols();
|
|
|
|
i != e; ++i) {
|
|
|
|
auto name = obj.getSymbolName(i);
|
|
|
|
if ((ec = name.getError()))
|
|
|
|
return ec;
|
|
|
|
|
|
|
|
// TODO: Add absolute symbols
|
|
|
|
if (i->st_shndx == llvm::ELF::SHN_ABS)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (i->st_shndx == llvm::ELF::SHN_UNDEF) {
|
|
|
|
if (!useShlibUndefines)
|
|
|
|
continue;
|
|
|
|
// Create an undefined atom.
|
|
|
|
if (!name->empty()) {
|
|
|
|
auto *newAtom =
|
|
|
|
new (file->_alloc) ELFUndefinedAtom<ELFT>(*file.get(), *name, &*i);
|
|
|
|
file->_undefinedAtoms._atoms.push_back(newAtom);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
file->_nameToSym[*name]._symbol = &*i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::move(file);
|
|
|
|
}
|
|
|
|
|
2013-02-12 07:03:35 +08:00
|
|
|
} // end namespace elf
|
|
|
|
} // end namespace lld
|
|
|
|
|
|
|
|
#endif
|