blob: e14ebcffe564df1b472ba1c737e56425fc953f6f [file] [log] [blame]
Sam Cleggc94d3932017-11-17 18:14:09 +00001//===- SymbolTable.cpp ----------------------------------------------------===//
2//
3// The LLVM Linker
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "SymbolTable.h"
11
12#include "Config.h"
Sam Clegg5fa274b2018-01-10 01:13:34 +000013#include "InputChunks.h"
Sam Cleggb8621592017-11-30 01:40:08 +000014#include "WriterUtils.h"
Sam Cleggc94d3932017-11-17 18:14:09 +000015#include "lld/Common/ErrorHandler.h"
Rui Ueyama2017d522017-11-28 20:39:17 +000016#include "lld/Common/Memory.h"
Rui Ueyama7d67dd12018-02-13 22:30:52 +000017#include "llvm/ADT/SetVector.h"
Sam Cleggc94d3932017-11-17 18:14:09 +000018
19#define DEBUG_TYPE "lld"
20
21using namespace llvm;
Sam Clegg20db3812018-01-10 00:52:20 +000022using namespace llvm::wasm;
Sam Cleggc94d3932017-11-17 18:14:09 +000023using namespace lld;
24using namespace lld::wasm;
25
26SymbolTable *lld::wasm::Symtab;
27
28void SymbolTable::addFile(InputFile *File) {
29 log("Processing: " + toString(File));
30 File->parse();
31
32 if (auto *F = dyn_cast<ObjFile>(File))
33 ObjectFiles.push_back(F);
34}
35
36void SymbolTable::reportRemainingUndefines() {
Rui Ueyama7d67dd12018-02-13 22:30:52 +000037 SetVector<Symbol *> Undefs;
Sam Clegg74fe0ba2017-12-07 01:51:24 +000038 for (Symbol *Sym : SymVector) {
Sam Cleggc94d3932017-11-17 18:14:09 +000039 if (Sym->isUndefined() && !Sym->isWeak() &&
40 Config->AllowUndefinedSymbols.count(Sym->getName()) == 0) {
41 Undefs.insert(Sym);
42 }
43 }
44
45 if (Undefs.empty())
46 return;
47
48 for (ObjFile *File : ObjectFiles)
49 for (Symbol *Sym : File->getSymbols())
50 if (Undefs.count(Sym))
51 error(toString(File) + ": undefined symbol: " + toString(*Sym));
52
53 for (Symbol *Sym : Undefs)
54 if (!Sym->getFile())
55 error("undefined symbol: " + toString(*Sym));
56}
57
58Symbol *SymbolTable::find(StringRef Name) {
Sam Clegga80d94d2017-11-27 23:16:06 +000059 auto It = SymMap.find(CachedHashStringRef(Name));
60 if (It == SymMap.end())
Sam Cleggc94d3932017-11-17 18:14:09 +000061 return nullptr;
62 return It->second;
63}
64
65std::pair<Symbol *, bool> SymbolTable::insert(StringRef Name) {
Sam Clegga80d94d2017-11-27 23:16:06 +000066 Symbol *&Sym = SymMap[CachedHashStringRef(Name)];
Sam Cleggc94d3932017-11-17 18:14:09 +000067 if (Sym)
68 return {Sym, false};
69 Sym = make<Symbol>(Name, false);
Sam Clegg74fe0ba2017-12-07 01:51:24 +000070 SymVector.emplace_back(Sym);
Sam Cleggc94d3932017-11-17 18:14:09 +000071 return {Sym, true};
72}
73
74void SymbolTable::reportDuplicate(Symbol *Existing, InputFile *NewFile) {
75 error("duplicate symbol: " + toString(*Existing) + "\n>>> defined in " +
76 toString(Existing->getFile()) + "\n>>> defined in " +
Sam Clegge2ed0922017-11-27 22:49:16 +000077 toString(NewFile));
Sam Cleggc94d3932017-11-17 18:14:09 +000078}
79
Sam Cleggb8621592017-11-30 01:40:08 +000080// Check the type of new symbol matches that of the symbol is replacing.
81// For functions this can also involve verifying that the signatures match.
82static void checkSymbolTypes(const Symbol &Existing, const InputFile &F,
Sam Clegg20db3812018-01-10 00:52:20 +000083 Symbol::Kind Kind, const WasmSignature *NewSig) {
Sam Cleggb8621592017-11-30 01:40:08 +000084 if (Existing.isLazy())
Sam Cleggc94d3932017-11-17 18:14:09 +000085 return;
86
Sam Clegg20db3812018-01-10 00:52:20 +000087 bool NewIsFunction = Kind == Symbol::Kind::UndefinedFunctionKind ||
88 Kind == Symbol::Kind::DefinedFunctionKind;
Sam Cleggb8621592017-11-30 01:40:08 +000089
90 // First check the symbol types match (i.e. either both are function
91 // symbols or both are data symbols).
92 if (Existing.isFunction() != NewIsFunction) {
Sam Clegg20db3812018-01-10 00:52:20 +000093 error("symbol type mismatch: " + Existing.getName() + "\n>>> defined as " +
Sam Cleggb8621592017-11-30 01:40:08 +000094 (Existing.isFunction() ? "Function" : "Global") + " in " +
95 toString(Existing.getFile()) + "\n>>> defined as " +
96 (NewIsFunction ? "Function" : "Global") + " in " + F.getName());
97 return;
98 }
99
100 // For function symbols, optionally check the function signature matches too.
101 if (!NewIsFunction || !Config->CheckSignatures)
Sam Cleggc94d3932017-11-17 18:14:09 +0000102 return;
Sam Clegg31de2f02017-12-07 03:19:53 +0000103 // Skip the signature check if the existing function has no signature (e.g.
104 // if it is an undefined symbol generated by --undefined command line flag).
105 if (!Existing.hasFunctionType())
106 return;
Sam Cleggc94d3932017-11-17 18:14:09 +0000107
Sam Clegg20db3812018-01-10 00:52:20 +0000108 DEBUG(dbgs() << "checkSymbolTypes: " << Existing.getName() << "\n");
Sam Cleggb8621592017-11-30 01:40:08 +0000109 assert(NewSig);
110
111 const WasmSignature &OldSig = Existing.getFunctionType();
112 if (*NewSig == OldSig)
113 return;
114
Sam Clegg20db3812018-01-10 00:52:20 +0000115 error("function signature mismatch: " + Existing.getName() +
116 "\n>>> defined as " + toString(OldSig) + " in " +
117 toString(Existing.getFile()) + "\n>>> defined as " + toString(*NewSig) +
118 " in " + F.getName());
Sam Cleggc94d3932017-11-17 18:14:09 +0000119}
120
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000121static void checkSymbolTypes(const Symbol &Existing, const InputFile &F,
122 Symbol::Kind Kind, const InputChunk *Chunk) {
Sam Cleggfc50c622018-01-28 19:57:02 +0000123 const WasmSignature *Sig = nullptr;
124 if (auto *F = dyn_cast_or_null<InputFunction>(Chunk))
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000125 Sig = &F->Signature;
126 return checkSymbolTypes(Existing, F, Kind, Sig);
127}
128
Sam Clegga1892302018-02-13 20:14:26 +0000129Symbol *SymbolTable::addSyntheticFunction(StringRef Name,
130 const WasmSignature *Type,
131 uint32_t Flags) {
132 DEBUG(dbgs() << "addSyntheticFunction: " << Name << "\n");
Sam Clegg50686852018-01-12 18:35:13 +0000133 Symbol *S;
134 bool WasInserted;
135 std::tie(S, WasInserted) = insert(Name);
Sam Clegga1892302018-02-13 20:14:26 +0000136 assert(WasInserted);
137 S->update(Symbol::DefinedFunctionKind, nullptr, Flags);
138 S->setFunctionType(Type);
Sam Clegg50686852018-01-12 18:35:13 +0000139 return S;
140}
141
Sam Clegga1892302018-02-13 20:14:26 +0000142Symbol *SymbolTable::addSyntheticGlobal(StringRef Name) {
143 DEBUG(dbgs() << "addSyntheticGlobal: " << Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000144 Symbol *S;
145 bool WasInserted;
146 std::tie(S, WasInserted) = insert(Name);
Sam Clegga1892302018-02-13 20:14:26 +0000147 assert(WasInserted);
148 S->update(Symbol::DefinedGlobalKind);
Sam Cleggc94d3932017-11-17 18:14:09 +0000149 return S;
150}
151
Sam Clegg20db3812018-01-10 00:52:20 +0000152Symbol *SymbolTable::addDefined(StringRef Name, Symbol::Kind Kind,
Sam Cleggfc50c622018-01-28 19:57:02 +0000153 uint32_t Flags, InputFile *F, InputChunk *Chunk,
154 uint32_t Address) {
Sam Clegg20db3812018-01-10 00:52:20 +0000155 DEBUG(dbgs() << "addDefined: " << Name << " addr:" << Address << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000156 Symbol *S;
157 bool WasInserted;
Sam Cleggc94d3932017-11-17 18:14:09 +0000158
Sam Clegg20db3812018-01-10 00:52:20 +0000159 std::tie(S, WasInserted) = insert(Name);
Sam Cleggc94d3932017-11-17 18:14:09 +0000160 if (WasInserted) {
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000161 S->update(Kind, F, Flags, Chunk, Address);
Sam Cleggb8621592017-11-30 01:40:08 +0000162 } else if (S->isLazy()) {
163 // The existing symbol is lazy. Replace it without checking types since
164 // lazy symbols don't have any type information.
Sam Clegg20db3812018-01-10 00:52:20 +0000165 DEBUG(dbgs() << "replacing existing lazy symbol: " << Name << "\n");
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000166 S->update(Kind, F, Flags, Chunk, Address);
Sam Cleggc94d3932017-11-17 18:14:09 +0000167 } else if (!S->isDefined()) {
168 // The existing symbol table entry is undefined. The new symbol replaces
Sam Clegg49ed9262017-12-01 00:53:21 +0000169 // it, after checking the type matches
Sam Clegg20db3812018-01-10 00:52:20 +0000170 DEBUG(dbgs() << "resolving existing undefined symbol: " << Name << "\n");
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000171 checkSymbolTypes(*S, *F, Kind, Chunk);
172 S->update(Kind, F, Flags, Chunk, Address);
Sam Clegg20db3812018-01-10 00:52:20 +0000173 } else if ((Flags & WASM_SYMBOL_BINDING_MASK) == WASM_SYMBOL_BINDING_WEAK) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000174 // the new symbol is weak we can ignore it
Sam Clegg49ed9262017-12-01 00:53:21 +0000175 DEBUG(dbgs() << "existing symbol takes precedence\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000176 } else if (S->isWeak()) {
177 // the new symbol is not weak and the existing symbol is, so we replace
178 // it
179 DEBUG(dbgs() << "replacing existing weak symbol\n");
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000180 checkSymbolTypes(*S, *F, Kind, Chunk);
181 S->update(Kind, F, Flags, Chunk, Address);
Sam Cleggc94d3932017-11-17 18:14:09 +0000182 } else {
Sam Clegg49ed9262017-12-01 00:53:21 +0000183 // neither symbol is week. They conflict.
Sam Cleggc94d3932017-11-17 18:14:09 +0000184 reportDuplicate(S, F);
185 }
186 return S;
187}
188
Sam Cleggb8621592017-11-30 01:40:08 +0000189Symbol *SymbolTable::addUndefinedFunction(StringRef Name,
190 const WasmSignature *Type) {
Sam Clegg096aa502018-01-20 01:40:17 +0000191 DEBUG(dbgs() << "addUndefinedFunction: " << Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000192 Symbol *S;
193 bool WasInserted;
194 std::tie(S, WasInserted) = insert(Name);
Sam Cleggb8621592017-11-30 01:40:08 +0000195 if (WasInserted) {
Sam Clegg8d146bb2018-01-09 23:56:44 +0000196 S->update(Symbol::UndefinedFunctionKind);
197 S->setFunctionType(Type);
Sam Cleggb8621592017-11-30 01:40:08 +0000198 } else if (!S->isFunction()) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000199 error("symbol type mismatch: " + Name);
Sam Cleggb8621592017-11-30 01:40:08 +0000200 }
Sam Cleggc94d3932017-11-17 18:14:09 +0000201 return S;
202}
203
Sam Clegg20db3812018-01-10 00:52:20 +0000204Symbol *SymbolTable::addUndefined(StringRef Name, Symbol::Kind Kind,
205 uint32_t Flags, InputFile *F,
Sam Clegg8d146bb2018-01-09 23:56:44 +0000206 const WasmSignature *Type) {
Sam Clegg20db3812018-01-10 00:52:20 +0000207 DEBUG(dbgs() << "addUndefined: " << Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000208 Symbol *S;
209 bool WasInserted;
Sam Clegg20db3812018-01-10 00:52:20 +0000210 std::tie(S, WasInserted) = insert(Name);
Sam Cleggc94d3932017-11-17 18:14:09 +0000211 if (WasInserted) {
Sam Clegg20db3812018-01-10 00:52:20 +0000212 S->update(Kind, F, Flags);
Sam Clegg8d146bb2018-01-09 23:56:44 +0000213 if (Type)
214 S->setFunctionType(Type);
Sam Cleggc94d3932017-11-17 18:14:09 +0000215 } else if (S->isLazy()) {
216 DEBUG(dbgs() << "resolved by existing lazy\n");
217 auto *AF = cast<ArchiveFile>(S->getFile());
218 AF->addMember(&S->getArchiveSymbol());
219 } else if (S->isDefined()) {
220 DEBUG(dbgs() << "resolved by existing\n");
Sam Clegg20db3812018-01-10 00:52:20 +0000221 checkSymbolTypes(*S, *F, Kind, Type);
Sam Cleggc94d3932017-11-17 18:14:09 +0000222 }
223 return S;
224}
225
226void SymbolTable::addLazy(ArchiveFile *F, const Archive::Symbol *Sym) {
Sam Clegga681a112017-12-06 03:10:39 +0000227 DEBUG(dbgs() << "addLazy: " << Sym->getName() << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000228 StringRef Name = Sym->getName();
229 Symbol *S;
230 bool WasInserted;
231 std::tie(S, WasInserted) = insert(Name);
232 if (WasInserted) {
233 S->update(Symbol::LazyKind, F);
234 S->setArchiveSymbol(*Sym);
235 } else if (S->isUndefined()) {
236 // There is an existing undefined symbol. The can load from the
237 // archive.
238 DEBUG(dbgs() << "replacing existing undefined\n");
239 F->addMember(Sym);
240 }
241}
Sam Clegge0f6fcd2018-01-12 22:25:17 +0000242
243bool SymbolTable::addComdat(StringRef Name, ObjFile *F) {
244 DEBUG(dbgs() << "addComdat: " << Name << "\n");
245 ObjFile *&File = ComdatMap[CachedHashStringRef(Name)];
246 if (File) {
247 DEBUG(dbgs() << "COMDAT already defined\n");
248 return false;
249 }
250 File = F;
251 return true;
252}
253
254ObjFile *SymbolTable::findComdat(StringRef Name) const {
255 auto It = ComdatMap.find(CachedHashStringRef(Name));
256 return It == ComdatMap.end() ? nullptr : It->second;
257}