|  | // 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 for interface conversions. | 
|  |  | 
|  | package escape | 
|  |  | 
|  | var sink interface{} | 
|  |  | 
|  | type M interface { | 
|  | M() | 
|  | } | 
|  |  | 
|  | func mescapes(m M) { // ERROR "leaking param: m" | 
|  | sink = m // ERROR "m escapes to heap" | 
|  | } | 
|  |  | 
|  | func mdoesnotescape(m M) { // ERROR "m does not escape" | 
|  | } | 
|  |  | 
|  | // Tests for type stored directly in iface and with value receiver method. | 
|  | type M0 struct { | 
|  | p *int | 
|  | } | 
|  |  | 
|  | func (M0) M() { | 
|  | } | 
|  |  | 
|  | func efaceEscape0() { | 
|  | { | 
|  | i := 0 | 
|  | v := M0{&i} // ERROR "&i does not escape" | 
|  | var x M = v // ERROR "v does not escape" | 
|  | _ = x | 
|  | } | 
|  | { | 
|  | i := 0      // ERROR "moved to heap: i" | 
|  | v := M0{&i} // ERROR "&i escapes to heap" | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | sink = x    // ERROR "x escapes to heap" | 
|  | } | 
|  | { | 
|  | i := 0 | 
|  | v := M0{&i} // ERROR "&i does not escape" | 
|  | var x M = v // ERROR "v does not escape" | 
|  | v1 := x.(M0) | 
|  | _ = v1 | 
|  | } | 
|  | { | 
|  | i := 0      // ERROR "moved to heap: i" | 
|  | v := M0{&i} // ERROR "&i escapes to heap" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | v1 := x.(M0) | 
|  | sink = v1 // ERROR "v1 escapes to heap" | 
|  | } | 
|  | { | 
|  | i := 0      // ERROR "moved to heap: i" | 
|  | v := M0{&i} // ERROR "&i escapes to heap" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | x.M() | 
|  | } | 
|  | { | 
|  | i := 0      // ERROR "moved to heap: i" | 
|  | v := M0{&i} // ERROR "&i escapes to heap" | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | mescapes(x) | 
|  | } | 
|  | { | 
|  | i := 0 | 
|  | v := M0{&i} // ERROR "&i does not escape" | 
|  | var x M = v // ERROR "v does not escape" | 
|  | mdoesnotescape(x) | 
|  | } | 
|  | } | 
|  |  | 
|  | // Tests for type stored indirectly in iface and with value receiver method. | 
|  | type M1 struct { | 
|  | p *int | 
|  | x int | 
|  | } | 
|  |  | 
|  | func (M1) M() { | 
|  | } | 
|  |  | 
|  | func efaceEscape1() { | 
|  | { | 
|  | i := 0 | 
|  | v := M1{&i, 0} // ERROR "&i does not escape" | 
|  | var x M = v    // ERROR "v does not escape" | 
|  | _ = x | 
|  | } | 
|  | { | 
|  | i := 0         // ERROR "moved to heap: i" | 
|  | v := M1{&i, 0} // ERROR "&i escapes to heap" | 
|  | var x M = v    // ERROR "v escapes to heap" | 
|  | sink = x       // ERROR "x escapes to heap" | 
|  | } | 
|  | { | 
|  | i := 0 | 
|  | v := M1{&i, 0} // ERROR "&i does not escape" | 
|  | var x M = v    // ERROR "v does not escape" | 
|  | v1 := x.(M1) | 
|  | _ = v1 | 
|  | } | 
|  | { | 
|  | i := 0         // ERROR "moved to heap: i" | 
|  | v := M1{&i, 0} // ERROR "&i escapes to heap" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | v1 := x.(M1) | 
|  | sink = v1 // ERROR "v1 escapes to heap" | 
|  | } | 
|  | { | 
|  | i := 0         // ERROR "moved to heap: i" | 
|  | v := M1{&i, 0} // ERROR "&i escapes to heap" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | x.M() | 
|  | } | 
|  | { | 
|  | i := 0         // ERROR "moved to heap: i" | 
|  | v := M1{&i, 0} // ERROR "&i escapes to heap" | 
|  | var x M = v    // ERROR "v escapes to heap" | 
|  | mescapes(x) | 
|  | } | 
|  | { | 
|  | i := 0 | 
|  | v := M1{&i, 0} // ERROR "&i does not escape" | 
|  | var x M = v    // ERROR "v does not escape" | 
|  | mdoesnotescape(x) | 
|  | } | 
|  | } | 
|  |  | 
|  | // Tests for type stored directly in iface and with pointer receiver method. | 
|  | type M2 struct { | 
|  | p *int | 
|  | } | 
|  |  | 
|  | func (*M2) M() { | 
|  | } | 
|  |  | 
|  | func efaceEscape2() { | 
|  | { | 
|  | i := 0 | 
|  | v := &M2{&i} // ERROR "&i does not escape" "&M2 literal does not escape" | 
|  | var x M = v  // ERROR "v does not escape" | 
|  | _ = x | 
|  | } | 
|  | { | 
|  | i := 0       // ERROR "moved to heap: i" | 
|  | v := &M2{&i} // ERROR "&i escapes to heap" "&M2 literal escapes to heap" | 
|  | var x M = v  // ERROR "v escapes to heap" | 
|  | sink = x     // ERROR "x escapes to heap" | 
|  | } | 
|  | { | 
|  | i := 0 | 
|  | v := &M2{&i} // ERROR "&i does not escape" "&M2 literal does not escape" | 
|  | var x M = v  // ERROR "v does not escape" | 
|  | v1 := x.(*M2) | 
|  | _ = v1 | 
|  | } | 
|  | { | 
|  | i := 0       // ERROR "moved to heap: i" | 
|  | v := &M2{&i} // ERROR "&i escapes to heap" "&M2 literal escapes to heap" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | v1 := x.(*M2) | 
|  | sink = v1 // ERROR "v1 escapes to heap" | 
|  | } | 
|  | { | 
|  | i := 0       // ERROR "moved to heap: i" | 
|  | v := &M2{&i} // ERROR "&i escapes to heap" "&M2 literal does not escape" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v does not escape" | 
|  | v1 := x.(*M2) | 
|  | sink = *v1 // ERROR "v1 escapes to heap" | 
|  | } | 
|  | { | 
|  | i := 0       // ERROR "moved to heap: i" | 
|  | v := &M2{&i} // ERROR "&i escapes to heap" "&M2 literal does not escape" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v does not escape" | 
|  | v1, ok := x.(*M2) | 
|  | sink = *v1 // ERROR "v1 escapes to heap" | 
|  | _ = ok | 
|  | } | 
|  | { | 
|  | i := 0       // ERROR "moved to heap: i" | 
|  | v := &M2{&i} // ERROR "&i escapes to heap" "&M2 literal escapes to heap" | 
|  | // BAD: v does not escape to heap here | 
|  | var x M = v // ERROR "v escapes to heap" | 
|  | x.M() | 
|  | } | 
|  | { | 
|  | i := 0       // ERROR "moved to heap: i" | 
|  | v := &M2{&i} // ERROR "&i escapes to heap" "&M2 literal escapes to heap" | 
|  | var x M = v  // ERROR "v escapes to heap" | 
|  | mescapes(x) | 
|  | } | 
|  | { | 
|  | i := 0 | 
|  | v := &M2{&i} // ERROR "&i does not escape" "&M2 literal does not escape" | 
|  | var x M = v  // ERROR "v does not escape" | 
|  | mdoesnotescape(x) | 
|  | } | 
|  | } | 
|  |  | 
|  | type T1 struct { | 
|  | p *int | 
|  | } | 
|  |  | 
|  | type T2 struct { | 
|  | T1 T1 | 
|  | } | 
|  |  | 
|  | func dotTypeEscape() *T2 { // #11931 | 
|  | var x interface{} | 
|  | x = &T1{p: new(int)} // ERROR "new\(int\) escapes to heap" "&T1 literal does not escape" | 
|  | return &T2{ | 
|  | T1: *(x.(*T1)), // ERROR "&T2 literal escapes to heap" | 
|  | } | 
|  | } | 
|  |  | 
|  | func dotTypeEscape2() { // #13805, #15796 | 
|  | { | 
|  | i := 0 | 
|  | j := 0 | 
|  | var v int | 
|  | var ok bool | 
|  | var x interface{} = i // ERROR "i does not escape" | 
|  | var y interface{} = j // ERROR "j does not escape" | 
|  |  | 
|  | *(&v) = x.(int) // ERROR "&v does not escape" | 
|  | *(&v), *(&ok) = y.(int) // ERROR "&v does not escape" "&ok does not escape" | 
|  | } | 
|  | { | 
|  | i := 0 | 
|  | j := 0 | 
|  | var ok bool | 
|  | var x interface{} = i // ERROR "i does not escape" | 
|  | var y interface{} = j // ERROR "j does not escape" | 
|  |  | 
|  | sink = x.(int)        // ERROR "x.\(int\) escapes to heap" | 
|  | sink, *(&ok) = y.(int)     // ERROR "&ok does not escape" | 
|  | } | 
|  | { | 
|  | i := 0 // ERROR "moved to heap: i" | 
|  | j := 0 // ERROR "moved to heap: j" | 
|  | var ok bool | 
|  | var x interface{} = &i // ERROR "&i escapes to heap" | 
|  | var y interface{} = &j // ERROR "&j escapes to heap" | 
|  |  | 
|  | sink = x.(*int)        // ERROR "x.\(\*int\) escapes to heap" | 
|  | sink, *(&ok) = y.(*int)     // ERROR "&ok does not escape" | 
|  | } | 
|  | } |