1
0
mirror of https://github.com/golang/go synced 2024-11-26 03:37:57 -07:00

cmd/compile/internal/inline: revise closure inl position fix

This patch revises the fix for issue 46234, fixing a bug that was
accidentally introduced by CL 320913. When inlining a chunk of code
with a closure expression, we want to avoid updating the source
positions in the function being closed over, but we do want to update
the position for the ClosureExpr itself (since it is part of the
function we are inlining). CL 320913 unintentionally did away with the
closure expr source position update; here we restore it again.

Updates #46234.
Fixes #49171.

Change-Id: Iaa51bc498e374b9e5a46fa0acd7db520edbbbfca
Reviewed-on: https://go-review.googlesource.com/c/go/+/366494
Trust: Than McIntosh <thanm@google.com>
Trust: Dan Scales <danscales@google.com>
Reviewed-by: Dan Scales <danscales@google.com>
This commit is contained in:
Than McIntosh 2021-11-23 08:19:45 -05:00
parent 47db3bb443
commit 465b402808
3 changed files with 16 additions and 11 deletions

View File

@ -1108,11 +1108,15 @@ func (subst *inlsubst) clovar(n *ir.Name) *ir.Name {
// closure does the necessary substitions for a ClosureExpr n and returns the new // closure does the necessary substitions for a ClosureExpr n and returns the new
// closure node. // closure node.
func (subst *inlsubst) closure(n *ir.ClosureExpr) ir.Node { func (subst *inlsubst) closure(n *ir.ClosureExpr) ir.Node {
// Prior to the subst edit, set a flag in the inlsubst to // Prior to the subst edit, set a flag in the inlsubst to indicate
// indicated that we don't want to update the source positions in // that we don't want to update the source positions in the new
// the new closure. If we do this, it will appear that the closure // closure function. If we do this, it will appear that the
// itself has things inlined into it, which is not the case. See // closure itself has things inlined into it, which is not the
// issue #46234 for more details. // case. See issue #46234 for more details. At the same time, we
// do want to update the position in the new ClosureExpr (which is
// part of the function we're working on). See #49171 for an
// example of what happens if we miss that update.
newClosurePos := subst.updatedPos(n.Pos())
defer func(prev bool) { subst.noPosUpdate = prev }(subst.noPosUpdate) defer func(prev bool) { subst.noPosUpdate = prev }(subst.noPosUpdate)
subst.noPosUpdate = true subst.noPosUpdate = true
@ -1175,6 +1179,7 @@ func (subst *inlsubst) closure(n *ir.ClosureExpr) ir.Node {
// Actually create the named function for the closure, now that // Actually create the named function for the closure, now that
// the closure is inlined in a specific function. // the closure is inlined in a specific function.
newclo := newfn.OClosure newclo := newfn.OClosure
newclo.SetPos(newClosurePos)
newclo.SetInit(subst.list(n.Init())) newclo.SetInit(subst.list(n.Init()))
return typecheck.Expr(newclo) return typecheck.Expr(newclo)
} }

View File

@ -94,10 +94,10 @@ func main() {
return x + 2 return x + 2
} }
y, sink = func() (func(int) int, int) { // ERROR "can inline main.func12" y, sink = func() (func(int) int, int) { // ERROR "can inline main.func12"
return func(x int) int { // ERROR "func literal does not escape" "can inline main.func12" return func(x int) int { // ERROR "can inline main.func12"
return x + 1 return x + 1
}, 42 }, 42
}() // ERROR "inlining call to main.func12" }() // ERROR "func literal does not escape" "inlining call to main.func12"
if y(40) != 41 { if y(40) != 41 {
ppanic("y(40) != 41") ppanic("y(40) != 41")
} }
@ -109,10 +109,10 @@ func main() {
return x + 2 return x + 2
} }
y, sink = func() (func(int) int, int) { // ERROR "can inline main.func13.2" y, sink = func() (func(int) int, int) { // ERROR "can inline main.func13.2"
return func(x int) int { // ERROR "func literal does not escape" "can inline main.func13.2" return func(x int) int { // ERROR "can inline main.func13.2"
return x + 1 return x + 1
}, 42 }, 42
}() // ERROR "inlining call to main.func13.2" }() // ERROR "func literal does not escape" "inlining call to main.func13.2"
if y(40) != 41 { if y(40) != 41 {
ppanic("y(40) != 41") ppanic("y(40) != 41")
} }

View File

@ -92,9 +92,9 @@ func o() int {
foo := func() int { return 1 } // ERROR "can inline o.func1" "func literal does not escape" foo := func() int { return 1 } // ERROR "can inline o.func1" "func literal does not escape"
func(x int) { // ERROR "can inline o.func2" func(x int) { // ERROR "can inline o.func2"
if x > 10 { if x > 10 {
foo = func() int { return 2 } // ERROR "func literal does not escape" "can inline o.func2" foo = func() int { return 2 } // ERROR "can inline o.func2"
} }
}(11) // ERROR "inlining call to o.func2" }(11) // ERROR "func literal does not escape" "inlining call to o.func2"
return foo() return foo()
} }