| // Copyright 2021 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. |
| |
| package types_test |
| |
| import ( |
| "bytes" |
| "errors" |
| "fmt" |
| "go/ast" |
| "go/importer" |
| "go/parser" |
| "go/token" |
| "go/types" |
| "runtime" |
| "strings" |
| "testing" |
| ) |
| |
| func checkMono(t *testing.T, body string) error { |
| if runtime.Compiler == "gccgo" { |
| t.Skip("skipping for gofronted: fails to import unsafe") |
| } |
| |
| fset := token.NewFileSet() |
| file, err := parser.ParseFile(fset, "x.go", "package x; import `unsafe`; var _ unsafe.Pointer;\n"+body, 0) |
| if err != nil { |
| t.Fatal(err) |
| } |
| files := []*ast.File{file} |
| |
| var buf bytes.Buffer |
| conf := types.Config{ |
| Error: func(err error) { fmt.Fprintln(&buf, err) }, |
| Importer: importer.Default(), |
| } |
| conf.Check("x", fset, files, nil) |
| if buf.Len() == 0 { |
| return nil |
| } |
| return errors.New(strings.TrimRight(buf.String(), "\n")) |
| } |
| |
| func TestMonoGood(t *testing.T) { |
| for i, good := range goods { |
| if err := checkMono(t, good); err != nil { |
| t.Errorf("%d: unexpected failure: %v", i, err) |
| } |
| } |
| } |
| |
| func TestMonoBad(t *testing.T) { |
| for i, bad := range bads { |
| if err := checkMono(t, bad); err == nil { |
| t.Errorf("%d: unexpected success", i) |
| } else { |
| t.Log(err) |
| } |
| } |
| } |
| |
| var goods = []string{ |
| "func F[T any](x T) { F(x) }", |
| "func F[T, U, V any]() { F[U, V, T](); F[V, T, U]() }", |
| "type Ring[A, B, C any] struct { L *Ring[B, C, A]; R *Ring[C, A, B] }", |
| "func F[T any]() { type U[T any] [unsafe.Sizeof(F[*T])]byte }", |
| "func F[T any]() { type U[T any] [unsafe.Sizeof(F[*T])]byte; var _ U[int] }", |
| "type U[T any] [unsafe.Sizeof(F[*T])]byte; func F[T any]() { var _ U[U[int]] }", |
| "func F[T any]() { type A = int; F[A]() }", |
| } |
| |
| // TODO(mdempsky): Validate specific error messages and positioning. |
| |
| var bads = []string{ |
| "func F[T any](x T) { F(&x) }", |
| "func F[T any]() { F[*T]() }", |
| "func F[T any]() { F[[]T]() }", |
| "func F[T any]() { F[[1]T]() }", |
| "func F[T any]() { F[chan T]() }", |
| "func F[T any]() { F[map[*T]int]() }", |
| "func F[T any]() { F[map[error]T]() }", |
| "func F[T any]() { F[func(T)]() }", |
| "func F[T any]() { F[func() T]() }", |
| "func F[T any]() { F[struct{ t T }]() }", |
| "func F[T any]() { F[interface{ t() T }]() }", |
| "type U[_ any] int; func F[T any]() { F[U[T]]() }", |
| "func F[T any]() { type U int; F[U]() }", |
| "func F[T any]() { type U int; F[*U]() }", |
| "type U[T any] int; func (U[T]) m() { var _ U[*T] }", |
| "type U[T any] int; func (*U[T]) m() { var _ U[*T] }", |
| "type U[T1 any] [unsafe.Sizeof(F[*T1])]byte; func F[T2 any]() { var _ U[T2] }", |
| "func F[A, B, C, D, E any]() { F[B, C, D, E, *A]() }", |
| "type U[_ any] int; const X = unsafe.Sizeof(func() { type A[T any] U[A[*T]] })", |
| "func F[T any]() { type A = *T; F[A]() }", |
| "type A[T any] struct { _ A[*T] }", |
| } |