-
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: generate external DWARF debuginfo archives directly #51692
Comments
CC @thanm |
Would it be sufficient to require a host linker when using this flag? Then the host linker could be in charge of writing out the archive in the standard format, and Go tools wouldn't have to know those details. |
This proposal has been added to the active column of the proposals project |
Most applications large enough to be running into these issues (e.g. binary with DWARF is "too big") are already using external linking due to the use of CGO in a dependency somewhere. When I do a build of k8s with "ldflags=-v", I can see that the external linker is already being invoked by default for a regular build. Given that k8s is already carefully curating linker flags, it seems that the easiest path forward would be to just pass the proper options to the host linker, as Russ suggests. |
Great. Doing it via the host linker sounds like a small amount of work. |
Based on the discussion above, this proposal seems like a likely accept. |
Separate .debug files are just elf (or mach-o) files so, in theory, the linker already knows how to write them. Letting the host linker do it is of course less work. Doing .pdb is much more complicated since the format is different and (afaik) only partially documented. |
No change in consensus, so accepted. 🎉 |
@rsc Doing this via the host linker would be great, but as far as I can tell neither GNU C and C++ builds will usually compile and link an integrated binary with full debuginfo and symbols, then post-process it to produce a stripped release binary and an external debuginfo archive.
In any case I don't see how this can be done with external linker flags alone. I suspect that to implement this, either:
It'd be good to make this easy for the golang ecosystem to adopt, so it becomes normal to publish debug symbols alongside release builds on github artifacts etc, and ideally even publish them on an elfutils A lot of useful info on this topic can be found at https://maskray.me/blog/2022-10-30-distribution-of-debug-information Sorry for the delayed response by the way. I tend to be flooded with noisy github notifications and it's hard to find the relevant ones. |
@thanm Are you aware of a set of linker flags that'll produce external debuginfo in a single pass? I'd love to do it that way, at least for now, but am not aware of a way to do so. |
Yes, this is what I had in mind. In external linking the Go linker prepares an intermediate object file (e.g. "go.o") that it then hands off to the external linker as a final step. Before doing that it seems reasonable to run llvm-objcopy (or equivalent) to split off the debug info to a separate file and then pass the striped go.o to the external linker. Implementing full split DWARF would obviously be more work. I don't think we would need to touch the compiler to do that, but it would require some changes in the linker. |
I'm a little late here, just discovered this proposal. I would prefer to keep PDB generation out of this proposal. If someday we support generating PDB files, then passing Having said this, would it make sense to change the flag name to |
TL;DR: Add an
-external-debuginfo
command line argument togo tool link
that causes thelink
command to write a.debug
archive containing symbol tables and DWARF debuginfo for the executable being linked.-w
and/or-s
will continue to have their current effects on the output executable. These flags will have no effect on the archive produced by-external-debuginfo
, which will always include the full symbol table and DWARF debuginfo.This could be implemented by using the functionality of elfutils
eu-strip
incmd/link
/pkg/tool/link
vialibelf
.For Windows targets a
.pdb
file could be produced, or external debuginfo support for Windows targets could be initially omitted if it's too hard to do at the same time.Rationale
Kubernetes project binaries and container images are currently built and distributed with the
-w -s
linker flags, so DWARF debuginfo is omitted then the executable is stripped of all symbols.This saves disk space by producing smaller binaries and container images, though it has no memory consumption or performance benefit at executable runtime.
The cost is that debugging such executables is extremely difficult - only numeric addresses are available, with no symbols for functions, variables, etc. If a problem only occurs in a production or near-production system and you want to inspect the running component, you will see backtraces like:
In traditional gcc or llvm based C/C++ build tool chains a balance between executable size and debuggability is maintained using external debuginfo archives. These are supported by
gcc
,lldb
and anything usingelfutils
orlibgdb
for symbol loading. Delve supports loading of external symbols from/usr/lib/debug/.build_id
and/usr/lib/debug/{path}
etc, likegdb
and other tools.Neither GNU
ld
and LLVMld.lld
provide a command line argument for writing external debuginfo archives. Instead the release process for most binaries includes a separate pass to split the original executable with debuginfo into separate stripped executable and debuginfo archive files usingobjcopy --only-keep-debug
,eu-strip
andobjcopy --add-gnu-debuglink
.The golang toolchain tries to be an integrated single-stop shop without external linker dependencies etc, so golang projects including the official k8s distribution don't tend to use this workflow. They're release-size conscious so they just don't provide any debuginfo. They just build stripped binaries for release. This is convenient but makes it much more difficult to diagnose problems.
go/link
should instead support directly generating an external debuginfo archive ready for upload to a debuginfod symbol server, inclusion in a container image's/usr/lib/debug
and/usr/lib/debug/.build_id
trees, and/or distribution in release tarballs.Note that
delve
already supports debuginfod. Ifgo/link
provided an easy way to generate external symbol archives, the k8s project or an interested 3rd party could host a public debuginfod with debuginfo for all future release binaries for use ingdb
,delve
,lldb
, etc.For example in gdb, one might add
https://debuginfod.k8s.io/
to theDEBUGINFOD_URLS
env-var, addset debuginfod enabled
to.gdbinit
, and have symbols automatically downloaded on demand when debugging k8s release binaries. Even over a remotegdbserver
. A similar approach is possible with Delve.Symbols can also be downloaded on-demand from a debuginfod with
eu-unstrip
then used as normal local detached symbol archives, andeu-make-debug-archive
can be used to make a full archive of a system or container's libraries and debuginfo for remote debugging use. But the symbols must be available for that to be possible.Interim workaround
Golang binaries can be built with debuginfo and symbols (neither
-s
nor-w
LDFLAGS
should be used) then split into separate debuginfo witheu-strip
for linux targets.This requires each build script to implement features that require dependencies that aren't part of the main go toolchain. So it's unlikely to see wide adoption until and unless the golang toolchain itself provides a standard way to do it like it has with
-s
for stripped executables.But it's not overly complicated:
will yield a stripped
mybin
with embedded external debuginfo link and a detached debuginfomybin.debug
. Ifmybin.debug
is placed in the appropriate location in/usr/lib/debug
,/usr/lib/debug/.build_id
, and/or on a debuginfod server, tools likedlv
andgdb
will automatically find its symbols when attaching to it.The text was updated successfully, but these errors were encountered: