mirror of
https://github.com/golang/go.git
synced 2025-05-05 23:53:05 +00:00
cmd/compile: distinguish bound calls/field access in getInstInfo
Given we have support for field access to type params with a single structural type, we need to distinguish between methods calls and field access when we have an OXDOT node on an expression which is a typeparam (or correspondingly a shape). We were missing checks in getInstInfo, which figures out the dictionary format, which then caused problems when we generate the dictionaries. We don't need/want dictionary entries for field access, only for bound method calls. Added a new function isBoundMethod() to distinguish OXDOT nodes which are bound calls vs. field accesses on a shape. Removed isShapeDeref() - we can't have field access or method call on a pointer to variable of type param type. Fixes #50690 Change-Id: Id692f65e6f427f28cd2cfe474dd30e53c71877a7 Reviewed-on: https://go-review.googlesource.com/c/go/+/379674 Trust: Dan Scales <danscales@google.com> Reviewed-by: Keith Randall <khr@golang.org>
This commit is contained in:
parent
f9df4ea0c9
commit
f88c3b9f4d
@ -1036,13 +1036,13 @@ func (subst *subster) node(n ir.Node) ir.Node {
|
|||||||
}
|
}
|
||||||
|
|
||||||
case ir.OXDOT:
|
case ir.OXDOT:
|
||||||
// Finish the transformation of an OXDOT, unless this was a
|
// Finish the transformation of an OXDOT, unless this is
|
||||||
// bound call (a direct call on a type param). A bound call
|
// bound call or field access on a type param. A bound call
|
||||||
// will be transformed during the dictPass. Otherwise, m
|
// or field access on a type param will be transformed during
|
||||||
// will be transformed to an OMETHVALUE node. It will be
|
// the dictPass. Otherwise, m will be transformed to an
|
||||||
// transformed to an ODOTMETH or ODOTINTER node if we find in
|
// OMETHVALUE node. It will be transformed to an ODOTMETH or
|
||||||
// the OCALL case below that the method value is actually
|
// ODOTINTER node if we find in the OCALL case below that the
|
||||||
// called.
|
// method value is actually called.
|
||||||
mse := m.(*ir.SelectorExpr)
|
mse := m.(*ir.SelectorExpr)
|
||||||
if src := mse.X.Type(); !src.IsShape() {
|
if src := mse.X.Type(); !src.IsShape() {
|
||||||
transformDot(mse, false)
|
transformDot(mse, false)
|
||||||
@ -1101,10 +1101,11 @@ func (subst *subster) node(n ir.Node) ir.Node {
|
|||||||
transformEarlyCall(call)
|
transformEarlyCall(call)
|
||||||
|
|
||||||
case ir.OXDOT:
|
case ir.OXDOT:
|
||||||
// This is the case of a bound call on a typeparam,
|
// This is the case of a bound call or a field access
|
||||||
// which will be handled in the dictPass.
|
// on a typeparam, which will be handled in the
|
||||||
// As with OFUNCINST, we must transform the arguments of the call now,
|
// dictPass. As with OFUNCINST, we must transform the
|
||||||
// so any needed CONVIFACE nodes are exposed.
|
// arguments of the call now, so any needed CONVIFACE
|
||||||
|
// nodes are exposed.
|
||||||
transformEarlyCall(call)
|
transformEarlyCall(call)
|
||||||
|
|
||||||
case ir.ODOTTYPE, ir.ODOTTYPE2:
|
case ir.ODOTTYPE, ir.ODOTTYPE2:
|
||||||
@ -1228,13 +1229,13 @@ func (g *genInst) dictPass(info *instInfo) {
|
|||||||
// No need for transformDot - buildClosure2 has already
|
// No need for transformDot - buildClosure2 has already
|
||||||
// transformed to OCALLINTER/ODOTINTER.
|
// transformed to OCALLINTER/ODOTINTER.
|
||||||
} else {
|
} else {
|
||||||
dst := info.dictInfo.shapeToBound[m.(*ir.SelectorExpr).X.Type()]
|
|
||||||
// If we can't find the selected method in the
|
// If we can't find the selected method in the
|
||||||
// AllMethods of the bound, then this must be an access
|
// AllMethods of the bound, then this must be an access
|
||||||
// to a field of a structural type. If so, we skip the
|
// to a field of a structural type. If so, we skip the
|
||||||
// dictionary lookups - transformDot() will convert to
|
// dictionary lookups - transformDot() will convert to
|
||||||
// the desired direct field access.
|
// the desired direct field access.
|
||||||
if typecheck.Lookdot1(mse, mse.Sel, dst, dst.AllMethods(), 1) != nil {
|
if isBoundMethod(info.dictInfo, mse) {
|
||||||
|
dst := info.dictInfo.shapeToBound[mse.X.Type()]
|
||||||
// Implement x.M as a conversion-to-bound-interface
|
// Implement x.M as a conversion-to-bound-interface
|
||||||
// 1) convert x to the bound interface
|
// 1) convert x to the bound interface
|
||||||
// 2) call M on that interface
|
// 2) call M on that interface
|
||||||
@ -1873,11 +1874,15 @@ func (g *genInst) getInstInfo(st *ir.Func, shapes []*types.Type, instInfo *instI
|
|||||||
info.subDictCalls = append(info.subDictCalls, subDictInfo{callNode: n, savedXNode: ce.X})
|
info.subDictCalls = append(info.subDictCalls, subDictInfo{callNode: n, savedXNode: ce.X})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ce.X.Op() == ir.OXDOT &&
|
// Note: this XDOT code is not actually needed as long as we
|
||||||
isShapeDeref(ce.X.(*ir.SelectorExpr).X.Type()) {
|
// continue to disable type parameters on RHS of type
|
||||||
|
// declarations (#45639).
|
||||||
|
if ce.X.Op() == ir.OXDOT {
|
||||||
callMap[ce.X] = true
|
callMap[ce.X] = true
|
||||||
infoPrint(" Optional subdictionary at generic bound call: %v\n", n)
|
if isBoundMethod(info, ce.X.(*ir.SelectorExpr)) {
|
||||||
info.subDictCalls = append(info.subDictCalls, subDictInfo{callNode: n, savedXNode: nil})
|
infoPrint(" Optional subdictionary at generic bound call: %v\n", n)
|
||||||
|
info.subDictCalls = append(info.subDictCalls, subDictInfo{callNode: n, savedXNode: nil})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
case ir.OCALLMETH:
|
case ir.OCALLMETH:
|
||||||
ce := n.(*ir.CallExpr)
|
ce := n.(*ir.CallExpr)
|
||||||
@ -1900,7 +1905,8 @@ func (g *genInst) getInstInfo(st *ir.Func, shapes []*types.Type, instInfo *instI
|
|||||||
info.itabConvs = append(info.itabConvs, n)
|
info.itabConvs = append(info.itabConvs, n)
|
||||||
}
|
}
|
||||||
case ir.OXDOT:
|
case ir.OXDOT:
|
||||||
if n.(*ir.SelectorExpr).X.Type().IsShape() {
|
se := n.(*ir.SelectorExpr)
|
||||||
|
if isBoundMethod(info, se) {
|
||||||
infoPrint(" Itab for bound call: %v\n", n)
|
infoPrint(" Itab for bound call: %v\n", n)
|
||||||
info.itabConvs = append(info.itabConvs, n)
|
info.itabConvs = append(info.itabConvs, n)
|
||||||
}
|
}
|
||||||
@ -1956,11 +1962,13 @@ func (g *genInst) getInstInfo(st *ir.Func, shapes []*types.Type, instInfo *instI
|
|||||||
info.dictLen = len(info.shapeParams) + len(info.derivedTypes) + len(info.subDictCalls) + len(info.itabConvs)
|
info.dictLen = len(info.shapeParams) + len(info.derivedTypes) + len(info.subDictCalls) + len(info.itabConvs)
|
||||||
}
|
}
|
||||||
|
|
||||||
// isShapeDeref returns true if t is either a shape or a pointer to a shape. (We
|
// isBoundMethod returns true if the selection indicated by se is a bound method of
|
||||||
// can't just use deref(t).IsShape(), since a shape type is a complex type and may
|
// se.X. se.X must be a shape type (i.e. substituted directly from a type param). If
|
||||||
// have a pointer as part of its shape.)
|
// isBoundMethod returns false, then the selection must be a field access of a
|
||||||
func isShapeDeref(t *types.Type) bool {
|
// structural type.
|
||||||
return t.IsShape() || t.IsPtr() && t.Elem().IsShape()
|
func isBoundMethod(info *dictInfo, se *ir.SelectorExpr) bool {
|
||||||
|
bound := info.shapeToBound[se.X.Type()]
|
||||||
|
return typecheck.Lookdot1(se, se.Sel, bound, bound.AllMethods(), 1) != nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// addType adds t to info.derivedTypes if it is parameterized type (which is not
|
// addType adds t to info.derivedTypes if it is parameterized type (which is not
|
||||||
|
62
test/typeparam/issue50690a.go
Normal file
62
test/typeparam/issue50690a.go
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
// run -gcflags=-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 main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Numeric expresses a type constraint satisfied by any numeric type.
|
||||||
|
type Numeric interface {
|
||||||
|
~uint | ~uint8 | ~uint16 | ~uint32 | ~uint64 |
|
||||||
|
~int | ~int8 | ~int16 | ~int32 | ~int64 |
|
||||||
|
~float32 | ~float64 |
|
||||||
|
~complex64 | ~complex128
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sum returns the sum of the provided arguments.
|
||||||
|
func Sum[T Numeric](args ...T) T {
|
||||||
|
var sum T
|
||||||
|
for i := 0; i < len(args); i++ {
|
||||||
|
sum += args[i]
|
||||||
|
}
|
||||||
|
return sum
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ledger is an identifiable, financial record.
|
||||||
|
type Ledger[T ~string, K Numeric] struct {
|
||||||
|
|
||||||
|
// ID identifies the ledger.
|
||||||
|
ID T
|
||||||
|
|
||||||
|
// Amounts is a list of monies associated with this ledger.
|
||||||
|
Amounts []K
|
||||||
|
|
||||||
|
// SumFn is a function that can be used to sum the amounts
|
||||||
|
// in this ledger.
|
||||||
|
SumFn func(...K) K
|
||||||
|
}
|
||||||
|
|
||||||
|
func PrintLedger[
|
||||||
|
T ~string,
|
||||||
|
K Numeric,
|
||||||
|
L ~struct {
|
||||||
|
ID T
|
||||||
|
Amounts []K
|
||||||
|
SumFn func(...K) K
|
||||||
|
},
|
||||||
|
](l L) {
|
||||||
|
fmt.Printf("%s has a sum of %v\n", l.ID, l.SumFn(l.Amounts...))
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
PrintLedger(Ledger[string, int]{
|
||||||
|
ID: "fake",
|
||||||
|
Amounts: []int{1, 2, 3},
|
||||||
|
SumFn: Sum[int],
|
||||||
|
})
|
||||||
|
}
|
1
test/typeparam/issue50690a.out
Normal file
1
test/typeparam/issue50690a.out
Normal file
@ -0,0 +1 @@
|
|||||||
|
fake has a sum of 6
|
41
test/typeparam/issue50690b.go
Normal file
41
test/typeparam/issue50690b.go
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
// run -gcflags=-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 main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Printer[T ~string] struct {
|
||||||
|
PrintFn func(T)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Print[T ~string](s T) {
|
||||||
|
fmt.Println(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func PrintWithPrinter[T ~string, S ~struct {
|
||||||
|
ID T
|
||||||
|
PrintFn func(T)
|
||||||
|
}](message T, obj S) {
|
||||||
|
obj.PrintFn(message)
|
||||||
|
}
|
||||||
|
|
||||||
|
type PrintShop[T ~string] struct {
|
||||||
|
ID T
|
||||||
|
PrintFn func(T)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
PrintWithPrinter(
|
||||||
|
"Hello, world.",
|
||||||
|
PrintShop[string]{
|
||||||
|
ID: "fake",
|
||||||
|
PrintFn: Print[string],
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
1
test/typeparam/issue50690b.out
Normal file
1
test/typeparam/issue50690b.out
Normal file
@ -0,0 +1 @@
|
|||||||
|
Hello, world.
|
36
test/typeparam/issue50690c.go
Normal file
36
test/typeparam/issue50690c.go
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
// run -gcflags=-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 main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Printer[T ~string] struct {
|
||||||
|
PrintFn func(T)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Print[T ~string](s T) {
|
||||||
|
fmt.Println(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func PrintWithPrinter[T ~string, S struct {
|
||||||
|
ID T
|
||||||
|
PrintFn func(T)
|
||||||
|
}](message T, obj S) {
|
||||||
|
obj.PrintFn(message)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
PrintWithPrinter(
|
||||||
|
"Hello, world.",
|
||||||
|
struct {
|
||||||
|
ID string
|
||||||
|
PrintFn func(string)
|
||||||
|
}{ID: "fake", PrintFn: Print[string]},
|
||||||
|
)
|
||||||
|
}
|
1
test/typeparam/issue50690c.out
Normal file
1
test/typeparam/issue50690c.out
Normal file
@ -0,0 +1 @@
|
|||||||
|
Hello, world.
|
Loading…
x
Reference in New Issue
Block a user