elf: Discard input .note.gnu.build-id sections
authorH.J. Lu <hjl.tools@gmail.com>
Wed, 1 Dec 2021 04:40:38 +0000 (20:40 -0800)
committerH.J. Lu <hjl.tools@gmail.com>
Thu, 2 Dec 2021 00:40:43 +0000 (16:40 -0800)
1. Discard input .note.gnu.build-id sections.
2. Clear the build ID field before writing.
3. Use bfd_make_section_anyway_with_flags to create the output
.note.gnu.build-id section.

PR ld/28639
* ldelf.c (ldelf_after_open): Discard input .note.gnu.build-id
sections, excluding the first one.
(write_build_id): Clear the build ID field before writing.
(ldelf_setup_build_id): Use bfd_make_section_anyway_with_flags
to create the output .note.gnu.build-id section.
* testsuite/ld-elf/build-id.exp: New file.
* testsuite/ld-elf/pr28639a.rd: Likewise.
* testsuite/ld-elf/pr28639b.rd: Likewise.
* testsuite/ld-elf/pr28639c.rd: Likewise.
* testsuite/ld-elf/pr28639d.rd: Likewise.

ld/ldelf.c
ld/testsuite/ld-elf/build-id.exp [new file with mode: 0644]
ld/testsuite/ld-elf/pr28639a.rd [new file with mode: 0644]
ld/testsuite/ld-elf/pr28639b.rd [new file with mode: 0644]
ld/testsuite/ld-elf/pr28639c.rd [new file with mode: 0644]
ld/testsuite/ld-elf/pr28639d.rd [new file with mode: 0644]

