]> git.ipfire.org Git - thirdparty/glibc.git/commitdiff
elf: Support DT_RELR relative relocation format [BZ #27924] maskray/relr
authorFangrui Song <maskray@google.com>
Wed, 9 Mar 2022 01:17:05 +0000 (17:17 -0800)
committerFangrui Song <maskray@google.com>
Wed, 9 Mar 2022 01:17:05 +0000 (17:17 -0800)
PIE and shared objects usually have many relative relocations. In
2017/2018, SHT_RELR/DT_RELR was proposed on
https://groups.google.com/g/generic-abi/c/bX460iggiKg/m/GxjM0L-PBAAJ
("Proposal for a new section type SHT_RELR") and is a pre-standard. RELR
usually takes 3% or smaller space than R_*_RELATIVE relocations. The
virtual memory size of a mostly statically linked PIE is typically 5~10%
smaller.

This patch adds ELF_DYNAMIC_DO_RELR to ELF_DYNAMIC_RELOCATE.
ELF_DYNAMIC_DO_RELR is ordered before ELF_DYNAMIC_DO_REL[A] so that ifunc
resolvers that require relocated got entries have them relocated. This is
needed for ppc64 according to Alan Modra.

configure
configure.ac
elf/Makefile
elf/dynamic-link.h
elf/elf.h
elf/get-dynamic-info.h
elf/tst-relr-no-pie.c [new file with mode: 0644]
elf/tst-relr.c [new file with mode: 0644]

index 8e5bee775a651fcbaaa96ede8039ae1f049e296e..c4e0d9e8e4b9c1f62fe0bcca70896dbfda42dbea 100755 (executable)
--- a/configure
+++ b/configure
@@ -6115,6 +6115,43 @@ $as_echo "$libc_linker_feature" >&6; }
 config_vars="$config_vars
 have-depaudit = $libc_cv_depaudit"
 
+{ $as_echo "$as_me:${as_lineno-$LINENO}: checking for linker that supports -z pack-relative-relocs" >&5
+$as_echo_n "checking for linker that supports -z pack-relative-relocs... " >&6; }
+libc_linker_feature=no
+if test x"$gnu_ld" = x"yes"; then
+  cat > conftest.c <<EOF
+int _start (void) { return 42; }
+EOF
+  if { ac_try='${CC-cc} $CFLAGS $CPPFLAGS $LDFLAGS $no_ssp
+                   -Wl,-z,pack-relative-relocs -nostdlib -nostartfiles
+                   -fPIC -shared -o conftest.so conftest.c
+                   1>&5'
+  { { eval echo "\"\$as_me\":${as_lineno-$LINENO}: \"$ac_try\""; } >&5
+  (eval $ac_try) 2>&5
+  ac_status=$?
+  $as_echo "$as_me:${as_lineno-$LINENO}: \$? = $ac_status" >&5
+  test $ac_status = 0; }; }
+  then
+    if ${CC-cc} $CFLAGS $CPPFLAGS $LDFLAGS $no_ssp -Wl,-z,pack-relative-relocs -nostdlib \
+       -nostartfiles -fPIC -shared -o conftest.so conftest.c 2>&1 \
+       | grep "warning: -z pack-relative-relocs ignored" > /dev/null 2>&1; then
+      true
+    else
+      libc_linker_feature=yes
+    fi
+  fi
+  rm -f conftest*
+fi
+if test $libc_linker_feature = yes; then
+  libc_cv_relr=yes
+else
+  libc_cv_relr=no
+fi
+{ $as_echo "$as_me:${as_lineno-$LINENO}: result: $libc_linker_feature" >&5
+$as_echo "$libc_linker_feature" >&6; }
+config_vars="$config_vars
+have-relr = $libc_cv_relr"
+
 { $as_echo "$as_me:${as_lineno-$LINENO}: checking for linker that supports --no-dynamic-linker" >&5
 $as_echo_n "checking for linker that supports --no-dynamic-linker... " >&6; }
 libc_linker_feature=no
index 87f67d25ec193ef03bdec7576d5e994fb34b2200..d6bb323fa6e9e5ea9feb96cac7240cd49b4b9002 100644 (file)
@@ -1367,6 +1367,10 @@ LIBC_LINKER_FEATURE([--depaudit], [-Wl,--depaudit,x],
                    [libc_cv_depaudit=yes], [libc_cv_depaudit=no])
 LIBC_CONFIG_VAR([have-depaudit], [$libc_cv_depaudit])
 
