-
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: package path of distinct types with unexported fields is the same #16616
Comments
One note to the title change: The |
I'm not suggesting a PkgPath check for the struct itself, but rather for the fields of the struct. But I'm not sure I'm right--I'm looking further. |
I think something is wrong in the export data generated by the compiler. a.go:
main.go:
I would expect the
and with 1.7 I see this:
Current guess: the type descriptors for the two different types are being merged even though that should not happen. |
If I remember well, checking But for example Storing |
gccgo gives a very interesting error message for this test case:
|
Again, I'm not talking about the |
@ianlancetaylor The export data seems correct since in both 1.6 and 1.7 it is not possible to call a.F(s) (given your example). The reason is that the exported package for field i is indeed different for th different types. But it is strange reflection doesn't show that difference. |
Simpler test case:
Running it (go tool compile a.go && go tool compile b.go && go tool link b.o && a.out) prints:
The package path reported via reflect is wrong here as well. It appears to be correct in the export data. |
CC @crawshaw |
The types supplied to reflect.ValueOf for a.V and V are identical, which leads to identical results. |
Preliminary analysis: It looks like the compiler canonicalizes unnamed types such as struct { i int } because it uses its string representation to identify the respective type information. Thus, the interfaces for both a.V and V have the same type information (which can also readily be seen from the assembly output generated by the compiler for b.go above). A quick hack to fully qualify unexported field names in the compiler's fmt.go (lines 1576ff) leads to different type strings for the two structs and in return to different types encountered by reflect.Value. However it doesn't quite fix the problem yet (I'm probably missing another place where the field names are encoded). Either way, the problem here is unrelated to export data but has to do with how unnamed type symbols are referenced. Assigning to @crawshaw for more insight. |
@griesemer's analysis looks right to me. And it looks like this is not a new bug but the choice of which type is treated as canonical jumped around between releases. I'll see if I can fix it. |
CL https://golang.org/cl/27791 mentions this issue. |
Go version and environment:
go version go1.6.3 windows/amd64
Issue:
If we have an exported function in a package, which has a parameter of anonymous struct type with unexported fields, e.g.:
We can't call this function from another package, trying to pass an anonymous struct value (with matching fields), e.g.:
We get a compile-time error:
Understandable, as the fields of the anonymous struct are unexported.
It is also understandable that we can use reflection to create a zero value of this type, and call
subplay.PrintAnonymous()
with this:Which runs, and doesn't panic.
But now if I try to pass a
reflect.Value
of a value of my own anonymous struct (with matching, unexported fields), with non-zero values, it still runs without panic:My question: is this normal? If yes, then this should be possible without using reflection, too. If without reflection the compile-time error is justified, then using reflection should result in runtime panic.
Inspired by this StackOverflow question.
The text was updated successfully, but these errors were encountered: