blob: d9a6fa1f04f50651f9693ae618d6990ff7598abb [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 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"
Sam Cleggc94d3932017-11-17 18:14:09 +000016
17#include <unordered_set>
18
19#define DEBUG_TYPE "lld"
20
21using namespace llvm;
22using 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() {
36 std::unordered_set<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};
68 Sym = make<Symbol>(Name, false);
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
73void SymbolTable::reportDuplicate(Symbol *Existing, InputFile *NewFile) {
74 error("duplicate symbol: " + toString(*Existing) + "\n>>> defined in " +
75 toString(Existing->getFile()) + "\n>>> defined in " +
Sam Clegge2ed0922017-11-27 22:49:16 +000076 toString(NewFile));
Sam Cleggc94d3932017-11-17 18:14:09 +000077}
78
Sam Cleggb8621592017-11-30 01:40:08 +000079// Get the signature for a given function symbol, either by looking
80// it up in function sections (for defined functions), of the imports section
81// (for imported functions).
82static const WasmSignature *getFunctionSig(const ObjFile &Obj,
83 const WasmSymbol &Sym) {
84 DEBUG(dbgs() << "getFunctionSig: " << Sym.Name << "\n");
85 const WasmObjectFile *WasmObj = Obj.getWasmObj();
86 uint32_t FunctionType;
87 if (Obj.isImportedFunction(Sym.ElementIndex)) {
88 const WasmImport &Import = WasmObj->imports()[Sym.ImportIndex];
89 FunctionType = Import.SigIndex;
90 } else {
91 uint32_t FuntionIndex = Sym.ElementIndex - Obj.NumFunctionImports();
92 FunctionType = WasmObj->functionTypes()[FuntionIndex];
93 }
94 return &WasmObj->types()[FunctionType];
95}
96
97// Check the type of new symbol matches that of the symbol is replacing.
98// For functions this can also involve verifying that the signatures match.
99static void checkSymbolTypes(const Symbol &Existing, const InputFile &F,
100 const WasmSymbol &New,
101 const WasmSignature *NewSig) {
102 if (Existing.isLazy())
Sam Cleggc94d3932017-11-17 18:14:09 +0000103 return;
104
Sam Cleggb8621592017-11-30 01:40:08 +0000105 bool NewIsFunction = New.Type == WasmSymbol::SymbolType::FUNCTION_EXPORT ||
106 New.Type == WasmSymbol::SymbolType::FUNCTION_IMPORT;
107
108 // First check the symbol types match (i.e. either both are function
109 // symbols or both are data symbols).
110 if (Existing.isFunction() != NewIsFunction) {
111 error("symbol type mismatch: " + New.Name + "\n>>> defined as " +
112 (Existing.isFunction() ? "Function" : "Global") + " in " +
113 toString(Existing.getFile()) + "\n>>> defined as " +
114 (NewIsFunction ? "Function" : "Global") + " in " + F.getName());
115 return;
116 }
117
118 // For function symbols, optionally check the function signature matches too.
119 if (!NewIsFunction || !Config->CheckSignatures)
Sam Cleggc94d3932017-11-17 18:14:09 +0000120 return;
Sam Clegg31de2f02017-12-07 03:19:53 +0000121 // Skip the signature check if the existing function has no signature (e.g.
122 // if it is an undefined symbol generated by --undefined command line flag).
123 if (!Existing.hasFunctionType())
124 return;
Sam Cleggc94d3932017-11-17 18:14:09 +0000125
Sam Cleggb8621592017-11-30 01:40:08 +0000126 DEBUG(dbgs() << "checkSymbolTypes: " << New.Name << "\n");
127 assert(NewSig);
128
129 const WasmSignature &OldSig = Existing.getFunctionType();
130 if (*NewSig == OldSig)
131 return;
132
133 error("function signature mismatch: " + New.Name + "\n>>> defined as " +
134 toString(OldSig) + " in " + toString(Existing.getFile()) +
135 "\n>>> defined as " + toString(*NewSig) + " in " + F.getName());
Sam Cleggc94d3932017-11-17 18:14:09 +0000136}
137
138Symbol *SymbolTable::addDefinedGlobal(StringRef Name) {
139 DEBUG(dbgs() << "addDefinedGlobal: " << Name << "\n");
140 Symbol *S;
141 bool WasInserted;
142 std::tie(S, WasInserted) = insert(Name);
143 if (WasInserted)
144 S->update(Symbol::DefinedGlobalKind);
145 else if (!S->isGlobal())
146 error("symbol type mismatch: " + Name);
147 return S;
148}
149
150Symbol *SymbolTable::addDefined(InputFile *F, const WasmSymbol *Sym,
151 const InputSegment *Segment) {
152 DEBUG(dbgs() << "addDefined: " << Sym->Name << "\n");
153 Symbol *S;
154 bool WasInserted;
155 Symbol::Kind Kind = Symbol::DefinedFunctionKind;
Sam Cleggb8621592017-11-30 01:40:08 +0000156 const WasmSignature *NewSig = nullptr;
Sam Cleggc94d3932017-11-17 18:14:09 +0000157 if (Sym->Type == WasmSymbol::SymbolType::GLOBAL_EXPORT)
158 Kind = Symbol::DefinedGlobalKind;
Sam Cleggb8621592017-11-30 01:40:08 +0000159 else
160 NewSig = getFunctionSig(*cast<ObjFile>(F), *Sym);
Sam Cleggc94d3932017-11-17 18:14:09 +0000161
162 std::tie(S, WasInserted) = insert(Sym->Name);
163 if (WasInserted) {
Sam Cleggb8621592017-11-30 01:40:08 +0000164 S->update(Kind, F, Sym, Segment, NewSig);
165 } else if (S->isLazy()) {
166 // The existing symbol is lazy. Replace it without checking types since
167 // lazy symbols don't have any type information.
168 DEBUG(dbgs() << "replacing existing lazy symbol: " << Sym->Name << "\n");
169 S->update(Kind, F, Sym, Segment, NewSig);
Sam Cleggc94d3932017-11-17 18:14:09 +0000170 } else if (!S->isDefined()) {
171 // The existing symbol table entry is undefined. The new symbol replaces
Sam Clegg49ed9262017-12-01 00:53:21 +0000172 // it, after checking the type matches
Sam Cleggc94d3932017-11-17 18:14:09 +0000173 DEBUG(dbgs() << "resolving existing undefined symbol: " << Sym->Name
174 << "\n");
Sam Cleggb8621592017-11-30 01:40:08 +0000175 checkSymbolTypes(*S, *F, *Sym, NewSig);
176 S->update(Kind, F, Sym, Segment, NewSig);
Sam Cleggc94d3932017-11-17 18:14:09 +0000177 } else if (Sym->isWeak()) {
178 // the new symbol is weak we can ignore it
Sam Clegg49ed9262017-12-01 00:53:21 +0000179 DEBUG(dbgs() << "existing symbol takes precedence\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000180 } else if (S->isWeak()) {
181 // the new symbol is not weak and the existing symbol is, so we replace
182 // it
183 DEBUG(dbgs() << "replacing existing weak symbol\n");
Sam Cleggb8621592017-11-30 01:40:08 +0000184 checkSymbolTypes(*S, *F, *Sym, NewSig);
185 S->update(Kind, F, Sym, Segment, NewSig);
Sam Cleggc94d3932017-11-17 18:14:09 +0000186 } else {
Sam Clegg49ed9262017-12-01 00:53:21 +0000187 // neither symbol is week. They conflict.
Sam Cleggc94d3932017-11-17 18:14:09 +0000188 reportDuplicate(S, F);
189 }
190 return S;
191}
192
Sam Cleggb8621592017-11-30 01:40:08 +0000193Symbol *SymbolTable::addUndefinedFunction(StringRef Name,
194 const WasmSignature *Type) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000195 Symbol *S;
196 bool WasInserted;
197 std::tie(S, WasInserted) = insert(Name);
Sam Cleggb8621592017-11-30 01:40:08 +0000198 if (WasInserted) {
199 S->update(Symbol::UndefinedFunctionKind, nullptr, nullptr, nullptr, Type);
200 } else if (!S->isFunction()) {
Sam Cleggc94d3932017-11-17 18:14:09 +0000201 error("symbol type mismatch: " + Name);
Sam Cleggb8621592017-11-30 01:40:08 +0000202 }
Sam Cleggc94d3932017-11-17 18:14:09 +0000203 return S;
204}
205
206Symbol *SymbolTable::addUndefined(InputFile *F, const WasmSymbol *Sym) {
Sam Clegga681a112017-12-06 03:10:39 +0000207 DEBUG(dbgs() << "addUndefined: " << Sym->Name << "\n");
Sam Cleggc94d3932017-11-17 18:14:09 +0000208 Symbol *S;
209 bool WasInserted;
210 Symbol::Kind Kind = Symbol::UndefinedFunctionKind;
Sam Cleggb8621592017-11-30 01:40:08 +0000211 const WasmSignature *NewSig = nullptr;
Sam Cleggc94d3932017-11-17 18:14:09 +0000212 if (Sym->Type == WasmSymbol::SymbolType::GLOBAL_IMPORT)
213 Kind = Symbol::UndefinedGlobalKind;
Sam Cleggb8621592017-11-30 01:40:08 +0000214 else
215 NewSig = getFunctionSig(*cast<ObjFile>(F), *Sym);
Sam Cleggc94d3932017-11-17 18:14:09 +0000216 std::tie(S, WasInserted) = insert(Sym->Name);
217 if (WasInserted) {
Sam Cleggb8621592017-11-30 01:40:08 +0000218 S->update(Kind, F, Sym, nullptr, NewSig);
Sam Cleggc94d3932017-11-17 18:14:09 +0000219 } else if (S->isLazy()) {
220 DEBUG(dbgs() << "resolved by existing lazy\n");
221 auto *AF = cast<ArchiveFile>(S->getFile());
222 AF->addMember(&S->getArchiveSymbol());
223 } else if (S->isDefined()) {
224 DEBUG(dbgs() << "resolved by existing\n");
Sam Cleggb8621592017-11-30 01:40:08 +0000225 checkSymbolTypes(*S, *F, *Sym, NewSig);
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) {
237 S->update(Symbol::LazyKind, F);
238 S->setArchiveSymbol(*Sym);
239 } else if (S->isUndefined()) {
240 // There is an existing undefined symbol. The can load from the
241 // archive.
242 DEBUG(dbgs() << "replacing existing undefined\n");
243 F->addMember(Sym);
244 }
245}