如何解决二等分 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.

我发现 GNU binutils2.31.1会导致此错误,但使用 version2.30可以解决问题。


来源:

我最终找到了答案这条聊天记录用户deviosity说:

并继续遇到可怕的:不支持的重定位类型: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);

相关内容