Index: docs/LangRef.rst =================================================================== --- docs/LangRef.rst +++ docs/LangRef.rst @@ -11606,6 +11606,230 @@ %r2 = call float @llvm.fmuladd.f32(float %a, float %b, float %c) ; yields float:r2 = (a * b) + c + +Vector Reduction Intrinsics +--------------------------- + +Horizontal reductions of vectors can be expressed using the following +intrinsics. Each one take a vector operand as an input and applies its +respective operation across all elements of the vector, returning a single +scalar result of the same element type. + +Not every target may currently support reductions expressed in these forms, +if the vector width is known and a power-of-two then then a shuffle sequence may +be used instead. + +'``llvm.vector.reduce.add.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.add..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.add.*``' intrinsics do an integer ``ADD`` reduction of a +vector, returning the result as a scalar. The return type matches the +element-type of the vector input. + +'``llvm.vector.reduce.fadd.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare float @llvm.vector.reduce.fadd.f32.( %a) + declare double @llvm.vector.reduce.fadd.f64.( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.fadd.*``' intrinsics do a floating point ``ADD`` +reduction of a vector, returning the result as a scalar. The return type matches +the element-type of the vector input. + +'``llvm.vector.reduce.mul.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.mul..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.mul.*``' intrinsics do an integer ``MUL`` reduction of a +vector, returning the result as a scalar. The return type matches the +element-type of the vector input. + +'``llvm.vector.reduce.and.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.and..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.and.*``' intrinsics do a bitwise ``AND`` reduction +of a vector, returning the result as a scalar. The return type matches the +element-type of the vector input. + +'``llvm.vector.reduce.or.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.or..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.or.*``' intrinsics do a bitwise ``OR`` reduction +of a vector, returning the result as a scalar. The return type matches the +element-type of the vector input. + +'``llvm.vector.reduce.eor.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.eor..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.eor.*``' intrinsics do a bitwise ``EOR`` reduction +of a vector, returning the result as a scalar. The return type matches the +element-type of the vector input. + +'``llvm.vector.reduce.smax.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.smax..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.smax.*``' intrinsics do a signed integer ``max`` +reduction of a vector, returning the result as a scalar. The return type matches +the element-type of the vector input. + +'``llvm.vector.reduce.smin.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.smin..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.smin.*``' intrinsics do a signed integer ``min`` +reduction of a vector, returning the result as a scalar. The return type matches +the element-type of the vector input. + +'``llvm.vector.reduce.umax.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.umax..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.umax.*``' intrinsics do an unsigned integer ``max`` +reduction of a vector, returning the result as a scalar. The return type matches the +element-type of the vector input. + +'``llvm.vector.reduce.umin.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare @llvm.vector.reduce.umin..( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.umin.*``' intrinsics do an unsigned integer ``min`` +reduction of a vector, returning the result as a scalar. The return type matches +the element-type of the vector input. + +'``llvm.vector.reduce.fmax.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare float @llvm.vector.reduce.fmax.f32.( %a) + declare double @llvm.vector.reduce.fmax.f64.( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.fmax.*``' intrinsics do a floating point ``max`` +reduction of a vector, returning the result as a scalar. The second operand is +an ``i32`` constant value which is ``1`` if it can be assumed that there are no +NaN values in the vector, and ``0`` otherwise. The return type matches the +element-type of the vector input. + +'``llvm.vector.reduce.fmin.*``' Intrinsic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Syntax: +""""""" + +:: + + declare float @llvm.vector.reduce.fmin.f32.( %a) + declare double @llvm.vector.reduce.fmin.f64.( %a) + +Overview: +""""""""" + +The '``llvm.vector.reduce.fmin.*``' intrinsics do a floating point ``min`` +reduction of a vector, returning the result as a scalar. The second operand is +an ``i32`` constant value which is ``1`` if it can be assumed that there are no +NaN values in the vector, and ``0`` otherwise. The return type matches the +element-type of the vector input. + Half Precision Floating Point Intrinsics ---------------------------------------- Index: include/llvm/Analysis/TargetTransformInfo.h =================================================================== --- include/llvm/Analysis/TargetTransformInfo.h +++ include/llvm/Analysis/TargetTransformInfo.h @@ -710,6 +710,11 @@ unsigned ChainSizeInBytes, VectorType *VecTy) const; + /// \returns True if the target wants to handle the given reduction idiom in + /// the intrinsics form instead of the shuffle form. + bool useReductionIntrinsic(unsigned Opcode, Type *Ty, bool IsMaxOp = false, + bool IsSigned = false, bool NoNaN = false) const; + /// @} private: @@ -860,6 +865,8 @@ virtual unsigned getStoreVectorFactor(unsigned VF, unsigned StoreSize, unsigned ChainSizeInBytes, VectorType *VecTy) const = 0; + virtual bool useReductionIntrinsic(unsigned Opcode, Type *Ty, bool IsMaxOp, + bool IsSigned, bool NoNaN) const = 0; }; template @@ -1151,6 +1158,10 @@ VectorType *VecTy) const override { return Impl.getStoreVectorFactor(VF, StoreSize, ChainSizeInBytes, VecTy); } + bool useReductionIntrinsic(unsigned Opcode, Type *Ty, bool IsMaxOp, + bool IsSigned, bool NoNaN) const override { + return Impl.useReductionIntrinsic(Opcode, Ty, IsMaxOp, IsSigned, NoNaN); + } }; template Index: include/llvm/Analysis/TargetTransformInfoImpl.h =================================================================== --- include/llvm/Analysis/TargetTransformInfoImpl.h +++ include/llvm/Analysis/TargetTransformInfoImpl.h @@ -438,6 +438,12 @@ VectorType *VecTy) const { return VF; } + + bool useReductionIntrinsic(unsigned Opcode, Type *Ty, bool IsMaxOp, + bool IsSigned, bool NoNaN) const { + return false; + } + protected: // Obtain the minimum required size to hold the value (without the sign) // In case of a vector it returns the min required size for one element. Index: include/llvm/CodeGen/ISDOpcodes.h =================================================================== --- include/llvm/CodeGen/ISDOpcodes.h +++ include/llvm/CodeGen/ISDOpcodes.h @@ -764,6 +764,13 @@ /// known nonzero constant. The only operand here is the chain. GET_DYNAMIC_AREA_OFFSET, + /// Generic reduction nodes. + VECREDUCE_FADD, VECREDUCE_FMUL, + VECREDUCE_ADD, VECREDUCE_MUL, + VECREDUCE_AND, VECREDUCE_OR, VECREDUCE_XOR, + VECREDUCE_SMAX, VECREDUCE_SMIN, VECREDUCE_UMAX, VECREDUCE_UMIN, + VECREDUCE_FMAX, VECREDUCE_FMIN, + /// BUILTIN_OP_END - This must be the last enum value in this list. /// The target-specific pre-isel opcode values start here. BUILTIN_OP_END Index: include/llvm/IR/IRBuilder.h =================================================================== --- include/llvm/IR/IRBuilder.h +++ include/llvm/IR/IRBuilder.h @@ -453,6 +453,43 @@ MDNode *ScopeTag = nullptr, MDNode *NoAliasTag = nullptr); + /// \brief Create a vector fadd reduction intrinsic of the source vector. + CallInst *CreateFAddReduce(Value *Src); + + /// \brief Create a vector fmul reduction intrinsic of the source vector. + CallInst *CreateFMulReduce(Value *Src); + + /// \brief Create a vector int add reduction intrinsic of the source vector. + CallInst *CreateAddReduce(Value *Src); + + /// \brief Create a vector int mul reduction intrinsic of the source vector. + CallInst *CreateMulReduce(Value *Src); + + /// \brief Create a vector int AND reduction intrinsic of the source vector. + CallInst *CreateAndReduce(Value *Src); + + /// \brief Create a vector int OR reduction intrinsic of the source vector. + CallInst *CreateOrReduce(Value *Src); + + /// \brief Create a vector int XOR reduction intrinsic of the source vector. + CallInst *CreateXorReduce(Value *Src); + + /// \brief Create a vector integer max reduction intrinsic of the source + /// vector. + CallInst *CreateIntMaxReduce(Value *Src, bool IsSigned = false); + + /// \brief Create a vector integer min reduction intrinsic of the source + /// vector. + CallInst *CreateIntMinReduce(Value *Src, bool IsSigned = false); + + /// \brief Create a vector float max reduction intrinsic of the source + /// vector. + CallInst *CreateFPMaxReduce(Value *Src, bool NoNaN = false); + + /// \brief Create a vector float min reduction intrinsic of the source + /// vector. + CallInst *CreateFPMinReduce(Value *Src, bool NoNaN = false); + /// \brief Create a lifetime.start intrinsic. /// /// If the pointer isn't i8* it will be converted. Index: include/llvm/IR/Intrinsics.td =================================================================== --- include/llvm/IR/Intrinsics.td +++ include/llvm/IR/Intrinsics.td @@ -781,6 +781,37 @@ [IntrArgMemOnly, NoCapture<0>, NoCapture<1>, WriteOnly<0>, ReadOnly<1>]>; +//===------------------------ Reduction Intrinsics ------------------------===// +// +def int_vector_reduce_fadd : Intrinsic<[llvm_anyfloat_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_fmul : Intrinsic<[llvm_anyfloat_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_add : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_mul : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_and : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_or : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_xor : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_smax : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_smin : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_umax : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_umin : Intrinsic<[llvm_anyint_ty], [llvm_anyvector_ty], + [IntrNoMem]>; +def int_vector_reduce_fmax : Intrinsic<[llvm_anyfloat_ty], + [llvm_anyvector_ty, llvm_i32_ty], + [IntrNoMem]>; +def int_vector_reduce_fmin : Intrinsic<[llvm_anyfloat_ty], + [llvm_anyvector_ty, llvm_i32_ty], + [IntrNoMem]>; + //===----- Intrinsics that are used to provide predicate information -----===// def int_ssa_copy : Intrinsic<[llvm_any_ty], [LLVMMatchType<0>], Index: include/llvm/Transforms/Utils/LoopUtils.h =================================================================== --- include/llvm/Transforms/Utils/LoopUtils.h +++ include/llvm/Transforms/Utils/LoopUtils.h @@ -42,6 +42,7 @@ class ScalarEvolution; class SCEV; class TargetLibraryInfo; +class TargetTransformInfo; /// \brief Captures loop safety information. /// It keep information for loop & its header may throw exception. @@ -489,6 +490,18 @@ LoopSafetyInfo *SafetyInfo, OptimizationRemarkEmitter *ORE = nullptr); +/// Create a target reduction of the given vector. The reduction must be simple, +/// that is, it must not be complex like a minmax reduction. +Value *createTargetReduction(IRBuilder<> &B, const TargetTransformInfo *TTI, + unsigned Opcode, Value *Src); + +/// Create a generic target reduction. This queries the target to determine if +/// it wants the given reduction type as an intrinsic or a log2 shuffle IR +/// pattern. +Value *createTargetReduction(IRBuilder<> &B, + const TargetTransformInfo *TTI, + RecurrenceDescriptor &Desc, Value *Src, + bool InOrder = false, bool NoNaN = false); } // end namespace llvm #endif // LLVM_TRANSFORMS_UTILS_LOOPUTILS_H Index: lib/Analysis/TargetTransformInfo.cpp =================================================================== --- lib/Analysis/TargetTransformInfo.cpp +++ lib/Analysis/TargetTransformInfo.cpp @@ -477,6 +477,12 @@ return TTIImpl->getStoreVectorFactor(VF, StoreSize, ChainSizeInBytes, VecTy); } +bool TargetTransformInfo::useReductionIntrinsic(unsigned Opcode, Type *Ty, + bool IsMaxOp, bool IsSigned, + bool NoNaN) const { + return TTIImpl->useReductionIntrinsic(Opcode, Ty, IsMaxOp, IsSigned, NoNaN); +} + TargetTransformInfo::Concept::~Concept() {} TargetIRAnalysis::TargetIRAnalysis() : TTICallback(&getDefaultTTI) {} Index: lib/Analysis/VectorUtils.cpp =================================================================== --- lib/Analysis/VectorUtils.cpp +++ lib/Analysis/VectorUtils.cpp @@ -23,6 +23,7 @@ #include "llvm/IR/PatternMatch.h" #include "llvm/IR/Value.h" #include "llvm/IR/Constants.h" +#include "llvm/IR/IRBuilder.h" using namespace llvm; using namespace llvm::PatternMatch; Index: lib/CodeGen/SelectionDAG/LegalizeTypes.h =================================================================== --- lib/CodeGen/SelectionDAG/LegalizeTypes.h +++ lib/CodeGen/SelectionDAG/LegalizeTypes.h @@ -671,6 +671,7 @@ // Vector Operand Splitting: <128 x ty> -> 2 x <64 x ty>. bool SplitVectorOperand(SDNode *N, unsigned OpNo); SDValue SplitVecOp_VSELECT(SDNode *N, unsigned OpNo); + SDValue SplitVecOp_VECREDUCE(SDNode *N, unsigned OpNo); SDValue SplitVecOp_UnaryOp(SDNode *N); SDValue SplitVecOp_TruncateHelper(SDNode *N); Index: lib/CodeGen/SelectionDAG/LegalizeVectorTypes.cpp =================================================================== --- lib/CodeGen/SelectionDAG/LegalizeVectorTypes.cpp +++ lib/CodeGen/SelectionDAG/LegalizeVectorTypes.cpp @@ -1471,6 +1471,20 @@ case ISD::FCANONICALIZE: Res = SplitVecOp_UnaryOp(N); break; + case ISD::VECREDUCE_FADD: + case ISD::VECREDUCE_FMUL: + case ISD::VECREDUCE_ADD: + case ISD::VECREDUCE_MUL: + case ISD::VECREDUCE_AND: + case ISD::VECREDUCE_OR: + case ISD::VECREDUCE_XOR: + case ISD::VECREDUCE_SMAX: + case ISD::VECREDUCE_SMIN: + case ISD::VECREDUCE_UMAX: + case ISD::VECREDUCE_UMIN: + case ISD::VECREDUCE_FMAX: + case ISD::VECREDUCE_FMIN: + Res = SplitVecOp_VECREDUCE(N, OpNo); } } @@ -1523,6 +1537,49 @@ return DAG.getNode(ISD::CONCAT_VECTORS, DL, Src0VT, LoSelect, HiSelect); } +SDValue DAGTypeLegalizer::SplitVecOp_VECREDUCE(SDNode *N, unsigned OpNo) { + assert(OpNo == 0 && "Can only split reduce vector operand"); + EVT ResVT = N->getValueType(0); + SDValue Lo, Hi; + SDLoc dl(N); + GetSplitVector(N->getOperand(0), Lo, Hi); + EVT LoOpVT, HiOpVT; + std::tie(LoOpVT, HiOpVT) = + DAG.GetSplitDestVTs(N->getOperand(0).getValueType()); + bool NoNaN = false; + + if (N->getNumOperands() > 1) + NoNaN = cast(N->getOperand(1))->getZExtValue(); + + unsigned CombineOpc = 0; + switch (N->getOpcode()) { + case ISD::VECREDUCE_FADD: CombineOpc = ISD::FADD; break; + case ISD::VECREDUCE_FMUL: CombineOpc = ISD::FMUL; break; + case ISD::VECREDUCE_ADD: CombineOpc = ISD::ADD; break; + case ISD::VECREDUCE_MUL: CombineOpc = ISD::MUL; break; + case ISD::VECREDUCE_AND: CombineOpc = ISD::AND; break; + case ISD::VECREDUCE_OR: CombineOpc = ISD::OR; break; + case ISD::VECREDUCE_XOR: CombineOpc = ISD::XOR; break; + case ISD::VECREDUCE_SMAX: CombineOpc = ISD::SMAX; break; + case ISD::VECREDUCE_SMIN: CombineOpc = ISD::SMIN; break; + case ISD::VECREDUCE_UMAX: CombineOpc = ISD::UMAX; break; + case ISD::VECREDUCE_UMIN: CombineOpc = ISD::UMIN; break; + case ISD::VECREDUCE_FMAX: + CombineOpc = NoNaN ? ISD::FMAXNUM : ISD::FMAXNAN; + break; + case ISD::VECREDUCE_FMIN: + CombineOpc = NoNaN ? ISD::FMINNUM : ISD::FMINNAN; + break; + default: + llvm_unreachable("Unexpected reduce ISD node"); + } + + // Use the appropriate scalar instruction on the split subvectors before + // reducing the now partially reduced smaller vector. + SDValue Partial = DAG.getNode(CombineOpc, dl, LoOpVT, Lo, Hi); + return DAG.getNode(N->getOpcode(), dl, ResVT, Partial); +} + SDValue DAGTypeLegalizer::SplitVecOp_UnaryOp(SDNode *N) { // The result has a legal vector type, but the input needs splitting. EVT ResVT = N->getValueType(0); Index: lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h =================================================================== --- lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h +++ lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h @@ -915,6 +915,8 @@ void visitGCRelocate(const GCRelocateInst &I); void visitGCResult(const GCResultInst &I); + void visitVectorReduce(const CallInst &I, unsigned Intrinsic); + void visitUserOp1(const Instruction &I) { llvm_unreachable("UserOp1 should not exist at instruction selection time!"); } Index: lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp =================================================================== --- lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp +++ lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp @@ -5768,6 +5768,24 @@ case Intrinsic::experimental_deoptimize: LowerDeoptimizeCall(&I); return nullptr; + + case Intrinsic::vector_reduce_fadd: + case Intrinsic::vector_reduce_fmul: + case Intrinsic::vector_reduce_add: + case Intrinsic::vector_reduce_mul: + case Intrinsic::vector_reduce_and: + case Intrinsic::vector_reduce_or: + case Intrinsic::vector_reduce_xor: + case Intrinsic::vector_reduce_smax: + case Intrinsic::vector_reduce_smin: + case Intrinsic::vector_reduce_umax: + case Intrinsic::vector_reduce_umin: + case Intrinsic::vector_reduce_fmax: + case Intrinsic::vector_reduce_fmin: { + visitVectorReduce(I, Intrinsic); + return nullptr; + } + } } @@ -7649,6 +7667,63 @@ FuncInfo.MF->getFrameInfo().setHasPatchPoint(); } +void SelectionDAGBuilder::visitVectorReduce(const CallInst &I, + unsigned Intrinsic) { + const TargetLowering &TLI = DAG.getTargetLoweringInfo(); + SDValue Op1 = getValue(I.getArgOperand(0)); + SDLoc dl = getCurSDLoc(); + EVT VT = TLI.getValueType(DAG.getDataLayout(), I.getType()); + SDValue Res; + switch (Intrinsic) { + case Intrinsic::vector_reduce_fadd: + Res = DAG.getNode(ISD::VECREDUCE_FADD, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_fmul: + Res = DAG.getNode(ISD::VECREDUCE_FMUL, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_add: + Res = DAG.getNode(ISD::VECREDUCE_ADD, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_mul: + Res = DAG.getNode(ISD::VECREDUCE_MUL, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_and: + Res = DAG.getNode(ISD::VECREDUCE_AND, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_or: + Res = DAG.getNode(ISD::VECREDUCE_OR, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_xor: + Res = DAG.getNode(ISD::VECREDUCE_XOR, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_smax: + Res = DAG.getNode(ISD::VECREDUCE_SMAX, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_smin: + Res = DAG.getNode(ISD::VECREDUCE_SMIN, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_umax: + Res = DAG.getNode(ISD::VECREDUCE_UMAX, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_umin: + Res = DAG.getNode(ISD::VECREDUCE_UMIN, dl, VT, Op1); + break; + case Intrinsic::vector_reduce_fmax: { + SDValue Op2 = getValue(I.getArgOperand(1)); + Res = DAG.getNode(ISD::VECREDUCE_FMAX, dl, VT, Op1, Op2); + break; + } + case Intrinsic::vector_reduce_fmin: { + SDValue Op2 = getValue(I.getArgOperand(1)); + Res = DAG.getNode(ISD::VECREDUCE_FMIN, dl, VT, Op1, Op2); + break; + } + default: + llvm_unreachable("Unhandled vector reduce intrinsic"); + } + setValue(&I, Res); +} + /// Returns an AttributeSet representing the attributes applied to the return /// value of the given call. static AttributeSet getReturnAttrs(TargetLowering::CallLoweringInfo &CLI) { Index: lib/CodeGen/SelectionDAG/SelectionDAGDumper.cpp =================================================================== --- lib/CodeGen/SelectionDAG/SelectionDAGDumper.cpp +++ lib/CodeGen/SelectionDAG/SelectionDAGDumper.cpp @@ -343,6 +343,19 @@ case ISD::SETFALSE: return "setfalse"; case ISD::SETFALSE2: return "setfalse2"; } + case ISD::VECREDUCE_FADD: return "vecreduce_fadd"; + case ISD::VECREDUCE_FMUL: return "vecreduce_fmul"; + case ISD::VECREDUCE_ADD: return "vecreduce_add"; + case ISD::VECREDUCE_MUL: return "vecreduce_mul"; + case ISD::VECREDUCE_AND: return "vecreduce_and"; + case ISD::VECREDUCE_OR: return "vecreduce_or"; + case ISD::VECREDUCE_XOR: return "vecreduce_xor"; + case ISD::VECREDUCE_SMAX: return "vecreduce_smax"; + case ISD::VECREDUCE_SMIN: return "vecreduce_smin"; + case ISD::VECREDUCE_UMAX: return "vecreduce_umax"; + case ISD::VECREDUCE_UMIN: return "vecreduce_umin"; + case ISD::VECREDUCE_FMAX: return "vecreduce_fmax"; + case ISD::VECREDUCE_FMIN: return "vecreduce_fmin"; } } Index: lib/IR/IRBuilder.cpp =================================================================== --- lib/IR/IRBuilder.cpp +++ lib/IR/IRBuilder.cpp @@ -161,6 +161,71 @@ return CI; } +static CallInst *getReductionIntrinsic(IRBuilderBase *Builder, Intrinsic::ID ID, + Value *Src) { + Module *M = Builder->GetInsertBlock()->getParent()->getParent(); + Value *Ops[] = {Src}; + Type *Tys[] = { Src->getType()->getVectorElementType(), Src->getType() }; + auto Decl = Intrinsic::getDeclaration(M, ID, Tys); + return createCallHelper(Decl, Ops, Builder); +} + +CallInst *IRBuilderBase::CreateFAddReduce(Value *Src) { + return getReductionIntrinsic(this, Intrinsic::vector_reduce_fadd, Src); +} + +CallInst *IRBuilderBase::CreateFMulReduce(Value *Src) { + return getReductionIntrinsic(this, Intrinsic::vector_reduce_fmul, Src); +} + +CallInst *IRBuilderBase::CreateAddReduce(Value *Src) { + return getReductionIntrinsic(this, Intrinsic::vector_reduce_add, Src); +} + +CallInst *IRBuilderBase::CreateMulReduce(Value *Src) { + return getReductionIntrinsic(this, Intrinsic::vector_reduce_mul, Src); +} + +CallInst *IRBuilderBase::CreateAndReduce(Value *Src) { + return getReductionIntrinsic(this, Intrinsic::vector_reduce_and, Src); +} + +CallInst *IRBuilderBase::CreateOrReduce(Value *Src) { + return getReductionIntrinsic(this, Intrinsic::vector_reduce_or, Src); +} + +CallInst *IRBuilderBase::CreateXorReduce(Value *Src) { + return getReductionIntrinsic(this, Intrinsic::vector_reduce_xor, Src); +} + +CallInst *IRBuilderBase::CreateIntMaxReduce(Value *Src, bool IsSigned) { + auto ID = + IsSigned ? Intrinsic::vector_reduce_smax : Intrinsic::vector_reduce_umax; + return getReductionIntrinsic(this, ID, Src); +} + +CallInst *IRBuilderBase::CreateIntMinReduce(Value *Src, bool IsSigned) { + auto ID = + IsSigned ? Intrinsic::vector_reduce_smin : Intrinsic::vector_reduce_umin; + return getReductionIntrinsic(this, ID, Src); +} + +CallInst *IRBuilderBase::CreateFPMaxReduce(Value *Src, bool NoNaN) { + Module *M = BB->getParent()->getParent(); + Value *Ops[] = { Src, NoNaN ? getInt32(1) : getInt32(0) }; + Type *Tys[] = {Src->getType()->getVectorElementType(), Src->getType()}; + auto Decl = Intrinsic::getDeclaration(M, Intrinsic::vector_reduce_fmax, Tys); + return createCallHelper(Decl, Ops, this); +} + +CallInst *IRBuilderBase::CreateFPMinReduce(Value *Src, bool NoNaN) { + Module *M = BB->getParent()->getParent(); + Value* Ops[] = { Src, NoNaN ? getInt32(1) : getInt32(0) }; + Type *Tys[] = {Src->getType()->getVectorElementType(), Src->getType()}; + auto Decl = Intrinsic::getDeclaration(M, Intrinsic::vector_reduce_fmin, Tys); + return createCallHelper(Decl, Ops, this); +} + CallInst *IRBuilderBase::CreateLifetimeStart(Value *Ptr, ConstantInt *Size) { assert(isa(Ptr->getType()) && "lifetime.start only applies to pointers."); Index: lib/Transforms/Utils/LoopUtils.cpp =================================================================== --- lib/Transforms/Utils/LoopUtils.cpp +++ lib/Transforms/Utils/LoopUtils.cpp @@ -18,6 +18,7 @@ #include "llvm/Analysis/GlobalsModRef.h" #include "llvm/Analysis/LoopInfo.h" #include "llvm/Analysis/LoopPass.h" +#include "llvm/Analysis/TargetTransformInfo.h" #include "llvm/Analysis/ScalarEvolution.h" #include "llvm/Analysis/ScalarEvolutionAliasAnalysis.h" #include "llvm/Analysis/ScalarEvolutionExpander.h" @@ -1111,3 +1112,149 @@ else return (FalseVal + (TrueVal / 2)) / TrueVal; } + +/// \brief Adds a 'fast' flag to floating point operations. +static Value *addFastMathFlag(Value *V) { + if (isa(V)) { + FastMathFlags Flags; + Flags.setUnsafeAlgebra(); + cast(V)->setFastMathFlags(Flags); + } + return V; +} + +// Helper to generate a log2 shuffle reduction. +static Value *getShuffleReduction( + IRBuilder<> &Builder, Value *Src, unsigned Op, + RecurrenceDescriptor::MinMaxRecurrenceKind *MinMaxKind = nullptr) { + unsigned VF = Src->getType()->getVectorNumElements(); + // VF is a power of 2 so we can emit the reduction using log2(VF) shuffles + // and vector ops, reducing the set of values being computed by half each + // round. + assert(isPowerOf2_32(VF) && + "Reduction emission only supported for pow2 vectors!"); + Value *TmpVec = Src; + SmallVector ShuffleMask(VF, nullptr); + for (unsigned i = VF; i != 1; i >>= 1) { + // Move the upper half of the vector to the lower half. + for (unsigned j = 0; j != i / 2; ++j) + ShuffleMask[j] = Builder.getInt32(i / 2 + j); + + // Fill the rest of the mask with undef. + std::fill(&ShuffleMask[i / 2], ShuffleMask.end(), + UndefValue::get(Builder.getInt32Ty())); + + Value *Shuf = Builder.CreateShuffleVector( + TmpVec, UndefValue::get(TmpVec->getType()), + ConstantVector::get(ShuffleMask), "rdx.shuf"); + + if (Op != Instruction::ICmp && Op != Instruction::FCmp) + // Floating point operations had to be 'fast' to enable the reduction. + TmpVec = addFastMathFlag(Builder.CreateBinOp((Instruction::BinaryOps)Op, + TmpVec, Shuf, "bin.rdx")); + else + TmpVec = RecurrenceDescriptor::createMinMaxOp(Builder, *MinMaxKind, TmpVec, + Shuf); + } + // The result is in the first element of the vector. + return Builder.CreateExtractElement(TmpVec, Builder.getInt32(0)); +} + +Value *llvm::createTargetReduction(IRBuilder<> &Builder, + const TargetTransformInfo *TTI, + unsigned Opcode, Value *Src) { + assert(isa(Src->getType()) && "Type must be a vector"); + assert((Opcode == Instruction::Add || Opcode == Instruction::FAdd) && + "Unhandled reduction opcode"); + // The reduction can't be signed or a minmax kind. + if (TTI->useReductionIntrinsic(Opcode, Src->getType(), false, false)) { + if (Opcode == Instruction::Add) + return Builder.CreateAddReduce(Src); + else + return Builder.CreateFAddReduce(Src); + } + + return getShuffleReduction(Builder, Src, Opcode); +} + +Value *llvm::createTargetReduction(IRBuilder<> &Builder, + const TargetTransformInfo *TTI, + RecurrenceDescriptor &Desc, Value *Src, + bool InOrder, bool NoNaN) { + assert(!InOrder && "In order reductions not handled yet"); + + RecurrenceDescriptor::RecurrenceKind RecKind = Desc.getRecurrenceKind(); + RecurrenceDescriptor::MinMaxRecurrenceKind MinMaxKind = + Desc.getMinMaxRecurrenceKind(); + unsigned Op = RecurrenceDescriptor::getRecurrenceBinOp(RecKind); + bool IsSigned = false, IsMaxOp = false; + + std::function BuildFunc; + switch (RecKind) { + case RecurrenceDescriptor::RK_FloatAdd: + BuildFunc = [&]() { return Builder.CreateFAddReduce(Src); }; + break; + case RecurrenceDescriptor::RK_FloatMult: + BuildFunc = [&]() { return Builder.CreateFMulReduce(Src); }; + break; + case RecurrenceDescriptor::RK_IntegerAdd: + BuildFunc = [&]() { return Builder.CreateAddReduce(Src); }; + break; + case RecurrenceDescriptor::RK_IntegerMult: + BuildFunc = [&]() { return Builder.CreateMulReduce(Src); }; + break; + case RecurrenceDescriptor::RK_IntegerAnd: + BuildFunc = [&]() { return Builder.CreateAndReduce(Src); }; + break; + case RecurrenceDescriptor::RK_IntegerOr: + BuildFunc = [&]() { return Builder.CreateOrReduce(Src); }; + break; + case RecurrenceDescriptor::RK_IntegerXor: + BuildFunc = [&]() { return Builder.CreateXorReduce(Src); }; + break; + case RecurrenceDescriptor::RK_IntegerMinMax: { + switch (Desc.getMinMaxRecurrenceKind()) { + case RecurrenceDescriptor::MRK_SIntMax: + IsSigned = true; + IsMaxOp = true; + BuildFunc = [&]() { return Builder.CreateIntMaxReduce(Src, true); }; + break; + case RecurrenceDescriptor::MRK_UIntMax: + IsMaxOp = true; + BuildFunc = [&]() { return Builder.CreateIntMaxReduce(Src, false); }; + break; + case RecurrenceDescriptor::MRK_SIntMin: + IsSigned = true; + BuildFunc = [&]() { return Builder.CreateIntMinReduce(Src, true); }; + break; + case RecurrenceDescriptor::MRK_UIntMin: + BuildFunc = [&]() { return Builder.CreateIntMinReduce(Src, false); }; + break; + default: + llvm_unreachable("Unhandled MRK"); + } + break; + } + case RecurrenceDescriptor::RK_FloatMinMax: { + switch (Desc.getMinMaxRecurrenceKind()) { + case RecurrenceDescriptor::MRK_FloatMin: + BuildFunc = [&]() { return Builder.CreateFPMinReduce(Src, NoNaN); }; + break; + case RecurrenceDescriptor::MRK_FloatMax: + IsMaxOp = true; + BuildFunc = [&]() { return Builder.CreateFPMaxReduce(Src, NoNaN); }; + break; + default: + llvm_unreachable("Unhandled MRK"); + } + break; + } + default: + llvm_unreachable("Unhandled RecKind"); + } + + if (TTI->useReductionIntrinsic(Op, Src->getType(), IsMaxOp, IsSigned, NoNaN)) + return BuildFunc(); + + return getShuffleReduction(Builder, Src, Op, &MinMaxKind); +} Index: lib/Transforms/Vectorize/LoopVectorize.cpp =================================================================== --- lib/Transforms/Vectorize/LoopVectorize.cpp +++ lib/Transforms/Vectorize/LoopVectorize.cpp @@ -1730,6 +1730,9 @@ /// access that can be widened. bool memoryInstructionCanBeWidened(Instruction *I, unsigned VF = 1); + // Returns true if the NoNaN attribute is set on the function. + bool hasFunNoNaNAttr() const { return HasFunNoNaNAttr; } + private: /// Check if a single basic block loop is vectorizable. /// At this point we know that this is a loop with a constant trip count @@ -4089,39 +4092,9 @@ } if (VF > 1) { - // VF is a power of 2 so we can emit the reduction using log2(VF) shuffles - // and vector ops, reducing the set of values being computed by half each - // round. - assert(isPowerOf2_32(VF) && - "Reduction emission only supported for pow2 vectors!"); - Value *TmpVec = ReducedPartRdx; - SmallVector ShuffleMask(VF, nullptr); - for (unsigned i = VF; i != 1; i >>= 1) { - // Move the upper half of the vector to the lower half. - for (unsigned j = 0; j != i / 2; ++j) - ShuffleMask[j] = Builder.getInt32(i / 2 + j); - - // Fill the rest of the mask with undef. - std::fill(&ShuffleMask[i / 2], ShuffleMask.end(), - UndefValue::get(Builder.getInt32Ty())); - - Value *Shuf = Builder.CreateShuffleVector( - TmpVec, UndefValue::get(TmpVec->getType()), - ConstantVector::get(ShuffleMask), "rdx.shuf"); - - if (Op != Instruction::ICmp && Op != Instruction::FCmp) - // Floating point operations had to be 'fast' to enable the reduction. - TmpVec = addFastMathFlag(Builder.CreateBinOp( - (Instruction::BinaryOps)Op, TmpVec, Shuf, "bin.rdx")); - else - TmpVec = RecurrenceDescriptor::createMinMaxOp(Builder, MinMaxKind, - TmpVec, Shuf); - } - - // The result is in the first element of the vector. - ReducedPartRdx = - Builder.CreateExtractElement(TmpVec, Builder.getInt32(0)); - + bool NoNaN = Legal->hasFunNoNaNAttr(); + ReducedPartRdx = createTargetReduction(Builder, TTI, RdxDesc, + ReducedPartRdx, false, NoNaN); // If the reduction can be performed in a smaller type, we need to extend // the reduction to the wider type before we branch to the original loop. if (Phi->getType() != RdxDesc.getRecurrenceType()) Index: lib/Transforms/Vectorize/SLPVectorizer.cpp =================================================================== --- lib/Transforms/Vectorize/SLPVectorizer.cpp +++ lib/Transforms/Vectorize/SLPVectorizer.cpp @@ -40,6 +40,7 @@ #include "llvm/Support/CommandLine.h" #include "llvm/Support/Debug.h" #include "llvm/Support/raw_ostream.h" +#include "llvm/Transforms/Utils/LoopUtils.h" #include "llvm/Transforms/Vectorize.h" #include #include @@ -4469,7 +4470,7 @@ // Emit a reduction. Value *ReducedSubTree = - emitReduction(VectorizedRoot, Builder, ReduxWidth); + emitReduction(VectorizedRoot, Builder, ReduxWidth, TTI); if (VectorizedTree) { Builder.SetCurrentDebugLocation(Loc); VectorizedTree = Builder.CreateBinOp(ReductionOpcode, VectorizedTree, @@ -4536,33 +4537,29 @@ /// \brief Emit a horizontal reduction of the vectorized value. Value *emitReduction(Value *VectorizedValue, IRBuilder<> &Builder, - unsigned ReduxWidth) { + unsigned ReduxWidth, const TargetTransformInfo *TTI) { assert(VectorizedValue && "Need to have a vectorized tree node"); assert(isPowerOf2_32(ReduxWidth) && "We only handle power-of-two reductions for now"); + if (!IsPairwiseReduction) + return createTargetReduction(Builder, TTI, ReductionOpcode, + VectorizedValue); + Value *TmpVec = VectorizedValue; for (unsigned i = ReduxWidth / 2; i != 0; i >>= 1) { - if (IsPairwiseReduction) { - Value *LeftMask = + Value *LeftMask = createRdxShuffleMask(ReduxWidth, i, true, true, Builder); - Value *RightMask = + Value *RightMask = createRdxShuffleMask(ReduxWidth, i, true, false, Builder); - Value *LeftShuf = Builder.CreateShuffleVector( + Value *LeftShuf = Builder.CreateShuffleVector( TmpVec, UndefValue::get(TmpVec->getType()), LeftMask, "rdx.shuf.l"); - Value *RightShuf = Builder.CreateShuffleVector( + Value *RightShuf = Builder.CreateShuffleVector( TmpVec, UndefValue::get(TmpVec->getType()), (RightMask), "rdx.shuf.r"); - TmpVec = Builder.CreateBinOp(ReductionOpcode, LeftShuf, RightShuf, - "bin.rdx"); - } else { - Value *UpperHalf = - createRdxShuffleMask(ReduxWidth, i, false, false, Builder); - Value *Shuf = Builder.CreateShuffleVector( - TmpVec, UndefValue::get(TmpVec->getType()), UpperHalf, "rdx.shuf"); - TmpVec = Builder.CreateBinOp(ReductionOpcode, TmpVec, Shuf, "bin.rdx"); - } + TmpVec = + Builder.CreateBinOp(ReductionOpcode, LeftShuf, RightShuf, "bin.rdx"); } // The result is in the first element of the vector.