[llvm] ead88c7 - [RISCV] Prefer (select (x < 0), y, z) -> x >> (XLEN - 1) & (y - z) + z even with Zicond. (#125772)
via llvm-commits
llvm-commits at lists.llvm.org
Thu Feb 6 08:53:45 PST 2025
Author: Craig Topper
Date: 2025-02-06T08:53:40-08:00
New Revision: ead88c787c4eba28b2148a5aaf190186bdb40820
URL: https://github.com/llvm/llvm-project/commit/ead88c787c4eba28b2148a5aaf190186bdb40820
DIFF: https://github.com/llvm/llvm-project/commit/ead88c787c4eba28b2148a5aaf190186bdb40820.diff
LOG: [RISCV] Prefer (select (x < 0), y, z) -> x >> (XLEN - 1) & (y - z) + z even with Zicond. (#125772)
The Zicond version of this requires an li instruction and an
additional register.
Without Zicond we match this in a DAGCombine on RISCVISD::SELECT_CC.
This PR has 2 commits. I'll pre-commit the test change if this looks
good.
Added:
Modified:
llvm/lib/Target/RISCV/RISCVISelLowering.cpp
Removed:
################################################################################
diff --git a/llvm/lib/Target/RISCV/RISCVISelLowering.cpp b/llvm/lib/Target/RISCV/RISCVISelLowering.cpp
index 165c71d8e03f16..d91ba33c235966 100644
--- a/llvm/lib/Target/RISCV/RISCVISelLowering.cpp
+++ b/llvm/lib/Target/RISCV/RISCVISelLowering.cpp
@@ -8460,6 +8460,33 @@ SDValue RISCVTargetLowering::lowerSELECT(SDValue Op, SelectionDAG &DAG) const {
if (isa<ConstantSDNode>(TrueV) && isa<ConstantSDNode>(FalseV)) {
const APInt &TrueVal = TrueV->getAsAPIntVal();
const APInt &FalseVal = FalseV->getAsAPIntVal();
+
+ // Prefer these over Zicond to avoid materializing an immediate:
+ // (select (x < 0), y, z) -> x >> (XLEN - 1) & (y - z) + z
+ // (select (x > -1), z, y) -> x >> (XLEN - 1) & (y - z) + z
+ if (CondV.getOpcode() == ISD::SETCC &&
+ CondV.getOperand(0).getValueType() == VT && CondV.hasOneUse()) {
+ ISD::CondCode CCVal = cast<CondCodeSDNode>(CondV.getOperand(2))->get();
+ if ((CCVal == ISD::SETLT && isNullConstant(CondV.getOperand(1))) ||
+ (CCVal == ISD::SETGT && isAllOnesConstant(CondV.getOperand(1)))) {
+ int64_t TrueImm = TrueVal.getSExtValue();
+ int64_t FalseImm = FalseVal.getSExtValue();
+ if (CCVal == ISD::SETGT)
+ std::swap(TrueImm, FalseImm);
+ if (isInt<12>(TrueImm) && isInt<12>(FalseImm) &&
+ isInt<12>(TrueImm - FalseImm)) {
+ SDValue SRA =
+ DAG.getNode(ISD::SRA, DL, VT, CondV.getOperand(0),
+ DAG.getConstant(Subtarget.getXLen() - 1, DL, VT));
+ SDValue AND =
+ DAG.getNode(ISD::AND, DL, VT, SRA,
+ DAG.getSignedConstant(TrueImm - FalseImm, DL, VT));
+ return DAG.getNode(ISD::ADD, DL, VT, AND,
+ DAG.getSignedConstant(FalseImm, DL, VT));
+ }
+ }
+ }
+
const int TrueValCost = RISCVMatInt::getIntMatCost(
TrueVal, Subtarget.getXLen(), Subtarget, /*CompressionCost=*/true);
const int FalseValCost = RISCVMatInt::getIntMatCost(
More information about the llvm-commits
mailing list