Commit e87fe0f1 authored by LE Manh Cuong's avatar LE Manh Cuong Committed by Matthew Dempsky

cmd/compile: make typecheck set n.Type.Nod when returning OTYPE

typecheck only set n.Type.Nod for declared type, and leave it nil for
anonymous types, type alias. It leads to compiler crashes, because
n.Type.Nod is nil at the time dowidth was called.

Fixing it by set n.Type.Nod right after n.Type initialization if n.Op is
OTYPE.

When embedding interface cycles involve in type alias, it also helps
pointing the error message to the position of the type alias
declaration, instead of position of embedding interface.

Fixes #31872

Change-Id: Ia18391e987036a91f42ba0c08b5506f52d07f683
Reviewed-on: https://go-review.googlesource.com/c/go/+/191540
Run-TryBot: Cuong Manh Le <cuong.manhle.vn@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarMatthew Dempsky <mdempsky@google.com>
parent 647dc1af
...@@ -189,16 +189,7 @@ func dowidth(t *types.Type) { ...@@ -189,16 +189,7 @@ func dowidth(t *types.Type) {
if t.Width == -2 { if t.Width == -2 {
if !t.Broke() { if !t.Broke() {
t.SetBroke(true) t.SetBroke(true)
// t.Nod should not be nil here, but in some cases is appears to be yyerrorl(asNode(t.Nod).Pos, "invalid recursive type %v", t)
// (see issue #23823). For now (temporary work-around) at a minimum
// don't crash and provide a meaningful error message.
// TODO(gri) determine the correct fix during a regular devel cycle
// (see issue #31872).
if t.Nod == nil {
yyerror("invalid recursive type %v", t)
} else {
yyerrorl(asNode(t.Nod).Pos, "invalid recursive type %v", t)
}
} }
t.Width = 0 t.Width = 0
......
...@@ -456,8 +456,7 @@ func typecheck1(n *Node, top int) (res *Node) { ...@@ -456,8 +456,7 @@ func typecheck1(n *Node, top int) (res *Node) {
t = types.NewArray(r.Type, bound) t = types.NewArray(r.Type, bound)
} }
n.Op = OTYPE setTypeNode(n, t)
n.Type = t
n.Left = nil n.Left = nil
n.Right = nil n.Right = nil
if !t.IsDDDArray() { if !t.IsDDDArray() {
...@@ -480,8 +479,8 @@ func typecheck1(n *Node, top int) (res *Node) { ...@@ -480,8 +479,8 @@ func typecheck1(n *Node, top int) (res *Node) {
if r.Type.NotInHeap() { if r.Type.NotInHeap() {
yyerror("go:notinheap map value not allowed") yyerror("go:notinheap map value not allowed")
} }
n.Op = OTYPE
n.Type = types.NewMap(l.Type, r.Type) setTypeNode(n, types.NewMap(l.Type, r.Type))
mapqueue = append(mapqueue, n) // check map keys when all types are settled mapqueue = append(mapqueue, n) // check map keys when all types are settled
n.Left = nil n.Left = nil
n.Right = nil n.Right = nil
...@@ -497,37 +496,28 @@ func typecheck1(n *Node, top int) (res *Node) { ...@@ -497,37 +496,28 @@ func typecheck1(n *Node, top int) (res *Node) {
if l.Type.NotInHeap() { if l.Type.NotInHeap() {
yyerror("chan of go:notinheap type not allowed") yyerror("chan of go:notinheap type not allowed")
} }
t := types.NewChan(l.Type, n.TChanDir())
n.Op = OTYPE setTypeNode(n, types.NewChan(l.Type, n.TChanDir()))
n.Type = t
n.Left = nil n.Left = nil
n.ResetAux() n.ResetAux()
case OTSTRUCT: case OTSTRUCT:
ok |= Etype ok |= Etype
n.Op = OTYPE t := tostruct(n.List.Slice())
n.Type = tostruct(n.List.Slice()) if t.Broke() {
if n.Type == nil || n.Type.Broke() {
n.Type = nil n.Type = nil
return n return n
} }
setTypeNode(n, t)
n.List.Set(nil) n.List.Set(nil)
case OTINTER: case OTINTER:
ok |= Etype ok |= Etype
n.Op = OTYPE setTypeNode(n, tointerface(n.List.Slice()))
n.Type = tointerface(n.List.Slice())
if n.Type == nil {
return n
}
case OTFUNC: case OTFUNC:
ok |= Etype ok |= Etype
n.Op = OTYPE setTypeNode(n, functype(n.Left, n.List.Slice(), n.Rlist.Slice()))
n.Type = functype(n.Left, n.List.Slice(), n.Rlist.Slice())
if n.Type == nil {
return n
}
n.Left = nil n.Left = nil
n.List.Set(nil) n.List.Set(nil)
n.Rlist.Set(nil) n.Rlist.Set(nil)
...@@ -543,8 +533,7 @@ func typecheck1(n *Node, top int) (res *Node) { ...@@ -543,8 +533,7 @@ func typecheck1(n *Node, top int) (res *Node) {
} }
if l.Op == OTYPE { if l.Op == OTYPE {
ok |= Etype ok |= Etype
n.Op = OTYPE setTypeNode(n, types.NewPtr(l.Type))
n.Type = types.NewPtr(l.Type)
// Ensure l.Type gets dowidth'd for the backend. Issue 20174. // Ensure l.Type gets dowidth'd for the backend. Issue 20174.
// Don't checkwidth [...] arrays, though, since they // Don't checkwidth [...] arrays, though, since they
// will be replaced by concrete-sized arrays. Issue 20333. // will be replaced by concrete-sized arrays. Issue 20333.
...@@ -3683,8 +3672,7 @@ func typecheckdef(n *Node) { ...@@ -3683,8 +3672,7 @@ func typecheckdef(n *Node) {
defercheckwidth() defercheckwidth()
} }
n.SetWalkdef(1) n.SetWalkdef(1)
n.Type = types.New(TFORW) setTypeNode(n, types.New(TFORW))
n.Type.Nod = asTypesNode(n)
n.Type.Sym = n.Sym // TODO(gri) this also happens in typecheckdeftype(n) - where should it happen? n.Type.Sym = n.Sym // TODO(gri) this also happens in typecheckdeftype(n) - where should it happen?
nerrors0 := nerrors nerrors0 := nerrors
typecheckdeftype(n) typecheckdeftype(n)
...@@ -3951,3 +3939,10 @@ func deadcodeexpr(n *Node) *Node { ...@@ -3951,3 +3939,10 @@ func deadcodeexpr(n *Node) *Node {
} }
return n return n
} }
// setTypeNode sets n to an OTYPE node representing t.
func setTypeNode(n *Node, t *types.Type) {
n.Op = OTYPE
n.Type = t
n.Type.Nod = asTypesNode(n)
}
...@@ -6,10 +6,10 @@ ...@@ -6,10 +6,10 @@
package p package p
type I1 = interface { type I1 = interface { // ERROR "invalid recursive type"
I2 I2
} }
type I2 interface { // ERROR "invalid recursive type" type I2 interface {
I1 I1
} }
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