From c9df51e6e959f81e451b5d65dfc274c3ef06d4c2 Mon Sep 17 00:00:00 2001 From: Craig Topper Date: Fri, 30 Jun 2017 07:37:42 +0000 Subject: [PATCH] [InstCombine] Add test cases to demonstrate failure to fold (a | b) ^ (~a | ~b) --> ~(a ^ b) and its commuted variants. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@306801 91177308-0d34-0410-b5e6-96231b3b80d8 --- test/Transforms/InstCombine/and-or-not.ll | 84 +++++++++++++++++++++++ test/Transforms/InstCombine/or-xor.ll | 68 ++++++++++++++++++ 2 files changed, 152 insertions(+) diff --git a/test/Transforms/InstCombine/and-or-not.ll b/test/Transforms/InstCombine/and-or-not.ll index f5fab1b5b55..16a23acb400 100644 --- a/test/Transforms/InstCombine/and-or-not.ll +++ b/test/Transforms/InstCombine/and-or-not.ll @@ -564,3 +564,87 @@ define i32 @or_to_nxor_multiuse(i32 %a, i32 %b) { %mul2 = mul i32 %mul1, %or2 ret i32 %mul2 } + +; (a | b) ^ (~a | ~b) --> ~(a ^ b) +define i32 @xor_to_xnor1(float %fa, float %fb) { +; CHECK-LABEL: @xor_to_xnor1( +; CHECK-NEXT: [[A:%.*]] = fptosi float [[FA:%.*]] to i32 +; CHECK-NEXT: [[B:%.*]] = fptosi float [[FB:%.*]] to i32 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A]], [[B]] +; CHECK-NEXT: [[OR2_DEMORGAN:%.*]] = and i32 [[A]], [[B]] +; CHECK-NEXT: [[OR2:%.*]] = xor i32 [[OR2_DEMORGAN]], -1 +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR1]], [[OR2]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %a = fptosi float %fa to i32 + %b = fptosi float %fb to i32 + %nota = xor i32 %a, -1 + %notb = xor i32 %b, -1 + %or1 = or i32 %a, %b + %or2 = or i32 %nota, %notb + %xor = xor i32 %or1, %or2 + ret i32 %xor +} + +; (a | b) ^ (~b | ~a) --> ~(a ^ b) +define i32 @xor_to_xnor2(float %fa, float %fb) { +; CHECK-LABEL: @xor_to_xnor2( +; CHECK-NEXT: [[A:%.*]] = fptosi float [[FA:%.*]] to i32 +; CHECK-NEXT: [[B:%.*]] = fptosi float [[FB:%.*]] to i32 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A]], [[B]] +; CHECK-NEXT: [[OR2_DEMORGAN:%.*]] = and i32 [[B]], [[A]] +; CHECK-NEXT: [[OR2:%.*]] = xor i32 [[OR2_DEMORGAN]], -1 +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR1]], [[OR2]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %a = fptosi float %fa to i32 + %b = fptosi float %fb to i32 + %nota = xor i32 %a, -1 + %notb = xor i32 %b, -1 + %or1 = or i32 %a, %b + %or2 = or i32 %notb, %nota + %xor = xor i32 %or1, %or2 + ret i32 %xor +} + +; (~a | ~b) ^ (a | b) --> ~(a ^ b) +define i32 @xor_to_xnor3(float %fa, float %fb) { +; CHECK-LABEL: @xor_to_xnor3( +; CHECK-NEXT: [[A:%.*]] = fptosi float [[FA:%.*]] to i32 +; CHECK-NEXT: [[B:%.*]] = fptosi float [[FB:%.*]] to i32 +; CHECK-NEXT: [[OR1_DEMORGAN:%.*]] = and i32 [[A]], [[B]] +; CHECK-NEXT: [[OR1:%.*]] = xor i32 [[OR1_DEMORGAN]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[B]] +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR2]], [[OR1]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %a = fptosi float %fa to i32 + %b = fptosi float %fb to i32 + %nota = xor i32 %a, -1 + %notb = xor i32 %b, -1 + %or1 = or i32 %nota, %notb + %or2 = or i32 %a, %b + %xor = xor i32 %or1, %or2 + ret i32 %xor +} + +; (~a | ~b) ^ (b | a) --> ~(a ^ b) +define i32 @xor_to_xnor4(float %fa, float %fb) { +; CHECK-LABEL: @xor_to_xnor4( +; CHECK-NEXT: [[A:%.*]] = fptosi float [[FA:%.*]] to i32 +; CHECK-NEXT: [[B:%.*]] = fptosi float [[FB:%.*]] to i32 +; CHECK-NEXT: [[OR1_DEMORGAN:%.*]] = and i32 [[A]], [[B]] +; CHECK-NEXT: [[OR1:%.*]] = xor i32 [[OR1_DEMORGAN]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[B]], [[A]] +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR2]], [[OR1]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %a = fptosi float %fa to i32 + %b = fptosi float %fb to i32 + %nota = xor i32 %a, -1 + %notb = xor i32 %b, -1 + %or1 = or i32 %nota, %notb + %or2 = or i32 %b, %a + %xor = xor i32 %or1, %or2 + ret i32 %xor +} diff --git a/test/Transforms/InstCombine/or-xor.ll b/test/Transforms/InstCombine/or-xor.ll index cff92a51a11..2164f0df8d2 100644 --- a/test/Transforms/InstCombine/or-xor.ll +++ b/test/Transforms/InstCombine/or-xor.ll @@ -344,3 +344,71 @@ define i8 @test18(i8 %A, i8 %B) { %res = mul i8 %or, %xor2 ; to increase the use count for the xor ret i8 %res } + +; ((x | y) ^ (~x | ~y)) -> ~(x ^ y) +define i32 @test19(i32 %x, i32 %y) { +; CHECK-LABEL: @test19( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: [[OR2_DEMORGAN:%.*]] = and i32 [[X]], [[Y]] +; CHECK-NEXT: [[OR2:%.*]] = xor i32 [[OR2_DEMORGAN]], -1 +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR1]], [[OR2]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %noty = xor i32 %y, -1 + %notx = xor i32 %x, -1 + %or1 = or i32 %x, %y + %or2 = or i32 %notx, %noty + %xor = xor i32 %or1, %or2 + ret i32 %xor +} + +; ((x | y) ^ (~y | ~x)) -> ~(x ^ y) +define i32 @test20(i32 %x, i32 %y) { +; CHECK-LABEL: @test20( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: [[OR2_DEMORGAN:%.*]] = and i32 [[Y]], [[X]] +; CHECK-NEXT: [[OR2:%.*]] = xor i32 [[OR2_DEMORGAN]], -1 +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR1]], [[OR2]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %noty = xor i32 %y, -1 + %notx = xor i32 %x, -1 + %or1 = or i32 %x, %y + %or2 = or i32 %noty, %notx + %xor = xor i32 %or1, %or2 + ret i32 %xor +} + +; ((~x | ~y) ^ (x | y)) -> ~(x ^ y) +define i32 @test21(i32 %x, i32 %y) { +; CHECK-LABEL: @test21( +; CHECK-NEXT: [[OR1_DEMORGAN:%.*]] = and i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: [[OR1:%.*]] = xor i32 [[OR1_DEMORGAN]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[X]], [[Y]] +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR2]], [[OR1]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %noty = xor i32 %y, -1 + %notx = xor i32 %x, -1 + %or1 = or i32 %notx, %noty + %or2 = or i32 %x, %y + %xor = xor i32 %or1, %or2 + ret i32 %xor +} + +; ((~x | ~y) ^ (y | x)) -> ~(x ^ y) +define i32 @test22(i32 %x, i32 %y) { +; CHECK-LABEL: @test22( +; CHECK-NEXT: [[OR1_DEMORGAN:%.*]] = and i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: [[OR1:%.*]] = xor i32 [[OR1_DEMORGAN]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[Y]], [[X]] +; CHECK-NEXT: [[XOR:%.*]] = xor i32 [[OR2]], [[OR1]] +; CHECK-NEXT: ret i32 [[XOR]] +; + %noty = xor i32 %y, -1 + %notx = xor i32 %x, -1 + %or1 = or i32 %notx, %noty + %or2 = or i32 %y, %x + %xor = xor i32 %or1, %or2 + ret i32 %xor +} -- 2.50.0