2017-10-20 05:37:38 +08:00
|
|
|
//===-- RISCVISelLowering.h - RISCV DAG Lowering Interface ------*- C++ -*-===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2017-10-20 05:37:38 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines the interfaces that RISCV uses to lower LLVM code into a
|
|
|
|
// selection DAG.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_LIB_TARGET_RISCV_RISCVISELLOWERING_H
|
|
|
|
#define LLVM_LIB_TARGET_RISCV_RISCVISELLOWERING_H
|
|
|
|
|
|
|
|
#include "RISCV.h"
|
|
|
|
#include "llvm/CodeGen/SelectionDAG.h"
|
2017-11-17 09:07:10 +08:00
|
|
|
#include "llvm/CodeGen/TargetLowering.h"
|
2017-10-20 05:37:38 +08:00
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
class RISCVSubtarget;
|
|
|
|
namespace RISCVISD {
|
|
|
|
enum NodeType : unsigned {
|
|
|
|
FIRST_NUMBER = ISD::BUILTIN_OP_END,
|
2017-11-08 21:41:21 +08:00
|
|
|
RET_FLAG,
|
[RISCV] Add support for _interrupt attribute
- Save/restore only registers that are used.
This includes Callee saved registers and Caller saved registers
(arguments and temporaries) for integer and FP registers.
- If there is a call in the interrupt handler, save/restore all
Caller saved registers (arguments and temporaries) and all FP registers.
- Emit special return instructions depending on "interrupt"
attribute type.
Based on initial patch by Zhaoshi Zheng.
Reviewers: asb
Reviewed By: asb
Subscribers: rkruppe, the_o, MartinMosbeck, brucehoult, rbar, johnrusso, simoncook, sabuasal, niosHD, kito-cheng, shiva0217, zzheng, edward-jones, mgrang, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D48411
llvm-svn: 338047
2018-07-27 01:49:43 +08:00
|
|
|
URET_FLAG,
|
|
|
|
SRET_FLAG,
|
|
|
|
MRET_FLAG,
|
2017-11-21 15:51:32 +08:00
|
|
|
CALL,
|
2018-04-12 13:34:25 +08:00
|
|
|
SELECT_CC,
|
|
|
|
BuildPairF64,
|
2018-05-24 06:44:08 +08:00
|
|
|
SplitF64,
|
2019-01-25 13:04:00 +08:00
|
|
|
TAIL,
|
|
|
|
// RV64I shifts, directly matching the semantics of the named RISC-V
|
|
|
|
// instructions.
|
|
|
|
SLLW,
|
|
|
|
SRAW,
|
2019-01-25 13:11:34 +08:00
|
|
|
SRLW,
|
|
|
|
// 32-bit operations from RV64M that can't be simply matched with a pattern
|
|
|
|
// at instruction selection time.
|
|
|
|
DIVW,
|
|
|
|
DIVUW,
|
2019-02-01 06:48:38 +08:00
|
|
|
REMUW,
|
|
|
|
// FPR32<->GPR transfer operations for RV64. Needed as an i32<->f32 bitcast
|
|
|
|
// is not legal on RV64. FMV_W_X_RV64 matches the semantics of the FMV.W.X.
|
|
|
|
// FMV_X_ANYEXTW_RV64 is similar to FMV.X.W but has an any-extended result.
|
|
|
|
// This is a more convenient semantic for producing dagcombines that remove
|
|
|
|
// unnecessary GPR->FPR->GPR moves.
|
|
|
|
FMV_W_X_RV64,
|
[RISCV] Support @llvm.readcyclecounter() Intrinsic
On RISC-V, the `cycle` CSR holds a 64-bit count of the number of clock
cycles executed by the core, from an arbitrary point in the past. This
matches the intended semantics of `@llvm.readcyclecounter()`, which we
currently leave to the default lowering (to the constant 0).
With this patch, we will now correctly lower this intrinsic to the
intended semantics, using the user-space instruction `rdcycle`. On
64-bit targets, we can directly lower to this instruction.
On 32-bit targets, we need to do more, as `rdcycle` only returns the low
32-bits of the `cycle` CSR. In this case, we perform a custom lowering,
based on the PowerPC lowering, using `rdcycleh` to obtain the high
32-bits of the `cycle` CSR. This custom lowering inserts a new basic
block which detects overflow in the high 32-bits of the `cycle` CSR
during reading (because multiple instructions are required to read). The
emitted assembly matches the suggested assembly in the RISC-V
specification.
Differential Revision: https://reviews.llvm.org/D64125
llvm-svn: 365201
2019-07-05 20:35:21 +08:00
|
|
|
FMV_X_ANYEXTW_RV64,
|
|
|
|
// READ_CYCLE_WIDE - A read of the 64-bit cycle CSR on a 32-bit target
|
|
|
|
// (returns (Lo, Hi)). It takes a chain operand.
|
|
|
|
READ_CYCLE_WIDE
|
2017-10-20 05:37:38 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
class RISCVTargetLowering : public TargetLowering {
|
|
|
|
const RISCVSubtarget &Subtarget;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit RISCVTargetLowering(const TargetMachine &TM,
|
|
|
|
const RISCVSubtarget &STI);
|
|
|
|
|
2018-09-19 18:54:22 +08:00
|
|
|
bool getTgtMemIntrinsic(IntrinsicInfo &Info, const CallInst &I,
|
|
|
|
MachineFunction &MF,
|
|
|
|
unsigned Intrinsic) const override;
|
2018-04-26 20:13:48 +08:00
|
|
|
bool isLegalAddressingMode(const DataLayout &DL, const AddrMode &AM, Type *Ty,
|
|
|
|
unsigned AS,
|
|
|
|
Instruction *I = nullptr) const override;
|
2018-04-26 21:15:17 +08:00
|
|
|
bool isLegalICmpImmediate(int64_t Imm) const override;
|
2018-04-26 21:00:37 +08:00
|
|
|
bool isLegalAddImmediate(int64_t Imm) const override;
|
2018-04-26 21:37:00 +08:00
|
|
|
bool isTruncateFree(Type *SrcTy, Type *DstTy) const override;
|
|
|
|
bool isTruncateFree(EVT SrcVT, EVT DstVT) const override;
|
2018-04-26 22:04:18 +08:00
|
|
|
bool isZExtFree(SDValue Val, EVT VT2) const override;
|
2018-11-30 17:56:54 +08:00
|
|
|
bool isSExtCheaperThanZExt(EVT SrcVT, EVT DstVT) const override;
|
[RISCV] Support Bit-Preserving FP in F/D Extensions
Summary:
This allows some integer bitwise operations to instead be performed by
hardware fp instructions. This is correct because the RISC-V spec
requires the F and D extensions to use the IEEE-754 standard
representation, and fp register loads and stores to be bit-preserving.
This is tested against the soft-float ABI, but with hardware float
extensions enabled, so that the tests also ensure the optimisation also
fires in this case.
Reviewers: asb, luismarques
Reviewed By: asb
Subscribers: hiraditya, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, kito-cheng, shiva0217, jrtc27, zzheng, edward-jones, rogfer01, MartinMosbeck, brucehoult, the_o, rkruppe, PkmX, jocewei, psnobl, benna, Jim, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62900
llvm-svn: 362790
2019-06-07 20:20:14 +08:00
|
|
|
|
|
|
|
bool hasBitPreservingFPLogic(EVT VT) const override;
|
2018-04-26 20:13:48 +08:00
|
|
|
|
2017-10-20 05:37:38 +08:00
|
|
|
// Provide custom lowering hooks for some operations.
|
|
|
|
SDValue LowerOperation(SDValue Op, SelectionDAG &DAG) const override;
|
2019-01-25 13:04:00 +08:00
|
|
|
void ReplaceNodeResults(SDNode *N, SmallVectorImpl<SDValue> &Results,
|
|
|
|
SelectionDAG &DAG) const override;
|
2017-10-20 05:37:38 +08:00
|
|
|
|
2018-10-04 07:30:16 +08:00
|
|
|
SDValue PerformDAGCombine(SDNode *N, DAGCombinerInfo &DCI) const override;
|
|
|
|
|
2019-01-25 13:04:00 +08:00
|
|
|
unsigned ComputeNumSignBitsForTargetNode(SDValue Op,
|
|
|
|
const APInt &DemandedElts,
|
|
|
|
const SelectionDAG &DAG,
|
|
|
|
unsigned Depth) const override;
|
|
|
|
|
2017-10-20 05:37:38 +08:00
|
|
|
// This method returns the name of a target specific DAG node.
|
|
|
|
const char *getTargetNodeName(unsigned Opcode) const override;
|
2018-01-11 04:05:09 +08:00
|
|
|
|
[RISCV] Support 'f' Inline Assembly Constraint
Summary:
This adds the 'f' inline assembly constraint, as supported by GCC. An
'f'-constrained operand is passed in a floating point register. Exactly
which kind of floating-point register (32-bit or 64-bit) is decided
based on the operand type and the available standard extensions (-f and
-d, respectively).
This patch adds support in both the clang frontend, and LLVM itself.
Reviewers: asb, lewis-revill
Reviewed By: asb
Subscribers: hiraditya, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, kito-cheng, shiva0217, jrtc27, MaskRay, zzheng, edward-jones, rogfer01, MartinMosbeck, brucehoult, the_o, rkruppe, PkmX, jocewei, psnobl, benna, Jim, s.egerton, cfe-commits, llvm-commits
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D65500
llvm-svn: 367403
2019-07-31 17:45:55 +08:00
|
|
|
ConstraintType getConstraintType(StringRef Constraint) const override;
|
2019-08-16 18:28:34 +08:00
|
|
|
|
|
|
|
unsigned getInlineAsmMemConstraint(StringRef ConstraintCode) const override;
|
|
|
|
|
2018-01-11 04:05:09 +08:00
|
|
|
std::pair<unsigned, const TargetRegisterClass *>
|
|
|
|
getRegForInlineAsmConstraint(const TargetRegisterInfo *TRI,
|
|
|
|
StringRef Constraint, MVT VT) const override;
|
2017-10-20 05:37:38 +08:00
|
|
|
|
2019-06-11 20:42:13 +08:00
|
|
|
void LowerAsmOperandForConstraint(SDValue Op, std::string &Constraint,
|
|
|
|
std::vector<SDValue> &Ops,
|
|
|
|
SelectionDAG &DAG) const override;
|
|
|
|
|
2017-11-21 15:51:32 +08:00
|
|
|
MachineBasicBlock *
|
|
|
|
EmitInstrWithCustomInserter(MachineInstr &MI,
|
|
|
|
MachineBasicBlock *BB) const override;
|
2018-02-02 10:43:18 +08:00
|
|
|
|
|
|
|
EVT getSetCCResultType(const DataLayout &DL, LLVMContext &Context,
|
|
|
|
EVT VT) const override;
|
2017-11-21 15:51:32 +08:00
|
|
|
|
2019-03-22 18:39:22 +08:00
|
|
|
bool convertSetCCLogicToBitwiseLogic(EVT VT) const override {
|
|
|
|
return VT.isScalarInteger();
|
|
|
|
}
|
|
|
|
|
2018-06-13 20:04:51 +08:00
|
|
|
bool shouldInsertFencesForAtomic(const Instruction *I) const override {
|
|
|
|
return isa<LoadInst>(I) || isa<StoreInst>(I);
|
|
|
|
}
|
|
|
|
Instruction *emitLeadingFence(IRBuilder<> &Builder, Instruction *Inst,
|
|
|
|
AtomicOrdering Ord) const override;
|
|
|
|
Instruction *emitTrailingFence(IRBuilder<> &Builder, Instruction *Inst,
|
|
|
|
AtomicOrdering Ord) const override;
|
|
|
|
|
2019-03-12 05:41:22 +08:00
|
|
|
ISD::NodeType getExtendForAtomicOps() const override {
|
|
|
|
return ISD::SIGN_EXTEND;
|
|
|
|
}
|
|
|
|
|
2019-04-16 22:38:32 +08:00
|
|
|
bool shouldExpandShift(SelectionDAG &DAG, SDNode *N) const override {
|
|
|
|
if (DAG.getMachineFunction().getFunction().hasMinSize())
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
[RISCV] Prevent re-ordering some adds after shifts
Summary:
DAGCombine will normally turn a `(shl (add x, c1), c2)` into `(add (shl x, c2), c1 << c2)`, where `c1` and `c2` are constants. This can be prevented by a callback in TargetLowering.
On RISC-V, materialising the constant `c1 << c2` can be more expensive than materialising `c1`, because materialising the former may take more instructions, and may use a register, where materialising the latter would not.
This patch implements the hook in RISCVTargetLowering to prevent this transform, in the cases where:
- `c1` fits into the immediate field in an `addi` instruction.
- `c1` takes fewer instructions to materialise than `c1 << c2`.
In future, DAGCombine could do the check to see whether `c1` fits into an add immediate, which might simplify more targets hooks than just RISC-V.
Reviewers: asb, luismarques, efriedma
Reviewed By: asb
Subscribers: xbolva00, lebedev.ri, craig.topper, lewis-revill, Jim, hiraditya, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, kito-cheng, shiva0217, jrtc27, zzheng, edward-jones, rogfer01, MartinMosbeck, brucehoult, the_o, rkruppe, PkmX, jocewei, psnobl, benna, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62857
llvm-svn: 363736
2019-06-19 04:38:08 +08:00
|
|
|
bool isDesirableToCommuteWithShift(const SDNode *N,
|
|
|
|
CombineLevel Level) const override;
|
2019-04-16 22:38:32 +08:00
|
|
|
|
2019-07-08 17:16:47 +08:00
|
|
|
/// If a physical register, this returns the register that receives the
|
|
|
|
/// exception address on entry to an EH pad.
|
|
|
|
unsigned
|
|
|
|
getExceptionPointerRegister(const Constant *PersonalityFn) const override;
|
|
|
|
|
|
|
|
/// If a physical register, this returns the register that receives the
|
|
|
|
/// exception typeid on entry to a landing pad.
|
|
|
|
unsigned
|
|
|
|
getExceptionSelectorRegister(const Constant *PersonalityFn) const override;
|
|
|
|
|
2019-08-29 07:40:37 +08:00
|
|
|
bool shouldExtendTypeInLibCall(EVT Type) const override;
|
|
|
|
|
2017-10-20 05:37:38 +08:00
|
|
|
private:
|
2017-12-11 20:49:02 +08:00
|
|
|
void analyzeInputArgs(MachineFunction &MF, CCState &CCInfo,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
|
|
bool IsRet) const;
|
|
|
|
void analyzeOutputArgs(MachineFunction &MF, CCState &CCInfo,
|
|
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs,
|
2018-01-11 03:41:03 +08:00
|
|
|
bool IsRet, CallLoweringInfo *CLI) const;
|
2017-10-20 05:37:38 +08:00
|
|
|
// Lower incoming arguments, copy physregs into vregs
|
|
|
|
SDValue LowerFormalArguments(SDValue Chain, CallingConv::ID CallConv,
|
|
|
|
bool IsVarArg,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
|
|
const SDLoc &DL, SelectionDAG &DAG,
|
|
|
|
SmallVectorImpl<SDValue> &InVals) const override;
|
2017-12-11 20:49:02 +08:00
|
|
|
bool CanLowerReturn(CallingConv::ID CallConv, MachineFunction &MF,
|
|
|
|
bool IsVarArg,
|
|
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
|
|
LLVMContext &Context) const override;
|
2017-10-20 05:37:38 +08:00
|
|
|
SDValue LowerReturn(SDValue Chain, CallingConv::ID CallConv, bool IsVarArg,
|
|
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
|
|
const SmallVectorImpl<SDValue> &OutVals, const SDLoc &DL,
|
|
|
|
SelectionDAG &DAG) const override;
|
2017-11-08 21:41:21 +08:00
|
|
|
SDValue LowerCall(TargetLowering::CallLoweringInfo &CLI,
|
|
|
|
SmallVectorImpl<SDValue> &InVals) const override;
|
2017-10-20 05:37:38 +08:00
|
|
|
bool shouldConvertConstantLoadToIntImm(const APInt &Imm,
|
|
|
|
Type *Ty) const override {
|
|
|
|
return true;
|
|
|
|
}
|
2019-04-01 22:42:56 +08:00
|
|
|
|
|
|
|
template <class NodeTy>
|
2019-06-11 20:57:47 +08:00
|
|
|
SDValue getAddr(NodeTy *N, SelectionDAG &DAG, bool IsLocal = true) const;
|
2019-04-01 22:42:56 +08:00
|
|
|
|
2019-06-19 16:40:59 +08:00
|
|
|
SDValue getStaticTLSAddr(GlobalAddressSDNode *N, SelectionDAG &DAG,
|
|
|
|
bool UseGOT) const;
|
|
|
|
SDValue getDynamicTLSAddr(GlobalAddressSDNode *N, SelectionDAG &DAG) const;
|
|
|
|
|
2019-06-17 18:54:12 +08:00
|
|
|
bool shouldConsiderGEPOffsetSplit() const override { return true; }
|
2017-11-08 21:24:21 +08:00
|
|
|
SDValue lowerGlobalAddress(SDValue Op, SelectionDAG &DAG) const;
|
[RISCV] Support and tests for a variety of additional LLVM IR constructs
Previous patches primarily ensured that codegen was possible for the standard
RISC-V instructions. However, there are a number of IR inputs that wouldn't be
appropriately lowered. This patch both adds test cases and supports lowering
for a number of these cases:
* Improved sext/zext/trunc support
* Support for setcc variants that don't map directly to RISC-V instructions
* Lowering mul, and hence support for external symbols
* addc, adde, subc, sube
* mulhs, srem, mulhu, urem, udiv, sdiv
* {srl,sra,shl}_parts
* brind
* br_jt
* bswap, ctlz, cttz, ctpop
* rotl, rotr
* BlockAddress operands
Differential Revision: https://reviews.llvm.org/D29938
llvm-svn: 318737
2017-11-21 16:11:03 +08:00
|
|
|
SDValue lowerBlockAddress(SDValue Op, SelectionDAG &DAG) const;
|
2018-03-20 21:26:12 +08:00
|
|
|
SDValue lowerConstantPool(SDValue Op, SelectionDAG &DAG) const;
|
2019-06-19 16:40:59 +08:00
|
|
|
SDValue lowerGlobalTLSAddress(SDValue Op, SelectionDAG &DAG) const;
|
2017-11-21 15:51:32 +08:00
|
|
|
SDValue lowerSELECT(SDValue Op, SelectionDAG &DAG) const;
|
2018-01-11 03:41:03 +08:00
|
|
|
SDValue lowerVASTART(SDValue Op, SelectionDAG &DAG) const;
|
2018-10-04 13:27:50 +08:00
|
|
|
SDValue lowerFRAMEADDR(SDValue Op, SelectionDAG &DAG) const;
|
|
|
|
SDValue lowerRETURNADDR(SDValue Op, SelectionDAG &DAG) const;
|
2019-04-16 22:38:32 +08:00
|
|
|
SDValue lowerShiftLeftParts(SDValue Op, SelectionDAG &DAG) const;
|
|
|
|
SDValue lowerShiftRightParts(SDValue Op, SelectionDAG &DAG, bool IsSRA) const;
|
2018-05-24 06:44:08 +08:00
|
|
|
|
2019-02-21 22:31:41 +08:00
|
|
|
bool isEligibleForTailCallOptimization(
|
|
|
|
CCState &CCInfo, CallLoweringInfo &CLI, MachineFunction &MF,
|
|
|
|
const SmallVector<CCValAssign, 16> &ArgLocs) const;
|
2018-09-19 18:54:22 +08:00
|
|
|
|
|
|
|
TargetLowering::AtomicExpansionKind
|
|
|
|
shouldExpandAtomicRMWInIR(AtomicRMWInst *AI) const override;
|
|
|
|
virtual Value *emitMaskedAtomicRMWIntrinsic(
|
|
|
|
IRBuilder<> &Builder, AtomicRMWInst *AI, Value *AlignedAddr, Value *Incr,
|
|
|
|
Value *Mask, Value *ShiftAmt, AtomicOrdering Ord) const override;
|
2018-11-30 04:43:42 +08:00
|
|
|
TargetLowering::AtomicExpansionKind
|
|
|
|
shouldExpandAtomicCmpXchgInIR(AtomicCmpXchgInst *CI) const override;
|
|
|
|
virtual Value *
|
|
|
|
emitMaskedAtomicCmpXchgIntrinsic(IRBuilder<> &Builder, AtomicCmpXchgInst *CI,
|
|
|
|
Value *AlignedAddr, Value *CmpVal,
|
|
|
|
Value *NewVal, Value *Mask,
|
|
|
|
AtomicOrdering Ord) const override;
|
2019-10-23 04:25:01 +08:00
|
|
|
|
|
|
|
/// Generate error diagnostics if any register used by CC has been marked
|
|
|
|
/// reserved.
|
|
|
|
void validateCCReservedRegs(
|
|
|
|
const SmallVectorImpl<std::pair<llvm::Register, llvm::SDValue>> &Regs,
|
|
|
|
MachineFunction &MF) const;
|
2017-10-20 05:37:38 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|