blob: 203250717f52103f00f6a8e0ea0bbcff2892d5ee [file] [log] [blame]
Derek Schuff9769deb2015-12-11 23:49:46 +00001; RUN: llc < %s -asm-verbose=false | FileCheck %s
2; RUN: llc < %s -asm-verbose=false -fast-isel | FileCheck %s
3
4
Dan Gohman0c6f5ac2016-01-07 03:19:23 +00005target datalayout = "e-m:e-p:32:32-i64:64-n32:64-S128"
Derek Schuff9769deb2015-12-11 23:49:46 +00006target triple = "wasm32-unknown-unknown"
7
8; CHECK-LABEL: alloca32:
Derek Schuff45cd5a72015-12-16 20:43:06 +00009; Check that there is an extra local for the stack pointer.
10; CHECK: .local i32, i32, i32, i32{{$}}
Derek Schuff9769deb2015-12-11 23:49:46 +000011define void @alloca32() {
12 ; CHECK: i32.const [[L1:.+]]=, __stack_pointer
13 ; CHECK-NEXT: i32.load [[L1]]=, 0([[L1]])
14 ; CHECK-NEXT: i32.const [[L2:.+]]=, 16
15 ; CHECK-NEXT: i32.sub [[SP:.+]]=, [[L1]], [[L2]]
16 %retval = alloca i32
17 ; CHECK: i32.const $push[[L3:.+]]=, 0
18 ; CHECK: i32.store {{.*}}=, 12([[SP]]), $pop[[L3]]
19 store i32 0, i32* %retval
20 ; CHECK: i32.const [[L4:.+]]=, 16
21 ; CHECK-NEXT: i32.add [[SP]]=, [[SP]], [[L4]]
22 ; CHECK-NEXT: i32.const [[L5:.+]]=, __stack_pointer
23 ; CHECK-NEXT: i32.store [[SP]]=, 0([[L5]]), [[SP]]
24 ret void
25}
26
27; CHECK-LABEL: alloca3264:
Derek Schuff45cd5a72015-12-16 20:43:06 +000028; CHECK: .local i32, i32, i32, i32{{$}}
Derek Schuff9769deb2015-12-11 23:49:46 +000029define void @alloca3264() {
30 ; CHECK: i32.const [[L1:.+]]=, __stack_pointer
31 ; CHECK-NEXT: i32.load [[L1]]=, 0([[L1]])
32 ; CHECK-NEXT: i32.const [[L2:.+]]=, 16
33 ; CHECK-NEXT: i32.sub [[SP:.+]]=, [[L1]], [[L2]]
34 %r1 = alloca i32
35 %r2 = alloca double
36 ; CHECK: i32.const $push[[L3:.+]]=, 0
37 ; CHECK: i32.store {{.*}}=, 12([[SP]]), $pop[[L3]]
38 store i32 0, i32* %r1
39 ; CHECK: i64.const $push[[L4:.+]]=, 0
40 ; CHECK: i64.store {{.*}}=, 0([[SP]]), $pop[[L4]]
41 store double 0.0, double* %r2
42 ; CHECK: i32.const [[L4:.+]]=, 16
43 ; CHECK-NEXT: i32.add [[SP]]=, [[SP]], [[L4]]
44 ; CHECK-NEXT: i32.const [[L5:.+]]=, __stack_pointer
45 ; CHECK-NEXT: i32.store [[SP]]=, 0([[L5]]), [[SP]]
46 ret void
47}
48
Derek Schuff45cd5a72015-12-16 20:43:06 +000049; CHECK-LABEL: allocarray:
Dan Gohman7e649172016-01-20 04:21:16 +000050; CHECK: .local i32, i32, i32, i32, i32{{$}}
Derek Schuff9769deb2015-12-11 23:49:46 +000051define void @allocarray() {
Dan Gohman7e649172016-01-20 04:21:16 +000052 ; CHECK-NEXT: i32.const [[L1:.+]]=, __stack_pointer
Derek Schuff9769deb2015-12-11 23:49:46 +000053 ; CHECK-NEXT: i32.load [[L1]]=, 0([[L1]])
Dan Gohman7e649172016-01-20 04:21:16 +000054 ; CHECK-NEXT: i32.const [[L2:.+]]=, 32{{$}}
Derek Schuff9769deb2015-12-11 23:49:46 +000055 ; CHECK-NEXT: i32.sub [[SP:.+]]=, [[L1]], [[L2]]
Dan Gohman7e649172016-01-20 04:21:16 +000056 ; CHECK-NEXT: i32.const [[L2]]=, __stack_pointer{{$}}
57 ; CHECK-NEXT: i32.store [[SP]]=, 0([[L2]]), [[SP]]
Derek Schuff9769deb2015-12-11 23:49:46 +000058 %r = alloca [5 x i32]
Dan Gohman7e649172016-01-20 04:21:16 +000059
Dan Gohmanbb372242016-01-26 03:39:31 +000060 ; CHECK-NEXT: i32.const $push[[L4:.+]]=, 12
Dan Gohman7e649172016-01-20 04:21:16 +000061 ; CHECK-NEXT: i32.const [[L5:.+]]=, 12
62 ; CHECK-NEXT: i32.add [[L5]]=, [[SP]], [[L5]]
63 ; CHECK-NEXT: i32.add $push[[L6:.+]]=, [[L5]], $pop[[L4]]
64 ; CHECK-NEXT: i32.const $push[[L9:.+]]=, 1{{$}}
65 ; CHECK-NEXT: i32.store $push[[L10:.+]]=, 12([[SP]]), $pop[[L9]]{{$}}
66 ; CHECK-NEXT: i32.store $discard=, 0($pop3), $pop[[L10]]{{$}}
Derek Schuff9769deb2015-12-11 23:49:46 +000067 %p = getelementptr [5 x i32], [5 x i32]* %r, i32 0, i32 0
68 store i32 1, i32* %p
Dan Gohmanbb372242016-01-26 03:39:31 +000069 %p2 = getelementptr [5 x i32], [5 x i32]* %r, i32 0, i32 3
Derek Schuff9769deb2015-12-11 23:49:46 +000070 store i32 1, i32* %p2
Dan Gohman7e649172016-01-20 04:21:16 +000071
72 ; CHECK-NEXT: i32.const [[L7:.+]]=, 32
Derek Schuff9769deb2015-12-11 23:49:46 +000073 ; CHECK-NEXT: i32.add [[SP]]=, [[SP]], [[L7]]
74 ; CHECK-NEXT: i32.const [[L8:.+]]=, __stack_pointer
Dan Gohman7e649172016-01-20 04:21:16 +000075 ; CHECK-NEXT: i32.store [[SP]]=, 0([[L8]]), [[SP]]
Derek Schuff9769deb2015-12-11 23:49:46 +000076 ret void
77}
78
Derek Schuff90d9e8d2016-01-26 22:47:43 +000079declare void @ext_func(i64* %ptr)
80; CHECK-LABEL: non_mem_use
Derek Schuffc97ba932016-01-30 21:43:08 +000081define void @non_mem_use(i8** %addr) {
82 ; CHECK: i32.const [[L2:.+]]=, 48
Derek Schuff90d9e8d2016-01-26 22:47:43 +000083 ; CHECK-NEXT: i32.sub [[SP:.+]]=, {{.+}}, [[L2]]
Derek Schuffc97ba932016-01-30 21:43:08 +000084 %buf = alloca [27 x i8], align 16
Derek Schuff90d9e8d2016-01-26 22:47:43 +000085 %r = alloca i64
86 %r2 = alloca i64
87 ; %r is at SP+8
88 ; CHECK: i32.const [[OFF:.+]]=, 8
89 ; CHECK-NEXT: i32.add [[ARG1:.+]]=, [[SP]], [[OFF]]
90 ; CHECK-NEXT: call ext_func@FUNCTION, [[ARG1]]
91 call void @ext_func(i64* %r)
92 ; %r2 is at SP+0, no add needed
93 ; CHECK-NEXT: call ext_func@FUNCTION, [[SP]]
94 call void @ext_func(i64* %r2)
Derek Schuffc97ba932016-01-30 21:43:08 +000095 ; Use as a value, but in a store
96 ; %buf is at SP+16
97 ; CHECK: i32.const [[OFF:.+]]=, 16
98 ; CHECK-NEXT: i32.add [[VAL:.+]]=, [[SP]], [[OFF]]
99 ; CHECK-NEXT: i32.store {{.*}}=, 0($0), [[VAL]]
100 %gep = getelementptr inbounds [27 x i8], [27 x i8]* %buf, i32 0, i32 0
101 store i8* %gep, i8** %addr
Derek Schuff90d9e8d2016-01-26 22:47:43 +0000102 ret void
103}
104
Dan Gohman7e649172016-01-20 04:21:16 +0000105; CHECK-LABEL: allocarray_inbounds:
106; CHECK: .local i32, i32, i32, i32{{$}}
Derek Schuff9bfea272016-01-07 18:55:52 +0000107define void @allocarray_inbounds() {
108 ; CHECK: i32.const [[L1:.+]]=, __stack_pointer
109 ; CHECK-NEXT: i32.load [[L1]]=, 0([[L1]])
110 ; CHECK-NEXT: i32.const [[L2:.+]]=, 32
111 ; CHECK-NEXT: i32.sub [[SP:.+]]=, [[L1]], [[L2]]
112 %r = alloca [5 x i32]
113 ; CHECK: i32.const $push[[L3:.+]]=, 1
114 ; CHECK: i32.store {{.*}}=, 12([[SP]]), $pop[[L3]]
115 %p = getelementptr inbounds [5 x i32], [5 x i32]* %r, i32 0, i32 0
116 store i32 1, i32* %p
117 ; This store should have both the GEP and the FI folded into it.
Dan Gohmanbb372242016-01-26 03:39:31 +0000118 ; CHECK-NEXT: i32.store {{.*}}=, 24([[SP]]), $pop
119 %p2 = getelementptr inbounds [5 x i32], [5 x i32]* %r, i32 0, i32 3
Derek Schuff9bfea272016-01-07 18:55:52 +0000120 store i32 1, i32* %p2
121 ; CHECK: i32.const [[L7:.+]]=, 32
122 ; CHECK-NEXT: i32.add [[SP]]=, [[SP]], [[L7]]
123 ; CHECK-NEXT: i32.const [[L8:.+]]=, __stack_pointer
124 ; CHECK-NEXT: i32.store [[SP]]=, 0([[L7]]), [[SP]]
125 ret void
126}
127
Dan Gohman7e649172016-01-20 04:21:16 +0000128; CHECK-LABEL: dynamic_alloca:
Derek Schuff8bb5f292015-12-16 23:21:30 +0000129define void @dynamic_alloca(i32 %alloc) {
Derek Schuff6ea637a2016-01-29 18:37:49 +0000130 ; CHECK: i32.const [[L0:.+]]=, __stack_pointer
131 ; CHECK-NEXT: i32.load [[SP:.+]]=, 0([[L0]])
132 ; CHECK-NEXT: copy_local [[FP:.+]]=, [[SP]]
133 ; Target independent codegen bumps the stack pointer
134 ; FIXME: we need to write the value back to memory
135 %r = alloca i32, i32 %alloc
136 ; Target-independent codegen also calculates the store addr
137 store i32 0, i32* %r
138 ; CHECK: i32.const [[L3:.+]]=, __stack_pointer
139 ; CHECK-NEXT: i32.store [[SP]]=, 0([[L3]]), [[FP]]
140 ret void
141}
142
143
144; CHECK-LABEL: dynamic_static_alloca:
145define void @dynamic_static_alloca(i32 %alloc) {
146 ; CHECK: i32.const [[L0:.+]]=, __stack_pointer
147 ; CHECK-NEXT: i32.load [[L0]]=, 0([[L0]])
148 ; CHECK-NEXT: i32.const [[L2:.+]]=, 16
149 ; CHECK-NEXT: i32.sub [[SP:.+]]=, [[L0]], [[L2]]
150 ; CHECK-NEXT: copy_local [[FP:.+]]=, [[SP]]
151 ; CHECK-NEXT: i32.const [[L3:.+]]=, __stack_pointer
152 ; CHECK-NEXT: i32.store {{.*}}=, 0([[L3]]), [[SP]]
153 %r1 = alloca i32
154 %r = alloca i32, i32 %alloc
155 store i32 0, i32* %r
156 ; CHECK: i32.const [[L3:.+]]=, 16
157 ; CHECK: i32.add [[SP]]=, [[FP]], [[L3]]
158 ; CHECK: i32.const [[L4:.+]]=, __stack_pointer
159 ; CHECK-NEXT: i32.store [[SP]]=, 0([[L4]]), [[SP]]
Derek Schuff8bb5f292015-12-16 23:21:30 +0000160 ret void
161}
Derek Schuffc97ba932016-01-30 21:43:08 +0000162
Derek Schuffaadc89c2016-02-16 18:18:36 +0000163; The use of the alloca in a phi causes a CopyToReg DAG node to be generated,
164; which has to have special handling because CopyToReg can't have a FI operand
165; CHECK-LABEL: copytoreg_fi:
166define void @copytoreg_fi(i1 %cond, i32* %b) {
167entry:
168 ; CHECK: i32.const [[L2:.+]]=, 16
169 ; CHECK-NEXT: i32.sub [[SP:.+]]=, {{.+}}, [[L2]]
170 %addr = alloca i32
171 ; CHECK: i32.const [[OFF:.+]]=, 12
172 ; CHECK-NEXT: i32.add [[ADDR:.+]]=, [[SP]], [[OFF]]
173 ; CHECK-NEXT: copy_local [[COPY:.+]]=, [[ADDR]]
174 br label %body
175body:
176 %a = phi i32* [%addr, %entry], [%b, %body]
177 store i32 1, i32* %a
178 ; CHECK: i32.store {{.*}}, 0([[COPY]]),
179 br i1 %cond, label %body, label %exit
180exit:
181 ret void
182}
183
Dan Gohman94c65662016-02-16 23:48:04 +0000184declare void @use_i8_star(i8*)
185declare i8* @llvm.frameaddress(i32)
186
187; Test __builtin_frame_address(0).
188; TODO: When the prolog/epilog sequences are optimized, refine these checks to
189; be more specific.
190
191; CHECK-LABEL: frameaddress_0:
192; CHECK: __stack_pointer
193; CHECK: load
194; CHECK: call use_i8_star
195; CHECK: __stack_pointer
196; CHECK: store
197define void @frameaddress_0() {
198 %t = call i8* @llvm.frameaddress(i32 0)
199 call void @use_i8_star(i8* %t)
200 ret void
201}
202
203; Test __builtin_frame_address(1).
204
205; CHECK-LABEL: frameaddress_1:
206; CHECK-NEXT: i32.const $push0=, 0{{$}}
207; CHECK-NEXT: call use_i8_star@FUNCTION, $pop0{{$}}
208; CHECK-NEXT: return{{$}}
209define void @frameaddress_1() {
210 %t = call i8* @llvm.frameaddress(i32 1)
211 call void @use_i8_star(i8* %t)
212 ret void
213}
214
Derek Schuffc97ba932016-01-30 21:43:08 +0000215; TODO: test over-aligned alloca