diff --git a/llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp b/llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp --- a/llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp +++ b/llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp @@ -3083,6 +3083,18 @@ TrueVal = Builder.CreateFreeze(TrueVal); return BinaryOperator::CreateAnd(FalseVal, Builder.CreateOr(C, TrueVal)); } + // select (a | ~c), a, b -> select a, true, (select c, b, false) + if (match(CondVal, m_c_Or(m_Specific(TrueVal), m_Not(m_Value(C)))) && + CondVal->hasOneUse()) { + Value *AndV = Builder.CreateSelect(C, FalseVal, Zero); + return SelectInst::Create(TrueVal, One, AndV); + } + // select (c & ~b), a, b -> select b, true, (select c, a, false) + if (match(CondVal, m_c_And(m_Value(C), m_Not(m_Specific(FalseVal)))) && + CondVal->hasOneUse()) { + Value *AndV = Builder.CreateSelect(C, TrueVal, Zero); + return SelectInst::Create(FalseVal, One, AndV); + } if (match(FalseVal, m_Zero()) || match(TrueVal, m_One())) { Use *Y = nullptr; diff --git a/llvm/test/Transforms/InstCombine/select-and-or.ll b/llvm/test/Transforms/InstCombine/select-and-or.ll --- a/llvm/test/Transforms/InstCombine/select-and-or.ll +++ b/llvm/test/Transforms/InstCombine/select-and-or.ll @@ -601,3 +601,176 @@ %r = select <2 x i1> %cond, <2 x i1> %a, <2 x i1> %b ret <2 x i1> %r } + +define i1 @or_and1(i1 %a, i1 %b, i1 %c) { +; CHECK-LABEL: @or_and1( +; CHECK-NEXT: [[TMP1:%.*]] = select i1 [[C:%.*]], i1 [[A:%.*]], i1 false +; CHECK-NEXT: [[R:%.*]] = select i1 [[B:%.*]], i1 true, i1 [[TMP1]] +; CHECK-NEXT: ret i1 [[R]] +; + %notb = xor i1 %b, true + %cond = and i1 %notb, %c + %r = select i1 %cond, i1 %a, i1 %b + ret i1 %r +} + +define i1 @or_and2(i1 %a, i1 %b, i1 %c) { +; CHECK-LABEL: @or_and2( +; CHECK-NEXT: [[TMP1:%.*]] = select i1 [[C:%.*]], i1 [[B:%.*]], i1 false +; CHECK-NEXT: [[R:%.*]] = select i1 [[A:%.*]], i1 true, i1 [[TMP1]] +; CHECK-NEXT: ret i1 [[R]] +; + %notc = xor i1 %c, true + %cond = or i1 %notc, %a + %r = select i1 %cond, i1 %a, i1 %b + ret i1 %r +} + +define i1 @or_and1_commuted(i1 %a, i1 %b, i1 %c) { +; CHECK-LABEL: @or_and1_commuted( +; CHECK-NEXT: [[TMP1:%.*]] = select i1 [[C:%.*]], i1 [[A:%.*]], i1 false +; CHECK-NEXT: [[R:%.*]] = select i1 [[B:%.*]], i1 true, i1 [[TMP1]] +; CHECK-NEXT: ret i1 [[R]] +; + %notb = xor i1 %b, true + %cond = and i1 %c, %notb + %r = select i1 %cond, i1 %a, i1 %b + ret i1 %r +} + +define i1 @or_and2_commuted(i1 %a, i1 %b, i1 %c) { +; CHECK-LABEL: @or_and2_commuted( +; CHECK-NEXT: [[TMP1:%.*]] = select i1 [[C:%.*]], i1 [[B:%.*]], i1 false +; CHECK-NEXT: [[R:%.*]] = select i1 [[A:%.*]], i1 true, i1 [[TMP1]] +; CHECK-NEXT: ret i1 [[R]] +; + %notc = xor i1 %c, true + %cond = or i1 %a, %notc + %r = select i1 %cond, i1 %a, i1 %b + ret i1 %r +} + +define i1 @or_and1_multiuse(i1 %a, i1 %b, i1 %c) { +; CHECK-LABEL: @or_and1_multiuse( +; CHECK-NEXT: [[NOTB:%.*]] = xor i1 [[B:%.*]], true +; CHECK-NEXT: [[COND:%.*]] = and i1 [[NOTB]], [[C:%.*]] +; CHECK-NEXT: call void @use(i1 [[COND]]) +; CHECK-NEXT: [[R:%.*]] = select i1 [[COND]], i1 [[A:%.*]], i1 [[B]] +; CHECK-NEXT: ret i1 [[R]] +; + %notb = xor i1 %b, true + %cond = and i1 %notb, %c + call void @use(i1 %cond) + %r = select i1 %cond, i1 %a, i1 %b + ret i1 %r +} + +define i1 @or_and2_multiuse(i1 %a, i1 %b, i1 %c) { +; CHECK-LABEL: @or_and2_multiuse( +; CHECK-NEXT: [[NOTC:%.*]] = xor i1 [[C:%.*]], true +; CHECK-NEXT: [[COND:%.*]] = or i1 [[NOTC]], [[A:%.*]] +; CHECK-NEXT: call void @use(i1 [[COND]]) +; CHECK-NEXT: [[R:%.*]] = select i1 [[COND]], i1 [[A]], i1 [[B:%.*]] +; CHECK-NEXT: ret i1 [[R]] +; + %notc = xor i1 %c, true + %cond = or i1 %notc, %a + call void @use(i1 %cond) + %r = select i1 %cond, i1 %a, i1 %b + ret i1 %r +} + +define <2 x i1> @or_and1_vec(<2 x i1> %a, <2 x i1> %b) { +; CHECK-LABEL: @or_and1_vec( +; CHECK-NEXT: [[C:%.*]] = call <2 x i1> @gen_v2i1() +; CHECK-NEXT: [[TMP1:%.*]] = select <2 x i1> [[C]], <2 x i1> [[A:%.*]], <2 x i1> zeroinitializer +; CHECK-NEXT: [[R:%.*]] = select <2 x i1> [[B:%.*]], <2 x i1> , <2 x i1> [[TMP1]] +; CHECK-NEXT: ret <2 x i1> [[R]] +; + %c = call <2 x i1> @gen_v2i1() + %notb = xor <2 x i1> %b, + %cond = and <2 x i1> %c, %notb + %r = select <2 x i1> %cond, <2 x i1> %a, <2 x i1> %b + ret <2 x i1> %r +} + +define <2 x i1> @or_and2_vec(<2 x i1> %a, <2 x i1> %b) { +; CHECK-LABEL: @or_and2_vec( +; CHECK-NEXT: [[C:%.*]] = call <2 x i1> @gen_v2i1() +; CHECK-NEXT: [[TMP1:%.*]] = select <2 x i1> [[C]], <2 x i1> [[B:%.*]], <2 x i1> zeroinitializer +; CHECK-NEXT: [[R:%.*]] = select <2 x i1> [[A:%.*]], <2 x i1> , <2 x i1> [[TMP1]] +; CHECK-NEXT: ret <2 x i1> [[R]] +; + %c = call <2 x i1> @gen_v2i1() + %notc = xor <2 x i1> %c, + %cond = or <2 x i1> %a, %notc + %r = select <2 x i1> %cond, <2 x i1> %a, <2 x i1> %b + ret <2 x i1> %r +} + +define <2 x i1> @or_and1_vec_commuted(<2 x i1> %a, <2 x i1> %b) { +; CHECK-LABEL: @or_and1_vec_commuted( +; CHECK-NEXT: [[C:%.*]] = call <2 x i1> @gen_v2i1() +; CHECK-NEXT: [[TMP1:%.*]] = select <2 x i1> [[C]], <2 x i1> [[A:%.*]], <2 x i1> zeroinitializer +; CHECK-NEXT: [[R:%.*]] = select <2 x i1> [[B:%.*]], <2 x i1> , <2 x i1> [[TMP1]] +; CHECK-NEXT: ret <2 x i1> [[R]] +; + %c = call <2 x i1> @gen_v2i1() + %notb = xor <2 x i1> %b, + %cond = and <2 x i1> %notb, %c + %r = select <2 x i1> %cond, <2 x i1> %a, <2 x i1> %b + ret <2 x i1> %r +} + +define <2 x i1> @or_and2_vec_commuted(<2 x i1> %a, <2 x i1> %b) { +; CHECK-LABEL: @or_and2_vec_commuted( +; CHECK-NEXT: [[C:%.*]] = call <2 x i1> @gen_v2i1() +; CHECK-NEXT: [[TMP1:%.*]] = select <2 x i1> [[C]], <2 x i1> [[B:%.*]], <2 x i1> zeroinitializer +; CHECK-NEXT: [[R:%.*]] = select <2 x i1> [[A:%.*]], <2 x i1> , <2 x i1> [[TMP1]] +; CHECK-NEXT: ret <2 x i1> [[R]] +; + %c = call <2 x i1> @gen_v2i1() + %notc = xor <2 x i1> %c, + %cond = or <2 x i1> %notc, %a + %r = select <2 x i1> %cond, <2 x i1> %a, <2 x i1> %b + ret <2 x i1> %r +} + +define i1 @or_and1_wrong_operand(i1 %a, i1 %b, i1 %c, i1 %d) { +; CHECK-LABEL: @or_and1_wrong_operand( +; CHECK-NEXT: [[NOTB:%.*]] = xor i1 [[B:%.*]], true +; CHECK-NEXT: [[COND:%.*]] = and i1 [[NOTB]], [[C:%.*]] +; CHECK-NEXT: [[R:%.*]] = select i1 [[COND]], i1 [[A:%.*]], i1 [[D:%.*]] +; CHECK-NEXT: ret i1 [[R]] +; + %notb = xor i1 %b, true + %cond = and i1 %c, %notb + %r = select i1 %cond, i1 %a, i1 %d + ret i1 %r +} + +define i1 @or_and2_wrong_operand(i1 %a, i1 %b, i1 %c, i1 %d) { +; CHECK-LABEL: @or_and2_wrong_operand( +; CHECK-NEXT: [[NOTC:%.*]] = xor i1 [[C:%.*]], true +; CHECK-NEXT: [[COND:%.*]] = or i1 [[NOTC]], [[A:%.*]] +; CHECK-NEXT: [[R:%.*]] = select i1 [[COND]], i1 [[D:%.*]], i1 [[B:%.*]] +; CHECK-NEXT: ret i1 [[R]] +; + %notc = xor i1 %c, true + %cond = or i1 %a, %notc + %r = select i1 %cond, i1 %d, i1 %b + ret i1 %r +} + +define i1 @pr64558(i1 noundef %a, i1 noundef %b) { +; CHECK-LABEL: @pr64558( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[COND_V:%.*]] = or i1 [[B:%.*]], [[A:%.*]] +; CHECK-NEXT: ret i1 [[COND_V]] +; +entry: + %lnot = xor i1 %b, true + %and11 = and i1 %lnot, %a + %cond.v = select i1 %and11, i1 %a, i1 %b + ret i1 %cond.v +}