123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372 |
- //===--- PS4CPU.cpp - PS4CPU ToolChain Implementations ----------*- C++ -*-===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- #include "PS4CPU.h"
- #include "CommonArgs.h"
- #include "clang/Driver/Compilation.h"
- #include "clang/Driver/Driver.h"
- #include "clang/Driver/DriverDiagnostic.h"
- #include "clang/Driver/Options.h"
- #include "clang/Driver/SanitizerArgs.h"
- #include "llvm/Option/ArgList.h"
- #include "llvm/Support/FileSystem.h"
- #include "llvm/Support/Path.h"
- #include <cstdlib> // ::getenv
- using namespace clang::driver;
- using namespace clang;
- using namespace llvm::opt;
- // Helper to paste bits of an option together and return a saved string.
- static const char *makeArgString(const ArgList &Args, const char *Prefix,
- const char *Base, const char *Suffix) {
- // Basically "Prefix + Base + Suffix" all converted to Twine then saved.
- return Args.MakeArgString(Twine(StringRef(Prefix), Base) + Suffix);
- }
- void tools::PScpu::addProfileRTArgs(const ToolChain &TC, const ArgList &Args,
- ArgStringList &CmdArgs) {
- assert(TC.getTriple().isPS());
- auto &PSTC = static_cast<const toolchains::PS4PS5Base &>(TC);
- if ((Args.hasFlag(options::OPT_fprofile_arcs, options::OPT_fno_profile_arcs,
- false) ||
- Args.hasFlag(options::OPT_fprofile_generate,
- options::OPT_fno_profile_generate, false) ||
- Args.hasFlag(options::OPT_fprofile_generate_EQ,
- options::OPT_fno_profile_generate, false) ||
- Args.hasFlag(options::OPT_fprofile_instr_generate,
- options::OPT_fno_profile_instr_generate, false) ||
- Args.hasFlag(options::OPT_fprofile_instr_generate_EQ,
- options::OPT_fno_profile_instr_generate, false) ||
- Args.hasFlag(options::OPT_fcs_profile_generate,
- options::OPT_fno_profile_generate, false) ||
- Args.hasFlag(options::OPT_fcs_profile_generate_EQ,
- options::OPT_fno_profile_generate, false) ||
- Args.hasArg(options::OPT_fcreate_profile) ||
- Args.hasArg(options::OPT_coverage)))
- CmdArgs.push_back(makeArgString(
- Args, "--dependent-lib=", PSTC.getProfileRTLibName(), ""));
- }
- void tools::PScpu::Assembler::ConstructJob(Compilation &C, const JobAction &JA,
- const InputInfo &Output,
- const InputInfoList &Inputs,
- const ArgList &Args,
- const char *LinkingOutput) const {
- auto &TC = static_cast<const toolchains::PS4PS5Base &>(getToolChain());
- claimNoWarnArgs(Args);
- ArgStringList CmdArgs;
- Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA, options::OPT_Xassembler);
- CmdArgs.push_back("-o");
- CmdArgs.push_back(Output.getFilename());
- assert(Inputs.size() == 1 && "Unexpected number of inputs.");
- const InputInfo &Input = Inputs[0];
- assert(Input.isFilename() && "Invalid input.");
- CmdArgs.push_back(Input.getFilename());
- std::string AsName = TC.qualifyPSCmdName("as");
- const char *Exec = Args.MakeArgString(TC.GetProgramPath(AsName.c_str()));
- C.addCommand(std::make_unique<Command>(JA, *this,
- ResponseFileSupport::AtFileUTF8(),
- Exec, CmdArgs, Inputs, Output));
- }
- void tools::PScpu::addSanitizerArgs(const ToolChain &TC, const ArgList &Args,
- ArgStringList &CmdArgs) {
- assert(TC.getTriple().isPS());
- auto &PSTC = static_cast<const toolchains::PS4PS5Base &>(TC);
- PSTC.addSanitizerArgs(Args, CmdArgs, "--dependent-lib=lib", ".a");
- }
- void toolchains::PS4CPU::addSanitizerArgs(const ArgList &Args,
- ArgStringList &CmdArgs,
- const char *Prefix,
- const char *Suffix) const {
- auto arg = [&](const char *Name) -> const char * {
- return makeArgString(Args, Prefix, Name, Suffix);
- };
- const SanitizerArgs &SanArgs = getSanitizerArgs(Args);
- if (SanArgs.needsUbsanRt())
- CmdArgs.push_back(arg("SceDbgUBSanitizer_stub_weak"));
- if (SanArgs.needsAsanRt())
- CmdArgs.push_back(arg("SceDbgAddressSanitizer_stub_weak"));
- }
- void toolchains::PS5CPU::addSanitizerArgs(const ArgList &Args,
- ArgStringList &CmdArgs,
- const char *Prefix,
- const char *Suffix) const {
- auto arg = [&](const char *Name) -> const char * {
- return makeArgString(Args, Prefix, Name, Suffix);
- };
- const SanitizerArgs &SanArgs = getSanitizerArgs(Args);
- if (SanArgs.needsUbsanRt())
- CmdArgs.push_back(arg("SceUBSanitizer_nosubmission_stub_weak"));
- if (SanArgs.needsAsanRt())
- CmdArgs.push_back(arg("SceAddressSanitizer_nosubmission_stub_weak"));
- if (SanArgs.needsTsanRt())
- CmdArgs.push_back(arg("SceThreadSanitizer_nosubmission_stub_weak"));
- }
- void tools::PScpu::Linker::ConstructJob(Compilation &C, const JobAction &JA,
- const InputInfo &Output,
- const InputInfoList &Inputs,
- const ArgList &Args,
- const char *LinkingOutput) const {
- auto &TC = static_cast<const toolchains::PS4PS5Base &>(getToolChain());
- const Driver &D = TC.getDriver();
- ArgStringList CmdArgs;
- // Silence warning for "clang -g foo.o -o foo"
- Args.ClaimAllArgs(options::OPT_g_Group);
- // and "clang -emit-llvm foo.o -o foo"
- Args.ClaimAllArgs(options::OPT_emit_llvm);
- // and for "clang -w foo.o -o foo". Other warning options are already
- // handled somewhere else.
- Args.ClaimAllArgs(options::OPT_w);
- if (!D.SysRoot.empty())
- CmdArgs.push_back(Args.MakeArgString("--sysroot=" + D.SysRoot));
- if (Args.hasArg(options::OPT_pie))
- CmdArgs.push_back("-pie");
- if (Args.hasArg(options::OPT_rdynamic))
- CmdArgs.push_back("-export-dynamic");
- if (Args.hasArg(options::OPT_shared))
- CmdArgs.push_back("--shared");
- if (Output.isFilename()) {
- CmdArgs.push_back("-o");
- CmdArgs.push_back(Output.getFilename());
- } else {
- assert(Output.isNothing() && "Invalid output.");
- }
- const bool UseLTO = D.isUsingLTO();
- const bool UseJMC =
- Args.hasFlag(options::OPT_fjmc, options::OPT_fno_jmc, false);
- const bool IsPS4 = TC.getTriple().isPS4();
- const bool IsPS5 = TC.getTriple().isPS5();
- assert(IsPS4 || IsPS5);
- auto AddCodeGenFlag = [&](Twine Flag) {
- const char *Prefix = nullptr;
- if (IsPS4 && D.getLTOMode() == LTOK_Thin)
- Prefix = "-lto-thin-debug-options=";
- else if (IsPS4 && D.getLTOMode() == LTOK_Full)
- Prefix = "-lto-debug-options=";
- else if (IsPS5)
- Prefix = "-plugin-opt=";
- else
- llvm_unreachable("new LTO mode?");
- CmdArgs.push_back(Args.MakeArgString(Twine(Prefix) + Flag));
- };
- if (UseLTO) {
- // We default to creating the arange section, but LTO does not. Enable it
- // here.
- AddCodeGenFlag("-generate-arange-section");
- // This tells LTO to perform JustMyCode instrumentation.
- if (UseJMC)
- AddCodeGenFlag("-enable-jmc-instrument");
- if (Arg *A = Args.getLastArg(options::OPT_fcrash_diagnostics_dir))
- AddCodeGenFlag(Twine("-crash-diagnostics-dir=") + A->getValue());
- }
- if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs))
- TC.addSanitizerArgs(Args, CmdArgs, "-l", "");
- Args.AddAllArgs(CmdArgs, options::OPT_L);
- Args.AddAllArgs(CmdArgs, options::OPT_T_Group);
- Args.AddAllArgs(CmdArgs, options::OPT_e);
- Args.AddAllArgs(CmdArgs, options::OPT_s);
- Args.AddAllArgs(CmdArgs, options::OPT_t);
- Args.AddAllArgs(CmdArgs, options::OPT_r);
- if (Args.hasArg(options::OPT_Z_Xlinker__no_demangle))
- CmdArgs.push_back("--no-demangle");
- AddLinkerInputs(TC, Inputs, Args, CmdArgs, JA);
- if (Args.hasArg(options::OPT_pthread)) {
- CmdArgs.push_back("-lpthread");
- }
- if (UseJMC) {
- CmdArgs.push_back("--whole-archive");
- if (IsPS4)
- CmdArgs.push_back("-lSceDbgJmc");
- else
- CmdArgs.push_back("-lSceJmc_nosubmission");
- CmdArgs.push_back("--no-whole-archive");
- }
- if (Args.hasArg(options::OPT_fuse_ld_EQ)) {
- D.Diag(diag::err_drv_unsupported_opt_for_target)
- << "-fuse-ld" << TC.getTriple().str();
- }
- std::string LdName = TC.qualifyPSCmdName(TC.getLinkerBaseName());
- const char *Exec = Args.MakeArgString(TC.GetProgramPath(LdName.c_str()));
- C.addCommand(std::make_unique<Command>(JA, *this,
- ResponseFileSupport::AtFileUTF8(),
- Exec, CmdArgs, Inputs, Output));
- }
- toolchains::PS4PS5Base::PS4PS5Base(const Driver &D, const llvm::Triple &Triple,
- const ArgList &Args, StringRef Platform,
- const char *EnvVar)
- : Generic_ELF(D, Triple, Args) {
- if (Args.hasArg(clang::driver::options::OPT_static))
- D.Diag(clang::diag::err_drv_unsupported_opt_for_target)
- << "-static" << Platform;
- // Determine where to find the PS4/PS5 libraries. We use the EnvVar
- // if it exists; otherwise use the driver's installation path, which
- // should be <SDK_DIR>/host_tools/bin.
- SmallString<512> SDKDir;
- if (const char *EnvValue = getenv(EnvVar)) {
- if (!llvm::sys::fs::exists(EnvValue))
- D.Diag(clang::diag::warn_drv_ps_sdk_dir) << EnvVar << EnvValue;
- SDKDir = EnvValue;
- } else {
- SDKDir = D.Dir;
- llvm::sys::path::append(SDKDir, "/../../");
- }
- // By default, the driver won't report a warning if it can't find the
- // SDK include or lib directories. This behavior could be changed if
- // -Weverything or -Winvalid-or-nonexistent-directory options are passed.
- // If -isysroot was passed, use that as the SDK base path.
- std::string PrefixDir;
- if (const Arg *A = Args.getLastArg(options::OPT_isysroot)) {
- PrefixDir = A->getValue();
- if (!llvm::sys::fs::exists(PrefixDir))
- D.Diag(clang::diag::warn_missing_sysroot) << PrefixDir;
- } else
- PrefixDir = std::string(SDKDir.str());
- SmallString<512> SDKIncludeDir(PrefixDir);
- llvm::sys::path::append(SDKIncludeDir, "target/include");
- if (!Args.hasArg(options::OPT_nostdinc) &&
- !Args.hasArg(options::OPT_nostdlibinc) &&
- !Args.hasArg(options::OPT_isysroot) &&
- !Args.hasArg(options::OPT__sysroot_EQ) &&
- !llvm::sys::fs::exists(SDKIncludeDir)) {
- D.Diag(clang::diag::warn_drv_unable_to_find_directory_expected)
- << Twine(Platform, " system headers").str() << SDKIncludeDir;
- }
- SmallString<512> SDKLibDir(SDKDir);
- llvm::sys::path::append(SDKLibDir, "target/lib");
- if (!Args.hasArg(options::OPT_nostdlib) &&
- !Args.hasArg(options::OPT_nodefaultlibs) &&
- !Args.hasArg(options::OPT__sysroot_EQ) && !Args.hasArg(options::OPT_E) &&
- !Args.hasArg(options::OPT_c) && !Args.hasArg(options::OPT_S) &&
- !Args.hasArg(options::OPT_emit_ast) &&
- !llvm::sys::fs::exists(SDKLibDir)) {
- D.Diag(clang::diag::warn_drv_unable_to_find_directory_expected)
- << Twine(Platform, " system libraries").str() << SDKLibDir;
- return;
- }
- getFilePaths().push_back(std::string(SDKLibDir.str()));
- }
- Tool *toolchains::PS4CPU::buildAssembler() const {
- return new tools::PScpu::Assembler(*this);
- }
- Tool *toolchains::PS5CPU::buildAssembler() const {
- // PS5 does not support an external assembler.
- getDriver().Diag(clang::diag::err_no_external_assembler);
- return nullptr;
- }
- Tool *toolchains::PS4PS5Base::buildLinker() const {
- return new tools::PScpu::Linker(*this);
- }
- SanitizerMask toolchains::PS4PS5Base::getSupportedSanitizers() const {
- SanitizerMask Res = ToolChain::getSupportedSanitizers();
- Res |= SanitizerKind::Address;
- Res |= SanitizerKind::PointerCompare;
- Res |= SanitizerKind::PointerSubtract;
- Res |= SanitizerKind::Vptr;
- return Res;
- }
- SanitizerMask toolchains::PS5CPU::getSupportedSanitizers() const {
- SanitizerMask Res = PS4PS5Base::getSupportedSanitizers();
- Res |= SanitizerKind::Thread;
- return Res;
- }
- void toolchains::PS4PS5Base::addClangTargetOptions(
- const ArgList &DriverArgs, ArgStringList &CC1Args,
- Action::OffloadKind DeviceOffloadingKind) const {
- // PS4/PS5 do not use init arrays.
- if (DriverArgs.hasArg(options::OPT_fuse_init_array)) {
- Arg *A = DriverArgs.getLastArg(options::OPT_fuse_init_array);
- getDriver().Diag(clang::diag::err_drv_unsupported_opt_for_target)
- << A->getAsString(DriverArgs) << getTriple().str();
- }
- CC1Args.push_back("-fno-use-init-array");
- const Arg *A =
- DriverArgs.getLastArg(options::OPT_fvisibility_from_dllstorageclass,
- options::OPT_fno_visibility_from_dllstorageclass);
- if (!A ||
- A->getOption().matches(options::OPT_fvisibility_from_dllstorageclass)) {
- CC1Args.push_back("-fvisibility-from-dllstorageclass");
- if (DriverArgs.hasArg(options::OPT_fvisibility_dllexport_EQ))
- DriverArgs.AddLastArg(CC1Args, options::OPT_fvisibility_dllexport_EQ);
- else
- CC1Args.push_back("-fvisibility-dllexport=protected");
- if (DriverArgs.hasArg(options::OPT_fvisibility_nodllstorageclass_EQ))
- DriverArgs.AddLastArg(CC1Args,
- options::OPT_fvisibility_nodllstorageclass_EQ);
- else
- CC1Args.push_back("-fvisibility-nodllstorageclass=hidden");
- if (DriverArgs.hasArg(options::OPT_fvisibility_externs_dllimport_EQ))
- DriverArgs.AddLastArg(CC1Args,
- options::OPT_fvisibility_externs_dllimport_EQ);
- else
- CC1Args.push_back("-fvisibility-externs-dllimport=default");
- if (DriverArgs.hasArg(
- options::OPT_fvisibility_externs_nodllstorageclass_EQ))
- DriverArgs.AddLastArg(
- CC1Args, options::OPT_fvisibility_externs_nodllstorageclass_EQ);
- else
- CC1Args.push_back("-fvisibility-externs-nodllstorageclass=default");
- }
- }
- // PS4 toolchain.
- toolchains::PS4CPU::PS4CPU(const Driver &D, const llvm::Triple &Triple,
- const llvm::opt::ArgList &Args)
- : PS4PS5Base(D, Triple, Args, "PS4", "SCE_ORBIS_SDK_DIR") {}
- // PS5 toolchain.
- toolchains::PS5CPU::PS5CPU(const Driver &D, const llvm::Triple &Triple,
- const llvm::opt::ArgList &Args)
- : PS4PS5Base(D, Triple, Args, "PS5", "SCE_PROSPERO_SDK_DIR") {}
|