Linuxカーネルを二分するときにR_X86_64_PLT32エラーを解決する方法

Linuxカーネルを二分するときにR_X86_64_PLT32エラーを解決する方法

バグを見つけるためにLinuxカーネルを二分するとき(私の場合はv4.4とv4.9の間)、次のビルドエラーが発生しました。

RELOCS  arch/x86/boot/compressed/vmlinux.relocs
Unsupported relocation type: R_X86_64_PLT32 (4)

これについて私たちは何ができますか?

人々がこの問題を他の場所で報告したことを発見しました。

答え1

最新バージョンに関連しているようですbinutils

2.31.1GNU binutilsでこのエラーが発生することがわかりましたが、バージョンを使用すると2.30問題が解決します。


源泉:

ついに答えを見つけました。このチャット履歴ユーザーはdeviosity次のように言いました。

引き続き恐ろしい: Unsupported relocation type: R_X86_64_PLT32 (4) エラーが発生します。通常、binutilsを2.30と2.31にダウングレードすると解決されます。

そしてこのコメントこれも確認されました(Ubuntu 16.04も以前のbinutilsバージョンを使用しています2.26.1)。

答え2

上記で次のパッチを見つけました。https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=b21ebf2fb4cde1618915a97cc773e287ff49173e

パッチが失敗したため、module.cを手動でパッチする必要がありました。

diff --git a/arch/x86/kernel/machine_kexec_64.c b/arch/x86/kernel/machine_kexec_64.c
index 1f790cf9d38f..3b7427aa7d85 100644
--- a/arch/x86/kernel/machine_kexec_64.c
+++ b/arch/x86/kernel/machine_kexec_64.c
@@ -542,6 +542,7 @@ int arch_kexec_apply_relocations_add(const Elf64_Ehdr *ehdr,
                goto overflow;
            break;
        case R_X86_64_PC32:
+       case R_X86_64_PLT32:
            value -= (u64)address;
            *(u32 *)location = value;
            break;
diff --git a/arch/x86/kernel/module.c b/arch/x86/kernel/module.c
index da0c160e5589..f58336af095c 100644
--- a/arch/x86/kernel/module.c
+++ b/arch/x86/kernel/module.c
@@ -191,6 +191,7 @@ int apply_relocate_add(Elf64_Shdr *sechdrs,
                goto overflow;
            break;
        case R_X86_64_PC32:
+       case R_X86_64_PLT32:
            if (*(u32 *)loc != 0)
                goto invalid_relocation;
            val -= (u64)loc;
diff --git a/arch/x86/tools/relocs.c b/arch/x86/tools/relocs.c
index 5d73c443e778..220e97841e49 100644
--- a/arch/x86/tools/relocs.c
+++ b/arch/x86/tools/relocs.c
@@ -770,9 +770,12 @@ static int do_reloc64(struct section *sec, Elf_Rel *rel, ElfW(Sym) *sym,
        break;

    case R_X86_64_PC32:
+   case R_X86_64_PLT32:
        /*
         * PC relative relocations don't need to be adjusted unless
         * referencing a percpu symbol.
+        *
+        * NB: R_X86_64_PLT32 can be treated as R_X86_64_PC32.
         */
        if (is_percpu_sym(sym, symname))
            add_reloc(&relocs32neg, offset);

関連情報