From 5670ff4ae593019c33b8095a6341deb657be34d7 Mon Sep 17 00:00:00 2001 From: korzhao Date: Mon, 16 Aug 2021 22:25:47 +0800 Subject: [PATCH] cmd/compile: fix conversions from TypeParam to interface If the TypeParam has all the methods of an interface, allow conversions from TypeParam to interface Fixes #47708 Change-Id: I40a82a31f6ea9354130dbe3bcfc83537094bf12c Reviewed-on: https://go-review.googlesource.com/c/go/+/342509 Reviewed-by: Dan Scales Reviewed-by: Robert Griesemer Trust: Dan Scales Trust: Keith Randall Trust: Robert Griesemer Run-TryBot: Robert Griesemer TryBot-Result: Go Bot --- src/cmd/compile/internal/typecheck/subr.go | 13 +++++-- test/typeparam/issue47708.go | 40 ++++++++++++++++++++++ 2 files changed, 50 insertions(+), 3 deletions(-) create mode 100644 test/typeparam/issue47708.go diff --git a/src/cmd/compile/internal/typecheck/subr.go b/src/cmd/compile/internal/typecheck/subr.go index 541e1907c03..4696b62cd2b 100644 --- a/src/cmd/compile/internal/typecheck/subr.go +++ b/src/cmd/compile/internal/typecheck/subr.go @@ -740,9 +740,16 @@ func implements(t, iface *types.Type, m, samename **types.Field, ptr *int) bool if t.IsInterface() || t.IsTypeParam() { if t.IsTypeParam() { - // A typeparam satisfies an interface if its type bound - // has all the methods of that interface. - t = t.Bound() + // If t is a simple type parameter T, its type and underlying is the same. + // If t is a type definition:'type P[T any] T', its type is P[T] and its + // underlying is T. Therefore we use 't.Underlying() != t' to distinguish them. + if t.Underlying() != t { + CalcMethods(t) + } else { + // A typeparam satisfies an interface if its type bound + // has all the methods of that interface. + t = t.Bound() + } } i := 0 tms := t.AllMethods().Slice() diff --git a/test/typeparam/issue47708.go b/test/typeparam/issue47708.go new file mode 100644 index 00000000000..261d6efb613 --- /dev/null +++ b/test/typeparam/issue47708.go @@ -0,0 +1,40 @@ +// run -gcflags=-G=3 + +// 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 main + +import ( + "fmt" +) + +type FooType[T any] interface { + Foo(BarType[T])string +} +type BarType[T any] interface { + Bar(FooType[T])string +} + +type Baz[T any] T +func (l Baz[T]) Foo(v BarType[T]) string { + return v.Bar(l) +} +type Bob[T any] T +func (l Bob[T]) Bar(v FooType[T]) string { + if v,ok := v.(Baz[T]);ok{ + return fmt.Sprintf("%v%v",v,l) + } + return "" +} + + +func main() { + var baz Baz[int] = 123 + var bob Bob[int] = 456 + + if got, want := baz.Foo(bob), "123456"; got != want { + panic(fmt.Sprintf("got %s want %s", got, want)) + } +}