You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			396 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			396 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===- SymbolTable.cpp ----------------------------------------------------===//
 | |
| //
 | |
| //                             The LLVM Linker
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "SymbolTable.h"
 | |
| #include "Config.h"
 | |
| #include "Driver.h"
 | |
| #include "LTO.h"
 | |
| #include "Symbols.h"
 | |
| #include "lld/Common/ErrorHandler.h"
 | |
| #include "lld/Common/Memory.h"
 | |
| #include "llvm/IR/LLVMContext.h"
 | |
| #include "llvm/Support/Debug.h"
 | |
| #include "llvm/Support/raw_ostream.h"
 | |
| #include <utility>
 | |
| 
 | |
| using namespace llvm;
 | |
| 
 | |
| namespace lld {
 | |
| namespace coff {
 | |
| 
 | |
| SymbolTable *Symtab;
 | |
| 
 | |
| void SymbolTable::addFile(InputFile *File) {
 | |
|   log("Reading " + toString(File));
 | |
|   File->parse();
 | |
| 
 | |
|   MachineTypes MT = File->getMachineType();
 | |
|   if (Config->Machine == IMAGE_FILE_MACHINE_UNKNOWN) {
 | |
|     Config->Machine = MT;
 | |
|   } else if (MT != IMAGE_FILE_MACHINE_UNKNOWN && Config->Machine != MT) {
 | |
|     fatal(toString(File) + ": machine type " + machineToStr(MT) +
 | |
|           " conflicts with " + machineToStr(Config->Machine));
 | |
|   }
 | |
| 
 | |
|   if (auto *F = dyn_cast<ObjFile>(File)) {
 | |
|     ObjFile::Instances.push_back(F);
 | |
|   } else if (auto *F = dyn_cast<BitcodeFile>(File)) {
 | |
|     BitcodeFile::Instances.push_back(F);
 | |
|   } else if (auto *F = dyn_cast<ImportFile>(File)) {
 | |
|     ImportFile::Instances.push_back(F);
 | |
|   }
 | |
| 
 | |
|   StringRef S = File->getDirectives();
 | |
|   if (S.empty())
 | |
|     return;
 | |
| 
 | |
|   log("Directives: " + toString(File) + ": " + S);
 | |
|   Driver->parseDirectives(S);
 | |
| }
 | |
| 
 | |
| static void errorOrWarn(const Twine &S) {
 | |
|   if (Config->Force)
 | |
|     warn(S);
 | |
|   else
 | |
|     error(S);
 | |
| }
 | |
| 
 | |
| void SymbolTable::reportRemainingUndefines() {
 | |
|   SmallPtrSet<Symbol *, 8> Undefs;
 | |
|   DenseMap<Symbol *, Symbol *> LocalImports;
 | |
| 
 | |
|   for (auto &I : SymMap) {
 | |
|     Symbol *Sym = I.second;
 | |
|     auto *Undef = dyn_cast<Undefined>(Sym);
 | |
|     if (!Undef)
 | |
|       continue;
 | |
|     if (!Sym->IsUsedInRegularObj)
 | |
|       continue;
 | |
| 
 | |
|     StringRef Name = Undef->getName();
 | |
| 
 | |
|     // A weak alias may have been resolved, so check for that.
 | |
|     if (Defined *D = Undef->getWeakAlias()) {
 | |
|       // We want to replace Sym with D. However, we can't just blindly
 | |
|       // copy sizeof(SymbolUnion) bytes from D to Sym because D may be an
 | |
|       // internal symbol, and internal symbols are stored as "unparented"
 | |
|       // Symbols. For that reason we need to check which type of symbol we
 | |
|       // are dealing with and copy the correct number of bytes.
 | |
|       if (isa<DefinedRegular>(D))
 | |
|         memcpy(Sym, D, sizeof(DefinedRegular));
 | |
|       else if (isa<DefinedAbsolute>(D))
 | |
|         memcpy(Sym, D, sizeof(DefinedAbsolute));
 | |
|       else
 | |
|         memcpy(Sym, D, sizeof(SymbolUnion));
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     // If we can resolve a symbol by removing __imp_ prefix, do that.
 | |
|     // This odd rule is for compatibility with MSVC linker.
 | |
|     if (Name.startswith("__imp_")) {
 | |
|       Symbol *Imp = find(Name.substr(strlen("__imp_")));
 | |
|       if (Imp && isa<Defined>(Imp)) {
 | |
|         auto *D = cast<Defined>(Imp);
 | |
|         replaceSymbol<DefinedLocalImport>(Sym, Name, D);
 | |
|         LocalImportChunks.push_back(cast<DefinedLocalImport>(Sym)->getChunk());
 | |
|         LocalImports[Sym] = D;
 | |
|         continue;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Remaining undefined symbols are not fatal if /force is specified.
 | |
|     // They are replaced with dummy defined symbols.
 | |
|     if (Config->Force)
 | |
|       replaceSymbol<DefinedAbsolute>(Sym, Name, 0);
 | |
|     Undefs.insert(Sym);
 | |
|   }
 | |
| 
 | |
|   if (Undefs.empty() && LocalImports.empty())
 | |
|     return;
 | |
| 
 | |
|   for (Symbol *B : Config->GCRoot) {
 | |
|     if (Undefs.count(B))
 | |
|       errorOrWarn("<root>: undefined symbol: " + B->getName());
 | |
|     if (Config->WarnLocallyDefinedImported)
 | |
|       if (Symbol *Imp = LocalImports.lookup(B))
 | |
|         warn("<root>: locally defined symbol imported: " + Imp->getName() +
 | |
|              " (defined in " + toString(Imp->getFile()) + ")");
 | |
|   }
 | |
| 
 | |
|   for (ObjFile *File : ObjFile::Instances) {
 | |
|     for (Symbol *Sym : File->getSymbols()) {
 | |
|       if (!Sym)
 | |
|         continue;
 | |
|       if (Undefs.count(Sym))
 | |
|         errorOrWarn(toString(File) + ": undefined symbol: " + Sym->getName());
 | |
|       if (Config->WarnLocallyDefinedImported)
 | |
|         if (Symbol *Imp = LocalImports.lookup(Sym))
 | |
|           warn(toString(File) + ": locally defined symbol imported: " +
 | |
|                Imp->getName() + " (defined in " + toString(Imp->getFile()) +
 | |
|                ")");
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| std::pair<Symbol *, bool> SymbolTable::insert(StringRef Name) {
 | |
|   Symbol *&Sym = SymMap[CachedHashStringRef(Name)];
 | |
|   if (Sym)
 | |
|     return {Sym, false};
 | |
|   Sym = (Symbol *)make<SymbolUnion>();
 | |
|   Sym->IsUsedInRegularObj = false;
 | |
|   Sym->PendingArchiveLoad = false;
 | |
|   return {Sym, true};
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::addUndefined(StringRef Name, InputFile *F,
 | |
|                                   bool IsWeakAlias) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(Name);
 | |
|   if (!F || !isa<BitcodeFile>(F))
 | |
|     S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || (isa<Lazy>(S) && IsWeakAlias)) {
 | |
|     replaceSymbol<Undefined>(S, Name);
 | |
|     return S;
 | |
|   }
 | |
|   if (auto *L = dyn_cast<Lazy>(S)) {
 | |
|     if (!S->PendingArchiveLoad) {
 | |
|       S->PendingArchiveLoad = true;
 | |
|       L->File->addMember(&L->Sym);
 | |
|     }
 | |
|   }
 | |
|   return S;
 | |
| }
 | |
| 
 | |
| void SymbolTable::addLazy(ArchiveFile *F, const Archive::Symbol Sym) {
 | |
|   StringRef Name = Sym.getName();
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(Name);
 | |
|   if (WasInserted) {
 | |
|     replaceSymbol<Lazy>(S, F, Sym);
 | |
|     return;
 | |
|   }
 | |
|   auto *U = dyn_cast<Undefined>(S);
 | |
|   if (!U || U->WeakAlias || S->PendingArchiveLoad)
 | |
|     return;
 | |
|   S->PendingArchiveLoad = true;
 | |
|   F->addMember(&Sym);
 | |
| }
 | |
| 
 | |
| void SymbolTable::reportDuplicate(Symbol *Existing, InputFile *NewFile) {
 | |
|   error("duplicate symbol: " + toString(*Existing) + " in " +
 | |
|         toString(Existing->getFile()) + " and in " + toString(NewFile));
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::addAbsolute(StringRef N, COFFSymbolRef Sym) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(N);
 | |
|   S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || isa<Undefined>(S) || isa<Lazy>(S))
 | |
|     replaceSymbol<DefinedAbsolute>(S, N, Sym);
 | |
|   else if (!isa<DefinedCOFF>(S))
 | |
|     reportDuplicate(S, nullptr);
 | |
|   return S;
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::addAbsolute(StringRef N, uint64_t VA) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(N);
 | |
|   S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || isa<Undefined>(S) || isa<Lazy>(S))
 | |
|     replaceSymbol<DefinedAbsolute>(S, N, VA);
 | |
|   else if (!isa<DefinedCOFF>(S))
 | |
|     reportDuplicate(S, nullptr);
 | |
|   return S;
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::addSynthetic(StringRef N, Chunk *C) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(N);
 | |
|   S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || isa<Undefined>(S) || isa<Lazy>(S))
 | |
|     replaceSymbol<DefinedSynthetic>(S, N, C);
 | |
|   else if (!isa<DefinedCOFF>(S))
 | |
|     reportDuplicate(S, nullptr);
 | |
|   return S;
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::addRegular(InputFile *F, StringRef N,
 | |
|                                 const coff_symbol_generic *Sym,
 | |
|                                 SectionChunk *C) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(N);
 | |
|   if (!isa<BitcodeFile>(F))
 | |
|     S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || !isa<DefinedRegular>(S))
 | |
|     replaceSymbol<DefinedRegular>(S, F, N, /*IsCOMDAT*/ false,
 | |
|                                   /*IsExternal*/ true, Sym, C);
 | |
|   else
 | |
|     reportDuplicate(S, F);
 | |
|   return S;
 | |
| }
 | |
| 
 | |
| std::pair<Symbol *, bool>
 | |
| SymbolTable::addComdat(InputFile *F, StringRef N,
 | |
|                        const coff_symbol_generic *Sym) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(N);
 | |
|   if (!isa<BitcodeFile>(F))
 | |
|     S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || !isa<DefinedRegular>(S)) {
 | |
|     replaceSymbol<DefinedRegular>(S, F, N, /*IsCOMDAT*/ true,
 | |
|                                   /*IsExternal*/ true, Sym, nullptr);
 | |
|     return {S, true};
 | |
|   }
 | |
|   if (!cast<DefinedRegular>(S)->isCOMDAT())
 | |
|     reportDuplicate(S, F);
 | |
|   return {S, false};
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::addCommon(InputFile *F, StringRef N, uint64_t Size,
 | |
|                                const coff_symbol_generic *Sym, CommonChunk *C) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(N);
 | |
|   if (!isa<BitcodeFile>(F))
 | |
|     S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || !isa<DefinedCOFF>(S))
 | |
|     replaceSymbol<DefinedCommon>(S, F, N, Size, Sym, C);
 | |
|   else if (auto *DC = dyn_cast<DefinedCommon>(S))
 | |
|     if (Size > DC->getSize())
 | |
|       replaceSymbol<DefinedCommon>(S, F, N, Size, Sym, C);
 | |
|   return S;
 | |
| }
 | |
