-
Notifications
You must be signed in to change notification settings - Fork 17.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
cmd/compile: escaping closure incorrectly allocated on the stack #17318
Comments
(Tip to reproduce - get godoc and install it in the distribution's bin directory.) I can reproduce with 1.7.1 but not with tip (darwin/amd64). So maybe this has already been fixed. Not sure what fixed it; might need to do some binary search. Possibly a candidate for 1.7.2 if it turns out to be a real bug. |
I lied, I can reproduce at tip with GOGC=1. |
Can you clarify what you mean? I'm not sure if you're referring to the test case or if there's some other mechanism of triggering this. |
Running "go test" in the go-closure-issue directory gives me a "godoc not found" error. The test is exec'ing godoc as part of the test. |
Ah, my bad! Didn't even think about that. But that makes sense. Basically the test just needs a few big files to rsync, |
/cc @aclements @RLH |
|
I can reproduce this on linux/amd64 as well. It happens with tip, 1.7 and 1.6, but apparently not 1.5. In fact, it happens almost every time with 1.6 (99 / 100) so it might even be easier to start debugging with that. |
This looks like a heap->stack pointer. The original stack-allocated closure is from github.com/havoc-io/go-closure-issue/server.go:50. It is passed in the call at line 59 to CreateDelta. CreateDelta (in /bitbucket.org/kardianos/rsync/rsync.go) makes the heap closure at line 247. (I don't understand why this closure is heap allocated. That's a separate issue I think.) I suspect the bug is that we need to make sure that if a pointer is captured in a heap-allocated closure, then its referents must be treated as escaping. |
Fails for me, too. |
Fails for me, go version go1.7.1 windows/amd64. |
I have a tiny reproducer. I think I also see the reason for the spurious escape of |
CL https://golang.org/cl/30693 mentions this issue. |
CL https://golang.org/cl/31290 mentions this issue. |
…od" to fixed point In some cases the members of the root set from which flood runs themselves escape, without their referents being also tagged as escaping. Fix this by reflooding from those roots whose escape increases, and also enhance the "leak" test to include reachability from a heap-escaped root. Fixes #17318. Change-Id: Ied1e75cee17ede8ca72a8b9302ce8201641ec593 Reviewed-on: https://go-review.googlesource.com/30693 Run-TryBot: David Chase <drchase@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Russ Cox <rsc@golang.org> Reviewed-on: https://go-review.googlesource.com/31290 Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org> Reviewed-by: David Chase <drchase@google.com>
…od" to fixed point In some cases the members of the root set from which flood runs themselves escape, without their referents being also tagged as escaping. Fix this by reflooding from those roots whose escape increases, and also enhance the "leak" test to include reachability from a heap-escaped root. Fixes golang#17318. Change-Id: Ied1e75cee17ede8ca72a8b9302ce8201641ec593 Reviewed-on: https://go-review.googlesource.com/30693 Run-TryBot: David Chase <drchase@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Russ Cox <rsc@golang.org> Reviewed-on: https://go-review.googlesource.com/31290 Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org> Reviewed-by: David Chase <drchase@google.com>
Please answer these questions before submitting your issue. Thanks!
What version of Go are you using (
go version
)?go version go1.7.1 darwin/amd64
What operating system and processor architecture are you using (
go env
)?GOARCH="amd64"
GOBIN=""
GOEXE=""
GOHOSTARCH="amd64"
GOHOSTOS="darwin"
GOOS="darwin"
GOPATH="/Users/jacob/Projects/gocode"
GORACE=""
GOROOT="/usr/local/Cellar/go/1.7.1/libexec"
GOTOOLDIR="/usr/local/Cellar/go/1.7.1/libexec/pkg/tool/darwin_amd64"
CC="clang"
GOGCCFLAGS="-fPIC -m64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -fdebug-prefix-map=/var/folders/d6/ynnsnz1x4m9188yk1r8xns3w0000gn/T/go-build882056681=/tmp/go-build -gno-record-gcc-switches -fno-common"
CXX="clang++"
CGO_ENABLED="1"
Also tested with same configuration using GOARCH=386 on Darwin (issue reproducible) and on windows/386 in a virtual machine (issue NOT reproducible).
What did you do?
Attempted to pass a closure as a callback to another function that generates values to be processed and invokes the callback with these values (see demo below).
What did you expect to see?
The program should run without issue.
What did you see instead?
After an indeterminate number of invocations, the closure seems to be invalidated, and invoking it as a callback generates a segmentation fault. Alternatively, with optimizations disabled, the garbage collector indicates that there is a reference to an invalid block of memory (which to my novice reading seems to be pointing at the closure).
A package demonstrating this problem (and including stack traces) is available here:
https://github.com/havoc-io/go-closure-issue
I apologize in advance for not being able to produce a more minimal test case - I really did try. This package is already extracted from a larger program.
Instructions and more details are included in the repository, but cloning and running
go test
(possibly 3 or 4 times) should reproduce the error.None of the code here uses unsafe or cgo, nor is the callback ever changed after defined. Using
GOGC=off
seems to stop the issue. Disabling SSA makes no difference. My suspicion is that the GC is somehow not realizing that the closure is referenced and is reclaiming it (potentially due to the complex structure of the function invoking the callback), but I am not even remotely an expert on Go's internals or GC implementation.Perhaps @kardianos can chime in since he wrote the underlying package.
The text was updated successfully, but these errors were encountered: