blob: 408b5addf8ea18ed019955ffd768b37d3c6cf983 [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"
Sam Cleggc94d3932017-11-17 18:14:09 +000011#include "Config.h"
Sam Clegg5fa274b2018-01-10 01:13:34 +000012#include "InputChunks.h"
Sam Cleggb8621592017-11-30 01:40:08 +000013#include "WriterUtils.h"
Sam Cleggc94d3932017-11-17 18:14:09 +000014#include "lld/Common/ErrorHandler.h"
Rui Ueyama2017d522017-11-28 20:39:17 +000015#include "lld/Common/Memory.h"
Rui Ueyama7d67dd12018-02-13 22:30:52 +000016#include "llvm/ADT/SetVector.h"
Sam Cleggc94d3932017-11-17 18:14:09 +000017
18#define DEBUG_TYPE "lld"
19
20using namespace llvm;
Sam Clegg20db3812018-01-10 00:52:20 +000021using namespace llvm::wasm;
Sam Cleggc94d3932017-11-17 18:14:09 +000022using namespace lld;
23using namespace lld::wasm;
24
25SymbolTable *lld::wasm::Symtab;
26
27void SymbolTable::addFile(InputFile *File) {
28 log("Processing: " + toString(File));
29 File->parse();
30
31 if (auto *F = dyn_cast<ObjFile>(File))
32 ObjectFiles.push_back(F);
33}
34
35void SymbolTable::reportRemainingUndefines() {
Rui Ueyama7d67dd12018-02-13 22:30:52 +000036 SetVector<Symbol *> Undefs;
Sam Clegg74fe0ba2017-12-07 01:51:24 +000037 for (Symbol *Sym : SymVector) {
Sam Cleggc94d3932017-11-17 18:14:09 +000038 if (Sym->isUndefined() && !Sym->isWeak() &&
39 Config->AllowUndefinedSymbols.count(Sym->getName()) == 0) {
40 Undefs.insert(Sym);
41 }
42 }
43
44 if (Undefs.empty())
45 return;
46
47 for (ObjFile *File : ObjectFiles)
48 for (Symbol *Sym : File->getSymbols())
49 if (Undefs.count(Sym))
50 error(toString(File) + ": undefined symbol: " + toString(*Sym));
51
52 for (Symbol *Sym : Undefs)
53 if (!Sym->getFile())
54 error("undefined symbol: " + toString(*Sym));
55}
56
57Symbol *SymbolTable::find(StringRef Name) {
Sam Clegga80d94d2017-11-27 23:16:06 +000058 auto It = SymMap.find(CachedHashStringRef(Name));
59 if (It == SymMap.end())
Sam Cleggc94d3932017-11-17 18:14:09 +000060 return nullptr;
61 return It->second;
62}
63
64std::pair<Symbol *, bool> SymbolTable::insert(StringRef Name) {
Sam Clegga80d94d2017-11-27 23:16:06 +000065 Symbol *&Sym = SymMap[CachedHashStringRef(Name)];
Sam Cleggc94d3932017-11-17 18:14:09 +000066 if (Sym)
67 return {Sym, false};
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000068 Sym = reinterpret_cast<Symbol *>(make<SymbolUnion>());
Sam Clegg74fe0ba2017-12-07 01:51:24 +000069 SymVector.emplace_back(Sym);
Sam Cleggc94d3932017-11-17 18:14:09 +000070 return {Sym, true};
71}
72
Sam Cleggb8621592017-11-30 01:40:08 +000073// Check the type of new symbol matches that of the symbol is replacing.
74// For functions this can also involve verifying that the signatures match.
75static void checkSymbolTypes(const Symbol &Existing, const InputFile &F,
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000076 bool NewIsFunction, const WasmSignature *NewSig) {
Sam Cleggb8621592017-11-30 01:40:08 +000077 if (Existing.isLazy())
Sam Cleggc94d3932017-11-17 18:14:09 +000078 return;
79
Sam Cleggb8621592017-11-30 01:40:08 +000080 // First check the symbol types match (i.e. either both are function
81 // symbols or both are data symbols).
Sam Cleggcaca8d52018-02-17 00:44:21 +000082 if (isa<FunctionSymbol>(Existing) != NewIsFunction) {
Sam Clegg20db3812018-01-10 00:52:20 +000083 error("symbol type mismatch: " + Existing.getName() + "\n>>> defined as " +
Sam Cleggcaca8d52018-02-17 00:44:21 +000084 (isa<FunctionSymbol>(Existing) ? "Function" : "Global") + " in " +
Sam Cleggb8621592017-11-30 01:40:08 +000085 toString(Existing.getFile()) + "\n>>> defined as " +
86 (NewIsFunction ? "Function" : "Global") + " in " + F.getName());
87 return;
88 }
89
90 // For function symbols, optionally check the function signature matches too.
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000091 auto *ExistingFunc = dyn_cast<FunctionSymbol>(&Existing);
92 if (!ExistingFunc || !Config->CheckSignatures)
Sam Cleggc94d3932017-11-17 18:14:09 +000093 return;
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000094
Sam Clegg3f8db982018-02-16 23:50:23 +000095 const WasmSignature *OldSig = ExistingFunc->getFunctionType();
96
Sam Clegg31de2f02017-12-07 03:19:53 +000097 // Skip the signature check if the existing function has no signature (e.g.
98 // if it is an undefined symbol generated by --undefined command line flag).
Sam Clegg3f8db982018-02-16 23:50:23 +000099 if (OldSig == nullptr)
Sam Clegg31de2f02017-12-07 03:19:53 +0000100 return;
Sam Cleggc94d3932017-11-17 18:14:09 +0000101
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000102 DEBUG(dbgs() << "checkSymbolTypes: " << ExistingFunc->getName() << "\n");
Sam Cleggb8621592017-11-30 01:40:08 +0000103 assert(NewSig);
104
Sam Clegg3f8db982018-02-16 23:50:23 +0000105 if (*NewSig == *OldSig)
Sam Cleggb8621592017-11-30 01:40:08 +0000106 return;
107
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000108 error("function signature mismatch: " + ExistingFunc->getName() +
Sam Clegg3f8db982018-02-16 23:50:23 +0000109 "\n>>> defined as " + toString(*OldSig) + " in " +
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000110 toString(ExistingFunc->getFile()) + "\n>>> defined as " +
111 toString(*NewSig) + " in " + F.getName());
Sam Cleggc94d3932017-11-17 18:14:09 +0000112}
113
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000114static void checkSymbolTypes(const Symbol &Existing, const InputFile &F,
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000115 bool IsFunction, const InputChunk *Chunk) {
Sam Cleggfc50c622018-01-28 19:57:02 +0000116 const WasmSignature *Sig = nullptr;
117 if (auto *F = dyn_cast_or_null<InputFunction>(Chunk))
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000118 Sig = &F->Signature;
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000119 return checkSymbolTypes(Existing, F, IsFunction, Sig);
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000120}
121
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000122DefinedFunction *SymbolTable::addSyntheticFunction(StringRef Name,
123 const WasmSignature *Type,
124 uint32_t Flags) {
Sam Clegga1892302018-02-13 20:14:26 +0000125 DEBUG(dbgs() << "addSyntheticFunction: " << Name << "\n");
Sam Clegg50686852018-01-12 18:35:13 +0000126 Symbol *S;
127 bool WasInserted;
128 std::tie(S, WasInserted) = insert(Name);
Sam Clegga1892302018-02-13 20:14:26 +0000129 assert(WasInserted);
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000130 return replaceSymbol<DefinedFunction>(S, Name, Flags, Type);
Sam Clegg50686852018-01-12 18:35:13 +0000131}
132
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000133DefinedGlobal *SymbolTable::addSyntheticGlobal(StringRef Name, uint32_t Flags) {
Sam Clegga1892302018-02-13 20:14:26 +0000134 DEBUG(dbgs() << "addSyntheticGlobal: " << Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000135 Symbol *S;
136 bool WasInserted;
137 std::tie(S, WasInserted) = insert(Name);
Sam Clegga1892302018-02-13 20:14:26 +0000138 assert(WasInserted);
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000139 return replaceSymbol<DefinedGlobal>(S, Name, Flags);
Sam Cleggc94d3932017-11-17 18:14:09 +0000140}
141
Sam Cleggf60b0d52018-02-20 18:55:07 +0000142static bool shouldReplace(const Symbol &Existing, InputFile *NewFile,
143 uint32_t NewFlags, InputChunk *NewChunk,
144 bool NewIsFunction) {
Rui Ueyamac03c9042018-02-20 21:08:47 +0000145 // If existing symbol is lazy, replace it without checking types since
146 // lazy symbols don't have any type information.
Sam Clegg93e559b2018-02-20 18:55:06 +0000147 if (Existing.isLazy()) {
Sam Clegg93e559b2018-02-20 18:55:06 +0000148 DEBUG(dbgs() << "replacing existing lazy symbol: " << Existing.getName()
149 << "\n");
Rui Ueyamac03c9042018-02-20 21:08:47 +0000150 return true;
Sam Cleggc94d3932017-11-17 18:14:09 +0000151 }
Sam Cleggc94d3932017-11-17 18:14:09 +0000152
Rui Ueyamac03c9042018-02-20 21:08:47 +0000153 // Now we have two wasm symbols, and all wasm symbols that have the same
154 // symbol name must have the same type, even if they are undefined. This
155 // is different from ELF because symbol types are not that significant
156 // in ELF, and undefined symbols in ELF don't have type in the first place.
157 checkSymbolTypes(Existing, *NewFile, NewIsFunction, NewChunk);
Sam Clegg93e559b2018-02-20 18:55:06 +0000158
Rui Ueyamac03c9042018-02-20 21:08:47 +0000159 // If existing symbol is undefined, replace it.
160 if (!Existing.isDefined()) {
161 DEBUG(dbgs() << "resolving existing undefined symbol: "
162 << Existing.getName() << "\n");
163 return true;
164 }
165
166 // Now we have two defined symbols. If the new one is weak, we can ignore it.
167 if ((NewFlags & WASM_SYMBOL_BINDING_MASK) == WASM_SYMBOL_BINDING_WEAK) {
168 DEBUG(dbgs() << "existing symbol takes precedence\n");
169 return false;
170 }
171
172 // If the existing symbol is weak, we should replace it.
173 if (Existing.isWeak()) {
174 DEBUG(dbgs() << "replacing existing weak symbol\n");
175 return true;
176 }
177
178 // Neither symbol is week. They conflict.
179 error("duplicate symbol: " + toString(Existing) + "\n>>> defined in " +
180 toString(Existing.getFile()) + "\n>>> defined in " + toString(NewFile));
181 return true;
Sam Clegg93e559b2018-02-20 18:55:06 +0000182}
183
184Symbol *SymbolTable::addDefinedFunction(StringRef Name, uint32_t Flags,
185 InputFile *F, InputFunction *Function) {
186 DEBUG(dbgs() << "addDefinedFunction: " << Name << "\n");
187 Symbol *S;
188 bool WasInserted;
189 std::tie(S, WasInserted) = insert(Name);
Sam Cleggf60b0d52018-02-20 18:55:07 +0000190 if (WasInserted || shouldReplace(*S, F, Flags, Function, true))
Sam Clegg93e559b2018-02-20 18:55:06 +0000191 replaceSymbol<DefinedFunction>(S, Name, Flags, F, Function);
192 return S;
193}
194
195Symbol *SymbolTable::addDefinedGlobal(StringRef Name, uint32_t Flags,
196 InputFile *F, InputSegment *Segment,
197 uint32_t Address) {
198 DEBUG(dbgs() << "addDefinedGlobal:" << Name << " addr:" << Address << "\n");
199 Symbol *S;
200 bool WasInserted;
201 std::tie(S, WasInserted) = insert(Name);
Sam Cleggf60b0d52018-02-20 18:55:07 +0000202 if (WasInserted || shouldReplace(*S, F, Flags, Segment, false))
Sam Clegg93e559b2018-02-20 18:55:06 +0000203 replaceSymbol<DefinedGlobal>(S, Name, Flags, F, Segment, Address);
Sam Cleggc94d3932017-11-17 18:14:09 +0000204 return S;
205}
206
Sam Clegg20db3812018-01-10 00:52:20 +0000207Symbol *SymbolTable::addUndefined(StringRef Name, Symbol::Kind Kind,
208 uint32_t Flags, InputFile *F,
Sam Clegg8d146bb2018-01-09 23:56:44 +0000209 const WasmSignature *Type) {
Sam Clegg20db3812018-01-10 00:52:20 +0000210 DEBUG(dbgs() << "addUndefined: " << Name << "\n");
Rui Ueyamac03c9042018-02-20 21:08:47 +0000211
Sam Cleggc94d3932017-11-17 18:14:09 +0000212 Symbol *S;
213 bool WasInserted;
Sam Clegg20db3812018-01-10 00:52:20 +0000214 std::tie(S, WasInserted) = insert(Name);
Rui Ueyamac03c9042018-02-20 21:08:47 +0000215
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000216 bool IsFunction = Kind == Symbol::UndefinedFunctionKind;
Sam Cleggc94d3932017-11-17 18:14:09 +0000217 if (WasInserted) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000218 if (IsFunction)
219 replaceSymbol<UndefinedFunction>(S, Name, Flags, F, Type);
220 else
221 replaceSymbol<UndefinedGlobal>(S, Name, Flags, F);
Rui Ueyamac03c9042018-02-20 21:08:47 +0000222 return S;
223 }
224
225 if (auto *Lazy = dyn_cast<LazySymbol>(S)) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000226 DEBUG(dbgs() << "resolved by existing lazy\n");
Rui Ueyamac03c9042018-02-20 21:08:47 +0000227 cast<ArchiveFile>(Lazy->getFile())->addMember(&Lazy->getArchiveSymbol());
228 return S;
229 }
230
231 if (S->isDefined()) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000232 DEBUG(dbgs() << "resolved by existing\n");
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000233 checkSymbolTypes(*S, *F, IsFunction, Type);
Sam Cleggc94d3932017-11-17 18:14:09 +0000234 }
235 return S;
236}
237
238void SymbolTable::addLazy(ArchiveFile *F, const Archive::Symbol *Sym) {
Sam Clegga681a112017-12-06 03:10:39 +0000239 DEBUG(dbgs() << "addLazy: " << Sym->getName() << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000240 StringRef Name = Sym->getName();
Rui Ueyamac03c9042018-02-20 21:08:47 +0000241
Sam Cleggc94d3932017-11-17 18:14:09 +0000242 Symbol *S;
243 bool WasInserted;
244 std::tie(S, WasInserted) = insert(Name);
Rui Ueyamac03c9042018-02-20 21:08:47 +0000245
Sam Cleggc94d3932017-11-17 18:14:09 +0000246 if (WasInserted) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000247 replaceSymbol<LazySymbol>(S, Name, F, *Sym);
Rui Ueyamac03c9042018-02-20 21:08:47 +0000248 return;
249 }
250
251 // If there is an existing undefined symbol, load a new one from the archive.
252 if (S->isUndefined()) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000253 DEBUG(dbgs() << "replacing existing undefined\n");
254 F->addMember(Sym);
255 }
256}
Sam Clegge0f6fcd2018-01-12 22:25:17 +0000257
258bool SymbolTable::addComdat(StringRef Name, ObjFile *F) {
259 DEBUG(dbgs() << "addComdat: " << Name << "\n");
260 ObjFile *&File = ComdatMap[CachedHashStringRef(Name)];
261 if (File) {
262 DEBUG(dbgs() << "COMDAT already defined\n");
263 return false;
264 }
265 File = F;
266 return true;
267}
268
269ObjFile *SymbolTable::findComdat(StringRef Name) const {
270 auto It = ComdatMap.find(CachedHashStringRef(Name));
271 return It == ComdatMap.end() ? nullptr : It->second;
272}