-
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/vet: spurious "context leak" due to imprecise 'defer' control-flow analysis in lostcancel #25720
Comments
Thanks for the report and the code to reproduce the issue. It indeed looks like vet's analysis isn't sophisticated enough. It already builds a control flow graph, so hopefully the fix won't be too complex. |
Here's a smaller repro:
If the defer is moved below the It seems to me like this is simply that vet doesn't handle defers properly. The @alandonovan do you think this would be a cfg or a vet fix? For example, when building the cfg, we could move the deferred blocks to be right before the appropriate return statements. But I'm not sure if that would be outside the scope of what a control flow graph should be. |
It just dawned on me that the cfg builder can't possibly do what I described above. For example, if we have:
The CFG will end up with a single return statement, and statically it's impossible to know if there will be a defer or not. So it seems to me like the fix must be after the CFG has been built. |
The error message is slightly confusing, but I think the real problem here is that you're deferring a function that uses an uninitialized variable. Admittedly WithCancel is not supposed to panic, but if it did, the deferred function would call a nil function. The right place for a cleanup operation is after the variable has been successfully initialized. I'm not convinced it is worth complicating the checker logic to handle this case. |
The original snippet does not have any uninitialized variables, and still produces the error |
@Wessie yes, but along the same lines, the cancel function that gets called isn’t guaranteed to be the intended one. |
discovered another variant that causes the same issues type Thing struct {
ctx context.Context
cancel context.CancelFunc
}
func main() {
ctx, cancel := context.WithCancel(parentCTX)
// There is some unrelated code here.
...
thing := Thing{
ctx: ctx,
cancel: cancel,
}
} Go vet doesn't complain from the above if the |
Yeah, this is the case. As long as you use cancel in any form or factor, vet won't complain. I just simply assign cancel to ctx, cancel := context.WithCancel(parentCTX)
// vet doesn't complain if I do this
_ = cancel
// Some code here
go func() {
// cancel on some condition
cancel()
}() |
I also encountered the same issue. And _ The reproduction is exactly same as included in #58850 under "sophisticated example" |
@jiangsunan Here is the sophisticated example in #58850 (comment) the playground https://go.dev/play/p/LpWXSoZ5Ck8. Here is the relevant bit:
If you uncomment the line |
I just ran into this as well. This is pretty clearly a bug in the linter / vetter. |
Certainly it is a bug in the linter, and the tool could definitely be smarter about the ramifications of defer, but it's not a straightforward change. It needs a more sophisticated control-dependent analysis of defer and the state of local variables. I continue to believe that the cases where the tool falls down are often the same places where humans also struggle to understand the code. |
Please answer these questions before submitting your issue. Thanks!
What version of Go are you using (
go version
)?go version go1.10 windows/amd64
Does this issue reproduce with the latest release?
yes, tested on go1.10.2 as well
What operating system and processor architecture are you using (
go env
)?What did you do?
run
go vet
on https://play.golang.org/p/X_vRJbYf7rOWhat did you expect to see?
no errors
What did you see instead?
The text was updated successfully, but these errors were encountered: