mirror of
https://github.com/golang/go
synced 2024-11-26 11:48:03 -07:00
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 <danscales@google.com> Reviewed-by: Robert Griesemer <gri@golang.org> Trust: Dan Scales <danscales@google.com> Trust: Keith Randall <khr@golang.org> Trust: Robert Griesemer <gri@golang.org> Run-TryBot: Robert Griesemer <gri@golang.org> TryBot-Result: Go Bot <gobot@golang.org>
This commit is contained in:
parent
717f337d1d
commit
5670ff4ae5
@ -740,9 +740,16 @@ func implements(t, iface *types.Type, m, samename **types.Field, ptr *int) bool
|
|||||||
|
|
||||||
if t.IsInterface() || t.IsTypeParam() {
|
if t.IsInterface() || t.IsTypeParam() {
|
||||||
if t.IsTypeParam() {
|
if t.IsTypeParam() {
|
||||||
// A typeparam satisfies an interface if its type bound
|
// If t is a simple type parameter T, its type and underlying is the same.
|
||||||
// has all the methods of that interface.
|
// If t is a type definition:'type P[T any] T', its type is P[T] and its
|
||||||
t = t.Bound()
|
// 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
|
i := 0
|
||||||
tms := t.AllMethods().Slice()
|
tms := t.AllMethods().Slice()
|
||||||
|
40
test/typeparam/issue47708.go
Normal file
40
test/typeparam/issue47708.go
Normal file
@ -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))
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user