2015-05-29 03:09:30 +08:00
|
|
|
//===- Symbols.cpp --------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Linker
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2016-12-08 07:17:02 +08:00
|
|
|
#include "Symbols.h"
|
2015-05-29 03:09:30 +08:00
|
|
|
#include "InputFiles.h"
|
2016-12-08 07:17:02 +08:00
|
|
|
#include "Strings.h"
|
[lld] unified COFF and ELF error handling on new Common/ErrorHandler
Summary:
The COFF linker and the ELF linker have long had similar but separate
Error.h and Error.cpp files to implement error handling. This change
introduces new error handling code in Common/ErrorHandler.h, changes the
COFF and ELF linkers to use it, and removes the old, separate
implementations.
Reviewers: ruiu
Reviewed By: ruiu
Subscribers: smeenai, jyknight, emaste, sdardis, nemanjai, nhaehnle, mgorny, javed.absar, kbarton, fedor.sergeev, llvm-commits
Differential Revision: https://reviews.llvm.org/D39259
llvm-svn: 316624
2017-10-26 06:28:38 +08:00
|
|
|
#include "lld/Common/ErrorHandler.h"
|
2017-11-29 04:39:17 +08:00
|
|
|
#include "lld/Common/Memory.h"
|
2015-05-29 03:09:30 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
|
2016-12-08 07:17:02 +08:00
|
|
|
using namespace llvm;
|
2015-05-29 03:09:30 +08:00
|
|
|
using namespace llvm::object;
|
|
|
|
|
2017-01-06 18:15:47 +08:00
|
|
|
// Returns a symbol name for an error message.
|
2017-11-04 05:21:47 +08:00
|
|
|
std::string lld::toString(coff::Symbol &B) {
|
2017-11-28 10:15:26 +08:00
|
|
|
if (Optional<std::string> S = coff::demangleMSVC(B.getName()))
|
2017-01-06 18:15:47 +08:00
|
|
|
return ("\"" + *S + "\" (" + B.getName() + ")").str();
|
|
|
|
return B.getName();
|
|
|
|
}
|
|
|
|
|
2015-05-29 03:09:30 +08:00
|
|
|
namespace lld {
|
|
|
|
namespace coff {
|
|
|
|
|
2017-11-04 05:21:47 +08:00
|
|
|
StringRef Symbol::getName() {
|
2017-02-03 07:58:14 +08:00
|
|
|
// COFF symbol names are read lazily for a performance reason.
|
[opt] Devirtualize the SymbolBody type hierarchy and start compacting
its members into the base class.
First, to help motivate this kind of change, understand that in
a self-link, LLD creates 5.5 million defined regular symbol bodies (and
6 million symbol bodies total). A significant portion of its time is
spent allocating the memory for these symbols, and befor ethis patch
the defined regular symbol body objects alone consumed some 420mb of
memory during the self link.
As a consequence, I think it is worth expending considerable effort to
make these objects as memory efficient as possible. This is the first of
several components of that. This change starts with the goal of removing
the virtual functins from SymbolBody so that it can avoid having a vptr
embedded in it when it already contains a "kind" member, and that member
can be much more compact than a vptr.
The primary way of doing this is to sink as much of the logic that we
would have to dispatch for into data in the base class. As part of this,
I made the various flags bits that will pack into a bitfield with the
kind tag. I also sank the Name down to eliminate the dispatch for that,
and used LLVM's RTTI-style dispatch for everything else (most of which
is cold and so doesn't matter terribly if we get minutely worse lowering
than a vtable dispatch).
As I was doing this, I wanted to make the RTTI-dispatch (which would
become much hotter than before) as efficient as possible, so I've
re-organized the tags somewhat. Notably, the common case (regular
defined symbols) is now zero which we can test for faster.
I also needed to rewrite the comparison routine used during resolving
symbols. This proved to be quite complex as the semantics of the
existing one were very subtle due to the back-and-forth virtual dispatch
caused by re-dispatching with reversed operands. I've consolidated it to
a single function and tried to comment it quite a bit more to help
explain what is going on. However, this may need more comments or other
explanations. It at least passes all the regression tests. I'm not
working on Windows, so I can't fully test it.
With all of these changes, the size of a DefinedRegular symbol on
a 64-bit build goes from 80 bytes to 64 bytes, and we save approximately
84mb or 20% of the memory consumed by these symbol bodies during the
link.
The link time appears marginally faster as well, and the profile hotness
of the memory allocation subsystem got a bit better, but there is still
a lot of allocation traffic.
Differential Revision: http://reviews.llvm.org/D10792
llvm-svn: 241001
2015-06-30 05:35:48 +08:00
|
|
|
// Non-external symbol names are never used by the linker except for logging
|
|
|
|
// or debugging. Their internal references are resolved not by name but by
|
|
|
|
// symbol index. And because they are not external, no one can refer them by
|
|
|
|
// name. Object files contain lots of non-external symbols, and creating
|
|
|
|
// StringRefs for them (which involves lots of strlen() on the string table)
|
|
|
|
// is a waste of time.
|
|
|
|
if (Name.empty()) {
|
|
|
|
auto *D = cast<DefinedCOFF>(this);
|
2017-07-27 07:05:24 +08:00
|
|
|
cast<ObjFile>(D->File)->getCOFFObj()->getSymbolName(D->Sym, Name);
|
[opt] Devirtualize the SymbolBody type hierarchy and start compacting
its members into the base class.
First, to help motivate this kind of change, understand that in
a self-link, LLD creates 5.5 million defined regular symbol bodies (and
6 million symbol bodies total). A significant portion of its time is
spent allocating the memory for these symbols, and befor ethis patch
the defined regular symbol body objects alone consumed some 420mb of
memory during the self link.
As a consequence, I think it is worth expending considerable effort to
make these objects as memory efficient as possible. This is the first of
several components of that. This change starts with the goal of removing
the virtual functins from SymbolBody so that it can avoid having a vptr
embedded in it when it already contains a "kind" member, and that member
can be much more compact than a vptr.
The primary way of doing this is to sink as much of the logic that we
would have to dispatch for into data in the base class. As part of this,
I made the various flags bits that will pack into a bitfield with the
kind tag. I also sank the Name down to eliminate the dispatch for that,
and used LLVM's RTTI-style dispatch for everything else (most of which
is cold and so doesn't matter terribly if we get minutely worse lowering
than a vtable dispatch).
As I was doing this, I wanted to make the RTTI-dispatch (which would
become much hotter than before) as efficient as possible, so I've
re-organized the tags somewhat. Notably, the common case (regular
defined symbols) is now zero which we can test for faster.
I also needed to rewrite the comparison routine used during resolving
symbols. This proved to be quite complex as the semantics of the
existing one were very subtle due to the back-and-forth virtual dispatch
caused by re-dispatching with reversed operands. I've consolidated it to
a single function and tried to comment it quite a bit more to help
explain what is going on. However, this may need more comments or other
explanations. It at least passes all the regression tests. I'm not
working on Windows, so I can't fully test it.
With all of these changes, the size of a DefinedRegular symbol on
a 64-bit build goes from 80 bytes to 64 bytes, and we save approximately
84mb or 20% of the memory consumed by these symbol bodies during the
link.
The link time appears marginally faster as well, and the profile hotness
of the memory allocation subsystem got a bit better, but there is still
a lot of allocation traffic.
Differential Revision: http://reviews.llvm.org/D10792
llvm-svn: 241001
2015-06-30 05:35:48 +08:00
|
|
|
}
|
|
|
|
return Name;
|
|
|
|
}
|
|
|
|
|
2017-11-04 05:21:47 +08:00
|
|
|
InputFile *Symbol::getFile() {
|
2016-12-08 07:17:02 +08:00
|
|
|
if (auto *Sym = dyn_cast<DefinedCOFF>(this))
|
|
|
|
return Sym->File;
|
|
|
|
if (auto *Sym = dyn_cast<Lazy>(this))
|
|
|
|
return Sym->File;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2017-11-04 05:21:47 +08:00
|
|
|
bool Symbol::isLive() const {
|
2017-07-28 02:25:59 +08:00
|
|
|
if (auto *R = dyn_cast<DefinedRegular>(this))
|
|
|
|
return R->getChunk()->isLive();
|
|
|
|
if (auto *Imp = dyn_cast<DefinedImportData>(this))
|
|
|
|
return Imp->File->Live;
|
|
|
|
if (auto *Imp = dyn_cast<DefinedImportThunk>(this))
|
|
|
|
return Imp->WrappedSym->File->Live;
|
|
|
|
// Assume any other kind of symbol is live.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-07-10 01:43:50 +08:00
|
|
|
COFFSymbolRef DefinedCOFF::getCOFFSymbol() {
|
2017-07-27 07:05:24 +08:00
|
|
|
size_t SymSize = cast<ObjFile>(File)->getCOFFObj()->getSymbolTableEntrySize();
|
2015-07-10 01:43:50 +08:00
|
|
|
if (SymSize == sizeof(coff_symbol16))
|
|
|
|
return COFFSymbolRef(reinterpret_cast<const coff_symbol16 *>(Sym));
|
|
|
|
assert(SymSize == sizeof(coff_symbol32));
|
|
|
|
return COFFSymbolRef(reinterpret_cast<const coff_symbol32 *>(Sym));
|
|
|
|
}
|
|
|
|
|
2017-06-23 07:33:04 +08:00
|
|
|
uint16_t DefinedAbsolute::OutputSectionIndex = 0;
|
|
|
|
|
2017-05-25 01:12:53 +08:00
|
|
|
static Chunk *makeImportThunk(DefinedImportData *S, uint16_t Machine) {
|
|
|
|
if (Machine == AMD64)
|
|
|
|
return make<ImportThunkChunkX64>(S);
|
|
|
|
if (Machine == I386)
|
|
|
|
return make<ImportThunkChunkX86>(S);
|
2017-07-11 15:22:44 +08:00
|
|
|
if (Machine == ARM64)
|
|
|
|
return make<ImportThunkChunkARM64>(S);
|
2017-05-25 01:12:53 +08:00
|
|
|
assert(Machine == ARMNT);
|
|
|
|
return make<ImportThunkChunkARM>(S);
|
|
|
|
}
|
|
|
|
|
2015-07-25 09:16:06 +08:00
|
|
|
DefinedImportThunk::DefinedImportThunk(StringRef Name, DefinedImportData *S,
|
2015-07-26 05:54:50 +08:00
|
|
|
uint16_t Machine)
|
2017-05-25 06:30:06 +08:00
|
|
|
: Defined(DefinedImportThunkKind, Name), WrappedSym(S),
|
2017-05-25 01:12:53 +08:00
|
|
|
Data(makeImportThunk(S, Machine)) {}
|
2015-07-25 09:16:06 +08:00
|
|
|
|
2015-07-04 13:28:41 +08:00
|
|
|
Defined *Undefined::getWeakAlias() {
|
|
|
|
// A weak alias may be a weak alias to another symbol, so check recursively.
|
2017-11-04 05:21:47 +08:00
|
|
|
for (Symbol *A = WeakAlias; A; A = cast<Undefined>(A)->WeakAlias)
|
2016-12-10 05:55:24 +08:00
|
|
|
if (auto *D = dyn_cast<Defined>(A))
|
2015-07-04 13:28:41 +08:00
|
|
|
return D;
|
|
|
|
return nullptr;
|
|
|
|
}
|
2017-01-06 18:04:08 +08:00
|
|
|
} // namespace coff
|
2015-05-29 03:09:30 +08:00
|
|
|
} // namespace lld
|