xref: /openbsd-src/gnu/llvm/lld/COFF/Symbols.cpp (revision dfe94b169149f14cc1aee2cf6dad58a8d9a1860c)
1ece8a530Spatrick //===- Symbols.cpp --------------------------------------------------------===//
2ece8a530Spatrick //
3ece8a530Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4ece8a530Spatrick // See https://llvm.org/LICENSE.txt for license information.
5ece8a530Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6ece8a530Spatrick //
7ece8a530Spatrick //===----------------------------------------------------------------------===//
8ece8a530Spatrick 
9ece8a530Spatrick #include "Symbols.h"
10*dfe94b16Srobert #include "COFFLinkerContext.h"
11ece8a530Spatrick #include "InputFiles.h"
12ece8a530Spatrick #include "lld/Common/ErrorHandler.h"
13ece8a530Spatrick #include "lld/Common/Memory.h"
14ece8a530Spatrick #include "lld/Common/Strings.h"
15ece8a530Spatrick #include "llvm/ADT/STLExtras.h"
16ece8a530Spatrick #include "llvm/Demangle/Demangle.h"
17ece8a530Spatrick #include "llvm/Support/Debug.h"
18ece8a530Spatrick #include "llvm/Support/raw_ostream.h"
19ece8a530Spatrick 
20ece8a530Spatrick using namespace llvm;
21ece8a530Spatrick using namespace llvm::object;
22ece8a530Spatrick 
23ece8a530Spatrick using namespace lld::coff;
24ece8a530Spatrick 
25ece8a530Spatrick namespace lld {
26ece8a530Spatrick 
27ece8a530Spatrick static_assert(sizeof(SymbolUnion) <= 48,
28ece8a530Spatrick               "symbols should be optimized for memory usage");
29ece8a530Spatrick 
30ece8a530Spatrick // Returns a symbol name for an error message.
maybeDemangleSymbol(const COFFLinkerContext & ctx,StringRef symName)31*dfe94b16Srobert static std::string maybeDemangleSymbol(const COFFLinkerContext &ctx,
32*dfe94b16Srobert                                        StringRef symName) {
33*dfe94b16Srobert   if (ctx.config.demangle) {
34ece8a530Spatrick     std::string prefix;
35ece8a530Spatrick     StringRef prefixless = symName;
36ece8a530Spatrick     if (prefixless.consume_front("__imp_"))
37ece8a530Spatrick       prefix = "__declspec(dllimport) ";
38ece8a530Spatrick     StringRef demangleInput = prefixless;
39*dfe94b16Srobert     if (ctx.config.machine == I386)
40ece8a530Spatrick       demangleInput.consume_front("_");
41*dfe94b16Srobert     std::string demangled = demangle(demangleInput.str());
42ece8a530Spatrick     if (demangled != demangleInput)
43*dfe94b16Srobert       return prefix + demangle(demangleInput.str());
44ece8a530Spatrick     return (prefix + prefixless).str();
45ece8a530Spatrick   }
46bb684c34Spatrick   return std::string(symName);
47ece8a530Spatrick }
toString(const COFFLinkerContext & ctx,coff::Symbol & b)48*dfe94b16Srobert std::string toString(const COFFLinkerContext &ctx, coff::Symbol &b) {
49*dfe94b16Srobert   return maybeDemangleSymbol(ctx, b.getName());
50ece8a530Spatrick }
toCOFFString(const COFFLinkerContext & ctx,const Archive::Symbol & b)51*dfe94b16Srobert std::string toCOFFString(const COFFLinkerContext &ctx,
52*dfe94b16Srobert                          const Archive::Symbol &b) {
53*dfe94b16Srobert   return maybeDemangleSymbol(ctx, b.getName());
54ece8a530Spatrick }
55ece8a530Spatrick 
56ece8a530Spatrick namespace coff {
57ece8a530Spatrick 
computeName()58bb684c34Spatrick void Symbol::computeName() {
59bb684c34Spatrick   assert(nameData == nullptr &&
60bb684c34Spatrick          "should only compute the name once for DefinedCOFF symbols");
61ece8a530Spatrick   auto *d = cast<DefinedCOFF>(this);
62bb684c34Spatrick   StringRef nameStr =
63bb684c34Spatrick       check(cast<ObjFile>(d->file)->getCOFFObj()->getSymbolName(d->sym));
64ece8a530Spatrick   nameData = nameStr.data();
65ece8a530Spatrick   nameSize = nameStr.size();
66ece8a530Spatrick   assert(nameSize == nameStr.size() && "name length truncated");
67ece8a530Spatrick }
68ece8a530Spatrick 
getFile()69ece8a530Spatrick InputFile *Symbol::getFile() {
70ece8a530Spatrick   if (auto *sym = dyn_cast<DefinedCOFF>(this))
71ece8a530Spatrick     return sym->file;
72ece8a530Spatrick   if (auto *sym = dyn_cast<LazyArchive>(this))
73ece8a530Spatrick     return sym->file;
74ece8a530Spatrick   if (auto *sym = dyn_cast<LazyObject>(this))
75ece8a530Spatrick     return sym->file;
761cf9926bSpatrick   if (auto *sym = dyn_cast<LazyDLLSymbol>(this))
771cf9926bSpatrick     return sym->file;
78ece8a530Spatrick   return nullptr;
79ece8a530Spatrick }
80ece8a530Spatrick 
isLive() const81ece8a530Spatrick bool Symbol::isLive() const {
82ece8a530Spatrick   if (auto *r = dyn_cast<DefinedRegular>(this))
83ece8a530Spatrick     return r->getChunk()->live;
84ece8a530Spatrick   if (auto *imp = dyn_cast<DefinedImportData>(this))
85ece8a530Spatrick     return imp->file->live;
86ece8a530Spatrick   if (auto *imp = dyn_cast<DefinedImportThunk>(this))
87ece8a530Spatrick     return imp->wrappedSym->file->thunkLive;
88ece8a530Spatrick   // Assume any other kind of symbol is live.
89ece8a530Spatrick   return true;
90ece8a530Spatrick }
91ece8a530Spatrick 
92ece8a530Spatrick // MinGW specific.
replaceKeepingName(Symbol * other,size_t size)93ece8a530Spatrick void Symbol::replaceKeepingName(Symbol *other, size_t size) {
94ece8a530Spatrick   StringRef origName = getName();
95ece8a530Spatrick   memcpy(this, other, size);
96ece8a530Spatrick   nameData = origName.data();
97ece8a530Spatrick   nameSize = origName.size();
98ece8a530Spatrick }
99ece8a530Spatrick 
getCOFFSymbol()100ece8a530Spatrick COFFSymbolRef DefinedCOFF::getCOFFSymbol() {
101ece8a530Spatrick   size_t symSize = cast<ObjFile>(file)->getCOFFObj()->getSymbolTableEntrySize();
102ece8a530Spatrick   if (symSize == sizeof(coff_symbol16))
103ece8a530Spatrick     return COFFSymbolRef(reinterpret_cast<const coff_symbol16 *>(sym));
104ece8a530Spatrick   assert(symSize == sizeof(coff_symbol32));
105ece8a530Spatrick   return COFFSymbolRef(reinterpret_cast<const coff_symbol32 *>(sym));
106ece8a530Spatrick }
107ece8a530Spatrick 
getRVA()108*dfe94b16Srobert uint64_t DefinedAbsolute::getRVA() { return va - ctx.config.imageBase; }
109ece8a530Spatrick 
makeImportThunk(COFFLinkerContext & ctx,DefinedImportData * s,uint16_t machine)110*dfe94b16Srobert static Chunk *makeImportThunk(COFFLinkerContext &ctx, DefinedImportData *s,
111*dfe94b16Srobert                               uint16_t machine) {
112ece8a530Spatrick   if (machine == AMD64)
113*dfe94b16Srobert     return make<ImportThunkChunkX64>(ctx, s);
114ece8a530Spatrick   if (machine == I386)
115*dfe94b16Srobert     return make<ImportThunkChunkX86>(ctx, s);
116ece8a530Spatrick   if (machine == ARM64)
117*dfe94b16Srobert     return make<ImportThunkChunkARM64>(ctx, s);
118ece8a530Spatrick   assert(machine == ARMNT);
119*dfe94b16Srobert   return make<ImportThunkChunkARM>(ctx, s);
120ece8a530Spatrick }
121ece8a530Spatrick 
DefinedImportThunk(COFFLinkerContext & ctx,StringRef name,DefinedImportData * s,uint16_t machine)122*dfe94b16Srobert DefinedImportThunk::DefinedImportThunk(COFFLinkerContext &ctx, StringRef name,
123*dfe94b16Srobert                                        DefinedImportData *s, uint16_t machine)
124ece8a530Spatrick     : Defined(DefinedImportThunkKind, name), wrappedSym(s),
125*dfe94b16Srobert       data(makeImportThunk(ctx, s, machine)) {}
126ece8a530Spatrick 
getWeakAlias()127ece8a530Spatrick Defined *Undefined::getWeakAlias() {
128ece8a530Spatrick   // A weak alias may be a weak alias to another symbol, so check recursively.
129ece8a530Spatrick   for (Symbol *a = weakAlias; a; a = cast<Undefined>(a)->weakAlias)
130ece8a530Spatrick     if (auto *d = dyn_cast<Defined>(a))
131ece8a530Spatrick       return d;
132ece8a530Spatrick   return nullptr;
133ece8a530Spatrick }
134ece8a530Spatrick 
getMemberBuffer()135ece8a530Spatrick MemoryBufferRef LazyArchive::getMemberBuffer() {
136ece8a530Spatrick   Archive::Child c =
137*dfe94b16Srobert       CHECK(sym.getMember(), "could not get the member for symbol " +
138*dfe94b16Srobert                                  toCOFFString(file->ctx, sym));
139ece8a530Spatrick   return CHECK(c.getMemoryBufferRef(),
140ece8a530Spatrick                "could not get the buffer for the member defining symbol " +
141*dfe94b16Srobert                    toCOFFString(file->ctx, sym));
142ece8a530Spatrick }
143ece8a530Spatrick } // namespace coff
144ece8a530Spatrick } // namespace lld
145