mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-01-18 01:35:44 +00:00
InstCombine: Add a bunch of combines of the form x | (y ^ z).
We usually catch this kind of optimization through InstSimplify's distributive magic, but or doesn't distribute over xor in general. "A | ~(A | B) -> A | ~B" hits 24 times on gcc.c. llvm-svn: 126081
This commit is contained in:
parent
1660c0bc25
commit
50cd35c25e
@ -1897,6 +1897,47 @@ Instruction *InstCombiner::visitOr(BinaryOperator &I) {
|
||||
return BinaryOperator::CreateNot(And);
|
||||
}
|
||||
|
||||
// Canonicalize xor to the RHS.
|
||||
if (match(Op0, m_Xor(m_Value(), m_Value())))
|
||||
std::swap(Op0, Op1);
|
||||
|
||||
// A | ( A ^ B) -> A | B
|
||||
// A | (~A ^ B) -> A | ~B
|
||||
if (match(Op1, m_Xor(m_Value(A), m_Value(B)))) {
|
||||
if (Op0 == A || Op0 == B)
|
||||
return BinaryOperator::CreateOr(A, B);
|
||||
|
||||
if (Op1->hasOneUse() && match(A, m_Not(m_Specific(Op0)))) {
|
||||
Value *Not = Builder->CreateNot(B, B->getName()+".not");
|
||||
return BinaryOperator::CreateOr(Not, Op0);
|
||||
}
|
||||
if (Op1->hasOneUse() && match(B, m_Not(m_Specific(Op0)))) {
|
||||
Value *Not = Builder->CreateNot(A, A->getName()+".not");
|
||||
return BinaryOperator::CreateOr(Not, Op0);
|
||||
}
|
||||
}
|
||||
|
||||
// A | ~(A | B) -> A | ~B
|
||||
// A | ~(A ^ B) -> A | ~B
|
||||
// A | ~(A & B) -> -1
|
||||
if (match(Op1, m_Not(m_Value(A))))
|
||||
if (BinaryOperator *B = dyn_cast<BinaryOperator>(A))
|
||||
if (Op0 == B->getOperand(0) || Op0 == B->getOperand(1))
|
||||
switch (B->getOpcode()) {
|
||||
default: break;
|
||||
case Instruction::Or:
|
||||
case Instruction::Xor:
|
||||
if (Op1->hasOneUse()) {
|
||||
Value *NotOp = Op0 == B->getOperand(0) ? B->getOperand(1) :
|
||||
B->getOperand(0);
|
||||
Value *Not = Builder->CreateNot(NotOp, NotOp->getName()+".not");
|
||||
return BinaryOperator::CreateOr(Not, Op0);
|
||||
}
|
||||
break;
|
||||
case Instruction::And:
|
||||
return ReplaceInstUsesWith(I, Constant::getAllOnesValue(I.getType()));
|
||||
}
|
||||
|
||||
if (ICmpInst *RHS = dyn_cast<ICmpInst>(I.getOperand(1)))
|
||||
if (ICmpInst *LHS = dyn_cast<ICmpInst>(I.getOperand(0)))
|
||||
if (Value *Res = FoldOrOfICmps(LHS, RHS))
|
||||
|
94
test/Transforms/InstCombine/or-xor.ll
Normal file
94
test/Transforms/InstCombine/or-xor.ll
Normal file
@ -0,0 +1,94 @@
|
||||
; RUN: opt -S -instcombine < %s | FileCheck %s
|
||||
|
||||
define i32 @test1(i32 %x, i32 %y) nounwind {
|
||||
%or = or i32 %x, %y
|
||||
%not = xor i32 %or, -1
|
||||
%z = or i32 %x, %not
|
||||
ret i32 %z
|
||||
; CHECK: @test1
|
||||
; CHECK-NEXT: %y.not = xor i32 %y, -1
|
||||
; CHECK-NEXT: %z = or i32 %y.not, %x
|
||||
; CHECK-NEXT: ret i32 %z
|
||||
}
|
||||
|
||||
define i32 @test2(i32 %x, i32 %y) nounwind {
|
||||
%or = or i32 %x, %y
|
||||
%not = xor i32 %or, -1
|
||||
%z = or i32 %y, %not
|
||||
ret i32 %z
|
||||
; CHECK: @test2
|
||||
; CHECK-NEXT: %x.not = xor i32 %x, -1
|
||||
; CHECK-NEXT: %z = or i32 %x.not, %y
|
||||
; CHECK-NEXT: ret i32 %z
|
||||
}
|
||||
|
||||
define i32 @test3(i32 %x, i32 %y) nounwind {
|
||||
%xor = xor i32 %x, %y
|
||||
%not = xor i32 %xor, -1
|
||||
%z = or i32 %x, %not
|
||||
ret i32 %z
|
||||
; CHECK: @test3
|
||||
; CHECK-NEXT: %y.not = xor i32 %y, -1
|
||||
; CHECK-NEXT: %z = or i32 %y.not, %x
|
||||
; CHECK-NEXT: ret i32 %z
|
||||
}
|
||||
|
||||
define i32 @test4(i32 %x, i32 %y) nounwind {
|
||||
%xor = xor i32 %x, %y
|
||||
%not = xor i32 %xor, -1
|
||||
%z = or i32 %y, %not
|
||||
ret i32 %z
|
||||
; CHECK: @test4
|
||||
; CHECK-NEXT: %x.not = xor i32 %x, -1
|
||||
; CHECK-NEXT: %z = or i32 %x.not, %y
|
||||
; CHECK-NEXT: ret i32 %z
|
||||
}
|
||||
|
||||
define i32 @test5(i32 %x, i32 %y) nounwind {
|
||||
%and = and i32 %x, %y
|
||||
%not = xor i32 %and, -1
|
||||
%z = or i32 %x, %not
|
||||
ret i32 %z
|
||||
; CHECK: @test5
|
||||
; CHECK-NEXT: ret i32 -1
|
||||
}
|
||||
|
||||
define i32 @test6(i32 %x, i32 %y) nounwind {
|
||||
%and = and i32 %x, %y
|
||||
%not = xor i32 %and, -1
|
||||
%z = or i32 %y, %not
|
||||
ret i32 %z
|
||||
; CHECK: @test6
|
||||
; CHECK-NEXT: ret i32 -1
|
||||
}
|
||||
|
||||
define i32 @test7(i32 %x, i32 %y) nounwind {
|
||||
%xor = xor i32 %x, %y
|
||||
%z = or i32 %y, %xor
|
||||
ret i32 %z
|
||||
; CHECK: @test7
|
||||
; CHECK-NEXT: %z = or i32 %x, %y
|
||||
; CHECK-NEXT: ret i32 %z
|
||||
}
|
||||
|
||||
define i32 @test8(i32 %x, i32 %y) nounwind {
|
||||
%not = xor i32 %y, -1
|
||||
%xor = xor i32 %x, %not
|
||||
%z = or i32 %y, %xor
|
||||
ret i32 %z
|
||||
; CHECK: @test8
|
||||
; CHECK-NEXT: %x.not = xor i32 %x, -1
|
||||
; CHECK-NEXT: %z = or i32 %x.not, %y
|
||||
; CHECK-NEXT: ret i32 %z
|
||||
}
|
||||
|
||||
define i32 @test9(i32 %x, i32 %y) nounwind {
|
||||
%not = xor i32 %x, -1
|
||||
%xor = xor i32 %not, %y
|
||||
%z = or i32 %x, %xor
|
||||
ret i32 %z
|
||||
; CHECK: @test9
|
||||
; CHECK-NEXT: %y.not = xor i32 %y, -1
|
||||
; CHECK-NEXT: %z = or i32 %y.not, %x
|
||||
; CHECK-NEXT: ret i32 %z
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user