Index: lib/Target/AMDGPU/SIISelLowering.cpp =================================================================== --- lib/Target/AMDGPU/SIISelLowering.cpp +++ lib/Target/AMDGPU/SIISelLowering.cpp @@ -812,10 +812,28 @@ SDValue SITargetLowering::LowerFrameIndex(SDValue Op, SelectionDAG &DAG) const { + SDLoc SL(Op); FrameIndexSDNode *FINode = cast(Op); unsigned FrameIndex = FINode->getIndex(); - return DAG.getTargetFrameIndex(FrameIndex, MVT::i32); + // A FrameIndex node represents a 32-bit offset into scratch memory. If + // the high bit of a frame index offset were to be set, this would mean + // that it represented an offset of ~2GB * 64 = ~128GB from the start of the + // scratch buffer, with 64 being the number of threads per wave. + // + // It may be theoretically possible to construct a system that makes + // 128 GB of memory accessible to the GPU, but for now we assume such + // machines do not exist. + // + // This allows us to mark the high bit of the FrameIndex node as known zero, + // which is important, because it means in most situations we can + // prove that values derived from FrameIndex nodes are non-negative. + // This enables us to take advantage of more addressing modes when + // accessing scratch buffers, since for scratch reads/writes, the register + // offset must always be positive. + return DAG.getNode(ISD::AssertAnd, SL, MVT::i32, + DAG.getTargetFrameIndex(FrameIndex, MVT::i32), + DAG.getConstant(APInt::getLowBitsSet(32,31), SL, MVT::i32)); } /// This transforms the control flow intrinsics to get the branch destination as @@ -2033,6 +2051,13 @@ } } +static bool isFrameIndexOp(SDValue Op) { + if (Op.getOpcode() == ISD::AssertAnd) + Op = Op.getOperand(0); + + return isa(Op); +} + /// \brief Legalize target independent instructions (e.g. INSERT_SUBREG) /// with frame index operands. /// LLVM assumes that inputs are to these instructions are registers. @@ -2041,7 +2066,7 @@ SmallVector Ops; for (unsigned i = 0; i < Node->getNumOperands(); ++i) { - if (!isa(Node->getOperand(i))) { + if (!isFrameIndexOp(Node->getOperand(i))) { Ops.push_back(Node->getOperand(i)); continue; }