Skip to content

Commit e1bd759

Browse files
committed
[InstCombine] allow more matches for logical-ands --> select
This allows patterns with real 'and' instructions because those are safe to transform: https://alive2.llvm.org/ce/z/7-U_Ak
1 parent f010b1e commit e1bd759

File tree

2 files changed

+16
-20
lines changed

2 files changed

+16
-20
lines changed

llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp

Lines changed: 11 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -2852,18 +2852,21 @@ Instruction *InstCombinerImpl::visitSelectInst(SelectInst &SI) {
28522852

28532853
// (C && A) || (!C && B) --> sel C, A, B
28542854
// (A && C) || (!C && B) --> sel C, A, B
2855-
if (match(FalseVal, m_LogicalAnd(m_Not(m_Value(C)), m_Value(B))) &&
2856-
match(CondVal, m_c_LogicalAnd(m_Specific(C), m_Value(A))))
2857-
return SelectInst::Create(C, A, B);
2858-
28592855
// (C && A) || (B && !C) --> sel C, A, B
2860-
// TODO: (A && C) || (B && !C) is safe to transform with real 'and' ops.
2861-
if (match(FalseVal, m_LogicalAnd(m_Value(B), m_Not(m_Value(C)))) &&
2862-
match(CondVal, m_LogicalAnd(m_Specific(C), m_Value(A))))
2863-
return SelectInst::Create(C, A, B);
2856+
// (A && C) || (B && !C) --> sel C, A, B (only with real 'and' ops)
2857+
if (match(FalseVal, m_c_LogicalAnd(m_Not(m_Value(C)), m_Value(B))) &&
2858+
match(CondVal, m_c_LogicalAnd(m_Specific(C), m_Value(A)))) {
2859+
auto *SelCond = dyn_cast<SelectInst>(CondVal);
2860+
auto *SelFVal = dyn_cast<SelectInst>(FalseVal);
2861+
if (!SelCond || !SelFVal ||
2862+
!match(SelFVal->getTrueValue(),
2863+
m_Not(m_Specific(SelCond->getTrueValue()))))
2864+
return SelectInst::Create(C, A, B);
2865+
}
28642866

28652867
// (!C && A) || (C && B) --> sel C, B, A
28662868
// (!C && A) || (B && C) --> sel C, B, A
2869+
// TODO: Allow more commutes as with the previous fold.
28672870
if (match(CondVal, m_LogicalAnd(m_Not(m_Value(C)), m_Value(A))) &&
28682871
match(FalseVal, m_c_LogicalAnd(m_Specific(C), m_Value(B))))
28692872
return SelectInst::Create(C, B, A);

llvm/test/Transforms/InstCombine/select-safe-transforms.ll

Lines changed: 5 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -529,6 +529,8 @@ define i1 @bools2_logical_commute2_and1_and2(i1 %a, i1 %c) {
529529
ret i1 %or
530530
}
531531

532+
; This is not safe to transform if 'c' could be poison.
533+
532534
define i1 @bools2_logical_commute3(i1 %a, i1 %b, i1 %c) {
533535
; CHECK-LABEL: @bools2_logical_commute3(
534536
; CHECK-NEXT: [[NOT:%.*]] = xor i1 [[C:%.*]], true
@@ -546,10 +548,7 @@ define i1 @bools2_logical_commute3(i1 %a, i1 %b, i1 %c) {
546548

547549
define i1 @bools2_logical_commute3_and1(i1 %a, i1 %b, i1 %c) {
548550
; CHECK-LABEL: @bools2_logical_commute3_and1(
549-
; CHECK-NEXT: [[NOT:%.*]] = xor i1 [[C:%.*]], true
550-
; CHECK-NEXT: [[AND1:%.*]] = and i1 [[A:%.*]], [[C]]
551-
; CHECK-NEXT: [[AND2:%.*]] = select i1 [[B:%.*]], i1 [[NOT]], i1 false
552-
; CHECK-NEXT: [[OR:%.*]] = select i1 [[AND1]], i1 true, i1 [[AND2]]
551+
; CHECK-NEXT: [[OR:%.*]] = select i1 [[C:%.*]], i1 [[A:%.*]], i1 [[B:%.*]]
553552
; CHECK-NEXT: ret i1 [[OR]]
554553
;
555554
%not = xor i1 %c, -1
@@ -562,10 +561,7 @@ define i1 @bools2_logical_commute3_and1(i1 %a, i1 %b, i1 %c) {
562561
define i1 @bools2_logical_commute3_and2(i1 %a, i1 %c) {
563562
; CHECK-LABEL: @bools2_logical_commute3_and2(
564563
; CHECK-NEXT: [[B:%.*]] = call i1 @gen1()
565-
; CHECK-NEXT: [[NOT:%.*]] = xor i1 [[C:%.*]], true
566-
; CHECK-NEXT: [[AND1:%.*]] = select i1 [[A:%.*]], i1 [[C]], i1 false
567-
; CHECK-NEXT: [[AND2:%.*]] = and i1 [[B]], [[NOT]]
568-
; CHECK-NEXT: [[OR:%.*]] = select i1 [[AND1]], i1 true, i1 [[AND2]]
564+
; CHECK-NEXT: [[OR:%.*]] = select i1 [[C:%.*]], i1 [[A:%.*]], i1 [[B]]
569565
; CHECK-NEXT: ret i1 [[OR]]
570566
;
571567
%b = call i1 @gen1()
@@ -579,10 +575,7 @@ define i1 @bools2_logical_commute3_and2(i1 %a, i1 %c) {
579575
define i1 @bools2_logical_commute3_and1_and2(i1 %a, i1 %c) {
580576
; CHECK-LABEL: @bools2_logical_commute3_and1_and2(
581577
; CHECK-NEXT: [[B:%.*]] = call i1 @gen1()
582-
; CHECK-NEXT: [[NOT:%.*]] = xor i1 [[C:%.*]], true
583-
; CHECK-NEXT: [[AND1:%.*]] = and i1 [[A:%.*]], [[C]]
584-
; CHECK-NEXT: [[AND2:%.*]] = and i1 [[B]], [[NOT]]
585-
; CHECK-NEXT: [[OR:%.*]] = select i1 [[AND1]], i1 true, i1 [[AND2]]
578+
; CHECK-NEXT: [[OR:%.*]] = select i1 [[C:%.*]], i1 [[A:%.*]], i1 [[B]]
586579
; CHECK-NEXT: ret i1 [[OR]]
587580
;
588581
%b = call i1 @gen1()

0 commit comments

Comments
 (0)