blob: c89c340ff91415df78ce2ce5ed35cf03b5a0f86d [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 Clegg8d146bb2018-01-09 23:56:44 +000013#include "InputFunction.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"
Sam Cleggc94d3932017-11-17 18:14:09 +000017
18#include <unordered_set>
19
20#define DEBUG_TYPE "lld"
21
22using namespace llvm;
23using 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() {
37 std::unordered_set<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,
83 const WasmSymbol &New,
84 const WasmSignature *NewSig) {
85 if (Existing.isLazy())
Sam Cleggc94d3932017-11-17 18:14:09 +000086 return;
87
Sam Cleggb8621592017-11-30 01:40:08 +000088 bool NewIsFunction = New.Type == WasmSymbol::SymbolType::FUNCTION_EXPORT ||
89 New.Type == WasmSymbol::SymbolType::FUNCTION_IMPORT;
90
91 // First check the symbol types match (i.e. either both are function
92 // symbols or both are data symbols).
93 if (Existing.isFunction() != NewIsFunction) {
94 error("symbol type mismatch: " + New.Name + "\n>>> defined as " +
95 (Existing.isFunction() ? "Function" : "Global") + " in " +
96 toString(Existing.getFile()) + "\n>>> defined as " +
97 (NewIsFunction ? "Function" : "Global") + " in " + F.getName());
98 return;
99 }
100
101 // For function symbols, optionally check the function signature matches too.
102 if (!NewIsFunction || !Config->CheckSignatures)
Sam Cleggc94d3932017-11-17 18:14:09 +0000103 return;
Sam Clegg31de2f02017-12-07 03:19:53 +0000104 // Skip the signature check if the existing function has no signature (e.g.
105 // if it is an undefined symbol generated by --undefined command line flag).
106 if (!Existing.hasFunctionType())
107 return;
Sam Cleggc94d3932017-11-17 18:14:09 +0000108
Sam Cleggb8621592017-11-30 01:40:08 +0000109 DEBUG(dbgs() << "checkSymbolTypes: " << New.Name << "\n");
110 assert(NewSig);
111
112 const WasmSignature &OldSig = Existing.getFunctionType();
113 if (*NewSig == OldSig)
114 return;
115
116 error("function signature mismatch: " + New.Name + "\n>>> defined as " +
117 toString(OldSig) + " in " + toString(Existing.getFile()) +
118 "\n>>> defined as " + toString(*NewSig) + " in " + F.getName());
Sam Cleggc94d3932017-11-17 18:14:09 +0000119}
120
121Symbol *SymbolTable::addDefinedGlobal(StringRef Name) {
122 DEBUG(dbgs() << "addDefinedGlobal: " << Name << "\n");
123 Symbol *S;
124 bool WasInserted;
125 std::tie(S, WasInserted) = insert(Name);
126 if (WasInserted)
127 S->update(Symbol::DefinedGlobalKind);
128 else if (!S->isGlobal())
129 error("symbol type mismatch: " + Name);
130 return S;
131}
132
133Symbol *SymbolTable::addDefined(InputFile *F, const WasmSymbol *Sym,
Sam Clegg8d146bb2018-01-09 23:56:44 +0000134 const InputSegment *Segment,
135 InputFunction *Function) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000136 DEBUG(dbgs() << "addDefined: " << Sym->Name << "\n");
137 Symbol *S;
138 bool WasInserted;
139 Symbol::Kind Kind = Symbol::DefinedFunctionKind;
140 if (Sym->Type == WasmSymbol::SymbolType::GLOBAL_EXPORT)
141 Kind = Symbol::DefinedGlobalKind;
142
143 std::tie(S, WasInserted) = insert(Sym->Name);
144 if (WasInserted) {
Sam Clegg8d146bb2018-01-09 23:56:44 +0000145 S->update(Kind, F, Sym, Segment, Function);
Sam Cleggb8621592017-11-30 01:40:08 +0000146 } else if (S->isLazy()) {
147 // The existing symbol is lazy. Replace it without checking types since
148 // lazy symbols don't have any type information.
149 DEBUG(dbgs() << "replacing existing lazy symbol: " << Sym->Name << "\n");
Sam Clegg8d146bb2018-01-09 23:56:44 +0000150 S->update(Kind, F, Sym, Segment, Function);
Sam Cleggc94d3932017-11-17 18:14:09 +0000151 } else if (!S->isDefined()) {
152 // The existing symbol table entry is undefined. The new symbol replaces
Sam Clegg49ed9262017-12-01 00:53:21 +0000153 // it, after checking the type matches
Sam Cleggc94d3932017-11-17 18:14:09 +0000154 DEBUG(dbgs() << "resolving existing undefined symbol: " << Sym->Name
155 << "\n");
Sam Clegg8d146bb2018-01-09 23:56:44 +0000156 checkSymbolTypes(*S, *F, *Sym, Function ? &Function->Signature : nullptr);
157 S->update(Kind, F, Sym, Segment, Function);
Sam Cleggc94d3932017-11-17 18:14:09 +0000158 } else if (Sym->isWeak()) {
159 // the new symbol is weak we can ignore it
Sam Clegg49ed9262017-12-01 00:53:21 +0000160 DEBUG(dbgs() << "existing symbol takes precedence\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000161 } else if (S->isWeak()) {
162 // the new symbol is not weak and the existing symbol is, so we replace
163 // it
164 DEBUG(dbgs() << "replacing existing weak symbol\n");
Sam Clegg8d146bb2018-01-09 23:56:44 +0000165 checkSymbolTypes(*S, *F, *Sym, Function ? &Function->Signature : nullptr);
166 S->update(Kind, F, Sym, Segment, Function);
Sam Cleggc94d3932017-11-17 18:14:09 +0000167 } else {
Sam Clegg49ed9262017-12-01 00:53:21 +0000168 // neither symbol is week. They conflict.
Sam Cleggc94d3932017-11-17 18:14:09 +0000169 reportDuplicate(S, F);
170 }
171 return S;
172}
173
Sam Cleggb8621592017-11-30 01:40:08 +0000174Symbol *SymbolTable::addUndefinedFunction(StringRef Name,
175 const WasmSignature *Type) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000176 Symbol *S;
177 bool WasInserted;
178 std::tie(S, WasInserted) = insert(Name);
Sam Cleggb8621592017-11-30 01:40:08 +0000179 if (WasInserted) {
Sam Clegg8d146bb2018-01-09 23:56:44 +0000180 S->update(Symbol::UndefinedFunctionKind);
181 S->setFunctionType(Type);
Sam Cleggb8621592017-11-30 01:40:08 +0000182 } else if (!S->isFunction()) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000183 error("symbol type mismatch: " + Name);
Sam Cleggb8621592017-11-30 01:40:08 +0000184 }
Sam Cleggc94d3932017-11-17 18:14:09 +0000185 return S;
186}
187
Sam Clegg8d146bb2018-01-09 23:56:44 +0000188Symbol *SymbolTable::addUndefined(InputFile *F, const WasmSymbol *Sym,
189 const WasmSignature *Type) {
Sam Clegga681a112017-12-06 03:10:39 +0000190 DEBUG(dbgs() << "addUndefined: " << Sym->Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000191 Symbol *S;
192 bool WasInserted;
193 Symbol::Kind Kind = Symbol::UndefinedFunctionKind;
194 if (Sym->Type == WasmSymbol::SymbolType::GLOBAL_IMPORT)
195 Kind = Symbol::UndefinedGlobalKind;
196 std::tie(S, WasInserted) = insert(Sym->Name);
197 if (WasInserted) {
Sam Clegg8d146bb2018-01-09 23:56:44 +0000198 S->update(Kind, F, Sym);
199 if (Type)
200 S->setFunctionType(Type);
Sam Cleggc94d3932017-11-17 18:14:09 +0000201 } else if (S->isLazy()) {
202 DEBUG(dbgs() << "resolved by existing lazy\n");
203 auto *AF = cast<ArchiveFile>(S->getFile());
204 AF->addMember(&S->getArchiveSymbol());
205 } else if (S->isDefined()) {
206 DEBUG(dbgs() << "resolved by existing\n");
Sam Clegg8d146bb2018-01-09 23:56:44 +0000207 checkSymbolTypes(*S, *F, *Sym, Type);
Sam Cleggc94d3932017-11-17 18:14:09 +0000208 }
209 return S;
210}
211
212void SymbolTable::addLazy(ArchiveFile *F, const Archive::Symbol *Sym) {
Sam Clegga681a112017-12-06 03:10:39 +0000213 DEBUG(dbgs() << "addLazy: " << Sym->getName() << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000214 StringRef Name = Sym->getName();
215 Symbol *S;
216 bool WasInserted;
217 std::tie(S, WasInserted) = insert(Name);
218 if (WasInserted) {
219 S->update(Symbol::LazyKind, F);
220 S->setArchiveSymbol(*Sym);
221 } else if (S->isUndefined()) {
222 // There is an existing undefined symbol. The can load from the
223 // archive.
224 DEBUG(dbgs() << "replacing existing undefined\n");
225 F->addMember(Sym);
226 }
227}