Index: llvm/lib/Target/AArch64/AArch64ISelLowering.h =================================================================== --- llvm/lib/Target/AArch64/AArch64ISelLowering.h +++ llvm/lib/Target/AArch64/AArch64ISelLowering.h @@ -155,6 +155,7 @@ // Scalar logical instructions with not operand SBIC, + SEON, // Conditional compares. Operands: left,right,falsecc,cc,flags CCMP, Index: llvm/lib/Target/AArch64/AArch64ISelLowering.cpp =================================================================== --- llvm/lib/Target/AArch64/AArch64ISelLowering.cpp +++ llvm/lib/Target/AArch64/AArch64ISelLowering.cpp @@ -2264,6 +2264,7 @@ MAKE_CASE(AArch64ISD::SBCS) MAKE_CASE(AArch64ISD::ANDS) MAKE_CASE(AArch64ISD::SBIC) + MAKE_CASE(AArch64ISD::SEON) MAKE_CASE(AArch64ISD::CCMP) MAKE_CASE(AArch64ISD::CCMN) MAKE_CASE(AArch64ISD::FCCMP) @@ -14988,6 +14989,47 @@ return SDValue(); } +// ((X & C) ^ Y) ^ C --> eon Y, (X | ~C) +static SDValue performXORCombineWithNotOp(SDNode *N, SelectionDAG &DAG) { + EVT VT = N->getValueType(0); + if (VT != MVT::i32 && VT != MVT::i64) + return SDValue(); + + SDValue LHS = N->getOperand(0); + auto *XorC = dyn_cast(N->getOperand(1)); + if (!XorC) + return SDValue(); + + SDLoc DL(N); + SDValue X, Y, C; + auto canGetNotFromXorAnd = [&](SDNode *N, SDValue Op0) { + if (Op0.getOpcode() != ISD::XOR || !Op0.hasOneUse()) + return false; + + Y = Op0.getOperand(1); + if (isa(Y)) + return false; + + SDValue AndVal = Op0.getOperand(0); + if (AndVal.getOpcode() != ISD::AND || !AndVal.hasOneUse()) + return false; + + ConstantSDNode *AndC = dyn_cast(AndVal.getOperand(1)); + if (!AndC || AndC->getSExtValue() != XorC->getSExtValue()) + return false; + + C = DAG.getNOT(DL, AndVal.getOperand(1), VT); + X = AndVal.getOperand(0); + return true; + }; + + if (canGetNotFromXorAnd(N, LHS)) { + SDValue OrVal = DAG.getNode(ISD::OR, DL, VT, X, C); + return DAG.getNode(AArch64ISD::SEON, DL, VT, Y, OrVal); + } + + return SDValue(); +} static SDValue performXorCombine(SDNode *N, SelectionDAG &DAG, TargetLowering::DAGCombinerInfo &DCI, @@ -14995,6 +15037,9 @@ if (DCI.isBeforeLegalizeOps()) return SDValue(); + if (SDValue R = performXORCombineWithNotOp(N, DAG)) + return R; + return foldVectorXorShiftIntoCmp(N, DAG, Subtarget); } Index: llvm/lib/Target/AArch64/AArch64InstrInfo.td =================================================================== --- llvm/lib/Target/AArch64/AArch64InstrInfo.td +++ llvm/lib/Target/AArch64/AArch64InstrInfo.td @@ -610,6 +610,7 @@ def AArch64fccmp : SDNode<"AArch64ISD::FCCMP", SDT_AArch64FCCMP>; def AArch64sbic : SDNode<"AArch64ISD::SBIC", SDTIntBinOp>; +def AArch64seon : SDNode<"AArch64ISD::SEON", SDTIntBinOp>; def AArch64threadpointer : SDNode<"AArch64ISD::THREAD_POINTER", SDTPtrLeaf>; @@ -2218,6 +2219,8 @@ def : Pat<(not GPR64:$Xm), (ORNXrr XZR, GPR64:$Xm)>; def : Pat<(AArch64sbic GPR32:$Wm, GPR32:$Wn), (BICWrr GPR32:$Wm, GPR32:$Wn)>; def : Pat<(AArch64sbic GPR64:$Xm, GPR64:$Xn), (BICXrr GPR64:$Xm, GPR64:$Xn)>; +def : Pat<(AArch64seon GPR32:$Wm, GPR32:$Wn), (EONWrr GPR32:$Wm, GPR32:$Wn)>; +def : Pat<(AArch64seon GPR64:$Xm, GPR64:$Xn), (EONXrr GPR64:$Xm, GPR64:$Xn)>; //===----------------------------------------------------------------------===// // One operand data processing instructions. Index: llvm/test/CodeGen/AArch64/logical-op-with-not.ll =================================================================== --- llvm/test/CodeGen/AArch64/logical-op-with-not.ll +++ llvm/test/CodeGen/AArch64/logical-op-with-not.ll @@ -197,9 +197,8 @@ define i64 @_Z6or_eon(i64 %0, i64 %1) { ; CHECK-LABEL: _Z6or_eon: ; CHECK: // %bb.0: -; CHECK-NEXT: and x8, x0, #0xfffffffffffff000 -; CHECK-NEXT: eor x8, x8, x1 -; CHECK-NEXT: eor x0, x8, #0xfffffffffffff000 +; CHECK-NEXT: orr x8, x0, #0xfff +; CHECK-NEXT: eon x0, x1, x8 ; CHECK-NEXT: ret %3 = and i64 %0, -4096 %4 = xor i64 %3, %1