[PATCH v2 12/13] armv8: layerscape: relocate spin table if EFI_LOADER is enabled

Heinrich Schuchardt xypron.glpk at gmx.de
Tue Jun 2 22:00:00 CEST 2020


On 6/1/20 9:53 PM, Michael Walle wrote:
> On ARM64, a 64kb region is reserved for the runtime services code.
> Unfortunately, this code overlaps with the spin table code, which also
> needs to be reserved. Thus now that the code is relocatable, allocate a
> new page from EFI, copy the spin table code into it, update any pointers
> to the old region and the start the secondary CPUs.
>
> Signed-off-by: Michael Walle <michael at walle.cc>
> ---
>  arch/arm/cpu/armv8/fsl-layerscape/mp.c | 36 ++++++++++++++++++++++++++
>  1 file changed, 36 insertions(+)
>
> diff --git a/arch/arm/cpu/armv8/fsl-layerscape/mp.c b/arch/arm/cpu/armv8/fsl-layerscape/mp.c
> index d50c5a437b..bd85351705 100644
> --- a/arch/arm/cpu/armv8/fsl-layerscape/mp.c
> +++ b/arch/arm/cpu/armv8/fsl-layerscape/mp.c
> @@ -79,6 +79,10 @@ int fsl_layerscape_wake_seconday_cores(void)
>  	u32 cores, cpu_up_mask = 1;
>  	int i, timeout = 10;
>  	u64 *table;
> +#ifdef CONFIG_EFI_LOADER
> +	u64 reloc_addr = U32_MAX;
> +	efi_status_t ret;
> +#endif
>
>  #ifdef COUNTER_FREQUENCY_REAL
>  	/* update for secondary cores */
> @@ -87,6 +91,38 @@ int fsl_layerscape_wake_seconday_cores(void)
>  			   (unsigned long)&__real_cntfrq + 8);
>  #endif
>
> +#ifdef CONFIG_EFI_LOADER
> +	/*
> +	 * EFI will reserve 64kb for its runtime services. This will probably
> +	 * overlap with our spin table code, which is why we have to relocate
> +	 * it.
> +	 * Keep this after the __real_cntfrq update, so we have it when we
> +	 * copy the complete section here.
> +	 */
> +	ret = efi_allocate_pages(EFI_ALLOCATE_MAX_ADDRESS,
> +				 EFI_RESERVED_MEMORY_TYPE,
> +				 efi_size_in_pages(secondary_boot_code_size),
> +				 &reloc_addr);
> +	if (ret == EFI_SUCCESS) {
> +		debug("Relocating spin table from %llx to %llx (size %lx)\n",
> +		      (u64)secondary_boot_code_start, reloc_addr,
> +		      secondary_boot_code_size);
> +		memcpy((void *)reloc_addr, secondary_boot_code_start,
> +		       secondary_boot_code_size);
> +		flush_dcache_range(reloc_addr,
> +				   reloc_addr + secondary_boot_code_size);
> +
> +		/* set new entry point for secondary cores */
> +		secondary_boot_addr += (void *)reloc_addr -
> +				       secondary_boot_code_start;
> +		flush_dcache_range((unsigned long)&secondary_boot_addr,
> +				   (unsigned long)&secondary_boot_addr + 8);

Wouldn't you want to flush the complete target range of memcpy() and
afterwards call invalidate_icache_all(). At least this is what we do in
other cases after copying instructions. Cf. efi_runtime_relocate().

Best regards

Heinrich

> +
> +		/* this will be used to reserve the memory */
> +		secondary_boot_code_start = (void *)reloc_addr;
> +	}
> +#endif
> +
>  	cores = cpu_mask();
>  	/* Clear spin table so that secondary processors
>  	 * observe the correct value after waking up from wfe.
>



More information about the U-Boot mailing list