| 
 | |
| DefinedImportData *SymbolTable::addImportData(StringRef N, ImportFile *F) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(N);
 | |
|   S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || isa<Undefined>(S) || isa<Lazy>(S)) {
 | |
|     replaceSymbol<DefinedImportData>(S, N, F);
 | |
|     return cast<DefinedImportData>(S);
 | |
|   }
 | |
| 
 | |
|   reportDuplicate(S, F);
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| DefinedImportThunk *SymbolTable::addImportThunk(StringRef Name,
 | |
|                                                DefinedImportData *ID,
 | |
|                                                uint16_t Machine) {
 | |
|   Symbol *S;
 | |
|   bool WasInserted;
 | |
|   std::tie(S, WasInserted) = insert(Name);
 | |
|   S->IsUsedInRegularObj = true;
 | |
|   if (WasInserted || isa<Undefined>(S) || isa<Lazy>(S)) {
 | |
|     replaceSymbol<DefinedImportThunk>(S, Name, ID, Machine);
 | |
|     return cast<DefinedImportThunk>(S);
 | |
|   }
 | |
| 
 | |
|   reportDuplicate(S, ID->File);
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| std::vector<Chunk *> SymbolTable::getChunks() {
 | |
|   std::vector<Chunk *> Res;
 | |
|   for (ObjFile *File : ObjFile::Instances) {
 | |
|     ArrayRef<Chunk *> V = File->getChunks();
 | |
|     Res.insert(Res.end(), V.begin(), V.end());
 | |
|   }
 | |
|   return Res;
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::find(StringRef Name) {
 | |
|   auto It = SymMap.find(CachedHashStringRef(Name));
 | |
|   if (It == SymMap.end())
 | |
|     return nullptr;
 | |
|   return It->second;
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::findUnderscore(StringRef Name) {
 | |
|   if (Config->Machine == I386)
 | |
|     return find(("_" + Name).str());
 | |
|   return find(Name);
 | |
| }
 | |
| 
 | |
| StringRef SymbolTable::findByPrefix(StringRef Prefix) {
 | |
|   for (auto Pair : SymMap) {
 | |
|     StringRef Name = Pair.first.val();
 | |
|     if (Name.startswith(Prefix))
 | |
|       return Name;
 | |
|   }
 | |
|   return "";
 | |
| }
 | |
| 
 | |
| StringRef SymbolTable::findMangle(StringRef Name) {
 | |
|   if (Symbol *Sym = find(Name))
 | |
|     if (!isa<Undefined>(Sym))
 | |
|       return Name;
 | |
|   if (Config->Machine != I386)
 | |
|     return findByPrefix(("?" + Name + "@@Y").str());
 | |
|   if (!Name.startswith("_"))
 | |
|     return "";
 | |
|   // Search for x86 stdcall function.
 | |
|   StringRef S = findByPrefix((Name + "@").str());
 | |
|   if (!S.empty())
 | |
|     return S;
 | |
|   // Search for x86 fastcall function.
 | |
|   S = findByPrefix(("@" + Name.substr(1) + "@").str());
 | |
|   if (!S.empty())
 | |
|     return S;
 | |
|   // Search for x86 vectorcall function.
 | |
|   S = findByPrefix((Name.substr(1) + "@@").str());
 | |
|   if (!S.empty())
 | |
|     return S;
 | |
|   // Search for x86 C++ non-member function.
 | |
|   return findByPrefix(("?" + Name.substr(1) + "@@Y").str());
 | |
| }
 | |
| 
 | |
| void SymbolTable::mangleMaybe(Symbol *B) {
 | |
|   auto *U = dyn_cast<Undefined>(B);
 | |
|   if (!U || U->WeakAlias)
 | |
|     return;
 | |
|   StringRef Alias = findMangle(U->getName());
 | |
|   if (!Alias.empty()) {
 | |
|     log(U->getName() + " aliased to " + Alias);
 | |
|     U->WeakAlias = addUndefined(Alias);
 | |
|   }
 | |
| }
 | |
| 
 | |
| Symbol *SymbolTable::addUndefined(StringRef Name) {
 | |
|   return addUndefined(Name, nullptr, false);
 | |
| }
 | |
| 
 | |
| std::vector<StringRef> SymbolTable::compileBitcodeFiles() {
 | |
|   LTO.reset(new BitcodeCompiler);
 | |
|   for (BitcodeFile *F : BitcodeFile::Instances)
 | |
|     LTO->add(*F);
 | |
|   return LTO->compile();
 | |
| }
 | |
| 
 | |
| void SymbolTable::addCombinedLTOObjects() {
 | |
|   if (BitcodeFile::Instances.empty())
 | |
|     return;
 | |
|   for (StringRef Object : compileBitcodeFiles()) {
 | |
|     auto *Obj = make<ObjFile>(MemoryBufferRef(Object, "lto.tmp"));
 | |
|     Obj->parse();
 | |
|     ObjFile::Instances.push_back(Obj);
 | |
|   }
 | |
| }
 | |
| 
 | |
| } // namespace coff
 | |
| } // namespace lld
 |