-
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/link: link
reports "symbol listed multiple times" if go:linkname into both vendor package and non-vendor package
#42401
Comments
I'll take a look. |
A note before getting into the details of the bug: please be aware that the behavior + implementation of "//go:linkname" is explicitly not covered by the Go 1 compatibility guarantee; the fact that importing "unsafe" is required is intended to reinforce this. As the toolchain changes over time, the implementation and behavior of "//go:linkname" can evolve. There are uses of "//go:linkname" that may work with version X of the Go toolchain and then not work with version X+1. For example, using "//go:linkname" to reach from package A into a dependent package B to access one of B's non-exported symbols is something that works today, but may very well not work with future versions of Go. Regarding the Go program in the bug: This program contains behavior that happened to work with the Go 1.14 toolchain implementation of "//go:linkname" but does not work with Go 1.15. In Go 1.14, linking and symbol resolution is done using an entirely name-driven basis; all (or virtuall all) symbols are named, and at link time two symbols with the same name wind up mapping to same linker symbol, regardless of which package or compilation unit the symbol was defined in. What this means is that for Go 1.14 when you write
this has the effect of making "v1" (defined in main.go) and "bar" (defined in foo.go) virtually the same symbol at link time. If you look in the final binary produced by Go 1.14 for this example, there are only two "foo.bar" variables. In Go 1.15 we moved to a new object file format and a new way of doing symbol resolution. For most regular Go symbol definitions (variables, functions) the compiler assigns the symbol an index within the object file; when the linker reads the object file, it sees the symbol (roughly) as a tuple <P,S> where P is a numeric package ID and S is a numeric symbol ID. What this means for Go 1.15 is that there are actually three "foo.bar" strings: one defined in package foo and two in package main; each has a unique identity using the new index-based scheme. Two of the three symbols happen to have the same name, which results in a collision during DWARF generation. |
I'm not sure I understand the issue.
Where do you vendor this? I didn't see any vendor directory or any vendor command. Also, do you run into this issue in real code, or just this synthetic example? |
I don't, I just write it there.
I ran into this issue in real code in our library, it is only used for unittests, so there are a lot of "hacks" in it. The code above is just a simplification. Some of our legacy projects use Maybe we should consider refactoring the library and get rid of |
We only use "//go:linkname" for unittests, not in our production code. There are some not-so-well-designed libraries that come with a lot of side-effects (performing I/O within The "//go:linkname" is a way we found to "hack" these libraries and set them into a certain state for our tests.
Should I consider "//go:linkname" as some sort of "undefined behavior" ? |
I wouldn't say that In general I would advise against using it. |
Change https://golang.org/cl/268178 mentions this issue: |
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?
main
package.What did you expect to see?
The
main
package builds just fine.What did you see instead?
I have also tried:
go 1.14.10
works fine.-ldflags='-w'
to the compiler also works fine.#11699
#15926
#28159
The text was updated successfully, but these errors were encountered: