67 const char *LinkingOutput)
const {
68 ArgStringList CmdArgs;
75 Args.MakeArgString(std::string(
"-out:") + Output.
getFilename()));
77 if (Args.hasArg(options::OPT_marm64x))
78 CmdArgs.push_back(
"-machine:arm64x");
79 else if (TC.getTriple().isWindowsArm64EC())
80 CmdArgs.push_back(
"-machine:arm64ec");
82 if (
const Arg *A = Args.getLastArg(options::OPT_fveclib)) {
83 StringRef
V = A->getValue();
85 CmdArgs.push_back(Args.MakeArgString(
"--dependent-lib=amath"));
91 if (Args.hasFlag(options::OPT_fsycl, options::OPT_fno_sycl,
false) &&
92 !Args.hasArg(options::OPT_nolibsycl) &&
93 !Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles)) {
97 bool HasStaticCRT =
false;
99 if (
const Arg *A = Args.getLastArg(options::OPT_fms_runtime_lib_EQ)) {
100 StringRef RuntimeLib = A->getValue();
101 if (RuntimeLib ==
"static" || RuntimeLib ==
"static_dbg")
105 if (
const Arg *A = Args.getLastArg(options::OPT__SLASH_M_Group)) {
106 if (A->getOption().matches(options::OPT__SLASH_MT) ||
107 A->getOption().matches(options::OPT__SLASH_MTd))
112 TC.getDriver().Diag(diag::err_drv_sycl_requires_dynamic_crt);
116 llvm::sys::path::append(LibPath,
"..",
"lib");
117 CmdArgs.push_back(Args.MakeArgString(Twine(
"-libpath:") + LibPath));
121 if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles) &&
122 !
C.getDriver().IsCLMode() && !
C.getDriver().IsFlangMode()) {
123 CmdArgs.push_back(
"-defaultlib:libcmt");
124 CmdArgs.push_back(
"-defaultlib:oldnames");
128 if (Args.hasFlag(options::OPT_fsycl, options::OPT_fno_sycl,
false) &&
129 !Args.hasArg(options::OPT_nolibsycl)) {
130 bool IsDebugBuild =
false;
131 if (
const Arg *A = Args.getLastArg(options::OPT_fms_runtime_lib_EQ)) {
132 StringRef RuntimeVal = A->getValue();
133 if (RuntimeVal ==
"dll_dbg")
136 CmdArgs.push_back(IsDebugBuild ?
"-defaultlib:LLVMSYCLd"
137 :
"-defaultlib:LLVMSYCL");
146 if (
const Arg *A = Args.getLastArg(options::OPT__SLASH_diasdkdir,
147 options::OPT__SLASH_winsysroot)) {
152 if (A->getOption().getID() == options::OPT__SLASH_winsysroot)
153 llvm::sys::path::append(DIAPath,
"DIA SDK");
156 llvm::sys::path::append(DIAPath,
"lib",
157 llvm::archToLegacyVCArch(TC.getArch()));
158 CmdArgs.push_back(Args.MakeArgString(Twine(
"-libpath:") + DIAPath));
160 if (!llvm::sys::Process::GetEnv(
"LIB") ||
161 Args.hasArg(options::OPT__SLASH_vctoolsdir,
162 options::OPT__SLASH_vctoolsversion,
163 options::OPT__SLASH_winsysroot)) {
164 CmdArgs.push_back(Args.MakeArgString(
166 TC.getSubDirectoryPath(llvm::SubDirectoryType::Lib)));
167 CmdArgs.push_back(Args.MakeArgString(
169 TC.getSubDirectoryPath(llvm::SubDirectoryType::Lib,
"atlmfc")));
171 if (!llvm::sys::Process::GetEnv(
"LIB") ||
172 Args.hasArg(options::OPT__SLASH_winsdkdir,
173 options::OPT__SLASH_winsdkversion,
174 options::OPT__SLASH_winsysroot)) {
175 if (TC.useUniversalCRT()) {
176 std::string UniversalCRTLibPath;
177 if (TC.getUniversalCRTLibraryPath(Args, UniversalCRTLibPath))
179 Args.MakeArgString(Twine(
"-libpath:") + UniversalCRTLibPath));
181 std::string WindowsSdkLibPath;
182 if (TC.getWindowsSDKLibraryPath(Args, WindowsSdkLibPath))
184 Args.MakeArgString(std::string(
"-libpath:") + WindowsSdkLibPath));
187 if (!
C.getDriver().IsCLMode() && Args.hasArg(options::OPT_L))
188 for (
const auto &LibPath : Args.getAllArgValues(options::OPT_L))
189 CmdArgs.push_back(Args.MakeArgString(
"-libpath:" + LibPath));
191 if (
C.getDriver().IsFlangMode() &&
192 !Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) {
193 TC.addFortranRuntimeLibraryPath(Args, CmdArgs);
194 TC.addFortranRuntimeLibs(Args, CmdArgs);
199 CmdArgs.push_back(
"/subsystem:console");
204 for (
const auto &LibPath : TC.getLibraryPaths()) {
205 if (TC.getVFS().exists(LibPath))
206 CmdArgs.push_back(Args.MakeArgString(
"-libpath:" + LibPath));
208 auto CRTPath = TC.getCompilerRTPath();
209 if (TC.getVFS().exists(CRTPath))
210 CmdArgs.push_back(Args.MakeArgString(
"-libpath:" + CRTPath));
215 if (Args.hasFlag(options::OPT_fsycl, options::OPT_fno_sycl,
false))
216 CmdArgs.push_back(
"/IGNORE:4078");
218 CmdArgs.push_back(
"-nologo");
220 if (Args.hasArg(options::OPT_g_Group, options::OPT__SLASH_Z7))
221 CmdArgs.push_back(
"-debug");
225 if (Args.hasArg(options::OPT_fms_hotpatch, options::OPT__SLASH_hotpatch))
226 CmdArgs.push_back(
"-functionpadmin");
230 bool DefaultIncrementalLinkerCompatible =
231 C.getDefaultToolChain().getTriple().isWindowsMSVCEnvironment();
232 if (!Args.hasFlag(options::OPT_mincremental_linker_compatible,
233 options::OPT_mno_incremental_linker_compatible,
234 DefaultIncrementalLinkerCompatible))
235 CmdArgs.push_back(
"-Brepro");
237 bool DLL = Args.hasArg(options::OPT__SLASH_LD, options::OPT__SLASH_LDd,
238 options::OPT_shared);
240 CmdArgs.push_back(Args.MakeArgString(
"-dll"));
243 llvm::sys::path::replace_extension(ImplibName,
"lib");
244 CmdArgs.push_back(Args.MakeArgString(std::string(
"-implib:") + ImplibName));
247 if (TC.getSanitizerArgs(Args).needsFuzzer()) {
248 if (!Args.hasArg(options::OPT_shared))
250 Args.MakeArgString(std::string(
"-wholearchive:") +
251 TC.getCompilerRTArgString(Args,
"fuzzer")));
252 CmdArgs.push_back(Args.MakeArgString(
"-debug"));
255 CmdArgs.push_back(Args.MakeArgString(
"-incremental:no"));
258 if (TC.getSanitizerArgs(Args).needsAsanRt()) {
259 CmdArgs.push_back(Args.MakeArgString(
"-debug"));
260 CmdArgs.push_back(Args.MakeArgString(
"-incremental:no"));
261 CmdArgs.push_back(TC.getCompilerRTArgString(Args,
"asan_dynamic"));
262 auto defines = Args.getAllArgValues(options::OPT_D);
263 if (Args.hasArg(options::OPT__SLASH_MD, options::OPT__SLASH_MDd) ||
264 llvm::is_contained(defines,
"_DLL")) {
267 CmdArgs.push_back(Args.MakeArgString(
268 TC.getArch() == llvm::Triple::x86
269 ?
"-include:___asan_seh_interceptor"
270 :
"-include:__asan_seh_interceptor"));
273 CmdArgs.push_back(Args.MakeArgString(
274 std::string(
"-wholearchive:") +
275 TC.getCompilerRT(Args,
"asan_dynamic_runtime_thunk")));
279 CmdArgs.push_back(Args.MakeArgString(
280 std::string(
"-wholearchive:") +
281 TC.getCompilerRT(Args,
"asan_static_runtime_thunk")));
285 if (
C.getDriver().isUsingLTO()) {
287 CmdArgs.push_back(Args.MakeArgString(std::string(
"-lto-sample-profile:") +
290 Args.AddAllArgValues(CmdArgs, options::OPT__SLASH_link);
293 for (
const Arg *A : Args.filtered(options::OPT__SLASH_guard)) {
294 StringRef GuardArgs = A->getValue();
295 if (GuardArgs.equals_insensitive(
"cf") ||
296 GuardArgs.equals_insensitive(
"cf,nochecks")) {
298 CmdArgs.push_back(
"-guard:cf");
299 }
else if (GuardArgs.equals_insensitive(
"cf-")) {
300 CmdArgs.push_back(
"-guard:cf-");
301 }
else if (GuardArgs.equals_insensitive(
"ehcont")) {
302 CmdArgs.push_back(
"-guard:ehcont");
303 }
else if (GuardArgs.equals_insensitive(
"ehcont-")) {
304 CmdArgs.push_back(
"-guard:ehcont-");
308 if (Args.hasFlag(options::OPT_fopenmp, options::OPT_fopenmp_EQ,
309 options::OPT_fno_openmp,
false)) {
310 CmdArgs.push_back(
"-nodefaultlib:vcomp.lib");
311 CmdArgs.push_back(
"-nodefaultlib:vcompd.lib");
312 CmdArgs.push_back(Args.MakeArgString(std::string(
"-libpath:") +
313 TC.getDriver().Dir +
"/../lib"));
314 switch (TC.getDriver().getOpenMPRuntime(Args)) {
316 CmdArgs.push_back(
"-defaultlib:libomp.lib");
319 CmdArgs.push_back(
"-defaultlib:libiomp5md.lib");
331 if (!Args.hasArg(options::OPT_nostdlib)) {
335 const Arg *A = Args.getLastArg(options::OPT_fuse_ld_EQ);
336 StringRef
Linker = A ? A->getValue() : TC.getDriver().getPreferredLinker();
343 if (Args.hasArg(options::OPT_gdwarf, options::OPT_gdwarf_2,
344 options::OPT_gdwarf_3, options::OPT_gdwarf_4,
345 options::OPT_gdwarf_5, options::OPT_gdwarf_6))
352 if (
Linker.equals_insensitive(
"lld"))
355 if (
Linker ==
"lld-link") {
356 for (Arg *A : Args.filtered(options::OPT_vfsoverlay))
358 Args.MakeArgString(std::string(
"/vfsoverlay:") + A->getValue()));
360 if (
C.getDriver().isUsingLTO() &&
361 Args.hasFlag(options::OPT_gsplit_dwarf, options::OPT_gno_split_dwarf,
363 CmdArgs.push_back(Args.MakeArgString(Twine(
"/dwodir:") +
368 for (
const auto &Input : Inputs) {
369 if (Input.isFilename()) {
370 CmdArgs.push_back(Input.getFilename());
374 const Arg &A = Input.getInputArg();
377 if (A.getOption().matches(options::OPT_l)) {
378 StringRef Lib = A.getValue();
379 const char *LinkLibArg;
380 if (Lib.ends_with(
".lib"))
381 LinkLibArg = Args.MakeArgString(Lib);
383 LinkLibArg = Args.MakeArgString(Lib +
".lib");
384 CmdArgs.push_back(LinkLibArg);
390 A.renderAsInput(Args, CmdArgs);
393 TC.addOffloadRTLibs(
C.getActiveOffloadKinds(), Args, CmdArgs);
395 TC.addProfileRTLibs(Args, CmdArgs);
397 std::vector<const char *> Environment;
402 if (
Linker.equals_insensitive(
"link")) {
408 if (!TC.FoundMSVCInstall() && !
canExecute(TC.getVFS(), linkPath)) {
410 ClPath = TC.GetProgramPath(
"cl.exe");
412 linkPath = llvm::sys::path::parent_path(ClPath);
413 llvm::sys::path::append(linkPath,
"link.exe");
415 C.getDriver().Diag(clang::diag::warn_drv_msvc_not_found);
417 C.getDriver().Diag(clang::diag::warn_drv_msvc_not_found);
423 if (TC.getSanitizerArgs(Args).needsAsanRt())
424 CmdArgs.push_back(
"/INFERASANLIBS:NO");
433 if (TC.getIsVS2017OrNewer() &&
434 llvm::Triple(llvm::sys::getProcessTriple()).getArch() != TC.getArch()) {
435 auto HostArch = llvm::Triple(llvm::sys::getProcessTriple()).getArch();
438 std::unique_ptr<wchar_t[], decltype(&FreeEnvironmentStringsW)>(
439 GetEnvironmentStringsW(), FreeEnvironmentStringsW);
441 goto SkipSettingEnvironment;
444 size_t EnvBlockLen = 0;
445 while (EnvBlockWide[EnvBlockLen] != L
'\0') {
447 EnvBlockLen += std::wcslen(&EnvBlockWide[EnvBlockLen]) +
452 std::string EnvBlock;
453 if (!llvm::convertUTF16ToUTF8String(
455 EnvBlockLen *
sizeof(EnvBlockWide[0])),
457 goto SkipSettingEnvironment;
459 Environment.reserve(EnvCount);
464 for (
const char *Cursor = EnvBlock.data(); *Cursor !=
'\0';) {
465 llvm::StringRef EnvVar(Cursor);
466 if (EnvVar.starts_with_insensitive(
"path=")) {
467 constexpr size_t PrefixLen = 5;
468 Environment.push_back(Args.MakeArgString(
469 EnvVar.substr(0, PrefixLen) +
470 TC.getSubDirectoryPath(llvm::SubDirectoryType::Bin) +
471 llvm::Twine(llvm::sys::EnvPathSeparator) +
472 TC.getSubDirectoryPath(llvm::SubDirectoryType::Bin, HostArch) +
473 (EnvVar.size() > PrefixLen
474 ? llvm::Twine(llvm::sys::EnvPathSeparator) +
475 EnvVar.substr(PrefixLen)
478 Environment.push_back(Args.MakeArgString(EnvVar));
480 Cursor += EnvVar.size() + 1 ;
483 SkipSettingEnvironment:;
486 linkPath = TC.GetProgramPath(
Linker.str().c_str());
489 auto LinkCmd = std::make_unique<Command>(
491 Args.MakeArgString(linkPath), CmdArgs, Inputs, Output);
492 if (!Environment.empty())
493 LinkCmd->setEnvironment(Environment);
494 C.addCommand(std::move(LinkCmd));
723 ArgStringList &CC1Args)
const {
724 if (DriverArgs.hasArg(options::OPT_nostdinc))
727 if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
733 for (
const auto &Path : DriverArgs.getAllArgValues(options::OPT__SLASH_imsvc))
736 auto AddSystemIncludesFromEnv = [&](StringRef Var) ->
bool {
737 if (
auto Val = llvm::sys::Process::GetEnv(Var)) {
739 StringRef(*Val).split(Dirs,
";", -1,
false);
749 for (
const auto &Var :
750 DriverArgs.getAllArgValues(options::OPT__SLASH_external_env)) {
751 AddSystemIncludesFromEnv(Var);
755 if (
const Arg *A = DriverArgs.getLastArg(options::OPT__SLASH_diasdkdir,
756 options::OPT__SLASH_winsysroot)) {
761 if (A->getOption().getID() == options::OPT__SLASH_winsysroot)
762 llvm::sys::path::append(DIASDKPath,
"DIA SDK");
767 if (DriverArgs.hasArg(options::OPT_nostdlibinc))
773 if (!DriverArgs.hasArg(
774 options::OPT__SLASH_vctoolsdir, options::OPT__SLASH_vctoolsversion,
775 options::OPT__SLASH_winsysroot, options::OPT__SLASH_winsdkdir,
776 options::OPT__SLASH_winsdkversion)) {
777 bool Found = AddSystemIncludesFromEnv(
"INCLUDE");
778 Found |= AddSystemIncludesFromEnv(
"EXTERNAL_INCLUDE");
785 if (!VCToolChainPath.empty()) {
793 std::string UniversalCRTSdkPath;
794 std::string UCRTVersion;
795 if (llvm::getUniversalCRTSdkDir(
getVFS(), WinSdkDir, WinSdkVersion,
796 WinSysRoot, UniversalCRTSdkPath,
798 if (!(WinSdkDir.has_value() || WinSysRoot.has_value()) &&
799 WinSdkVersion.has_value())
800 UCRTVersion = *WinSdkVersion;
802 "Include", UCRTVersion,
"ucrt");
806 std::string WindowsSDKDir;
808 std::string windowsSDKIncludeVersion;
809 std::string windowsSDKLibVersion;
810 if (llvm::getWindowsSDKDir(
getVFS(), WinSdkDir, WinSdkVersion, WinSysRoot,
811 WindowsSDKDir, major, windowsSDKIncludeVersion,
812 windowsSDKLibVersion)) {
814 if (!(WinSdkDir.has_value() || WinSysRoot.has_value()) &&
815 WinSdkVersion.has_value())
816 windowsSDKIncludeVersion = windowsSDKLibVersion = *WinSdkVersion;
821 "Include", windowsSDKIncludeVersion,
824 "Include", windowsSDKIncludeVersion,
827 "Include", windowsSDKIncludeVersion,
830 llvm::VersionTuple Tuple;
831 if (!Tuple.tryParse(windowsSDKIncludeVersion) &&
832 Tuple.getSubminor().value_or(0) >= 17134) {
834 "Include", windowsSDKIncludeVersion,
850 const StringRef Paths[] = {
851 "C:/Program Files/Microsoft Visual Studio 10.0/VC/include",
852 "C:/Program Files/Microsoft Visual Studio 9.0/VC/include",
853 "C:/Program Files/Microsoft Visual Studio 9.0/VC/PlatformSDK/Include",
854 "C:/Program Files/Microsoft Visual Studio 8/VC/include",
855 "C:/Program Files/Microsoft Visual Studio 8/VC/PlatformSDK/Include"
924 bool SupportsForcingFramePointer,
925 const char *ExpandChar,
const OptTable &Opts) {
926 assert(A->getOption().matches(options::OPT__SLASH_O));
928 StringRef OptStr = A->getValue();
929 for (
size_t I = 0, E = OptStr.size(); I != E; ++I) {
930 const char &OptChar = *(OptStr.data() + I);
940 if (&OptChar != ExpandChar) {
944 if (OptChar ==
'd') {
945 DAL.AddFlagArg(A, Opts.getOption(options::OPT_O0));
947 if (OptChar ==
'1') {
948 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"s");
949 }
else if (OptChar ==
'2' || OptChar ==
'x') {
950 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fbuiltin));
951 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"3");
953 if (SupportsForcingFramePointer &&
954 !DAL.hasArgNoClaim(options::OPT_fno_omit_frame_pointer))
955 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fomit_frame_pointer));
956 if (OptChar ==
'1' || OptChar ==
'2')
957 DAL.AddFlagArg(A, Opts.getOption(options::OPT_ffunction_sections));
961 if (I + 1 != E && isdigit(OptStr[I + 1])) {
962 switch (OptStr[I + 1]) {
964 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fno_inline));
967 DAL.AddFlagArg(A, Opts.getOption(options::OPT_finline_hint_functions));
971 DAL.AddFlagArg(A, Opts.getOption(options::OPT_finline_functions));
981 if (I + 1 != E && OptStr[I + 1] ==
'-') {
983 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fno_builtin));
985 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fbuiltin));
989 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"s");
992 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"3");
995 bool OmitFramePointer =
true;
996 if (I + 1 != E && OptStr[I + 1] ==
'-') {
997 OmitFramePointer =
false;
1000 if (SupportsForcingFramePointer) {
1001 if (OmitFramePointer)
1003 Opts.getOption(options::OPT_fomit_frame_pointer));
1006 A, Opts.getOption(options::OPT_fno_omit_frame_pointer));