clang-cl: Support the run-time selection options (/MD, /MT et al.)
authorHans Wennborg <hans@hanshq.net>
Thu, 8 Aug 2013 00:17:41 +0000 (00:17 +0000)
committerHans Wennborg <hans@hanshq.net>
Thu, 8 Aug 2013 00:17:41 +0000 (00:17 +0000)
These flags set some preprocessor macros and injects a dependency
on the runtime library into the object file, which later is picked up
by the linker.

This also adds a new CC1 flag for adding a dependent library.

Differential Revision: http://llvm-reviews.chandlerc.com/D1315

llvm-svn: 187945

clang/include/clang/Driver/CC1Options.td
clang/include/clang/Driver/CLCompatOptions.td
clang/include/clang/Frontend/CodeGenOptions.h
clang/lib/CodeGen/ModuleBuilder.cpp
clang/lib/Driver/Tools.cpp
clang/lib/Driver/Tools.h
clang/lib/Frontend/CompilerInvocation.cpp
clang/test/CodeGen/dependent-lib.c [new file with mode: 0644]
clang/test/Driver/cl-options.c
clang/test/Driver/cl-runtime-flags.c [new file with mode: 0644]

index fdc073b5a2b97a152d4e85ed0889547e657c6c52..6ac40f5d208313049ee01367a4b8e4bb55ad1a1e 100644 (file)
@@ -210,6 +210,8 @@ def vectorize_slp : Flag<["-"], "vectorize-slp">,
   HelpText<"Run the SLP vectorization passes">;
 def vectorize_slp_aggressive : Flag<["-"], "vectorize-slp-aggressive">,
   HelpText<"Run the BB vectorization passes">;
+def dependent_lib : Joined<["--"], "dependent-lib=">,
+  HelpText<"Add dependent library">;
 
 //===----------------------------------------------------------------------===//
 // Dependency Output Options
index b29fc8802db0ebc0ca3e0693219546996cc7376d..e98d8e851649752c870ae9ccd55fbf0574d2f4b4 100644 (file)
@@ -86,6 +86,12 @@ def _SLASH_Zs : CLFlag<"Zs">, HelpText<"Syntax-check only">,
 def _SLASH_Fo : CLJoined<"Fo">,
   HelpText<"Set output object file, or directory (ends in / or \\)">,
   MetaVarName<"<file or directory>">;
+def _SLASH_MD : CLFlag<"MD">,
+  HelpText<"Use DLL run-time">;
+def _SLASH_MDd : CLFlag<"MDd">,
+  HelpText<"Use DLL debug run-time">;
+def _SLASH_MT : CLFlag<"MT">, HelpText<"Use static run-time">;
+def _SLASH_MTd : CLFlag<"MTd">, HelpText<"Use static debug run-time">;
 def _SLASH_Tc : CLJoinedOrSeparate<"Tc">, HelpText<"Specify a C source file">,
   MetaVarName<"<filename>">;
 def _SLASH_TC : CLFlag<"TC">, HelpText<"Treat all source files as C">;
@@ -119,10 +125,6 @@ def _SLASH_GS : CLFlag<"GS">;
 def _SLASH_Gy : CLFlag<"Gy">;
 def _SLASH_Gy_ : CLFlag<"Gy-">;
 def _SLASH_GZ : CLFlag<"GZ">;
-def _SLASH_MD : CLFlag<"MD">;
-def _SLASH_MT : CLFlag<"MT">;
-def _SLASH_MDd : CLFlag<"MDd">;
-def _SLASH_MTd : CLFlag<"MTd">;
 def _SLASH_Oi : CLFlag<"Oi">;
 def _SLASH_RTC : CLJoined<"RTC">;
 def _SLASH_showIncludes : CLJoined<"showIncludes">;
index 6717791466a3fbd511d6ef8258fefb382fc7fce2..45d2bda985bc0fdaa9492ec72755cc5eec660bb6 100644 (file)
@@ -128,6 +128,9 @@ public:
   /// A list of command-line options to forward to the LLVM backend.
   std::vector<std::string> BackendOptions;
 
+  /// A list of dependent libraries.
+  std::vector<std::string> DependentLibraries;
+
 public:
   // Define accessors/mutators for code generation options of enumeration type.
 #define CODEGENOPT(Name, Bits, Default)