index 0c39eb6024f26465c416647b8b7439346670b14c..529992b02ae2d14a26646ed9e0ca09c0dee1645a 100644 (file)
@@ -1043,6 +1043,15 @@ ldelf_after_open (int use_libpath, int native, int is_linux, int is_freebsd,
   /* Do not allow executable files to be used as inputs to the link.  */
   for (abfd = link_info.input_bfds; abfd; abfd = abfd->link.next)
     {
+      /* Discard input .note.gnu.build-id sections.  */
+      s = bfd_get_section_by_name (abfd, ".note.gnu.build-id");
+      while (s != NULL)
+       {
+         if (s != elf_tdata (link_info.output_bfd)->o->build_id.sec)
+           s->flags |= SEC_EXCLUDE;
+         s = bfd_get_next_section_by_name (NULL, s);
+       }
+
       if (abfd->xvec->flavour == bfd_target_elf_flavour
          && !bfd_input_just_syms (abfd)
          && elf_tdata (abfd) != NULL
@@ -1386,6 +1395,9 @@ write_build_id (bfd *abfd)
   id_bits = contents + size;
   size = asec->size - size;
 
+  /* Clear the build ID field.  */
+  memset (id_bits, 0, size);
+
   bfd_h_put_32 (abfd, sizeof "GNU", &e_note->namesz);
   bfd_h_put_32 (abfd, size, &e_note->descsz);
   bfd_h_put_32 (abfd, NT_GNU_BUILD_ID, &e_note->type);
@@ -1417,7 +1429,8 @@ ldelf_setup_build_id (bfd *ibfd)
 
   flags = (SEC_ALLOC | SEC_LOAD | SEC_IN_MEMORY
           | SEC_LINKER_CREATED | SEC_READONLY | SEC_DATA);
-  s = bfd_make_section_with_flags (ibfd, ".note.gnu.build-id", flags);
+  s = bfd_make_section_anyway_with_flags (ibfd, ".note.gnu.build-id",
+                                         flags);
   if (s != NULL && bfd_set_section_alignment (s, 2))
     {
       struct elf_obj_tdata *t = elf_tdata (link_info.output_bfd);
diff --git a/ld/testsuite/ld-elf/build-id.exp b/ld/testsuite/ld-elf/build-id.exp
new file mode 100644 (file)
index 0000000..19c22a7
--- /dev/null
@@ -0,0 +1,77 @@
+# Expect script for --build-id tests.
+#   Copyright (C) 2021 Free Software Foundation, Inc.
+#
+# This file is part of the GNU Binutils.
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 3 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
+# MA 02110-1301, USA.
+#
+
+# Exclude non-ELF targets.
+
+if ![is_elf_format] {
+    return
+}
+
+if { [istarget frv-*-*] || [istarget lm32-*-*] } {
+    return
+}
+
+if { !([istarget *-*-linux*]
+       || [istarget arm*-*-uclinuxfdpiceabi]
+       || [istarget *-*-nacl*]
+       || [istarget *-*-gnu*]) } then {
+    return
+}
+
+run_ld_link_tests [list \
+    [list \
+       "pr28639a.o" \
+       "-r --build-id=md5" \
+       "" \
+       "" \
+       {start.s} \
+       {{readelf {--notes} pr28639a.rd}} \
+       "pr28639a.o" \
+    ] \
+    [list \
+       "pr28639a.o" \
+       "-r --build-id" \
+       "" \
+       "" \
+       {dummy.s} \
+       {{readelf {--notes} pr28639b.rd}} \
+       "pr28639b.o" \
+    ] \
+    [list \
+       "pr28639a" \
+       "--build-id tmpdir/pr28639a.o tmpdir/pr28639b.o" \
+       "" \
+       "" \
+       {dummy.s} \
+       {{readelf {--notes} pr28639b.rd}  \
+        {readelf {--notes} pr28639c.rd}} \
+       "pr28639a" \
+    ] \
+    [list \
+       "pr28639b" \
+       "--build-id=none tmpdir/pr28639a.o tmpdir/pr28639b.o" \
+       "" \
+       "" \
+       {dummy.s} \
+       {{readelf {--notes} pr28639d.rd}} \
+       "pr28639b" \
+    ] \
+]
diff --git a/ld/testsuite/ld-elf/pr28639a.rd b/ld/testsuite/ld-elf/pr28639a.rd
new file mode 100644 (file)
index 0000000..e850870
--- /dev/null
@@ -0,0 +1,6 @@
+#...
+Displaying notes found in: \.note\.gnu\.build-id
+  Owner                Data size       Description
+  GNU                  0x00000010      NT_GNU_BUILD_ID \(unique build ID bitstring\)
+    Build ID: [0-9a-f]+
+#pass
diff --git a/ld/testsuite/ld-elf/pr28639b.rd b/ld/testsuite/ld-elf/pr28639b.rd
new file mode 100644 (file)
index 0000000..04dcb04
--- /dev/null
@@ -0,0 +1,6 @@
+#...
+Displaying notes found in: \.note\.gnu\.build-id
+  Owner                Data size       Description
+  GNU                  0x00000014      NT_GNU_BUILD_ID \(unique build ID bitstring\)
+    Build ID: [0-9a-f]+
+#pass
diff --git a/ld/testsuite/ld-elf/pr28639c.rd b/ld/testsuite/ld-elf/pr28639c.rd
new file mode 100644 (file)
index 0000000..64221e5
--- /dev/null
@@ -0,0 +1,10 @@
+#failif
+#...
+Displaying notes found in: \.note\.gnu\.build-id
+  Owner                Data size       Description
+  GNU                  0x[0-9a-f]+     NT_GNU_BUILD_ID \(unique build ID bitstring\)
+    Build ID: [0-9a-f]+
+  Owner                Data size       Description
+  GNU                  0x[0-9a-f]+     NT_GNU_BUILD_ID \(unique build ID bitstring\)
+    Build ID: [0-9a-f]+
+#...
diff --git a/ld/testsuite/ld-elf/pr28639d.rd b/ld/testsuite/ld-elf/pr28639d.rd
new file mode 100644 (file)
index 0000000..897c849
--- /dev/null
@@ -0,0 +1,4 @@
+#failif
+#...
+Displaying notes found in: \.note\.gnu\.build-id
+#...