Commit a80c5f05 authored by Robert Griesemer's avatar Robert Griesemer

go/types: allow embedding overlapping interfaces

Quietly drop duplicate methods from embedded interfaces
if they have an identical signature to existing methods.

Instead of adjusting the prior syntax-based only method set
computation where methods don't have signature information
(and thus where de-duplication according to the new rules
would have been somewhat tricky to get right), this change
completely rewrites interface method set computation, taking
a page from the cmd/compiler's implementation. In a first
pass, when type-checking interfaces, explicit methods and
embedded interfaces are collected, but the interfaces are
not "expanded", that is the final method set computation
is done lazily, either when needed for method lookup, or
at the end of type-checking.

While this is a substantial rewrite, it allows us to get
rid of the separate (duplicate and delicate) syntactical
method set computation and generally simplifies checking
of interface types significantly. A few (esoteric) test
cases now have slightly different error messages but all
tests that are accepted by cmd/compile are also accepted
by go/types.

(This is a replacement for golang.org/cl/190258.)

Updates #6977.

Change-Id: Ic8b9321374ab4f617498d97c12871b69d1119735
Reviewed-on: https://go-review.googlesource.com/c/go/+/191257Reviewed-by: default avatarMatthew Dempsky <mdempsky@google.com>
parent 89f02eb8
...@@ -559,7 +559,7 @@ func (check *Checker) builtin(x *operand, call *ast.CallExpr, id builtinId) (_ b ...@@ -559,7 +559,7 @@ func (check *Checker) builtin(x *operand, call *ast.CallExpr, id builtinId) (_ b
base := derefStructPtr(x.typ) base := derefStructPtr(x.typ)
sel := selx.Sel.Name sel := selx.Sel.Name
obj, index, indirect := LookupFieldOrMethod(base, false, check.pkg, sel) obj, index, indirect := check.LookupFieldOrMethod(base, false, check.pkg, sel)
switch obj.(type) { switch obj.(type) {
case nil: case nil:
check.invalidArg(x.pos(), "%s has no single field %s", base, sel) check.invalidArg(x.pos(), "%s has no single field %s", base, sel)
......
...@@ -370,7 +370,7 @@ func (check *Checker) selector(x *operand, e *ast.SelectorExpr) { ...@@ -370,7 +370,7 @@ func (check *Checker) selector(x *operand, e *ast.SelectorExpr) {
goto Error goto Error
} }
obj, index, indirect = LookupFieldOrMethod(x.typ, x.mode == variable, check.pkg, sel) obj, index, indirect = check.LookupFieldOrMethod(x.typ, x.mode == variable, check.pkg, sel)
if obj == nil { if obj == nil {
switch { switch {
case index != nil: case index != nil:
...@@ -437,6 +437,10 @@ func (check *Checker) selector(x *operand, e *ast.SelectorExpr) { ...@@ -437,6 +437,10 @@ func (check *Checker) selector(x *operand, e *ast.SelectorExpr) {
if debug { if debug {
// Verify that LookupFieldOrMethod and MethodSet.Lookup agree. // Verify that LookupFieldOrMethod and MethodSet.Lookup agree.
// TODO(gri) This only works because we call LookupFieldOrMethod
// _before_ calling NewMethodSet: LookupFieldOrMethod completes
// any incomplete interfaces so they are avaible to NewMethodSet
// (which assumes that interfaces have been completed already).
typ := x.typ typ := x.typ
if x.mode == variable { if x.mode == variable {
// If typ is not an (unnamed) pointer or an interface, // If typ is not an (unnamed) pointer or an interface,
......
...@@ -76,8 +76,9 @@ type Checker struct { ...@@ -76,8 +76,9 @@ type Checker struct {
fset *token.FileSet fset *token.FileSet
pkg *Package pkg *Package
*Info *Info
objMap map[Object]*declInfo // maps package-level objects and (non-interface) methods to declaration info objMap map[Object]*declInfo // maps package-level objects and (non-interface) methods to declaration info
impMap map[importKey]*Package // maps (import path, source directory) to (complete or fake) package impMap map[importKey]*Package // maps (import path, source directory) to (complete or fake) package
posMap map[*Interface][]token.Pos // maps interface types to lists of embedded interface positions
// information collected during type-checking of a set of package files // information collected during type-checking of a set of package files
// (initialized by Files, valid only for the duration of check.Files; // (initialized by Files, valid only for the duration of check.Files;
...@@ -86,12 +87,10 @@ type Checker struct { ...@@ -86,12 +87,10 @@ type Checker struct {
unusedDotImports map[*Scope]map[*Package]token.Pos // positions of unused dot-imported packages for each file scope unusedDotImports map[*Scope]map[*Package]token.Pos // positions of unused dot-imported packages for each file scope
firstErr error // first error encountered firstErr error // first error encountered
methods map[*TypeName][]*Func // maps package scope type names to associated non-blank, non-interface methods methods map[*TypeName][]*Func // maps package scope type names to associated non-blank (non-interface) methods
// TODO(gri) move interfaces up to the group of fields persistent across check.Files invocations (see also comment in Checker.initFiles) untyped map[ast.Expr]exprInfo // map of expressions without final type
interfaces map[*TypeName]*ifaceInfo // maps interface type names to corresponding interface infos delayed []func() // stack of delayed actions
untyped map[ast.Expr]exprInfo // map of expressions without final type objPath []Object // path of object dependencies during type inference (for cycle reporting)
delayed []func() // stack of delayed actions
objPath []Object // path of object dependencies during type inference (for cycle reporting)
// context within which the current object is type-checked // context within which the current object is type-checked
// (valid only for the duration of type-checking a specific object) // (valid only for the duration of type-checking a specific object)
...@@ -181,6 +180,7 @@ func NewChecker(conf *Config, fset *token.FileSet, pkg *Package, info *Info) *Ch ...@@ -181,6 +180,7 @@ func NewChecker(conf *Config, fset *token.FileSet, pkg *Package, info *Info) *Ch
Info: info, Info: info,
objMap: make(map[Object]*declInfo), objMap: make(map[Object]*declInfo),
impMap: make(map[importKey]*Package), impMap: make(map[importKey]*Package),
posMap: make(map[*Interface][]token.Pos),
} }
} }
...@@ -193,15 +193,6 @@ func (check *Checker) initFiles(files []*ast.File) { ...@@ -193,15 +193,6 @@ func (check *Checker) initFiles(files []*ast.File) {
check.firstErr = nil check.firstErr = nil
check.methods = nil check.methods = nil
// Don't clear the interfaces cache! It's important that we don't recompute
// ifaceInfos repeatedly (due to multiple check.Files calls) because when
// they are recomputed, they are not used in the context of their original
// declaration (because those types are already type-checked, typically) and
// then they will get the wrong receiver types, which matters for go/types
// clients. It is also safe to not reset the interfaces cache because files
// added to a package cannot change (add methods to) existing interface types;
// they can only add new interfaces. See also the respective comment in
// checker.infoFromTypeName (interfaces.go). Was bug - see issue #29029.
check.untyped = nil check.untyped = nil
check.delayed = nil check.delayed = nil
......
...@@ -97,6 +97,7 @@ var tests = [][]string{ ...@@ -97,6 +97,7 @@ var tests = [][]string{
{"testdata/issue23203a.src"}, {"testdata/issue23203a.src"},
{"testdata/issue23203b.src"}, {"testdata/issue23203b.src"},
{"testdata/issue28251.src"}, {"testdata/issue28251.src"},
{"testdata/issue6977.src"},
} }
var fset = token.NewFileSet() var fset = token.NewFileSet()
......
...@@ -82,6 +82,10 @@ func (check *Checker) err(pos token.Pos, msg string, soft bool) { ...@@ -82,6 +82,10 @@ func (check *Checker) err(pos token.Pos, msg string, soft bool) {
check.firstErr = err check.firstErr = err
} }
if trace {
check.trace(pos, "ERROR: %s", msg)
}
f := check.conf.Error f := check.conf.Error
if f == nil { if f == nil {
panic(bailout{}) // report only first error panic(bailout{}) // report only first error
......
This diff is collapsed.
...@@ -6,11 +6,6 @@ ...@@ -6,11 +6,6 @@
package types package types
// Internal use of LookupFieldOrMethod: If the obj result is a method
// associated with a concrete (non-interface) type, the method's signature
// may not be fully set up. Call Checker.objDecl(obj, nil) before accessing
// the method's type.
// LookupFieldOrMethod looks up a field or method with given package and name // LookupFieldOrMethod looks up a field or method with given package and name
// in T and returns the corresponding *Var or *Func, an index sequence, and a // in T and returns the corresponding *Var or *Func, an index sequence, and a
// bool indicating if there were any pointer indirections on the path to the // bool indicating if there were any pointer indirections on the path to the
...@@ -38,6 +33,19 @@ package types ...@@ -38,6 +33,19 @@ package types
// the method's formal receiver base type, nor was the receiver addressable. // the method's formal receiver base type, nor was the receiver addressable.
// //
func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (obj Object, index []int, indirect bool) { func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (obj Object, index []int, indirect bool) {
return (*Checker)(nil).LookupFieldOrMethod(T, addressable, pkg, name)
}
// Internal use of Checker.LookupFieldOrMethod: If the obj result is a method
// associated with a concrete (non-interface) type, the method's signature
// may not be fully set up. Call Checker.objDecl(obj, nil) before accessing
// the method's type.
// TODO(gri) Now that we provide the *Checker, we can probably remove this
// caveat by calling Checker.objDecl from LookupFieldOrMethod. Investigate.
// LookupFieldOrMethod is like the external version but completes interfaces
// as necessary.
func (check *Checker) LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (obj Object, index []int, indirect bool) {
// Methods cannot be associated to a named pointer type // Methods cannot be associated to a named pointer type
// (spec: "The type denoted by T is called the receiver base type; // (spec: "The type denoted by T is called the receiver base type;
// it must not be a pointer or interface type and it must be declared // it must not be a pointer or interface type and it must be declared
...@@ -47,7 +55,7 @@ func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o ...@@ -47,7 +55,7 @@ func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o
// not have found it for T (see also issue 8590). // not have found it for T (see also issue 8590).
if t, _ := T.(*Named); t != nil { if t, _ := T.(*Named); t != nil {
if p, _ := t.underlying.(*Pointer); p != nil { if p, _ := t.underlying.(*Pointer); p != nil {
obj, index, indirect = lookupFieldOrMethod(p, false, pkg, name) obj, index, indirect = check.lookupFieldOrMethod(p, false, pkg, name)
if _, ok := obj.(*Func); ok { if _, ok := obj.(*Func); ok {
return nil, nil, false return nil, nil, false
} }
...@@ -55,7 +63,7 @@ func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o ...@@ -55,7 +63,7 @@ func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o
} }
} }
return lookupFieldOrMethod(T, addressable, pkg, name) return check.lookupFieldOrMethod(T, addressable, pkg, name)
} }
// TODO(gri) The named type consolidation and seen maps below must be // TODO(gri) The named type consolidation and seen maps below must be
...@@ -63,7 +71,8 @@ func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o ...@@ -63,7 +71,8 @@ func LookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o
// types always have only one representation (even when imported // types always have only one representation (even when imported
// indirectly via different packages.) // indirectly via different packages.)
func lookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (obj Object, index []int, indirect bool) { // lookupFieldOrMethod should only be called by LookupFieldOrMethod and missingMethod.
func (check *Checker) lookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (obj Object, index []int, indirect bool) {
// WARNING: The code in this function is extremely subtle - do not modify casually! // WARNING: The code in this function is extremely subtle - do not modify casually!
// This function and NewMethodSet should be kept in sync. // This function and NewMethodSet should be kept in sync.
...@@ -166,6 +175,7 @@ func lookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o ...@@ -166,6 +175,7 @@ func lookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o
case *Interface: case *Interface:
// look for a matching method // look for a matching method
// TODO(gri) t.allMethods is sorted - use binary search // TODO(gri) t.allMethods is sorted - use binary search
check.completeInterface(t)
if i, m := lookupMethod(t.allMethods, pkg, name); m != nil { if i, m := lookupMethod(t.allMethods, pkg, name); m != nil {
assert(m.typ != nil) assert(m.typ != nil)
index = concat(e.index, i) index = concat(e.index, i)
...@@ -261,6 +271,8 @@ func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType b ...@@ -261,6 +271,8 @@ func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType b
// an exported API call (such as MissingMethod), i.e., when all // an exported API call (such as MissingMethod), i.e., when all
// methods have been type-checked. // methods have been type-checked.
func (check *Checker) missingMethod(V Type, T *Interface, static bool) (method *Func, wrongType bool) { func (check *Checker) missingMethod(V Type, T *Interface, static bool) (method *Func, wrongType bool) {
check.completeInterface(T)
// fast path for common case // fast path for common case
if T.Empty() { if T.Empty() {
return return
...@@ -269,6 +281,7 @@ func (check *Checker) missingMethod(V Type, T *Interface, static bool) (method * ...@@ -269,6 +281,7 @@ func (check *Checker) missingMethod(V Type, T *Interface, static bool) (method *
// TODO(gri) Consider using method sets here. Might be more efficient. // TODO(gri) Consider using method sets here. Might be more efficient.
if ityp, _ := V.Underlying().(*Interface); ityp != nil { if ityp, _ := V.Underlying().(*Interface); ityp != nil {
check.completeInterface(ityp)
// TODO(gri) allMethods is sorted - can do this more efficiently // TODO(gri) allMethods is sorted - can do this more efficiently
for _, m := range T.allMethods { for _, m := range T.allMethods {
_, obj := lookupMethod(ityp.allMethods, m.pkg, m.name) _, obj := lookupMethod(ityp.allMethods, m.pkg, m.name)
...@@ -286,7 +299,7 @@ func (check *Checker) missingMethod(V Type, T *Interface, static bool) (method * ...@@ -286,7 +299,7 @@ func (check *Checker) missingMethod(V Type, T *Interface, static bool) (method *
// A concrete type implements T if it implements all methods of T. // A concrete type implements T if it implements all methods of T.
for _, m := range T.allMethods { for _, m := range T.allMethods {
obj, _, _ := lookupFieldOrMethod(V, false, m.pkg, m.name) obj, _, _ := check.lookupFieldOrMethod(V, false, m.pkg, m.name)
// we must have a method (not a field of matching function type) // we must have a method (not a field of matching function type)
f, _ := obj.(*Func) f, _ := obj.(*Func)
......
...@@ -62,6 +62,11 @@ func (s *MethodSet) Lookup(pkg *Package, name string) *Selection { ...@@ -62,6 +62,11 @@ func (s *MethodSet) Lookup(pkg *Package, name string) *Selection {
// Shared empty method set. // Shared empty method set.
var emptyMethodSet MethodSet var emptyMethodSet MethodSet
// Note: NewMethodSet is intended for external use only as it
// requires interfaces to be complete. If may be used
// internally if LookupFieldOrMethod completed the same
// interfaces beforehand.
// NewMethodSet returns the method set for the given type T. // NewMethodSet returns the method set for the given type T.
// It always returns a non-nil method set, even if it is empty. // It always returns a non-nil method set, even if it is empty.
func NewMethodSet(T Type) *MethodSet { func NewMethodSet(T Type) *MethodSet {
......
...@@ -22,13 +22,13 @@ func _() { ...@@ -22,13 +22,13 @@ func _() {
t.f(t) t.f(t)
t.f(u) t.f(u)
u.f(t) u.f(t)
u.f(u) u.f(u)
} }
// Test case for issue 6589. // Test case for issues #6589, #33656.
type A interface { type A interface {
a() interface { a() interface {
...@@ -45,26 +45,28 @@ type B interface { ...@@ -45,26 +45,28 @@ type B interface {
type AB interface { type AB interface {
a() interface { a() interface {
A A
B /* ERROR a redeclared */ // TODO(gri) there shouldn't be an error here. See issue #33656.
B // ERROR duplicate method a
} }
b() interface { b() interface {
A A
B /* ERROR a redeclared */ // TODO(gri) there shouldn't be an error here. See issue #33656.
B // ERROR duplicate method a
} }
} }
var x AB var x AB
var y interface { var y interface {
A A
B /* ERROR a redeclared */ B
} }
var _ = x /* ERROR cannot compare */ == y var _ = x == y
// Test case for issue 6638. // Test case for issue 6638.
type T interface { type T interface {
m() [T /* ERROR no value */ (nil).m()[0]]int m() [T(nil).m /* ERROR undefined */ ()[0]]int
} }
// Variations of this test case. // Variations of this test case.
......
...@@ -139,7 +139,7 @@ type ( ...@@ -139,7 +139,7 @@ type (
} }
I3 interface { I3 interface {
m1() m1()
m1 /* ERROR "redeclared" */ () m1 /* ERROR "duplicate method" */ ()
} }
I4 interface { I4 interface {
m1(x, y, x /* ERROR "redeclared" */ float32) m1(x, y, x /* ERROR "redeclared" */ float32)
......
// Copyright 2019 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 p
import "io"
// Alan's initial report.
type I interface { f(); String() string }
type J interface { g(); String() string }
type IJ1 = interface { I; J }
type IJ2 = interface { f(); g(); String() string }
var _ = (*IJ1)(nil) == (*IJ2)(nil) // static assert that IJ1 and IJ2 are identical types
// The canonical example.
type ReadWriteCloser interface { io.ReadCloser; io.WriteCloser }
// Some more cases.
type M interface { m() }
type M32 interface { m() int32 }
type M64 interface { m() int64 }
type U1 interface { m() }
type U2 interface { m(); M }
type U3 interface { M; m() }
type U4 interface { M; M; M }
type U5 interface { U1; U2; U3; U4 }
type U6 interface { m(); m /* ERROR duplicate method */ () }
type U7 interface { M32 /* ERROR duplicate method */ ; m() }
type U8 interface { m(); M32 /* ERROR duplicate method */ }
type U9 interface { M32; M64 /* ERROR duplicate method */ }
// Verify that repeated embedding of the same interface(s)
// eliminates duplicate methods early (rather than at the
// end) to prevent exponential memory and time use.
// Without early elimination, computing T29 may take dozens
// of minutes.
type (
T0 interface { m() }
T1 interface { T0; T0 }
T2 interface { T1; T1 }
T3 interface { T2; T2 }
T4 interface { T3; T3 }
T5 interface { T4; T4 }
T6 interface { T5; T5 }
T7 interface { T6; T6 }
T8 interface { T7; T7 }
T9 interface { T8; T8 }
T10 interface { T9; T9 }
T11 interface { T10; T10 }
T12 interface { T11; T11 }
T13 interface { T12; T12 }
T14 interface { T13; T13 }
T15 interface { T14; T14 }
T16 interface { T15; T15 }
T17 interface { T16; T16 }
T18 interface { T17; T17 }
T19 interface { T18; T18 }
T20 interface { T19; T19 }
T21 interface { T20; T20 }
T22 interface { T21; T21 }
T23 interface { T22; T22 }
T24 interface { T23; T23 }
T25 interface { T24; T24 }
T26 interface { T25; T25 }
T27 interface { T26; T26 }
T28 interface { T27; T27 }
T29 interface { T28; T28 }
)
// Verify that m is present.
var x T29
var _ = x.m
...@@ -91,7 +91,7 @@ func issue10979() { ...@@ -91,7 +91,7 @@ func issue10979() {
nosuchpkg /* ERROR undeclared name: nosuchpkg */ .Nosuchtype nosuchpkg /* ERROR undeclared name: nosuchpkg */ .Nosuchtype
} }
type I interface { type I interface {
I /* ERROR I\.m \(value of type func\(I\)\) is not a type */ .m I.m /* ERROR no field or method m */
m() m()
} }
} }
...@@ -259,10 +259,9 @@ type E = interface { ...@@ -259,10 +259,9 @@ type E = interface {
m() m()
} }
// Test case from issue. Eventually we may disallow this due // Test case from issue.
// to the cycle via the alias type name. But for now we make // cmd/compile reports a cycle as well.
// sure this is accepted. type issue25301b /* ERROR cycle */ = interface {
type issue25301b = interface {
m() interface{ issue25301b } m() interface{ issue25301b }
} }
......
...@@ -284,31 +284,27 @@ func NewInterfaceType(methods []*Func, embeddeds []Type) *Interface { ...@@ -284,31 +284,27 @@ func NewInterfaceType(methods []*Func, embeddeds []Type) *Interface {
return typ return typ
} }
var mset objset // set method receivers if necessary
for _, m := range methods { for _, m := range methods {
if mset.insert(m) != nil {
panic("multiple methods with the same name")
}
// set receiver if we don't have one
if sig := m.typ.(*Signature); sig.recv == nil { if sig := m.typ.(*Signature); sig.recv == nil {
sig.recv = NewVar(m.pos, m.pkg, "", typ) sig.recv = NewVar(m.pos, m.pkg, "", typ)
} }
} }
sort.Sort(byUniqueMethodName(methods))
if len(embeddeds) > 0 { // All embedded types should be interfaces; however, defined types
// All embedded types should be interfaces; however, defined types // may not yet be fully resolved. Only verify that non-defined types
// may not yet be fully resolved. Only verify that non-defined types // are interfaces. This matches the behavior of the code before the
// are interfaces. This matches the behavior of the code before the // fix for #25301 (issue #25596).
// fix for #25301 (issue #25596). for _, t := range embeddeds {
for _, t := range embeddeds { if _, ok := t.(*Named); !ok && !IsInterface(t) {
if _, ok := t.(*Named); !ok && !IsInterface(t) { panic("embedded type is not an interface")
panic("embedded type is not an interface")
}
} }
sort.Stable(byUniqueTypeName(embeddeds))
} }
// sort for API stability
sort.Sort(byUniqueMethodName(methods))
sort.Stable(byUniqueTypeName(embeddeds))
typ.methods = methods typ.methods = methods
typ.embeddeds = embeddeds typ.embeddeds = embeddeds
return typ return typ
...@@ -346,28 +342,45 @@ func (t *Interface) Empty() bool { return len(t.allMethods) == 0 } ...@@ -346,28 +342,45 @@ func (t *Interface) Empty() bool { return len(t.allMethods) == 0 }
// Complete computes the interface's method set. It must be called by users of // Complete computes the interface's method set. It must be called by users of
// NewInterfaceType and NewInterface after the interface's embedded types are // NewInterfaceType and NewInterface after the interface's embedded types are
// fully defined and before using the interface type in any way other than to // fully defined and before using the interface type in any way other than to
// form other types. Complete returns the receiver. // form other types. The interface must not contain duplicate methods or a
// panic occurs. Complete returns the receiver.
func (t *Interface) Complete() *Interface { func (t *Interface) Complete() *Interface {
// TODO(gri) consolidate this method with Checker.completeInterface
if t.allMethods != nil { if t.allMethods != nil {
return t return t
} }
// collect all methods t.allMethods = markComplete // avoid infinite recursion
var allMethods []*Func
allMethods = append(allMethods, t.methods...) var methods []*Func
for _, et := range t.embeddeds { var seen objset
it := et.Underlying().(*Interface) addMethod := func(m *Func, explicit bool) {
it.Complete() switch alt := seen.insert(m); {
// copy embedded methods unchanged (see issue #28282) case alt == nil:
allMethods = append(allMethods, it.allMethods...) methods = append(methods, m)
case explicit || !Identical(m.Type(), alt.Type()):
panic("duplicate method " + m.name)
default:
// silently drop method m
}
}
for _, m := range t.methods {
addMethod(m, true)
}
for _, typ := range t.embeddeds {
typ := typ.Underlying().(*Interface)
typ.Complete()
for _, m := range typ.allMethods {
addMethod(m, false)
}
} }
sort.Sort(byUniqueMethodName(allMethods))
// t.methods and/or t.embeddeds may have been empty if methods != nil {
if allMethods == nil { sort.Sort(byUniqueMethodName(methods))
allMethods = markComplete t.allMethods = methods
} }
t.allMethods = allMethods
return t return t
} }
......
...@@ -472,171 +472,127 @@ func (check *Checker) declareInSet(oset *objset, pos token.Pos, obj Object) bool ...@@ -472,171 +472,127 @@ func (check *Checker) declareInSet(oset *objset, pos token.Pos, obj Object) bool
} }
func (check *Checker) interfaceType(ityp *Interface, iface *ast.InterfaceType, def *Named) { func (check *Checker) interfaceType(ityp *Interface, iface *ast.InterfaceType, def *Named) {
// fast-track empty interface for _, f := range iface.Methods.List {
if iface.Methods.List == nil { if len(f.Names) > 0 {
ityp.allMethods = markComplete // We have a method with name f.Names[0].
return // (The parser ensures that there's only one method
} // and we don't care if a constructed AST has more.)
name := f.Names[0]
// collect embedded interfaces if name.Name == "_" {
// Only needed for printing and API. Delay collection check.errorf(name.Pos(), "invalid method name _")
// to end of type-checking (for package-global interfaces) continue // ignore
// when all types are complete. Local interfaces are handled }
// after each statement (as each statement processes delayed
// functions).
interfaceContext := check.context // capture for use in closure below
check.later(func() {
if trace {
check.trace(iface.Pos(), "-- delayed checking embedded interfaces of %v", iface)
check.indent++
defer func() {
check.indent--
}()
}
// The context must be restored since for local interfaces typ := check.indirectType(f.Type)
// delayed functions are processed after each statement sig, _ := typ.(*Signature)
// (was issue #24140). if sig == nil {
defer func(ctxt context) { if typ != Typ[Invalid] {
check.context = ctxt check.invalidAST(f.Type.Pos(), "%s is not a method signature", typ)
}(check.context)
check.context = interfaceContext
for _, f := range iface.Methods.List {
if len(f.Names) == 0 {
typ := check.indirectType(f.Type)
// typ should be a named type denoting an interface
// (the parser will make sure it's a named type but
// constructed ASTs may be wrong).
if typ == Typ[Invalid] {
continue // error reported before
} }
embed, _ := typ.Underlying().(*Interface) continue // ignore
if embed == nil { }
// use named receiver type if available (for better error messages)
var recvTyp Type = ityp
if def != nil {
recvTyp = def
}
sig.recv = NewVar(name.Pos(), check.pkg, "", recvTyp)
m := NewFunc(name.Pos(), check.pkg, name.Name, sig)
check.recordDef(name, m)
ityp.methods = append(ityp.methods, m)
} else {
// We have an embedded interface and f.Type is its
// (possibly qualified) embedded type name. Collect
// it if it's a valid interface.
typ := check.typ(f.Type)
if _, ok := underlying(typ).(*Interface); !ok {
if typ != Typ[Invalid] {
check.errorf(f.Type.Pos(), "%s is not an interface", typ) check.errorf(f.Type.Pos(), "%s is not an interface", typ)
continue
} }
// Correct embedded interfaces must be complete - continue
// don't just assert, but report error since this
// used to be the underlying cause for issue #18395.
if embed.allMethods == nil {
check.dump("%v: incomplete embedded interface %s", f.Type.Pos(), typ)
unreachable()
}
// collect interface
ityp.embeddeds = append(ityp.embeddeds, typ)
} }
ityp.embeddeds = append(ityp.embeddeds, typ)
check.posMap[ityp] = append(check.posMap[ityp], f.Type.Pos())
} }
// sort to match NewInterface/NewInterface2
// TODO(gri) we may be able to switch to source order
sort.Stable(byUniqueTypeName(ityp.embeddeds))
})
// compute method set
var tname *TypeName
var path []*TypeName
if def != nil {
tname = def.obj
path = []*TypeName{tname}
} }
info := check.infoFromTypeLit(check.scope, iface, tname, path)
if info == nil || info == &emptyIfaceInfo { if len(ityp.methods) == 0 && len(ityp.embeddeds) == 0 {
// we got an error or the empty interface - exit early // empty interface
ityp.allMethods = markComplete ityp.allMethods = markComplete
return return
} }
// use named receiver type if available (for better error messages) // sort for API stability
var recvTyp Type = ityp sort.Sort(byUniqueMethodName(ityp.methods))
if def != nil { sort.Stable(byUniqueTypeName(ityp.embeddeds))
recvTyp = def
check.later(func() { check.completeInterface(ityp) })
}
func (check *Checker) completeInterface(ityp *Interface) {
if ityp.allMethods != nil {
return
} }
// Correct receiver type for all methods explicitly declared // completeInterface may be called via the LookupFieldOrMethod or
// by this interface after we're done with type-checking at // MissingMethod external API in which case check will be nil. In
// this level. See comment below for details. // this case, type-checking must be finished and all interfaces
check.later(func() { // should have been completed.
for _, m := range ityp.methods { if check == nil {
m.typ.(*Signature).recv.typ = recvTyp panic("internal error: incomplete interface")
}
})
// collect methods
var sigfix []*methodInfo
for i, minfo := range info.methods {
fun := minfo.fun
if fun == nil {
name := minfo.src.Names[0]
pos := name.Pos()
// Don't type-check signature yet - use an
// empty signature now and update it later.
// But set up receiver since we know it and
// its position, and because interface method
// signatures don't get a receiver via regular
// type-checking (there isn't a receiver in the
// method's AST). Setting the receiver type is
// also important for ptrRecv() (see methodset.go).
//
// Note: For embedded methods, the receiver type
// should be the type of the interface that declared
// the methods in the first place. Since we get the
// methods here via methodInfo, which may be computed
// before we have all relevant interface types, we use
// the current interface's type (recvType). This may be
// the type of the interface embedding the interface that
// declared the methods. This doesn't matter for type-
// checking (we only care about the receiver type for
// the ptrRecv predicate, and it's never a pointer recv
// for interfaces), but it matters for go/types clients
// and for printing. We correct the receiver after type-
// checking.
//
// TODO(gri) Consider marking methods signatures
// as incomplete, for better error messages. See
// also the T4 and T5 tests in testdata/cycles2.src.
sig := new(Signature)
sig.recv = NewVar(pos, check.pkg, "", recvTyp)
fun = NewFunc(pos, check.pkg, name.Name, sig)
minfo.fun = fun
check.recordDef(name, fun)
sigfix = append(sigfix, minfo)
}
// fun != nil
if i < info.explicits {
ityp.methods = append(ityp.methods, fun)
}
ityp.allMethods = append(ityp.allMethods, fun)
} }
// fix signatures now that we have collected all methods if trace {
savedContext := check.context check.trace(token.NoPos, "complete %s", ityp)
for _, minfo := range sigfix { check.indent++
// (possibly embedded) methods must be type-checked within their scope and defer func() {
// type-checking them must not affect the current context (was issue #23914) check.indent--
check.context = context{scope: minfo.scope} check.trace(token.NoPos, "=> %s", ityp)
typ := check.indirectType(minfo.src.Type) }()
sig, _ := typ.(*Signature) }
if sig == nil {
if typ != Typ[Invalid] { ityp.allMethods = markComplete // avoid infinite recursion
check.invalidAST(minfo.src.Type.Pos(), "%s is not a method signature", typ)
} var methods []*Func
continue // keep method with empty method signature var seen objset
addMethod := func(m *Func, explicit bool) {
switch alt := seen.insert(m); {
case alt == nil:
methods = append(methods, m)
case explicit || !Identical(m.Type(), alt.Type()):
check.errorf(m.pos, "duplicate method %s", m.name)
// We use "other" rather than "previous" here because
// the first declaration seen may not be textually
// earlier in the source.
check.errorf(alt.Pos(), "\tother declaration of %s", m) // secondary error, \t indented
default:
// silently drop method m
} }
// update signature, but keep recv that was set up before
old := minfo.fun.typ.(*Signature)
sig.recv = old.recv
*old = *sig // update signature (don't replace pointer!)
} }
check.context = savedContext
// sort to match NewInterface/NewInterface2 for _, m := range ityp.methods {
// TODO(gri) we may be able to switch to source order addMethod(m, true)
sort.Sort(byUniqueMethodName(ityp.methods)) }
if ityp.allMethods == nil { posList := check.posMap[ityp]
ityp.allMethods = markComplete for i, typ := range ityp.embeddeds {
} else { pos := posList[i] // embedding position
sort.Sort(byUniqueMethodName(ityp.allMethods)) typ := typ.Underlying().(*Interface)
check.completeInterface(typ)
for _, m := range typ.allMethods {
copy := *m
copy.pos = pos // preserve embedding position
addMethod(&copy, false)
}
}
if methods != nil {
sort.Sort(byUniqueMethodName(methods))
ityp.allMethods = methods
} }
} }
......
// errorcheck
// Copyright 2009 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
type R interface { duplicate() }
type S interface { duplicate() }
type T interface { R; S } // ERROR "duplicate"
func main() {
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment