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));
742 ArgStringList &CC1Args)
const {
743 if (DriverArgs.hasArg(options::OPT_nostdinc))
746 if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
752 for (
const auto &Path : DriverArgs.getAllArgValues(options::OPT__SLASH_imsvc))
755 auto AddSystemIncludesFromEnv = [&](StringRef Var) ->
bool {
756 if (
auto Val = llvm::sys::Process::GetEnv(Var)) {
758 StringRef(*Val).split(Dirs,
";", -1,
false);
768 for (
const auto &Var :
769 DriverArgs.getAllArgValues(options::OPT__SLASH_external_env)) {
770 AddSystemIncludesFromEnv(Var);
774 if (
const Arg *A = DriverArgs.getLastArg(options::OPT__SLASH_diasdkdir,
775 options::OPT__SLASH_winsysroot)) {
780 if (A->getOption().getID() == options::OPT__SLASH_winsysroot)
781 llvm::sys::path::append(DIASDKPath,
"DIA SDK");
786 if (DriverArgs.hasArg(options::OPT_nostdlibinc))
795 llvm::sys::path::append(Dir, M.includeSuffix());
804 if (!DriverArgs.hasArg(
805 options::OPT__SLASH_vctoolsdir, options::OPT__SLASH_vctoolsversion,
806 options::OPT__SLASH_winsysroot, options::OPT__SLASH_winsdkdir,
807 options::OPT__SLASH_winsdkversion)) {
808 bool Found = AddSystemIncludesFromEnv(
"INCLUDE");
809 Found |= AddSystemIncludesFromEnv(
"EXTERNAL_INCLUDE");
816 if (!VCToolChainPath.empty()) {
824 std::string UniversalCRTSdkPath;
825 std::string UCRTVersion;
826 if (llvm::getUniversalCRTSdkDir(
getVFS(), WinSdkDir, WinSdkVersion,
827 WinSysRoot, UniversalCRTSdkPath,
829 if (!(WinSdkDir.has_value() || WinSysRoot.has_value()) &&
830 WinSdkVersion.has_value())
831 UCRTVersion = *WinSdkVersion;
833 "Include", UCRTVersion,
"ucrt");
837 std::string WindowsSDKDir;
839 std::string windowsSDKIncludeVersion;
840 std::string windowsSDKLibVersion;
841 if (llvm::getWindowsSDKDir(
getVFS(), WinSdkDir, WinSdkVersion, WinSysRoot,
842 WindowsSDKDir, major, windowsSDKIncludeVersion,
843 windowsSDKLibVersion)) {
845 if (!(WinSdkDir.has_value() || WinSysRoot.has_value()) &&
846 WinSdkVersion.has_value())
847 windowsSDKIncludeVersion = windowsSDKLibVersion = *WinSdkVersion;
852 "Include", windowsSDKIncludeVersion,
855 "Include", windowsSDKIncludeVersion,
858 "Include", windowsSDKIncludeVersion,
861 llvm::VersionTuple Tuple;
862 if (!Tuple.tryParse(windowsSDKIncludeVersion) &&
863 Tuple.getSubminor().value_or(0) >= 17134) {
865 "Include", windowsSDKIncludeVersion,
881 const StringRef Paths[] = {
882 "C:/Program Files/Microsoft Visual Studio 10.0/VC/include",
883 "C:/Program Files/Microsoft Visual Studio 9.0/VC/include",
884 "C:/Program Files/Microsoft Visual Studio 9.0/VC/PlatformSDK/Include",
885 "C:/Program Files/Microsoft Visual Studio 8/VC/include",
886 "C:/Program Files/Microsoft Visual Studio 8/VC/PlatformSDK/Include"
955 bool SupportsForcingFramePointer,
956 const char *ExpandChar,
const OptTable &Opts) {
957 assert(A->getOption().matches(options::OPT__SLASH_O));
959 StringRef OptStr = A->getValue();
960 for (
size_t I = 0, E = OptStr.size(); I != E; ++I) {
961 const char &OptChar = *(OptStr.data() + I);
971 if (&OptChar != ExpandChar) {
975 if (OptChar ==
'd') {
976 DAL.AddFlagArg(A, Opts.getOption(options::OPT_O0));
978 if (OptChar ==
'1') {
979 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"s");
980 }
else if (OptChar ==
'2' || OptChar ==
'x') {
981 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fbuiltin));
982 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"3");
984 if (SupportsForcingFramePointer &&
985 !DAL.hasArgNoClaim(options::OPT_fno_omit_frame_pointer))
986 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fomit_frame_pointer));
987 if (OptChar ==
'1' || OptChar ==
'2')
988 DAL.AddFlagArg(A, Opts.getOption(options::OPT_ffunction_sections));
992 if (I + 1 != E && isdigit(OptStr[I + 1])) {
993 switch (OptStr[I + 1]) {
995 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fno_inline));
998 DAL.AddFlagArg(A, Opts.getOption(options::OPT_finline_hint_functions));
1002 DAL.AddFlagArg(A, Opts.getOption(options::OPT_finline_functions));
1012 if (I + 1 != E && OptStr[I + 1] ==
'-') {
1014 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fno_builtin));
1016 DAL.AddFlagArg(A, Opts.getOption(options::OPT_fbuiltin));
1020 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"s");
1023 DAL.AddJoinedArg(A, Opts.getOption(options::OPT_O),
"3");
1026 bool OmitFramePointer =
true;
1027 if (I + 1 != E && OptStr[I + 1] ==
'-') {
1028 OmitFramePointer =
false;
1031 if (SupportsForcingFramePointer) {
1032 if (OmitFramePointer)
1034 Opts.getOption(options::OPT_fomit_frame_pointer));
1037 A, Opts.getOption(options::OPT_fno_omit_frame_pointer));