Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / llvm / lib / Target / RISCV / RISCVSubtarget.cpp
blob3e6af1abc5d408ba291195d2c867e88c09147893
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.
11 //===----------------------------------------------------------------------===//
13 #include "RISCVSubtarget.h"
14 #include "GISel/RISCVCallLowering.h"
15 #include "GISel/RISCVLegalizerInfo.h"
16 #include "GISel/RISCVRegisterBankInfo.h"
17 #include "RISCV.h"
18 #include "RISCVFrameLowering.h"
19 #include "RISCVMacroFusion.h"
20 #include "RISCVTargetMachine.h"
21 #include "llvm/MC/TargetRegistry.h"
22 #include "llvm/Support/ErrorHandling.h"
24 using namespace llvm;
26 #define DEBUG_TYPE "riscv-subtarget"
28 #define GET_SUBTARGETINFO_TARGET_DESC
29 #define GET_SUBTARGETINFO_CTOR
30 #include "RISCVGenSubtargetInfo.inc"
32 namespace llvm::RISCVTuneInfoTable {
34 #define GET_RISCVTuneInfoTable_IMPL
35 #include "RISCVGenSearchableTables.inc"
36 } // namespace llvm::RISCVTuneInfoTable
38 static cl::opt<bool> EnableSubRegLiveness("riscv-enable-subreg-liveness",
39 cl::init(true), cl::Hidden);
41 static cl::opt<unsigned> RVVVectorLMULMax(
42 "riscv-v-fixed-length-vector-lmul-max",
43 cl::desc("The maximum LMUL value to use for fixed length vectors. "
44 "Fractional LMUL values are not supported."),
45 cl::init(8), cl::Hidden);
47 static cl::opt<bool> RISCVDisableUsingConstantPoolForLargeInts(
48 "riscv-disable-using-constant-pool-for-large-ints",
49 cl::desc("Disable using constant pool for large integers."),
50 cl::init(false), cl::Hidden);
52 static cl::opt<unsigned> RISCVMaxBuildIntsCost(
53 "riscv-max-build-ints-cost",
54 cl::desc("The maximum cost used for building integers."), cl::init(0),
55 cl::Hidden);
57 static cl::opt<bool> UseAA("riscv-use-aa", cl::init(true),
58 cl::desc("Enable the use of AA during codegen."));
60 void RISCVSubtarget::anchor() {}
62 RISCVSubtarget &
63 RISCVSubtarget::initializeSubtargetDependencies(const Triple &TT, StringRef CPU,
64 StringRef TuneCPU, StringRef FS,
65 StringRef ABIName) {
66 // Determine default and user-specified characteristics
67 bool Is64Bit = TT.isArch64Bit();
68 if (CPU.empty() || CPU == "generic")
69 CPU = Is64Bit ? "generic-rv64" : "generic-rv32";
71 if (TuneCPU.empty())
72 TuneCPU = CPU;
74 TuneInfo = RISCVTuneInfoTable::getRISCVTuneInfo(TuneCPU);
75 // If there is no TuneInfo for this CPU, we fail back to generic.
76 if (!TuneInfo)
77 TuneInfo = RISCVTuneInfoTable::getRISCVTuneInfo("generic");
78 assert(TuneInfo && "TuneInfo shouldn't be nullptr!");
80 ParseSubtargetFeatures(CPU, TuneCPU, FS);
81 TargetABI = RISCVABI::computeTargetABI(TT, getFeatureBits(), ABIName);
82 RISCVFeatures::validate(TT, getFeatureBits());
83 return *this;
86 RISCVSubtarget::RISCVSubtarget(const Triple &TT, StringRef CPU,
87 StringRef TuneCPU, StringRef FS,
88 StringRef ABIName, unsigned RVVVectorBitsMin,
89 unsigned RVVVectorBitsMax,
90 const TargetMachine &TM)
91 : RISCVGenSubtargetInfo(TT, CPU, TuneCPU, FS),
92 RVVVectorBitsMin(RVVVectorBitsMin), RVVVectorBitsMax(RVVVectorBitsMax),
93 FrameLowering(
94 initializeSubtargetDependencies(TT, CPU, TuneCPU, FS, ABIName)),
95 InstrInfo(*this), RegInfo(getHwMode()), TLInfo(TM, *this) {
96 CallLoweringInfo.reset(new RISCVCallLowering(*getTargetLowering()));
97 Legalizer.reset(new RISCVLegalizerInfo(*this));
99 auto *RBI = new RISCVRegisterBankInfo(getHwMode());
100 RegBankInfo.reset(RBI);
101 InstSelector.reset(createRISCVInstructionSelector(
102 *static_cast<const RISCVTargetMachine *>(&TM), *this, *RBI));
105 const CallLowering *RISCVSubtarget::getCallLowering() const {
106 return CallLoweringInfo.get();
109 InstructionSelector *RISCVSubtarget::getInstructionSelector() const {
110 return InstSelector.get();
113 const LegalizerInfo *RISCVSubtarget::getLegalizerInfo() const {
114 return Legalizer.get();
117 const RegisterBankInfo *RISCVSubtarget::getRegBankInfo() const {
118 return RegBankInfo.get();
121 bool RISCVSubtarget::useConstantPoolForLargeInts() const {
122 return !RISCVDisableUsingConstantPoolForLargeInts;
125 unsigned RISCVSubtarget::getMaxBuildIntsCost() const {
126 // Loading integer from constant pool needs two instructions (the reason why
127 // the minimum cost is 2): an address calculation instruction and a load
128 // instruction. Usually, address calculation and instructions used for
129 // building integers (addi, slli, etc.) can be done in one cycle, so here we
130 // set the default cost to (LoadLatency + 1) if no threshold is provided.
131 return RISCVMaxBuildIntsCost == 0
132 ? getSchedModel().LoadLatency + 1
133 : std::max<unsigned>(2, RISCVMaxBuildIntsCost);
136 unsigned RISCVSubtarget::getMaxRVVVectorSizeInBits() const {
137 assert(hasVInstructions() &&
138 "Tried to get vector length without Zve or V extension support!");
140 // ZvlLen specifies the minimum required vlen. The upper bound provided by
141 // riscv-v-vector-bits-max should be no less than it.
142 if (RVVVectorBitsMax != 0 && RVVVectorBitsMax < ZvlLen)
143 report_fatal_error("riscv-v-vector-bits-max specified is lower "
144 "than the Zvl*b limitation");
146 return RVVVectorBitsMax;
149 unsigned RISCVSubtarget::getMinRVVVectorSizeInBits() const {
150 assert(hasVInstructions() &&
151 "Tried to get vector length without Zve or V extension support!");
153 if (RVVVectorBitsMin == -1U)
154 return ZvlLen;
156 // ZvlLen specifies the minimum required vlen. The lower bound provided by
157 // riscv-v-vector-bits-min should be no less than it.
158 if (RVVVectorBitsMin != 0 && RVVVectorBitsMin < ZvlLen)
159 report_fatal_error("riscv-v-vector-bits-min specified is lower "
160 "than the Zvl*b limitation");
162 return RVVVectorBitsMin;
165 unsigned RISCVSubtarget::getMaxLMULForFixedLengthVectors() const {
166 assert(hasVInstructions() &&
167 "Tried to get vector length without Zve or V extension support!");
168 assert(RVVVectorLMULMax <= 8 &&
169 llvm::has_single_bit<uint32_t>(RVVVectorLMULMax) &&
170 "V extension requires a LMUL to be at most 8 and a power of 2!");
171 return llvm::bit_floor(std::clamp<unsigned>(RVVVectorLMULMax, 1, 8));
174 bool RISCVSubtarget::useRVVForFixedLengthVectors() const {
175 return hasVInstructions() && getMinRVVVectorSizeInBits() != 0;
178 bool RISCVSubtarget::enableSubRegLiveness() const {
179 // FIXME: Enable subregister liveness by default for RVV to better handle
180 // LMUL>1 and segment load/store.
181 return EnableSubRegLiveness;
184 void RISCVSubtarget::getPostRAMutations(
185 std::vector<std::unique_ptr<ScheduleDAGMutation>> &Mutations) const {
186 Mutations.push_back(createRISCVMacroFusionDAGMutation());
189 /// Enable use of alias analysis during code generation (during MI
190 /// scheduling, DAGCombine, etc.).
191 bool RISCVSubtarget::useAA() const { return UseAA; }