aboutsummaryrefslogtreecommitdiff
path: root/ld
diff options
context:
space:
mode:
authorEric Botcazou <ebotcazou@gcc.gnu.org>2026-03-30 19:10:26 +0200
committerEric Botcazou <ebotcazou@gcc.gnu.org>2026-03-30 19:12:46 +0200
commit84acfa08c7a40f01b4ae254cc4658e5865edf3c6 (patch)
tree6c567fa68df7e8e2d4f78b458c0665cc93451ce5 /ld
parent82e460365ba9729e1e66b5f146c17a2ed1b4646c (diff)
downloadfsf-binutils-gdb-84acfa08c7a40f01b4ae254cc4658e5865edf3c6.tar.gz
fsf-binutils-gdb-84acfa08c7a40f01b4ae254cc4658e5865edf3c6.tar.bz2
fsf-binutils-gdb-84acfa08c7a40f01b4ae254cc4658e5865edf3c6.zip
PE-COFF: Fix link failure of C++ code with debug info after partial linking
If you apply the following recipe to the attached C/C++ files with a PE-COFF toolchain, you get the specified output: 1. g++ -c clib.cpp cpplib.cpp test.c -g 2. g++ -o pl.o clib.o cpplib.o -nostdlib -Wl,-r 3. objcopy pl.o objcopy.exe: pl.o: warning: COMDAT symbol '.debug_frame$_ZNSt12_Vector_baseIiSaIiEE12_Vector_implD1Ev' does not match section name '.debug_frame' 4. g++ -o test test.o pl.o ld.exe: pl.o: warning: COMDAT symbol '.debug_frame$_ZNSt12_Vector_baseIiSaIiEE12_Vector_implD1Ev' does not match section name '.debug_frame' pl.o:clib.cpp:(.pdata$_ZNSt12_Vector_baseIiSaIiEE13_M_deallocateEPiy+0x0): relocation truncated to fit: IMAGE_REL_AMD64_ADDR32NB against `.text$_ZNSt12_Vector_baseIiSaIiEE13_M_deallocateEPiy' pl.o:clib.cpp:(.pdata$_ZNSt12_Vector_baseIiSaIiEE13_M_deallocateEPiy+0x4): relocation truncated to fit: IMAGE_REL_AMD64_ADDR32NB against `.text$_ZNSt12_Vector_baseIiSaIiEE13_M_deallocateEPiy' pl.o:clib.cpp:(.pdata$_ZNSt12_Vector_baseIiSaIiEE19_M_get_Tp_allocatorEv+0x0): relocation truncated to fit: IMAGE_REL_AMD64_ADDR32NB against `.text$_ZNSt12_Vector_baseIiSaIiEE19_M_get_Tp_allocatorEv' pl.o:clib.cpp:(.pdata$_ZNSt12_Vector_baseIiSaIiEE19_M_get_Tp_allocatorEv+0x4): relocation truncated to fit: IMAGE_REL_AMD64_ADDR32NB against `.text$_ZNSt12_Vector_baseIiSaIiEE19_M_get_Tp_allocatorEv' pl.o:clib.cpp:(.pdata$_ZNSt15__new_allocatorIiE10deallocateEPiy+0x0): relocation truncated to fit: IMAGE_REL_AMD64_ADDR32NB against `.text$_ZNSt15__new_allocatorIiE10deallocateEPiy' pl.o:clib.cpp:(.pdata$_ZNSt15__new_allocatorIiE10deallocateEPiy+0x4): relocation truncated to fit: IMAGE_REL_AMD64_ADDR32NB against `.text$_ZNSt15__new_allocatorIiE10deallocateEPiy' collect2.exe: error: ld returned 1 exit status The problem pertains to section symbols generated for COMDAT sections: they are marked as local symbols as per Microsoft's PE-COFF specification, but partial linking discards the duplicate COMDAT sections without being able to either merge them, or remove them when they are used in a relocation. So they end up as undefined local symbols after the partial link, which in turn may cause the final link to fail (in practice you need e.g. a call to objcopy in between, because it moves them to the end of the symbol list). This change instructs the linker to "relocate" them instead, that is to say to attach them to the one COMDAT section that is output among the multiple COMDAT sections that are duplicate. It also prevents partial linking from prematurely globing the .[z]debug_frame* sections together, as already done for the .eh_frame* sections. bfd/ * cofflink.c (_bfd_coff_link_input_bfd): For a relocatable output, relocate section symbols for input sections that are not going to be emitted because they are duplicate of another one in the link. ld/ * scripttempl/pe.sc (.debug_frame): Do not glob all .debug_frame* sections together when not relocating. (.zdebug_frame): Likewise for .zdebug_frame* sections. * scripttempl/pep.sc (.debug_frame): Likewise. (.zdebug_frame): Likewise.
Diffstat (limited to 'ld')
-rw-r--r--ld/scripttempl/pe.sc4
-rw-r--r--ld/scripttempl/pep.sc4
2 files changed, 4 insertions, 4 deletions
diff --git a/ld/scripttempl/pe.sc b/ld/scripttempl/pe.sc
index f8d7c8ef6e7..a669f279eda 100644
--- a/ld/scripttempl/pe.sc
+++ b/ld/scripttempl/pe.sc
@@ -364,11 +364,11 @@ SECTIONS
.debug_frame ${RELOCATING+BLOCK(__section_alignment__)} ${RELOCATING+(NOLOAD)} :
{
- *(.debug_frame*)
+ *(.debug_frame${RELOCATING+*})
}
.zdebug_frame ${RELOCATING+BLOCK(__section_alignment__)} ${RELOCATING+(NOLOAD)} :
{
- *(.zdebug_frame*)
+ *(.zdebug_frame${RELOCATING+*})
}
.debug_str ${RELOCATING+BLOCK(__section_alignment__)} ${RELOCATING+(NOLOAD)} :
diff --git a/ld/scripttempl/pep.sc b/ld/scripttempl/pep.sc
index a5b4679544c..45e785492c5 100644
--- a/ld/scripttempl/pep.sc
+++ b/ld/scripttempl/pep.sc
@@ -373,11 +373,11 @@ SECTIONS
.debug_frame ${RELOCATING+BLOCK(__section_alignment__)} ${RELOCATING+(NOLOAD)} :
{
- *(.debug_frame*)
+ *(.debug_frame${RELOCATING+*})
}
.zdebug_frame ${RELOCATING+BLOCK(__section_alignment__)} ${RELOCATING+(NOLOAD)} :
{
- *(.zdebug_frame*)
+ *(.zdebug_frame${RELOCATING+*})
}
.debug_str ${RELOCATING+BLOCK(__section_alignment__)} ${RELOCATING+(NOLOAD)} :