/src/llvm-project/clang/lib/Driver/ToolChains/Arch/CSKY.cpp
Line | Count | Source (jump to first uncovered line) |
1 | | //===--- CSKY.cpp - CSKY Helpers for Tools --------------------*- C++ -*-===// |
2 | | // |
3 | | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | | // See https://llvm.org/LICENSE.txt for license information. |
5 | | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | | // |
7 | | //===----------------------------------------------------------------------===// |
8 | | |
9 | | #include "CSKY.h" |
10 | | #include "ToolChains/CommonArgs.h" |
11 | | #include "clang/Basic/CharInfo.h" |
12 | | #include "clang/Driver/Driver.h" |
13 | | #include "clang/Driver/DriverDiagnostic.h" |
14 | | #include "clang/Driver/Options.h" |
15 | | #include "llvm/ADT/StringSwitch.h" |
16 | | #include "llvm/Option/ArgList.h" |
17 | | #include "llvm/Support/raw_ostream.h" |
18 | | #include "llvm/TargetParser/CSKYTargetParser.h" |
19 | | #include "llvm/TargetParser/Host.h" |
20 | | #include "llvm/TargetParser/TargetParser.h" |
21 | | |
22 | | using namespace clang::driver; |
23 | | using namespace clang::driver::tools; |
24 | | using namespace clang; |
25 | | using namespace llvm::opt; |
26 | | |
27 | | std::optional<llvm::StringRef> |
28 | | csky::getCSKYArchName(const Driver &D, const ArgList &Args, |
29 | 0 | const llvm::Triple &Triple) { |
30 | 0 | if (const Arg *A = Args.getLastArg(options::OPT_march_EQ)) { |
31 | 0 | llvm::CSKY::ArchKind ArchKind = llvm::CSKY::parseArch(A->getValue()); |
32 | |
|
33 | 0 | if (ArchKind == llvm::CSKY::ArchKind::INVALID) { |
34 | 0 | D.Diag(clang::diag::err_drv_invalid_arch_name) << A->getAsString(Args); |
35 | 0 | return std::nullopt; |
36 | 0 | } |
37 | 0 | return std::optional<llvm::StringRef>(A->getValue()); |
38 | 0 | } |
39 | | |
40 | 0 | if (const Arg *A = Args.getLastArg(clang::driver::options::OPT_mcpu_EQ)) { |
41 | 0 | llvm::CSKY::ArchKind ArchKind = llvm::CSKY::parseCPUArch(A->getValue()); |
42 | 0 | if (ArchKind == llvm::CSKY::ArchKind::INVALID) { |
43 | 0 | D.Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args); |
44 | 0 | return std::nullopt; |
45 | 0 | } |
46 | 0 | return std::optional<llvm::StringRef>(llvm::CSKY::getArchName(ArchKind)); |
47 | 0 | } |
48 | | |
49 | 0 | return std::optional<llvm::StringRef>("ck810"); |
50 | 0 | } |
51 | | |
52 | 0 | csky::FloatABI csky::getCSKYFloatABI(const Driver &D, const ArgList &Args) { |
53 | 0 | csky::FloatABI ABI = FloatABI::Soft; |
54 | 0 | if (Arg *A = |
55 | 0 | Args.getLastArg(options::OPT_msoft_float, options::OPT_mhard_float, |
56 | 0 | options::OPT_mfloat_abi_EQ)) { |
57 | 0 | if (A->getOption().matches(options::OPT_msoft_float)) { |
58 | 0 | ABI = FloatABI::Soft; |
59 | 0 | } else if (A->getOption().matches(options::OPT_mhard_float)) { |
60 | 0 | ABI = FloatABI::Hard; |
61 | 0 | } else { |
62 | 0 | ABI = llvm::StringSwitch<csky::FloatABI>(A->getValue()) |
63 | 0 | .Case("soft", FloatABI::Soft) |
64 | 0 | .Case("softfp", FloatABI::SoftFP) |
65 | 0 | .Case("hard", FloatABI::Hard) |
66 | 0 | .Default(FloatABI::Invalid); |
67 | 0 | if (ABI == FloatABI::Invalid) { |
68 | 0 | D.Diag(diag::err_drv_invalid_mfloat_abi) << A->getAsString(Args); |
69 | 0 | ABI = FloatABI::Soft; |
70 | 0 | } |
71 | 0 | } |
72 | 0 | } |
73 | |
|
74 | 0 | return ABI; |
75 | 0 | } |
76 | | |
77 | | // Handle -mfpu=. |
78 | | static llvm::CSKY::CSKYFPUKind |
79 | | getCSKYFPUFeatures(const Driver &D, const Arg *A, const ArgList &Args, |
80 | 0 | StringRef FPU, std::vector<StringRef> &Features) { |
81 | |
|
82 | 0 | llvm::CSKY::CSKYFPUKind FPUID = |
83 | 0 | llvm::StringSwitch<llvm::CSKY::CSKYFPUKind>(FPU) |
84 | 0 | .Case("auto", llvm::CSKY::FK_AUTO) |
85 | 0 | .Case("fpv2", llvm::CSKY::FK_FPV2) |
86 | 0 | .Case("fpv2_divd", llvm::CSKY::FK_FPV2_DIVD) |
87 | 0 | .Case("fpv2_sf", llvm::CSKY::FK_FPV2_SF) |
88 | 0 | .Case("fpv3", llvm::CSKY::FK_FPV3) |
89 | 0 | .Case("fpv3_hf", llvm::CSKY::FK_FPV3_HF) |
90 | 0 | .Case("fpv3_hsf", llvm::CSKY::FK_FPV3_HSF) |
91 | 0 | .Case("fpv3_sdf", llvm::CSKY::FK_FPV3_SDF) |
92 | 0 | .Default(llvm::CSKY::FK_INVALID); |
93 | 0 | if (FPUID == llvm::CSKY::FK_INVALID) { |
94 | 0 | D.Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args); |
95 | 0 | return llvm::CSKY::FK_INVALID; |
96 | 0 | } |
97 | | |
98 | 0 | auto RemoveTargetFPUFeature = |
99 | 0 | [&Features](ArrayRef<const char *> FPUFeatures) { |
100 | 0 | for (auto FPUFeature : FPUFeatures) { |
101 | 0 | auto it = llvm::find(Features, FPUFeature); |
102 | 0 | if (it != Features.end()) |
103 | 0 | Features.erase(it); |
104 | 0 | } |
105 | 0 | }; |
106 | |
|
107 | 0 | RemoveTargetFPUFeature({"+fpuv2_sf", "+fpuv2_df", "+fdivdu", "+fpuv3_hi", |
108 | 0 | "+fpuv3_hf", "+fpuv3_sf", "+fpuv3_df"}); |
109 | |
|
110 | 0 | if (!llvm::CSKY::getFPUFeatures(FPUID, Features)) { |
111 | 0 | D.Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args); |
112 | 0 | return llvm::CSKY::FK_INVALID; |
113 | 0 | } |
114 | | |
115 | 0 | return FPUID; |
116 | 0 | } |
117 | | |
118 | | void csky::getCSKYTargetFeatures(const Driver &D, const llvm::Triple &Triple, |
119 | | const ArgList &Args, ArgStringList &CmdArgs, |
120 | 0 | std::vector<llvm::StringRef> &Features) { |
121 | 0 | llvm::StringRef archName; |
122 | 0 | llvm::StringRef cpuName; |
123 | 0 | llvm::CSKY::ArchKind ArchKind = llvm::CSKY::ArchKind::INVALID; |
124 | 0 | if (const Arg *A = Args.getLastArg(options::OPT_march_EQ)) { |
125 | 0 | ArchKind = llvm::CSKY::parseArch(A->getValue()); |
126 | 0 | if (ArchKind == llvm::CSKY::ArchKind::INVALID) { |
127 | 0 | D.Diag(clang::diag::err_drv_invalid_arch_name) << A->getAsString(Args); |
128 | 0 | return; |
129 | 0 | } |
130 | 0 | archName = A->getValue(); |
131 | 0 | } |
132 | | |
133 | 0 | if (const Arg *A = Args.getLastArg(clang::driver::options::OPT_mcpu_EQ)) { |
134 | 0 | llvm::CSKY::ArchKind Kind = llvm::CSKY::parseCPUArch(A->getValue()); |
135 | 0 | if (Kind == llvm::CSKY::ArchKind::INVALID) { |
136 | 0 | D.Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args); |
137 | 0 | return; |
138 | 0 | } |
139 | 0 | if (!archName.empty() && Kind != ArchKind) { |
140 | 0 | D.Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args); |
141 | 0 | return; |
142 | 0 | } |
143 | 0 | cpuName = A->getValue(); |
144 | 0 | if (archName.empty()) |
145 | 0 | archName = llvm::CSKY::getArchName(Kind); |
146 | 0 | } |
147 | | |
148 | 0 | if (archName.empty() && cpuName.empty()) { |
149 | 0 | archName = "ck810"; |
150 | 0 | cpuName = "ck810"; |
151 | 0 | } else if (!archName.empty() && cpuName.empty()) { |
152 | 0 | cpuName = archName; |
153 | 0 | } |
154 | |
|
155 | 0 | csky::FloatABI FloatABI = csky::getCSKYFloatABI(D, Args); |
156 | |
|
157 | 0 | if (FloatABI == csky::FloatABI::Hard) { |
158 | 0 | Features.push_back("+hard-float-abi"); |
159 | 0 | Features.push_back("+hard-float"); |
160 | 0 | } else if (FloatABI == csky::FloatABI::SoftFP) { |
161 | 0 | Features.push_back("+hard-float"); |
162 | 0 | } |
163 | |
|
164 | 0 | uint64_t Extension = llvm::CSKY::getDefaultExtensions(cpuName); |
165 | 0 | llvm::CSKY::getExtensionFeatures(Extension, Features); |
166 | |
|
167 | 0 | if (const Arg *FPUArg = Args.getLastArg(options::OPT_mfpu_EQ)) |
168 | 0 | getCSKYFPUFeatures(D, FPUArg, Args, FPUArg->getValue(), Features); |
169 | 0 | } |