From: Ross Lagerwall Date: Wed, 10 Jun 2015 09:57:18 +0000 (+0200) Subject: efi: avoid calling boot services after ExitBootServices() X-Git-Tag: archive/raspbian/4.8.0-1+rpi1~1^2~3120 X-Git-Url: https://dgit.raspbian.org/?a=commitdiff_plain;h=d4300db3a03a0cd999745135d7879fc4b6b5aa61;p=xen.git efi: avoid calling boot services after ExitBootServices() After the first call to ExitBootServices(), avoid calling any boot services (except GetMemoryMap() and ExitBootServices()) by setting setting efi_bs to NULL and halting in blexit(). Only GetMemoryMap() and ExitBootServices() are explicitly allowed to be called after the first call to ExitBootServices() and so are are called via SystemTable->BootServices. Suggested-by: Jan Beulich Signed-off-by: Ross Lagerwall Reviewed-by: Jan Beulich Acked-by: Ian Campbell --- diff --git a/xen/arch/arm/efi/efi-boot.h b/xen/arch/arm/efi/efi-boot.h index 3297f27684..2a7aa13707 100644 --- a/xen/arch/arm/efi/efi-boot.h +++ b/xen/arch/arm/efi/efi-boot.h @@ -6,6 +6,7 @@ #include #include #include +#include void noreturn efi_xen_start(void *fdt_ptr, uint32_t fdt_size); @@ -522,6 +523,11 @@ static void __init efi_arch_blexit(void) efi_bs->FreePool(memmap); } +static void __init efi_arch_halt(void) +{ + stop_cpu(); +} + static void __init efi_arch_load_addr_check(EFI_LOADED_IMAGE *loaded_image) { if ( (unsigned long)loaded_image->ImageBase & ((1 << 12) - 1) ) diff --git a/xen/arch/x86/efi/efi-boot.h b/xen/arch/x86/efi/efi-boot.h index cd14c19bf2..9f417937bf 100644 --- a/xen/arch/x86/efi/efi-boot.h +++ b/xen/arch/x86/efi/efi-boot.h @@ -614,6 +614,13 @@ static void __init efi_arch_blexit(void) efi_bs->FreePages(ucode.addr, PFN_UP(ucode.size)); } +static void __init efi_arch_halt(void) +{ + local_irq_disable(); + for ( ; ; ) + halt(); +} + static void __init efi_arch_load_addr_check(EFI_LOADED_IMAGE *loaded_image) { xen_phys_start = (UINTN)loaded_image->ImageBase; diff --git a/xen/common/efi/boot.c b/xen/common/efi/boot.c index 60c1b8dc32..4b816f2862 100644 --- a/xen/common/efi/boot.c +++ b/xen/common/efi/boot.c @@ -216,6 +216,9 @@ static void __init noreturn blexit(const CHAR16 *str) PrintStr((CHAR16 *)str); PrintStr(newline); + if ( !efi_bs ) + efi_arch_halt(); + if ( cfg.addr ) efi_bs->FreePages(cfg.addr, PFN_UP(cfg.size)); if ( kernel.addr ) @@ -1063,8 +1066,10 @@ efi_start(EFI_HANDLE ImageHandle, EFI_SYSTEM_TABLE *SystemTable) for ( retry = 0; ; retry = 1 ) { efi_memmap_size = map_alloc_size; - status = efi_bs->GetMemoryMap(&efi_memmap_size, efi_memmap, &map_key, - &efi_mdesc_size, &mdesc_ver); + status = SystemTable->BootServices->GetMemoryMap(&efi_memmap_size, + efi_memmap, &map_key, + &efi_mdesc_size, + &mdesc_ver); if ( EFI_ERROR(status) ) PrintErrMesg(L"Cannot obtain memory map", status); @@ -1073,7 +1078,9 @@ efi_start(EFI_HANDLE ImageHandle, EFI_SYSTEM_TABLE *SystemTable) efi_arch_pre_exit_boot(); - status = efi_bs->ExitBootServices(ImageHandle, map_key); + status = SystemTable->BootServices->ExitBootServices(ImageHandle, + map_key); + efi_bs = NULL; if ( status != EFI_INVALID_PARAMETER || retry ) break; }