+LIBC_LINKER_FEATURE([-z pack-relative-relocs], [-Wl,-z,pack-relative-relocs],
+                   [libc_cv_relr=yes], [libc_cv_relr=no])
+LIBC_CONFIG_VAR([have-relr], [$libc_cv_relr])
+
 LIBC_LINKER_FEATURE([--no-dynamic-linker],
                    [-Wl,--no-dynamic-linker],
                    [libc_cv_no_dynamic_linker=yes],
index c96924e9c2e60acdbbd7c87f3c839bbb80a11ecc..d75353eed0bcf8adf97128da158abe03e2fff04e 100644 (file)
@@ -541,6 +541,13 @@ tests-special += \
   # tests-special
 endif
 endif
+ifeq ($(have-relr),yes)
+tests += tst-relr tst-relr-no-pie
+tests-pie += tst-relr
+tests-no-pie += tst-relr-no-pie
+LDFLAGS-tst-relr += -z pack-relative-relocs
+LDFLAGS-tst-relr-no-pie += -z pack-relative-relocs
+endif
 endif
 
 ifeq ($(run-built-tests),yes)
index 25dd7ca4f2eca3c97fdd1649c15e1b5106e70dc1..169745f9a409aec7995f7418736ae64bfeebc9d6 100644 (file)
@@ -146,12 +146,46 @@ elf_machine_lazy_rel (struct link_map *map, struct r_scope_elem *scope[],
 #  define ELF_DYNAMIC_DO_RELA(map, scope, lazy, skip_ifunc) /* Nothing to do.  */
 # endif
 
+# define ELF_DYNAMIC_DO_RELR(map)                                            \
+  do {                                                                       \
+    ElfW(Addr) l_addr = (map)->l_addr, *where = 0;                           \
+    const ElfW(Relr) *r, *end;                                               \
+    if (!(map)->l_info[DT_RELR])                                             \
+      break;                                                                 \
+    r = (const ElfW(Relr) *)D_PTR((map), l_info[DT_RELR]);                   \
+    end = (const ElfW(Relr) *)((const char *)r +                             \
+                               (map)->l_info[DT_RELRSZ]->d_un.d_val);        \
+    for (; r < end; r++)                                                     \
+      {                                                                              \
+       ElfW(Relr) entry = *r;                                                \
+       if ((entry & 1) == 0)                                                 \
+         {                                                                   \
+           where = (ElfW(Addr) *)(l_addr + entry);                           \
+           *where++ += l_addr;                                               \
+         }                                                                   \
+       else                                                                  \
+         {                                                                   \
+           for (long i = 0; (entry >>= 1) != 0; i++)                         \
+             if ((entry & 1) != 0)                                           \
+               where[i] += l_addr;                                           \
+           where += CHAR_BIT * sizeof(ElfW(Relr)) - 1;                       \
+         }                                                                   \
+      }                                                                              \
+  } while (0);
+
 /* This can't just be an inline function because GCC is too dumb
    to inline functions containing inlines themselves.  */
+# ifdef RTLD_BOOTSTRAP
+#  define DO_RTLD_BOOTSTRAP 1
+# else
+#  define DO_RTLD_BOOTSTRAP 0
+# endif
 # define ELF_DYNAMIC_RELOCATE(map, scope, lazy, consider_profile, skip_ifunc) \
   do {                                                                       \
     int edr_lazy = elf_machine_runtime_setup ((map), (scope), (lazy),        \
                                              (consider_profile));            \
+    if ((map) != &GL(dl_rtld_map) || DO_RTLD_BOOTSTRAP)                              \
+      ELF_DYNAMIC_DO_RELR (map);                                             \
     ELF_DYNAMIC_DO_REL ((map), (scope), edr_lazy, skip_ifunc);               \
     ELF_DYNAMIC_DO_RELA ((map), (scope), edr_lazy, skip_ifunc);                      \
   } while (0)
index 0735f6b579f72a32cfec19a883395306aa7f6d0d..01950291885d1257324322fff7bafb4c6695eef8 100644 (file)
--- a/elf/elf.h
+++ b/elf/elf.h
@@ -443,7 +443,8 @@ typedef struct
 #define SHT_PREINIT_ARRAY 16           /* Array of pre-constructors */
 #define SHT_GROUP        17            /* Section group */
 #define SHT_SYMTAB_SHNDX  18           /* Extended section indices */
-#define        SHT_NUM           19            /* Number of defined types.  */
+#define SHT_RELR         19            /* RELR relative relocations */
+#define        SHT_NUM           20            /* Number of defined types.  */
 #define SHT_LOOS         0x60000000    /* Start OS-specific.  */
 #define SHT_GNU_ATTRIBUTES 0x6ffffff5  /* Object attributes.  */
 #define SHT_GNU_HASH     0x6ffffff6    /* GNU-style hash table.  */
@@ -662,6 +663,11 @@ typedef struct
   Elf64_Sxword r_addend;               /* Addend */
 } Elf64_Rela;
 
+/* RELR relocation table entry */
+
+typedef Elf32_Word     Elf32_Relr;
+typedef Elf64_Xword    Elf64_Relr;
+
 /* How to extract and insert information held in the r_info field.  */
 
 #define ELF32_R_SYM(val)               ((val) >> 8)
@@ -887,7 +893,10 @@ typedef struct
 #define DT_PREINIT_ARRAY 32            /* Array with addresses of preinit fct*/
 #define DT_PREINIT_ARRAYSZ 33          /* size in bytes of DT_PREINIT_ARRAY */
 #define DT_SYMTAB_SHNDX        34              /* Address of SYMTAB_SHNDX section */
-#define        DT_NUM          35              /* Number used */
+#define DT_RELRSZ      35              /* Total size of RELR relative relocations */
+#define DT_RELR                36              /* Address of RELR relative relocations */
+#define DT_RELRENT     37              /* Size of one RELR relative relocaction */
+#define        DT_NUM          38              /* Number used */
 #define DT_LOOS                0x6000000d      /* Start of OS-specific */
 #define DT_HIOS                0x6ffff000      /* End of OS-specific */
 #define DT_LOPROC      0x70000000      /* Start of processor-specific */
index 6c2ccd6db4b1721fcd96775a772e5a76d22102c9..6c2a3a12b162cb0d5f30d60bb4731110853d033e 100644 (file)
@@ -89,6 +89,7 @@ elf_get_dynamic_info (struct link_map *l, bool bootstrap,
 # if ! ELF_MACHINE_NO_REL
       ADJUST_DYN_INFO (DT_REL);
 # endif
+      ADJUST_DYN_INFO (DT_RELR);
       ADJUST_DYN_INFO (DT_JMPREL);
       ADJUST_DYN_INFO (VERSYMIDX (DT_VERSYM));
       ADJUST_DYN_INFO (ADDRIDX (DT_GNU_HASH));
@@ -113,6 +114,8 @@ elf_get_dynamic_info (struct link_map *l, bool bootstrap,
   if (info[DT_REL] != NULL)
     assert (info[DT_RELENT]->d_un.d_val == sizeof (ElfW(Rel)));
 #endif
+  if (info[DT_RELR] != NULL)
+    assert (info[DT_RELRENT]->d_un.d_val == sizeof (ElfW(Relr)));
   if (bootstrap || static_pie_bootstrap)
     {
       assert (info[DT_RUNPATH] == NULL);
diff --git a/elf/tst-relr-no-pie.c b/elf/tst-relr-no-pie.c
new file mode 100644 (file)
index 0000000..7df0cdb
--- /dev/null
@@ -0,0 +1 @@
+#include "tst-relr.c"
diff --git a/elf/tst-relr.c b/elf/tst-relr.c
new file mode 100644 (file)
index 0000000..0b0fe5e
--- /dev/null
@@ -0,0 +1,45 @@
+#include <link.h>
+#include <stdbool.h>
+#include <stdio.h>
+
+static int o, x;
+
+#define ELEMS O O O O O O O O X X X X X X X O O X O O X X X E X E E O X O E
+#define E 0,
+
+#define O &o,
+#define X &x,
+void *arr[] = { ELEMS };
+#undef O
+#undef X
+
+#define O 1,
+#define X 2,
+static char val[] = { ELEMS };
+
+static int
+do_test (void)
+{
+  ElfW(Dyn) *d = _DYNAMIC;
+  if (d)
+    {
+      bool has_relr = false;
+      for (; d->d_tag != DT_NULL; d++)
+       if (d->d_tag == DT_RELR)
+         has_relr = true;
+      if (!has_relr)
+       {
+         fprintf (stderr, "no DT_RELR\n");
+         return 1;
+       }
+    }
+
+  for (int i = 0; i < sizeof (arr) / sizeof (arr[0]); i++)
+    if (!((arr[i] == 0 && val[i] == 0) ||
+         (arr[i] == &o && val[i] == 1) ||
+         (arr[i] == &x && val[i] == 2)))
+      return 1;
+  return 0;
+}
+
+#include <support/test-driver.c>