[llvm] 052b77e - [InstCombine] Add some extra tests for truncated saturates. NFC

David Green via llvm-commits llvm-commits at lists.llvm.org
Sun Oct 17 05:21:34 PDT 2021


Author: David Green
Date: 2021-10-17T13:21:28+01:00
New Revision: 052b77e49f50e5796ee84a55c8a409b3ee41af22

URL: https://github.com/llvm/llvm-project/commit/052b77e49f50e5796ee84a55c8a409b3ee41af22
DIFF: https://github.com/llvm/llvm-project/commit/052b77e49f50e5796ee84a55c8a409b3ee41af22.diff

LOG: [InstCombine] Add some extra tests for truncated saturates. NFC

Added: 
    llvm/test/Transforms/InstCombine/truncating-saturate.ll

Modified: 
    

Removed: 
    


################################################################################
diff  --git a/llvm/test/Transforms/InstCombine/truncating-saturate.ll b/llvm/test/Transforms/InstCombine/truncating-saturate.ll
new file mode 100644
index 0000000000000..07899b9490cf1
--- /dev/null
+++ b/llvm/test/Transforms/InstCombine/truncating-saturate.ll
@@ -0,0 +1,585 @@
+; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
+; RUN: opt < %s -instcombine -S | FileCheck %s
+target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64"
+
+declare void @use(i32)
+declare void @use1(i1)
+
+define i8 @testi16i8(i16 %add) {
+; CHECK-LABEL: @testi16i8(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i16 [[ADD:%.*]], 8
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i16 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i16 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 7
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i16 [[ADD]], 15
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i16 [[SHR4_I]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 127
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i16 %add, 8
+  %conv.i = trunc i16 %sh to i8
+  %conv1.i = trunc i16 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 7
+  %cmp.not.i = icmp eq i8 %shr2.i, %conv.i
+  %shr4.i = ashr i16 %add, 15
+  %conv5.i = trunc i16 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 127
+  %cond.i = select i1 %cmp.not.i, i8 %conv1.i, i8 %xor.i
+  ret i8 %cond.i
+}
+
+define i32 @testi64i32(i64 %add) {
+; CHECK-LABEL: @testi64i32(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i64 [[ADD:%.*]], 32
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i64 [[SH]] to i32
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i64 [[ADD]] to i32
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i32 [[CONV1_I]], 31
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i32 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i64 [[ADD]], 63
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i64 [[SHR4_I]] to i32
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i32 [[CONV5_I]], 2147483647
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i32 [[CONV1_I]], i32 [[XOR_I]]
+; CHECK-NEXT:    ret i32 [[COND_I]]
+;
+  %sh = lshr i64 %add, 32
+  %conv.i = trunc i64 %sh to i32
+  %conv1.i = trunc i64 %add to i32
+  %shr2.i = ashr i32 %conv1.i, 31
+  %cmp.not.i = icmp eq i32 %shr2.i, %conv.i
+  %shr4.i = ashr i64 %add, 63
+  %conv5.i = trunc i64 %shr4.i to i32
+  %xor.i = xor i32 %conv5.i, 2147483647
+  %cond.i = select i1 %cmp.not.i, i32 %conv1.i, i32 %xor.i
+  ret i32 %cond.i
+}
+
+define i16 @testi32i16i8(i32 %add) {
+; CHECK-LABEL: @testi32i16i8(
+; CHECK-NEXT:    [[A:%.*]] = add i32 [[ADD:%.*]], 128
+; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A]], 256
+; CHECK-NEXT:    [[T:%.*]] = trunc i32 [[ADD]] to i16
+; CHECK-NEXT:    [[C:%.*]] = icmp sgt i32 [[ADD]], -1
+; CHECK-NEXT:    [[F:%.*]] = select i1 [[C]], i16 127, i16 -128
+; CHECK-NEXT:    [[R:%.*]] = select i1 [[CMP]], i16 [[T]], i16 [[F]]
+; CHECK-NEXT:    ret i16 [[R]]
+;
+  %a = add i32 %add, 128
+  %cmp = icmp ult i32 %a, 256
+  %t = trunc i32 %add to i16
+  %c = icmp sgt i32 %add, -1
+  %f = select i1 %c, i16 127, i16 -128
+  %r = select i1 %cmp, i16 %t, i16 %f
+  ret i16 %r
+}
+
+define <4 x i16> @testv4i32i16i8(<4 x i32> %add) {
+; CHECK-LABEL: @testv4i32i16i8(
+; CHECK-NEXT:    [[A:%.*]] = add <4 x i32> [[ADD:%.*]], <i32 128, i32 128, i32 128, i32 128>
+; CHECK-NEXT:    [[CMP:%.*]] = icmp ult <4 x i32> [[A]], <i32 256, i32 256, i32 256, i32 256>
+; CHECK-NEXT:    [[T:%.*]] = trunc <4 x i32> [[ADD]] to <4 x i16>
+; CHECK-NEXT:    [[C:%.*]] = icmp sgt <4 x i32> [[ADD]], <i32 -1, i32 -1, i32 -1, i32 -1>
+; CHECK-NEXT:    [[F:%.*]] = select <4 x i1> [[C]], <4 x i16> <i16 127, i16 127, i16 127, i16 127>, <4 x i16> <i16 -128, i16 -128, i16 -128, i16 -128>
+; CHECK-NEXT:    [[R:%.*]] = select <4 x i1> [[CMP]], <4 x i16> [[T]], <4 x i16> [[F]]
+; CHECK-NEXT:    ret <4 x i16> [[R]]
+;
+  %a = add <4 x i32> %add, <i32 128, i32 128, i32 128, i32 128>
+  %cmp = icmp ult <4 x i32> %a, <i32 256, i32 256, i32 256, i32 256>
+  %t = trunc <4 x i32> %add to <4 x i16>
+  %c = icmp sgt <4 x i32> %add, <i32 -1, i32 -1, i32 -1, i32 -1>
+  %f = select <4 x i1> %c, <4 x i16> <i16 127, i16 127, i16 127, i16 127>, <4 x i16> <i16 -128, i16 -128, i16 -128, i16 -128>
+  %r = select <4 x i1> %cmp, <4 x i16> %t, <4 x i16> %f
+  ret <4 x i16> %r
+}
+
+define i32 @testi32i32i8(i32 %add) {
+; CHECK-LABEL: @testi32i32i8(
+; CHECK-NEXT:    [[TMP1:%.*]] = icmp sgt i32 [[ADD:%.*]], -128
+; CHECK-NEXT:    [[TMP2:%.*]] = select i1 [[TMP1]], i32 [[ADD]], i32 -128
+; CHECK-NEXT:    [[TMP3:%.*]] = icmp slt i32 [[TMP2]], 127
+; CHECK-NEXT:    [[R:%.*]] = select i1 [[TMP3]], i32 [[TMP2]], i32 127
+; CHECK-NEXT:    ret i32 [[R]]
+;
+  %a = add i32 %add, 128
+  %cmp = icmp ult i32 %a, 256
+  %c = icmp sgt i32 %add, -1
+  %f = select i1 %c, i32 127, i32 -128
+  %r = select i1 %cmp, i32 %add, i32 %f
+  ret i32 %r
+}
+
+define i16 @test_truncfirst(i32 %add) {
+; CHECK-LABEL: @test_truncfirst(
+; CHECK-NEXT:    [[T:%.*]] = trunc i32 [[ADD:%.*]] to i16
+; CHECK-NEXT:    [[TMP1:%.*]] = icmp sgt i16 [[T]], -128
+; CHECK-NEXT:    [[TMP2:%.*]] = select i1 [[TMP1]], i16 [[T]], i16 -128
+; CHECK-NEXT:    [[TMP3:%.*]] = icmp slt i16 [[TMP2]], 127
+; CHECK-NEXT:    [[R:%.*]] = select i1 [[TMP3]], i16 [[TMP2]], i16 127
+; CHECK-NEXT:    ret i16 [[R]]
+;
+  %t = trunc i32 %add to i16
+  %a = add i16 %t, 128
+  %cmp = icmp ult i16 %a, 256
+  %c = icmp sgt i16 %t, -1
+  %f = select i1 %c, i16 127, i16 -128
+  %r = select i1 %cmp, i16 %t, i16 %f
+  ret i16 %r
+}
+
+define i16 @testtrunclowhigh(i32 %add, i16 %low, i16 %high) {
+; CHECK-LABEL: @testtrunclowhigh(
+; CHECK-NEXT:    [[A:%.*]] = add i32 [[ADD:%.*]], 128
+; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A]], 256
+; CHECK-NEXT:    [[T:%.*]] = trunc i32 [[ADD]] to i16
+; CHECK-NEXT:    [[C:%.*]] = icmp sgt i32 [[ADD]], -1
+; CHECK-NEXT:    [[F:%.*]] = select i1 [[C]], i16 [[HIGH:%.*]], i16 [[LOW:%.*]]
+; CHECK-NEXT:    [[R:%.*]] = select i1 [[CMP]], i16 [[T]], i16 [[F]]
+; CHECK-NEXT:    ret i16 [[R]]
+;
+  %a = add i32 %add, 128
+  %cmp = icmp ult i32 %a, 256
+  %t = trunc i32 %add to i16
+  %c = icmp sgt i32 %add, -1
+  %f = select i1 %c, i16 %high, i16 %low
+  %r = select i1 %cmp, i16 %t, i16 %f
+  ret i16 %r
+}
+
+define i32 @testi64i32addsat(i32 %a, i32 %b) {
+; CHECK-LABEL: @testi64i32addsat(
+; CHECK-NEXT:    [[SA:%.*]] = sext i32 [[A:%.*]] to i64
+; CHECK-NEXT:    [[SB:%.*]] = sext i32 [[B:%.*]] to i64
+; CHECK-NEXT:    [[ADD:%.*]] = add nsw i64 [[SA]], [[SB]]
+; CHECK-NEXT:    [[SH:%.*]] = lshr i64 [[ADD]], 32
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i64 [[SH]] to i32
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i64 [[ADD]] to i32
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i32 [[CONV1_I]], 31
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i32 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i64 [[ADD]], 63
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i64 [[SHR4_I]] to i32
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i32 [[CONV5_I]], 2147483647
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i32 [[CONV1_I]], i32 [[XOR_I]]
+; CHECK-NEXT:    ret i32 [[COND_I]]
+;
+  %sa = sext i32 %a to i64
+  %sb = sext i32 %b to i64
+  %add = add i64 %sa, %sb
+  %sh = lshr i64 %add, 32
+  %conv.i = trunc i64 %sh to i32
+  %conv1.i = trunc i64 %add to i32
+  %shr2.i = ashr i32 %conv1.i, 31
+  %cmp.not.i = icmp eq i32 %shr2.i, %conv.i
+  %shr4.i = ashr i64 %add, 63
+  %conv5.i = trunc i64 %shr4.i to i32
+  %xor.i = xor i32 %conv5.i, 2147483647
+  %cond.i = select i1 %cmp.not.i, i32 %conv1.i, i32 %xor.i
+  ret i32 %cond.i
+}
+
+define <4 x i8> @testv4i16i8(<4 x i16> %add) {
+; CHECK-LABEL: @testv4i16i8(
+; CHECK-NEXT:    [[SH:%.*]] = lshr <4 x i16> [[ADD:%.*]], <i16 8, i16 8, i16 8, i16 8>
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc <4 x i16> [[SH]] to <4 x i8>
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc <4 x i16> [[ADD]] to <4 x i8>
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr <4 x i8> [[CONV1_I]], <i8 7, i8 7, i8 7, i8 7>
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq <4 x i8> [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr <4 x i16> [[ADD]], <i16 15, i16 15, i16 15, i16 15>
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc <4 x i16> [[SHR4_I]] to <4 x i8>
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor <4 x i8> [[CONV5_I]], <i8 127, i8 127, i8 127, i8 127>
+; CHECK-NEXT:    [[COND_I:%.*]] = select <4 x i1> [[CMP_NOT_I]], <4 x i8> [[CONV1_I]], <4 x i8> [[XOR_I]]
+; CHECK-NEXT:    ret <4 x i8> [[COND_I]]
+;
+  %sh = lshr <4 x i16> %add, <i16 8, i16 8, i16 8, i16 8>
+  %conv.i = trunc <4 x i16> %sh to <4 x i8>
+  %conv1.i = trunc <4 x i16> %add to <4 x i8>
+  %shr2.i = ashr <4 x i8> %conv1.i, <i8 7, i8 7, i8 7, i8 7>
+  %cmp.not.i = icmp eq <4 x i8> %shr2.i, %conv.i
+  %shr4.i = ashr <4 x i16> %add, <i16 15, i16 15, i16 15, i16 15>
+  %conv5.i = trunc <4 x i16> %shr4.i to <4 x i8>
+  %xor.i = xor <4 x i8> %conv5.i, <i8 127, i8 127, i8 127, i8 127>
+  %cond.i = select <4 x i1> %cmp.not.i, <4 x i8> %conv1.i, <4 x i8> %xor.i
+  ret <4 x i8> %cond.i
+}
+
+define <4 x i8> @testv4i16i8add(<4 x i8> %a, <4 x i8> %b) {
+; CHECK-LABEL: @testv4i16i8add(
+; CHECK-NEXT:    [[SA:%.*]] = sext <4 x i8> [[A:%.*]] to <4 x i16>
+; CHECK-NEXT:    [[SB:%.*]] = sext <4 x i8> [[B:%.*]] to <4 x i16>
+; CHECK-NEXT:    [[ADD:%.*]] = add nsw <4 x i16> [[SA]], [[SB]]
+; CHECK-NEXT:    [[SH:%.*]] = lshr <4 x i16> [[ADD]], <i16 8, i16 8, i16 8, i16 8>
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc <4 x i16> [[SH]] to <4 x i8>
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc <4 x i16> [[ADD]] to <4 x i8>
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr <4 x i8> [[CONV1_I]], <i8 7, i8 7, i8 7, i8 7>
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq <4 x i8> [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr <4 x i16> [[ADD]], <i16 15, i16 15, i16 15, i16 15>
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc <4 x i16> [[SHR4_I]] to <4 x i8>
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor <4 x i8> [[CONV5_I]], <i8 127, i8 127, i8 127, i8 127>
+; CHECK-NEXT:    [[COND_I:%.*]] = select <4 x i1> [[CMP_NOT_I]], <4 x i8> [[CONV1_I]], <4 x i8> [[XOR_I]]
+; CHECK-NEXT:    ret <4 x i8> [[COND_I]]
+;
+  %sa = sext <4 x i8> %a to <4 x i16>
+  %sb = sext <4 x i8> %b to <4 x i16>
+  %add = add <4 x i16> %sa, %sb
+  %sh = lshr <4 x i16> %add, <i16 8, i16 8, i16 8, i16 8>
+  %conv.i = trunc <4 x i16> %sh to <4 x i8>
+  %conv1.i = trunc <4 x i16> %add to <4 x i8>
+  %shr2.i = ashr <4 x i8> %conv1.i, <i8 7, i8 7, i8 7, i8 7>
+  %cmp.not.i = icmp eq <4 x i8> %shr2.i, %conv.i
+  %shr4.i = ashr <4 x i16> %add, <i16 15, i16 15, i16 15, i16 15>
+  %conv5.i = trunc <4 x i16> %shr4.i to <4 x i8>
+  %xor.i = xor <4 x i8> %conv5.i, <i8 127, i8 127, i8 127, i8 127>
+  %cond.i = select <4 x i1> %cmp.not.i, <4 x i8> %conv1.i, <4 x i8> %xor.i
+  ret <4 x i8> %cond.i
+}
+
+define i8 @testi16i8_revcmp(i16 %add) {
+; CHECK-LABEL: @testi16i8_revcmp(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i16 [[ADD:%.*]], 8
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i16 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i16 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 7
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i16 [[ADD]], 15
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i16 [[SHR4_I]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 127
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i16 %add, 8
+  %conv.i = trunc i16 %sh to i8
+  %conv1.i = trunc i16 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 7
+  %cmp.not.i = icmp eq i8 %conv.i, %shr2.i
+  %shr4.i = ashr i16 %add, 15
+  %conv5.i = trunc i16 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 127
+  %cond.i = select i1 %cmp.not.i, i8 %conv1.i, i8 %xor.i
+  ret i8 %cond.i
+}
+
+define i8 @testi16i8_revselect(i16 %add) {
+; CHECK-LABEL: @testi16i8_revselect(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i16 [[ADD:%.*]], 8
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i16 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i16 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 7
+; CHECK-NEXT:    [[CMP_NOT_I_NOT:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i16 [[ADD]], 15
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i16 [[SHR4_I]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 127
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I_NOT]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i16 %add, 8
+  %conv.i = trunc i16 %sh to i8
+  %conv1.i = trunc i16 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 7
+  %cmp.not.i = icmp ne i8 %conv.i, %shr2.i
+  %shr4.i = ashr i16 %add, 15
+  %conv5.i = trunc i16 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 127
+  %cond.i = select i1 %cmp.not.i, i8 %xor.i, i8 %conv1.i
+  ret i8 %cond.i
+}
+
+define i8 @testi32i8(i32 %add) {
+; CHECK-LABEL: @testi32i8(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i32 [[ADD:%.*]], 8
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i32 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i32 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 7
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[TMP1:%.*]] = lshr i32 [[ADD]], 15
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i32 [[TMP1]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 127
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i32 %add, 8
+  %conv.i = trunc i32 %sh to i8
+  %conv1.i = trunc i32 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 7
+  %cmp.not.i = icmp eq i8 %shr2.i, %conv.i
+  %shr4.i = ashr i32 %add, 15
+  %conv5.i = trunc i32 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 127
+  %cond.i = select i1 %cmp.not.i, i8 %conv1.i, i8 %xor.i
+  ret i8 %cond.i
+}
+
+define i16 @
diff erentconsts(i32 %x, i16 %replacement_low, i16 %replacement_high) {
+; CHECK-LABEL: @
diff erentconsts(
+; CHECK-NEXT:    [[T0:%.*]] = icmp slt i32 [[X:%.*]], 128
+; CHECK-NEXT:    [[T1:%.*]] = select i1 [[T0]], i16 256, i16 -1
+; CHECK-NEXT:    [[T2:%.*]] = add i32 [[X]], 16
+; CHECK-NEXT:    [[T3:%.*]] = icmp ult i32 [[T2]], 144
+; CHECK-NEXT:    [[T4:%.*]] = trunc i32 [[X]] to i16
+; CHECK-NEXT:    [[R:%.*]] = select i1 [[T3]], i16 [[T4]], i16 [[T1]]
+; CHECK-NEXT:    ret i16 [[R]]
+;
+  %t0 = icmp slt i32 %x, 128
+  %t1 = select i1 %t0, i16 256, i16 65535
+  %t2 = add i32 %x, 16
+  %t3 = icmp ult i32 %t2, 144
+  %t4 = trunc i32 %x to i16
+  %r = select i1 %t3, i16 %t4, i16 %t1
+  ret i16 %r
+}
+
+define i8 @badimm1(i16 %add) {
+; CHECK-LABEL: @badimm1(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i16 [[ADD:%.*]], 9
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i16 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i16 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 7
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i16 [[ADD]], 15
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i16 [[SHR4_I]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 127
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i16 %add, 9
+  %conv.i = trunc i16 %sh to i8
+  %conv1.i = trunc i16 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 7
+  %cmp.not.i = icmp eq i8 %shr2.i, %conv.i
+  %shr4.i = ashr i16 %add, 15
+  %conv5.i = trunc i16 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 127
+  %cond.i = select i1 %cmp.not.i, i8 %conv1.i, i8 %xor.i
+  ret i8 %cond.i
+}
+
+define i8 @badimm2(i16 %add) {
+; CHECK-LABEL: @badimm2(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i16 [[ADD:%.*]], 8
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i16 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i16 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 6
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i16 [[ADD]], 15
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i16 [[SHR4_I]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 127
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i16 %add, 8
+  %conv.i = trunc i16 %sh to i8
+  %conv1.i = trunc i16 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 6
+  %cmp.not.i = icmp eq i8 %shr2.i, %conv.i
+  %shr4.i = ashr i16 %add, 15
+  %conv5.i = trunc i16 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 127
+  %cond.i = select i1 %cmp.not.i, i8 %conv1.i, i8 %xor.i
+  ret i8 %cond.i
+}
+
+define i8 @badimm3(i16 %add) {
+; CHECK-LABEL: @badimm3(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i16 [[ADD:%.*]], 8
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i16 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i16 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 7
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i16 [[ADD]], 14
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i16 [[SHR4_I]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 127
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i16 %add, 8
+  %conv.i = trunc i16 %sh to i8
+  %conv1.i = trunc i16 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 7
+  %cmp.not.i = icmp eq i8 %shr2.i, %conv.i
+  %shr4.i = ashr i16 %add, 14
+  %conv5.i = trunc i16 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 127
+  %cond.i = select i1 %cmp.not.i, i8 %conv1.i, i8 %xor.i
+  ret i8 %cond.i
+}
+
+define i8 @badimm4(i16 %add) {
+; CHECK-LABEL: @badimm4(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i16 [[ADD:%.*]], 8
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i16 [[SH]] to i8
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i16 [[ADD]] to i8
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i8 [[CONV1_I]], 7
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i8 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i16 [[ADD]], 15
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i16 [[SHR4_I]] to i8
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i8 [[CONV5_I]], 126
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i8 [[CONV1_I]], i8 [[XOR_I]]
+; CHECK-NEXT:    ret i8 [[COND_I]]
+;
+  %sh = lshr i16 %add, 8
+  %conv.i = trunc i16 %sh to i8
+  %conv1.i = trunc i16 %add to i8
+  %shr2.i = ashr i8 %conv1.i, 7
+  %cmp.not.i = icmp eq i8 %shr2.i, %conv.i
+  %shr4.i = ashr i16 %add, 15
+  %conv5.i = trunc i16 %shr4.i to i8
+  %xor.i = xor i8 %conv5.i, 126
+  %cond.i = select i1 %cmp.not.i, i8 %conv1.i, i8 %xor.i
+  ret i8 %cond.i
+}
+
+; One use checks
+
+define i32 @oneusexor(i64 %add) {
+; CHECK-LABEL: @oneusexor(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i64 [[ADD:%.*]], 32
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i64 [[SH]] to i32
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i64 [[ADD]] to i32
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i32 [[CONV1_I]], 31
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i32 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i64 [[ADD]], 63
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i64 [[SHR4_I]] to i32
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i32 [[CONV5_I]], 2147483647
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i32 [[CONV1_I]], i32 [[XOR_I]]
+; CHECK-NEXT:    call void @use(i32 [[XOR_I]])
+; CHECK-NEXT:    ret i32 [[COND_I]]
+;
+  %sh = lshr i64 %add, 32
+  %conv.i = trunc i64 %sh to i32
+  %conv1.i = trunc i64 %add to i32
+  %shr2.i = ashr i32 %conv1.i, 31
+  %cmp.not.i = icmp eq i32 %shr2.i, %conv.i
+  %shr4.i = ashr i64 %add, 63
+  %conv5.i = trunc i64 %shr4.i to i32
+  %xor.i = xor i32 %conv5.i, 2147483647
+  %cond.i = select i1 %cmp.not.i, i32 %conv1.i, i32 %xor.i
+  call void @use(i32 %xor.i)
+  ret i32 %cond.i
+}
+
+define i32 @oneuseconv(i64 %add) {
+; CHECK-LABEL: @oneuseconv(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i64 [[ADD:%.*]], 32
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i64 [[SH]] to i32
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i64 [[ADD]] to i32
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i32 [[CONV1_I]], 31
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i32 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i64 [[ADD]], 63
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i64 [[SHR4_I]] to i32
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i32 [[CONV5_I]], 2147483647
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i32 [[CONV1_I]], i32 [[XOR_I]]
+; CHECK-NEXT:    call void @use(i32 [[CONV1_I]])
+; CHECK-NEXT:    ret i32 [[COND_I]]
+;
+  %sh = lshr i64 %add, 32
+  %conv.i = trunc i64 %sh to i32
+  %conv1.i = trunc i64 %add to i32
+  %shr2.i = ashr i32 %conv1.i, 31
+  %cmp.not.i = icmp eq i32 %shr2.i, %conv.i
+  %shr4.i = ashr i64 %add, 63
+  %conv5.i = trunc i64 %shr4.i to i32
+  %xor.i = xor i32 %conv5.i, 2147483647
+  %cond.i = select i1 %cmp.not.i, i32 %conv1.i, i32 %xor.i
+  call void @use(i32 %conv1.i)
+  ret i32 %cond.i
+}
+
+define i32 @oneusecmp(i64 %add) {
+; CHECK-LABEL: @oneusecmp(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i64 [[ADD:%.*]], 32
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i64 [[SH]] to i32
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i64 [[ADD]] to i32
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i32 [[CONV1_I]], 31
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i32 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i64 [[ADD]], 63
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i64 [[SHR4_I]] to i32
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i32 [[CONV5_I]], 2147483647
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i32 [[CONV1_I]], i32 [[XOR_I]]
+; CHECK-NEXT:    call void @use1(i1 [[CMP_NOT_I]])
+; CHECK-NEXT:    ret i32 [[COND_I]]
+;
+  %sh = lshr i64 %add, 32
+  %conv.i = trunc i64 %sh to i32
+  %conv1.i = trunc i64 %add to i32
+  %shr2.i = ashr i32 %conv1.i, 31
+  %cmp.not.i = icmp eq i32 %shr2.i, %conv.i
+  %shr4.i = ashr i64 %add, 63
+  %conv5.i = trunc i64 %shr4.i to i32
+  %xor.i = xor i32 %conv5.i, 2147483647
+  %cond.i = select i1 %cmp.not.i, i32 %conv1.i, i32 %xor.i
+  call void @use1(i1 %cmp.not.i)
+  ret i32 %cond.i
+}
+
+define i32 @oneuseboth(i64 %add) {
+; CHECK-LABEL: @oneuseboth(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i64 [[ADD:%.*]], 32
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i64 [[SH]] to i32
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i64 [[ADD]] to i32
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i32 [[CONV1_I]], 31
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i32 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i64 [[ADD]], 63
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i64 [[SHR4_I]] to i32
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i32 [[CONV5_I]], 2147483647
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i32 [[CONV1_I]], i32 [[XOR_I]]
+; CHECK-NEXT:    call void @use(i32 [[XOR_I]])
+; CHECK-NEXT:    call void @use(i32 [[CONV1_I]])
+; CHECK-NEXT:    ret i32 [[COND_I]]
+;
+  %sh = lshr i64 %add, 32
+  %conv.i = trunc i64 %sh to i32
+  %conv1.i = trunc i64 %add to i32
+  %shr2.i = ashr i32 %conv1.i, 31
+  %cmp.not.i = icmp eq i32 %shr2.i, %conv.i
+  %shr4.i = ashr i64 %add, 63
+  %conv5.i = trunc i64 %shr4.i to i32
+  %xor.i = xor i32 %conv5.i, 2147483647
+  %cond.i = select i1 %cmp.not.i, i32 %conv1.i, i32 %xor.i
+  call void @use(i32 %xor.i)
+  call void @use(i32 %conv1.i)
+  ret i32 %cond.i
+}
+
+define i32 @oneusethree(i64 %add) {
+; CHECK-LABEL: @oneusethree(
+; CHECK-NEXT:    [[SH:%.*]] = lshr i64 [[ADD:%.*]], 32
+; CHECK-NEXT:    [[CONV_I:%.*]] = trunc i64 [[SH]] to i32
+; CHECK-NEXT:    [[CONV1_I:%.*]] = trunc i64 [[ADD]] to i32
+; CHECK-NEXT:    [[SHR2_I:%.*]] = ashr i32 [[CONV1_I]], 31
+; CHECK-NEXT:    [[CMP_NOT_I:%.*]] = icmp eq i32 [[SHR2_I]], [[CONV_I]]
+; CHECK-NEXT:    [[SHR4_I:%.*]] = ashr i64 [[ADD]], 63
+; CHECK-NEXT:    [[CONV5_I:%.*]] = trunc i64 [[SHR4_I]] to i32
+; CHECK-NEXT:    [[XOR_I:%.*]] = xor i32 [[CONV5_I]], 2147483647
+; CHECK-NEXT:    [[COND_I:%.*]] = select i1 [[CMP_NOT_I]], i32 [[CONV1_I]], i32 [[XOR_I]]
+; CHECK-NEXT:    call void @use(i32 [[XOR_I]])
+; CHECK-NEXT:    call void @use(i32 [[CONV1_I]])
+; CHECK-NEXT:    call void @use1(i1 [[CMP_NOT_I]])
+; CHECK-NEXT:    ret i32 [[COND_I]]
+;
+  %sh = lshr i64 %add, 32
+  %conv.i = trunc i64 %sh to i32
+  %conv1.i = trunc i64 %add to i32
+  %shr2.i = ashr i32 %conv1.i, 31
+  %cmp.not.i = icmp eq i32 %shr2.i, %conv.i
+  %shr4.i = ashr i64 %add, 63
+  %conv5.i = trunc i64 %shr4.i to i32
+  %xor.i = xor i32 %conv5.i, 2147483647
+  %cond.i = select i1 %cmp.not.i, i32 %conv1.i, i32 %xor.i
+  call void @use(i32 %xor.i)
+  call void @use(i32 %conv1.i)
+  call void @use1(i1 %cmp.not.i)
+  ret i32 %cond.i
+}
+
+define i8 @C0zero(i8 %X, i8 %y, i8 %z) {
+; CHECK-LABEL: @C0zero(
+; CHECK-NEXT:    [[C:%.*]] = icmp slt i8 [[X:%.*]], -10
+; CHECK-NEXT:    [[F:%.*]] = select i1 [[C]], i8 [[Y:%.*]], i8 [[Z:%.*]]
+; CHECK-NEXT:    ret i8 [[F]]
+;
+  %a = add i8 %X, 10
+  %cmp = icmp ult i8 %a, 0
+  %c = icmp slt i8 %X, -10
+  %f = select i1 %c, i8 %y, i8 %z
+  %r = select i1 %cmp, i8 %X, i8 %f
+  ret i8 %r
+}


        


More information about the llvm-commits mailing list