Index: COFF/CMakeLists.txt =================================================================== --- COFF/CMakeLists.txt +++ COFF/CMakeLists.txt @@ -17,6 +17,7 @@ LTO.cpp MapFile.cpp MarkLive.cpp + MinGW.cpp PDB.cpp Strings.cpp SymbolTable.cpp Index: COFF/Driver.cpp =================================================================== --- COFF/Driver.cpp +++ COFF/Driver.cpp @@ -12,6 +12,7 @@ #include "Error.h" #include "InputFiles.h" #include "Memory.h" +#include "MinGW.h" #include "SymbolTable.h" #include "Symbols.h" #include "Writer.h" @@ -548,113 +549,6 @@ } } -// Logic for deciding what symbols to export, when exporting all -// symbols for MinGW. -class AutoExporter { -public: - AutoExporter() { - if (Config->Machine == I386) - ExcludeSymbols = { - "__NULL_IMPORT_DESCRIPTOR", - "__pei386_runtime_relocator", - "_do_pseudo_reloc", - "_impure_ptr", - "__impure_ptr", - "__fmode", - "_environ", - "___dso_handle", - // These are the MinGW names that differ from the standard - // ones (lacking an extra underscore). - "_DllMain@12", - "_DllEntryPoint@12", - "_DllMainCRTStartup@12", - }; - else - ExcludeSymbols = { - "_NULL_IMPORT_DESCRIPTOR", - "_pei386_runtime_relocator", - "do_pseudo_reloc", - "impure_ptr", - "_impure_ptr", - "_fmode", - "environ", - "__dso_handle", - // These are the MinGW names that differ from the standard - // ones (lacking an extra underscore). - "DllMain", - "DllEntryPoint", - "DllMainCRTStartup", - }; - } - - StringSet<> ExcludeSymbols; - StringSet<> ExcludeLibs = { - "libgcc", - "libgcc_s", - "libstdc++", - "libmingw32", - "libmingwex", - "libg2c", - "libsupc++", - "libobjc", - "libgcj", - "libclang_rt.builtins-aarch64", - "libclang_rt.builtins-arm", - "libclang_rt.builtins-i386", - "libclang_rt.builtins-x86_64", - }; - StringSet<> ExcludeObjects = { - "crt0.o", - "crt1.o", - "crt1u.o", - "crt2.o", - "crt2u.o", - "dllcrt1.o", - "dllcrt2.o", - "gcrt0.o", - "gcrt1.o", - "gcrt2.o", - "crtbegin.o", - "crtend.o", - }; - - bool shouldExport(Defined *Sym) const { - if (!Sym || !Sym->isLive() || !Sym->getChunk()) - return false; - if (ExcludeSymbols.count(Sym->getName())) - return false; - StringRef LibName = sys::path::filename(Sym->getFile()->ParentName); - // Drop the file extension. - LibName = LibName.substr(0, LibName.rfind('.')); - if (ExcludeLibs.count(LibName)) - return false; - StringRef FileName = sys::path::filename(Sym->getFile()->getName()); - if (LibName.empty() && ExcludeObjects.count(FileName)) - return false; - return true; - } -}; - -// This is MinGW specific. -static void writeDefFile(StringRef Name) { - std::error_code EC; - raw_fd_ostream OS(Name, EC, sys::fs::F_None); - if (EC) - fatal("cannot open " + Name + ": " + EC.message()); - - OS << "EXPORTS\n"; - for (Export &E : Config->Exports) { - OS << " " << E.ExportName << " " - << "@" << E.Ordinal; - if (auto *Def = dyn_cast_or_null(E.Sym)) { - if (Def && Def->getChunk() && - !(Def->getChunk()->getPermissions() & IMAGE_SCN_MEM_EXECUTE)) - OS << " DATA"; - } - OS << "\n"; - } -} - // A helper function for filterBitcodeFiles. static bool needsRebuilding(MemoryBufferRef MB) { // The MSVC linker doesn't support thin archives, so if it's a thin Index: COFF/MinGW.h =================================================================== --- /dev/null +++ COFF/MinGW.h @@ -0,0 +1,41 @@ +//===- MinGW.h --------------------------------------------------*- C++ -*-===// +// +// The LLVM Linker +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#ifndef LLD_COFF_MINGW_H +#define LLD_COFF_MINGW_H + +#include "Config.h" +#include "Symbols.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/ADT/StringSet.h" + +namespace lld { +namespace coff { + +using llvm::StringSet; + +// Logic for deciding what symbols to export, when exporting all +// symbols for MinGW. +class AutoExporter { +public: + AutoExporter(); + + StringSet<> ExcludeSymbols; + StringSet<> ExcludeLibs; + StringSet<> ExcludeObjects; + + bool shouldExport(Defined *Sym) const; +}; + +void writeDefFile(StringRef Name); + +} // namespace coff +} // namespace lld + +#endif Index: COFF/MinGW.cpp =================================================================== --- /dev/null +++ COFF/MinGW.cpp @@ -0,0 +1,123 @@ +//===- MinGW.cpp ----------------------------------------------------------===// +// +// The LLVM Linker +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "MinGW.h" +#include "Error.h" +#include "llvm/Object/COFF.h" +#include "llvm/Support/Path.h" +#include "llvm/Support/raw_ostream.h" + +using namespace llvm; +using namespace llvm::COFF; + +namespace lld { +namespace coff { + +AutoExporter::AutoExporter() { + if (Config->Machine == I386) + ExcludeSymbols = { + "__NULL_IMPORT_DESCRIPTOR", + "__pei386_runtime_relocator", + "_do_pseudo_reloc", + "_impure_ptr", + "__impure_ptr", + "__fmode", + "_environ", + "___dso_handle", + // These are the MinGW names that differ from the standard + // ones (lacking an extra underscore). + "_DllMain@12", + "_DllEntryPoint@12", + "_DllMainCRTStartup@12", + }; + else + ExcludeSymbols = { + "_NULL_IMPORT_DESCRIPTOR", + "_pei386_runtime_relocator", + "do_pseudo_reloc", + "impure_ptr", + "_impure_ptr", + "_fmode", + "environ", + "__dso_handle", + // These are the MinGW names that differ from the standard + // ones (lacking an extra underscore). + "DllMain", + "DllEntryPoint", + "DllMainCRTStartup", + }; + + ExcludeLibs = { + "libgcc", + "libgcc_s", + "libstdc++", + "libmingw32", + "libmingwex", + "libg2c", + "libsupc++", + "libobjc", + "libgcj", + "libclang_rt.builtins-aarch64", + "libclang_rt.builtins-arm", + "libclang_rt.builtins-i386", + "libclang_rt.builtins-x86_64", + }; + ExcludeObjects = { + "crt0.o", + "crt1.o", + "crt1u.o", + "crt2.o", + "crt2u.o", + "dllcrt1.o", + "dllcrt2.o", + "gcrt0.o", + "gcrt1.o", + "gcrt2.o", + "crtbegin.o", + "crtend.o", + }; +} + +bool AutoExporter::shouldExport(Defined *Sym) const { + if (!Sym || !Sym->isLive() || !Sym->getChunk()) + return false; + if (ExcludeSymbols.count(Sym->getName())) + return false; + StringRef LibName = sys::path::filename(Sym->getFile()->ParentName); + // Drop the file extension. + LibName = LibName.substr(0, LibName.rfind('.')); + if (ExcludeLibs.count(LibName)) + return false; + StringRef FileName = sys::path::filename(Sym->getFile()->getName()); + if (LibName.empty() && ExcludeObjects.count(FileName)) + return false; + return true; +} + +void writeDefFile(StringRef Name) { + std::error_code EC; + raw_fd_ostream OS(Name, EC, sys::fs::F_None); + if (EC) + fatal("cannot open " + Name + ": " + EC.message()); + + OS << "EXPORTS\n"; + for (Export &E : Config->Exports) { + OS << " " << E.ExportName << " " + << "@" << E.Ordinal; + if (auto *Def = dyn_cast_or_null(E.Sym)) { + if (Def && Def->getChunk() && + !(Def->getChunk()->getPermissions() & IMAGE_SCN_MEM_EXECUTE)) + OS << " DATA"; + } + OS << "\n"; + } +} + +} // namespace coff +} // namespace lld