Commit 026b8579 authored by Ard Biesheuvel's avatar Ard Biesheuvel

efi/zboot: arm64: Grab code size from ELF symbol in payload

Instead of relying on a dodgy dd hack to copy the image code size from
the uncompressed image's PE header to the end of the compressed image,
let's grab the code size from the symbol that is injected into the ELF
object by the Kbuild rules that generate the compressed payload.
Signed-off-by: default avatarArd Biesheuvel <ardb@kernel.org>
Acked-by: default avatarMark Rutland <mark.rutland@arm.com>
parent 45dd403d
...@@ -24,21 +24,13 @@ comp-type-$(CONFIG_KERNEL_ZSTD) := zstd22 ...@@ -24,21 +24,13 @@ comp-type-$(CONFIG_KERNEL_ZSTD) := zstd22
# causing the original tools to complain when checking image integrity. # causing the original tools to complain when checking image integrity.
# So disregard it when calculating the payload size in the zimage header. # So disregard it when calculating the payload size in the zimage header.
zboot-method-y := $(comp-type-y)_with_size zboot-method-y := $(comp-type-y)_with_size
zboot-size-len-y := 12 zboot-size-len-y := 4
zboot-method-$(CONFIG_KERNEL_GZIP) := gzip zboot-method-$(CONFIG_KERNEL_GZIP) := gzip
zboot-size-len-$(CONFIG_KERNEL_GZIP) := 8 zboot-size-len-$(CONFIG_KERNEL_GZIP) := 0
# Copy the SizeOfHeaders and SizeOfCode fields from the payload to the end of
# the compressed image. Note that this presupposes a PE header offset of 64
# bytes, which is what arm64, RISC-V and LoongArch use.
quiet_cmd_compwithsize = $(quiet_cmd_$(zboot-method-y))
cmd_compwithsize = $(cmd_$(zboot-method-y)) && ( \
dd status=none if=$< bs=4 count=1 skip=37 ; \
dd status=none if=$< bs=4 count=1 skip=23 ) >> $@
$(obj)/vmlinuz: $(obj)/vmlinux.bin FORCE $(obj)/vmlinuz: $(obj)/vmlinux.bin FORCE
$(call if_changed,compwithsize) $(call if_changed,$(zboot-method-y))
OBJCOPYFLAGS_vmlinuz.o := -I binary -O $(EFI_ZBOOT_BFD_TARGET) $(EFI_ZBOOT_OBJCOPY_FLAGS) \ OBJCOPYFLAGS_vmlinuz.o := -I binary -O $(EFI_ZBOOT_BFD_TARGET) $(EFI_ZBOOT_OBJCOPY_FLAGS) \
--rename-section .data=.gzdata,load,alloc,readonly,contents --rename-section .data=.gzdata,load,alloc,readonly,contents
......
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
#include <linux/efi.h> #include <linux/efi.h>
#include <asm/efi.h> #include <asm/efi.h>
#include <asm/image.h>
#include <asm/memory.h> #include <asm/memory.h>
#include <asm/sysreg.h> #include <asm/sysreg.h>
...@@ -88,9 +89,10 @@ efi_status_t check_platform_features(void) ...@@ -88,9 +89,10 @@ efi_status_t check_platform_features(void)
#define DCTYPE "cvau" #define DCTYPE "cvau"
#endif #endif
u32 __weak code_size;
void efi_cache_sync_image(unsigned long image_base, void efi_cache_sync_image(unsigned long image_base,
unsigned long alloc_size, unsigned long alloc_size)
unsigned long code_size)
{ {
u32 ctr = read_cpuid_effective_cachetype(); u32 ctr = read_cpuid_effective_cachetype();
u64 lsize = 4 << cpuid_feature_extract_unsigned_field(ctr, u64 lsize = 4 << cpuid_feature_extract_unsigned_field(ctr,
...@@ -98,16 +100,21 @@ void efi_cache_sync_image(unsigned long image_base, ...@@ -98,16 +100,21 @@ void efi_cache_sync_image(unsigned long image_base,
/* only perform the cache maintenance if needed for I/D coherency */ /* only perform the cache maintenance if needed for I/D coherency */
if (!(ctr & BIT(CTR_EL0_IDC_SHIFT))) { if (!(ctr & BIT(CTR_EL0_IDC_SHIFT))) {
unsigned long base = image_base;
unsigned long size = code_size;
do { do {
asm("dc " DCTYPE ", %0" :: "r"(image_base)); asm("dc " DCTYPE ", %0" :: "r"(base));
image_base += lsize; base += lsize;
code_size -= lsize; size -= lsize;
} while (code_size >= lsize); } while (size >= lsize);
} }
asm("ic ialluis"); asm("ic ialluis");
dsb(ish); dsb(ish);
isb(); isb();
efi_remap_image(image_base, alloc_size, code_size);
} }
unsigned long __weak primary_entry_offset(void) unsigned long __weak primary_entry_offset(void)
......
...@@ -1066,8 +1066,7 @@ struct screen_info *__alloc_screen_info(void); ...@@ -1066,8 +1066,7 @@ struct screen_info *__alloc_screen_info(void);
void free_screen_info(struct screen_info *si); void free_screen_info(struct screen_info *si);
void efi_cache_sync_image(unsigned long image_base, void efi_cache_sync_image(unsigned long image_base,
unsigned long alloc_size, unsigned long alloc_size);
unsigned long code_size);
struct efi_smbios_record { struct efi_smbios_record {
u8 type; u8 type;
......
...@@ -50,8 +50,7 @@ static unsigned long alloc_preferred_address(unsigned long alloc_size) ...@@ -50,8 +50,7 @@ static unsigned long alloc_preferred_address(unsigned long alloc_size)
} }
void __weak efi_cache_sync_image(unsigned long image_base, void __weak efi_cache_sync_image(unsigned long image_base,
unsigned long alloc_size, unsigned long alloc_size)
unsigned long code_size)
{ {
// Provided by the arch to perform the cache maintenance necessary for // Provided by the arch to perform the cache maintenance necessary for
// executable code loaded into memory to be safe for execution. // executable code loaded into memory to be safe for execution.
...@@ -66,7 +65,7 @@ asmlinkage efi_status_t __efiapi ...@@ -66,7 +65,7 @@ asmlinkage efi_status_t __efiapi
efi_zboot_entry(efi_handle_t handle, efi_system_table_t *systab) efi_zboot_entry(efi_handle_t handle, efi_system_table_t *systab)
{ {
unsigned long compressed_size = _gzdata_end - _gzdata_start; unsigned long compressed_size = _gzdata_end - _gzdata_start;
unsigned long image_base, alloc_size, code_size; unsigned long image_base, alloc_size;
efi_loaded_image_t *image; efi_loaded_image_t *image;
efi_status_t status; efi_status_t status;
char *cmdline_ptr; char *cmdline_ptr;
...@@ -91,13 +90,9 @@ efi_zboot_entry(efi_handle_t handle, efi_system_table_t *systab) ...@@ -91,13 +90,9 @@ efi_zboot_entry(efi_handle_t handle, efi_system_table_t *systab)
efi_info("Decompressing Linux Kernel...\n"); efi_info("Decompressing Linux Kernel...\n");
// SizeOfImage from the compressee's PE/COFF header // SizeOfImage from the compressee's PE/COFF header
alloc_size = round_up(get_unaligned_le32(_gzdata_end - 12), alloc_size = round_up(get_unaligned_le32(_gzdata_end - 4),
EFI_ALLOC_ALIGN); EFI_ALLOC_ALIGN);
// SizeOfHeaders and SizeOfCode from the compressee's PE/COFF header
code_size = get_unaligned_le32(_gzdata_end - 4) +
get_unaligned_le32(_gzdata_end - 8);
// If the architecture has a preferred address for the image, // If the architecture has a preferred address for the image,
// try that first. // try that first.
image_base = alloc_preferred_address(alloc_size); image_base = alloc_preferred_address(alloc_size);
...@@ -140,9 +135,7 @@ efi_zboot_entry(efi_handle_t handle, efi_system_table_t *systab) ...@@ -140,9 +135,7 @@ efi_zboot_entry(efi_handle_t handle, efi_system_table_t *systab)
goto free_image; goto free_image;
} }
efi_cache_sync_image(image_base, alloc_size, code_size); efi_cache_sync_image(image_base, alloc_size);
efi_remap_image(image_base, alloc_size, code_size);
status = efi_stub_common(handle, image, image_base, cmdline_ptr); status = efi_stub_common(handle, image, image_base, cmdline_ptr);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment