From ee0a3b5c776cac769ae68dce369728eaae286671 Mon Sep 17 00:00:00 2001 From: =?utf8?q?Martin=20Storsj=C3=B6?= Date: Thu, 12 Dec 2019 11:08:18 +0200 Subject: [PATCH] [MinGW] Implicitly add .exe suffix if not provided GCC implicitly adds an .exe suffix if it is given an output file name, but the file name doesn't contain a suffix, and there are certain users of GCC that rely on this behaviour (and run into issues when trying to use Clang instead of GCC). And MSVC's cl.exe also does the same (but not link.exe). However, GCC only does this when actually running on windows, not when operating as a cross compiler. As GCC doesn't have this behaviour when cross compiling, we definitely shouldn't introduce the behaviour in such cases (as it would break at least as many cases as this fixes). Differential Revision: https://reviews.llvm.org/D71400 --- clang/lib/Driver/ToolChains/MinGW.cpp | 14 +++++++++++++- clang/test/Driver/mingw-implicit-extension-cross.c | 9 +++++++++ clang/test/Driver/mingw-implicit-extension-windows.c | 14 ++++++++++++++ 3 files changed, 36 insertions(+), 1 deletion(-) create mode 100644 clang/test/Driver/mingw-implicit-extension-cross.c create mode 100644 clang/test/Driver/mingw-implicit-extension-windows.c diff --git a/clang/lib/Driver/ToolChains/MinGW.cpp b/clang/lib/Driver/ToolChains/MinGW.cpp index 0d85111..8f24384 100644 --- a/clang/lib/Driver/ToolChains/MinGW.cpp +++ b/clang/lib/Driver/ToolChains/MinGW.cpp @@ -160,7 +160,19 @@ void tools::MinGW::Linker::ConstructJob(Compilation &C, const JobAction &JA, } CmdArgs.push_back("-o"); - CmdArgs.push_back(Output.getFilename()); + const char *OutputFile = Output.getFilename(); + // GCC implicitly adds an .exe extension if it is given an output file name + // that lacks an extension. However, GCC only does this when actually + // running on windows, not when operating as a cross compiler. As some users + // have come to rely on this behaviour, try to replicate it. +#ifdef _WIN32 + if (!llvm::sys::path::has_extension(OutputFile)) + CmdArgs.push_back(Args.MakeArgString(Twine(OutputFile) + ".exe")); + else + CmdArgs.push_back(OutputFile); +#else + CmdArgs.push_back(OutputFile); +#endif Args.AddAllArgs(CmdArgs, options::OPT_e); // FIXME: add -N, -n flags diff --git a/clang/test/Driver/mingw-implicit-extension-cross.c b/clang/test/Driver/mingw-implicit-extension-cross.c new file mode 100644 index 0000000..2cf24dc --- /dev/null +++ b/clang/test/Driver/mingw-implicit-extension-cross.c @@ -0,0 +1,9 @@ +// Test how an implicit .exe extension is added. If not running the compiler +// on windows, no implicit extension is added. (Therefore, this test is skipped +// when running on windows.) + +// UNSUPPORTED: system-windows + +// RUN: %clang -target i686-windows-gnu -### --sysroot=%S/Inputs/mingw_clang_tree/mingw32 %s -o outputname 2>&1 | FileCheck %s + +// CHECK: "-o" "outputname" diff --git a/clang/test/Driver/mingw-implicit-extension-windows.c b/clang/test/Driver/mingw-implicit-extension-windows.c new file mode 100644 index 0000000..9c60308 --- /dev/null +++ b/clang/test/Driver/mingw-implicit-extension-windows.c @@ -0,0 +1,14 @@ +// Test how an implicit .exe extension is added. If running the compiler +// on windows, an implicit extension is added if none is provided in the +// given name. (Therefore, this test is skipped when not running on windows.) + +// REQUIRES: system-windows + +// RUN: %clang -target i686-windows-gnu -### --sysroot=%S/Inputs/mingw_clang_tree/mingw32 %s -o outputname 2>&1 | FileCheck %s --check-prefix=CHECK-OUTPUTNAME-EXE + +// RUN: %clang -target i686-windows-gnu -### --sysroot=%S/Inputs/mingw_clang_tree/mingw32 %s -o outputname.exe 2>&1 | FileCheck %s --check-prefix=CHECK-OUTPUTNAME-EXE + +// RUN: %clang -target i686-windows-gnu -### --sysroot=%S/Inputs/mingw_clang_tree/mingw32 %s -o outputname.q 2>&1 | FileCheck %s --check-prefix=CHECK-OUTPUTNAME-Q + +// CHECK-OUTPUTNAME-EXE: "-o" "outputname.exe" +// CHECK-OUTPUTNAME-Q: "-o" "outputname.q" -- 2.7.4