15#include "llvm/Option/ArgList.h"
16#include "llvm/Support/Error.h"
17#include "llvm/TargetParser/Host.h"
18#include "llvm/TargetParser/RISCVISAInfo.h"
19#include "llvm/TargetParser/RISCVTargetParser.h"
28 std::vector<StringRef> &Features,
29 const ArgList &Args) {
30 bool EnableExperimentalExtensions =
31 Args.hasArg(options::OPT_menable_experimental_extensions);
33 llvm::RISCVISAInfo::parseArchString(
Arch, EnableExperimentalExtensions);
35 handleAllErrors(ISAInfo.takeError(), [&](llvm::StringError &ErrMsg) {
36 D.Diag(diag::err_drv_invalid_riscv_arch_name)
37 << Arch << ErrMsg.getMessage();
43 for (
const std::string &Str : (*ISAInfo)->toFeatures(
true,
45 Features.push_back(Args.MakeArgString(Str));
47 if (EnableExperimentalExtensions)
48 Features.push_back(Args.MakeArgString(
"+experimental"));
54 const llvm::Triple &Triple, StringRef Mcpu) {
55 bool Is64Bit = Triple.isRISCV64();
56 if (!llvm::RISCV::parseCPU(Mcpu, Is64Bit)) {
58 if (llvm::RISCV::parseCPU(Mcpu, !Is64Bit))
59 D.
Diag(clang::diag::err_drv_invalid_riscv_cpu_name_for_target)
62 D.
Diag(clang::diag::err_drv_unsupported_option_argument)
63 << A->getSpelling() << Mcpu;
71 std::vector<StringRef> &Features) {
77 bool CPUFastScalarUnaligned =
false;
78 bool CPUFastVectorUnaligned =
false;
82 if (Arg *A = Args.getLastArg(options::OPT_mcpu_EQ)) {
83 StringRef
CPU = A->getValue();
85 CPU = llvm::sys::getHostCPUName();
90 if (llvm::RISCV::hasFastScalarUnalignedAccess(CPU))
91 CPUFastScalarUnaligned =
true;
92 if (llvm::RISCV::hasFastVectorUnalignedAccess(CPU))
93 CPUFastVectorUnaligned =
true;
97#define RESERVE_REG(REG) \
98 if (Args.hasArg(options::OPT_ffixed_##REG)) \
99 Features.push_back("+reserve-" #REG);
134 if (Args.hasFlag(options::OPT_mrelax, options::OPT_mno_relax,
true))
135 Features.push_back(
"+relax");
137 Features.push_back(
"-relax");
143 if (
const Arg *A = Args.getLastArg(
144 options::OPT_mno_strict_align, options::OPT_mscalar_strict_align,
145 options::OPT_mstrict_align, options::OPT_mno_scalar_strict_align)) {
146 if (A->getOption().matches(options::OPT_mno_strict_align) ||
147 A->getOption().matches(options::OPT_mno_scalar_strict_align)) {
148 Features.push_back(
"+unaligned-scalar-mem");
150 Features.push_back(
"-unaligned-scalar-mem");
152 }
else if (CPUFastScalarUnaligned || Triple.isAndroid()) {
153 Features.push_back(
"+unaligned-scalar-mem");
160 if (
const Arg *A = Args.getLastArg(
161 options::OPT_mno_strict_align, options::OPT_mvector_strict_align,
162 options::OPT_mstrict_align, options::OPT_mno_vector_strict_align)) {
163 if (A->getOption().matches(options::OPT_mno_strict_align) ||
164 A->getOption().matches(options::OPT_mno_vector_strict_align)) {
165 Features.push_back(
"+unaligned-vector-mem");
167 Features.push_back(
"-unaligned-vector-mem");
169 }
else if (CPUFastVectorUnaligned || Triple.isAndroid()) {
170 Features.push_back(
"+unaligned-vector-mem");
173 if (Triple.isRISCV32()) {
180 if (
const Arg *A = Args.getLastArg(
181 options::OPT_mstrict_align, options::OPT_mscalar_strict_align,
182 options::OPT_mzilsd_word_align, options::OPT_mno_strict_align,
183 options::OPT_mno_scalar_strict_align,
184 options::OPT_mzilsd_strict_align)) {
185 if (A->getOption().matches(options::OPT_mno_strict_align) ||
186 A->getOption().matches(options::OPT_mno_scalar_strict_align) ||
187 A->getOption().matches(options::OPT_mzilsd_word_align)) {
188 Features.push_back(
"+zilsd-word-align");
190 Features.push_back(
"-zilsd-word-align");
195 if (
const Arg *A = Args.getLastArg(options::OPT_mzilsd_word_align,
196 options::OPT_mzilsd_strict_align)) {
197 D.
Diag(clang::diag::err_drv_unsupported_opt_for_target)
198 << A->getSpelling() << Triple.getTriple();
205 for (
const std::string &TF : TuneFeatures)
206 Features.push_back(Args.MakeArgString(TF));
211 options::OPT_m_riscv_Features_Group);
215 assert(Triple.isRISCV() &&
"Unexpected triple");
238 if (
const Arg *A = Args.getLastArg(options::OPT_mabi_EQ))
239 return A->getValue();
251 auto ParseResult = llvm::RISCVISAInfo::parseArchString(
254 if (!llvm::errorToBool(ParseResult.takeError()))
255 return (*ParseResult)->computeDefaultABI();
262 if (Triple.isRISCV32()) {
263 if (Triple.getOS() == llvm::Triple::UnknownOS)
268 if (Triple.getOS() == llvm::Triple::UnknownOS)
276 const llvm::Triple &Triple) {
277 assert(Triple.isRISCV() &&
"Unexpected triple");
306 if (
const Arg *A = Args.getLastArg(options::OPT_march_EQ)) {
307 StringRef MArch = A->getValue();
308 if (MArch !=
"unset")
313 if (
const Arg *A = Args.getLastArg(options::OPT_mcpu_EQ)) {
314 StringRef CPU = A->getValue();
315 if (CPU ==
"native") {
316 CPU = llvm::sys::getHostCPUName();
318 if (CPU.starts_with(
"generic")) {
319 auto FeatureMap = llvm::sys::getHostCPUFeatures();
321 if (!FeatureMap.empty()) {
322 std::vector<std::string> Features;
323 for (
auto &F : FeatureMap)
324 Features.push_back(((F.second ?
"+" :
"-") + F.first()).str());
325 auto ParseResult = llvm::RISCVISAInfo::parseFeatures(
326 Triple.isRISCV32() ? 32 : 64, Features);
328 return (*ParseResult)->toString();
333 StringRef MArch = llvm::RISCV::getMArchFromMcpu(CPU);
345 if (
const Arg *A = Args.getLastArg(options::OPT_mabi_EQ)) {
346 StringRef MABI = A->getValue();
348 if (MABI.equals_insensitive(
"ilp32e"))
350 if (MABI.equals_insensitive(
"lp64e"))
352 if (MABI.starts_with_insensitive(
"ilp32"))
354 if (MABI.starts_with_insensitive(
"lp64")) {
355 if (Triple.isAndroid())
356 return "rv64imafdcv_zba_zbb_zbs";
357 if (Triple.isOSFuchsia())
368 if (Triple.isRISCV32()) {
369 if (Triple.getOS() == llvm::Triple::UnknownOS)
374 if (Triple.getOS() == llvm::Triple::UnknownOS)
376 if (Triple.isAndroid())
377 return "rv64imafdcv_zba_zbb_zbs";
378 if (Triple.isOSFuchsia())
384 const llvm::Triple &Triple) {
387 if (
const Arg *A = Args.getLastArg(options::OPT_mcpu_EQ))
392 CPU = llvm::sys::getHostCPUName();
397 return Triple.isRISCV64() ?
"generic-rv64" :
"generic-rv32";
400std::optional<StringRef>
403 const Arg *MTuneArg = Args.getLastArg(options::OPT_mtune_EQ);
407 StringRef MTune = MTuneArg->getValue();
409 auto [TuneCPU, TFString] = MTune.split(
':');
410 if (!Args.hasFlag(options::OPT_mexperimental_mtune_syntax,
411 options::OPT_mno_experimental_mtune_syntax,
false)) {
412 if (!TFString.empty()) {
416 D.
Diag(diag::err_drv_invalid_riscv_mtune_string)
418 <<
"require '-mexperimental-mtune-syntax' to use with tune feature "
425 if (!TuneFeatures || TFString.empty())
427 if (
auto E = llvm::RISCV::parseTuneFeatureString(TuneCPU, TFString,
429 D.
Diag(diag::err_drv_invalid_riscv_mtune_string)
430 << 1 << TFString << llvm::toString(std::move(E));
Driver - Encapsulate logic for constructing compilation processes from a set of gcc-driver-like comma...
DiagnosticBuilder Diag(unsigned DiagID) const
The JSON file list parser is used to communicate input to InstallAPI.