[PATCH] arm64: fix KASAN_INLINE
Mark Rutland
mark.rutland at arm.com
Wed Jul 20 07:53:41 PDT 2022
On Wed, Jul 13, 2022 at 03:09:49PM +0100, Mark Rutland wrote:
> Since commit:
>
> a004393f45d9a55e ("arm64: idreg-override: use early FDT mapping in ID map")
>
> Kernels built with KASAN_INLINE=y die early in boot before producing any
> console output. This is because the accesses made to the FDT (e.g. in
> generic string processing functions) are instrumented with KASAN, and
> with KASAN_INLINE=y any access to an address in TTBR0 results in a bogus
> shadow VA, resulting in a data abort.
>
> This patch fixes this by reverting commits:
>
> 7559d9f97581654f ("arm64: setup: drop early FDT pointer helpers")
> bd0c3fa21878b6d0 ("arm64: idreg-override: use early FDT mapping in ID map")
>
> ... and using the TTBR1 fixmap mapping of the FDT.
>
> Note that due to a later commit:
>
> b65e411d6cc2f12a ("arm64: Save state of HCR_EL2.E2H before switch to EL1")
>
> ... which altered the prototype of init_feature_override() (and
> invocation from head.S), commit bd0c3fa21878b6d0 does not revert
> cleanly, and I've fixed that up manually.
>
Whoops; this was meant to have:
Signed-off-by: Mark Rutland <mark.rutland at arm.com>
... but I somehow messed that up.
Will, are you happy to fold that in?
Mark.
> Fixes: a004393f45d9a55e ("arm64: idreg-override: use early FDT mapping in ID map")
> Cc: Ard Biesheuvel <ardb at kernel.org>
> Cc: Catalin Marinas <catalin.marinas at arm.com>
> Cc: Marc Zyngier <maz at kernel.org>
> Cc: Will Deacon <will at kernel.org>
> ---
> arch/arm64/include/asm/setup.h | 3 +++
> arch/arm64/kernel/head.S | 5 +++--
> arch/arm64/kernel/idreg-override.c | 17 +++++++++++------
> arch/arm64/kernel/setup.c | 15 +++++++++++++++
> 4 files changed, 32 insertions(+), 8 deletions(-)
>
> diff --git a/arch/arm64/include/asm/setup.h b/arch/arm64/include/asm/setup.h
> index 5f147a418281..6437df661700 100644
> --- a/arch/arm64/include/asm/setup.h
> +++ b/arch/arm64/include/asm/setup.h
> @@ -5,6 +5,9 @@
>
> #include <uapi/asm/setup.h>
>
> +void *get_early_fdt_ptr(void);
> +void early_fdt_map(u64 dt_phys);
> +
> /*
> * These two variables are used in the head.S file.
> */
> diff --git a/arch/arm64/kernel/head.S b/arch/arm64/kernel/head.S
> index 29d641290293..cefe6a73ee54 100644
> --- a/arch/arm64/kernel/head.S
> +++ b/arch/arm64/kernel/head.S
> @@ -456,8 +456,9 @@ SYM_FUNC_START_LOCAL(__primary_switched)
> #if defined(CONFIG_KASAN_GENERIC) || defined(CONFIG_KASAN_SW_TAGS)
> bl kasan_early_init
> #endif
> - mov x0, x22 // pass FDT address in x0
> - mov x1, x20 // pass the full boot status
> + mov x0, x21 // pass FDT address in x0
> + bl early_fdt_map // Try mapping the FDT early
> + mov x0, x20 // pass the full boot status
> bl init_feature_override // Parse cpu feature overrides
> mov x0, x20
> bl finalise_el2 // Prefer VHE if possible
> diff --git a/arch/arm64/kernel/idreg-override.c b/arch/arm64/kernel/idreg-override.c
> index 42883657f711..1b0542c69738 100644
> --- a/arch/arm64/kernel/idreg-override.c
> +++ b/arch/arm64/kernel/idreg-override.c
> @@ -262,11 +262,16 @@ static __init void __parse_cmdline(const char *cmdline, bool parse_aliases)
> } while (1);
> }
>
> -static __init const u8 *get_bootargs_cmdline(const void *fdt)
> +static __init const u8 *get_bootargs_cmdline(void)
> {
> const u8 *prop;
> + void *fdt;
> int node;
>
> + fdt = get_early_fdt_ptr();
> + if (!fdt)
> + return NULL;
> +
> node = fdt_path_offset(fdt, "/chosen");
> if (node < 0)
> return NULL;
> @@ -278,9 +283,9 @@ static __init const u8 *get_bootargs_cmdline(const void *fdt)
> return strlen(prop) ? prop : NULL;
> }
>
> -static __init void parse_cmdline(const void *fdt)
> +static __init void parse_cmdline(void)
> {
> - const u8 *prop = get_bootargs_cmdline(fdt);
> + const u8 *prop = get_bootargs_cmdline();
>
> if (IS_ENABLED(CONFIG_CMDLINE_FORCE) || !prop)
> __parse_cmdline(CONFIG_CMDLINE, true);
> @@ -290,9 +295,9 @@ static __init void parse_cmdline(const void *fdt)
> }
>
> /* Keep checkers quiet */
> -void init_feature_override(const void *fdt, u64 boot_status);
> +void init_feature_override(u64 boot_status);
>
> -asmlinkage void __init init_feature_override(const void *fdt, u64 boot_status)
> +asmlinkage void __init init_feature_override(u64 boot_status)
> {
> int i;
>
> @@ -305,7 +310,7 @@ asmlinkage void __init init_feature_override(const void *fdt, u64 boot_status)
>
> __boot_status = boot_status;
>
> - parse_cmdline(fdt);
> + parse_cmdline();
>
> for (i = 0; i < ARRAY_SIZE(regs); i++) {
> if (regs[i]->override)
> diff --git a/arch/arm64/kernel/setup.c b/arch/arm64/kernel/setup.c
> index d0e6c7a291da..fea3223704b6 100644
> --- a/arch/arm64/kernel/setup.c
> +++ b/arch/arm64/kernel/setup.c
> @@ -163,6 +163,21 @@ static void __init smp_build_mpidr_hash(void)
> pr_warn("Large number of MPIDR hash buckets detected\n");
> }
>
> +static void *early_fdt_ptr __initdata;
> +
> +void __init *get_early_fdt_ptr(void)
> +{
> + return early_fdt_ptr;
> +}
> +
> +asmlinkage void __init early_fdt_map(u64 dt_phys)
> +{
> + int fdt_size;
> +
> + early_fixmap_init();
> + early_fdt_ptr = fixmap_remap_fdt(dt_phys, &fdt_size, PAGE_KERNEL);
> +}
> +
> static void __init setup_machine_fdt(phys_addr_t dt_phys)
> {
> int size;
> --
> 2.30.2
>
More information about the linux-arm-kernel
mailing list