diff options
author | Martin Storsjo <martin@martin.st> | 2017-10-23 09:08:24 +0000 |
---|---|---|
committer | Martin Storsjo <martin@martin.st> | 2017-10-23 09:08:24 +0000 |
commit | ddb094ad367d0ab180ccebd27d86c940d7c250f7 (patch) | |
tree | d0c12e654f7b1d9f3bebef2325845258fecee321 | |
parent | 843cbbddebf8451a8661334cacc4dd608a1f0683 (diff) | |
download | bcm5719-llvm-ddb094ad367d0ab180ccebd27d86c940d7c250f7.tar.gz bcm5719-llvm-ddb094ad367d0ab180ccebd27d86c940d7c250f7.zip |
[COFF] Fix exporting of functions starting with underscores, etc
This fixes exporting functions in the following cases:
- functions starting with an underscore in def files
- functions starting with an underscore, via dllexport attributes, for mingw
- fastcall and vectorcall functions when declared undecorated in def files
- vectorcall functions when declared decorated in def files
- stdcall functions when declared decorated in def files for mingw
This still exports the stdcall functions with the wrong name
in the normal msvc/link.exe mode, if declared with decoration in
the def file though (this is not a regression though). Exporting
functions via def files including decoration is not something I
believe is routinely done though, but is tested to try to match
link.exe's behaviour as far as easily possible.
Differential Revision: https://reviews.llvm.org/D39170
llvm-svn: 316317
-rw-r--r-- | lld/COFF/Driver.cpp | 7 | ||||
-rw-r--r-- | lld/COFF/SymbolTable.cpp | 14 | ||||
-rw-r--r-- | lld/test/COFF/def-export-stdcall.s | 78 | ||||
-rw-r--r-- | lld/test/COFF/dllexport-mingw.s | 7 |
4 files changed, 95 insertions, 11 deletions
diff --git a/lld/COFF/Driver.cpp b/lld/COFF/Driver.cpp index c039740715c..988fb4b1498 100644 --- a/lld/COFF/Driver.cpp +++ b/lld/COFF/Driver.cpp @@ -212,7 +212,8 @@ void LinkerDriver::enqueueArchiveMember(const Archive::Child &C, } static bool isDecorated(StringRef Sym) { - return Sym.startswith("_") || Sym.startswith("@") || Sym.startswith("?"); + return Sym.startswith("@") || Sym.contains("@@") || Sym.startswith("?") || + (!Config->MinGW && Sym.contains('@')); } // Parses .drectve section contents and returns a list of files @@ -510,8 +511,8 @@ static void createImportLibrary(bool AsLib) { static void parseModuleDefs(StringRef Path) { std::unique_ptr<MemoryBuffer> MB = check( MemoryBuffer::getFile(Path, -1, false, true), "could not open " + Path); - COFFModuleDefinition M = - check(parseCOFFModuleDefinition(MB->getMemBufferRef(), Config->Machine)); + COFFModuleDefinition M = check(parseCOFFModuleDefinition( + MB->getMemBufferRef(), Config->Machine, Config->MinGW)); if (Config->OutputFile.empty()) Config->OutputFile = Saver.save(M.OutputFile); diff --git a/lld/COFF/SymbolTable.cpp b/lld/COFF/SymbolTable.cpp index 40fbea91cdc..3b3004c6897 100644 --- a/lld/COFF/SymbolTable.cpp +++ b/lld/COFF/SymbolTable.cpp @@ -351,10 +351,18 @@ StringRef SymbolTable::findMangle(StringRef Name) { return findByPrefix(("?" + Name + "@@Y").str()); if (!Name.startswith("_")) return ""; - // Search for x86 C function. + // 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()); } @@ -364,8 +372,10 @@ void SymbolTable::mangleMaybe(SymbolBody *B) { if (!U || U->WeakAlias) return; StringRef Alias = findMangle(U->getName()); - if (!Alias.empty()) + if (!Alias.empty()) { + log(U->getName() + " aliased to " + Alias); U->WeakAlias = addUndefined(Alias); + } } SymbolBody *SymbolTable::addUndefined(StringRef Name) { diff --git a/lld/test/COFF/def-export-stdcall.s b/lld/test/COFF/def-export-stdcall.s index 17473f7f976..851ac6d975b 100644 --- a/lld/test/COFF/def-export-stdcall.s +++ b/lld/test/COFF/def-export-stdcall.s @@ -1,21 +1,89 @@ # REQUIRES: x86 # RUN: llvm-mc -filetype=obj -triple=i686-windows-msvc %s -o %t.obj -# RUN: echo -e "LIBRARY foo\nEXPORTS\n stdcall" > %t.def +# RUN: echo -e "LIBRARY foo\nEXPORTS\n stdcall\n fastcall\n vectorcall\n _underscored" > %t.def # RUN: lld-link -entry:dllmain -dll -def:%t.def %t.obj -out:%t.dll -implib:%t.lib -# RUN: llvm-readobj %t.lib | FileCheck %s -# CHECK: Name type: undecorate -# CHECK: __imp__stdcall@8 -# CHECK: _stdcall@8 +# RUN: llvm-readobj %t.lib | FileCheck -check-prefix UNDECORATED-IMPLIB %s +# RUN: llvm-readobj -coff-exports %t.dll | FileCheck -check-prefix UNDECORATED-EXPORTS %s + +# UNDECORATED-IMPLIB: Name type: noprefix +# UNDECORATED-IMPLIB-NEXT: __imp___underscored +# UNDECORATED-IMPLIB-NEXT: __underscored +# UNDECORATED-IMPLIB: Name type: undecorate +# UNDECORATED-IMPLIB-NEXT: __imp_@fastcall@8 +# UNDECORATED-IMPLIB-NEXT: fastcall@8 +# UNDECORATED-IMPLIB: Name type: undecorate +# UNDECORATED-IMPLIB-NEXT: __imp__stdcall@8 +# UNDECORATED-IMPLIB-NEXT: _stdcall@8 +# UNDECORATED-IMPLIB: Name type: undecorate +# UNDECORATED-IMPLIB-NEXT: __imp_vectorcall@@8 +# UNDECORATED-IMPLIB-NEXT: vectorcall@@8 + +# UNDECORATED-EXPORTS: Name: _underscored +# UNDECORATED-EXPORTS: Name: fastcall +# UNDECORATED-EXPORTS: Name: stdcall +# UNDECORATED-EXPORTS: Name: vectorcall + + +# RUN: echo -e "LIBRARY foo\nEXPORTS\n _stdcall@8\n @fastcall@8\n vectorcall@@8" > %t.def +# RUN: lld-link -entry:dllmain -dll -def:%t.def %t.obj -out:%t.dll -implib:%t.lib +# RUN: llvm-readobj %t.lib | FileCheck -check-prefix DECORATED-IMPLIB %s +# RUN: llvm-readobj -coff-exports %t.dll | FileCheck -check-prefix DECORATED-EXPORTS %s + +# DECORATED-IMPLIB: Name type: name +# DECORATED-IMPLIB-NEXT: __imp_@fastcall@8 +# DECORATED-IMPLIB-NEXT: @fastcall@8 +# TODO: To match link.exe, this one should also be Name type: name. +# DECORATED-IMPLIB: Name type: noprefix +# DECORATED-IMPLIB-NEXT: __imp__stdcall@8 +# DECORATED-IMPLIB-NEXT: _stdcall@8 +# DECORATED-IMPLIB: Name type: name +# DECORATED-IMPLIB-NEXT: __imp_vectorcall@@8 +# DECORATED-IMPLIB-NEXT: vectorcall@@8 + +# DECORATED-EXPORTS: Name: @fastcall@8 +# TODO: To match link.exe, this one should actually be _stdcall@8 +# DECORATED-EXPORTS: Name: stdcall@8 +# DECORATED-EXPORTS: Name: vectorcall@@8 + + +# RUN: echo -e "LIBRARY foo\nEXPORTS\n stdcall@8\n @fastcall@8" > %t.def +# RUN: lld-link -lldmingw -entry:dllmain -dll -def:%t.def %t.obj -out:%t.dll -implib:%t.lib +# RUN: llvm-readobj %t.lib | FileCheck -check-prefix DECORATED-MINGW-IMPLIB %s +# RUN: llvm-readobj -coff-exports %t.dll | FileCheck -check-prefix DECORATED-MINGW-EXPORTS %s + +# DECORATED-MINGW-IMPLIB: Name type: name +# DECORATED-MINGW-IMPLIB-NEXT: __imp_@fastcall@8 +# DECORATED-MINGW-IMPLIB-NEXT: fastcall@8 +# DECORATED-MINGW-IMPLIB: Name type: noprefix +# DECORATED-MINGW-IMPLIB-NEXT: __imp__stdcall@8 +# DECORATED-MINGW-IMPLIB-NEXT: _stdcall@8 + +# DECORATED-MINGW-EXPORTS: Name: @fastcall@8 +# DECORATED-MINGW-EXPORTS: Name: stdcall@8 + .def _stdcall@8; .scl 2; .type 32; .endef .globl _stdcall@8 + .globl @fastcall@8 + .globl vectorcall@@8 + .globl __underscored _stdcall@8: movl 8(%esp), %eax addl 4(%esp), %eax retl $8 +@fastcall@8: + movl 8(%esp), %eax + addl 4(%esp), %eax + retl $8 +vectorcall@@8: + movl 8(%esp), %eax + addl 4(%esp), %eax + retl $8 +__underscored: + ret .def _dllmain; .scl 2; diff --git a/lld/test/COFF/dllexport-mingw.s b/lld/test/COFF/dllexport-mingw.s index 06c2e225817..8bf035b36dc 100644 --- a/lld/test/COFF/dllexport-mingw.s +++ b/lld/test/COFF/dllexport-mingw.s @@ -5,15 +5,20 @@ # RUN: lld-link -lldmingw -dll -out:%t.dll -entry:main %t.obj -implib:%t.lib # RUN: llvm-readobj %t.lib | FileCheck %s +# CHECK: Symbol: __imp___underscoredFunc +# CHECK: Symbol: __underscoredFunc # CHECK: Symbol: __imp__func # CHECK: Symbol: _func .global _main .global _func +.global __underscoredFunc .text _main: ret _func: ret +__underscoredFunc: + ret .section .drectve -.ascii "-export:func" +.ascii "-export:func -export:_underscoredFunc" |