-
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: value assignment to pointer converted from unsafe pointer exceeds structure boundary on 32-bit arm #56979
Comments
This is unsupported behavior and is not expected to work. |
@randall77 isn't the size is 24 on 32-bit platforms? 20 for netip.Addr (2 uint64 + 1 pointer), 2 for a uint16, 2 for padding. @database64128 your conversion violates the 1st rule of unsafe.Pointer conversion https://pkg.go.dev/unsafe#Pointer
Your |
@cuonglm Yeah sorry, I can't do math. |
This custom address type is used in places where the address is either a domain name or an IP. It used to consist of a // For space efficiency, the IP address and the domain string share the same space.
// The [netip.Addr] is stored in its original layout.
// The domain string's data pointer is stored in the ip.z field.
// Its length is stored at the beginning of the structure.
// This is essentially an unsafe "enum".
It's not clear to me what "equivalent" means. I think this needs to be clarified. And I still don't see how the memory layout is different. Can you elaborate on that? Here's the memory layout of type uint128 struct {
hi uint64
lo uint64
}
type Addr struct {
addr uint128
z *intern.Value
}
type AddrPort struct {
ip Addr
port uint16
} If you change my type Addr struct {
_ [0]func()
addr netipAddrHeader
port uint16
af addressFamily
} to: type Addr struct {
_ [0]func()
ap netip.AddrPort
af addressFamily
} It still has the same memory layout. And a pointer to an |
It does not have the same memory layout. The padding is different. |
Closing because this is not a bug in Go. |
Thanks. I understand that this is undefined behavior. But still, I feel like this doesn't have to be like this way. The unaligned |
The same behavior will happen on 64 bit for larger structs (add 2 more |
What version of Go are you using (
go version
)?Does this issue reproduce with the latest release?
Yes.
What operating system and processor architecture are you using (
go env
)?go env
OutputWhat did you do?
Full minimal reproduction code: https://go.dev/play/p/r2c8IBscDxV (Can only be reproduced on 32-bit ARM)
Most relevant snippet:
Sorry that this example is a bit complex. I don't have 32-bit ARM devices to test it, so I'm sticking to the original code that can absolutely trigger the problem.
What did you expect to see?
The
AddrFromIPPort
function works on all platforms.What did you see instead?
The function works fine on amd64 and arm64. On 32-bit ARM,
*(*netip.AddrPort)(unsafe.Pointer(&addr)) = addrPort
overwrites the next byteaf
and causes the returnedaddr
to haveaf
set to 0.The above code snippet was taken out from https://github.com/database64128/shadowsocks-go/blob/main/conn/addr.go. A user reported the issue (database64128/shadowsocks-go#30) and confirmed that my fix (database64128/shadowsocks-go@ea642fe) works:
The text was updated successfully, but these errors were encountered: