blob: a414addc738d470f1b89168b3eb752570154de7e [file] [log] [blame]
Russ Cox00d2f912014-09-24 15:20:03 -04001// errorcheck -0 -N -m -l
2
3// Copyright 2010 The Go Authors. All rights reserved.
4// Use of this source code is governed by a BSD-style
5// license that can be found in the LICENSE file.
6
7// Test, using compiler diagnostic flags, that the escape analysis is working.
8// Compiles but does not run. Inlining is disabled.
9// Registerization is disabled too (-N), which should
10// have no effect on escape analysis.
11
12package foo
13
14import (
15 "fmt"
16 "unsafe"
17)
18
19var gxx *int
20
Dmitry Vyukov76477412015-02-19 22:00:11 +030021func foo1(x int) { // ERROR "moved to heap: x$"
22 gxx = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -040023}
24
Dmitry Vyukov76477412015-02-19 22:00:11 +030025func foo2(yy *int) { // ERROR "leaking param: yy$"
Russ Cox00d2f912014-09-24 15:20:03 -040026 gxx = yy
27}
28
Dmitry Vyukov76477412015-02-19 22:00:11 +030029func foo3(x int) *int { // ERROR "moved to heap: x$"
30 return &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -040031}
32
33type T *T
34
Dmitry Vyukov76477412015-02-19 22:00:11 +030035func foo3b(t T) { // ERROR "leaking param: t$"
Russ Cox00d2f912014-09-24 15:20:03 -040036 *t = t
37}
38
39// xx isn't going anywhere, so use of yy is ok
Dmitry Vyukov76477412015-02-19 22:00:11 +030040func foo4(xx, yy *int) { // ERROR "foo4 xx does not escape$" "foo4 yy does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -040041 xx = yy
42}
43
44// xx isn't going anywhere, so taking address of yy is ok
Dmitry Vyukov76477412015-02-19 22:00:11 +030045func foo5(xx **int, yy *int) { // ERROR "foo5 xx does not escape$" "foo5 yy does not escape$"
46 xx = &yy // ERROR "foo5 &yy does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -040047}
48
Dmitry Vyukov76477412015-02-19 22:00:11 +030049func foo6(xx **int, yy *int) { // ERROR "foo6 xx does not escape$" "leaking param: yy$"
Russ Cox00d2f912014-09-24 15:20:03 -040050 *xx = yy
51}
52
Dmitry Vyukov76477412015-02-19 22:00:11 +030053func foo7(xx **int, yy *int) { // ERROR "foo7 xx does not escape$" "foo7 yy does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -040054 **xx = *yy
55}
56
Dmitry Vyukov76477412015-02-19 22:00:11 +030057func foo8(xx, yy *int) int { // ERROR "foo8 xx does not escape$" "foo8 yy does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -040058 xx = yy
59 return *xx
60}
61
Dmitry Vyukov76477412015-02-19 22:00:11 +030062func foo9(xx, yy *int) *int { // ERROR "leaking param: xx to result ~r2$" "leaking param: yy to result ~r2$"
Russ Cox00d2f912014-09-24 15:20:03 -040063 xx = yy
64 return xx
65}
66
Dmitry Vyukov76477412015-02-19 22:00:11 +030067func foo10(xx, yy *int) { // ERROR "foo10 xx does not escape$" "foo10 yy does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -040068 *xx = *yy
69}
70
71func foo11() int {
72 x, y := 0, 42
Dmitry Vyukov76477412015-02-19 22:00:11 +030073 xx := &x // ERROR "foo11 &x does not escape$"
74 yy := &y // ERROR "foo11 &y does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -040075 *xx = *yy
76 return x
77}
78
79var xxx **int
80
Dmitry Vyukov76477412015-02-19 22:00:11 +030081func foo12(yyy **int) { // ERROR "leaking param: yyy$"
Russ Cox00d2f912014-09-24 15:20:03 -040082 xxx = yyy
83}
84
85// Must treat yyy as leaking because *yyy leaks, and the escape analysis
86// summaries in exported metadata do not distinguish these two cases.
Dmitry Vyukov76477412015-02-19 22:00:11 +030087func foo13(yyy **int) { // ERROR "leaking param: yyy$"
Russ Cox00d2f912014-09-24 15:20:03 -040088 *xxx = *yyy
89}
90
Dmitry Vyukov76477412015-02-19 22:00:11 +030091func foo14(yyy **int) { // ERROR "foo14 yyy does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -040092 **xxx = **yyy
93}
94
Dmitry Vyukov76477412015-02-19 22:00:11 +030095func foo15(yy *int) { // ERROR "moved to heap: yy$"
96 xxx = &yy // ERROR "&yy escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -040097}
98
Dmitry Vyukov76477412015-02-19 22:00:11 +030099func foo16(yy *int) { // ERROR "leaking param: yy$"
Russ Cox00d2f912014-09-24 15:20:03 -0400100 *xxx = yy
101}
102
Dmitry Vyukov76477412015-02-19 22:00:11 +0300103func foo17(yy *int) { // ERROR "foo17 yy does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400104 **xxx = *yy
105}
106
Dmitry Vyukov76477412015-02-19 22:00:11 +0300107func foo18(y int) { // ERROR "moved to heap: y$"
108 *xxx = &y // ERROR "&y escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400109}
110
111func foo19(y int) {
112 **xxx = y
113}
114
115type Bar struct {
116 i int
117 ii *int
118}
119
120func NewBar() *Bar {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300121 return &Bar{42, nil} // ERROR "&Bar literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400122}
123
Dmitry Vyukov76477412015-02-19 22:00:11 +0300124func NewBarp(x *int) *Bar { // ERROR "leaking param: x$"
125 return &Bar{42, x} // ERROR "&Bar literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400126}
127
Dmitry Vyukov76477412015-02-19 22:00:11 +0300128func NewBarp2(x *int) *Bar { // ERROR "NewBarp2 x does not escape$"
129 return &Bar{*x, nil} // ERROR "&Bar literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400130}
131
Dmitry Vyukov76477412015-02-19 22:00:11 +0300132func (b *Bar) NoLeak() int { // ERROR "\(\*Bar\).NoLeak b does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400133 return *(b.ii)
134}
135
Dmitry Vyukov76477412015-02-19 22:00:11 +0300136func (b *Bar) Leak() *int { // ERROR "leaking param: b to result ~r0$"
137 return &b.i // ERROR "&b.i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400138}
139
Dmitry Vyukov76477412015-02-19 22:00:11 +0300140func (b *Bar) AlsoNoLeak() *int { // ERROR "\(\*Bar\).AlsoNoLeak leaking param b content to result ~r0$"
Russ Cox00d2f912014-09-24 15:20:03 -0400141 return b.ii
142}
143
Dmitry Vyukov76477412015-02-19 22:00:11 +0300144func (b Bar) AlsoLeak() *int { // ERROR "leaking param: b to result ~r0$"
Russ Cox00d2f912014-09-24 15:20:03 -0400145 return b.ii
146}
147
Dmitry Vyukov76477412015-02-19 22:00:11 +0300148func (b Bar) LeaksToo() *int { // ERROR "leaking param: b to result ~r0$"
149 v := 0 // ERROR "moved to heap: v$"
150 b.ii = &v // ERROR "&v escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400151 return b.ii
152}
153
Dmitry Vyukov76477412015-02-19 22:00:11 +0300154func (b *Bar) LeaksABit() *int { // ERROR "\(\*Bar\).LeaksABit leaking param b content to result ~r0$"
155 v := 0 // ERROR "moved to heap: v$"
156 b.ii = &v // ERROR "&v escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400157 return b.ii
158}
159
Dmitry Vyukov76477412015-02-19 22:00:11 +0300160func (b Bar) StillNoLeak() int { // ERROR "Bar.StillNoLeak b does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400161 v := 0
Dmitry Vyukov76477412015-02-19 22:00:11 +0300162 b.ii = &v // ERROR "Bar.StillNoLeak &v does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400163 return b.i
164}
165
Dmitry Vyukov76477412015-02-19 22:00:11 +0300166func goLeak(b *Bar) { // ERROR "leaking param: b$"
Russ Cox00d2f912014-09-24 15:20:03 -0400167 go b.NoLeak()
168}
169
170type Bar2 struct {
171 i [12]int
172 ii []int
173}
174
175func NewBar2() *Bar2 {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300176 return &Bar2{[12]int{42}, nil} // ERROR "&Bar2 literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400177}
178
Dmitry Vyukov76477412015-02-19 22:00:11 +0300179func (b *Bar2) NoLeak() int { // ERROR "\(\*Bar2\).NoLeak b does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400180 return b.i[0]
181}
182
Dmitry Vyukov76477412015-02-19 22:00:11 +0300183func (b *Bar2) Leak() []int { // ERROR "leaking param: b to result ~r0$"
184 return b.i[:] // ERROR "b.i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400185}
186
Dmitry Vyukov76477412015-02-19 22:00:11 +0300187func (b *Bar2) AlsoNoLeak() []int { // ERROR "\(\*Bar2\).AlsoNoLeak leaking param b content to result ~r0$"
Russ Cox00d2f912014-09-24 15:20:03 -0400188 return b.ii[0:1]
189}
190
Dmitry Vyukov76477412015-02-19 22:00:11 +0300191func (b Bar2) AgainNoLeak() [12]int { // ERROR "Bar2.AgainNoLeak b does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400192 return b.i
193}
194
Dmitry Vyukov76477412015-02-19 22:00:11 +0300195func (b *Bar2) LeakSelf() { // ERROR "leaking param: b$"
196 b.ii = b.i[0:4] // ERROR "b.i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400197}
198
Dmitry Vyukov76477412015-02-19 22:00:11 +0300199func (b *Bar2) LeakSelf2() { // ERROR "leaking param: b$"
Russ Cox00d2f912014-09-24 15:20:03 -0400200 var buf []int
Dmitry Vyukov76477412015-02-19 22:00:11 +0300201 buf = b.i[0:] // ERROR "b.i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400202 b.ii = buf
203}
204
205func foo21() func() int {
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300206 x := 42
Dmitry Vyukov76477412015-02-19 22:00:11 +0300207 return func() int { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300208 return x
209 }
210}
211
212func foo21a() func() int {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300213 x := 42 // ERROR "moved to heap: x$"
214 return func() int { // ERROR "func literal escapes to heap$"
215 x++ // ERROR "&x escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300216 return x
Russ Cox00d2f912014-09-24 15:20:03 -0400217 }
218}
219
220func foo22() int {
221 x := 42
Dmitry Vyukov76477412015-02-19 22:00:11 +0300222 return func() int { // ERROR "foo22 func literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400223 return x
224 }()
225}
226
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300227func foo23(x int) func() int {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300228 return func() int { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300229 return x
Russ Cox00d2f912014-09-24 15:20:03 -0400230 }
231}
232
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300233func foo23a(x int) func() int {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300234 f := func() int { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300235 return x
Russ Cox00d2f912014-09-24 15:20:03 -0400236 }
237 return f
238}
239
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300240func foo23b(x int) *(func() int) {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300241 f := func() int { return x } // ERROR "func literal escapes to heap$" "moved to heap: f$"
242 return &f // ERROR "&f escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400243}
244
Dmitry Vyukov76477412015-02-19 22:00:11 +0300245func foo23c(x int) func() int { // ERROR "moved to heap: x$"
246 return func() int { // ERROR "func literal escapes to heap$"
247 x++ // ERROR "&x escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300248 return x
249 }
250}
251
Russ Cox00d2f912014-09-24 15:20:03 -0400252func foo24(x int) int {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300253 return func() int { // ERROR "foo24 func literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400254 return x
255 }()
256}
257
258var x *int
259
Dmitry Vyukov76477412015-02-19 22:00:11 +0300260func fooleak(xx *int) int { // ERROR "leaking param: xx$"
Russ Cox00d2f912014-09-24 15:20:03 -0400261 x = xx
262 return *x
263}
264
Dmitry Vyukov76477412015-02-19 22:00:11 +0300265func foonoleak(xx *int) int { // ERROR "foonoleak xx does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400266 return *x + *xx
267}
268
Dmitry Vyukov76477412015-02-19 22:00:11 +0300269func foo31(x int) int { // ERROR "moved to heap: x$"
270 return fooleak(&x) // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400271}
272
273func foo32(x int) int {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300274 return foonoleak(&x) // ERROR "foo32 &x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400275}
276
277type Foo struct {
278 xx *int
279 x int
280}
281
282var F Foo
283var pf *Foo
284
Dmitry Vyukov76477412015-02-19 22:00:11 +0300285func (f *Foo) fooleak() { // ERROR "leaking param: f$"
Russ Cox00d2f912014-09-24 15:20:03 -0400286 pf = f
287}
288
Dmitry Vyukov76477412015-02-19 22:00:11 +0300289func (f *Foo) foonoleak() { // ERROR "\(\*Foo\).foonoleak f does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400290 F.x = f.x
291}
292
Dmitry Vyukov76477412015-02-19 22:00:11 +0300293func (f *Foo) Leak() { // ERROR "leaking param: f$"
Russ Cox00d2f912014-09-24 15:20:03 -0400294 f.fooleak()
295}
296
Dmitry Vyukov76477412015-02-19 22:00:11 +0300297func (f *Foo) NoLeak() { // ERROR "\(\*Foo\).NoLeak f does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400298 f.foonoleak()
299}
300
Dmitry Vyukov76477412015-02-19 22:00:11 +0300301func foo41(x int) { // ERROR "moved to heap: x$"
302 F.xx = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400303}
304
Dmitry Vyukov76477412015-02-19 22:00:11 +0300305func (f *Foo) foo42(x int) { // ERROR "\(\*Foo\).foo42 f does not escape$" "moved to heap: x$"
306 f.xx = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400307}
308
Dmitry Vyukov76477412015-02-19 22:00:11 +0300309func foo43(f *Foo, x int) { // ERROR "foo43 f does not escape$" "moved to heap: x$"
310 f.xx = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400311}
312
Dmitry Vyukov76477412015-02-19 22:00:11 +0300313func foo44(yy *int) { // ERROR "leaking param: yy$"
Russ Cox00d2f912014-09-24 15:20:03 -0400314 F.xx = yy
315}
316
Dmitry Vyukov76477412015-02-19 22:00:11 +0300317func (f *Foo) foo45() { // ERROR "\(\*Foo\).foo45 f does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400318 F.x = f.x
319}
320
321// See foo13 above for explanation of why f leaks.
Dmitry Vyukov76477412015-02-19 22:00:11 +0300322func (f *Foo) foo46() { // ERROR "leaking param: f$"
Russ Cox00d2f912014-09-24 15:20:03 -0400323 F.xx = f.xx
324}
325
Dmitry Vyukov76477412015-02-19 22:00:11 +0300326func (f *Foo) foo47() { // ERROR "leaking param: f$"
327 f.xx = &f.x // ERROR "&f.x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400328}
329
330var ptrSlice []*int
331
Dmitry Vyukov76477412015-02-19 22:00:11 +0300332func foo50(i *int) { // ERROR "leaking param: i$"
Russ Cox00d2f912014-09-24 15:20:03 -0400333 ptrSlice[0] = i
334}
335
336var ptrMap map[*int]*int
337
Dmitry Vyukov76477412015-02-19 22:00:11 +0300338func foo51(i *int) { // ERROR "leaking param: i$"
Russ Cox00d2f912014-09-24 15:20:03 -0400339 ptrMap[i] = i
340}
341
Dmitry Vyukov76477412015-02-19 22:00:11 +0300342func indaddr1(x int) *int { // ERROR "moved to heap: x$"
343 return &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400344}
345
Dmitry Vyukov76477412015-02-19 22:00:11 +0300346func indaddr2(x *int) *int { // ERROR "leaking param: x to result ~r1$"
347 return *&x // ERROR "indaddr2 &x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400348}
349
Dmitry Vyukov76477412015-02-19 22:00:11 +0300350func indaddr3(x *int32) *int { // ERROR "leaking param: x to result ~r1$"
351 return *(**int)(unsafe.Pointer(&x)) // ERROR "indaddr3 &x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400352}
353
354// From package math:
355
356func Float32bits(f float32) uint32 {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300357 return *(*uint32)(unsafe.Pointer(&f)) // ERROR "Float32bits &f does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400358}
359
360func Float32frombits(b uint32) float32 {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300361 return *(*float32)(unsafe.Pointer(&b)) // ERROR "Float32frombits &b does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400362}
363
364func Float64bits(f float64) uint64 {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300365 return *(*uint64)(unsafe.Pointer(&f)) // ERROR "Float64bits &f does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400366}
367
368func Float64frombits(b uint64) float64 {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300369 return *(*float64)(unsafe.Pointer(&b)) // ERROR "Float64frombits &b does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400370}
371
372// contrast with
Dmitry Vyukov76477412015-02-19 22:00:11 +0300373func float64bitsptr(f float64) *uint64 { // ERROR "moved to heap: f$"
374 return (*uint64)(unsafe.Pointer(&f)) // ERROR "&f escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400375}
376
Dmitry Vyukov76477412015-02-19 22:00:11 +0300377func float64ptrbitsptr(f *float64) *uint64 { // ERROR "leaking param: f to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400378 return (*uint64)(unsafe.Pointer(f))
379}
380
Dmitry Vyukov76477412015-02-19 22:00:11 +0300381func typesw(i interface{}) *int { // ERROR "leaking param: i to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400382 switch val := i.(type) {
383 case *int:
384 return val
385 case *int8:
Dmitry Vyukov76477412015-02-19 22:00:11 +0300386 v := int(*val) // ERROR "moved to heap: v$"
387 return &v // ERROR "&v escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400388 }
389 return nil
390}
391
Dmitry Vyukov76477412015-02-19 22:00:11 +0300392func exprsw(i *int) *int { // ERROR "leaking param: i to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400393 switch j := i; *j + 110 {
394 case 12:
395 return j
396 case 42:
397 return nil
398 }
399 return nil
400
401}
402
403// assigning to an array element is like assigning to the array
Dmitry Vyukov76477412015-02-19 22:00:11 +0300404func foo60(i *int) *int { // ERROR "leaking param: i to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400405 var a [12]*int
406 a[0] = i
407 return a[1]
408}
409
Dmitry Vyukov76477412015-02-19 22:00:11 +0300410func foo60a(i *int) *int { // ERROR "foo60a i does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400411 var a [12]*int
412 a[0] = i
413 return nil
414}
415
416// assigning to a struct field is like assigning to the struct
Dmitry Vyukov76477412015-02-19 22:00:11 +0300417func foo61(i *int) *int { // ERROR "leaking param: i to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400418 type S struct {
419 a, b *int
420 }
421 var s S
422 s.a = i
423 return s.b
424}
425
Dmitry Vyukov76477412015-02-19 22:00:11 +0300426func foo61a(i *int) *int { // ERROR "foo61a i does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400427 type S struct {
428 a, b *int
429 }
430 var s S
431 s.a = i
432 return nil
433}
434
435// assigning to a struct field is like assigning to the struct but
436// here this subtlety is lost, since s.a counts as an assignment to a
437// track-losing dereference.
Dmitry Vyukov76477412015-02-19 22:00:11 +0300438func foo62(i *int) *int { // ERROR "leaking param: i$"
Russ Cox00d2f912014-09-24 15:20:03 -0400439 type S struct {
440 a, b *int
441 }
Dmitry Vyukov76477412015-02-19 22:00:11 +0300442 s := new(S) // ERROR "foo62 new\(S\) does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400443 s.a = i
444 return nil // s.b
445}
446
447type M interface {
448 M()
449}
450
Dmitry Vyukov76477412015-02-19 22:00:11 +0300451func foo63(m M) { // ERROR "foo63 m does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400452}
453
Dmitry Vyukov76477412015-02-19 22:00:11 +0300454func foo64(m M) { // ERROR "leaking param: m$"
Russ Cox00d2f912014-09-24 15:20:03 -0400455 m.M()
456}
457
Dmitry Vyukov76477412015-02-19 22:00:11 +0300458func foo64b(m M) { // ERROR "leaking param: m$"
Russ Cox00d2f912014-09-24 15:20:03 -0400459 defer m.M()
460}
461
462type MV int
463
464func (MV) M() {}
465
466func foo65() {
467 var mv MV
Dmitry Vyukov76477412015-02-19 22:00:11 +0300468 foo63(&mv) // ERROR "foo65 &mv does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400469}
470
471func foo66() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300472 var mv MV // ERROR "moved to heap: mv$"
473 foo64(&mv) // ERROR "&mv escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400474}
475
476func foo67() {
477 var mv MV
Dmitry Vyukov76477412015-02-19 22:00:11 +0300478 foo63(mv) // ERROR "foo67 mv does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400479}
480
481func foo68() {
482 var mv MV
Dmitry Vyukovedcc0622015-02-19 16:27:32 +0300483 // escapes but it's an int so irrelevant
Dmitry Vyukov76477412015-02-19 22:00:11 +0300484 foo64(mv) // ERROR "mv escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400485}
486
Dmitry Vyukov76477412015-02-19 22:00:11 +0300487func foo69(m M) { // ERROR "leaking param: m$"
Russ Cox00d2f912014-09-24 15:20:03 -0400488 foo64(m)
489}
490
Dmitry Vyukov76477412015-02-19 22:00:11 +0300491func foo70(mv1 *MV, m M) { // ERROR "leaking param: m$" "leaking param: mv1$"
492 m = mv1 // ERROR "mv1 escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400493 foo64(m)
494}
495
Dmitry Vyukov76477412015-02-19 22:00:11 +0300496func foo71(x *int) []*int { // ERROR "leaking param: x$"
Russ Cox00d2f912014-09-24 15:20:03 -0400497 var y []*int
498 y = append(y, x)
499 return y
500}
501
Dmitry Vyukov76477412015-02-19 22:00:11 +0300502func foo71a(x int) []*int { // ERROR "moved to heap: x$"
Russ Cox00d2f912014-09-24 15:20:03 -0400503 var y []*int
Dmitry Vyukov76477412015-02-19 22:00:11 +0300504 y = append(y, &x) // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400505 return y
506}
507
508func foo72() {
509 var x int
510 var y [1]*int
Dmitry Vyukov76477412015-02-19 22:00:11 +0300511 y[0] = &x // ERROR "foo72 &x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400512}
513
514func foo72aa() [10]*int {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300515 var x int // ERROR "moved to heap: x$"
Russ Cox00d2f912014-09-24 15:20:03 -0400516 var y [10]*int
Dmitry Vyukov76477412015-02-19 22:00:11 +0300517 y[0] = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400518 return y
519}
520
521func foo72a() {
522 var y [10]*int
523 for i := 0; i < 10; i++ {
524 // escapes its scope
Dmitry Vyukov76477412015-02-19 22:00:11 +0300525 x := i // ERROR "moved to heap: x$"
526 y[i] = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400527 }
528 return
529}
530
531func foo72b() [10]*int {
532 var y [10]*int
533 for i := 0; i < 10; i++ {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300534 x := i // ERROR "moved to heap: x$"
535 y[i] = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400536 }
537 return y
538}
539
540// issue 2145
541func foo73() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300542 s := []int{3, 2, 1} // ERROR "foo73 \[\]int literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400543 for _, v := range s {
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300544 vv := v
545 // actually just escapes its scope
Dmitry Vyukov76477412015-02-19 22:00:11 +0300546 defer func() { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300547 println(vv)
548 }()
549 }
550}
551
552func foo731() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300553 s := []int{3, 2, 1} // ERROR "foo731 \[\]int literal does not escape$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300554 for _, v := range s {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300555 vv := v // ERROR "moved to heap: vv$"
Russ Cox00d2f912014-09-24 15:20:03 -0400556 // actually just escapes its scope
Dmitry Vyukov76477412015-02-19 22:00:11 +0300557 defer func() { // ERROR "func literal escapes to heap$"
558 vv = 42 // ERROR "&vv escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300559 println(vv)
Russ Cox00d2f912014-09-24 15:20:03 -0400560 }()
561 }
562}
563
564func foo74() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300565 s := []int{3, 2, 1} // ERROR "foo74 \[\]int literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400566 for _, v := range s {
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300567 vv := v
568 // actually just escapes its scope
Dmitry Vyukov76477412015-02-19 22:00:11 +0300569 fn := func() { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300570 println(vv)
571 }
572 defer fn()
573 }
574}
575
576func foo74a() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300577 s := []int{3, 2, 1} // ERROR "foo74a \[\]int literal does not escape$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300578 for _, v := range s {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300579 vv := v // ERROR "moved to heap: vv$"
Russ Cox00d2f912014-09-24 15:20:03 -0400580 // actually just escapes its scope
Dmitry Vyukov76477412015-02-19 22:00:11 +0300581 fn := func() { // ERROR "func literal escapes to heap$"
582 vv += 1 // ERROR "&vv escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300583 println(vv)
Russ Cox00d2f912014-09-24 15:20:03 -0400584 }
585 defer fn()
586 }
587}
588
589// issue 3975
590func foo74b() {
591 var array [3]func()
Dmitry Vyukov76477412015-02-19 22:00:11 +0300592 s := []int{3, 2, 1} // ERROR "foo74b \[\]int literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400593 for i, v := range s {
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300594 vv := v
595 // actually just escapes its scope
Dmitry Vyukov76477412015-02-19 22:00:11 +0300596 array[i] = func() { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300597 println(vv)
598 }
599 }
600}
601
602func foo74c() {
603 var array [3]func()
Dmitry Vyukov76477412015-02-19 22:00:11 +0300604 s := []int{3, 2, 1} // ERROR "foo74c \[\]int literal does not escape$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +0300605 for i, v := range s {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300606 vv := v // ERROR "moved to heap: vv$"
Russ Cox00d2f912014-09-24 15:20:03 -0400607 // actually just escapes its scope
Dmitry Vyukov76477412015-02-19 22:00:11 +0300608 array[i] = func() { // ERROR "func literal escapes to heap$"
609 println(&vv) // ERROR "&vv escapes to heap$" "<S> &vv does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400610 }
611 }
612}
613
Dmitry Vyukov76477412015-02-19 22:00:11 +0300614func myprint(y *int, x ...interface{}) *int { // ERROR "leaking param: y to result ~r2$" "myprint x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400615 return y
616}
617
Dmitry Vyukov76477412015-02-19 22:00:11 +0300618func myprint1(y *int, x ...interface{}) *interface{} { // ERROR "leaking param: x to result ~r2$" "myprint1 y does not escape$"
619 return &x[0] // ERROR "&x\[0\] escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400620}
621
Dmitry Vyukov76477412015-02-19 22:00:11 +0300622func foo75(z *int) { // ERROR "foo75 z does not escape$"
623 myprint(z, 1, 2, 3) // ERROR "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "foo75 ... argument does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400624}
625
Dmitry Vyukov76477412015-02-19 22:00:11 +0300626func foo75a(z *int) { // ERROR "foo75a z does not escape$"
627 myprint1(z, 1, 2, 3) // ERROR "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "foo75a ... argument does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400628}
629
Dmitry Vyukov76477412015-02-19 22:00:11 +0300630func foo75esc(z *int) { // ERROR "leaking param: z$"
631 gxx = myprint(z, 1, 2, 3) // ERROR "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "foo75esc ... argument does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400632}
633
Dmitry Vyukov76477412015-02-19 22:00:11 +0300634func foo75aesc(z *int) { // ERROR "foo75aesc z does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400635 var ppi **interface{} // assignments to pointer dereferences lose track
Dmitry Vyukov76477412015-02-19 22:00:11 +0300636 *ppi = myprint1(z, 1, 2, 3) // ERROR "... argument escapes to heap$" "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400637}
638
Dmitry Vyukov76477412015-02-19 22:00:11 +0300639func foo75aesc1(z *int) { // ERROR "foo75aesc1 z does not escape$"
640 sink = myprint1(z, 1, 2, 3) // ERROR "... argument escapes to heap$" "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "myprint1\(z, 1, 2, 3\) escapes to heap$"
Dmitry Vyukov1f5617e2015-02-19 17:54:55 +0300641}
642
643// BAD: z does not escape here
Dmitry Vyukov76477412015-02-19 22:00:11 +0300644func foo76(z *int) { // ERROR "leaking param: z$"
645 myprint(nil, z) // ERROR "foo76 ... argument does not escape$" "z escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400646}
647
Dmitry Vyukov1f5617e2015-02-19 17:54:55 +0300648// BAD: z does not escape here
Dmitry Vyukov76477412015-02-19 22:00:11 +0300649func foo76a(z *int) { // ERROR "leaking param: z$"
650 myprint1(nil, z) // ERROR "foo76a ... argument does not escape$" "z escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400651}
652
653func foo76b() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300654 myprint(nil, 1, 2, 3) // ERROR "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "foo76b ... argument does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400655}
656
657func foo76c() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300658 myprint1(nil, 1, 2, 3) // ERROR "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "foo76c ... argument does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400659}
660
661func foo76d() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300662 defer myprint(nil, 1, 2, 3) // ERROR "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "foo76d ... argument does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400663}
664
665func foo76e() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300666 defer myprint1(nil, 1, 2, 3) // ERROR "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$" "foo76e ... argument does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400667}
668
669func foo76f() {
670 for {
671 // TODO: This one really only escapes its scope, but we don't distinguish yet.
Dmitry Vyukov76477412015-02-19 22:00:11 +0300672 defer myprint(nil, 1, 2, 3) // ERROR "... argument escapes to heap$" "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400673 }
674}
675
676func foo76g() {
677 for {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300678 defer myprint1(nil, 1, 2, 3) // ERROR "... argument escapes to heap$" "1 escapes to heap$" "2 escapes to heap$" "3 escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400679 }
680}
681
Dmitry Vyukov76477412015-02-19 22:00:11 +0300682func foo77(z []interface{}) { // ERROR "foo77 z does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400683 myprint(nil, z...) // z does not escape
684}
685
Dmitry Vyukov76477412015-02-19 22:00:11 +0300686func foo77a(z []interface{}) { // ERROR "foo77a z does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400687 myprint1(nil, z...)
688}
689
Dmitry Vyukov76477412015-02-19 22:00:11 +0300690func foo77b(z []interface{}) { // ERROR "leaking param: z$"
Russ Cox00d2f912014-09-24 15:20:03 -0400691 var ppi **interface{}
692 *ppi = myprint1(nil, z...)
693}
694
Dmitry Vyukov76477412015-02-19 22:00:11 +0300695func foo77c(z []interface{}) { // ERROR "leaking param: z$"
696 sink = myprint1(nil, z...) // ERROR "myprint1\(nil, z...\) escapes to heap$"
Dmitry Vyukov1f5617e2015-02-19 17:54:55 +0300697}
698
699func dotdotdot() {
700 // BAD: i should not escape here
Dmitry Vyukov76477412015-02-19 22:00:11 +0300701 i := 0 // ERROR "moved to heap: i$"
702 myprint(nil, &i) // ERROR "&i escapes to heap$" "dotdotdot ... argument does not escape$"
Dmitry Vyukov1f5617e2015-02-19 17:54:55 +0300703
704 // BAD: j should not escape here
Dmitry Vyukov76477412015-02-19 22:00:11 +0300705 j := 0 // ERROR "moved to heap: j$"
706 myprint1(nil, &j) // ERROR "&j escapes to heap$" "dotdotdot ... argument does not escape$"
Dmitry Vyukov1f5617e2015-02-19 17:54:55 +0300707}
708
Dmitry Vyukov76477412015-02-19 22:00:11 +0300709func foo78(z int) *int { // ERROR "moved to heap: z$"
710 return &z // ERROR "&z escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400711}
712
Dmitry Vyukov76477412015-02-19 22:00:11 +0300713func foo78a(z int) *int { // ERROR "moved to heap: z$"
714 y := &z // ERROR "&z escapes to heap$"
715 x := &y // ERROR "foo78a &y does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400716 return *x // really return y
717}
718
719func foo79() *int {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300720 return new(int) // ERROR "new\(int\) escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400721}
722
723func foo80() *int {
724 var z *int
725 for {
726 // Really just escapes its scope but we don't distinguish
Dmitry Vyukov76477412015-02-19 22:00:11 +0300727 z = new(int) // ERROR "new\(int\) escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400728 }
729 _ = z
730 return nil
731}
732
733func foo81() *int {
734 for {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300735 z := new(int) // ERROR "foo81 new\(int\) does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400736 _ = z
737 }
738 return nil
739}
740
Dmitry Vyukov76477412015-02-19 22:00:11 +0300741func tee(p *int) (x, y *int) { return p, p } // ERROR "leaking param: p to result x$" "leaking param: p to result y$"
Russ Cox00d2f912014-09-24 15:20:03 -0400742
Dmitry Vyukov76477412015-02-19 22:00:11 +0300743func noop(x, y *int) {} // ERROR "noop x does not escape$" "noop y does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400744
745func foo82() {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300746 var x, y, z int // ERROR "moved to heap: x$" "moved to heap: y$" "moved to heap: z$"
747 go noop(tee(&z)) // ERROR "&z escapes to heap$"
748 go noop(&x, &y) // ERROR "&x escapes to heap$" "&y escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400749 for {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300750 var u, v, w int // ERROR "moved to heap: u$" "moved to heap: v$" "moved to heap: w$"
751 defer noop(tee(&u)) // ERROR "&u escapes to heap$"
752 defer noop(&v, &w) // ERROR "&v escapes to heap$" "&w escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400753 }
754}
755
756type Fooer interface {
757 Foo()
758}
759
760type LimitedFooer struct {
761 Fooer
762 N int64
763}
764
Dmitry Vyukov76477412015-02-19 22:00:11 +0300765func LimitFooer(r Fooer, n int64) Fooer { // ERROR "leaking param: r$"
766 return &LimitedFooer{r, n} // ERROR "&LimitedFooer literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400767}
768
Dmitry Vyukov76477412015-02-19 22:00:11 +0300769func foo90(x *int) map[*int]*int { // ERROR "leaking param: x$"
770 return map[*int]*int{nil: x} // ERROR "map\[\*int\]\*int literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400771}
772
Dmitry Vyukov76477412015-02-19 22:00:11 +0300773func foo91(x *int) map[*int]*int { // ERROR "leaking param: x$"
774 return map[*int]*int{x: nil} // ERROR "map\[\*int\]\*int literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400775}
776
Dmitry Vyukov76477412015-02-19 22:00:11 +0300777func foo92(x *int) [2]*int { // ERROR "leaking param: x to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400778 return [2]*int{x, nil}
779}
780
781// does not leak c
Dmitry Vyukov76477412015-02-19 22:00:11 +0300782func foo93(c chan *int) *int { // ERROR "foo93 c does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400783 for v := range c {
784 return v
785 }
786 return nil
787}
788
789// does not leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300790func foo94(m map[*int]*int, b bool) *int { // ERROR "foo94 m does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400791 for k, v := range m {
792 if b {
793 return k
794 }
795 return v
796 }
797 return nil
798}
799
800// does leak x
Dmitry Vyukov76477412015-02-19 22:00:11 +0300801func foo95(m map[*int]*int, x *int) { // ERROR "foo95 m does not escape$" "leaking param: x$"
Russ Cox00d2f912014-09-24 15:20:03 -0400802 m[x] = x
803}
804
805// does not leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300806func foo96(m []*int) *int { // ERROR "foo96 m does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400807 return m[0]
808}
809
810// does leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300811func foo97(m [1]*int) *int { // ERROR "leaking param: m to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400812 return m[0]
813}
814
815// does not leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300816func foo98(m map[int]*int) *int { // ERROR "foo98 m does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400817 return m[0]
818}
819
820// does leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300821func foo99(m *[1]*int) []*int { // ERROR "leaking param: m to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400822 return m[:]
823}
824
825// does not leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300826func foo100(m []*int) *int { // ERROR "foo100 m does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400827 for _, v := range m {
828 return v
829 }
830 return nil
831}
832
833// does leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300834func foo101(m [1]*int) *int { // ERROR "leaking param: m to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400835 for _, v := range m {
836 return v
837 }
838 return nil
839}
840
841// does not leak m
Dmitry Vyukov76477412015-02-19 22:00:11 +0300842func foo101a(m [1]*int) *int { // ERROR "foo101a m does not escape$"
843 for i := range m { // ERROR "moved to heap: i$"
844 return &i // ERROR "&i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400845 }
846 return nil
847}
848
849// does leak x
Dmitry Vyukov76477412015-02-19 22:00:11 +0300850func foo102(m []*int, x *int) { // ERROR "foo102 m does not escape$" "leaking param: x$"
Russ Cox00d2f912014-09-24 15:20:03 -0400851 m[0] = x
852}
853
854// does not leak x
Dmitry Vyukov76477412015-02-19 22:00:11 +0300855func foo103(m [1]*int, x *int) { // ERROR "foo103 m does not escape$" "foo103 x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400856 m[0] = x
857}
858
859var y []*int
860
861// does not leak x
Dmitry Vyukov76477412015-02-19 22:00:11 +0300862func foo104(x []*int) { // ERROR "foo104 x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400863 copy(y, x)
864}
865
866// does not leak x
Dmitry Vyukov76477412015-02-19 22:00:11 +0300867func foo105(x []*int) { // ERROR "foo105 x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400868 _ = append(y, x...)
869}
870
871// does leak x
Dmitry Vyukov76477412015-02-19 22:00:11 +0300872func foo106(x *int) { // ERROR "leaking param: x$"
Russ Cox00d2f912014-09-24 15:20:03 -0400873 _ = append(y, x)
874}
875
Dmitry Vyukov76477412015-02-19 22:00:11 +0300876func foo107(x *int) map[*int]*int { // ERROR "leaking param: x$"
877 return map[*int]*int{x: nil} // ERROR "map\[\*int\]\*int literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400878}
879
Dmitry Vyukov76477412015-02-19 22:00:11 +0300880func foo108(x *int) map[*int]*int { // ERROR "leaking param: x$"
881 return map[*int]*int{nil: x} // ERROR "map\[\*int\]\*int literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400882}
883
Dmitry Vyukov76477412015-02-19 22:00:11 +0300884func foo109(x *int) *int { // ERROR "leaking param: x$"
885 m := map[*int]*int{x: nil} // ERROR "foo109 map\[\*int\]\*int literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400886 for k, _ := range m {
887 return k
888 }
889 return nil
890}
891
Dmitry Vyukov76477412015-02-19 22:00:11 +0300892func foo110(x *int) *int { // ERROR "leaking param: x$"
893 m := map[*int]*int{nil: x} // ERROR "foo110 map\[\*int\]\*int literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400894 return m[nil]
895}
896
Dmitry Vyukov76477412015-02-19 22:00:11 +0300897func foo111(x *int) *int { // ERROR "leaking param: x$"
898 m := []*int{x} // ERROR "foo111 \[\]\*int literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400899 return m[0]
900}
901
Dmitry Vyukov76477412015-02-19 22:00:11 +0300902func foo112(x *int) *int { // ERROR "leaking param: x to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400903 m := [1]*int{x}
904 return m[0]
905}
906
Dmitry Vyukov76477412015-02-19 22:00:11 +0300907func foo113(x *int) *int { // ERROR "leaking param: x to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400908 m := Bar{ii: x}
909 return m.ii
910}
911
Dmitry Vyukov76477412015-02-19 22:00:11 +0300912func foo114(x *int) *int { // ERROR "leaking param: x to result ~r1$"
913 m := &Bar{ii: x} // ERROR "foo114 &Bar literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -0400914 return m.ii
915}
916
Dmitry Vyukov76477412015-02-19 22:00:11 +0300917func foo115(x *int) *int { // ERROR "leaking param: x to result ~r1$"
Russ Cox00d2f912014-09-24 15:20:03 -0400918 return (*int)(unsafe.Pointer(uintptr(unsafe.Pointer(x)) + 1))
919}
920
921func foo116(b bool) *int {
922 if b {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300923 x := 1 // ERROR "moved to heap: x$"
924 return &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400925 } else {
Dmitry Vyukov76477412015-02-19 22:00:11 +0300926 y := 1 // ERROR "moved to heap: y$"
927 return &y // ERROR "&y escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400928 }
929 return nil
930}
931
Dmitry Vyukov76477412015-02-19 22:00:11 +0300932func foo117(unknown func(interface{})) { // ERROR "foo117 unknown does not escape$"
933 x := 1 // ERROR "moved to heap: x$"
934 unknown(&x) // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400935}
936
Dmitry Vyukov76477412015-02-19 22:00:11 +0300937func foo118(unknown func(*int)) { // ERROR "foo118 unknown does not escape$"
938 x := 1 // ERROR "moved to heap: x$"
939 unknown(&x) // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -0400940}
941
942func external(*int)
943
Dmitry Vyukov76477412015-02-19 22:00:11 +0300944func foo119(x *int) { // ERROR "leaking param: x$"
Russ Cox00d2f912014-09-24 15:20:03 -0400945 external(x)
946}
947
948func foo120() {
949 // formerly exponential time analysis
950L1:
951L2:
952L3:
953L4:
954L5:
955L6:
956L7:
957L8:
958L9:
959L10:
960L11:
961L12:
962L13:
963L14:
964L15:
965L16:
966L17:
967L18:
968L19:
969L20:
970L21:
971L22:
972L23:
973L24:
974L25:
975L26:
976L27:
977L28:
978L29:
979L30:
980L31:
981L32:
982L33:
983L34:
984L35:
985L36:
986L37:
987L38:
988L39:
989L40:
990L41:
991L42:
992L43:
993L44:
994L45:
995L46:
996L47:
997L48:
998L49:
999L50:
1000L51:
1001L52:
1002L53:
1003L54:
1004L55:
1005L56:
1006L57:
1007L58:
1008L59:
1009L60:
1010L61:
1011L62:
1012L63:
1013L64:
1014L65:
1015L66:
1016L67:
1017L68:
1018L69:
1019L70:
1020L71:
1021L72:
1022L73:
1023L74:
1024L75:
1025L76:
1026L77:
1027L78:
1028L79:
1029L80:
1030L81:
1031L82:
1032L83:
1033L84:
1034L85:
1035L86:
1036L87:
1037L88:
1038L89:
1039L90:
1040L91:
1041L92:
1042L93:
1043L94:
1044L95:
1045L96:
1046L97:
1047L98:
1048L99:
1049L100:
1050 // use the labels to silence compiler errors
1051 goto L1
1052 goto L2
1053 goto L3
1054 goto L4
1055 goto L5
1056 goto L6
1057 goto L7
1058 goto L8
1059 goto L9
1060 goto L10
1061 goto L11
1062 goto L12
1063 goto L13
1064 goto L14
1065 goto L15
1066 goto L16
1067 goto L17
1068 goto L18
1069 goto L19
1070 goto L20
1071 goto L21
1072 goto L22
1073 goto L23
1074 goto L24
1075 goto L25
1076 goto L26
1077 goto L27
1078 goto L28
1079 goto L29
1080 goto L30
1081 goto L31
1082 goto L32
1083 goto L33
1084 goto L34
1085 goto L35
1086 goto L36
1087 goto L37
1088 goto L38
1089 goto L39
1090 goto L40
1091 goto L41
1092 goto L42
1093 goto L43
1094 goto L44
1095 goto L45
1096 goto L46
1097 goto L47
1098 goto L48
1099 goto L49
1100 goto L50
1101 goto L51
1102 goto L52
1103 goto L53
1104 goto L54
1105 goto L55
1106 goto L56
1107 goto L57
1108 goto L58
1109 goto L59
1110 goto L60
1111 goto L61
1112 goto L62
1113 goto L63
1114 goto L64
1115 goto L65
1116 goto L66
1117 goto L67
1118 goto L68
1119 goto L69
1120 goto L70
1121 goto L71
1122 goto L72
1123 goto L73
1124 goto L74
1125 goto L75
1126 goto L76
1127 goto L77
1128 goto L78
1129 goto L79
1130 goto L80
1131 goto L81
1132 goto L82
1133 goto L83
1134 goto L84
1135 goto L85
1136 goto L86
1137 goto L87
1138 goto L88
1139 goto L89
1140 goto L90
1141 goto L91
1142 goto L92
1143 goto L93
1144 goto L94
1145 goto L95
1146 goto L96
1147 goto L97
1148 goto L98
1149 goto L99
1150 goto L100
1151}
1152
1153func foo121() {
1154 for i := 0; i < 10; i++ {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001155 defer myprint(nil, i) // ERROR "... argument escapes to heap$" "i escapes to heap$"
1156 go myprint(nil, i) // ERROR "... argument escapes to heap$" "i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001157 }
1158}
1159
1160// same as foo121 but check across import
1161func foo121b() {
1162 for i := 0; i < 10; i++ {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001163 defer fmt.Printf("%d", i) // ERROR "... argument escapes to heap$" "i escapes to heap$"
1164 go fmt.Printf("%d", i) // ERROR "... argument escapes to heap$" "i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001165 }
1166}
1167
1168// a harmless forward jump
1169func foo122() {
1170 var i *int
1171
1172 goto L1
1173L1:
Dmitry Vyukov76477412015-02-19 22:00:11 +03001174 i = new(int) // ERROR "foo122 new\(int\) does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001175 _ = i
1176}
1177
1178// a backward jump, increases loopdepth
1179func foo123() {
1180 var i *int
1181
1182L1:
Dmitry Vyukov76477412015-02-19 22:00:11 +03001183 i = new(int) // ERROR "new\(int\) escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001184
1185 goto L1
1186 _ = i
1187}
1188
Dmitry Vyukov76477412015-02-19 22:00:11 +03001189func foo124(x **int) { // ERROR "foo124 x does not escape$"
1190 var i int // ERROR "moved to heap: i$"
1191 p := &i // ERROR "&i escapes to heap$"
1192 func() { // ERROR "foo124 func literal does not escape$"
1193 *x = p // ERROR "leaking closure reference p$"
Russ Cox00d2f912014-09-24 15:20:03 -04001194 }()
1195}
1196
Dmitry Vyukov76477412015-02-19 22:00:11 +03001197func foo125(ch chan *int) { // ERROR "foo125 ch does not escape$"
1198 var i int // ERROR "moved to heap: i$"
1199 p := &i // ERROR "&i escapes to heap$"
1200 func() { // ERROR "foo125 func literal does not escape$"
1201 ch <- p // ERROR "leaking closure reference p$"
Russ Cox00d2f912014-09-24 15:20:03 -04001202 }()
1203}
1204
1205func foo126() {
1206 var px *int // loopdepth 0
1207 for {
1208 // loopdepth 1
Dmitry Vyukov76477412015-02-19 22:00:11 +03001209 var i int // ERROR "moved to heap: i$"
1210 func() { // ERROR "foo126 func literal does not escape$"
1211 px = &i // ERROR "&i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001212 }()
1213 }
1214 _ = px
1215}
1216
1217var px *int
1218
1219func foo127() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001220 var i int // ERROR "moved to heap: i$"
1221 p := &i // ERROR "&i escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001222 q := p
1223 px = q
1224}
1225
1226func foo128() {
1227 var i int
Dmitry Vyukov76477412015-02-19 22:00:11 +03001228 p := &i // ERROR "foo128 &i does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001229 q := p
1230 _ = q
1231}
1232
1233func foo129() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001234 var i int // ERROR "moved to heap: i$"
1235 p := &i // ERROR "&i escapes to heap$"
1236 func() { // ERROR "foo129 func literal does not escape$"
1237 q := p // ERROR "leaking closure reference p$"
1238 func() { // ERROR "<S> func literal does not escape$"
1239 r := q // ERROR "leaking closure reference q$"
Russ Cox00d2f912014-09-24 15:20:03 -04001240 px = r
1241 }()
1242 }()
1243}
1244
1245func foo130() {
1246 for {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001247 var i int // ERROR "moved to heap: i$"
1248 func() { // ERROR "foo130 func literal does not escape$"
1249 px = &i // ERROR "&i escapes to heap$" "leaking closure reference i$"
Russ Cox00d2f912014-09-24 15:20:03 -04001250 }()
1251 }
1252}
1253
1254func foo131() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001255 var i int // ERROR "moved to heap: i$"
1256 func() { // ERROR "foo131 func literal does not escape$"
1257 px = &i // ERROR "&i escapes to heap$" "leaking closure reference i$"
Russ Cox00d2f912014-09-24 15:20:03 -04001258 }()
1259}
1260
1261func foo132() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001262 var i int // ERROR "moved to heap: i$"
1263 go func() { // ERROR "func literal escapes to heap$"
1264 px = &i // ERROR "&i escapes to heap$" "leaking closure reference i$"
Russ Cox00d2f912014-09-24 15:20:03 -04001265 }()
1266}
1267
1268func foo133() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001269 var i int // ERROR "moved to heap: i$"
1270 defer func() { // ERROR "foo133 func literal does not escape$"
1271 px = &i // ERROR "&i escapes to heap$" "leaking closure reference i$"
Russ Cox00d2f912014-09-24 15:20:03 -04001272 }()
1273}
1274
1275func foo134() {
1276 var i int
Dmitry Vyukov76477412015-02-19 22:00:11 +03001277 p := &i // ERROR "foo134 &i does not escape$"
1278 func() { // ERROR "foo134 func literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001279 q := p
Dmitry Vyukov76477412015-02-19 22:00:11 +03001280 func() { // ERROR "<S> func literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001281 r := q
1282 _ = r
1283 }()
1284 }()
1285}
1286
1287func foo135() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001288 var i int // ERROR "moved to heap: i$"
1289 p := &i // ERROR "&i escapes to heap$"
1290 go func() { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +03001291 q := p
Dmitry Vyukov76477412015-02-19 22:00:11 +03001292 func() { // ERROR "<S> func literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001293 r := q
1294 _ = r
1295 }()
1296 }()
1297}
1298
1299func foo136() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001300 var i int // ERROR "moved to heap: i$"
1301 p := &i // ERROR "&i escapes to heap$"
1302 go func() { // ERROR "func literal escapes to heap$"
1303 q := p // ERROR "leaking closure reference p$"
1304 func() { // ERROR "<S> func literal does not escape$"
1305 r := q // ERROR "leaking closure reference q$"
Russ Cox00d2f912014-09-24 15:20:03 -04001306 px = r
1307 }()
1308 }()
1309}
1310
1311func foo137() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001312 var i int // ERROR "moved to heap: i$"
1313 p := &i // ERROR "&i escapes to heap$"
1314 func() { // ERROR "foo137 func literal does not escape$"
1315 q := p // ERROR "leaking closure reference p$"
1316 go func() { // ERROR "func literal escapes to heap$"
Dmitry Vyukov0e80b2e2015-01-19 22:59:58 +03001317 r := q
Russ Cox00d2f912014-09-24 15:20:03 -04001318 _ = r
1319 }()
1320 }()
1321}
1322
1323func foo138() *byte {
1324 type T struct {
1325 x [1]byte
1326 }
Dmitry Vyukov76477412015-02-19 22:00:11 +03001327 t := new(T) // ERROR "new\(T\) escapes to heap$"
1328 return &t.x[0] // ERROR "&t.x\[0\] escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001329}
1330
1331func foo139() *byte {
1332 type T struct {
1333 x struct {
1334 y byte
1335 }
1336 }
Dmitry Vyukov76477412015-02-19 22:00:11 +03001337 t := new(T) // ERROR "new\(T\) escapes to heap$"
1338 return &t.x.y // ERROR "&t.x.y escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001339}
1340
1341// issue 4751
1342func foo140() interface{} {
1343 type T struct {
1344 X string
1345 }
1346 type U struct {
1347 X string
1348 T *T
1349 }
Dmitry Vyukov76477412015-02-19 22:00:11 +03001350 t := &T{} // ERROR "&T literal escapes to heap$"
1351 return U{ // ERROR "U literal escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001352 X: t.X,
1353 T: t,
1354 }
1355}
1356
1357//go:noescape
1358
1359func F1([]byte)
1360
1361func F2([]byte)
1362
1363//go:noescape
1364
Dmitry Vyukov76477412015-02-19 22:00:11 +03001365func F3(x []byte) // ERROR "F3 x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001366
1367func F4(x []byte)
1368
1369func G() {
1370 var buf1 [10]byte
Dmitry Vyukov76477412015-02-19 22:00:11 +03001371 F1(buf1[:]) // ERROR "G buf1 does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001372
Dmitry Vyukov76477412015-02-19 22:00:11 +03001373 var buf2 [10]byte // ERROR "moved to heap: buf2$"
1374 F2(buf2[:]) // ERROR "buf2 escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001375
1376 var buf3 [10]byte
Dmitry Vyukov76477412015-02-19 22:00:11 +03001377 F3(buf3[:]) // ERROR "G buf3 does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001378
Dmitry Vyukov76477412015-02-19 22:00:11 +03001379 var buf4 [10]byte // ERROR "moved to heap: buf4$"
1380 F4(buf4[:]) // ERROR "buf4 escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001381}
1382
1383type Tm struct {
1384 x int
1385}
1386
Dmitry Vyukov76477412015-02-19 22:00:11 +03001387func (t *Tm) M() { // ERROR "\(\*Tm\).M t does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001388}
1389
1390func foo141() {
1391 var f func()
1392
Dmitry Vyukov76477412015-02-19 22:00:11 +03001393 t := new(Tm) // ERROR "new\(Tm\) escapes to heap$"
1394 f = t.M // ERROR "foo141 t.M does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001395 _ = f
1396}
1397
1398var gf func()
1399
1400func foo142() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001401 t := new(Tm) // ERROR "new\(Tm\) escapes to heap$"
1402 gf = t.M // ERROR "t.M escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001403}
1404
1405// issue 3888.
1406func foo143() {
1407 for i := 0; i < 1000; i++ {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001408 func() { // ERROR "foo143 func literal does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001409 for i := 0; i < 1; i++ {
1410 var t Tm
Dmitry Vyukov76477412015-02-19 22:00:11 +03001411 t.M() // ERROR "<S> t does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001412 }
1413 }()
1414 }
1415}
1416
1417// issue 5773
1418// Check that annotations take effect regardless of whether they
1419// are before or after the use in the source code.
1420
1421//go:noescape
1422
1423func foo144a(*int)
1424
1425func foo144() {
1426 var x int
Dmitry Vyukov76477412015-02-19 22:00:11 +03001427 foo144a(&x) // ERROR "foo144 &x does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001428 var y int
Dmitry Vyukov76477412015-02-19 22:00:11 +03001429 foo144b(&y) // ERROR "foo144 &y does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001430}
1431
1432//go:noescape
1433
1434func foo144b(*int)
1435
1436// issue 7313: for loop init should not be treated as "in loop"
1437
1438type List struct {
1439 Next *List
1440}
1441
Dmitry Vyukov76477412015-02-19 22:00:11 +03001442func foo145(l List) { // ERROR "foo145 l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001443 var p *List
Dmitry Vyukov76477412015-02-19 22:00:11 +03001444 for p = &l; p.Next != nil; p = p.Next { // ERROR "foo145 &l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001445 }
1446}
1447
Dmitry Vyukov76477412015-02-19 22:00:11 +03001448func foo146(l List) { // ERROR "foo146 l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001449 var p *List
Dmitry Vyukov76477412015-02-19 22:00:11 +03001450 p = &l // ERROR "foo146 &l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001451 for ; p.Next != nil; p = p.Next {
1452 }
1453}
1454
Dmitry Vyukov76477412015-02-19 22:00:11 +03001455func foo147(l List) { // ERROR "foo147 l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001456 var p *List
Dmitry Vyukov76477412015-02-19 22:00:11 +03001457 p = &l // ERROR "foo147 &l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001458 for p.Next != nil {
1459 p = p.Next
1460 }
1461}
1462
Dmitry Vyukov76477412015-02-19 22:00:11 +03001463func foo148(l List) { // ERROR "foo148 l does not escape$"
1464 for p := &l; p.Next != nil; p = p.Next { // ERROR "foo148 &l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001465 }
1466}
1467
1468// related: address of variable should have depth of variable, not of loop
1469
Dmitry Vyukov76477412015-02-19 22:00:11 +03001470func foo149(l List) { // ERROR "foo149 l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001471 var p *List
1472 for {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001473 for p = &l; p.Next != nil; p = p.Next { // ERROR "foo149 &l does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001474 }
1475 }
1476}
1477
1478// issue 7934: missed ... if element type had no pointers
1479
1480var save150 []byte
1481
Dmitry Vyukov76477412015-02-19 22:00:11 +03001482func foo150(x ...byte) { // ERROR "leaking param: x$"
Russ Cox00d2f912014-09-24 15:20:03 -04001483 save150 = x
1484}
1485
1486func bar150() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001487 foo150(1, 2, 3) // ERROR "... argument escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001488}
1489
1490// issue 7931: bad handling of slice of array
1491
1492var save151 *int
1493
Dmitry Vyukov76477412015-02-19 22:00:11 +03001494func foo151(x *int) { // ERROR "leaking param: x$"
Russ Cox00d2f912014-09-24 15:20:03 -04001495 save151 = x
1496}
1497
1498func bar151() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001499 var a [64]int // ERROR "moved to heap: a$"
Russ Cox00d2f912014-09-24 15:20:03 -04001500 a[4] = 101
Dmitry Vyukov76477412015-02-19 22:00:11 +03001501 foo151(&(&a)[4:8][0]) // ERROR "&\(&a\)\[4:8\]\[0\] escapes to heap$" "&a escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001502}
1503
1504func bar151b() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001505 var a [10]int // ERROR "moved to heap: a$"
1506 b := a[:] // ERROR "a escapes to heap$"
1507 foo151(&b[4:8][0]) // ERROR "&b\[4:8\]\[0\] escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001508}
1509
1510func bar151c() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001511 var a [64]int // ERROR "moved to heap: a$"
Russ Cox00d2f912014-09-24 15:20:03 -04001512 a[4] = 101
Dmitry Vyukov76477412015-02-19 22:00:11 +03001513 foo151(&(&a)[4:8:8][0]) // ERROR "&\(&a\)\[4:8:8\]\[0\] escapes to heap$" "&a escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001514}
1515
1516func bar151d() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001517 var a [10]int // ERROR "moved to heap: a$"
1518 b := a[:] // ERROR "a escapes to heap$"
1519 foo151(&b[4:8:8][0]) // ERROR "&b\[4:8:8\]\[0\] escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001520}
1521
1522// issue 8120
1523
1524type U struct {
1525 s *string
1526}
1527
Dmitry Vyukov76477412015-02-19 22:00:11 +03001528func (u *U) String() *string { // ERROR "\(\*U\).String leaking param u content to result ~r0$"
Russ Cox00d2f912014-09-24 15:20:03 -04001529 return u.s
1530}
1531
1532type V struct {
1533 s *string
1534}
1535
Dmitry Vyukov76477412015-02-19 22:00:11 +03001536func NewV(u U) *V { // ERROR "leaking param: u$"
1537 return &V{u.String()} // ERROR "&V literal escapes to heap$" "NewV u does not escape$"
Russ Cox00d2f912014-09-24 15:20:03 -04001538}
1539
1540func foo152() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001541 a := "a" // ERROR "moved to heap: a$"
1542 u := U{&a} // ERROR "&a escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001543 v := NewV(u)
1544 println(v)
1545}
1546
1547// issue 8176 - &x in type switch body not marked as escaping
1548
Dmitry Vyukov76477412015-02-19 22:00:11 +03001549func foo153(v interface{}) *int { // ERROR "leaking param: v$"
Russ Cox00d2f912014-09-24 15:20:03 -04001550 switch x := v.(type) {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001551 case int: // ERROR "moved to heap: x$"
1552 return &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001553 }
1554 panic(0)
1555}
1556
1557// issue 8185 - &result escaping into result
1558
Dmitry Vyukov76477412015-02-19 22:00:11 +03001559func f() (x int, y *int) { // ERROR "moved to heap: x$"
1560 y = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001561 return
1562}
1563
Dmitry Vyukov76477412015-02-19 22:00:11 +03001564func g() (x interface{}) { // ERROR "moved to heap: x$"
1565 x = &x // ERROR "&x escapes to heap$"
Russ Cox00d2f912014-09-24 15:20:03 -04001566 return
1567}
Dmitry Vyukov1b87f012015-01-19 23:46:22 +03001568
1569var sink interface{}
1570
1571type Lit struct {
1572 p *int
1573}
1574
1575func ptrlitNoescape() {
1576 // Both literal and element do not escape.
1577 i := 0
Dmitry Vyukov76477412015-02-19 22:00:11 +03001578 x := &Lit{&i} // ERROR "ptrlitNoescape &Lit literal does not escape$" "ptrlitNoescape &i does not escape$"
Dmitry Vyukov1b87f012015-01-19 23:46:22 +03001579 _ = x
1580}
1581
1582func ptrlitNoEscape2() {
1583 // Literal does not escape, but element does.
Dmitry Vyukov76477412015-02-19 22:00:11 +03001584 i := 0 // ERROR "moved to heap: i$"
1585 x := &Lit{&i} // ERROR "&i escapes to heap$" "ptrlitNoEscape2 &Lit literal does not escape$"
1586 sink = *x // ERROR "\*x escapes to heap$"
Dmitry Vyukov1b87f012015-01-19 23:46:22 +03001587}
1588
1589func ptrlitEscape() {
1590 // Both literal and element escape.
Dmitry Vyukov76477412015-02-19 22:00:11 +03001591 i := 0 // ERROR "moved to heap: i$"
1592 x := &Lit{&i} // ERROR "&Lit literal escapes to heap$" "&i escapes to heap$"
1593 sink = x // ERROR "x escapes to heap$"
Dmitry Vyukov1b87f012015-01-19 23:46:22 +03001594}
Dmitry Vyukov22c16b42015-01-16 23:30:35 +03001595
1596// self-assignments
1597
1598type Buffer struct {
1599 arr [64]byte
1600 buf1 []byte
1601 buf2 []byte
1602 str1 string
1603 str2 string
1604}
1605
Dmitry Vyukov76477412015-02-19 22:00:11 +03001606func (b *Buffer) foo() { // ERROR "\(\*Buffer\).foo b does not escape$"
1607 b.buf1 = b.buf1[1:2] // ERROR "\(\*Buffer\).foo ignoring self-assignment to b.buf1$"
1608 b.buf1 = b.buf1[1:2:3] // ERROR "\(\*Buffer\).foo ignoring self-assignment to b.buf1$"
1609 b.buf1 = b.buf2[1:2] // ERROR "\(\*Buffer\).foo ignoring self-assignment to b.buf1$"
1610 b.buf1 = b.buf2[1:2:3] // ERROR "\(\*Buffer\).foo ignoring self-assignment to b.buf1$"
Dmitry Vyukov22c16b42015-01-16 23:30:35 +03001611}
1612
Dmitry Vyukov76477412015-02-19 22:00:11 +03001613func (b *Buffer) bar() { // ERROR "leaking param: b$"
1614 b.buf1 = b.arr[1:2] // ERROR "b.arr escapes to heap$"
Dmitry Vyukov22c16b42015-01-16 23:30:35 +03001615}
1616
Dmitry Vyukov76477412015-02-19 22:00:11 +03001617func (b *Buffer) baz() { // ERROR "\(\*Buffer\).baz b does not escape$"
1618 b.str1 = b.str1[1:2] // ERROR "\(\*Buffer\).baz ignoring self-assignment to b.str1$"
1619 b.str1 = b.str2[1:2] // ERROR "\(\*Buffer\).baz ignoring self-assignment to b.str1$"
Dmitry Vyukov22c16b42015-01-16 23:30:35 +03001620}
1621
Dmitry Vyukov76477412015-02-19 22:00:11 +03001622func (b *Buffer) bat() { // ERROR "leaking param: b$"
1623 o := new(Buffer) // ERROR "new\(Buffer\) escapes to heap$"
Dmitry Vyukov22c16b42015-01-16 23:30:35 +03001624 o.buf1 = b.buf1[1:2]
Dmitry Vyukov76477412015-02-19 22:00:11 +03001625 sink = o // ERROR "o escapes to heap$"
Dmitry Vyukov22c16b42015-01-16 23:30:35 +03001626}
1627
Dmitry Vyukov76477412015-02-19 22:00:11 +03001628func quux(sp *string, bp *[]byte) { // ERROR "quux bp does not escape$" "quux sp does not escape$"
1629 *sp = (*sp)[1:2] // ERROR "quux ignoring self-assignment to \*sp$"
1630 *bp = (*bp)[1:2] // ERROR "quux ignoring self-assignment to \*bp$"
Dmitry Vyukov22c16b42015-01-16 23:30:35 +03001631}
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001632
1633type StructWithString struct {
1634 p *int
1635 s string
1636}
1637
1638// This is escape analysis false negative.
1639// We assign the pointer to x.p but leak x.s. Escape analysis coarsens flows
1640// to just x, and thus &i looks escaping.
1641func fieldFlowTracking() {
1642 var x StructWithString
Dmitry Vyukov76477412015-02-19 22:00:11 +03001643 i := 0 // ERROR "moved to heap: i$"
1644 x.p = &i // ERROR "&i escapes to heap$"
1645 sink = x.s // ERROR "x.s escapes to heap$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001646}
1647
1648// String operations.
1649
1650func slicebytetostring0() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001651 b := make([]byte, 20) // ERROR "slicebytetostring0 make\(\[\]byte, 20\) does not escape$"
1652 s := string(b) // ERROR "slicebytetostring0 string\(b\) does not escape$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001653 _ = s
1654}
1655
1656func slicebytetostring1() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001657 b := make([]byte, 20) // ERROR "slicebytetostring1 make\(\[\]byte, 20\) does not escape$"
1658 s := string(b) // ERROR "slicebytetostring1 string\(b\) does not escape$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001659 s1 := s[0:1]
1660 _ = s1
1661}
1662
1663func slicebytetostring2() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001664 b := make([]byte, 20) // ERROR "slicebytetostring2 make\(\[\]byte, 20\) does not escape$"
1665 s := string(b) // ERROR "string\(b\) escapes to heap$"
1666 s1 := s[0:1] // ERROR "moved to heap: s1$"
1667 sink = &s1 // ERROR "&s1 escapes to heap$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001668}
1669
1670func slicebytetostring3() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001671 b := make([]byte, 20) // ERROR "slicebytetostring3 make\(\[\]byte, 20\) does not escape$"
1672 s := string(b) // ERROR "string\(b\) escapes to heap$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001673 s1 := s[0:1]
Dmitry Vyukov76477412015-02-19 22:00:11 +03001674 sink = s1 // ERROR "s1 escapes to heap$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001675}
1676
1677func addstr0() {
1678 s0 := "a"
1679 s1 := "b"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001680 s := s0 + s1 // ERROR "addstr0 s0 \+ s1 does not escape$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001681 _ = s
1682}
1683
1684func addstr1() {
1685 s0 := "a"
1686 s1 := "b"
1687 s := "c"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001688 s += s0 + s1 // ERROR "addstr1 s0 \+ s1 does not escape$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001689 _ = s
1690}
1691
1692func addstr2() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001693 b := make([]byte, 20) // ERROR "addstr2 make\(\[\]byte, 20\) does not escape$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001694 s0 := "a"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001695 s := string(b) + s0 // ERROR "addstr2 string\(b\) \+ s0 does not escape$" "addstr2 string\(b\) does not escape$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001696 _ = s
1697}
1698
1699func addstr3() {
1700 s0 := "a"
1701 s1 := "b"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001702 s := s0 + s1 // ERROR "s0 \+ s1 escapes to heap$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001703 s2 := s[0:1]
Dmitry Vyukov76477412015-02-19 22:00:11 +03001704 sink = s2 // ERROR "s2 escapes to heap$"
Dmitry Vyukove6fac082015-01-21 17:37:59 +03001705}
Dmitry Vyukov4ce4d8b2015-01-28 08:42:20 +03001706
1707func intstring0() bool {
1708 // string does not escape
1709 x := '0'
Dmitry Vyukov76477412015-02-19 22:00:11 +03001710 s := string(x) // ERROR "intstring0 string\(x\) does not escape$"
Dmitry Vyukov4ce4d8b2015-01-28 08:42:20 +03001711 return s == "0"
1712}
1713
1714func intstring1() string {
1715 // string does not escape, but the buffer does
1716 x := '0'
Dmitry Vyukov76477412015-02-19 22:00:11 +03001717 s := string(x) // ERROR "string\(x\) escapes to heap$"
Dmitry Vyukov4ce4d8b2015-01-28 08:42:20 +03001718 return s
1719}
1720
1721func intstring2() {
1722 // string escapes to heap
1723 x := '0'
Dmitry Vyukov76477412015-02-19 22:00:11 +03001724 s := string(x) // ERROR "moved to heap: s$" "string\(x\) escapes to heap$"
1725 sink = &s // ERROR "&s escapes to heap$"
Dmitry Vyukov4ce4d8b2015-01-28 08:42:20 +03001726}
Dmitry Vyukov95681262015-01-30 09:14:13 +03001727
1728func stringtoslicebyte0() {
1729 s := "foo"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001730 x := []byte(s) // ERROR "stringtoslicebyte0 \(\[\]byte\)\(s\) does not escape$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001731 _ = x
1732}
1733
1734func stringtoslicebyte1() []byte {
1735 s := "foo"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001736 return []byte(s) // ERROR "\(\[\]byte\)\(s\) escapes to heap$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001737}
1738
1739func stringtoslicebyte2() {
1740 s := "foo"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001741 sink = []byte(s) // ERROR "\(\[\]byte\)\(s\) escapes to heap$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001742}
1743
1744func stringtoslicerune0() {
1745 s := "foo"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001746 x := []rune(s) // ERROR "stringtoslicerune0 \(\[\]rune\)\(s\) does not escape$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001747 _ = x
1748}
1749
1750func stringtoslicerune1() []rune {
1751 s := "foo"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001752 return []rune(s) // ERROR "\(\[\]rune\)\(s\) escapes to heap$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001753}
1754
1755func stringtoslicerune2() {
1756 s := "foo"
Dmitry Vyukov76477412015-02-19 22:00:11 +03001757 sink = []rune(s) // ERROR "\(\[\]rune\)\(s\) escapes to heap$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001758}
1759
1760func slicerunetostring0() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001761 r := []rune{1, 2, 3} // ERROR "slicerunetostring0 \[\]rune literal does not escape$"
1762 s := string(r) // ERROR "slicerunetostring0 string\(r\) does not escape$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001763 _ = s
1764}
1765
1766func slicerunetostring1() string {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001767 r := []rune{1, 2, 3} // ERROR "slicerunetostring1 \[\]rune literal does not escape$"
1768 return string(r) // ERROR "string\(r\) escapes to heap$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001769}
1770
1771func slicerunetostring2() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001772 r := []rune{1, 2, 3} // ERROR "slicerunetostring2 \[\]rune literal does not escape$"
1773 sink = string(r) // ERROR "string\(r\) escapes to heap$"
Dmitry Vyukov95681262015-01-30 09:14:13 +03001774}
Dmitry Vyukovb3be3602015-01-29 19:40:02 +03001775
1776func makemap0() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001777 m := make(map[int]int) // ERROR "makemap0 make\(map\[int\]int\) does not escape$"
Dmitry Vyukovb3be3602015-01-29 19:40:02 +03001778 m[0] = 0
1779 m[1]++
1780 delete(m, 1)
Dmitry Vyukov76477412015-02-19 22:00:11 +03001781 sink = m[0] // ERROR "m\[0\] escapes to heap$"
Dmitry Vyukovb3be3602015-01-29 19:40:02 +03001782}
1783
1784func makemap1() map[int]int {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001785 return make(map[int]int) // ERROR "make\(map\[int\]int\) escapes to heap$"
Dmitry Vyukovb3be3602015-01-29 19:40:02 +03001786}
1787
1788func makemap2() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001789 m := make(map[int]int) // ERROR "make\(map\[int\]int\) escapes to heap$"
1790 sink = m // ERROR "m escapes to heap$"
Dmitry Vyukovedcc0622015-02-19 16:27:32 +03001791}
1792
Dmitry Vyukov76477412015-02-19 22:00:11 +03001793func nonescapingEface(m map[interface{}]bool) bool { // ERROR "nonescapingEface m does not escape$"
1794 return m["foo"] // ERROR "nonescapingEface .foo. does not escape$"
Dmitry Vyukovedcc0622015-02-19 16:27:32 +03001795}
1796
Dmitry Vyukov76477412015-02-19 22:00:11 +03001797func nonescapingIface(m map[M]bool) bool { // ERROR "nonescapingIface m does not escape$"
1798 return m[MV(0)] // ERROR "nonescapingIface MV\(0\) does not escape$"
Dmitry Vyukovb3be3602015-01-29 19:40:02 +03001799}
Dmitry Vyukov878a86a2015-04-06 18:17:20 +03001800
1801func issue10353() {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001802 x := new(int) // ERROR "new\(int\) escapes to heap$"
Dmitry Vyukov878a86a2015-04-06 18:17:20 +03001803 issue10353a(x)()
1804}
1805
Dmitry Vyukov76477412015-02-19 22:00:11 +03001806func issue10353a(x *int) func() { // ERROR "leaking param: x$"
1807 return func() { // ERROR "func literal escapes to heap$"
Dmitry Vyukov878a86a2015-04-06 18:17:20 +03001808 println(*x)
1809 }
1810}
1811
1812func issue10353b() {
1813 var f func()
1814 for {
Dmitry Vyukov76477412015-02-19 22:00:11 +03001815 x := new(int) // ERROR "new\(int\) escapes to heap$"
1816 f = func() { // ERROR "func literal escapes to heap$"
Dmitry Vyukov878a86a2015-04-06 18:17:20 +03001817 println(*x)
1818 }
1819 }
1820 _ = f
1821}