]> Cypherpunks repositories - gostls13.git/commitdiff
cmd/compile: pop instantiations of local types when leaving scope
authorDan Scales <danscales@google.com>
Thu, 16 Dec 2021 02:42:00 +0000 (18:42 -0800)
committerDan Scales <danscales@google.com>
Tue, 4 Jan 2022 22:05:15 +0000 (22:05 +0000)
Since we use existing instantiations from the symbol table when possible
(to make sure each instantiation is unique), we need to pop
instantiations of local types when leaving the containing scope.
g.stmts() now pushes and pops scope, and we do a Pushdcl() in g.typ0()
when creating an instantiation of a local type.

Non-instantiated local types (generic or not) are translated directly
from types2, so they don't need to be pushed/popped. We don't export
function bodies with local types, so there is no issue during import.

We still don't support local types in generic functions/methods.

Fixes #50177

Change-Id: If2d2fe71aec003d13f0338565c7a0da2c9580a14
Reviewed-on: https://go-review.googlesource.com/c/go/+/372654
Reviewed-by: Keith Randall <khr@golang.org>
Trust: Dan Scales <danscales@google.com>
Run-TryBot: Dan Scales <danscales@google.com>
TryBot-Result: Gopher Robot <gobot@golang.org>

src/cmd/compile/internal/noder/stmt.go
src/cmd/compile/internal/noder/types.go
test/typeparam/issue50177.go [new file with mode: 0644]

index 1e996b95c48ffa506d61573f7bde134cc80be639..a349a7ef10e03f3b6010bfb396f065d74261690a 100644 (file)
@@ -13,8 +13,10 @@ import (
        "cmd/internal/src"
 )
 
+// stmts creates nodes for a slice of statements that form a scope.
 func (g *irgen) stmts(stmts []syntax.Stmt) []ir.Node {
        var nodes []ir.Node
+       types.Markdcl()
        for _, stmt := range stmts {
                switch s := g.stmt(stmt).(type) {
                case nil: // EmptyStmt
@@ -24,6 +26,7 @@ func (g *irgen) stmts(stmts []syntax.Stmt) []ir.Node {
                        nodes = append(nodes, s)
                }
        }
+       types.Popdcl()
        return nodes
 }
 
index 4f6d8287201d4b685c3a8822b57e6b6f6b3f7ab8..ed816b4955a9419f2a3a4de46cc8dc45e10b9e1c 100644 (file)
@@ -123,7 +123,14 @@ func (g *irgen) typ0(typ types2.Type) *types.Type {
                        // Make sure the base generic type exists in type1 (it may
                        // not yet if we are referecing an imported generic type, as
                        // opposed to a generic type declared in this package).
-                       _ = g.obj(typ.Origin().Obj())
+                       base := g.obj(typ.Origin().Obj())
+                       if base.Class == ir.PAUTO {
+                               // If the base type is a local type, we want to pop
+                               // this instantiated type symbol/definition when we
+                               // leave the containing block, so we don't use it
+                               // incorrectly later.
+                               types.Pushdcl(s)
+                       }
 
                        // Create a forwarding type first and put it in the g.typs
                        // map, in order to deal with recursive generic types
diff --git a/test/typeparam/issue50177.go b/test/typeparam/issue50177.go
new file mode 100644 (file)
index 0000000..5fd62ad
--- /dev/null
@@ -0,0 +1,101 @@
+// compile -G=3
+
+// Copyright 2021 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 main
+
+import "fmt"
+
+type Fn[T any] func(T)
+type FnErr[T any] func(T) error
+
+// Test that local generic types across functions don't conflict, and they also don't
+// conflict with local non-generic types and local variables.
+func caller0() {
+       type X[T any] struct {
+               fn Fn[int]
+       }
+
+       x := X[int]{func(v int) { fmt.Println(v) }}
+       x.fn(0)
+}
+
+func caller1(val int) {
+       type X[T any] struct {
+               fn FnErr[int]
+       }
+
+       x := X[int]{func(v int) error { fmt.Println(v); return nil }}
+       x.fn(0)
+}
+
+func caller1a(val int) {
+       type X struct {
+               fn func(float64) error
+       }
+
+       x := X{func(v float64) error { fmt.Println(v); return nil }}
+       x.fn(float64(3.2))
+}
+
+func caller1b(val int) {
+       type Y struct {
+               fn func(float64) error
+       }
+
+       X := Y{func(v float64) error { fmt.Println(v); return nil }}
+       X.fn(float64(3.2))
+}
+
+// Test that local generic types within different if clauses don't conflict.
+func caller2(val int) {
+       if val > 2 {
+               type X[T any] struct {
+                       fn func(v int) float64
+               }
+
+               x := X[int]{func(v int) float64 { fmt.Println(v); return 1.5 }}
+               x.fn(0)
+       } else {
+               type X[T any] struct {
+                       fn func(v int) int
+               }
+               x := X[int]{func(v int) int { fmt.Println(v); return 5 }}
+               x.fn(0)
+       }
+}
+
+// Test that local generic types within different cases don't conflict with each
+// other or with local non-generic types or local variables.
+func caller3(val int) {
+       switch val {
+       case 0:
+               type X[T any] struct {
+                       fn func(v int) float64
+               }
+
+               x := X[int]{func(v int) float64 { fmt.Println(v); return 1.5 }}
+               x.fn(0)
+       case 1:
+               type X[T any] struct {
+                       fn func(v int) int
+               }
+               x := X[int]{func(v int) int { fmt.Println(v); return 5 }}
+               x.fn(0)
+       case 2:
+               type X struct {
+                       fn func(v int) bool
+               }
+               x := X{func(v int) bool { fmt.Println(v); return false }}
+               x.fn(0)
+       case 3:
+               type Y struct {
+                       fn func(v int) bool
+               }
+               X := Y{func(v int) bool { fmt.Println(v); return false }}
+               X.fn(0)
+
+       }
+}