index 7e0e3aac072526abfecfe8f9986909d3665e7d0e..c6d40330e0cdfd3dc1ec301112d7f8221ca78ea4 100644 (file)
@@ -60,6 +60,9 @@ namespace {
       TD.reset(new llvm::DataLayout(Ctx->getTargetInfo().getTargetDescription()));
       Builder.reset(new CodeGen::CodeGenModule(Context, CodeGenOpts, *M, *TD,
                                                Diags));
+
+      for (size_t i = 0, e = CodeGenOpts.DependentLibraries.size(); i < e; ++i)
+        HandleDependentLibrary(CodeGenOpts.DependentLibraries[i]);
     }
 
     virtual void HandleCXXStaticMemberVarInstantiation(VarDecl *VD) {
index 4a3ce42f0a11a5d896d300184f9108003b0bbe18..e4acadbb510145dd25a2afd7c6c0d6724687ff15 100644 (file)
@@ -2516,6 +2516,10 @@ void Clang::ConstructJob(Compilation &C, const JobAction &JA,
     break;
   }
 
+  // Add clang-cl arguments.
+  if (getToolChain().getDriver().IsCLMode())
+    AddClangCLArgs(Args, CmdArgs);
+
   // Pass the linker version in use.
   if (Arg *A = Args.getLastArg(options::OPT_mlinker_version_EQ)) {
     CmdArgs.push_back("-target-linker-version");
@@ -3803,6 +3807,47 @@ ObjCRuntime Clang::AddObjCRuntimeArgs(const ArgList &args,
   return runtime;
 }
 
+void Clang::AddClangCLArgs(const ArgList &Args, ArgStringList &CmdArgs) const {
+  unsigned RTOptionID = options::OPT__SLASH_MT;
+
+  if (Arg *A = Args.getLastArg(options::OPT__SLASH_MD,
+                               options::OPT__SLASH_MDd,
+                               options::OPT__SLASH_MT,
+                               options::OPT__SLASH_MTd)) {
+    RTOptionID = A->getOption().getID();
+  }
+
+  switch(RTOptionID) {
+    case options::OPT__SLASH_MD:
+      CmdArgs.push_back("-D_MT");
+      CmdArgs.push_back("-D_DLL");
+      CmdArgs.push_back("--dependent-lib=msvcrt");
+      break;
+    case options::OPT__SLASH_MDd:
+      CmdArgs.push_back("-D_DEBUG");
+      CmdArgs.push_back("-D_MT");
+      CmdArgs.push_back("-D_DLL");
+      CmdArgs.push_back("--dependent-lib=msvcrtd");
+      break;
+    case options::OPT__SLASH_MT:
+      CmdArgs.push_back("-D_MT");
+      CmdArgs.push_back("--dependent-lib=libcmt");
+      break;
+    case options::OPT__SLASH_MTd:
+      CmdArgs.push_back("-D_DEBUG");
+      CmdArgs.push_back("-D_MT");
+      CmdArgs.push_back("--dependent-lib=libcmtd");
+      break;
+    default:
+      llvm_unreachable("Unexpected option ID.");
+  }
+
+  // This provides POSIX compatibility (maps 'open' to '_open'), which most users
+  // want.  MSVC has a switch to turn off this autolinking, but it's not
+  // implemented in clang yet.
+  CmdArgs.push_back("--dependent-lib=oldnames");
+}
+
 void ClangAs::ConstructJob(Compilation &C, const JobAction &JA,
                            const InputInfo &Output,
                            const InputInfoList &Inputs,
index 9138ddf88ba5cdadb4afa2143f43a54202187908..8c1e695aa201145bd6a373074b3f99a17eed3866 100644 (file)
@@ -74,6 +74,9 @@ using llvm::opt::ArgStringList;
                                    llvm::opt::ArgStringList &cmdArgs,
                                    RewriteKind rewrite) const;
 
+    void AddClangCLArgs(const llvm::opt::ArgList &Args,
+                        llvm::opt::ArgStringList &CmdArgs) const;
+
   public:
     Clang(const ToolChain &TC) : Tool("clang", "clang frontend", TC) {}
 
index 27ddd84d52fa4cfcba7d70f46b2d6aaa6c7d201b..a4c93fa5b502de5ad438de9001413310cf3df426 100644 (file)
@@ -502,6 +502,8 @@ static bool ParseCodeGenArgs(CodeGenOptions &Opts, ArgList &Args, InputKind IK,
     }
   }
 
+  Opts.DependentLibraries = Args.getAllArgValues(OPT_dependent_lib);
+
   return Success;
 }
 
