|  | // errorcheck -0 -m -l | 
|  |  | 
|  | // Copyright 2015 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | // Test escape analysis with respect to field assignments. | 
|  |  | 
|  | package escape | 
|  |  | 
|  | var sink interface{} | 
|  |  | 
|  | type X struct { | 
|  | p1 *int | 
|  | p2 *int | 
|  | a  [2]*int | 
|  | } | 
|  |  | 
|  | type Y struct { | 
|  | x X | 
|  | } | 
|  |  | 
|  | func field0() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | x.p1 = &i | 
|  | sink = x.p1 | 
|  | } | 
|  |  | 
|  | func field1() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | // BAD: &i should not escape | 
|  | x.p1 = &i | 
|  | sink = x.p2 | 
|  | } | 
|  |  | 
|  | func field3() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | x.p1 = &i | 
|  | sink = x // ERROR "x escapes to heap" | 
|  | } | 
|  |  | 
|  | func field4() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var y Y | 
|  | y.x.p1 = &i | 
|  | x := y.x | 
|  | sink = x // ERROR "x escapes to heap" | 
|  | } | 
|  |  | 
|  | func field5() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | // BAD: &i should not escape here | 
|  | x.a[0] = &i | 
|  | sink = x.a[1] | 
|  | } | 
|  |  | 
|  | // BAD: we are not leaking param x, only x.p2 | 
|  | func field6(x *X) { // ERROR "leaking param content: x$" | 
|  | sink = x.p2 | 
|  | } | 
|  |  | 
|  | func field6a() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | // BAD: &i should not escape | 
|  | x.p1 = &i | 
|  | field6(&x) | 
|  | } | 
|  |  | 
|  | func field7() { | 
|  | i := 0 | 
|  | var y Y | 
|  | y.x.p1 = &i | 
|  | x := y.x | 
|  | var y1 Y | 
|  | y1.x = x | 
|  | _ = y1.x.p1 | 
|  | } | 
|  |  | 
|  | func field8() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var y Y | 
|  | y.x.p1 = &i | 
|  | x := y.x | 
|  | var y1 Y | 
|  | y1.x = x | 
|  | sink = y1.x.p1 | 
|  | } | 
|  |  | 
|  | func field9() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var y Y | 
|  | y.x.p1 = &i | 
|  | x := y.x | 
|  | var y1 Y | 
|  | y1.x = x | 
|  | sink = y1.x // ERROR "y1\.x escapes to heap" | 
|  | } | 
|  |  | 
|  | func field10() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var y Y | 
|  | // BAD: &i should not escape | 
|  | y.x.p1 = &i | 
|  | x := y.x | 
|  | var y1 Y | 
|  | y1.x = x | 
|  | sink = y1.x.p2 | 
|  | } | 
|  |  | 
|  | func field11() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | x := X{p1: &i} | 
|  | sink = x.p1 | 
|  | } | 
|  |  | 
|  | func field12() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | // BAD: &i should not escape | 
|  | x := X{p1: &i} | 
|  | sink = x.p2 | 
|  | } | 
|  |  | 
|  | func field13() { | 
|  | i := 0          // ERROR "moved to heap: i$" | 
|  | x := &X{p1: &i} // ERROR "&X{...} does not escape$" | 
|  | sink = x.p1 | 
|  | } | 
|  |  | 
|  | func field14() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | // BAD: &i should not escape | 
|  | x := &X{p1: &i} // ERROR "&X{...} does not escape$" | 
|  | sink = x.p2 | 
|  | } | 
|  |  | 
|  | func field15() { | 
|  | i := 0          // ERROR "moved to heap: i$" | 
|  | x := &X{p1: &i} // ERROR "&X{...} escapes to heap$" | 
|  | sink = x | 
|  | } | 
|  |  | 
|  | func field16() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | // BAD: &i should not escape | 
|  | x.p1 = &i | 
|  | var iface interface{} = x // ERROR "x does not escape" | 
|  | x1 := iface.(X) | 
|  | sink = x1.p2 | 
|  | } | 
|  |  | 
|  | func field17() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | x.p1 = &i | 
|  | var iface interface{} = x // ERROR "x does not escape" | 
|  | x1 := iface.(X) | 
|  | sink = x1.p1 | 
|  | } | 
|  |  | 
|  | func field18() { | 
|  | i := 0 // ERROR "moved to heap: i$" | 
|  | var x X | 
|  | // BAD: &i should not escape | 
|  | x.p1 = &i | 
|  | var iface interface{} = x // ERROR "x does not escape" | 
|  | y, _ := iface.(Y)         // Put X, but extracted Y. The cast will fail, so y is zero initialized. | 
|  | sink = y                  // ERROR "y escapes to heap" | 
|  | } |