// errorcheck -+ // Copyright 2016 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 type-checking errors for go:notinheap. package p //go:notinheap type nih struct{} // Types embedding notinheap types must be notinheap. type embed1 struct { // ERROR "must be go:notinheap" x nih } type embed2 [1]nih // ERROR "must be go:notinheap" type embed3 struct { // ERROR "must be go:notinheap" x [1]nih } type embed4 map[nih]int // ERROR "incomplete \(or unallocatable\) map key not allowed" type embed5 map[int]nih // ERROR "incomplete \(or unallocatable\) map value not allowed" type emebd6 chan nih // ERROR "chan of incomplete \(or unallocatable\) type not allowed" type okay1 *nih type okay2 []nih type okay3 func(x nih) nih type okay4 interface { f(x nih) nih } // Type conversions don't let you sneak past notinheap. type t1 struct{ x int } //go:notinheap type t2 t1 //go:notinheap type t3 byte //go:notinheap type t4 rune // Type aliases inherit the go:notinheap-ness of the type they alias. type nihAlias = nih type embedAlias1 struct { // ERROR "must be go:notinheap" x nihAlias } type embedAlias2 [1]nihAlias // ERROR "must be go:notinheap" var sink interface{} func i() { sink = new(t1) // no error sink = (*t2)(new(t1)) // ERROR "cannot convert(.|\n)*t2 is incomplete \(or unallocatable\)" sink = (*t2)(new(struct{ x int })) // ERROR "cannot convert(.|\n)*t2 is incomplete \(or unallocatable\)" sink = []t3("foo") // ERROR "cannot convert(.|\n)*t3 is incomplete \(or unallocatable\)" sink = []t4("bar") // ERROR "cannot convert(.|\n)*t4 is incomplete \(or unallocatable\)" }