LLVM 23.0.0git
RISCVSubtarget.cpp
Go to the documentation of this file.
1//===-- RISCVSubtarget.cpp - RISC-V Subtarget Information -----------------===//
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// This file implements the RISC-V specific subclass of TargetSubtargetInfo.
10//
11//===----------------------------------------------------------------------===//
12
13#include "RISCVSubtarget.h"
17#include "RISCV.h"
18#include "RISCVFrameLowering.h"
20#include "RISCVTargetMachine.h"
24
25using namespace llvm;
26
27#define DEBUG_TYPE "riscv-subtarget"
28
29#define GET_SUBTARGETINFO_TARGET_DESC
30#define GET_SUBTARGETINFO_CTOR
31#include "RISCVGenSubtargetInfo.inc"
32
33#define GET_RISCV_MACRO_FUSION_PRED_IMPL
34#include "RISCVGenMacroFusion.inc"
35
37
38#define GET_RISCVTuneInfoTable_IMPL
39#include "RISCVGenSearchableTables.inc"
40} // namespace llvm::RISCVTuneInfoTable
41
43 "riscv-v-fixed-length-vector-lmul-max",
44 cl::desc("The maximum LMUL value to use for fixed length vectors. "
45 "Fractional LMUL values are not supported."),
47
49 "riscv-disable-using-constant-pool-for-large-ints",
50 cl::desc("Disable using constant pool for large integers."),
51 cl::init(false), cl::Hidden);
52
54 "riscv-max-build-ints-cost",
55 cl::desc("The maximum cost used for building integers."), cl::init(0),
57
58static cl::opt<bool> UseAA("riscv-use-aa", cl::init(true),
59 cl::desc("Enable the use of AA during codegen."));
60
62 "riscv-min-jump-table-entries", cl::Hidden,
63 cl::desc("Set minimum number of entries to use a jump table on RISCV"));
64
66 "use-riscv-mips-load-store-pairs",
67 cl::desc("Enable the load/store pair optimization pass"), cl::init(false),
69
70static cl::opt<bool> UseMIPSCCMovInsn("use-riscv-mips-ccmov",
71 cl::desc("Use 'mips.ccmov' instruction"),
72 cl::init(true), cl::Hidden);
73
74void RISCVSubtarget::anchor() {}
75
77RISCVSubtarget::initializeSubtargetDependencies(const Triple &TT, StringRef CPU,
78 StringRef TuneCPU, StringRef FS,
79 StringRef ABIName) {
80 // Determine default and user-specified characteristics
81 bool Is64Bit = TT.isArch64Bit();
82 if (CPU.empty() || CPU == "generic")
83 CPU = Is64Bit ? "generic-rv64" : "generic-rv32";
84
85 if (TuneCPU.empty())
86 TuneCPU = CPU;
87 if (TuneCPU == "generic")
88 TuneCPU = Is64Bit ? "generic-rv64" : "generic-rv32";
89
90 TuneInfo = RISCVTuneInfoTable::getRISCVTuneInfo(TuneCPU);
91 // If there is no TuneInfo for this CPU, we fail back to generic.
92 if (!TuneInfo)
93 TuneInfo = RISCVTuneInfoTable::getRISCVTuneInfo("generic");
94 assert(TuneInfo && "TuneInfo shouldn't be nullptr!");
95
96 ParseSubtargetFeatures(CPU, TuneCPU, FS);
97
99
100 // Re-sync the flags.
101 HasStdExtZcd = hasFeature(RISCV::FeatureStdExtZcd);
102 HasStdExtZcf = hasFeature(RISCV::FeatureStdExtZcf);
103 HasStdExtC = hasFeature(RISCV::FeatureStdExtC);
104 HasStdExtZce = hasFeature(RISCV::FeatureStdExtZce);
105
106 TargetABI = RISCVABI::computeTargetABI(TT, getFeatureBits(), ABIName);
107 RISCVFeatures::validate(TT, getFeatureBits());
108 return *this;
109}
110
112 StringRef TuneCPU, StringRef FS,
113 StringRef ABIName, unsigned RVVVectorBitsMin,
114 unsigned RVVVectorBitsMax,
115 const TargetMachine &TM)
116 : RISCVGenSubtargetInfo(TT, CPU, TuneCPU, FS),
117 IsLittleEndian(TT.isLittleEndian()), RVVVectorBitsMin(RVVVectorBitsMin),
118 RVVVectorBitsMax(RVVVectorBitsMax),
119 FrameLowering(
120 initializeSubtargetDependencies(TT, CPU, TuneCPU, FS, ABIName)),
121 InstrInfo(*this), TLInfo(TM, *this) {
122 TSInfo = std::make_unique<RISCVSelectionDAGInfo>();
123}
124
126
130
137
143
145 if (!InstSelector) {
147 *static_cast<const RISCVTargetMachine *>(&TLInfo.getTargetMachine()),
148 *this, *getRegBankInfo()));
149 }
150 return InstSelector.get();
151}
152
154 if (!Legalizer)
155 Legalizer.reset(new RISCVLegalizerInfo(*this));
156 return Legalizer.get();
157}
158
160 if (!RegBankInfo)
161 RegBankInfo.reset(new RISCVRegisterBankInfo(getHwMode()));
162 return RegBankInfo.get();
163}
164
168
169// Returns true if VT is a P extension packed SIMD type.
171 if (!HasStdExtP)
172 return false;
173
174 // RV32 supports 32-bit and 64-bit vectors. RV64 only support 64-bit vectors.
175 if (!is64Bit() && (VT == MVT::v4i8 || VT == MVT::v2i16))
176 return true;
177
178 return VT == MVT::v8i8 || VT == MVT::v4i16 || VT == MVT::v2i32;
179}
180
181// Returns true if VT is a P extension packed double-wide SIMD type.
183 if (!HasStdExtP || is64Bit())
184 return false;
185
186 return VT == MVT::v8i8 || VT == MVT::v4i16 || VT == MVT::v2i32;
187}
188
190 // Loading integer from constant pool needs two instructions (the reason why
191 // the minimum cost is 2): an address calculation instruction and a load
192 // instruction. Usually, address calculation and instructions used for
193 // building integers (addi, slli, etc.) can be done in one cycle, so here we
194 // set the default cost to (LoadLatency + 1) if no threshold is provided.
195 return RISCVMaxBuildIntsCost == 0
196 ? getSchedModel().LoadLatency + 1
197 : std::max<unsigned>(2, RISCVMaxBuildIntsCost);
198}
199
202 "Tried to get vector length without Zve or V extension support!");
203
204 // ZvlLen specifies the minimum required vlen. The upper bound provided by
205 // riscv-v-vector-bits-max should be no less than it.
206 if (RVVVectorBitsMax != 0 && RVVVectorBitsMax < ZvlLen)
207 report_fatal_error("riscv-v-vector-bits-max specified is lower "
208 "than the Zvl*b limitation");
209
210 return RVVVectorBitsMax;
211}
212
215 "Tried to get vector length without Zve or V extension support!");
216
217 if (RVVVectorBitsMin == -1U)
218 return ZvlLen;
219
220 // ZvlLen specifies the minimum required vlen. The lower bound provided by
221 // riscv-v-vector-bits-min should be no less than it.
222 if (RVVVectorBitsMin != 0 && RVVVectorBitsMin < ZvlLen)
223 report_fatal_error("riscv-v-vector-bits-min specified is lower "
224 "than the Zvl*b limitation");
225
226 return RVVVectorBitsMin;
227}
228
231 "Tried to get vector length without Zve or V extension support!");
234 "V extension requires a LMUL to be at most 8 and a power of 2!");
235 return llvm::bit_floor(std::clamp<unsigned>(RVVVectorLMULMax, 1, 8));
236}
237
242
243bool RISCVSubtarget::enableSubRegLiveness() const { return true; }
244
246 return getSchedModel().hasInstrSchedModel();
247}
248
250 // We usually compute max call frame size after ISel. Do the computation now
251 // if the .mir file didn't specify it. Note that this will probably give you
252 // bogus values after PEI has eliminated the callframe setup/destroy pseudo
253 // instructions, specify explicitly if you need it to be correct.
254 MachineFrameInfo &MFI = MF.getFrameInfo();
257}
258
259 /// Enable use of alias analysis during code generation (during MI
260 /// scheduling, DAGCombine, etc.).
261bool RISCVSubtarget::useAA() const { return UseAA; }
262
264 return RISCVMinimumJumpTableEntries.getNumOccurrences() > 0
266 : TuneInfo->MinimumJumpTableEntries;
267}
268
270 const SchedRegion &Region) const {
271 // Do bidirectional scheduling since it provides a more balanced scheduling
272 // leading to better performance. This will increase compile time.
273 Policy.OnlyTopDown = false;
274 Policy.OnlyBottomUp = false;
275
276 // Disabling the latency heuristic can reduce the number of spills/reloads but
277 // will cause some regressions on some cores.
278 Policy.DisableLatencyHeuristic = DisableLatencySchedHeuristic;
279
280 // Spilling is generally expensive on all RISC-V cores, so always enable
281 // register-pressure tracking. This will increase compile time.
282 Policy.ShouldTrackPressure = true;
283}
284
286 MachineSchedPolicy &Policy, const SchedRegion &Region) const {
287 MISched::Direction PostRASchedDirection = getPostRASchedDirection();
288 if (PostRASchedDirection == MISched::TopDown) {
289 Policy.OnlyTopDown = true;
290 Policy.OnlyBottomUp = false;
291 } else if (PostRASchedDirection == MISched::BottomUp) {
292 Policy.OnlyTopDown = false;
293 Policy.OnlyBottomUp = true;
294 } else if (PostRASchedDirection == MISched::Bidirectional) {
295 Policy.OnlyTopDown = false;
296 Policy.OnlyBottomUp = false;
297 }
298}
299
301 return UseMIPSLoadStorePairsOpt && HasVendorXMIPSLSP;
302}
303
305 return UseMIPSCCMovInsn && HasVendorXMIPSCMov;
306}
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static cl::opt< bool > UseAA("aarch64-use-aa", cl::init(true), cl::desc("Enable the use of AA during codegen."))
This file describes how to lower LLVM calls to machine code calls.
This file describes how to lower LLVM inline asm to machine code INLINEASM.
This file declares the targeting of the Machinelegalizer class for RISC-V.
static cl::opt< unsigned > RVVVectorLMULMax("riscv-v-fixed-length-vector-lmul-max", cl::desc("The maximum LMUL value to use for fixed length vectors. " "Fractional LMUL values are not supported."), cl::init(8), cl::Hidden)
static cl::opt< bool > UseAA("riscv-use-aa", cl::init(true), cl::desc("Enable the use of AA during codegen."))
static cl::opt< bool > UseMIPSCCMovInsn("use-riscv-mips-ccmov", cl::desc("Use 'mips.ccmov' instruction"), cl::init(true), cl::Hidden)
static cl::opt< unsigned > RISCVMinimumJumpTableEntries("riscv-min-jump-table-entries", cl::Hidden, cl::desc("Set minimum number of entries to use a jump table on RISCV"))
static cl::opt< bool > UseMIPSLoadStorePairsOpt("use-riscv-mips-load-store-pairs", cl::desc("Enable the load/store pair optimization pass"), cl::init(false), cl::Hidden)
static cl::opt< bool > RISCVDisableUsingConstantPoolForLargeInts("riscv-disable-using-constant-pool-for-large-ints", cl::desc("Disable using constant pool for large integers."), cl::init(false), cl::Hidden)
static cl::opt< unsigned > RISCVMaxBuildIntsCost("riscv-max-build-ints-cost", cl::desc("The maximum cost used for building integers."), cl::init(0), cl::Hidden)
Machine Value Type.
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
LLVM_ABI void computeMaxCallFrameSize(MachineFunction &MF, std::vector< MachineBasicBlock::iterator > *FrameSDOps=nullptr)
Computes the maximum size of a callframe.
bool isMaxCallFrameSizeComputed() const
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
This class provides the information for the target register banks.
unsigned getMinimumJumpTableEntries() const
const LegalizerInfo * getLegalizerInfo() const override
void overrideSchedPolicy(MachineSchedPolicy &Policy, const SchedRegion &Region) const override
std::unique_ptr< LegalizerInfo > Legalizer
unsigned getMaxLMULForFixedLengthVectors() const
bool isPExtPackedDoubleType(MVT VT) const
bool useMIPSLoadStorePairs() const
const InlineAsmLowering * getInlineAsmLowering() const override
bool useRVVForFixedLengthVectors() const
MISched::Direction getPostRASchedDirection() const
std::unique_ptr< InlineAsmLowering > InlineAsmLoweringInfo
bool isPExtPackedType(MVT VT) const
unsigned getMinRVVVectorSizeInBits() const
std::unique_ptr< InstructionSelector > InstSelector
RISCVSubtarget(const Triple &TT, StringRef CPU, StringRef TuneCPU, StringRef FS, StringRef ABIName, unsigned RVVVectorBitsMin, unsigned RVVVectorLMULMax, const TargetMachine &TM)
bool useMIPSCCMovInsn() const
const RISCVRegisterBankInfo * getRegBankInfo() const override
const CallLowering * getCallLowering() const override
InstructionSelector * getInstructionSelector() const override
unsigned getMaxBuildIntsCost() const
std::unique_ptr< const SelectionDAGTargetInfo > TSInfo
bool hasVInstructions() const
bool useAA() const override
Enable use of alias analysis during code generation (during MI scheduling, DAGCombine,...
bool enableMachinePipeliner() const override
bool useConstantPoolForLargeInts() const
bool isLittleEndian() const
~RISCVSubtarget() override
unsigned getMaxRVVVectorSizeInBits() const
void ParseSubtargetFeatures(StringRef CPU, StringRef TuneCPU, StringRef FS)
std::unique_ptr< RISCVRegisterBankInfo > RegBankInfo
void mirFileLoaded(MachineFunction &MF) const override
std::unique_ptr< CallLowering > CallLoweringInfo
const RISCVTargetLowering * getTargetLowering() const override
void overridePostRASchedPolicy(MachineSchedPolicy &Policy, const SchedRegion &Region) const override
bool enableSubRegLiveness() const override
const SelectionDAGTargetInfo * getSelectionDAGInfo() const override
Targets can subclass this to parameterize the SelectionDAG lowering and instruction selection process...
Represent a constant reference to a string, i.e.
Definition StringRef.h:56
constexpr bool empty() const
Check if the string is empty.
Definition StringRef.h:141
Primary interface to the complete machine description for the target machine.
Triple - Helper class for working with autoconf configuration names.
Definition Triple.h:47
ABI computeTargetABI(const Triple &TT, const FeatureBitset &FeatureBits, StringRef ABIName)
void validate(const Triple &TT, const FeatureBitset &FeatureBits)
void updateCZceFeatureImplications(MCSubtargetInfo &STI)
static constexpr unsigned RVVBitsPerBlock
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
InstructionSelector * createRISCVInstructionSelector(const RISCVTargetMachine &TM, const RISCVSubtarget &Subtarget, const RISCVRegisterBankInfo &RBI)
constexpr bool has_single_bit(T Value) noexcept
Definition bit.h:149
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
Definition Error.cpp:163
T bit_floor(T Value)
Returns the largest integral power of two no greater than Value if Value is nonzero.
Definition bit.h:347
Define a generic scheduling policy for targets that don't provide their own MachineSchedStrategy.
A region of an MBB for scheduling.