Received: by 2002:a05:6358:16cc:b0:ea:6187:17c9 with SMTP id r12csp8781747rwl; Tue, 10 Jan 2023 19:28:33 -0800 (PST) X-Google-Smtp-Source: AMrXdXta9lTXi2ZvXyGTJDYfU0zkVkWjuvx/UTh9pWZkWeEpz3I5+q8EHNSGjtYPwK/cKoUgNHXv X-Received: by 2002:a17:902:9046:b0:194:4196:7dae with SMTP id w6-20020a170902904600b0019441967daemr3515349plz.3.1673407713759; Tue, 10 Jan 2023 19:28:33 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1673407713; cv=none; d=google.com; s=arc-20160816; b=JVwhGY/NGHi5enT1H/Fcr4dWr5q8YJsFOUR+KVSE6wPkedSdLjsIH//YPOgnCyqLNx MskUvAJU7GiHG53qUfBvnFFagFYOeGQxK2FFQgIiindfNGP1t4WjajXGzUh1/fh/DSgj lYHlfukIqXq1GI/BGsyu2ImQAyG1ERWYeeHDWnG1R41lOVhz/dPDhQFc6al4tWg6Xlj8 smzQbOo50d9hngYWP+tvn1LCSDHMjnnsdPRA6HQdHLDDUy2VnoSsIHFy/udMWyyMqLee cL/qRq65rtTzaqB4hpgWbi44mpZJW134gUTDRqXp3f+IJpMoZPXp6Eypl775xMfIeXm1 zwXg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:in-reply-to :mime-version:user-agent:date:message-id:from:cc:references:to :subject; bh=KuSZmt2F7X7Uq658c1hgS2aICAtZ9JZCH/S0JlkIwJY=; b=xc/JUyagt6mB9aScc3UuyzC9PfoIJefDAadck4OJqZxYqP8OnOo4QaxzE0o9rSBXpf e0R1Ue2JIjkbqajkevWeiqvzcvM04plfegObgSh3K+/dY0zlfIqrpV+10KmF3CR81QCr L2XvaX2PSBlWoxU67jUS7ywXr/liGDsZ+A/+nYo+52tMr8HyabnPZYEUZLiH8+7I3NSj hBidBaq+xXhM9PcpKKNAN42yYNLGR+xPPnYx84LB0oVjr2her2BPfhgE/OtX5SomHdgd 8Rwydv5a+6jRwSp9FfDVCR2x43nP04Szgqh5f/ma8qyymF5pmKnV8EyS9sbi5pv9FLiL Of9w== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id ik28-20020a170902ab1c00b00193287d238bsi8315918plb.440.2023.01.10.19.28.27; Tue, 10 Jan 2023 19:28:33 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231429AbjAKDTi (ORCPT + 53 others); Tue, 10 Jan 2023 22:19:38 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:59972 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S235482AbjAKDTK (ORCPT ); Tue, 10 Jan 2023 22:19:10 -0500 Received: from loongson.cn (mail.loongson.cn [114.242.206.163]) by lindbergh.monkeyblade.net (Postfix) with ESMTP id 88412178AB for ; Tue, 10 Jan 2023 19:18:55 -0800 (PST) Received: from loongson.cn (unknown [192.168.200.1]) by gateway (Coremail) with SMTP id _____8Cxe+qdKr5j2+EAAA--.2617S3; Wed, 11 Jan 2023 11:18:53 +0800 (CST) Received: from [0.0.0.0] (unknown [192.168.200.1]) by localhost.localdomain (Coremail) with SMTP id AQAAf8Bxzr6bKr5j0roXAA--.52803S3; Wed, 11 Jan 2023 11:18:52 +0800 (CST) Subject: Re: [PATCH 3/4] LoongArch: Add support for kernel relocation To: Huacai Chen , Xi Ruoyao , Jinyang He References: <1673255274-18238-1-git-send-email-tangyouling@loongson.cn> <1673255274-18238-4-git-send-email-tangyouling@loongson.cn> Cc: Xuerui Wang , loongarch@lists.linux.dev, linux-kernel@vger.kernel.org From: Youling Tang Message-ID: <3b281ec1-c432-eb7b-3131-3049a0d0ea95@loongson.cn> Date: Wed, 11 Jan 2023 11:18:51 +0800 User-Agent: Mozilla/5.0 (X11; Linux mips64; rv:45.0) Gecko/20100101 Thunderbird/45.4.0 MIME-Version: 1.0 In-Reply-To: <1673255274-18238-4-git-send-email-tangyouling@loongson.cn> Content-Type: text/plain; charset=windows-1252; format=flowed Content-Transfer-Encoding: 7bit X-CM-TRANSID: AQAAf8Bxzr6bKr5j0roXAA--.52803S3 X-CM-SenderInfo: 5wdqw5prxox03j6o00pqjv00gofq/ X-Coremail-Antispam: 1Uk129KBjvJXoW3XryfGrWDWw1rKFWxZr1DWrg_yoW3tr4xpr ZrZr4kJr4xGF17Jr9aqa4ruryUZws7Ww43W3ZrK34rCF1avFn5Ww4vgrnrXFW0qws5Kr40 qFyF9asFya1UAaDanT9S1TB71UUUUU7qnTZGkaVYY2UrUUUUj1kv1TuYvTs0mT0YCTnIWj qI5I8CrVACY4xI64kE6c02F40Ex7xfYxn0WfASr-VFAUDa7-sFnT9fnUUIcSsGvfJTRUUU bOxFc2x0x2IEx4CE42xK8VAvwI8IcIk0rVWrJVCq3wA2ocxC64kIII0Yj41l84x0c7CEw4 AK67xGY2AK021l84ACjcxK6xIIjxv20xvE14v26r1I6r4UM28EF7xvwVC0I7IYx2IY6xkF 7I0E14v26r4j6F4UM28EF7xvwVC2z280aVAFwI0_Gr1j6F4UJwA2z4x0Y4vEx4A2jsIEc7 CjxVAFwI0_Gr1j6F4UJwAS0I0E0xvYzxvE52x082IY62kv0487Mc804VCY07AIYIkI8VC2 zVCFFI0UMc02F40EFcxC0VAKzVAqx4xG6I80ewAv7VCjz48v1sIEY20_WwAm72CE4IkC6x 0Yz7v_Jr0_Gr1lF7xvr2IY64vIr41lc7I2V7IY0VAS07AlzVAYIcxG8wCF04k20xvY0x0E wIxGrwCF04k20xvE74AGY7Cv6cx26rWl4I8I3I0E4IkC6x0Yz7v_Jr0_Gr1lx2IqxVAqx4 xG67AKxVWUJVWUGwC20s026x8GjcxK67AKxVWUGVWUWwC2zVAF1VAY17CE14v26r126r1D MIIYrxkI7VAKI48JMIIF0xvE2Ix0cI8IcVAFwI0_Jr0_JF4lIxAIcVC0I7IYx2IY6xkF7I 0E14v26r1j6r4UMIIF0xvE42xK8VAvwI8IcIk0rVWUJVWUCwCI42IY6I8E87Iv67AKxVWU JVW8JwCI42IY6I8E87Iv6xkF7I0E14v26r1j6r4UYxBIdaVFxhVjvjDU0xZFpf9x0zRVWl kUUUUU= X-Spam-Status: No, score=-1.9 required=5.0 tests=BAYES_00,NICE_REPLY_A, SPF_HELO_PASS,SPF_PASS autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 01/09/2023 05:07 PM, Youling Tang wrote: > arch/loongarch/kernel/relocate.c contains the functions necessary to > relocate the kernel elsewhere in memory. > > The kernel makes a copy of itself at the new address. It uses the > relocation table inserted by the relocs tool to fix symbol references > within the new image. > > If copy/relocation is successful then the entry point of the new kernel > is returned, otherwise fall back to starting the kernel in place. > > Signed-off-by: Youling Tang > --- > arch/loongarch/Kconfig | 15 +++++ > arch/loongarch/Makefile | 5 ++ > arch/loongarch/kernel/Makefile | 2 + > arch/loongarch/kernel/head.S | 18 ++++++ > arch/loongarch/kernel/relocate.c | 96 +++++++++++++++++++++++++++++ > arch/loongarch/kernel/vmlinux.lds.S | 11 +++- > 6 files changed, 145 insertions(+), 2 deletions(-) > create mode 100644 arch/loongarch/kernel/relocate.c > > diff --git a/arch/loongarch/Kconfig b/arch/loongarch/Kconfig > index 9cc8b84f7eb0..089a4695b1b3 100644 > --- a/arch/loongarch/Kconfig > +++ b/arch/loongarch/Kconfig > @@ -48,6 +48,7 @@ config LOONGARCH > select ARCH_SUPPORTS_ATOMIC_RMW > select ARCH_SUPPORTS_HUGETLBFS > select ARCH_SUPPORTS_NUMA_BALANCING > + select SYS_SUPPORTS_RELOCATABLE > select ARCH_USE_BUILTIN_BSWAP > select ARCH_USE_CMPXCHG_LOCKREF > select ARCH_USE_QUEUED_RWLOCKS > @@ -229,6 +230,11 @@ config SCHED_OMIT_FRAME_POINTER > config AS_HAS_EXPLICIT_RELOCS > def_bool $(as-instr,x:pcalau12i \$t0$(comma)%pc_hi20(x)) > > +config SYS_SUPPORTS_RELOCATABLE > + bool > + help > + Selected if the platform supports relocating the kernel. > + > menu "Kernel type and options" > > source "kernel/Kconfig.hz" > @@ -474,6 +480,15 @@ config PHYSICAL_START > specified in the "crashkernel=YM@XM" command line boot parameter > passed to the panic-ed kernel). > > +config RELOCATABLE > + bool "Relocatable kernel" > + depends on SYS_SUPPORTS_RELOCATABLE > + help > + This builds the kernel as a Position Independent Executable (PIE), > + which retains all relocation metadata required to relocate the > + kernel binary at runtime to a different virtual address than the > + address it was linked at. > + > config SECCOMP > bool "Enable seccomp to safely compute untrusted bytecode" > depends on PROC_FS > diff --git a/arch/loongarch/Makefile b/arch/loongarch/Makefile > index 4402387d2755..27b5a70ff31c 100644 > --- a/arch/loongarch/Makefile > +++ b/arch/loongarch/Makefile > @@ -71,6 +71,11 @@ KBUILD_AFLAGS_MODULE += -Wa,-mla-global-with-abs > KBUILD_CFLAGS_MODULE += -fplt -Wa,-mla-global-with-abs,-mla-local-with-abs > endif > > +ifeq ($(CONFIG_RELOCATABLE),y) > +LDFLAGS_vmlinux += -static -pie --no-dynamic-linker -z notext > +KBUILD_CFLAGS_KERNEL += -fPIE > +endif > + > cflags-y += -ffreestanding > cflags-y += $(call cc-option, -mno-check-zero-division) > > diff --git a/arch/loongarch/kernel/Makefile b/arch/loongarch/kernel/Makefile > index fcaa024a685e..33787d22e6f4 100644 > --- a/arch/loongarch/kernel/Makefile > +++ b/arch/loongarch/kernel/Makefile > @@ -31,6 +31,8 @@ endif > obj-$(CONFIG_MODULES) += module.o module-sections.o > obj-$(CONFIG_STACKTRACE) += stacktrace.o > > +obj-$(CONFIG_RELOCATABLE) += relocate.o > + > obj-$(CONFIG_PROC_FS) += proc.o > > obj-$(CONFIG_SMP) += smp.o > diff --git a/arch/loongarch/kernel/head.S b/arch/loongarch/kernel/head.S > index e8a4bf9d7599..6db1549177ad 100644 > --- a/arch/loongarch/kernel/head.S > +++ b/arch/loongarch/kernel/head.S > @@ -88,7 +88,25 @@ SYM_CODE_START(kernel_entry) # kernel entry point > PTR_ADD sp, sp, tp > set_saved_sp sp, t0, t1 > > +#ifdef CONFIG_RELOCATABLE > + /* Copy kernel and apply the relocations */ > + bl relocate_kernel > + > + /* Repoint the sp into the new kernel image */ > + PTR_LI sp, (_THREAD_SIZE - 32 - PT_SIZE) > + PTR_ADD sp, sp, tp > + set_saved_sp sp, t0, t1 > + PTR_ADDI sp, sp, -4 * SZREG # init stack pointer PTR_ADDI sp, sp, -4 * SZREG # init stack pointer This line needs to be removed. > + > + /* > + * relocate_kernel returns the entry point either > + * in the relocated kernel or the original if for > + * some reason relocation failed. > + */ > + jr a0 > +#else > bl start_kernel > +#endif > ASM_BUG() > > SYM_CODE_END(kernel_entry) > diff --git a/arch/loongarch/kernel/relocate.c b/arch/loongarch/kernel/relocate.c > new file mode 100644 > index 000000000000..a58551c0698d > --- /dev/null > +++ b/arch/loongarch/kernel/relocate.c > @@ -0,0 +1,96 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * Support for Kernel relocation at boot time > + * > + * Copyright (C) 2023 Loongson Technology Corporation Limited > + */ > + > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +#define RELOCATED(x) ((void *)((long)x + offset)) > + > +extern long __rela_dyn_start; > +extern long __rela_dyn_end; > + > +/* > + * Choose a new address for the kernel, for now we'll hard > + * code the destination. > + */ > +static inline void __init *determine_relocation_address(void) > +{ > + return (void *)(CACHE_BASE + 0x02000000); > +} > + > +static inline int __init relocation_addr_valid(void *loc_new) > +{ > + if ((unsigned long)loc_new & 0x0000ffff) { > + /* Inappropriately aligned new location */ > + return 0; > + } > + if ((unsigned long)loc_new < (unsigned long)_end) { > + /* New location overlaps original kernel */ > + return 0; > + } > + return 1; > +} > + > +void *__init relocate_kernel(void) > +{ > + Elf64_Rela *rela, *rela_end; > + void *loc_new; > + unsigned long kernel_length; > + long offset = 0; > + int res = 1; > + /* Default to original kernel entry point */ > + void *kernel_entry = start_kernel; > + > + kernel_length = (long)(_end) - (long)(_text); > + > + loc_new = determine_relocation_address(); > + > + /* Sanity check relocation address */ > + if (relocation_addr_valid(loc_new)) > + offset = (unsigned long)loc_new - (unsigned long)(_text); > + > + if (offset) { > + /* Copy the kernel to it's new location */ > + memcpy(loc_new, _text, kernel_length); > + > + /* Sync the caches ready for execution of new kernel */ > + __asm__ __volatile__ ( > + "ibar 0 \t\n" > + "dbar 0 \t\n"); > + > + rela = (Elf64_Rela *)RELOCATED(&__rela_dyn_start); > + rela_end = (Elf64_Rela *)RELOCATED(&__rela_dyn_end); > + > + for ( ; rela < rela_end; rela++) { > + Elf64_Addr addr = rela->r_offset; > + Elf64_Addr relocated_addr = rela->r_addend; > + > + if (rela->r_info != R_LARCH_RELATIVE) > + continue; > + > + if (relocated_addr >= VMLINUX_LOAD_ADDRESS) > + relocated_addr = RELOCATED(relocated_addr); > + > + *(Elf64_Addr *)RELOCATED(addr) = relocated_addr; > + > + } > + > + /* The current thread is now within the relocated image */ > + __current_thread_info = RELOCATED(__current_thread_info); > + > + /* Return the new kernel's entry point */ > + kernel_entry = RELOCATED(start_kernel); > + } > +out: > + return kernel_entry; > +} > diff --git a/arch/loongarch/kernel/vmlinux.lds.S b/arch/loongarch/kernel/vmlinux.lds.S > index 733b16e8d55d..aec0b6567d24 100644 > --- a/arch/loongarch/kernel/vmlinux.lds.S > +++ b/arch/loongarch/kernel/vmlinux.lds.S > @@ -70,6 +70,8 @@ SECTIONS > .plt : ALIGN(16) { *(.plt) } > .got.plt : ALIGN(16) { *(.got.plt) } > > + .data.rel : { *(.data.rel*) } > + > . = ALIGN(PECOFF_SEGMENT_ALIGN); > __init_begin = .; > __inittext_begin = .; > @@ -93,8 +95,6 @@ SECTIONS > PERCPU_SECTION(1 << CONFIG_L1_CACHE_SHIFT) > #endif > > - .rela.dyn : ALIGN(8) { *(.rela.dyn) *(.rela*) } > - > .init.bss : { > *(.init.bss) > } > @@ -107,6 +107,12 @@ SECTIONS > RO_DATA(4096) > RW_DATA(1 << CONFIG_L1_CACHE_SHIFT, PAGE_SIZE, THREAD_SIZE) > > + .rela.dyn : ALIGN(8) { > + __rela_dyn_start = .; > + *(.rela.dyn) *(.rela*) > + __rela_dyn_end = .; > + } > + > .sdata : { > *(.sdata) > } > @@ -133,6 +139,7 @@ SECTIONS > > DISCARDS > /DISCARD/ : { > + *(.dynamic .dynsym .dynstr .hash .gnu.hash) > *(.gnu.attributes) > *(.options) > *(.eh_frame) >