aboutsummaryrefslogtreecommitdiff
path: root/src/cmd/compile/internal/noder
diff options
context:
space:
mode:
authorKeith Randall <khr@golang.org>2021-09-10 15:24:16 -0700
committerKeith Randall <khr@golang.org>2021-09-27 20:42:34 +0000
commit301f6c8019bea813b039c3d376a6464a0e117dce (patch)
tree9f357af4032039785de2e13bfb63a71a01aad4cb /src/cmd/compile/internal/noder
parentdfd875d015fb67671a2374c229e2159388d37693 (diff)
downloadgo-301f6c8019bea813b039c3d376a6464a0e117dce.tar.xz
cmd/compile: keep methods on generic types from being deadcode eliminated
We currently make dictionaries contain a relocation pointing to methods that generic code might use, so that those methods are not deadcode eliminated. However, with inlining we can end up not using the dictionary, making the reference from the dictionary to the method no longer keep the method alive. Fix this by keeping the dictionary alive at generic interface call sites. It's a bit of overkill, as we only need to keep the dictionary statically alive. We don't actually need it dynamically alive, which is what KeepAlive does. But it works. It ends up generating a LEAQ + stack spill that aren't necessary, but that's pretty low overhead. To make this work, I needed to stop generating methods on shape types. We should do this anyway, as we shouldn't ever need them. But currently we do use them! issue44688.go has a test that only works because it calls a method on a shape type. I've disabled that test for now, will work on it in a subsequent CL. Fixes #48047 Change-Id: I78968868d6486c1745f51b8b43be0898931432a2 Reviewed-on: https://go-review.googlesource.com/c/go/+/349169 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> Reviewed-by: Than McIntosh <thanm@google.com>
Diffstat (limited to 'src/cmd/compile/internal/noder')
-rw-r--r--src/cmd/compile/internal/noder/helpers.go2
-rw-r--r--src/cmd/compile/internal/noder/stencil.go22
-rw-r--r--src/cmd/compile/internal/noder/transform.go15
3 files changed, 26 insertions, 13 deletions
diff --git a/src/cmd/compile/internal/noder/helpers.go b/src/cmd/compile/internal/noder/helpers.go
index e8a1540307..83830a5d31 100644
--- a/src/cmd/compile/internal/noder/helpers.go
+++ b/src/cmd/compile/internal/noder/helpers.go
@@ -187,7 +187,7 @@ func Call(pos src.XPos, typ *types.Type, fun ir.Node, args []ir.Node, dots bool)
// If no type params, do the normal call transformations. This
// will convert OCALL to OCALLFUNC.
typed(typ, n)
- transformCall(n)
+ transformCall(n, nil)
return n
}
diff --git a/src/cmd/compile/internal/noder/stencil.go b/src/cmd/compile/internal/noder/stencil.go
index 772fcca46a..e49702c04c 100644
--- a/src/cmd/compile/internal/noder/stencil.go
+++ b/src/cmd/compile/internal/noder/stencil.go
@@ -124,7 +124,7 @@ func (g *irgen) stencil() {
// it before installing the instantiation, so we are
// checking against non-shape param types in
// typecheckaste.
- transformCall(call)
+ transformCall(call, nil)
// Replace the OFUNCINST with a direct reference to the
// new stenciled function
@@ -162,7 +162,7 @@ func (g *irgen) stencil() {
// Transform the Call now, which changes OCALL
// to OCALLFUNC and does typecheckaste/assignconvfn.
- transformCall(call)
+ transformCall(call, nil)
st := g.getInstantiation(gf, targs, true).fun
dictValue, usingSubdict := g.getDictOrSubdict(declInfo, n, gf, targs, true)
@@ -258,7 +258,7 @@ func (g *irgen) stencil() {
assert(l == len(g.instInfoMap))
}
-// buildClosure makes a closure to implement x, a OFUNCINST or OMETHEXPR
+// buildClosure makes a closure to implement x, a OFUNCINST or OMETHEXPR/OMETHVALUE
// of generic type. outer is the containing function (or nil if closure is
// in a global assignment instead of a function).
func (g *irgen) buildClosure(outer *ir.Func, x ir.Node) ir.Node {
@@ -1053,14 +1053,14 @@ func (subst *subster) node(n ir.Node) ir.Node {
// transform the call.
call.X.(*ir.SelectorExpr).SetOp(ir.OXDOT)
transformDot(call.X.(*ir.SelectorExpr), true)
- transformCall(call)
+ transformCall(call, subst.info.dictParam)
case ir.ODOT, ir.ODOTPTR:
// An OXDOT for a generic receiver was resolved to
// an access to a field which has a function
// value. Transform the call to that function, now
// that the OXDOT was resolved.
- transformCall(call)
+ transformCall(call, subst.info.dictParam)
case ir.ONAME:
name := call.X.Name()
@@ -1077,24 +1077,24 @@ func (subst *subster) node(n ir.Node) ir.Node {
// This is the case of a function value that was a
// type parameter (implied to be a function via a
// structural constraint) which is now resolved.
- transformCall(call)
+ transformCall(call, subst.info.dictParam)
}
case ir.OCLOSURE:
- transformCall(call)
+ transformCall(call, subst.info.dictParam)
case ir.ODEREF, ir.OINDEX, ir.OINDEXMAP, ir.ORECV:
// Transform a call that was delayed because of the
// use of typeparam inside an expression that required
// a pointer dereference, array indexing, map indexing,
// or channel receive to compute function value.
- transformCall(call)
+ transformCall(call, subst.info.dictParam)
case ir.OCALL, ir.OCALLFUNC, ir.OCALLMETH, ir.OCALLINTER:
- transformCall(call)
+ transformCall(call, subst.info.dictParam)
case ir.OCONVNOP:
- transformCall(call)
+ transformCall(call, subst.info.dictParam)
case ir.OFUNCINST:
// A call with an OFUNCINST will get transformed
@@ -1239,7 +1239,7 @@ func (g *irgen) dictPass(info *instInfo) {
m.(*ir.CallExpr).X.(*ir.SelectorExpr).SetOp(ir.OXDOT)
transformDot(m.(*ir.CallExpr).X.(*ir.SelectorExpr), true)
}
- transformCall(m.(*ir.CallExpr))
+ transformCall(m.(*ir.CallExpr), info.dictParam)
}
case ir.OCONVIFACE:
diff --git a/src/cmd/compile/internal/noder/transform.go b/src/cmd/compile/internal/noder/transform.go
index f7115904fe..9076db2822 100644
--- a/src/cmd/compile/internal/noder/transform.go
+++ b/src/cmd/compile/internal/noder/transform.go
@@ -132,7 +132,9 @@ func transformConvCall(n *ir.CallExpr) ir.Node {
// transformCall transforms a normal function/method call. Corresponds to last half
// (non-conversion, non-builtin part) of typecheck.tcCall. This code should work even
// in the case of OCALL/OFUNCINST.
-func transformCall(n *ir.CallExpr) {
+// The dict parameter is used for OCALLINTER nodes to ensure that the called method
+// is retained by the linker.
+func transformCall(n *ir.CallExpr, dict *ir.Name) {
// n.Type() can be nil for calls with no return value
assert(n.Typecheck() == 1)
transformArgs(n)
@@ -142,6 +144,17 @@ func transformCall(n *ir.CallExpr) {
switch l.Op() {
case ir.ODOTINTER:
n.SetOp(ir.OCALLINTER)
+ if n.X.(*ir.SelectorExpr).X.Type().HasShape() {
+ if dict == nil {
+ base.Fatalf("calls on shape interfaces need a dictionary reference")
+ }
+ dict.SetAddrtaken(true)
+ // KeepAlive isn't exactly the right thing here, as we only
+ // need to keep the dictionary live in the linker-deadcode
+ // sense, not the at-runtime sense. But the at-runtime sense
+ // is stronger, so it works. See issue 48047.
+ n.KeepAlive = append(n.KeepAlive, dict)
+ }
case ir.ODOTMETH:
l := l.(*ir.SelectorExpr)