[InstCombine] Signed saturation patterns
[llvm-complete.git] / test / Transforms / InstCombine / load-cmp.ll
blob7b24ce309a98c7ac129fd478326231eac70ce910
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt -instcombine -S -data-layout="p:32:32:32-p1:16:16:16-n8:16:32:64" < %s | FileCheck %s
4 @G16 = internal constant [10 x i16] [i16 35, i16 82, i16 69, i16 81, i16 85,
5                                      i16 73, i16 82, i16 69, i16 68, i16 0]
7 @G16_as1 = internal addrspace(1) constant [10 x i16] [i16 35, i16 82, i16 69, i16 81, i16 85,
8                                                       i16 73, i16 82, i16 69, i16 68, i16 0]
10 @GD = internal constant [6 x double]
11    [double -10.0, double 1.0, double 4.0, double 2.0, double -20.0, double -40.0]
13 %Foo = type { i32, i32, i32, i32 }
15 @GS = internal constant %Foo { i32 1, i32 4, i32 9, i32 14 }
17 @GStructArr = internal constant [4 x %Foo] [ %Foo { i32 1, i32 4, i32 9, i32 14 },
18                                              %Foo { i32 5, i32 4, i32 6, i32 11 },
19                                              %Foo { i32 6, i32 5, i32 9, i32 20 },
20                                              %Foo { i32 12, i32 3, i32 9, i32 8 } ]
23 define i1 @test1(i32 %X) {
24 ; CHECK-LABEL: @test1(
25 ; CHECK-NEXT:    [[R:%.*]] = icmp eq i32 [[X:%.*]], 9
26 ; CHECK-NEXT:    ret i1 [[R]]
28   %P = getelementptr inbounds [10 x i16], [10 x i16]* @G16, i32 0, i32 %X
29   %Q = load i16, i16* %P
30   %R = icmp eq i16 %Q, 0
31   ret i1 %R
34 define i1 @test1_noinbounds(i32 %X) {
35 ; CHECK-LABEL: @test1_noinbounds(
36 ; CHECK-NEXT:    [[R:%.*]] = icmp eq i32 [[X:%.*]], 9
37 ; CHECK-NEXT:    ret i1 [[R]]
39   %P = getelementptr [10 x i16], [10 x i16]* @G16, i32 0, i32 %X
40   %Q = load i16, i16* %P
41   %R = icmp eq i16 %Q, 0
42   ret i1 %R
45 define i1 @test1_noinbounds_i64(i64 %X) {
46 ; CHECK-LABEL: @test1_noinbounds_i64(
47 ; CHECK-NEXT:    [[TMP1:%.*]] = trunc i64 [[X:%.*]] to i32
48 ; CHECK-NEXT:    [[R:%.*]] = icmp eq i32 [[TMP1]], 9
49 ; CHECK-NEXT:    ret i1 [[R]]
51   %P = getelementptr [10 x i16], [10 x i16]* @G16, i64 0, i64 %X
52   %Q = load i16, i16* %P
53   %R = icmp eq i16 %Q, 0
54   ret i1 %R
57 define i1 @test1_noinbounds_as1(i32 %x) {
58 ; CHECK-LABEL: @test1_noinbounds_as1(
59 ; CHECK-NEXT:    [[TMP1:%.*]] = trunc i32 [[X:%.*]] to i16
60 ; CHECK-NEXT:    [[R:%.*]] = icmp eq i16 [[TMP1]], 9
61 ; CHECK-NEXT:    ret i1 [[R]]
63   %p = getelementptr [10 x i16], [10 x i16] addrspace(1)* @G16_as1, i16 0, i32 %x
64   %q = load i16, i16 addrspace(1)* %p
65   %r = icmp eq i16 %q, 0
66   ret i1 %r
70 define i1 @test2(i32 %X) {
71 ; CHECK-LABEL: @test2(
72 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i32 [[X:%.*]], 4
73 ; CHECK-NEXT:    ret i1 [[R]]
75   %P = getelementptr inbounds [10 x i16], [10 x i16]* @G16, i32 0, i32 %X
76   %Q = load i16, i16* %P
77   %R = icmp slt i16 %Q, 85
78   ret i1 %R
81 define i1 @test3(i32 %X) {
82 ; CHECK-LABEL: @test3(
83 ; CHECK-NEXT:    [[R:%.*]] = icmp eq i32 [[X:%.*]], 1
84 ; CHECK-NEXT:    ret i1 [[R]]
86   %P = getelementptr inbounds [6 x double], [6 x double]* @GD, i32 0, i32 %X
87   %Q = load double, double* %P
88   %R = fcmp oeq double %Q, 1.0
89   ret i1 %R
93 define i1 @test4(i32 %X) {
94 ; CHECK-LABEL: @test4(
95 ; CHECK-NEXT:    [[TMP1:%.*]] = lshr i32 933, [[X:%.*]]
96 ; CHECK-NEXT:    [[TMP2:%.*]] = and i32 [[TMP1]], 1
97 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i32 [[TMP2]], 0
98 ; CHECK-NEXT:    ret i1 [[R]]
100   %P = getelementptr inbounds [10 x i16], [10 x i16]* @G16, i32 0, i32 %X
101   %Q = load i16, i16* %P
102   %R = icmp sle i16 %Q, 73
103   ret i1 %R
106 define i1 @test4_i16(i16 %X) {
107 ; CHECK-LABEL: @test4_i16(
108 ; CHECK-NEXT:    [[TMP1:%.*]] = zext i16 [[X:%.*]] to i32
109 ; CHECK-NEXT:    [[TMP2:%.*]] = lshr i32 933, [[TMP1]]
110 ; CHECK-NEXT:    [[TMP3:%.*]] = and i32 [[TMP2]], 1
111 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i32 [[TMP3]], 0
112 ; CHECK-NEXT:    ret i1 [[R]]
114   %P = getelementptr inbounds [10 x i16], [10 x i16]* @G16, i32 0, i16 %X
115   %Q = load i16, i16* %P
116   %R = icmp sle i16 %Q, 73
117   ret i1 %R
120 define i1 @test5(i32 %X) {
121 ; CHECK-LABEL: @test5(
122 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp eq i32 [[X:%.*]], 2
123 ; CHECK-NEXT:    [[TMP2:%.*]] = icmp eq i32 [[X]], 7
124 ; CHECK-NEXT:    [[R:%.*]] = or i1 [[TMP1]], [[TMP2]]
125 ; CHECK-NEXT:    ret i1 [[R]]
127   %P = getelementptr inbounds [10 x i16], [10 x i16]* @G16, i32 0, i32 %X
128   %Q = load i16, i16* %P
129   %R = icmp eq i16 %Q, 69
130   ret i1 %R
133 define i1 @test6(i32 %X) {
134 ; CHECK-LABEL: @test6(
135 ; CHECK-NEXT:    [[TMP1:%.*]] = add i32 [[X:%.*]], -1
136 ; CHECK-NEXT:    [[R:%.*]] = icmp ult i32 [[TMP1]], 3
137 ; CHECK-NEXT:    ret i1 [[R]]
139   %P = getelementptr inbounds [6 x double], [6 x double]* @GD, i32 0, i32 %X
140   %Q = load double, double* %P
141   %R = fcmp ogt double %Q, 0.0
142   ret i1 %R
145 define i1 @test7(i32 %X) {
146 ; CHECK-LABEL: @test7(
147 ; CHECK-NEXT:    [[TMP1:%.*]] = add i32 [[X:%.*]], -1
148 ; CHECK-NEXT:    [[R:%.*]] = icmp ugt i32 [[TMP1]], 2
149 ; CHECK-NEXT:    ret i1 [[R]]
151   %P = getelementptr inbounds [6 x double], [6 x double]* @GD, i32 0, i32 %X
152   %Q = load double, double* %P
153   %R = fcmp olt double %Q, 0.0
154   ret i1 %R
157 define i1 @test8(i32 %X) {
158 ; CHECK-LABEL: @test8(
159 ; CHECK-NEXT:    [[TMP1:%.*]] = or i32 [[X:%.*]], 1
160 ; CHECK-NEXT:    [[TMP2:%.*]] = icmp eq i32 [[TMP1]], 9
161 ; CHECK-NEXT:    ret i1 [[TMP2]]
163   %P = getelementptr inbounds [10 x i16], [10 x i16]* @G16, i32 0, i32 %X
164   %Q = load i16, i16* %P
165   %R = and i16 %Q, 3
166   %S = icmp eq i16 %R, 0
167   ret i1 %S
170 @GA = internal constant [4 x { i32, i32 } ] [
171   { i32, i32 } { i32 1, i32 0 },
172   { i32, i32 } { i32 2, i32 1 },
173   { i32, i32 } { i32 3, i32 1 },
174   { i32, i32 } { i32 4, i32 0 }
177 define i1 @test9(i32 %X) {
178 ; CHECK-LABEL: @test9(
179 ; CHECK-NEXT:    [[TMP1:%.*]] = add i32 [[X:%.*]], -1
180 ; CHECK-NEXT:    [[TMP2:%.*]] = icmp ult i32 [[TMP1]], 2
181 ; CHECK-NEXT:    ret i1 [[TMP2]]
183   %P = getelementptr inbounds [4 x { i32, i32 } ], [4 x { i32, i32 } ]* @GA, i32 0, i32 %X, i32 1
184   %Q = load i32, i32* %P
185   %R = icmp eq i32 %Q, 1
186   ret i1 %R
189 define i1 @test10_struct(i32 %x) {
190 ; CHECK-LABEL: @test10_struct(
191 ; CHECK-NEXT:    ret i1 false
193   %p = getelementptr inbounds %Foo, %Foo* @GS, i32 %x, i32 0
194   %q = load i32, i32* %p
195   %r = icmp eq i32 %q, 9
196   ret i1 %r
199 define i1 @test10_struct_noinbounds(i32 %x) {
200 ; CHECK-LABEL: @test10_struct_noinbounds(
201 ; CHECK-NEXT:    [[P:%.*]] = getelementptr [[FOO:%.*]], %Foo* @GS, i32 [[X:%.*]], i32 0
202 ; CHECK-NEXT:    [[Q:%.*]] = load i32, i32* [[P]], align 8
203 ; CHECK-NEXT:    [[R:%.*]] = icmp eq i32 [[Q]], 9
204 ; CHECK-NEXT:    ret i1 [[R]]
206   %p = getelementptr %Foo, %Foo* @GS, i32 %x, i32 0
207   %q = load i32, i32* %p
208   %r = icmp eq i32 %q, 9
209   ret i1 %r
212 ; Test that the GEP indices are converted before we ever get here
213 ; Index < ptr size
214 define i1 @test10_struct_i16(i16 %x){
215 ; CHECK-LABEL: @test10_struct_i16(
216 ; CHECK-NEXT:    ret i1 false
218   %p = getelementptr inbounds %Foo, %Foo* @GS, i16 %x, i32 0
219   %q = load i32, i32* %p
220   %r = icmp eq i32 %q, 0
221   ret i1 %r
224 ; Test that the GEP indices are converted before we ever get here
225 ; Index > ptr size
226 define i1 @test10_struct_i64(i64 %x){
227 ; CHECK-LABEL: @test10_struct_i64(
228 ; CHECK-NEXT:    ret i1 false
230   %p = getelementptr inbounds %Foo, %Foo* @GS, i64 %x, i32 0
231   %q = load i32, i32* %p
232   %r = icmp eq i32 %q, 0
233   ret i1 %r
236 define i1 @test10_struct_noinbounds_i16(i16 %x) {
237 ; CHECK-LABEL: @test10_struct_noinbounds_i16(
238 ; CHECK-NEXT:    [[TMP1:%.*]] = sext i16 [[X:%.*]] to i32
239 ; CHECK-NEXT:    [[P:%.*]] = getelementptr [[FOO:%.*]], %Foo* @GS, i32 [[TMP1]], i32 0
240 ; CHECK-NEXT:    [[Q:%.*]] = load i32, i32* [[P]], align 8
241 ; CHECK-NEXT:    [[R:%.*]] = icmp eq i32 [[Q]], 0
242 ; CHECK-NEXT:    ret i1 [[R]]
244   %p = getelementptr %Foo, %Foo* @GS, i16 %x, i32 0
245   %q = load i32, i32* %p
246   %r = icmp eq i32 %q, 0
247   ret i1 %r
250 define i1 @test10_struct_arr(i32 %x) {
251 ; CHECK-LABEL: @test10_struct_arr(
252 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i32 [[X:%.*]], 1
253 ; CHECK-NEXT:    ret i1 [[R]]
255   %p = getelementptr inbounds [4 x %Foo], [4 x %Foo]* @GStructArr, i32 0, i32 %x, i32 2
256   %q = load i32, i32* %p
257   %r = icmp eq i32 %q, 9
258   ret i1 %r
261 define i1 @test10_struct_arr_noinbounds(i32 %x) {
262 ; CHECK-LABEL: @test10_struct_arr_noinbounds(
263 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i32 [[X:%.*]], 1
264 ; CHECK-NEXT:    ret i1 [[R]]
266   %p = getelementptr [4 x %Foo], [4 x %Foo]* @GStructArr, i32 0, i32 %x, i32 2
267   %q = load i32, i32* %p
268   %r = icmp eq i32 %q, 9
269   ret i1 %r
272 define i1 @test10_struct_arr_i16(i16 %x) {
273 ; CHECK-LABEL: @test10_struct_arr_i16(
274 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i16 [[X:%.*]], 1
275 ; CHECK-NEXT:    ret i1 [[R]]
277   %p = getelementptr inbounds [4 x %Foo], [4 x %Foo]* @GStructArr, i16 0, i16 %x, i32 2
278   %q = load i32, i32* %p
279   %r = icmp eq i32 %q, 9
280   ret i1 %r
283 define i1 @test10_struct_arr_i64(i64 %x) {
284 ; CHECK-LABEL: @test10_struct_arr_i64(
285 ; CHECK-NEXT:    [[TMP1:%.*]] = trunc i64 [[X:%.*]] to i32
286 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i32 [[TMP1]], 1
287 ; CHECK-NEXT:    ret i1 [[R]]
289   %p = getelementptr inbounds [4 x %Foo], [4 x %Foo]* @GStructArr, i64 0, i64 %x, i32 2
290   %q = load i32, i32* %p
291   %r = icmp eq i32 %q, 9
292   ret i1 %r
295 define i1 @test10_struct_arr_noinbounds_i16(i16 %x) {
296 ; CHECK-LABEL: @test10_struct_arr_noinbounds_i16(
297 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i16 [[X:%.*]], 1
298 ; CHECK-NEXT:    ret i1 [[R]]
300   %p = getelementptr [4 x %Foo], [4 x %Foo]* @GStructArr, i32 0, i16 %x, i32 2
301   %q = load i32, i32* %p
302   %r = icmp eq i32 %q, 9
303   ret i1 %r
306 define i1 @test10_struct_arr_noinbounds_i64(i64 %x) {
307 ; CHECK-LABEL: @test10_struct_arr_noinbounds_i64(
308 ; CHECK-NEXT:    [[TMP1:%.*]] = trunc i64 [[X:%.*]] to i32
309 ; CHECK-NEXT:    [[R:%.*]] = icmp ne i32 [[TMP1]], 1
310 ; CHECK-NEXT:    ret i1 [[R]]
312   %p = getelementptr [4 x %Foo], [4 x %Foo]* @GStructArr, i32 0, i64 %x, i32 2
313   %q = load i32, i32* %p
314   %r = icmp eq i32 %q, 9
315   ret i1 %r