diff options
| author | Sanjay Patel <spatel@rotateright.com> | 2017-04-30 18:57:12 +0000 |
|---|---|---|
| committer | Sanjay Patel <spatel@rotateright.com> | 2017-04-30 18:57:12 +0000 |
| commit | 0c6086f493b0b5208d2e58ad5278322523440ddd (patch) | |
| tree | 6cf210b871f8efabbe0608ec2749e7d8440997b7 /llvm/test/Transforms/InstCombine/demorgan.ll | |
| parent | ad22f42efb2b50211182c711de7ee535fadfc399 (diff) | |
| download | bcm5719-llvm-0c6086f493b0b5208d2e58ad5278322523440ddd.tar.gz bcm5719-llvm-0c6086f493b0b5208d2e58ad5278322523440ddd.zip | |
[InstCombine] consolidate tests for DeMorgan folds; NFC
I'm proposing to add tests and change behavior in D32665.
llvm-svn: 301774
Diffstat (limited to 'llvm/test/Transforms/InstCombine/demorgan.ll')
| -rw-r--r-- | llvm/test/Transforms/InstCombine/demorgan.ll | 320 |
1 files changed, 320 insertions, 0 deletions
diff --git a/llvm/test/Transforms/InstCombine/demorgan.ll b/llvm/test/Transforms/InstCombine/demorgan.ll new file mode 100644 index 00000000000..30413ce521f --- /dev/null +++ b/llvm/test/Transforms/InstCombine/demorgan.ll @@ -0,0 +1,320 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt < %s -instcombine -S | FileCheck %s + +; (~A | ~B) == ~(A & B) + +define i43 @demorgan_or_apint1(i43 %A, i43 %B) { +; CHECK-LABEL: @demorgan_or_apint1( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = and i43 %A, %B +; CHECK-NEXT: [[C:%.*]] = xor i43 [[C_DEMORGAN]], -1 +; CHECK-NEXT: ret i43 [[C]] +; + %NotA = xor i43 %A, -1 + %NotB = xor i43 %B, -1 + %C = or i43 %NotA, %NotB + ret i43 %C +} + +; (~A | ~B) == ~(A & B) + +define i129 @demorgan_or_apint2(i129 %A, i129 %B) { +; CHECK-LABEL: @demorgan_or_apint2( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = and i129 %A, %B +; CHECK-NEXT: [[C:%.*]] = xor i129 [[C_DEMORGAN]], -1 +; CHECK-NEXT: ret i129 [[C]] +; + %NotA = xor i129 %A, -1 + %NotB = xor i129 %B, -1 + %C = or i129 %NotA, %NotB + ret i129 %C +} + +; (~A & ~B) == ~(A | B) + +define i477 @demorgan_and_apint1(i477 %A, i477 %B) { +; CHECK-LABEL: @demorgan_and_apint1( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = or i477 %A, %B +; CHECK-NEXT: [[C:%.*]] = xor i477 [[C_DEMORGAN]], -1 +; CHECK-NEXT: ret i477 [[C]] +; + %NotA = xor i477 %A, -1 + %NotB = xor i477 %B, -1 + %C = and i477 %NotA, %NotB + ret i477 %C +} + +; (~A & ~B) == ~(A | B) + +define i129 @demorgan_and_apint2(i129 %A, i129 %B) { +; CHECK-LABEL: @demorgan_and_apint2( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = or i129 %A, %B +; CHECK-NEXT: [[C:%.*]] = xor i129 [[C_DEMORGAN]], -1 +; CHECK-NEXT: ret i129 [[C]] +; + %NotA = xor i129 %A, -1 + %NotB = xor i129 %B, -1 + %C = and i129 %NotA, %NotB + ret i129 %C +} + +; (~A & ~B) == ~(A | B) + +define i65 @demorgan_and_apint3(i65 %A, i65 %B) { +; CHECK-LABEL: @demorgan_and_apint3( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = or i65 %A, %B +; CHECK-NEXT: [[C:%.*]] = xor i65 [[C_DEMORGAN]], -1 +; CHECK-NEXT: ret i65 [[C]] +; + %NotA = xor i65 %A, -1 + %NotB = xor i65 -1, %B + %C = and i65 %NotA, %NotB + ret i65 %C +} + +; (~A & ~B) == ~(A | B) + +define i66 @demorgan_and_apint4(i66 %A, i66 %B) { +; CHECK-LABEL: @demorgan_and_apint4( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = or i66 %A, %B +; CHECK-NEXT: [[C:%.*]] = xor i66 [[C_DEMORGAN]], -1 +; CHECK-NEXT: ret i66 [[C]] +; + %NotA = xor i66 %A, -1 + %NotB = xor i66 %B, -1 + %C = and i66 %NotA, %NotB + ret i66 %C +} + +; (~A & ~B) == ~(A | B) + +define i47 @demorgan_and_apint5(i47 %A, i47 %B) { +; CHECK-LABEL: @demorgan_and_apint5( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = or i47 %A, %B +; CHECK-NEXT: [[C:%.*]] = xor i47 [[C_DEMORGAN]], -1 +; CHECK-NEXT: ret i47 [[C]] +; + %NotA = xor i47 %A, -1 + %NotB = xor i47 %B, -1 + %C = and i47 %NotA, %NotB + ret i47 %C +} + +; This is confirming that 2 transforms work together: +; ~(~A & ~B) --> A | B + +define i32 @test3(i32 %A, i32 %B) { +; CHECK-LABEL: @test3( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = or i32 %A, %B +; CHECK-NEXT: ret i32 [[C_DEMORGAN]] +; + %nota = xor i32 %A, -1 + %notb = xor i32 %B, -1 + %c = and i32 %nota, %notb + %notc = xor i32 %c, -1 + ret i32 %notc +} + +; Invert a constant if needed: +; ~(~A & 5) --> A | ~5 + +define i32 @test4(i32 %A) { +; CHECK-LABEL: @test4( +; CHECK-NEXT: [[NOTC1:%.*]] = or i32 %A, -6 +; CHECK-NEXT: ret i32 [[NOTC1]] +; + %nota = xor i32 %A, -1 + %c = and i32 %nota, 5 + %notc = xor i32 %c, -1 + ret i32 %notc +} + +; Test the mirror of DeMorgan's law with an extra 'not'. +; ~(~A | ~B) --> A & B + +define i32 @test5(i32 %A, i32 %B) { +; CHECK-LABEL: @test5( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = and i32 %A, %B +; CHECK-NEXT: ret i32 [[C_DEMORGAN]] +; + %nota = xor i32 %A, -1 + %notb = xor i32 %B, -1 + %c = or i32 %nota, %notb + %notc = xor i32 %c, -1 + ret i32 %notc +} + +; Repeat with weird types for extra coverage. +; ~(~A & ~B) --> A | B + +define i47 @test3_apint(i47 %A, i47 %B) { +; CHECK-LABEL: @test3_apint( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = or i47 %A, %B +; CHECK-NEXT: ret i47 [[C_DEMORGAN]] +; + %nota = xor i47 %A, -1 + %notb = xor i47 %B, -1 + %c = and i47 %nota, %notb + %notc = xor i47 %c, -1 + ret i47 %notc +} + +; ~(~A & 5) --> A | ~5 + +define i61 @test4_apint(i61 %A) { +; CHECK-LABEL: @test4_apint( +; CHECK-NEXT: [[NOTA:%.*]] = and i61 %A, 5 +; CHECK-NEXT: [[C:%.*]] = xor i61 [[NOTA]], 5 +; CHECK-NEXT: ret i61 [[C]] +; + %nota = xor i61 %A, -1 + %c = and i61 %nota, 5 ; 5 = ~c2 + %notc = xor i61 %c, -1 + ret i61 %c +} + +; ~(~A | ~B) --> A & B + +define i71 @test5_apint(i71 %A, i71 %B) { +; CHECK-LABEL: @test5_apint( +; CHECK-NEXT: [[C_DEMORGAN:%.*]] = and i71 %A, %B +; CHECK-NEXT: ret i71 [[C_DEMORGAN]] +; + %nota = xor i71 %A, -1 + %notb = xor i71 %B, -1 + %c = or i71 %nota, %notb + %notc = xor i71 %c, -1 + ret i71 %notc +} + +; FIXME: Do not apply DeMorgan's Law to constants. We prefer 'not' ops. + +define i32 @demorganize_constant1(i32 %a) { +; CHECK-LABEL: @demorganize_constant1( +; CHECK-NEXT: [[A_NOT:%.*]] = or i32 %a, -16 +; CHECK-NEXT: [[AND1:%.*]] = xor i32 [[A_NOT]], 15 +; CHECK-NEXT: ret i32 [[AND1]] +; + %and = and i32 %a, 15 + %and1 = xor i32 %and, -1 + ret i32 %and1 +} + +; FIXME: Do not apply DeMorgan's Law to constants. We prefer 'not' ops. + +define i32 @demorganize_constant2(i32 %a) { +; CHECK-LABEL: @demorganize_constant2( +; CHECK-NEXT: [[A_NOT:%.*]] = and i32 %a, -16 +; CHECK-NEXT: [[AND1:%.*]] = xor i32 [[A_NOT]], -16 +; CHECK-NEXT: ret i32 [[AND1]] +; + %and = or i32 %a, 15 + %and1 = xor i32 %and, -1 + ret i32 %and1 +} + +; PR22723: Recognize DeMorgan's Laws when obfuscated by zexts. + +define i32 @demorgan_or_zext(i1 %X, i1 %Y) { +; CHECK-LABEL: @demorgan_or_zext( +; CHECK-NEXT: [[OR1_DEMORGAN:%.*]] = and i1 %X, %Y +; CHECK-NEXT: [[OR1:%.*]] = xor i1 [[OR1_DEMORGAN]], true +; CHECK-NEXT: [[OR:%.*]] = zext i1 [[OR:%.*]]1 to i32 +; CHECK-NEXT: ret i32 [[OR]] +; + %zextX = zext i1 %X to i32 + %zextY = zext i1 %Y to i32 + %notX = xor i32 %zextX, 1 + %notY = xor i32 %zextY, 1 + %or = or i32 %notX, %notY + ret i32 %or +} + +define i32 @demorgan_and_zext(i1 %X, i1 %Y) { +; CHECK-LABEL: @demorgan_and_zext( +; CHECK-NEXT: [[AND1_DEMORGAN:%.*]] = or i1 %X, %Y +; CHECK-NEXT: [[AND1:%.*]] = xor i1 [[AND1_DEMORGAN]], true +; CHECK-NEXT: [[AND:%.*]] = zext i1 [[AND:%.*]]1 to i32 +; CHECK-NEXT: ret i32 [[AND]] +; + %zextX = zext i1 %X to i32 + %zextY = zext i1 %Y to i32 + %notX = xor i32 %zextX, 1 + %notY = xor i32 %zextY, 1 + %and = and i32 %notX, %notY + ret i32 %and +} + +define <2 x i32> @demorgan_or_zext_vec(<2 x i1> %X, <2 x i1> %Y) { +; CHECK-LABEL: @demorgan_or_zext_vec( +; CHECK-NEXT: [[OR1_DEMORGAN:%.*]] = and <2 x i1> %X, %Y +; CHECK-NEXT: [[OR1:%.*]] = xor <2 x i1> [[OR1_DEMORGAN]], <i1 true, i1 true> +; CHECK-NEXT: [[OR:%.*]] = zext <2 x i1> [[OR:%.*]]1 to <2 x i32> +; CHECK-NEXT: ret <2 x i32> [[OR]] +; + %zextX = zext <2 x i1> %X to <2 x i32> + %zextY = zext <2 x i1> %Y to <2 x i32> + %notX = xor <2 x i32> %zextX, <i32 1, i32 1> + %notY = xor <2 x i32> %zextY, <i32 1, i32 1> + %or = or <2 x i32> %notX, %notY + ret <2 x i32> %or +} + +define <2 x i32> @demorgan_and_zext_vec(<2 x i1> %X, <2 x i1> %Y) { +; CHECK-LABEL: @demorgan_and_zext_vec( +; CHECK-NEXT: [[AND1_DEMORGAN:%.*]] = or <2 x i1> %X, %Y +; CHECK-NEXT: [[AND1:%.*]] = xor <2 x i1> [[AND1_DEMORGAN]], <i1 true, i1 true> +; CHECK-NEXT: [[AND:%.*]] = zext <2 x i1> [[AND:%.*]]1 to <2 x i32> +; CHECK-NEXT: ret <2 x i32> [[AND]] +; + %zextX = zext <2 x i1> %X to <2 x i32> + %zextY = zext <2 x i1> %Y to <2 x i32> + %notX = xor <2 x i32> %zextX, <i32 1, i32 1> + %notY = xor <2 x i32> %zextY, <i32 1, i32 1> + %and = and <2 x i32> %notX, %notY + ret <2 x i32> %and +} + +define i32 @PR28476(i32 %x, i32 %y) { +; CHECK-LABEL: @PR28476( +; CHECK-NEXT: [[CMP0:%.*]] = icmp eq i32 %x, 0 +; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i32 %y, 0 +; CHECK-NEXT: [[TMP1:%.*]] = or i1 [[CMP1]], [[CMP0]] +; CHECK-NEXT: [[COND:%.*]] = zext i1 [[TMP1]] to i32 +; CHECK-NEXT: ret i32 [[COND]] +; + %cmp0 = icmp ne i32 %x, 0 + %cmp1 = icmp ne i32 %y, 0 + %and = and i1 %cmp0, %cmp1 + %zext = zext i1 %and to i32 + %cond = xor i32 %zext, 1 + ret i32 %cond +} + +; ~(~(a | b) | (a & b)) --> (a | b) & ~(a & b) -> a ^ b + +define i32 @demorgan_plus_and_to_xor(i32 %a, i32 %b) { +; CHECK-LABEL: @demorgan_plus_and_to_xor( +; CHECK-NEXT: [[NOT:%.*]] = xor i32 %b, %a +; CHECK-NEXT: ret i32 [[NOT]] +; + %or = or i32 %b, %a + %notor = xor i32 %or, -1 + %and = and i32 %b, %a + %or2 = or i32 %and, %notor + %not = xor i32 %or2, -1 + ret i32 %not +} + +define <4 x i32> @demorgan_plus_and_to_xor_vec(<4 x i32> %a, <4 x i32> %b) { +; CHECK-LABEL: @demorgan_plus_and_to_xor_vec( +; CHECK-NEXT: [[NOT:%.*]] = xor <4 x i32> %a, %b +; CHECK-NEXT: ret <4 x i32> [[NOT]] +; + %or = or <4 x i32> %a, %b + %notor = xor <4 x i32> %or, < i32 -1, i32 -1, i32 -1, i32 -1 > + %and = and <4 x i32> %a, %b + %or2 = or <4 x i32> %and, %notor + %not = xor <4 x i32> %or2, < i32 -1, i32 -1, i32 -1, i32 -1 > + ret <4 x i32> %not +} + |

