blob: 32a7c3dab25f361b212861e79bb97f39f170177d [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};
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000069 Sym = reinterpret_cast<Symbol *>(make<SymbolUnion>());
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
Sam Cleggb8621592017-11-30 01:40:08 +000074// Check the type of new symbol matches that of the symbol is replacing.
75// For functions this can also involve verifying that the signatures match.
76static void checkSymbolTypes(const Symbol &Existing, const InputFile &F,
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000077 bool NewIsFunction, const WasmSignature *NewSig) {
Sam Cleggb8621592017-11-30 01:40:08 +000078 if (Existing.isLazy())
Sam Cleggc94d3932017-11-17 18:14:09 +000079 return;
80
Sam Cleggb8621592017-11-30 01:40:08 +000081 // First check the symbol types match (i.e. either both are function
82 // symbols or both are data symbols).
Sam Cleggcaca8d52018-02-17 00:44:21 +000083 if (isa<FunctionSymbol>(Existing) != NewIsFunction) {
Sam Clegg20db3812018-01-10 00:52:20 +000084 error("symbol type mismatch: " + Existing.getName() + "\n>>> defined as " +
Sam Cleggcaca8d52018-02-17 00:44:21 +000085 (isa<FunctionSymbol>(Existing) ? "Function" : "Global") + " in " +
Sam Cleggb8621592017-11-30 01:40:08 +000086 toString(Existing.getFile()) + "\n>>> defined as " +
87 (NewIsFunction ? "Function" : "Global") + " in " + F.getName());
88 return;
89 }
90
91 // For function symbols, optionally check the function signature matches too.
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000092 auto *ExistingFunc = dyn_cast<FunctionSymbol>(&Existing);
93 if (!ExistingFunc || !Config->CheckSignatures)
Sam Cleggc94d3932017-11-17 18:14:09 +000094 return;
Sam Cleggdfb0b2c2018-02-14 18:27:59 +000095
Sam Clegg3f8db982018-02-16 23:50:23 +000096 const WasmSignature *OldSig = ExistingFunc->getFunctionType();
97
Sam Clegg31de2f02017-12-07 03:19:53 +000098 // Skip the signature check if the existing function has no signature (e.g.
99 // if it is an undefined symbol generated by --undefined command line flag).
Sam Clegg3f8db982018-02-16 23:50:23 +0000100 if (OldSig == nullptr)
Sam Clegg31de2f02017-12-07 03:19:53 +0000101 return;
Sam Cleggc94d3932017-11-17 18:14:09 +0000102
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000103 DEBUG(dbgs() << "checkSymbolTypes: " << ExistingFunc->getName() << "\n");
Sam Cleggb8621592017-11-30 01:40:08 +0000104 assert(NewSig);
105
Sam Clegg3f8db982018-02-16 23:50:23 +0000106 if (*NewSig == *OldSig)
Sam Cleggb8621592017-11-30 01:40:08 +0000107 return;
108
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000109 error("function signature mismatch: " + ExistingFunc->getName() +
Sam Clegg3f8db982018-02-16 23:50:23 +0000110 "\n>>> defined as " + toString(*OldSig) + " in " +
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000111 toString(ExistingFunc->getFile()) + "\n>>> defined as " +
112 toString(*NewSig) + " in " + F.getName());
Sam Cleggc94d3932017-11-17 18:14:09 +0000113}
114
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000115static void checkSymbolTypes(const Symbol &Existing, const InputFile &F,
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000116 bool IsFunction, const InputChunk *Chunk) {
Sam Cleggfc50c622018-01-28 19:57:02 +0000117 const WasmSignature *Sig = nullptr;
118 if (auto *F = dyn_cast_or_null<InputFunction>(Chunk))
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000119 Sig = &F->Signature;
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000120 return checkSymbolTypes(Existing, F, IsFunction, Sig);
Sam Clegg24b3dcd2018-01-28 19:57:01 +0000121}
122
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000123DefinedFunction *SymbolTable::addSyntheticFunction(StringRef Name,
124 const WasmSignature *Type,
125 uint32_t Flags) {
Sam Clegga1892302018-02-13 20:14:26 +0000126 DEBUG(dbgs() << "addSyntheticFunction: " << Name << "\n");
Sam Clegg50686852018-01-12 18:35:13 +0000127 Symbol *S;
128 bool WasInserted;
129 std::tie(S, WasInserted) = insert(Name);
Sam Clegga1892302018-02-13 20:14:26 +0000130 assert(WasInserted);
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000131 return replaceSymbol<DefinedFunction>(S, Name, Flags, Type);
Sam Clegg50686852018-01-12 18:35:13 +0000132}
133
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000134DefinedGlobal *SymbolTable::addSyntheticGlobal(StringRef Name, uint32_t Flags) {
Sam Clegga1892302018-02-13 20:14:26 +0000135 DEBUG(dbgs() << "addSyntheticGlobal: " << Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000136 Symbol *S;
137 bool WasInserted;
138 std::tie(S, WasInserted) = insert(Name);
Sam Clegga1892302018-02-13 20:14:26 +0000139 assert(WasInserted);
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000140 return replaceSymbol<DefinedGlobal>(S, Name, Flags);
Sam Cleggc94d3932017-11-17 18:14:09 +0000141}
142
Sam Cleggf60b0d52018-02-20 18:55:07 +0000143static bool shouldReplace(const Symbol &Existing, InputFile *NewFile,
144 uint32_t NewFlags, InputChunk *NewChunk,
145 bool NewIsFunction) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000146 bool Replace = false;
147 bool CheckTypes = false;
Sam Cleggc94d3932017-11-17 18:14:09 +0000148
Sam Clegg93e559b2018-02-20 18:55:06 +0000149 if (Existing.isLazy()) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000150 // Existing symbol is lazy. Replace it without checking types since
Sam Cleggb8621592017-11-30 01:40:08 +0000151 // lazy symbols don't have any type information.
Sam Clegg93e559b2018-02-20 18:55:06 +0000152 DEBUG(dbgs() << "replacing existing lazy symbol: " << Existing.getName()
153 << "\n");
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000154 Replace = true;
Sam Clegg93e559b2018-02-20 18:55:06 +0000155 } else if (!Existing.isDefined()) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000156 // Existing symbol is undefined: replace it, while check types.
Sam Clegg93e559b2018-02-20 18:55:06 +0000157 DEBUG(dbgs() << "resolving existing undefined symbol: "
158 << Existing.getName() << "\n");
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000159 Replace = true;
160 CheckTypes = true;
Sam Cleggf60b0d52018-02-20 18:55:07 +0000161 } else if ((NewFlags & WASM_SYMBOL_BINDING_MASK) == WASM_SYMBOL_BINDING_WEAK) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000162 // the new symbol is weak we can ignore it
Sam Clegg49ed9262017-12-01 00:53:21 +0000163 DEBUG(dbgs() << "existing symbol takes precedence\n");
Sam Cleggc7e885b2018-02-20 17:20:44 +0000164 CheckTypes = true;
Sam Clegg93e559b2018-02-20 18:55:06 +0000165 } else if (Existing.isWeak()) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000166 // the existing symbol is, so we replace it
Sam Cleggc94d3932017-11-17 18:14:09 +0000167 DEBUG(dbgs() << "replacing existing weak symbol\n");
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000168 Replace = true;
169 CheckTypes = true;
Sam Cleggc94d3932017-11-17 18:14:09 +0000170 } else {
Sam Clegg49ed9262017-12-01 00:53:21 +0000171 // neither symbol is week. They conflict.
Sam Clegg93e559b2018-02-20 18:55:06 +0000172 error("duplicate symbol: " + toString(Existing) + "\n>>> defined in " +
173 toString(Existing.getFile()) + "\n>>> defined in " +
Sam Cleggf60b0d52018-02-20 18:55:07 +0000174 toString(NewFile));
Sam Cleggc94d3932017-11-17 18:14:09 +0000175 }
Sam Cleggc94d3932017-11-17 18:14:09 +0000176
Sam Clegg93e559b2018-02-20 18:55:06 +0000177 if (CheckTypes)
Sam Cleggf60b0d52018-02-20 18:55:07 +0000178 checkSymbolTypes(Existing, *NewFile, NewIsFunction, NewChunk);
Sam Clegg93e559b2018-02-20 18:55:06 +0000179
180 return Replace;
181}
182
183Symbol *SymbolTable::addDefinedFunction(StringRef Name, uint32_t Flags,
184 InputFile *F, InputFunction *Function) {
185 DEBUG(dbgs() << "addDefinedFunction: " << Name << "\n");
186 Symbol *S;
187 bool WasInserted;
188 std::tie(S, WasInserted) = insert(Name);
Sam Cleggf60b0d52018-02-20 18:55:07 +0000189 if (WasInserted || shouldReplace(*S, F, Flags, Function, true))
Sam Clegg93e559b2018-02-20 18:55:06 +0000190 replaceSymbol<DefinedFunction>(S, Name, Flags, F, Function);
191 return S;
192}
193
194Symbol *SymbolTable::addDefinedGlobal(StringRef Name, uint32_t Flags,
195 InputFile *F, InputSegment *Segment,
196 uint32_t Address) {
197 DEBUG(dbgs() << "addDefinedGlobal:" << Name << " addr:" << Address << "\n");
198 Symbol *S;
199 bool WasInserted;
200 std::tie(S, WasInserted) = insert(Name);
Sam Cleggf60b0d52018-02-20 18:55:07 +0000201 if (WasInserted || shouldReplace(*S, F, Flags, Segment, false))
Sam Clegg93e559b2018-02-20 18:55:06 +0000202 replaceSymbol<DefinedGlobal>(S, Name, Flags, F, Segment, Address);
Sam Cleggc94d3932017-11-17 18:14:09 +0000203 return S;
204}
205
Sam Clegg20db3812018-01-10 00:52:20 +0000206Symbol *SymbolTable::addUndefined(StringRef Name, Symbol::Kind Kind,
207 uint32_t Flags, InputFile *F,
Sam Clegg8d146bb2018-01-09 23:56:44 +0000208 const WasmSignature *Type) {
Sam Clegg20db3812018-01-10 00:52:20 +0000209 DEBUG(dbgs() << "addUndefined: " << Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000210 Symbol *S;
211 bool WasInserted;
Sam Clegg20db3812018-01-10 00:52:20 +0000212 std::tie(S, WasInserted) = insert(Name);
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000213 bool IsFunction = Kind == Symbol::UndefinedFunctionKind;
Sam Cleggc94d3932017-11-17 18:14:09 +0000214 if (WasInserted) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000215 if (IsFunction)
216 replaceSymbol<UndefinedFunction>(S, Name, Flags, F, Type);
217 else
218 replaceSymbol<UndefinedGlobal>(S, Name, Flags, F);
219 } else if (auto *LazySym = dyn_cast<LazySymbol>(S)) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000220 DEBUG(dbgs() << "resolved by existing lazy\n");
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000221 auto *AF = cast<ArchiveFile>(LazySym->getFile());
222 AF->addMember(&LazySym->getArchiveSymbol());
Sam Cleggc94d3932017-11-17 18:14:09 +0000223 } else if (S->isDefined()) {
224 DEBUG(dbgs() << "resolved by existing\n");
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000225 checkSymbolTypes(*S, *F, IsFunction, Type);
Sam Cleggc94d3932017-11-17 18:14:09 +0000226 }
227 return S;
228}
229
230void SymbolTable::addLazy(ArchiveFile *F, const Archive::Symbol *Sym) {
Sam Clegga681a112017-12-06 03:10:39 +0000231 DEBUG(dbgs() << "addLazy: " << Sym->getName() << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000232 StringRef Name = Sym->getName();
233 Symbol *S;
234 bool WasInserted;
235 std::tie(S, WasInserted) = insert(Name);
236 if (WasInserted) {
Sam Cleggdfb0b2c2018-02-14 18:27:59 +0000237 replaceSymbol<LazySymbol>(S, Name, F, *Sym);
Sam Cleggc94d3932017-11-17 18:14:09 +0000238 } else if (S->isUndefined()) {
239 // There is an existing undefined symbol. The can load from the
240 // archive.
241 DEBUG(dbgs() << "replacing existing undefined\n");
242 F->addMember(Sym);
243 }
244}
Sam Clegge0f6fcd2018-01-12 22:25:17 +0000245
246bool SymbolTable::addComdat(StringRef Name, ObjFile *F) {
247 DEBUG(dbgs() << "addComdat: " << Name << "\n");
248 ObjFile *&File = ComdatMap[CachedHashStringRef(Name)];
249 if (File) {
250 DEBUG(dbgs() << "COMDAT already defined\n");
251 return false;
252 }
253 File = F;
254 return true;
255}
256
257ObjFile *SymbolTable::findComdat(StringRef Name) const {
258 auto It = ComdatMap.find(CachedHashStringRef(Name));
259 return It == ComdatMap.end() ? nullptr : It->second;
260}