// Copyright 2022 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 types // validType verifies that the given type does not "expand" indefinitely // producing a cycle in the type graph. Cycles are detected by marking // defined types. // (Cycles involving alias types, as in "type A = [10]A" are detected // earlier, via the objDecl cycle detection mechanism.) func (check *Checker) validType(typ *Named) { check.validType0(typ, nil, nil) } type typeInfo uint // validType0 checks if the given type is valid. If typ is a type parameter // its value is looked up in the provided environment. The environment is // nil if typ is not part of (the RHS of) an instantiated type, in that case // any type parameter encountered must be from an enclosing function and can // be ignored. The path is the list of type names that lead to the current typ. func (check *Checker) validType0(typ Type, env *tparamEnv, path []Object) typeInfo { const ( unknown typeInfo = iota marked valid invalid ) switch t := typ.(type) { case nil: // We should never see a nil type but be conservative and panic // only in debug mode. if debug { panic("validType0(nil)") } case *Array: return check.validType0(t.elem, env, path) case *Struct: for _, f := range t.fields { if check.validType0(f.typ, env, path) == invalid { return invalid } } case *Union: for _, t := range t.terms { if check.validType0(t.typ, env, path) == invalid { return invalid } } case *Interface: for _, etyp := range t.embeddeds { if check.validType0(etyp, env, path) == invalid { return invalid } } case *Named: // Don't report a 2nd error if we already know the type is invalid // Note: ensure that t.orig is fully resolved by calling Underlying(). if t.Underlying() == Typ[Invalid] { check.infoMap[t] = invalid return invalid } switch check.infoMap[t] { case unknown: check.infoMap[t] = marked check.infoMap[t] = check.validType0(t.Origin().fromRHS, env.push(t), append(path, t.obj)) case marked: // We have seen type t before and thus must have a cycle. check.infoMap[t] = invalid // t cannot be in an imported package otherwise that package // would have reported a type cycle and couldn't have been // imported in the first place. assert(t.obj.pkg == check.pkg) t.underlying = Typ[Invalid] // t is in the current package (no race possibility) // Find the starting point of the cycle and report it. for i, tn := range path { if tn == t.obj { check.cycleError(path[i:]) return invalid } } panic("cycle start not found") } return check.infoMap[t] case *TypeParam: // A type parameter stands for the type (argument) it was instantiated with. // Check the corresponding type argument for validity if we have one. if env != nil { if targ := env.tmap[t]; targ != nil { // Type arguments found in targ must be looked // up in the enclosing environment env.link. return check.validType0(targ, env.link, path) } } } return valid } // A tparamEnv provides the environment for looking up the type arguments // with which type parameters for a given instance were instantiated. // If we don't have an instance, the corresponding tparamEnv is nil. type tparamEnv struct { tmap substMap link *tparamEnv } func (env *tparamEnv) push(typ *Named) *tparamEnv { // If typ is not an instantiated type there are no typ-specific // type parameters to look up and we don't need an environment. targs := typ.TypeArgs() if targs == nil { return nil // no instance => nil environment } // Populate tmap: remember the type argument for each type parameter. // We cannot use makeSubstMap because the number of type parameters // and arguments may not match due to errors in the source (too many // or too few type arguments). Populate tmap "manually". tparams := typ.TypeParams() n, m := targs.Len(), tparams.Len() if n > m { n = m // too many targs } tmap := make(substMap, n) for i := 0; i < n; i++ { tmap[tparams.At(i)] = targs.At(i) } return &tparamEnv{tmap: tmap, link: env} } // TODO(gri) Alternative implementation: // We may not need to build a stack of environments to // look up the type arguments for type parameters. The // same information should be available via the path: // We should be able to just walk the path backwards // and find the type arguments in the instance objects.