1
0
mirror of https://github.com/golang/go synced 2024-11-17 13:25:15 -07:00

go/types: prepare for delayed type-checking of methods to when they are used

Remove assumption that methods associated to concrete (non-interface)
types have a fully set up signature. Such methods are found through
LookupFieldOrMethod or lookupMethod, or indexed method access from
a Named type. Make sure that the method's signature is type-checked
before use in those cases.

(MethodSets also hold methods but the type checker is not using
them but for internal verification. API clients will be using it
after all methods have been type-checked.)

Some functions such as MissingMethod may now have to type-check a
method and for that they need a *Checker. Add helper functions as
necessary to provide the additional (receiver) parameter but permit
it to be nil if the respective functions are invoked through the API
(at which point we know that all methods have a proper signature and
thus we don't need the delayed type-check).

Since all package-level objects eventually are type-checked through
the top-level loop in Checker.packageObjects we are guaranteed that
all methods will be type-checked as well.

Updates #23203.
Updates #26854.

Change-Id: I6e48f0016cefd498aa70b776e84a48215a9042c5
Reviewed-on: https://go-review.googlesource.com/c/139425
Reviewed-by: Alan Donovan <adonovan@google.com>
This commit is contained in:
Robert Griesemer 2018-10-03 17:50:02 -07:00
parent f2c1c7acf8
commit bf9240681d
11 changed files with 66 additions and 22 deletions

View File

@ -353,20 +353,20 @@ func (conf *Config) Check(path string, fset *token.FileSet, files []*ast.File, i
// AssertableTo reports whether a value of type V can be asserted to have type T. // AssertableTo reports whether a value of type V can be asserted to have type T.
func AssertableTo(V *Interface, T Type) bool { func AssertableTo(V *Interface, T Type) bool {
m, _ := assertableTo(V, T) m, _ := (*Checker)(nil).assertableTo(V, T)
return m == nil return m == nil
} }
// AssignableTo reports whether a value of type V is assignable to a variable of type T. // AssignableTo reports whether a value of type V is assignable to a variable of type T.
func AssignableTo(V, T Type) bool { func AssignableTo(V, T Type) bool {
x := operand{mode: value, typ: V} x := operand{mode: value, typ: V}
return x.assignableTo(nil, T, nil) // config not needed for non-constant x return x.assignableTo(nil, T, nil) // check not needed for non-constant x
} }
// ConvertibleTo reports whether a value of type V is convertible to a value of type T. // ConvertibleTo reports whether a value of type V is convertible to a value of type T.
func ConvertibleTo(V, T Type) bool { func ConvertibleTo(V, T Type) bool {
x := operand{mode: value, typ: V} x := operand{mode: value, typ: V}
return x.convertibleTo(nil, T) // config not needed for non-constant x return x.convertibleTo(nil, T) // check not needed for non-constant x
} }
// Implements reports whether type V implements interface T. // Implements reports whether type V implements interface T.

View File

@ -57,7 +57,7 @@ func (check *Checker) assignment(x *operand, T Type, context string) {
return return
} }
if reason := ""; !x.assignableTo(check.conf, T, &reason) { if reason := ""; !x.assignableTo(check, T, &reason) {
if reason != "" { if reason != "" {
check.errorf(x.pos(), "cannot use %s as %s value in %s: %s", x, T, context, reason) check.errorf(x.pos(), "cannot use %s as %s value in %s: %s", x, T, context, reason)
} else { } else {

View File

@ -95,7 +95,7 @@ func (check *Checker) builtin(x *operand, call *ast.CallExpr, id builtinId) (_ b
// spec: "As a special case, append also accepts a first argument assignable // spec: "As a special case, append also accepts a first argument assignable
// to type []byte with a second argument of string type followed by ... . // to type []byte with a second argument of string type followed by ... .
// This form appends the bytes of the string. // This form appends the bytes of the string.
if nargs == 2 && call.Ellipsis.IsValid() && x.assignableTo(check.conf, NewSlice(universeByte), nil) { if nargs == 2 && call.Ellipsis.IsValid() && x.assignableTo(check, NewSlice(universeByte), nil) {
arg(x, 1) arg(x, 1)
if x.mode == invalid { if x.mode == invalid {
return return
@ -345,7 +345,7 @@ func (check *Checker) builtin(x *operand, call *ast.CallExpr, id builtinId) (_ b
return return
} }
if !x.assignableTo(check.conf, m.key, nil) { if !x.assignableTo(check, m.key, nil) {
check.invalidArg(x.pos(), "%s is not assignable to %s", x, m.key) check.invalidArg(x.pos(), "%s is not assignable to %s", x, m.key)
return return
} }

View File

@ -383,6 +383,11 @@ func (check *Checker) selector(x *operand, e *ast.SelectorExpr) {
goto Error goto Error
} }
// methods may not have a fully set up signature yet
if m, _ := obj.(*Func); m != nil {
check.objDecl(m, nil)
}
if x.mode == typexpr { if x.mode == typexpr {
// method expression // method expression
m, _ := obj.(*Func) m, _ := obj.(*Func)

View File

@ -18,7 +18,7 @@ func (check *Checker) conversion(x *operand, T Type) {
case constArg && isConstType(T): case constArg && isConstType(T):
// constant conversion // constant conversion
switch t := T.Underlying().(*Basic); { switch t := T.Underlying().(*Basic); {
case representableConst(x.val, check.conf, t, &x.val): case representableConst(x.val, check, t, &x.val):
ok = true ok = true
case isInteger(x.typ) && isString(t): case isInteger(x.typ) && isString(t):
codepoint := int64(-1) codepoint := int64(-1)
@ -31,7 +31,7 @@ func (check *Checker) conversion(x *operand, T Type) {
x.val = constant.MakeString(string(codepoint)) x.val = constant.MakeString(string(codepoint))
ok = true ok = true
} }
case x.convertibleTo(check.conf, T): case x.convertibleTo(check, T):
// non-constant conversion // non-constant conversion
x.mode = value x.mode = value
ok = true ok = true
@ -76,9 +76,12 @@ func (check *Checker) conversion(x *operand, T Type) {
// is tricky because we'd have to run updateExprType on the argument first. // is tricky because we'd have to run updateExprType on the argument first.
// (Issue #21982.) // (Issue #21982.)
func (x *operand) convertibleTo(conf *Config, T Type) bool { // convertibleTo reports whether T(x) is valid.
// The check parameter may be nil if convertibleTo is invoked through an
// exported API call, i.e., when all methods have been type-checked.
func (x *operand) convertibleTo(check *Checker, T Type) bool {
// "x is assignable to T" // "x is assignable to T"
if x.assignableTo(conf, T, nil) { if x.assignableTo(check, T, nil) {
return true return true
} }

View File

@ -586,6 +586,10 @@ func (check *Checker) addMethodDecls(obj *TypeName) {
} }
// type-check // type-check
// TODO(gri): This call is not needed anymore because the code can handle
// method signatures that have not yet been type-checked.
// Remove in separate CL to make it easy to isolate issues
// that might be introduced by this change.
check.objDecl(m, nil) check.objDecl(m, nil)
if base != nil { if base != nil {

View File

@ -187,11 +187,20 @@ func roundFloat64(x constant.Value) constant.Value {
// representable floating-point and complex values, and to an Int // representable floating-point and complex values, and to an Int
// value for integer values; it is left alone otherwise. // value for integer values; it is left alone otherwise.
// It is ok to provide the addressof the first argument for rounded. // It is ok to provide the addressof the first argument for rounded.
func representableConst(x constant.Value, conf *Config, typ *Basic, rounded *constant.Value) bool { //
// The check parameter may be nil if representableConst is invoked
// (indirectly) through an exported API call (AssignableTo, ConvertibleTo)
// because we don't need the Checker's config for those calls.
func representableConst(x constant.Value, check *Checker, typ *Basic, rounded *constant.Value) bool {
if x.Kind() == constant.Unknown { if x.Kind() == constant.Unknown {
return true // avoid follow-up errors return true // avoid follow-up errors
} }
var conf *Config
if check != nil {
conf = check.conf
}
switch { switch {
case isInteger(typ): case isInteger(typ):
x := constant.ToInt(x) x := constant.ToInt(x)
@ -323,7 +332,7 @@ func representableConst(x constant.Value, conf *Config, typ *Basic, rounded *con
// representable checks that a constant operand is representable in the given basic type. // representable checks that a constant operand is representable in the given basic type.
func (check *Checker) representable(x *operand, typ *Basic) { func (check *Checker) representable(x *operand, typ *Basic) {
assert(x.mode == constant_) assert(x.mode == constant_)
if !representableConst(x.val, check.conf, typ, &x.val) { if !representableConst(x.val, check, typ, &x.val) {
var msg string var msg string
if isNumeric(x.typ) && isNumeric(typ) { if isNumeric(x.typ) && isNumeric(typ) {
// numeric conversion : error msg // numeric conversion : error msg
@ -576,7 +585,7 @@ func (check *Checker) comparison(x, y *operand, op token.Token) {
// spec: "In any comparison, the first operand must be assignable // spec: "In any comparison, the first operand must be assignable
// to the type of the second operand, or vice versa." // to the type of the second operand, or vice versa."
err := "" err := ""
if x.assignableTo(check.conf, y.typ, nil) || y.assignableTo(check.conf, x.typ, nil) { if x.assignableTo(check, y.typ, nil) || y.assignableTo(check, x.typ, nil) {
defined := false defined := false
switch op { switch op {
case token.EQL, token.NEQ: case token.EQL, token.NEQ:
@ -1547,7 +1556,7 @@ func keyVal(x constant.Value) interface{} {
// typeAssertion checks that x.(T) is legal; xtyp must be the type of x. // typeAssertion checks that x.(T) is legal; xtyp must be the type of x.
func (check *Checker) typeAssertion(pos token.Pos, x *operand, xtyp *Interface, T Type) { func (check *Checker) typeAssertion(pos token.Pos, x *operand, xtyp *Interface, T Type) {
method, wrongType := assertableTo(xtyp, T) method, wrongType := check.assertableTo(xtyp, T)
if method == nil { if method == nil {
return return
} }

View File

@ -6,6 +6,11 @@
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
@ -112,7 +117,7 @@ func lookupFieldOrMethod(T Type, addressable bool, pkg *Package, name string) (o
// look for a matching attached method // look for a matching attached method
if i, m := lookupMethod(named.methods, pkg, name); m != nil { if i, m := lookupMethod(named.methods, pkg, name); m != nil {
// potential match // potential match
assert(m.typ != nil) // caution: method may not have a proper signature yet
index = concat(e.index, i) index = concat(e.index, i)
if obj != nil || e.multiples { if obj != nil || e.multiples {
return nil, index, false // collision return nil, index, false // collision
@ -248,6 +253,14 @@ func lookupType(m map[Type]int, typ Type) (int, bool) {
// x is of interface type V). // x is of interface type V).
// //
func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType bool) { func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType bool) {
return (*Checker)(nil).missingMethod(V, T, static)
}
// missingMethod is like MissingMethod but accepts a receiver.
// The receiver may be nil if missingMethod is invoked through
// an exported API call (such as MissingMethod), i.e., when all
// methods have been type-checked.
func (check *Checker) missingMethod(V Type, T *Interface, static bool) (method *Func, wrongType bool) {
// fast path for common case // fast path for common case
if T.Empty() { if T.Empty() {
return return
@ -275,11 +288,17 @@ func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType b
for _, m := range T.allMethods { for _, m := range T.allMethods {
obj, _, _ := lookupFieldOrMethod(V, false, m.pkg, m.name) obj, _, _ := lookupFieldOrMethod(V, false, m.pkg, m.name)
// we must have a method (not a field of matching function type)
f, _ := obj.(*Func) f, _ := obj.(*Func)
if f == nil { if f == nil {
return m, false return m, false
} }
// methods may not have a fully set up signature yet
if check != nil {
check.objDecl(f, nil)
}
if !Identical(f.typ, m.typ) { if !Identical(f.typ, m.typ) {
return m, true return m, true
} }
@ -291,14 +310,16 @@ func MissingMethod(V Type, T *Interface, static bool) (method *Func, wrongType b
// assertableTo reports whether a value of type V can be asserted to have type T. // 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 // 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. // method required by V and whether it is missing or just has the wrong type.
func assertableTo(V *Interface, T Type) (method *Func, wrongType bool) { // 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.
func (check *Checker) assertableTo(V *Interface, T Type) (method *Func, wrongType bool) {
// no static check is required if T is an interface // no static check is required if T is an interface
// spec: "If T is an interface type, x.(T) asserts that the // spec: "If T is an interface type, x.(T) asserts that the
// dynamic type of x implements the interface T." // dynamic type of x implements the interface T."
if _, ok := T.Underlying().(*Interface); ok && !strict { if _, ok := T.Underlying().(*Interface); ok && !strict {
return return
} }
return MissingMethod(T, V, false) return check.missingMethod(T, V, false)
} }
// deref dereferences typ if it is a *Pointer and returns its base and true. // deref dereferences typ if it is a *Pointer and returns its base and true.

View File

@ -201,7 +201,9 @@ func (x *operand) isNil() bool {
// assignableTo reports whether x is assignable to a variable of type T. // assignableTo reports whether x is assignable to a variable of type T.
// If the result is false and a non-nil reason is provided, it may be set // If the result is false and a non-nil reason is provided, it may be set
// to a more detailed explanation of the failure (result != ""). // to a more detailed explanation of the failure (result != "").
func (x *operand) assignableTo(conf *Config, T Type, reason *string) bool { // The check parameter may be nil if assignableTo is invoked through
// an exported API call, i.e., when all methods have been type-checked.
func (x *operand) assignableTo(check *Checker, T Type, reason *string) bool {
if x.mode == invalid || T == Typ[Invalid] { if x.mode == invalid || T == Typ[Invalid] {
return true // avoid spurious errors return true // avoid spurious errors
} }
@ -226,7 +228,7 @@ func (x *operand) assignableTo(conf *Config, T Type, reason *string) bool {
return true return true
} }
if x.mode == constant_ { if x.mode == constant_ {
return representableConst(x.val, conf, t, nil) return representableConst(x.val, check, t, nil)
} }
// The result of a comparison is an untyped boolean, // The result of a comparison is an untyped boolean,
// but may not be a constant. // but may not be a constant.
@ -249,7 +251,7 @@ func (x *operand) assignableTo(conf *Config, T Type, reason *string) bool {
// T is an interface type and x implements T // T is an interface type and x implements T
if Ti, ok := Tu.(*Interface); ok { if Ti, ok := Tu.(*Interface); ok {
if m, wrongType := MissingMethod(x.typ, Ti, true); m != nil /* Implements(x.typ, Ti) */ { if m, wrongType := check.missingMethod(x.typ, Ti, true); m != nil /* Implements(x.typ, Ti) */ {
if reason != nil { if reason != nil {
if wrongType { if wrongType {
*reason = "wrong type for method " + m.Name() *reason = "wrong type for method " + m.Name()

View File

@ -424,7 +424,7 @@ func (c *Chan) Elem() Type { return c.elem }
type Named struct { type Named struct {
obj *TypeName // corresponding declared object obj *TypeName // corresponding declared object
underlying Type // possibly a *Named during setup; never a *Named once set up completely underlying Type // possibly a *Named during setup; never a *Named once set up completely
methods []*Func // methods declared for this type (not the method set of this type) methods []*Func // methods declared for this type (not the method set of this type); signatures are type-checked lazily
} }
// NewNamed returns a new named type for the given type name, underlying type, and associated methods. // NewNamed returns a new named type for the given type name, underlying type, and associated methods.

View File

@ -391,7 +391,7 @@ func (check *Checker) arrayLength(e ast.Expr) int64 {
} }
if isUntyped(x.typ) || isInteger(x.typ) { if isUntyped(x.typ) || isInteger(x.typ) {
if val := constant.ToInt(x.val); val.Kind() == constant.Int { if val := constant.ToInt(x.val); val.Kind() == constant.Int {
if representableConst(val, check.conf, Typ[Int], nil) { if representableConst(val, check, Typ[Int], nil) {
if n, ok := constant.Int64Val(val); ok && n >= 0 { if n, ok := constant.Int64Val(val); ok && n >= 0 {
return n return n
} }