aarch64: Allow PC-relative relocations against protected STT_FUNC for -shared
authorFangrui Song <maskray@google.com>
Thu, 23 Jun 2022 08:10:44 +0000 (01:10 -0700)
committerFangrui Song <i@maskray.me>
Thu, 23 Jun 2022 08:10:44 +0000 (01:10 -0700)
    __attribute__((visibility("protected"))) void *foo() {
      return (void *)foo;
    }

gcc -fpic -shared -fuse-ld=bfd fails with the confusing diagnostic:

    relocation R_AARCH64_ADR_PREL_PG_HI21 against symbol `foo' which may bind externally can not be used when making a shared object; recompile with -fPIC

Call _bfd_elf_symbol_refs_local_p with local_protected==true to suppress
the error.  The new behavior matches gold and ld.lld.

Note: if some code tries to use direct access relocations to take the
address of foo (likely due to -fno-pic), the pointer equality will
break, but the error should be reported on the executable link, not on
the innocent shared object link.  glibc 2.36 will give a warning at
relocation resolving time.

bfd/elfnn-aarch64.c
ld/testsuite/ld-aarch64/aarch64-elf.exp
ld/testsuite/ld-aarch64/pcrel-protected.s [new file with mode: 0644]
ld/testsuite/ld-aarch64/pcrel_pic_protected.d [new file with mode: 0644]

index ce2632ecd3015175d66eb088bd8b71900dc2feba..c688b7cff2dd31f7601cd165ce45e5b1bd8c7c4e 100644 (file)
@@ -5888,7 +5888,7 @@ elfNN_aarch64_final_link_relocate (reloc_howto_type *howto,
       if (bfd_link_pic (info)
          && (input_section->flags & SEC_ALLOC) != 0
          && (input_section->flags & SEC_READONLY) != 0
-         && !SYMBOL_REFERENCES_LOCAL (info, h))
+         && !_bfd_elf_symbol_refs_local_p (h, info, 1))
        {
          int howto_index = bfd_r_type - BFD_RELOC_AARCH64_RELOC_START;
 
index 31162277bd99aeee995f3eb7cfb0e881bb160119..3c45f87151f82b2a3a5fcb10cea4b0cab534ab40 100644 (file)
@@ -248,6 +248,7 @@ run_dump_test_lp64 "local-addend-r"
 # test error handling on pcrel relocation for shared libraries.
 run_dump_test_lp64 "pcrel_pic_undefined"
 run_dump_test_lp64 "pcrel_pic_defined"
+run_dump_test_lp64 "pcrel_pic_protected"
 
 run_dump_test "limit-b"
 run_dump_test "limit-bl"
diff --git a/ld/testsuite/ld-aarch64/pcrel-protected.s b/ld/testsuite/ld-aarch64/pcrel-protected.s
new file mode 100644 (file)
index 0000000..24c6232
--- /dev/null
@@ -0,0 +1,11 @@
+.protected protected_a, protected_b, protected_c
+.type protected_b, %object
+.type protected_c, %function
+
+.text
+       adrp    x0, protected_a
+       add     x0, x0, :lo12:protected_a
+       adrp    x0, protected_b
+       add     x0, x0, :lo12:protected_b
+       adrp    x0, protected_c
+       add     x0, x0, :lo12:protected_c
diff --git a/ld/testsuite/ld-aarch64/pcrel_pic_protected.d b/ld/testsuite/ld-aarch64/pcrel_pic_protected.d
new file mode 100644 (file)
index 0000000..1b6f24c
--- /dev/null
@@ -0,0 +1,7 @@
+#name: PC-Rel relocation against protected
+#source: pcrel-protected.s
+#target: [check_shared_lib_support]
+#ld: -shared -e0 --defsym protected_a=0x1000 --defsym protected_b=0x1010 --defsym protected_c=0x1020
+#readelf: -r -W
+#...
+There are no relocations in this file.