11#include "clang/Config/config.h"
16#include "llvm/Config/llvm-config.h"
17#include "llvm/Option/ArgList.h"
18#include "llvm/Support/FileSystem.h"
19#include "llvm/Support/Path.h"
20#include "llvm/Support/VirtualFileSystem.h"
30std::string WebAssembly::getMultiarchTriple(
const Driver &D,
31 const llvm::Triple &TargetTriple,
32 StringRef SysRoot)
const {
33 return (TargetTriple.getArchName() +
"-" +
34 TargetTriple.getOSAndEnvironmentName()).str();
41 if (DriverArgs.getLastArg(options::OPT_fwasm_exceptions))
48 if (
const Arg* A = Args.getLastArg(options::OPT_fuse_ld_EQ)) {
49 StringRef UseLinker = A->getValue();
50 if (!UseLinker.empty()) {
51 if (llvm::sys::path::is_absolute(UseLinker) &&
52 llvm::sys::fs::can_execute(UseLinker))
53 return std::string(UseLinker);
58 if (UseLinker ==
"lld") {
63 if (UseLinker !=
"ld")
65 << A->getAsString(Args);
75 return TargetTriple.isOSWASI() && TargetTriple.getOSName() !=
"wasip1" &&
76 TargetTriple.getOSName() !=
"wasi";
79static bool WantsPthread(
const llvm::Triple &Triple,
const ArgList &Args) {
81 Args.hasFlag(options::OPT_pthread, options::OPT_no_pthread,
false);
85 if (Triple.isOSWASI() && Triple.getEnvironmentName() ==
"threads")
92 const ArgList &Args) {
93 return Triple.getOS() == llvm::Triple::WASIp3;
100 const char *LinkingOutput)
const {
104 ArgStringList CmdArgs;
106 CmdArgs.push_back(
"-m");
108 CmdArgs.push_back(
"wasm64");
110 CmdArgs.push_back(
"wasm32");
112 if (Args.hasArg(options::OPT_s))
113 CmdArgs.push_back(
"--strip-all");
119 if (llvm::sys::path::stem(
Linker).ends_with_insensitive(
120 "wasm-component-ld")) {
121 CmdArgs.push_back(
"--wasm-ld-path");
125 Args.addAllArgs(CmdArgs, {options::OPT_L, options::OPT_u});
129 bool IsCommand =
true;
131 const char *Entry =
nullptr;
135 if (Args.hasArg(options::OPT_shared))
138 if (
const Arg *A = Args.getLastArg(options::OPT_mexec_model_EQ)) {
139 StringRef CM = A->getValue();
140 if (CM ==
"command") {
142 }
else if (CM ==
"reactor") {
146 << CM << A->getOption().getName();
159 Crt1 =
"crt1-command.o";
161 Crt1 =
"crt1-reactor.o";
162 Entry =
"_initialize";
165 if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles))
168 CmdArgs.push_back(Args.MakeArgString(
"--entry"));
169 CmdArgs.push_back(Args.MakeArgString(Entry));
172 if (Args.hasArg(options::OPT_shared))
173 CmdArgs.push_back(Args.MakeArgString(
"-shared"));
178 CmdArgs.push_back(
"--libcall-thread-context");
181 CmdArgs.push_back(
"--shared-memory");
183 if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) {
188 CmdArgs.push_back(
"-lpthread");
190 CmdArgs.push_back(
"-lc");
196 CmdArgs.push_back(
"-o");
203 bool RunWasmOpt = Args.hasFlag(options::OPT_wasm_opt,
204 options::OPT_no_wasm_opt, WasmOptDefault);
208 std::string WasmOptPath;
209 if (RunWasmOpt && Args.getLastArg(options::OPT_O_Group)) {
211 if (WasmOptPath ==
"wasm-opt") {
216 if (!WasmOptPath.empty()) {
217 CmdArgs.push_back(
"--keep-section=target_features");
220 C.addCommand(std::make_unique<Command>(JA, *
this,
222 Linker, CmdArgs, Inputs, Output));
224 if (Arg *A = Args.getLastArg(options::OPT_O_Group)) {
225 if (!WasmOptPath.empty()) {
226 StringRef OOpt =
"s";
227 if (A->getOption().matches(options::OPT_O4) ||
228 A->getOption().matches(options::OPT_Ofast))
230 else if (A->getOption().matches(options::OPT_O0))
232 else if (A->getOption().matches(options::OPT_O))
233 OOpt = A->getValue();
236 const char *WasmOpt = Args.MakeArgString(WasmOptPath);
237 ArgStringList OptArgs;
239 OptArgs.push_back(Args.MakeArgString(llvm::Twine(
"-O") + OOpt));
240 OptArgs.push_back(
"-o");
242 C.addCommand(std::make_unique<Command>(
254 const llvm::opt::ArgList &Args,
255 const std::string &Dir) {
259 Paths.push_back(Dir +
"/llvm-lto/" LLVM_VERSION_STRING);
261 Paths.push_back(Dir);
265 const llvm::opt::ArgList &Args)
266 : ToolChain(D, Triple, Args) {
268 assert(Triple.isArch32Bit() != Triple.isArch64Bit());
270 getProgramPaths().push_back(getDriver().Dir);
272 auto SysRoot = getDriver().SysRoot;
273 if (
getTriple().getOS() == llvm::Triple::UnknownOS) {
278 getFilePaths().push_back(SysRoot +
"/lib");
280 const std::string MultiarchTriple =
281 getMultiarchTriple(getDriver(), Triple, SysRoot);
282 std::string TripleLibDir = SysRoot +
"/lib/" + MultiarchTriple;
293 if (
getTriple().getOS() == llvm::Triple::WASI) {
294 D.
Diag(diag::warn_drv_deprecated_custom)
295 <<
"--target=wasm32-wasi"
296 <<
"use --target=wasm32-wasip1 instead";
300const char *WebAssembly::getDefaultLinker()
const {
302 return "wasm-component-ld";
314bool WebAssembly::isPIEDefault(
const llvm::opt::ArgList &Args)
const {
327void WebAssembly::addClangTargetOptions(
const ArgList &DriverArgs,
328 ArgStringList &CC1Args,
331 if (!DriverArgs.hasFlag(options::OPT_fuse_init_array,
332 options::OPT_fno_use_init_array,
true))
333 CC1Args.push_back(
"-fno-use-init-array");
337 if (DriverArgs.hasFlag(options::OPT_mno_atomics, options::OPT_matomics,
342 if (DriverArgs.hasFlag(options::OPT_mno_bulk_memory,
343 options::OPT_mbulk_memory,
false))
346 <<
"-mno-bulk-memory";
347 if (DriverArgs.hasFlag(options::OPT_mno_mutable_globals,
348 options::OPT_mmutable_globals,
false))
351 <<
"-mno-mutable-globals";
352 if (DriverArgs.hasFlag(options::OPT_mno_sign_ext, options::OPT_msign_ext,
357 CC1Args.push_back(
"-target-feature");
358 CC1Args.push_back(
"+atomics");
359 CC1Args.push_back(
"-target-feature");
360 CC1Args.push_back(
"+bulk-memory");
361 CC1Args.push_back(
"-target-feature");
362 CC1Args.push_back(
"+mutable-globals");
363 CC1Args.push_back(
"-target-feature");
364 CC1Args.push_back(
"+sign-ext");
367 if (!DriverArgs.hasFlag(options::OPT_mmutable_globals,
368 options::OPT_mno_mutable_globals,
false)) {
371 llvm::Reloc::Model RelocationModel;
374 std::tie(RelocationModel, PICLevel, IsPIE) =
376 if (RelocationModel == llvm::Reloc::PIC_) {
377 if (DriverArgs.hasFlag(options::OPT_mno_mutable_globals,
378 options::OPT_mmutable_globals,
false)) {
381 <<
"-mno-mutable-globals";
383 CC1Args.push_back(
"-target-feature");
384 CC1Args.push_back(
"+mutable-globals");
388 bool HasBannedIncompatibleOptionsForWasmEHSjLj =
false;
389 bool HasEnabledFeaturesForWasmEHSjLj =
false;
393 auto BanIncompatibleOptionsForWasmEHSjLj = [&](StringRef CurOption) {
394 if (HasBannedIncompatibleOptionsForWasmEHSjLj)
396 HasBannedIncompatibleOptionsForWasmEHSjLj =
true;
397 if (DriverArgs.hasFlag(options::OPT_mno_exception_handing,
398 options::OPT_mexception_handing,
false))
400 << CurOption <<
"-mno-exception-handling";
402 if (DriverArgs.hasFlag(options::OPT_mno_multivalue,
403 options::OPT_mmultivalue,
false))
405 << CurOption <<
"-mno-multivalue";
406 if (DriverArgs.hasFlag(options::OPT_mno_reference_types,
407 options::OPT_mreference_types,
false))
409 << CurOption <<
"-mno-reference-types";
411 for (
const Arg *A : DriverArgs.filtered(options::OPT_mllvm)) {
412 for (
const auto *Option :
413 {
"-enable-emscripten-cxx-exceptions",
"-enable-emscripten-sjlj",
414 "-emscripten-cxx-exceptions-allowed"}) {
415 if (StringRef(A->getValue(0)) == Option)
417 << CurOption << Option;
423 auto EnableFeaturesForWasmEHSjLj = [&]() {
424 if (HasEnabledFeaturesForWasmEHSjLj)
426 HasEnabledFeaturesForWasmEHSjLj =
true;
427 CC1Args.push_back(
"-target-feature");
428 CC1Args.push_back(
"+exception-handling");
430 CC1Args.push_back(
"-target-feature");
431 CC1Args.push_back(
"+multivalue");
432 CC1Args.push_back(
"-target-feature");
433 CC1Args.push_back(
"+reference-types");
435 CC1Args.push_back(
"-exception-model=wasm");
438 if (DriverArgs.getLastArg(options::OPT_fwasm_exceptions)) {
439 BanIncompatibleOptionsForWasmEHSjLj(
"-fwasm-exceptions");
440 EnableFeaturesForWasmEHSjLj();
442 CC1Args.push_back(
"-mllvm");
443 CC1Args.push_back(
"-wasm-enable-eh");
446 for (
const Arg *A : DriverArgs.filtered(options::OPT_mllvm)) {
447 StringRef Opt = A->getValue(0);
448 if (Opt.starts_with(
"-emscripten-cxx-exceptions-allowed")) {
451 bool EmEHArgExists =
false;
452 for (
const Arg *A : DriverArgs.filtered(options::OPT_mllvm)) {
453 if (StringRef(A->getValue(0)) ==
"-enable-emscripten-cxx-exceptions") {
454 EmEHArgExists =
true;
460 <<
"-mllvm -emscripten-cxx-exceptions-allowed"
461 <<
"-mllvm -enable-emscripten-cxx-exceptions";
465 StringRef FuncNamesStr = Opt.split(
'=').second;
466 SmallVector<StringRef, 4> FuncNames;
467 FuncNamesStr.split(FuncNames,
',');
468 for (
auto Name : FuncNames) {
469 CC1Args.push_back(
"-mllvm");
470 CC1Args.push_back(DriverArgs.MakeArgString(
"--force-attribute=" + Name +
475 for (
const auto *Option :
476 {
"-wasm-enable-eh",
"-wasm-enable-sjlj",
"-wasm-use-legacy-eh"}) {
477 if (Opt.starts_with(Option)) {
478 BanIncompatibleOptionsForWasmEHSjLj(Option);
479 EnableFeaturesForWasmEHSjLj();
490WebAssembly::GetCXXStdlibType(
const ArgList &Args)
const {
491 if (Arg *A = Args.getLastArg(options::OPT_stdlib_EQ)) {
492 StringRef
Value = A->getValue();
493 if (
Value ==
"libc++")
495 else if (
Value ==
"libstdc++")
499 << A->getAsString(Args);
504void WebAssembly::AddClangSystemIncludeArgs(
const ArgList &DriverArgs,
505 ArgStringList &CC1Args)
const {
506 if (DriverArgs.hasArg(options::OPT_nostdinc))
511 if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
512 SmallString<128> P(D.ResourceDir);
513 llvm::sys::path::append(P,
"include");
517 if (DriverArgs.hasArg(options::OPT_nostdlibinc))
521 StringRef CIncludeDirs(C_INCLUDE_DIRS);
522 if (CIncludeDirs !=
"") {
523 SmallVector<StringRef, 5> dirs;
524 CIncludeDirs.split(dirs,
":");
525 for (StringRef dir : dirs) {
527 llvm::sys::path::is_absolute(dir) ?
"" : StringRef(D.SysRoot);
534 const std::string MultiarchTriple =
535 getMultiarchTriple(D,
getTriple(), D.SysRoot);
536 addSystemInclude(DriverArgs, CC1Args, D.SysRoot +
"/include/" + MultiarchTriple);
541void WebAssembly::AddClangCXXStdlibIncludeArgs(
const ArgList &DriverArgs,
542 ArgStringList &CC1Args)
const {
544 if (DriverArgs.hasArg(options::OPT_nostdlibinc, options::OPT_nostdinc,
545 options::OPT_nostdincxx))
548 switch (GetCXXStdlibType(DriverArgs)) {
550 addLibCxxIncludePaths(DriverArgs, CC1Args);
553 addLibStdCXXIncludePaths(DriverArgs, CC1Args);
558void WebAssembly::AddCXXStdlibLibArgs(
const llvm::opt::ArgList &Args,
559 llvm::opt::ArgStringList &CmdArgs)
const {
561 switch (GetCXXStdlibType(Args)) {
563 CmdArgs.push_back(
"-lc++");
564 if (Args.hasArg(options::OPT_fexperimental_library))
565 CmdArgs.push_back(
"-lc++experimental");
566 CmdArgs.push_back(
"-lc++abi");
569 CmdArgs.push_back(
"-lstdc++");
574SanitizerMask WebAssembly::getSupportedSanitizers(
579 Res |= SanitizerKind::Vptr | SanitizerKind::Leak;
583 Res |= SanitizerKind::Address;
593 return new tools::wasm::Linker(*
this);
596void WebAssembly::addLibCxxIncludePaths(
597 const llvm::opt::ArgList &DriverArgs,
598 llvm::opt::ArgStringList &CC1Args)
const {
601 std::string LibPath = SysRoot +
"/include";
602 const std::string MultiarchTriple =
603 getMultiarchTriple(D,
getTriple(), SysRoot);
604 bool IsKnownOs = (
getTriple().getOS() != llvm::Triple::UnknownOS);
613 std::string TargetDir = LibPath +
"/" + MultiarchTriple;
614 std::string Suffix =
"/c++/" + Version;
625void WebAssembly::addLibStdCXXIncludePaths(
626 const llvm::opt::ArgList &DriverArgs,
627 llvm::opt::ArgStringList &CC1Args)
const {
634 std::string LibPath = SysRoot +
"/include";
635 const std::string MultiarchTriple =
636 getMultiarchTriple(D,
getTriple(), SysRoot);
637 bool IsKnownOs = (
getTriple().getOS() != llvm::Triple::UnknownOS);
643 Generic_GCC::GCCVersion MaxVersion =
645 SmallString<128> Path(LibPath);
646 llvm::sys::path::append(Path,
"c++");
647 for (llvm::vfs::directory_iterator LI =
getVFS().dir_begin(Path, EC), LE;
648 !EC && LI !=
LE; LI = LI.increment(EC)) {
649 StringRef VersionText = llvm::sys::path::filename(LI->path());
650 if (VersionText[0] !=
'v') {
652 if (Version > MaxVersion)
653 MaxVersion = Version;
656 if (MaxVersion.
Major > 0)
657 Version = MaxVersion.
Text;
665 std::string TargetDir = LibPath +
"/c++/" + Version +
"/" + MultiarchTriple;
672 addSystemInclude(DriverArgs, CC1Args, LibPath +
"/c++/" + Version +
"/backward");
static StringRef getTriple(const Command &Job)
Compilation - A set of tasks to perform for a single driver invocation.
Driver - Encapsulate logic for constructing compilation processes from a set of gcc-driver-like comma...
DiagnosticBuilder Diag(unsigned DiagID) const
SmallVector< InputInfo, 4 > InputInfoList
bool LE(InterpState &S, CodePtr OpPC)
The JSON file list parser is used to communicate input to InstallAPI.
bool(*)(llvm::ArrayRef< const char * >, llvm::raw_ostream &, llvm::raw_ostream &, bool, bool) Driver
static constexpr ResponseFileSupport AtFileCurCP()