blob: 4b455448b380b39bc35fe5d87493c2d96ee789fb [file] [log] [blame]
Alex Bradbury89718422017-10-19 21:37:38 +00001//===-- RISCVISelDAGToDAG.cpp - A dag to dag inst selector for RISCV ------===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// 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
Alex Bradbury89718422017-10-19 21:37:38 +00006//
7//===----------------------------------------------------------------------===//
8//
9// This file defines an instruction selector for the RISCV target.
10//
11//===----------------------------------------------------------------------===//
12
Alex Bradbury89718422017-10-19 21:37:38 +000013#include "MCTargetDesc/RISCVMCTargetDesc.h"
Alex Bradbury2146e8f2018-11-16 10:14:16 +000014#include "RISCV.h"
Alex Bradbury89718422017-10-19 21:37:38 +000015#include "RISCVTargetMachine.h"
Alex Bradbury2146e8f2018-11-16 10:14:16 +000016#include "Utils/RISCVMatInt.h"
Alex Bradbury660bcce2017-12-11 11:53:54 +000017#include "llvm/CodeGen/MachineFrameInfo.h"
Alex Bradbury89718422017-10-19 21:37:38 +000018#include "llvm/CodeGen/SelectionDAGISel.h"
19#include "llvm/Support/Debug.h"
20#include "llvm/Support/MathExtras.h"
21#include "llvm/Support/raw_ostream.h"
22using namespace llvm;
23
24#define DEBUG_TYPE "riscv-isel"
25
26// RISCV-specific code to select RISCV machine instructions for
27// SelectionDAG operations.
28namespace {
29class RISCVDAGToDAGISel final : public SelectionDAGISel {
Alex Bradbury0c7b3642017-11-21 08:23:08 +000030 const RISCVSubtarget *Subtarget;
31
Alex Bradbury89718422017-10-19 21:37:38 +000032public:
33 explicit RISCVDAGToDAGISel(RISCVTargetMachine &TargetMachine)
34 : SelectionDAGISel(TargetMachine) {}
35
36 StringRef getPassName() const override {
37 return "RISCV DAG->DAG Pattern Instruction Selection";
38 }
39
Alex Bradbury0c7b3642017-11-21 08:23:08 +000040 bool runOnMachineFunction(MachineFunction &MF) override {
41 Subtarget = &MF.getSubtarget<RISCVSubtarget>();
42 return SelectionDAGISel::runOnMachineFunction(MF);
43 }
44
Alex Bradbury0171a9f2018-03-19 11:54:28 +000045 void PostprocessISelDAG() override;
46
Alex Bradbury89718422017-10-19 21:37:38 +000047 void Select(SDNode *Node) override;
48
Alex Bradbury9330e642018-01-10 20:05:09 +000049 bool SelectInlineAsmMemoryOperand(const SDValue &Op, unsigned ConstraintID,
50 std::vector<SDValue> &OutOps) override;
51
Alex Bradbury660bcce2017-12-11 11:53:54 +000052 bool SelectAddrFI(SDValue Addr, SDValue &Base);
53
Alex Bradbury89718422017-10-19 21:37:38 +000054// Include the pieces autogenerated from the target description.
55#include "RISCVGenDAGISel.inc"
Alex Bradbury0171a9f2018-03-19 11:54:28 +000056
57private:
58 void doPeepholeLoadStoreADDI();
Alex Bradbury89718422017-10-19 21:37:38 +000059};
60}
61
Alex Bradbury0b4175f2018-04-12 05:34:25 +000062void RISCVDAGToDAGISel::PostprocessISelDAG() {
63 doPeepholeLoadStoreADDI();
Alex Bradbury0b4175f2018-04-12 05:34:25 +000064}
Alex Bradbury0171a9f2018-03-19 11:54:28 +000065
Alex Bradbury2146e8f2018-11-16 10:14:16 +000066static SDNode *selectImm(SelectionDAG *CurDAG, const SDLoc &DL, int64_t Imm,
67 MVT XLenVT) {
68 RISCVMatInt::InstSeq Seq;
69 RISCVMatInt::generateInstSeq(Imm, XLenVT == MVT::i64, Seq);
70
71 SDNode *Result;
72 SDValue SrcReg = CurDAG->getRegister(RISCV::X0, XLenVT);
73 for (RISCVMatInt::Inst &Inst : Seq) {
74 SDValue SDImm = CurDAG->getTargetConstant(Inst.Imm, DL, XLenVT);
75 if (Inst.Opc == RISCV::LUI)
76 Result = CurDAG->getMachineNode(RISCV::LUI, DL, XLenVT, SDImm);
77 else
78 Result = CurDAG->getMachineNode(Inst.Opc, DL, XLenVT, SrcReg, SDImm);
79
80 // Only the first instruction has X0 as its source.
81 SrcReg = SDValue(Result, 0);
82 }
83
84 return Result;
85}
86
Alex Bradburybc96a982018-11-30 09:38:44 +000087// Returns true if the Node is an ISD::AND with a constant argument. If so,
88// set Mask to that constant value.
89static bool isConstantMask(SDNode *Node, uint64_t &Mask) {
90 if (Node->getOpcode() == ISD::AND &&
91 Node->getOperand(1).getOpcode() == ISD::Constant) {
92 Mask = cast<ConstantSDNode>(Node->getOperand(1))->getZExtValue();
93 return true;
94 }
95 return false;
96}
97
Alex Bradbury89718422017-10-19 21:37:38 +000098void RISCVDAGToDAGISel::Select(SDNode *Node) {
Alex Bradburyd33ffe92018-10-03 13:13:13 +000099 // If we have a custom node, we have already selected.
Alex Bradbury89718422017-10-19 21:37:38 +0000100 if (Node->isMachineOpcode()) {
Nicola Zaghend34e60c2018-05-14 12:53:11 +0000101 LLVM_DEBUG(dbgs() << "== "; Node->dump(CurDAG); dbgs() << "\n");
Alex Bradbury89718422017-10-19 21:37:38 +0000102 Node->setNodeId(-1);
103 return;
104 }
105
Alex Bradbury0c7b3642017-11-21 08:23:08 +0000106 // Instruction Selection not handled by the auto-generated tablegen selection
107 // should be handled here.
Alex Bradburyd33ffe92018-10-03 13:13:13 +0000108 unsigned Opcode = Node->getOpcode();
109 MVT XLenVT = Subtarget->getXLenVT();
110 SDLoc DL(Node);
Alex Bradbury0c7b3642017-11-21 08:23:08 +0000111 EVT VT = Node->getValueType(0);
Alex Bradburyd33ffe92018-10-03 13:13:13 +0000112
113 switch (Opcode) {
114 case ISD::Constant: {
115 auto ConstNode = cast<ConstantSDNode>(Node);
116 if (VT == XLenVT && ConstNode->isNullValue()) {
Alex Bradbury9caefe32017-11-21 12:00:19 +0000117 SDValue New = CurDAG->getCopyFromReg(CurDAG->getEntryNode(), SDLoc(Node),
118 RISCV::X0, XLenVT);
119 ReplaceNode(Node, New.getNode());
120 return;
Alex Bradbury0c7b3642017-11-21 08:23:08 +0000121 }
Alex Bradbury2146e8f2018-11-16 10:14:16 +0000122 int64_t Imm = ConstNode->getSExtValue();
123 if (XLenVT == MVT::i64) {
124 ReplaceNode(Node, selectImm(CurDAG, SDLoc(Node), Imm, XLenVT));
125 return;
126 }
Alex Bradburyd33ffe92018-10-03 13:13:13 +0000127 break;
Alex Bradbury0c7b3642017-11-21 08:23:08 +0000128 }
Alex Bradburyd33ffe92018-10-03 13:13:13 +0000129 case ISD::FrameIndex: {
Alex Bradbury660bcce2017-12-11 11:53:54 +0000130 SDValue Imm = CurDAG->getTargetConstant(0, DL, XLenVT);
Craig Topper781aa182018-05-05 01:57:00 +0000131 int FI = cast<FrameIndexSDNode>(Node)->getIndex();
Alex Bradbury660bcce2017-12-11 11:53:54 +0000132 SDValue TFI = CurDAG->getTargetFrameIndex(FI, VT);
133 ReplaceNode(Node, CurDAG->getMachineNode(RISCV::ADDI, DL, VT, TFI, Imm));
134 return;
135 }
Alex Bradburybc96a982018-11-30 09:38:44 +0000136 case ISD::SRL: {
137 if (!Subtarget->is64Bit())
138 break;
139 SDValue Op0 = Node->getOperand(0);
140 SDValue Op1 = Node->getOperand(1);
141 uint64_t Mask;
142 // Match (srl (and val, mask), imm) where the result would be a
143 // zero-extended 32-bit integer. i.e. the mask is 0xffffffff or the result
144 // is equivalent to this (SimplifyDemandedBits may have removed lower bits
145 // from the mask that aren't necessary due to the right-shifting).
146 if (Op1.getOpcode() == ISD::Constant &&
147 isConstantMask(Op0.getNode(), Mask)) {
148 uint64_t ShAmt = cast<ConstantSDNode>(Op1.getNode())->getZExtValue();
149
150 if ((Mask | maskTrailingOnes<uint64_t>(ShAmt)) == 0xffffffff) {
151 SDValue ShAmtVal =
152 CurDAG->getTargetConstant(ShAmt, SDLoc(Node), XLenVT);
153 CurDAG->SelectNodeTo(Node, RISCV::SRLIW, XLenVT, Op0.getOperand(0),
154 ShAmtVal);
155 return;
156 }
157 }
Alex Bradbury1b9cd442019-01-22 07:22:00 +0000158 break;
Alex Bradburybc96a982018-11-30 09:38:44 +0000159 }
Alex Bradburyd33ffe92018-10-03 13:13:13 +0000160 }
Alex Bradbury0c7b3642017-11-21 08:23:08 +0000161
Alex Bradbury89718422017-10-19 21:37:38 +0000162 // Select the default instruction.
163 SelectCode(Node);
164}
165
Alex Bradbury9330e642018-01-10 20:05:09 +0000166bool RISCVDAGToDAGISel::SelectInlineAsmMemoryOperand(
167 const SDValue &Op, unsigned ConstraintID, std::vector<SDValue> &OutOps) {
168 switch (ConstraintID) {
169 case InlineAsm::Constraint_i:
170 case InlineAsm::Constraint_m:
171 // We just support simple memory operands that have a single address
172 // operand and need no special handling.
173 OutOps.push_back(Op);
174 return false;
175 default:
176 break;
177 }
178
179 return true;
180}
181
Alex Bradbury660bcce2017-12-11 11:53:54 +0000182bool RISCVDAGToDAGISel::SelectAddrFI(SDValue Addr, SDValue &Base) {
183 if (auto FIN = dyn_cast<FrameIndexSDNode>(Addr)) {
184 Base = CurDAG->getTargetFrameIndex(FIN->getIndex(), Subtarget->getXLenVT());
185 return true;
186 }
187 return false;
188}
189
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000190// Merge an ADDI into the offset of a load/store instruction where possible.
191// (load (add base, off), 0) -> (load base, off)
192// (store val, (add base, off)) -> (store val, base, off)
193void RISCVDAGToDAGISel::doPeepholeLoadStoreADDI() {
194 SelectionDAG::allnodes_iterator Position(CurDAG->getRoot().getNode());
195 ++Position;
196
197 while (Position != CurDAG->allnodes_begin()) {
198 SDNode *N = &*--Position;
199 // Skip dead nodes and any non-machine opcodes.
200 if (N->use_empty() || !N->isMachineOpcode())
201 continue;
202
203 int OffsetOpIdx;
204 int BaseOpIdx;
205
206 // Only attempt this optimisation for I-type loads and S-type stores.
207 switch (N->getMachineOpcode()) {
208 default:
209 continue;
210 case RISCV::LB:
211 case RISCV::LH:
212 case RISCV::LW:
213 case RISCV::LBU:
214 case RISCV::LHU:
215 case RISCV::LWU:
216 case RISCV::LD:
217 case RISCV::FLW:
218 case RISCV::FLD:
219 BaseOpIdx = 0;
220 OffsetOpIdx = 1;
221 break;
222 case RISCV::SB:
223 case RISCV::SH:
224 case RISCV::SW:
225 case RISCV::SD:
226 case RISCV::FSW:
227 case RISCV::FSD:
228 BaseOpIdx = 1;
229 OffsetOpIdx = 2;
230 break;
231 }
232
233 // Currently, the load/store offset must be 0 to be considered for this
234 // peephole optimisation.
235 if (!isa<ConstantSDNode>(N->getOperand(OffsetOpIdx)) ||
236 N->getConstantOperandVal(OffsetOpIdx) != 0)
237 continue;
238
239 SDValue Base = N->getOperand(BaseOpIdx);
240
Alex Bradbury099c7202018-04-18 19:02:31 +0000241 // If the base is an ADDI, we can merge it in to the load/store.
242 if (!Base.isMachineOpcode() || Base.getMachineOpcode() != RISCV::ADDI)
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000243 continue;
244
Alex Bradbury099c7202018-04-18 19:02:31 +0000245 SDValue ImmOperand = Base.getOperand(1);
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000246
Alex Bradbury099c7202018-04-18 19:02:31 +0000247 if (auto Const = dyn_cast<ConstantSDNode>(ImmOperand)) {
248 ImmOperand = CurDAG->getTargetConstant(
249 Const->getSExtValue(), SDLoc(ImmOperand), ImmOperand.getValueType());
250 } else if (auto GA = dyn_cast<GlobalAddressSDNode>(ImmOperand)) {
251 ImmOperand = CurDAG->getTargetGlobalAddress(
252 GA->getGlobal(), SDLoc(ImmOperand), ImmOperand.getValueType(),
253 GA->getOffset(), GA->getTargetFlags());
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000254 } else {
Alex Bradbury099c7202018-04-18 19:02:31 +0000255 continue;
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000256 }
257
Nicola Zaghend34e60c2018-05-14 12:53:11 +0000258 LLVM_DEBUG(dbgs() << "Folding add-immediate into mem-op:\nBase: ");
259 LLVM_DEBUG(Base->dump(CurDAG));
260 LLVM_DEBUG(dbgs() << "\nN: ");
261 LLVM_DEBUG(N->dump(CurDAG));
262 LLVM_DEBUG(dbgs() << "\n");
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000263
264 // Modify the offset operand of the load/store.
265 if (BaseOpIdx == 0) // Load
Alex Bradbury099c7202018-04-18 19:02:31 +0000266 CurDAG->UpdateNodeOperands(N, Base.getOperand(0), ImmOperand,
267 N->getOperand(2));
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000268 else // Store
Alex Bradbury099c7202018-04-18 19:02:31 +0000269 CurDAG->UpdateNodeOperands(N, N->getOperand(0), Base.getOperand(0),
270 ImmOperand, N->getOperand(3));
271
272 // The add-immediate may now be dead, in which case remove it.
273 if (Base.getNode()->use_empty())
274 CurDAG->RemoveDeadNode(Base.getNode());
Alex Bradbury0171a9f2018-03-19 11:54:28 +0000275 }
276}
277
Alex Bradbury89718422017-10-19 21:37:38 +0000278// This pass converts a legalized DAG into a RISCV-specific DAG, ready
279// for instruction scheduling.
280FunctionPass *llvm::createRISCVISelDag(RISCVTargetMachine &TM) {
281 return new RISCVDAGToDAGISel(TM);
282}