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