mirror of
https://github.com/golang/go
synced 2024-11-16 20:04:52 -07:00
go/types, types2: report an error when using a broken alias
The type checker doesn't have a general mechanism to "use" the type of a type alias whose type depends on a recursive type declaration which is not yet completely type-checked. In some cases, the type of a type alias is needed before it is determined; the type is incorrect (invalid) in that case but no error is reported. The type-checker is happy with this (incorrect type), but the compiler may crash under some circumstances. A correct fix will likely require some form of forwarding type which is a fairly pervasive change and may also affect the type checker API. This CL introduces a simple side table, a map of broken type aliases, which is consulted before the type associated with a type alias is used. If the type alias is broken, an error is reported. This is a stop-gap solution that prevents the compiler from crashing. The reported error refers to the corresponding issue which suggests a work-around that may be applicable in some cases. Also fix a minor error related to type cycles: If we have a cycle that doesn't start with a type, don't use a compiler error message that explicitly mentions "type". Fixes #50259. Fixes #50276. Fixes #50779. For #50729. Change-Id: Ie8e38f49ef724e742e8e78625e6d4f3d4014a52c Reviewed-on: https://go-review.googlesource.com/c/go/+/379916 Trust: Robert Griesemer <gri@golang.org> Run-TryBot: Robert Griesemer <gri@golang.org> Reviewed-by: Robert Findley <rfindley@google.com> TryBot-Result: Gopher Robot <gobot@golang.org>
This commit is contained in:
parent
fef14fdd1d
commit
fe85c24431
@ -130,6 +130,7 @@ type Checker struct {
|
||||
imports []*PkgName // list of imported packages
|
||||
dotImportMap map[dotImportKey]*PkgName // maps dot-imported objects to the package they were dot-imported through
|
||||
recvTParamMap map[*syntax.Name]*TypeParam // maps blank receiver type parameters to their type
|
||||
brokenAliases map[*TypeName]bool // set of aliases with broken (not yet determined) types
|
||||
unionTypeSets map[*Union]*_TypeSet // computed type sets for union types
|
||||
mono monoGraph // graph for detecting non-monomorphizable instantiation loops
|
||||
|
||||
@ -160,6 +161,27 @@ func (check *Checker) addDeclDep(to Object) {
|
||||
from.addDep(to)
|
||||
}
|
||||
|
||||
// brokenAlias records that alias doesn't have a determined type yet.
|
||||
// It also sets alias.typ to Typ[Invalid].
|
||||
func (check *Checker) brokenAlias(alias *TypeName) {
|
||||
if check.brokenAliases == nil {
|
||||
check.brokenAliases = make(map[*TypeName]bool)
|
||||
}
|
||||
check.brokenAliases[alias] = true
|
||||
alias.typ = Typ[Invalid]
|
||||
}
|
||||
|
||||
// validAlias records that alias has the valid type typ (possibly Typ[Invalid]).
|
||||
func (check *Checker) validAlias(alias *TypeName, typ Type) {
|
||||
delete(check.brokenAliases, alias)
|
||||
alias.typ = typ
|
||||
}
|
||||
|
||||
// isBrokenAlias reports whether alias doesn't have a determined type yet.
|
||||
func (check *Checker) isBrokenAlias(alias *TypeName) bool {
|
||||
return alias.typ == Typ[Invalid] && check.brokenAliases[alias]
|
||||
}
|
||||
|
||||
func (check *Checker) rememberUntyped(e syntax.Expr, lhs bool, mode operandMode, typ *Basic, val constant.Value) {
|
||||
m := check.untyped
|
||||
if m == nil {
|
||||
@ -333,6 +355,7 @@ func (check *Checker) checkFiles(files []*syntax.File) (err error) {
|
||||
check.pkgPathMap = nil
|
||||
check.seenPkgMap = nil
|
||||
check.recvTParamMap = nil
|
||||
check.brokenAliases = nil
|
||||
check.unionTypeSets = nil
|
||||
check.defTypes = nil
|
||||
check.ctxt = nil
|
||||
|
@ -314,8 +314,13 @@ func (check *Checker) cycleError(cycle []Object) {
|
||||
// cycle? That would be more consistent with other error messages.
|
||||
i := firstInSrc(cycle)
|
||||
obj := cycle[i]
|
||||
// If obj is a type alias, mark it as valid (not broken) in order to avoid follow-on errors.
|
||||
tname, _ := obj.(*TypeName)
|
||||
if tname != nil && tname.IsAlias() {
|
||||
check.validAlias(tname, Typ[Invalid])
|
||||
}
|
||||
var err error_
|
||||
if check.conf.CompilerErrorMessages {
|
||||
if tname != nil && check.conf.CompilerErrorMessages {
|
||||
err.errorf(obj, "invalid recursive type %s", obj.Name())
|
||||
} else {
|
||||
err.errorf(obj, "illegal cycle in declaration of %s", obj.Name())
|
||||
@ -502,9 +507,9 @@ func (check *Checker) typeDecl(obj *TypeName, tdecl *syntax.TypeDecl, def *Named
|
||||
check.versionErrorf(tdecl, "go1.9", "type aliases")
|
||||
}
|
||||
|
||||
obj.typ = Typ[Invalid]
|
||||
check.brokenAlias(obj)
|
||||
rhs = check.varType(tdecl.Type)
|
||||
obj.typ = rhs
|
||||
check.validAlias(obj, rhs)
|
||||
return
|
||||
}
|
||||
|
||||
|
@ -135,7 +135,7 @@ type (
|
||||
type (
|
||||
a struct{ *b }
|
||||
b = c
|
||||
c struct{ *b }
|
||||
c struct{ *b /* ERROR invalid use of type alias */ }
|
||||
)
|
||||
|
||||
// issue #24939
|
||||
@ -145,7 +145,7 @@ type (
|
||||
}
|
||||
|
||||
M interface {
|
||||
F() P
|
||||
F() P // ERROR invalid use of type alias
|
||||
}
|
||||
|
||||
P = interface {
|
||||
|
18
src/cmd/compile/internal/types2/testdata/fixedbugs/issue50259.go2
vendored
Normal file
18
src/cmd/compile/internal/types2/testdata/fixedbugs/issue50259.go2
vendored
Normal file
@ -0,0 +1,18 @@
|
||||
// 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 p
|
||||
|
||||
var x T[B]
|
||||
|
||||
type T[_ any] struct{}
|
||||
type A T[B /* ERROR invalid use of type alias */ ]
|
||||
type B = T[A]
|
||||
|
||||
// test case from issue
|
||||
|
||||
var v Box[Step]
|
||||
type Box[T any] struct{}
|
||||
type Step = Box[StepBox]
|
||||
type StepBox Box[Step /* ERROR invalid use of type alias */ ]
|
39
src/cmd/compile/internal/types2/testdata/fixedbugs/issue50276.go2
vendored
Normal file
39
src/cmd/compile/internal/types2/testdata/fixedbugs/issue50276.go2
vendored
Normal file
@ -0,0 +1,39 @@
|
||||
// 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 p
|
||||
|
||||
// simplified test case
|
||||
|
||||
type transform[T any] struct{}
|
||||
type pair[S any] struct {}
|
||||
|
||||
var _ transform[step]
|
||||
|
||||
type box transform[step /* ERROR invalid use of type alias */ ]
|
||||
type step = pair[box]
|
||||
|
||||
// test case from issue
|
||||
|
||||
type Transform[T any] struct{ hold T }
|
||||
type Pair[S, T any] struct {
|
||||
First S
|
||||
Second T
|
||||
}
|
||||
|
||||
var first Transform[Step]
|
||||
|
||||
// This line doesn't use the Step alias, and it compiles fine if you uncomment it.
|
||||
var second Transform[Pair[Box, interface{}]]
|
||||
|
||||
type Box *Transform[Step /* ERROR invalid use of type alias */ ]
|
||||
|
||||
// This line is the same as the `first` line, but it comes after the Box declaration and
|
||||
// does not break the compile.
|
||||
var third Transform[Step]
|
||||
|
||||
type Step = Pair[Box, interface{}]
|
||||
|
||||
// This line also does not break the compile
|
||||
var fourth Transform[Step]
|
23
src/cmd/compile/internal/types2/testdata/fixedbugs/issue50779.go2
vendored
Normal file
23
src/cmd/compile/internal/types2/testdata/fixedbugs/issue50779.go2
vendored
Normal file
@ -0,0 +1,23 @@
|
||||
// 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 p
|
||||
|
||||
type AC interface {
|
||||
C
|
||||
}
|
||||
|
||||
type ST []int
|
||||
|
||||
type R[S any, P any] struct{}
|
||||
|
||||
type SR = R[SS, ST]
|
||||
|
||||
type SS interface {
|
||||
NSR(any) *SR // ERROR invalid use of type alias SR in recursive type
|
||||
}
|
||||
|
||||
type C interface {
|
||||
NSR(any) *SR
|
||||
}
|
@ -99,6 +99,10 @@ func (check *Checker) ident(x *operand, e *syntax.Name, def *Named, wantType boo
|
||||
x.mode = constant_
|
||||
|
||||
case *TypeName:
|
||||
if check.isBrokenAlias(obj) {
|
||||
check.errorf(e, "invalid use of type alias %s in recursive type (see issue #50729)", obj.name)
|
||||
return
|
||||
}
|
||||
x.mode = typexpr
|
||||
|
||||
case *Var:
|
||||
|
@ -137,6 +137,7 @@ type Checker struct {
|
||||
imports []*PkgName // list of imported packages
|
||||
dotImportMap map[dotImportKey]*PkgName // maps dot-imported objects to the package they were dot-imported through
|
||||
recvTParamMap map[*ast.Ident]*TypeParam // maps blank receiver type parameters to their type
|
||||
brokenAliases map[*TypeName]bool // set of aliases with broken (not yet determined) types
|
||||
unionTypeSets map[*Union]*_TypeSet // computed type sets for union types
|
||||
mono monoGraph // graph for detecting non-monomorphizable instantiation loops
|
||||
|
||||
@ -167,6 +168,27 @@ func (check *Checker) addDeclDep(to Object) {
|
||||
from.addDep(to)
|
||||
}
|
||||
|
||||
// brokenAlias records that alias doesn't have a determined type yet.
|
||||
// It also sets alias.typ to Typ[Invalid].
|
||||
func (check *Checker) brokenAlias(alias *TypeName) {
|
||||
if check.brokenAliases == nil {
|
||||
check.brokenAliases = make(map[*TypeName]bool)
|
||||
}
|
||||
check.brokenAliases[alias] = true
|
||||
alias.typ = Typ[Invalid]
|
||||
}
|
||||
|
||||
// validAlias records that alias has the valid type typ (possibly Typ[Invalid]).
|
||||
func (check *Checker) validAlias(alias *TypeName, typ Type) {
|
||||
delete(check.brokenAliases, alias)
|
||||
alias.typ = typ
|
||||
}
|
||||
|
||||
// isBrokenAlias reports whether alias doesn't have a determined type yet.
|
||||
func (check *Checker) isBrokenAlias(alias *TypeName) bool {
|
||||
return alias.typ == Typ[Invalid] && check.brokenAliases[alias]
|
||||
}
|
||||
|
||||
func (check *Checker) rememberUntyped(e ast.Expr, lhs bool, mode operandMode, typ *Basic, val constant.Value) {
|
||||
m := check.untyped
|
||||
if m == nil {
|
||||
@ -326,6 +348,7 @@ func (check *Checker) checkFiles(files []*ast.File) (err error) {
|
||||
check.pkgPathMap = nil
|
||||
check.seenPkgMap = nil
|
||||
check.recvTParamMap = nil
|
||||
check.brokenAliases = nil
|
||||
check.unionTypeSets = nil
|
||||
check.defTypes = nil
|
||||
check.ctxt = nil
|
||||
|
@ -313,7 +313,16 @@ func (check *Checker) cycleError(cycle []Object) {
|
||||
// cycle? That would be more consistent with other error messages.
|
||||
i := firstInSrc(cycle)
|
||||
obj := cycle[i]
|
||||
check.errorf(obj, _InvalidDeclCycle, "illegal cycle in declaration of %s", obj.Name())
|
||||
// If obj is a type alias, mark it as valid (not broken) in order to avoid follow-on errors.
|
||||
tname, _ := obj.(*TypeName)
|
||||
if tname != nil && tname.IsAlias() {
|
||||
check.validAlias(tname, Typ[Invalid])
|
||||
}
|
||||
if tname != nil && compilerErrorMessages {
|
||||
check.errorf(obj, _InvalidDeclCycle, "invalid recursive type %s", obj.Name())
|
||||
} else {
|
||||
check.errorf(obj, _InvalidDeclCycle, "illegal cycle in declaration of %s", obj.Name())
|
||||
}
|
||||
for range cycle {
|
||||
check.errorf(obj, _InvalidDeclCycle, "\t%s refers to", obj.Name()) // secondary error, \t indented
|
||||
i++
|
||||
@ -555,9 +564,9 @@ func (check *Checker) typeDecl(obj *TypeName, tdecl *ast.TypeSpec, def *Named) {
|
||||
check.errorf(atPos(tdecl.Assign), _BadDecl, "type aliases requires go1.9 or later")
|
||||
}
|
||||
|
||||
obj.typ = Typ[Invalid]
|
||||
check.brokenAlias(obj)
|
||||
rhs = check.varType(tdecl.Type)
|
||||
obj.typ = rhs
|
||||
check.validAlias(obj, rhs)
|
||||
return
|
||||
}
|
||||
|
||||
|
4
src/go/types/testdata/check/cycles5.src
vendored
4
src/go/types/testdata/check/cycles5.src
vendored
@ -135,7 +135,7 @@ type (
|
||||
type (
|
||||
a struct{ *b }
|
||||
b = c
|
||||
c struct{ *b }
|
||||
c struct{ *b /* ERROR invalid use of type alias */ }
|
||||
)
|
||||
|
||||
// issue #24939
|
||||
@ -145,7 +145,7 @@ type (
|
||||
}
|
||||
|
||||
M interface {
|
||||
F() P
|
||||
F() P // ERROR invalid use of type alias
|
||||
}
|
||||
|
||||
P = interface {
|
||||
|
18
src/go/types/testdata/fixedbugs/issue50259.go2
vendored
Normal file
18
src/go/types/testdata/fixedbugs/issue50259.go2
vendored
Normal file
@ -0,0 +1,18 @@
|
||||
// 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 p
|
||||
|
||||
var x T[B]
|
||||
|
||||
type T[_ any] struct{}
|
||||
type A T[B /* ERROR invalid use of type alias */ ]
|
||||
type B = T[A]
|
||||
|
||||
// test case from issue
|
||||
|
||||
var v Box[Step]
|
||||
type Box[T any] struct{}
|
||||
type Step = Box[StepBox]
|
||||
type StepBox Box[Step /* ERROR invalid use of type alias */ ]
|
39
src/go/types/testdata/fixedbugs/issue50276.go2
vendored
Normal file
39
src/go/types/testdata/fixedbugs/issue50276.go2
vendored
Normal file
@ -0,0 +1,39 @@
|
||||
// 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 p
|
||||
|
||||
// simplified test case
|
||||
|
||||
type transform[T any] struct{}
|
||||
type pair[S any] struct {}
|
||||
|
||||
var _ transform[step]
|
||||
|
||||
type box transform[step /* ERROR invalid use of type alias */ ]
|
||||
type step = pair[box]
|
||||
|
||||
// test case from issue
|
||||
|
||||
type Transform[T any] struct{ hold T }
|
||||
type Pair[S, T any] struct {
|
||||
First S
|
||||
Second T
|
||||
}
|
||||
|
||||
var first Transform[Step]
|
||||
|
||||
// This line doesn't use the Step alias, and it compiles fine if you uncomment it.
|
||||
var second Transform[Pair[Box, interface{}]]
|
||||
|
||||
type Box *Transform[Step /* ERROR invalid use of type alias */ ]
|
||||
|
||||
// This line is the same as the `first` line, but it comes after the Box declaration and
|
||||
// does not break the compile.
|
||||
var third Transform[Step]
|
||||
|
||||
type Step = Pair[Box, interface{}]
|
||||
|
||||
// This line also does not break the compile
|
||||
var fourth Transform[Step]
|
23
src/go/types/testdata/fixedbugs/issue50779.go2
vendored
Normal file
23
src/go/types/testdata/fixedbugs/issue50779.go2
vendored
Normal file
@ -0,0 +1,23 @@
|
||||
// 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 p
|
||||
|
||||
type AC interface {
|
||||
C
|
||||
}
|
||||
|
||||
type ST []int
|
||||
|
||||
type R[S any, P any] struct{}
|
||||
|
||||
type SR = R[SS, ST]
|
||||
|
||||
type SS interface {
|
||||
NSR(any) *SR // ERROR invalid use of type alias SR in recursive type
|
||||
}
|
||||
|
||||
type C interface {
|
||||
NSR(any) *SR
|
||||
}
|
@ -96,6 +96,10 @@ func (check *Checker) ident(x *operand, e *ast.Ident, def *Named, wantType bool)
|
||||
x.mode = constant_
|
||||
|
||||
case *TypeName:
|
||||
if check.isBrokenAlias(obj) {
|
||||
check.errorf(e, _InvalidDeclCycle, "invalid use of type alias %s in recursive type (see issue #50729)", obj.name)
|
||||
return
|
||||
}
|
||||
x.mode = typexpr
|
||||
|
||||
case *Var:
|
||||
|
13
test/typeparam/issue50259.go
Normal file
13
test/typeparam/issue50259.go
Normal file
@ -0,0 +1,13 @@
|
||||
// errorcheck -G=3
|
||||
|
||||
// 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 p
|
||||
|
||||
var x T[B]
|
||||
|
||||
type T[_ any] struct{}
|
||||
type A T[B] // ERROR "invalid use of type alias B in recursive type"
|
||||
type B = T[A]
|
Loading…
Reference in New Issue
Block a user