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 @@ -3578,6 +3578,15 @@ } } + // select(X | Y, X | Y, Y) -> X | Y + ICmpInst::Predicate Pred; + Value *X; + Value *Y; + if (match(CondVal, m_ICmp(Pred, m_Specific(FalseVal), m_Zero())) && + match(FalseVal, m_Or(m_Value(X), m_Value(Y))) && + (match(TrueVal, m_Specific(X)) || match(TrueVal, m_Specific(Y)))) + return replaceInstUsesWith(SI, FalseVal); + // Try to simplify a binop sandwiched between 2 selects with the same // condition. This is not valid for div/rem because the select might be // preventing a division-by-zero. diff --git a/llvm/test/Transforms/InstCombine/select_or.ll b/llvm/test/Transforms/InstCombine/select_or.ll new file mode 100644 --- /dev/null +++ b/llvm/test/Transforms/InstCombine/select_or.ll @@ -0,0 +1,78 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S -passes=instcombine < %s | FileCheck %s + +; select(Y | X == 0, X, Y | X) +define i32 @select_or_1(i32 %0, i32 %1) { +; CHECK-LABEL: @select_or_1( +; CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP1:%.*]], [[TMP0:%.*]] +; CHECK-NEXT: ret i32 [[TMP3]] +; + %3 = or i32 %1, %0 + %4 = icmp eq i32 %3, 0 + %5 = select i1 %4, i32 %0, i32 %3 + ret i32 %5 +} + +; select(Y | X == 0, Y, Y | X) +define i32 @select_or_2(i32 %0, i32 %1) { +; CHECK-LABEL: @select_or_2( +; CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP1:%.*]], [[TMP0:%.*]] +; CHECK-NEXT: ret i32 [[TMP3]] +; + %3 = or i32 %1, %0 + %4 = icmp eq i32 %3, 0 + %5 = select i1 %4, i32 %1, i32 %3 + ret i32 %5 +} + +; select(Y | X != 0, Y | X, X) +define i32 @select_or_3(i32 %0, i32 %1) { +; CHECK-LABEL: @select_or_3( +; CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP1:%.*]], [[TMP0:%.*]] +; CHECK-NEXT: ret i32 [[TMP3]] +; + %3 = or i32 %1, %0 + %4 = icmp ne i32 %3, 0 + %5 = select i1 %4, i32 %3, i32 %0 + ret i32 %5 +} + +; select(Y | X != 0, Y | X, Y) +define i32 @select_or_4(i32 %0, i32 %1) { +; CHECK-LABEL: @select_or_4( +; CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP1:%.*]], [[TMP0:%.*]] +; CHECK-NEXT: ret i32 [[TMP3]] +; + %3 = or i32 %1, %0 + %4 = icmp ne i32 %3, 0 + %5 = select i1 %4, i32 %3, i32 %1 + ret i32 %5 +} + +; select(Y | X == 0, Y | X, Y) +define i32 @select_or_not_1(i32 %0, i32 %1) { +; CHECK-LABEL: @select_or_not_1( +; CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP1:%.*]], [[TMP0:%.*]] +; CHECK-NEXT: [[TMP4:%.*]] = icmp eq i32 [[TMP3]], 0 +; CHECK-NEXT: [[TMP5:%.*]] = select i1 [[TMP4]], i32 0, i32 [[TMP1]] +; CHECK-NEXT: ret i32 [[TMP5]] +; + %3 = or i32 %1, %0 + %4 = icmp eq i32 %3, 0 + %5 = select i1 %4, i32 %3, i32 %1 + ret i32 %5 +} +; select(Y | X != 0, Y, Y | X) +define i32 @select_or_not_2(i32 %0, i32 %1) { +; CHECK-LABEL: @select_or_not_2( +; CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP1:%.*]], [[TMP0:%.*]] +; CHECK-NEXT: [[DOTNOT:%.*]] = icmp eq i32 [[TMP3]], 0 +; CHECK-NEXT: [[TMP4:%.*]] = select i1 [[DOTNOT]], i32 0, i32 [[TMP1]] +; CHECK-NEXT: ret i32 [[TMP4]] +; + %3 = or i32 %1, %0 + %4 = icmp ne i32 %3, 0 + %5 = select i1 %4, i32 %1, i32 %3 + ret i32 %5 +} +