-
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: len(*f())
does not panic when f() returns (*[k]T)(nil)
#72844
Comments
The spec says:
Let's look at your two examples. In foo, there is a function call, therefore the argument must be evaluated, and should panic: func foo() {
println(len(*nilPtrToArray()))
} In bar, there are no channel receives or function calls, so the len argument is not evaluated, and there is no panic: func bar() {
ptr := nilPtrToArray()
println(len(*ptr))
} So I believe x/tools/go/ssa is correct. Congratulations, you have found a bug in the compiler. |
len
parameter is dereferenced when it shouldn't belen(*f())
does not panic when f() returns (*[k]T)(nil)
@aykevl thanks. I appreciate you documenting tinygo-org/tinygo#4786 here. I found this first when the go2hx transpiler was producing different behavior. |
The ssa output is some surprising, but the execution behaviors are both correct. Regard to channel receives or function calls, there are no differences between var c = func() chan int {
x := make(chan int, 1)
x <- 123
return x
}()
var s []byte
const X = len([1]int{<-c})
const Y = cap([1]int{len(s)}) |
This seems like a bug in the order pass. It rewrites
to
Which is not a semantics-preserving transformation in this strange case.
|
A tricky case:
My read of the spec is that this shouldn't panic, as there's an implicit |
Change https://go.dev/cl/658096 mentions this issue: |
I sent 2 CLs for this, which mostly fix it. |
Change https://go.dev/cl/658097 mentions this issue: |
Thank you for looking into this! |
func f() *[4]int { return nil } _ = len(f()) should not panic. We evaluate f, but there isn't a dereference according to the spec (just "arg is evaluated"). Update #72844 Change-Id: Ia32cefc1b7aa091cd1c13016e015842b4d12d5b4 Reviewed-on: https://go-review.googlesource.com/c/go/+/658096 LUCI-TryBot-Result: Go LUCI <[email protected]> Reviewed-by: Robert Griesemer <[email protected]> Reviewed-by: Keith Randall <[email protected]>
Go version
go version go1.24.0 linux/arm64
Output of
go env
in your module/workspace:What did you do?
I have the following file, called repro.go:
I ran
ssadump
on it:What did you see happen?
foo
andbar
are equivalent. However, they produce different output.foo
results in a nil pointer dereference, whilebar
does not.What did you expect to see?
I think
foo
andbar
should have produced equivalent SSA.I think the following part of the language spec is relevant here:
If I'm reading this correctly, the
bar
case is the correct way this code should be read. Also, when testing withgo run
, the Go compiler doesn't panic because of a nil pointer in either case.Also see: tinygo-org/tinygo#4786
The text was updated successfully, but these errors were encountered: