1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt < %s -instcombine -S | FileCheck %s
4 target datalayout = "n32"
6 define i1 @is_rem2_neg_i8(i8 %x) {
7 ; CHECK-LABEL: @is_rem2_neg_i8(
8 ; CHECK-NEXT: [[TMP1:%.*]] = and i8 [[X:%.*]], -127
9 ; CHECK-NEXT: [[R:%.*]] = icmp eq i8 [[TMP1]], -127
10 ; CHECK-NEXT: ret i1 [[R]]
13 %r = icmp slt i8 %s, 0
17 define <2 x i1> @is_rem2_pos_v2i8(<2 x i8> %x) {
18 ; CHECK-LABEL: @is_rem2_pos_v2i8(
19 ; CHECK-NEXT: [[TMP1:%.*]] = and <2 x i8> [[X:%.*]], <i8 -127, i8 -127>
20 ; CHECK-NEXT: [[R:%.*]] = icmp eq <2 x i8> [[TMP1]], <i8 1, i8 1>
21 ; CHECK-NEXT: ret <2 x i1> [[R]]
23 %s = srem <2 x i8> %x, <i8 2, i8 2>
24 %r = icmp sgt <2 x i8> %s, zeroinitializer
28 ; i8 -97 == 159 == 0b10011111
30 define i1 @is_rem32_pos_i8(i8 %x) {
31 ; CHECK-LABEL: @is_rem32_pos_i8(
32 ; CHECK-NEXT: [[TMP1:%.*]] = and i8 [[X:%.*]], -97
33 ; CHECK-NEXT: [[R:%.*]] = icmp sgt i8 [[TMP1]], 0
34 ; CHECK-NEXT: ret i1 [[R]]
37 %r = icmp sgt i8 %s, 0
41 ; i16 -32765 == 32771 == 0b1000000000000011
43 define i1 @is_rem4_neg_i16(i16 %x) {
44 ; CHECK-LABEL: @is_rem4_neg_i16(
45 ; CHECK-NEXT: [[TMP1:%.*]] = and i16 [[X:%.*]], -32765
46 ; CHECK-NEXT: [[R:%.*]] = icmp ugt i16 [[TMP1]], -32768
47 ; CHECK-NEXT: ret i1 [[R]]
50 %r = icmp slt i16 %s, 0
54 declare void @use(i32)
56 ; TODO: This is still worth folding because srem is difficult?
58 define i1 @is_rem32_neg_i32_extra_use(i32 %x) {
59 ; CHECK-LABEL: @is_rem32_neg_i32_extra_use(
60 ; CHECK-NEXT: [[S:%.*]] = srem i32 [[X:%.*]], 32
61 ; CHECK-NEXT: call void @use(i32 [[S]])
62 ; CHECK-NEXT: [[R:%.*]] = icmp slt i32 [[S]], 0
63 ; CHECK-NEXT: ret i1 [[R]]
66 call void @use(i32 %s)
67 %r = icmp slt i32 %s, 0
71 ; Negative test - wrong compare constant
73 define i1 @is_rem8_nonneg_i16(i16 %x) {
74 ; CHECK-LABEL: @is_rem8_nonneg_i16(
75 ; CHECK-NEXT: [[S:%.*]] = srem i16 [[X:%.*]], 8
76 ; CHECK-NEXT: [[R:%.*]] = icmp sgt i16 [[S]], -1
77 ; CHECK-NEXT: ret i1 [[R]]
80 %r = icmp sgt i16 %s, -1
84 ; Negative test - wrong remainder constant
86 define i1 @is_rem3_neg_i8(i8 %x) {
87 ; CHECK-LABEL: @is_rem3_neg_i8(
88 ; CHECK-NEXT: [[S:%.*]] = srem i8 [[X:%.*]], 3
89 ; CHECK-NEXT: [[R:%.*]] = icmp slt i8 [[S]], 0
90 ; CHECK-NEXT: ret i1 [[R]]
93 %r = icmp slt i8 %s, 0
97 ; Negative test - wrong compare constant
99 define i1 @is_rem16_something_i8(i8 %x) {
100 ; CHECK-LABEL: @is_rem16_something_i8(
101 ; CHECK-NEXT: [[S:%.*]] = srem i8 [[X:%.*]], 16
102 ; CHECK-NEXT: [[R:%.*]] = icmp slt i8 [[S]], 7
103 ; CHECK-NEXT: ret i1 [[R]]
106 %r = icmp slt i8 %s, 7
110 ; PR30281 - https://llvm.org/bugs/show_bug.cgi?id=30281
112 ; All of these tests contain foldable division-by-constant instructions, but we
113 ; can't assert that those folds have occurred before we process the later icmp.
115 define i32 @icmp_div(i16 %a, i16 %c) {
116 ; CHECK-LABEL: @icmp_div(
118 ; CHECK-NEXT: [[TOBOOL:%.*]] = icmp eq i16 [[A:%.*]], 0
119 ; CHECK-NEXT: br i1 [[TOBOOL]], label [[THEN:%.*]], label [[EXIT:%.*]]
121 ; CHECK-NEXT: [[CMP:%.*]] = icmp eq i16 [[C:%.*]], 0
122 ; CHECK-NEXT: [[PHI_BO:%.*]] = sext i1 [[CMP]] to i32
123 ; CHECK-NEXT: br label [[EXIT]]
125 ; CHECK-NEXT: [[PHI:%.*]] = phi i32 [ -1, [[ENTRY:%.*]] ], [ [[PHI_BO]], [[THEN]] ]
126 ; CHECK-NEXT: ret i32 [[PHI]]
129 %tobool = icmp eq i16 %a, 0
130 br i1 %tobool, label %then, label %exit
133 %div = sdiv i16 %c, -1
134 %cmp = icmp ne i16 %div, 0
138 %phi = phi i1 [ false, %entry ], [ %cmp, %then ]
139 %zext = zext i1 %phi to i32
140 %add = add nsw i32 %zext, -1
144 define i32 @icmp_div2(i16 %a, i16 %c) {
145 ; CHECK-LABEL: @icmp_div2(
147 ; CHECK-NEXT: [[TOBOOL:%.*]] = icmp eq i16 [[A:%.*]], 0
148 ; CHECK-NEXT: br i1 [[TOBOOL]], label [[THEN:%.*]], label [[EXIT:%.*]]
150 ; CHECK-NEXT: br label [[EXIT]]
152 ; CHECK-NEXT: ret i32 -1
155 %tobool = icmp eq i16 %a, 0
156 br i1 %tobool, label %then, label %exit
159 %div = sdiv i16 %c, 0
160 %cmp = icmp ne i16 %div, 0
164 %phi = phi i1 [ false, %entry ], [ %cmp, %then ]
165 %zext = zext i1 %phi to i32
166 %add = add nsw i32 %zext, -1
170 define i32 @icmp_div3(i16 %a, i16 %c) {
171 ; CHECK-LABEL: @icmp_div3(
173 ; CHECK-NEXT: [[TOBOOL:%.*]] = icmp eq i16 [[A:%.*]], 0
174 ; CHECK-NEXT: br i1 [[TOBOOL]], label [[THEN:%.*]], label [[EXIT:%.*]]
176 ; CHECK-NEXT: [[CMP:%.*]] = icmp eq i16 [[C:%.*]], 0
177 ; CHECK-NEXT: [[PHI_BO:%.*]] = sext i1 [[CMP]] to i32
178 ; CHECK-NEXT: br label [[EXIT]]
180 ; CHECK-NEXT: [[PHI:%.*]] = phi i32 [ -1, [[ENTRY:%.*]] ], [ [[PHI_BO]], [[THEN]] ]
181 ; CHECK-NEXT: ret i32 [[PHI]]
184 %tobool = icmp eq i16 %a, 0
185 br i1 %tobool, label %then, label %exit
188 %div = sdiv i16 %c, 1
189 %cmp = icmp ne i16 %div, 0
193 %phi = phi i1 [ false, %entry ], [ %cmp, %then ]
194 %zext = zext i1 %phi to i32
195 %add = add nsw i32 %zext, -1