From 9f2fe2d721a10cd1066c52e22efc49a59a81b76d Mon Sep 17 00:00:00 2001 From: Robert Griesemer Date: Wed, 1 Mar 2023 08:06:42 -0800 Subject: [PATCH] go/types, types2: change missingMethod to match MissingMethod signature This simplifies the use of missingMethod and also opens the door to further missingMethod-internal simplifications. Change-Id: I74d9694b6fca67c4103aea04d08916a69ad0e3c5 Reviewed-on: https://go-review.googlesource.com/c/go/+/472495 Reviewed-by: Robert Griesemer TryBot-Result: Gopher Robot Reviewed-by: Robert Findley Run-TryBot: Robert Griesemer Auto-Submit: Robert Griesemer --- src/cmd/compile/internal/types2/expr.go | 3 +- src/cmd/compile/internal/types2/lookup.go | 34 ++++++++++------------- src/go/types/expr.go | 3 +- src/go/types/lookup.go | 34 ++++++++++------------- 4 files changed, 32 insertions(+), 42 deletions(-) diff --git a/src/cmd/compile/internal/types2/expr.go b/src/cmd/compile/internal/types2/expr.go index 0abb3fa3b5dc97..2baa80c4faab3c 100644 --- a/src/cmd/compile/internal/types2/expr.go +++ b/src/cmd/compile/internal/types2/expr.go @@ -1795,8 +1795,7 @@ func keyVal(x constant.Value) interface{} { // typeAssertion checks x.(T). The type of x must be an interface. func (check *Checker) typeAssertion(e syntax.Expr, x *operand, T Type, typeSwitch bool) { var cause string - method, _ := check.assertableTo(x.typ, T, &cause) - if method == nil { + if check.assertableTo(x.typ, T, &cause) { return // success } diff --git a/src/cmd/compile/internal/types2/lookup.go b/src/cmd/compile/internal/types2/lookup.go index be0e6b44293bd0..7c20a281362602 100644 --- a/src/cmd/compile/internal/types2/lookup.go +++ b/src/cmd/compile/internal/types2/lookup.go @@ -305,9 +305,7 @@ func (l *instanceLookup) add(inst *Named) { // present in V have matching types (e.g., for a type assertion x.(T) where // x is of interface type V). func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType bool) { - m, alt := (*Checker)(nil).missingMethod(V, T, static, Identical, nil) - // Only report a wrong type if the alternative method has the same name as m. - return m, alt != nil && alt.name == m.name // alt != nil implies m != nil + return (*Checker)(nil).missingMethod(V, T, static, Identical, nil) } // missingMethod is like MissingMethod but accepts a *Checker as receiver, @@ -318,17 +316,14 @@ func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType b // The underlying type of T must be an interface; T (rather than its under- // lying type) is used for better error messages (reported through *cause). // The comparator is used to compare signatures. -// If a method is missing and cause is not nil, *cause is set to the error cause. -// -// If a method is missing on T but is found on *T, or if a method is found -// on T when looked up with case-folding, this alternative method is returned -// as the second result. -func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y Type) bool, cause *string) (method, alt *Func) { +// If a method is missing and cause is not nil, *cause describes the error. +func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y Type) bool, cause *string) (method *Func, wrongType bool) { methods := under(T).(*Interface).typeSet().methods // T must be an interface if len(methods) == 0 { return } + var alt *Func if cause != nil { defer func() { if method != nil { @@ -347,11 +342,12 @@ func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y if !static { continue } - return m, nil + return m, false } if !equivalent(f.typ, m.typ) { - return m, f + alt = f + return m, true } } @@ -376,7 +372,7 @@ func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y // we must have a method (not a struct field) f, _ := obj.(*Func) if f == nil { - return m, nil + return m, false } // methods may not have a fully set up signature yet @@ -385,7 +381,8 @@ func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y } if !found || !equivalent(f.typ, m.typ) { - return m, f + alt = f + return m, f.name == m.name } } @@ -470,22 +467,21 @@ func (check *Checker) funcString(f *Func, pkgInfo bool) string { } // assertableTo reports whether a value of type V can be asserted to have type T. -// It returns (nil, false) as affirmative answer. Otherwise it returns a missing -// method required by V and whether it is missing or just has the wrong type. // The receiver may be nil if assertableTo is invoked through an exported API call // (such as AssertableTo), i.e., when all methods have been type-checked. // The underlying type of V must be an interface. -// If the result is negative and cause is not nil, *cause is set to the error cause. +// If the result is false and cause is not nil, *cause describes the error. // TODO(gri) replace calls to this function with calls to newAssertableTo. -func (check *Checker) assertableTo(V, T Type, cause *string) (method, wrongType *Func) { +func (check *Checker) assertableTo(V, T Type, cause *string) bool { // no static check is required if T is an interface // spec: "If T is an interface type, x.(T) asserts that the // dynamic type of x implements the interface T." if IsInterface(T) { - return + return true } // TODO(gri) fix this for generalized interfaces - return check.missingMethod(T, V, false, Identical, cause) + m, _ := check.missingMethod(T, V, false, Identical, cause) + return m == nil } // newAssertableTo reports whether a value of type V can be asserted to have type T. diff --git a/src/go/types/expr.go b/src/go/types/expr.go index c9ddef347326b5..7c87702bd8675f 100644 --- a/src/go/types/expr.go +++ b/src/go/types/expr.go @@ -1742,8 +1742,7 @@ func keyVal(x constant.Value) interface{} { // typeAssertion checks x.(T). The type of x must be an interface. func (check *Checker) typeAssertion(e ast.Expr, x *operand, T Type, typeSwitch bool) { var cause string - method, _ := check.assertableTo(x.typ, T, &cause) - if method == nil { + if check.assertableTo(x.typ, T, &cause) { return // success } diff --git a/src/go/types/lookup.go b/src/go/types/lookup.go index 893f3b8afc5710..09597888992d20 100644 --- a/src/go/types/lookup.go +++ b/src/go/types/lookup.go @@ -307,9 +307,7 @@ func (l *instanceLookup) add(inst *Named) { // present in V have matching types (e.g., for a type assertion x.(T) where // x is of interface type V). func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType bool) { - m, alt := (*Checker)(nil).missingMethod(V, T, static, Identical, nil) - // Only report a wrong type if the alternative method has the same name as m. - return m, alt != nil && alt.name == m.name // alt != nil implies m != nil + return (*Checker)(nil).missingMethod(V, T, static, Identical, nil) } // missingMethod is like MissingMethod but accepts a *Checker as receiver, @@ -320,17 +318,14 @@ func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType b // The underlying type of T must be an interface; T (rather than its under- // lying type) is used for better error messages (reported through *cause). // The comparator is used to compare signatures. -// If a method is missing and cause is not nil, *cause is set to the error cause. -// -// If a method is missing on T but is found on *T, or if a method is found -// on T when looked up with case-folding, this alternative method is returned -// as the second result. -func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y Type) bool, cause *string) (method, alt *Func) { +// If a method is missing and cause is not nil, *cause describes the error. +func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y Type) bool, cause *string) (method *Func, wrongType bool) { methods := under(T).(*Interface).typeSet().methods // T must be an interface if len(methods) == 0 { return } + var alt *Func if cause != nil { defer func() { if method != nil { @@ -349,11 +344,12 @@ func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y if !static { continue } - return m, nil + return m, false } if !equivalent(f.typ, m.typ) { - return m, f + alt = f + return m, true } } @@ -378,7 +374,7 @@ func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y // we must have a method (not a struct field) f, _ := obj.(*Func) if f == nil { - return m, nil + return m, false } // methods may not have a fully set up signature yet @@ -387,7 +383,8 @@ func (check *Checker) missingMethod(V, T Type, static bool, equivalent func(x, y } if !found || !equivalent(f.typ, m.typ) { - return m, f + alt = f + return m, f.name == m.name } } @@ -472,22 +469,21 @@ func (check *Checker) funcString(f *Func, pkgInfo bool) string { } // assertableTo reports whether a value of type V can be asserted to have type T. -// It returns (nil, false) as affirmative answer. Otherwise it returns a missing -// method required by V and whether it is missing or just has the wrong type. // The receiver may be nil if assertableTo is invoked through an exported API call // (such as AssertableTo), i.e., when all methods have been type-checked. // The underlying type of V must be an interface. -// If the result is negative and cause is not nil, *cause is set to the error cause. +// If the result is false and cause is not nil, *cause describes the error. // TODO(gri) replace calls to this function with calls to newAssertableTo. -func (check *Checker) assertableTo(V, T Type, cause *string) (method, wrongType *Func) { +func (check *Checker) assertableTo(V, T Type, cause *string) bool { // no static check is required if T is an interface // spec: "If T is an interface type, x.(T) asserts that the // dynamic type of x implements the interface T." if IsInterface(T) { - return + return true } // TODO(gri) fix this for generalized interfaces - return check.missingMethod(T, V, false, Identical, cause) + m, _ := check.missingMethod(T, V, false, Identical, cause) + return m == nil } // newAssertableTo reports whether a value of type V can be asserted to have type T.