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 for (
const auto &LibPath : TC.getFilePaths()) {
209 if (LibPath.length() > 0)
210 CmdArgs.push_back(Args.MakeArgString(
"-libpath:" + LibPath));
212 auto CRTPath = TC.getCompilerRTPath();
213 if (TC.getVFS().exists(CRTPath))
214 CmdArgs.push_back(Args.MakeArgString(
"-libpath:" + CRTPath));
219 if (Args.hasFlag(options::OPT_fsycl, options::OPT_fno_sycl,
false))
220 CmdArgs.push_back(
"/IGNORE:4078");
222 CmdArgs.push_back(
"-nologo");
224 if (Args.hasArg(options::OPT_g_Group, options::OPT__SLASH_Z7))
225 CmdArgs.push_back(
"-debug");
229 if (Args.hasArg(options::OPT_fms_hotpatch, options::OPT__SLASH_hotpatch))
230 CmdArgs.push_back(
"-functionpadmin");
234 bool DefaultIncrementalLinkerCompatible =
235 C.getDefaultToolChain().getTriple().isWindowsMSVCEnvironment();
236 if (!Args.hasFlag(options::OPT_mincremental_linker_compatible,
237 options::OPT_mno_incremental_linker_compatible,
238 DefaultIncrementalLinkerCompatible))
239 CmdArgs.push_back(
"-Brepro");
241 bool DLL = Args.hasArg(options::OPT__SLASH_LD, options::OPT__SLASH_LDd,
242 options::OPT_shared);
244 CmdArgs.push_back(Args.MakeArgString(
"-dll"));
247 llvm::sys::path::replace_extension(ImplibName,
"lib");
248 CmdArgs.push_back(Args.MakeArgString(std::string(
"-implib:") + ImplibName));
251 if (TC.getSanitizerArgs(Args).needsFuzzer()) {
252 if (!Args.hasArg(options::OPT_shared))
254 Args.MakeArgString(std::string(
"-wholearchive:") +
255 TC.getCompilerRTArgString(Args,
"fuzzer")));
256 CmdArgs.push_back(Args.MakeArgString(
"-debug"));
259 CmdArgs.push_back(Args.MakeArgString(
"-incremental:no"));
262 if (TC.getSanitizerArgs(Args).needsAsanRt()) {
263 CmdArgs.push_back(Args.MakeArgString(
"-debug"));
264 CmdArgs.push_back(Args.MakeArgString(
"-incremental:no"));
265 CmdArgs.push_back(TC.getCompilerRTArgString(Args,
"asan_dynamic"));
266 auto defines = Args.getAllArgValues(options::OPT_D);
267 if (Args.hasArg(options::OPT__SLASH_MD, options::OPT__SLASH_MDd) ||
268 llvm::is_contained(defines,
"_DLL")) {
271 CmdArgs.push_back(Args.MakeArgString(
272 TC.getArch() == llvm::Triple::x86
273 ?
"-include:___asan_seh_interceptor"
274 :
"-include:__asan_seh_interceptor"));
277 CmdArgs.push_back(Args.MakeArgString(
278 std::string(
"-wholearchive:") +
279 TC.getCompilerRT(Args,
"asan_dynamic_runtime_thunk")));
283 CmdArgs.push_back(Args.MakeArgString(
284 std::string(
"-wholearchive:") +
285 TC.getCompilerRT(Args,
"asan_static_runtime_thunk")));
289 if (TC.isUsingLTO(Args)) {
291 CmdArgs.push_back(Args.MakeArgString(std::string(
"-lto-sample-profile:") +
294 Args.AddAllArgValues(CmdArgs, options::OPT__SLASH_link);
297 for (
const Arg *A : Args.filtered(options::OPT__SLASH_guard)) {
298 StringRef GuardArgs = A->getValue();
299 if (GuardArgs.equals_insensitive(
"cf") ||
300 GuardArgs.equals_insensitive(
"cf,nochecks")) {
302 CmdArgs.push_back(
"-guard:cf");
303 }
else if (GuardArgs.equals_insensitive(
"cf-")) {
304 CmdArgs.push_back(
"-guard:cf-");
305 }
else if (GuardArgs.equals_insensitive(
"ehcont")) {
306 CmdArgs.push_back(
"-guard:ehcont");
307 }
else if (GuardArgs.equals_insensitive(
"ehcont-")) {
308 CmdArgs.push_back(
"-guard:ehcont-");
312 if (Args.hasFlag(options::OPT_fopenmp, options::OPT_fopenmp_EQ,
313 options::OPT_fno_openmp,
false)) {
314 CmdArgs.push_back(
"-nodefaultlib:vcomp.lib");
315 CmdArgs.push_back(
"-nodefaultlib:vcompd.lib");
316 CmdArgs.push_back(Args.MakeArgString(std::string(
"-libpath:") +
317 TC.getDriver().Dir +
"/../lib"));
318 switch (TC.getDriver().getOpenMPRuntime(Args)) {
320 CmdArgs.push_back(
"-defaultlib:libomp.lib");
323 CmdArgs.push_back(
"-defaultlib:libiomp5md.lib");
335 if (!Args.hasArg(options::OPT_nostdlib)) {
339 const Arg *A = Args.getLastArg(options::OPT_fuse_ld_EQ);
340 StringRef
Linker = A ? A->getValue() : TC.getDriver().getPreferredLinker();
347 if (Args.hasArg(options::OPT_gdwarf, options::OPT_gdwarf_2,
348 options::OPT_gdwarf_3, options::OPT_gdwarf_4,
349 options::OPT_gdwarf_5, options::OPT_gdwarf_6))
356 if (
Linker.equals_insensitive(
"lld"))
359 if (
Linker ==
"lld-link") {
360 for (Arg *A : Args.filtered(options::OPT_vfsoverlay))
362 Args.MakeArgString(std::string(
"/vfsoverlay:") + A->getValue()));
364 if (TC.isUsingLTO(Args) &&
365 Args.hasFlag(options::OPT_gsplit_dwarf, options::OPT_gno_split_dwarf,
367 CmdArgs.push_back(Args.MakeArgString(Twine(
"/dwodir:") +
372 for (
const auto &Input : Inputs) {
373 if (Input.isFilename()) {
374 CmdArgs.push_back(Input.getFilename());
378 const Arg &A = Input.getInputArg();
381 if (A.getOption().matches(options::OPT_l)) {
382 StringRef Lib = A.getValue();
383 const char *LinkLibArg;
384 if (Lib.ends_with(
".lib"))
385 LinkLibArg = Args.MakeArgString(Lib);
387 LinkLibArg = Args.MakeArgString(Lib +
".lib");
388 CmdArgs.push_back(LinkLibArg);
394 A.renderAsInput(Args, CmdArgs);
397 TC.addOffloadRTLibs(
C.getActiveOffloadKinds(), Args, CmdArgs);
399 TC.addProfileRTLibs(Args, CmdArgs);
401 std::vector<const char *> Environment;
406 if (
Linker.equals_insensitive(
"link")) {
412 if (!TC.FoundMSVCInstall() && !
canExecute(TC.getVFS(), linkPath)) {
414 ClPath = TC.GetProgramPath(
"cl.exe");
416 linkPath = llvm::sys::path::parent_path(ClPath);
417 llvm::sys::path::append(linkPath,
"link.exe");
419 C.getDriver().Diag(clang::diag::warn_drv_msvc_not_found);
421 C.getDriver().Diag(clang::diag::warn_drv_msvc_not_found);
427 if (TC.getSanitizerArgs(Args).needsAsanRt())
428 CmdArgs.push_back(
"/INFERASANLIBS:NO");
437 if (TC.getIsVS2017OrNewer() &&
438 llvm::Triple(llvm::sys::getProcessTriple()).getArch() != TC.getArch()) {
439 auto HostArch = llvm::Triple(llvm::sys::getProcessTriple()).getArch();
442 std::unique_ptr<wchar_t[], decltype(&FreeEnvironmentStringsW)>(
443 GetEnvironmentStringsW(), FreeEnvironmentStringsW);
445 goto SkipSettingEnvironment;
448 size_t EnvBlockLen = 0;
449 while (EnvBlockWide[EnvBlockLen] != L
'\0') {
451 EnvBlockLen += std::wcslen(&EnvBlockWide[EnvBlockLen]) +
456 std::string EnvBlock;
457 if (!llvm::convertUTF16ToUTF8String(
459 EnvBlockLen *
sizeof(EnvBlockWide[0])),
461 goto SkipSettingEnvironment;
463 Environment.reserve(EnvCount);
468 for (
const char *Cursor = EnvBlock.data(); *Cursor !=
'\0';) {
469 llvm::StringRef EnvVar(Cursor);
470 if (EnvVar.starts_with_insensitive(
"path=")) {
471 constexpr size_t PrefixLen = 5;
472 Environment.push_back(Args.MakeArgString(
473 EnvVar.substr(0, PrefixLen) +
474 TC.getSubDirectoryPath(llvm::SubDirectoryType::Bin) +
475 llvm::Twine(llvm::sys::EnvPathSeparator) +
476 TC.getSubDirectoryPath(llvm::SubDirectoryType::Bin, HostArch) +
477 (EnvVar.size() > PrefixLen
478 ? llvm::Twine(llvm::sys::EnvPathSeparator) +
479 EnvVar.substr(PrefixLen)
482 Environment.push_back(Args.MakeArgString(EnvVar));
484 Cursor += EnvVar.size() + 1 ;
487 SkipSettingEnvironment:;
490 linkPath = TC.GetProgramPath(
Linker.str().c_str());
493 auto LinkCmd = std::make_unique<Command>(
495 Args.MakeArgString(linkPath), CmdArgs, Inputs, Output);
496 if (!Environment.empty())
497 LinkCmd->setEnvironment(Environment);
498 C.addCommand(std::move(LinkCmd));
736 ArgStringList &CC1Args)
const {
737 if (DriverArgs.hasArg(options::OPT_nostdinc))
740 if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
746 for (
const auto &Path : DriverArgs.getAllArgValues(options::OPT__SLASH_imsvc))
749 auto AddSystemIncludesFromEnv = [&](StringRef Var) ->
bool {
750 if (
auto Val = llvm::sys::Process::GetEnv(Var)) {
752 StringRef(*Val).split(Dirs,
";", -1,
false);
762 for (
const auto &Var :
763 DriverArgs.getAllArgValues(options::OPT__SLASH_external_env)) {
764 AddSystemIncludesFromEnv(Var);
768 if (
const Arg *A = DriverArgs.getLastArg(options::OPT__SLASH_diasdkdir,
769 options::OPT__SLASH_winsysroot)) {
774 if (A->getOption().getID() == options::OPT__SLASH_winsysroot)
775 llvm::sys::path::append(DIASDKPath,
"DIA SDK");
780 if (DriverArgs.hasArg(options::OPT_nostdlibinc))
789 llvm::sys::path::append(Dir, M.includeSuffix());
798 if (!DriverArgs.hasArg(
799 options::OPT__SLASH_vctoolsdir, options::OPT__SLASH_vctoolsversion,
800 options::OPT__SLASH_winsysroot, options::OPT__SLASH_winsdkdir,
801 options::OPT__SLASH_winsdkversion)) {
802 bool Found = AddSystemIncludesFromEnv(
"INCLUDE");
803 Found |= AddSystemIncludesFromEnv(
"EXTERNAL_INCLUDE");
810 if (!VCToolChainPath.empty()) {
818 std::string UniversalCRTSdkPath;
819 std::string UCRTVersion;
820 if (llvm::getUniversalCRTSdkDir(
getVFS(), WinSdkDir, WinSdkVersion,
821 WinSysRoot, UniversalCRTSdkPath,
823 if (!(WinSdkDir.has_value() || WinSysRoot.has_value()) &&
824 WinSdkVersion.has_value())
825 UCRTVersion = *WinSdkVersion;
827 "Include", UCRTVersion,
"ucrt");
831 std::string WindowsSDKDir;
833 std::string windowsSDKIncludeVersion;
834 std::string windowsSDKLibVersion;
835 if (llvm::getWindowsSDKDir(
getVFS(), WinSdkDir, WinSdkVersion, WinSysRoot,
836 WindowsSDKDir, major, windowsSDKIncludeVersion,
837 windowsSDKLibVersion)) {
839 if (!(WinSdkDir.has_value() || WinSysRoot.has_value()) &&
840 WinSdkVersion.has_value())
841 windowsSDKIncludeVersion = windowsSDKLibVersion = *WinSdkVersion;
846 "Include", windowsSDKIncludeVersion,
849 "Include", windowsSDKIncludeVersion,
852 "Include", windowsSDKIncludeVersion,
855 llvm::VersionTuple Tuple;
856 if (!Tuple.tryParse(windowsSDKIncludeVersion) &&
857 Tuple.getSubminor().value_or(0) >= 17134) {
859 "Include", windowsSDKIncludeVersion,
875 const StringRef Paths[] = {
876 "C:/Program Files/Microsoft Visual Studio 10.0/VC/include",
877 "C:/Program Files/Microsoft Visual Studio 9.0/VC/include",
878 "C:/Program Files/Microsoft Visual Studio 9.0/VC/PlatformSDK/Include",
879 "C:/Program Files/Microsoft Visual Studio 8/VC/include",
880 "C:/Program Files/Microsoft Visual Studio 8/VC/PlatformSDK/Include"
949 bool SupportsForcingFramePointer,
950 const char *ExpandChar,
const OptTable &Opts) {
951 assert(A->getOption().matches(options::OPT__SLASH_O));
953 StringRef OptStr = A->getValue();
954 for (
size_t I = 0, E = OptStr.size(); I != E; ++I) {
955 const char &OptChar = *(OptStr.data() + I);
965 if (&OptChar != ExpandChar) {
969 if (OptChar ==
'd') {
970 DAL.AddFlagArg(A, Opts.getOption(options::OPT_O0));
972 if (OptChar ==
'1') {
973 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"s");
974 }
else if (OptChar ==
'2' || OptChar ==
'x') {
975 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fbuiltin));
976 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"3");
978 if (SupportsForcingFramePointer &&
979 !DAL.hasArgNoClaim(options::OPT_fno_omit_frame_pointer))
980 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fomit_frame_pointer));
981 if (OptChar ==
'1' || OptChar ==
'2')
982 DAL.AddFlagArg(A, Opts.getOption(options::OPT_ffunction_sections));
986 if (I + 1 != E && isdigit(OptStr[I + 1])) {
987 switch (OptStr[I + 1]) {
989 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fno_inline));
992 DAL.AddFlagArg(A, Opts.getOption(options::OPT_finline_hint_functions));
996 DAL.AddFlagArg(A, Opts.getOption(options::OPT_finline_functions));
1006 if (I + 1 != E && OptStr[I + 1] ==
'-') {
1008 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fno_builtin));
1010 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fbuiltin));
1014 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"s");
1017 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"3");
1020 bool OmitFramePointer =
true;
1021 if (I + 1 != E && OptStr[I + 1] ==
'-') {
1022 OmitFramePointer =
false;
1025 if (SupportsForcingFramePointer) {
1026 if (OmitFramePointer)
1028 Opts.getOption(options::OPT_fomit_frame_pointer));
1031 A, Opts.getOption(options::OPT_fno_omit_frame_pointer));