diff --git a/clang/test/CodeGen/dependent-lib.c b/clang/test/CodeGen/dependent-lib.c
new file mode 100644 (file)
index 0000000..df4aaf0
--- /dev/null
@@ -0,0 +1,15 @@
+// RUN: %clang_cc1 %s --dependent-lib=msvcrt -triple i686-pc-win32 -emit-llvm -o - | FileCheck %s
+// RUN: %clang_cc1 %s --dependent-lib=msvcrt -triple x86_64-pc-win32 -emit-llvm -o - | FileCheck %s
+// RUN: %clang_cc1 %s --dependent-lib=msvcrt -triple i686-pc-linux -emit-llvm -o - | FileCheck -check-prefix LINUX %s
+
+// CHECK: !llvm.module.flags = !{!0}
+// CHECK: !0 = metadata !{i32 6, metadata !"Linker Options", metadata ![[link_opts:[0-9]+]]}
+// CHECK: ![[link_opts]] = metadata !{metadata ![[msvcrt:[0-9]+]]}
+// CHECK: ![[msvcrt]] = metadata !{metadata !"/DEFAULTLIB:msvcrt.lib"}
+
+// LINUX: !llvm.module.flags = !{!0}
+// LINUX: !0 = metadata !{i32 6, metadata !"Linker Options", metadata ![[link_opts:[0-9]+]]}
+// LINUX: ![[link_opts]] = metadata !{metadata ![[msvcrt:[0-9]+]]}
+// LINUX: ![[msvcrt]] = metadata !{metadata !"-lmsvcrt"}
+
+int f();
index ba1c1f30144abbb35e2a554d8894fbc6d9a2de1b..d37e88e456a4f395e79f40c360cfd22230d020dc 100644 (file)
@@ -89,6 +89,6 @@
 // Unsupported but parsed options. Check that we don't error on them.
 // (/Zs is for syntax-only)
 // RUN: %clang_cl /Zs /EHsc /Fdfoo /fp:precise /Gd /GL /GL- -- %s 2>&1
-// RUN: %clang_cl /Zs /Gm /Gm- /GS /Gy /Gy- /GZ /MD /MT /MDd /MTd /Oi -- %s 2>&1
+// RUN: %clang_cl /Zs /Gm /Gm- /GS /Gy /Gy- /GZ /Oi -- %s 2>&1
 // RUN: %clang_cl /Zs /RTC1 /wfoo /Zc:wchar_t- -- %s 2>&1
 // RUN: %clang_cl /Zs /ZI /Zi /showIncludes -- %s 2>&1
diff --git a/clang/test/Driver/cl-runtime-flags.c b/clang/test/Driver/cl-runtime-flags.c
new file mode 100644 (file)
index 0000000..520093c
--- /dev/null
@@ -0,0 +1,41 @@
+// Don't attempt slash switches on msys bash.\r
+// REQUIRES: shell-preserves-root\r
+\r
+// Note: %s must be preceded by --, otherwise it may be interpreted as a\r
+// command-line option, e.g. on Mac where %s is commonly under /Users.\r
+\r
+// First check that regular clang doesn't do any of this stuff.\r
+// RUN: %clang -### %s 2>&1 | FileCheck -check-prefix=CHECK-CLANG %s\r
+// CHECK-CLANG-NOT: "-D_DEBUG"\r
+// CHECK-CLANG-NOT: "-D_MT"\r
+// CHECK-CLANG-NOT: "-D_DLL"\r
+// CHECK-CLANG-NOT: --dependent-lib\r
+\r
+// RUN: %clang_cl -### -- %s 2>&1 | FileCheck -check-prefix=CHECK-MT %s\r
+// RUN: %clang_cl -### /MT -- %s 2>&1 | FileCheck -check-prefix=CHECK-MT %s\r
+// CHECK-MT-NOT: "-D_DEBUG"\r
+// CHECK-MT: "-D_MT"\r
+// CHECK-MT-NOT: "-D_DLL"\r
+// CHECK-MT: "--dependent-lib=libcmt"\r
+// CHECK-MT: "--dependent-lib=oldnames"\r
+\r
+// RUN: %clang_cl -### /MTd -- %s 2>&1 | FileCheck -check-prefix=CHECK-MTd %s\r
+// CHECK-MTd: "-D_DEBUG"\r
+// CHECK-MTd: "-D_MT"\r
+// CHECK-MTd-NOT: "-D_DLL"\r
+// CHECK-MTd: "--dependent-lib=libcmtd"\r
+// CHECK-MTd: "--dependent-lib=oldnames"\r
+\r
+// RUN: %clang_cl -### /MD -- %s 2>&1 | FileCheck -check-prefix=CHECK-MD %s\r
+// CHECK-MD-NOT: "-D_DEBUG"\r
+// CHECK-MD: "-D_MT"\r
+// CHECK-MD: "-D_DLL"\r
+// CHECK-MD: "--dependent-lib=msvcrt"\r
+// CHECK-MD: "--dependent-lib=oldnames"\r
+\r
+// RUN: %clang_cl -### /MDd -- %s 2>&1 | FileCheck -check-prefix=CHECK-MDd %s\r
+// CHECK-MDd: "-D_DEBUG"\r
+// CHECK-MDd: "-D_MT"\r
+// CHECK-MDd: "-D_DLL"\r
+// CHECK-MDd: "--dependent-lib=msvcrtd"\r
+// CHECK-MDd: "--dependent-lib=oldnames"\r