ARM: use CHECK-LABEL on a test.
[oota-llvm.git] / test / CodeGen / ARM / fast-isel-intrinsic.ll
1 ; RUN: llc < %s -O0 -fast-isel-abort -relocation-model=dynamic-no-pic -mtriple=armv7-apple-ios -verify-machineinstrs | FileCheck %s --check-prefix=ARM
2 ; RUN: llc < %s -O0 -fast-isel-abort -relocation-model=dynamic-no-pic -mtriple=armv7-linux-gnueabi -verify-machineinstrs | FileCheck %s --check-prefix=ARM
3 ; RUN: llc < %s -O0 -fast-isel-abort -relocation-model=dynamic-no-pic -mtriple=thumbv7-apple-ios -verify-machineinstrs | FileCheck %s --check-prefix=THUMB
4 ; RUN: llc < %s -O0 -fast-isel-abort -relocation-model=dynamic-no-pic -mtriple=armv7-apple-ios -arm-long-calls -verify-machineinstrs | FileCheck %s --check-prefix=ARM-LONG
5 ; RUN: llc < %s -O0 -fast-isel-abort -relocation-model=dynamic-no-pic -mtriple=armv7-linux-gnueabi -arm-long-calls -verify-machineinstrs | FileCheck %s --check-prefix=ARM-LONG
6 ; RUN: llc < %s -O0 -fast-isel-abort -relocation-model=dynamic-no-pic -mtriple=thumbv7-apple-ios -arm-long-calls -verify-machineinstrs | FileCheck %s --check-prefix=THUMB-LONG
7
8 ; XFAIL: vg_leak
9
10 ; Note that some of these tests assume that relocations are either
11 ; movw/movt or constant pool loads. Different platforms will select
12 ; different approaches.
13
14 @message1 = global [60 x i8] c"The LLVM Compiler Infrastructure\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00\00", align 1
15 @temp = common global [60 x i8] zeroinitializer, align 1
16
17 define void @t1() nounwind ssp {
18 ; ARM-LABEL: t1:
19 ; ARM: {{(movw r0, :lower16:_?message1)|(ldr r0, .LCPI)}}
20 ; ARM: {{(movt r0, :upper16:_?message1)|(ldr r0, \[r0\])}}
21 ; ARM: add r0, r0, #5
22 ; ARM: movw r1, #64
23 ; ARM: movw r2, #10
24 ; ARM: and r1, r1, #255
25 ; ARM: bl {{_?}}memset
26 ; ARM-LONG-LABEL: t1:
27 ; ARM-LONG: {{(movw r3, :lower16:L_memset\$non_lazy_ptr)|(ldr r3, .LCPI)}}
28 ; ARM-LONG: {{(movt r3, :upper16:L_memset\$non_lazy_ptr)?}}
29 ; ARM-LONG: ldr r3, [r3]
30 ; ARM-LONG: blx r3
31 ; THUMB-LABEL: t1:
32 ; THUMB: {{(movw r0, :lower16:_?message1)|(ldr.n r0, .LCPI)}}
33 ; THUMB: {{(movt r0, :upper16:_?message1)|(ldr r0, \[r0\])}}
34 ; THUMB: adds r0, #5
35 ; THUMB: movs r1, #64
36 ; THUMB: movt r1, #0
37 ; THUMB: movs r2, #10
38 ; THUMB: movt r2, #0
39 ; THUMB: and r1, r1, #255
40 ; THUMB: bl {{_?}}memset
41 ; THUMB-LONG-LABEL: t1:
42 ; THUMB-LONG: movw r3, :lower16:L_memset$non_lazy_ptr
43 ; THUMB-LONG: movt r3, :upper16:L_memset$non_lazy_ptr
44 ; THUMB-LONG: ldr r3, [r3]
45 ; THUMB-LONG: blx r3
46   call void @llvm.memset.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @message1, i32 0, i32 5), i8 64, i32 10, i32 4, i1 false)
47   ret void
48 }
49
50 declare void @llvm.memset.p0i8.i32(i8* nocapture, i8, i32, i32, i1) nounwind
51
52 define void @t2() nounwind ssp {
53 ; ARM-LABEL: t2:
54 ; ARM: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr r0, .LCPI)}}
55 ; ARM: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
56 ; ARM: ldr r0, [r0]
57 ; ARM: add r1, r0, #4
58 ; ARM: add r0, r0, #16
59 ; ARM: movw r2, #17
60 ; ARM: str r0, [sp[[SLOT:[, #0-9]*]]] @ 4-byte Spill
61 ; ARM: mov r0, r1
62 ; ARM: ldr r1, [sp[[SLOT]]] @ 4-byte Reload
63 ; ARM: bl {{_?}}memcpy
64 ; ARM-LONG-LABEL: t2:
65 ; ARM-LONG: {{(movw r3, :lower16:L_memcpy\$non_lazy_ptr)|(ldr r3, .LCPI)}}
66 ; ARM-LONG: {{(movt r3, :upper16:L_memcpy\$non_lazy_ptr)?}}
67 ; ARM-LONG: ldr r3, [r3]
68 ; ARM-LONG: blx r3
69 ; THUMB-LABEL: t2:
70 ; THUMB: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr.n r0, .LCPI)}}
71 ; THUMB: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
72 ; THUMB: ldr r0, [r0]
73 ; THUMB: adds r1, r0, #4
74 ; THUMB: adds r0, #16
75 ; THUMB: movs r2, #17
76 ; THUMB: movt r2, #0
77 ; THUMB: str r0, [sp[[SLOT:[, #0-9]*]]] @ 4-byte Spill
78 ; THUMB: mov r0, r1
79 ; THUMB: ldr r1,  [sp[[SLOT]]] @ 4-byte Reload
80 ; THUMB: bl {{_?}}memcpy
81 ; THUMB-LONG-LABEL: t2:
82 ; THUMB-LONG: movw r3, :lower16:L_memcpy$non_lazy_ptr
83 ; THUMB-LONG: movt r3, :upper16:L_memcpy$non_lazy_ptr
84 ; THUMB-LONG: ldr r3, [r3]
85 ; THUMB-LONG: blx r3
86   call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 4), i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 16), i32 17, i32 4, i1 false)
87   ret void
88 }
89
90 declare void @llvm.memcpy.p0i8.p0i8.i32(i8* nocapture, i8* nocapture, i32, i32, i1) nounwind
91
92 define void @t3() nounwind ssp {
93 ; ARM-LABEL: t3:
94 ; ARM: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr r0, .LCPI)}}
95 ; ARM: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
96 ; ARM: ldr r0, [r0]
97 ; ARM: add r1, r0, #4
98 ; ARM: add r0, r0, #16
99 ; ARM: movw r2, #10
100 ; ARM: mov r0, r1
101 ; ARM: bl {{_?}}memmove
102 ; ARM-LONG-LABEL: t3:
103 ; ARM-LONG: {{(movw r3, :lower16:L_memmove\$non_lazy_ptr)|(ldr r3, .LCPI)}}
104 ; ARM-LONG: {{(movt r3, :upper16:L_memmove\$non_lazy_ptr)?}}
105 ; ARM-LONG: ldr r3, [r3]
106 ; ARM-LONG: blx r3
107 ; THUMB-LABEL: t3:
108 ; THUMB: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr.n r0, .LCPI)}}
109 ; THUMB: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
110 ; THUMB: ldr r0, [r0]
111 ; THUMB: adds r1, r0, #4
112 ; THUMB: adds r0, #16
113 ; THUMB: movs r2, #10
114 ; THUMB: movt r2, #0
115 ; THUMB: str r0, [sp[[SLOT:[, #0-9]*]]] @ 4-byte Spill
116 ; THUMB: mov r0, r1
117 ; THUMB: ldr r1,  [sp[[SLOT]]] @ 4-byte Reload
118 ; THUMB: bl {{_?}}memmove
119 ; THUMB-LONG-LABEL: t3:
120 ; THUMB-LONG: movw r3, :lower16:L_memmove$non_lazy_ptr
121 ; THUMB-LONG: movt r3, :upper16:L_memmove$non_lazy_ptr
122 ; THUMB-LONG: ldr r3, [r3]
123 ; THUMB-LONG: blx r3
124   call void @llvm.memmove.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 4), i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 16), i32 10, i32 1, i1 false)
125   ret void
126 }
127
128 define void @t4() nounwind ssp {
129 ; ARM-LABEL: t4:
130 ; ARM: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr r0, .LCPI)}}
131 ; ARM: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
132 ; ARM: ldr r0, [r0]
133 ; ARM: ldr r1, [r0, #16]
134 ; ARM: str r1, [r0, #4]
135 ; ARM: ldr r1, [r0, #20]
136 ; ARM: str r1, [r0, #8]
137 ; ARM: ldrh r1, [r0, #24]
138 ; ARM: strh r1, [r0, #12]
139 ; ARM: bx lr
140 ; THUMB-LABEL: t4:
141 ; THUMB: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr.n r0, .LCPI)}}
142 ; THUMB: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
143 ; THUMB: ldr r0, [r0]
144 ; THUMB: ldr r1, [r0, #16]
145 ; THUMB: str r1, [r0, #4]
146 ; THUMB: ldr r1, [r0, #20]
147 ; THUMB: str r1, [r0, #8]
148 ; THUMB: ldrh r1, [r0, #24]
149 ; THUMB: strh r1, [r0, #12]
150 ; THUMB: bx lr
151   call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 4), i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 16), i32 10, i32 4, i1 false)
152   ret void
153 }
154
155 declare void @llvm.memmove.p0i8.p0i8.i32(i8* nocapture, i8* nocapture, i32, i32, i1) nounwind
156
157 define void @t5() nounwind ssp {
158 ; ARM-LABEL: t5:
159 ; ARM: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr r0, .LCPI)}}
160 ; ARM: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
161 ; ARM: ldr r0, [r0]
162 ; ARM: ldrh r1, [r0, #16]
163 ; ARM: strh r1, [r0, #4]
164 ; ARM: ldrh r1, [r0, #18]
165 ; ARM: strh r1, [r0, #6]
166 ; ARM: ldrh r1, [r0, #20]
167 ; ARM: strh r1, [r0, #8]
168 ; ARM: ldrh r1, [r0, #22]
169 ; ARM: strh r1, [r0, #10]
170 ; ARM: ldrh r1, [r0, #24]
171 ; ARM: strh r1, [r0, #12]
172 ; ARM: bx lr
173 ; THUMB-LABEL: t5:
174 ; THUMB: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr.n r0, .LCPI)}}
175 ; THUMB: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
176 ; THUMB: ldr r0, [r0]
177 ; THUMB: ldrh r1, [r0, #16]
178 ; THUMB: strh r1, [r0, #4]
179 ; THUMB: ldrh r1, [r0, #18]
180 ; THUMB: strh r1, [r0, #6]
181 ; THUMB: ldrh r1, [r0, #20]
182 ; THUMB: strh r1, [r0, #8]
183 ; THUMB: ldrh r1, [r0, #22]
184 ; THUMB: strh r1, [r0, #10]
185 ; THUMB: ldrh r1, [r0, #24]
186 ; THUMB: strh r1, [r0, #12]
187 ; THUMB: bx lr
188   call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 4), i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 16), i32 10, i32 2, i1 false)
189   ret void
190 }
191
192 define void @t6() nounwind ssp {
193 ; ARM-LABEL: t6:
194 ; ARM: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr r0, .LCPI)}}
195 ; ARM: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
196 ; ARM: ldr r0, [r0]
197 ; ARM: ldrb r1, [r0, #16]
198 ; ARM: strb r1, [r0, #4]
199 ; ARM: ldrb r1, [r0, #17]
200 ; ARM: strb r1, [r0, #5]
201 ; ARM: ldrb r1, [r0, #18]
202 ; ARM: strb r1, [r0, #6]
203 ; ARM: ldrb r1, [r0, #19]
204 ; ARM: strb r1, [r0, #7]
205 ; ARM: ldrb r1, [r0, #20]
206 ; ARM: strb r1, [r0, #8]
207 ; ARM: ldrb r1, [r0, #21]
208 ; ARM: strb r1, [r0, #9]
209 ; ARM: ldrb r1, [r0, #22]
210 ; ARM: strb r1, [r0, #10]
211 ; ARM: ldrb r1, [r0, #23]
212 ; ARM: strb r1, [r0, #11]
213 ; ARM: ldrb r1, [r0, #24]
214 ; ARM: strb r1, [r0, #12]
215 ; ARM: ldrb r1, [r0, #25]
216 ; ARM: strb r1, [r0, #13]
217 ; ARM: bx lr
218 ; THUMB-LABEL: t6:
219 ; THUMB: {{(movw r0, :lower16:L_temp\$non_lazy_ptr)|(ldr.n r0, .LCPI)}}
220 ; THUMB: {{(movt r0, :upper16:L_temp\$non_lazy_ptr)?}}
221 ; THUMB: ldr r0, [r0]
222 ; THUMB: ldrb r1, [r0, #16]
223 ; THUMB: strb r1, [r0, #4]
224 ; THUMB: ldrb r1, [r0, #17]
225 ; THUMB: strb r1, [r0, #5]
226 ; THUMB: ldrb r1, [r0, #18]
227 ; THUMB: strb r1, [r0, #6]
228 ; THUMB: ldrb r1, [r0, #19]
229 ; THUMB: strb r1, [r0, #7]
230 ; THUMB: ldrb r1, [r0, #20]
231 ; THUMB: strb r1, [r0, #8]
232 ; THUMB: ldrb r1, [r0, #21]
233 ; THUMB: strb r1, [r0, #9]
234 ; THUMB: ldrb r1, [r0, #22]
235 ; THUMB: strb r1, [r0, #10]
236 ; THUMB: ldrb r1, [r0, #23]
237 ; THUMB: strb r1, [r0, #11]
238 ; THUMB: ldrb r1, [r0, #24]
239 ; THUMB: strb r1, [r0, #12]
240 ; THUMB: ldrb r1, [r0, #25]
241 ; THUMB: strb r1, [r0, #13]
242 ; THUMB: bx lr
243   call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 4), i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 16), i32 10, i32 1, i1 false)
244   ret void
245 }
246
247 ; rdar://13202135
248 define void @t7() nounwind ssp {
249 ; Just make sure this doesn't assert when we have an odd length and an alignment of 2.
250   call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 4), i8* getelementptr inbounds ([60 x i8]* @temp, i32 0, i32 16), i32 3, i32 2, i1 false)
251   ret void
252 }
253
254 define i32 @t8(i32 %x) nounwind {
255 entry:
256 ; ARM-LABEL: t8:
257 ; ARM-NOT: FastISel missed call:   %expval = call i32 @llvm.expect.i32(i32 %x, i32 1)
258 ; THUMB-LABEL: t8:
259 ; THUMB-NOT: FastISel missed call:   %expval = call i32 @llvm.expect.i32(i32 %x, i32 1)
260   %expval = call i32 @llvm.expect.i32(i32 %x, i32 1)
261   ret i32 %expval
262 }
263
264 declare i32 @llvm.expect.i32(i32, i32) nounwind readnone