]> Cypherpunks repositories - gostls13.git/commitdiff
[dev.typeparams] cmd/compile: allow types with the same underlying type to have the...
authorKeith Randall <khr@golang.org>
Fri, 30 Jul 2021 16:06:38 +0000 (09:06 -0700)
committerKeith Randall <khr@golang.org>
Fri, 30 Jul 2021 21:01:32 +0000 (21:01 +0000)
First baby step to sharing the underlying implementation among several types.

Change-Id: I6a156176d2b7f0131a87285a03b881ce380c26ed
Reviewed-on: https://go-review.googlesource.com/c/go/+/338610
Trust: Keith Randall <khr@golang.org>
Trust: Dan Scales <danscales@google.com>
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Go Bot <gobot@golang.org>
Reviewed-by: Dan Scales <danscales@google.com>
src/cmd/compile/internal/reflectdata/reflect.go
src/cmd/compile/internal/typecheck/subr.go
test/typeparam/shape1.go [new file with mode: 0644]
test/typeparam/shape1.out [new file with mode: 0644]

index 3bf248a7ade2a8e90b46129d75b7596fb6c83c85..dca8de74f303ff508164a4a60ab0c1beeef4861c 100644 (file)
@@ -1920,7 +1920,7 @@ func methodWrapper(rcvr *types.Type, method *types.Field, forItab bool) *obj.LSy
                        // Target method uses shaped names.
                        targs2 := make([]*types.Type, len(targs))
                        for i, t := range targs {
-                               targs2[i] = typecheck.Shaped[t]
+                               targs2[i] = typecheck.Shapify(t)
                        }
                        targs = targs2
 
index 53221bc1cdfef3f3b71e4a6fc9bbc42bb4216938..e2f0a57e7133393984d11d4295bb13984e4abd50 100644 (file)
@@ -1348,32 +1348,26 @@ func Shapify(t *types.Type) *types.Type {
        if t.IsShape() {
                return t // TODO: is this right?
        }
-       if s := Shaped[t]; s != nil {
+       // Map all types with the same underlying type to the same shape.
+       u := t.Underlying()
+
+       if s := shaped[u]; s != nil {
                return s //TODO: keep?
        }
 
-       // For now, there is a 1-1 mapping between regular types and shape types.
        sym := Lookup(fmt.Sprintf(".shape%d", snum))
        snum++
-       name := ir.NewDeclNameAt(t.Pos(), ir.OTYPE, sym)
+       name := ir.NewDeclNameAt(u.Pos(), ir.OTYPE, sym)
        s := types.NewNamed(name)
-       s.SetUnderlying(t.Underlying())
+       s.SetUnderlying(u)
        s.SetIsShape(true)
        name.SetType(s)
        name.SetTypecheck(1)
        // TODO: add methods to s that the bound has?
-       Shaped[t] = s
+       shaped[u] = s
        return s
 }
 
 var snum int
 
-var Shaped = map[*types.Type]*types.Type{}
-
-func ShapifyList(targs []*types.Type) []*types.Type {
-       r := make([]*types.Type, len(targs))
-       for i, t := range targs {
-               r[i] = Shapify(t)
-       }
-       return r
-}
+var shaped = map[*types.Type]*types.Type{}
diff --git a/test/typeparam/shape1.go b/test/typeparam/shape1.go
new file mode 100644 (file)
index 0000000..3c9e71e
--- /dev/null
@@ -0,0 +1,33 @@
+// run -gcflags=-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
+
+type I interface {
+       foo() int
+}
+
+// There should be a single instantiation of f in this program.
+func f[T I](x T) int {
+       return x.foo()
+}
+
+type squarer int
+
+func (x squarer) foo() int {
+       return int(x*x)
+}
+
+type doubler int
+
+func (x doubler) foo() int {
+       return int(2*x)
+}
+
+func main() {
+       println(f(squarer(5)))
+       println(f(doubler(5)))
+}
diff --git a/test/typeparam/shape1.out b/test/typeparam/shape1.out
new file mode 100644 (file)
index 0000000..28391fd
--- /dev/null
@@ -0,0 +1,2 @@
+25
+10