Commit 3b8f44fc authored by Ard Biesheuvel's avatar Ard Biesheuvel

efi/libstub/x86: Use Exit() boot service to exit the stub on errors

Currently, we either return with an error [from efi_pe_entry()] or
enter a deadloop [in efi_main()] if any fatal errors occur during
execution of the EFI stub. Let's switch to calling the Exit() EFI boot
service instead in both cases, so that we
a) can get rid of the deadloop, and simply return to the boot manager
   if any errors occur during execution of the stub, including during
   the call to ExitBootServices(),
b) can also return cleanly from efi_pe_entry() or efi_main() in mixed
   mode, once we introduce support for LoadImage/StartImage based mixed
   mode in the next patch.

Note that on systems running downstream GRUBs [which do not use LoadImage
or StartImage to boot the kernel, and instead, pass their own image
handle as the loaded image handle], calling Exit() will exit from GRUB
rather than from the kernel, but this is a tolerable side effect.
Signed-off-by: default avatarArd Biesheuvel <ardb@kernel.org>
parent f7b85b33
...@@ -270,6 +270,11 @@ static inline void *efi64_zero_upper(void *p) ...@@ -270,6 +270,11 @@ static inline void *efi64_zero_upper(void *p)
return p; return p;
} }
static inline u32 efi64_convert_status(efi_status_t status)
{
return (u32)(status | (u64)status >> 32);
}
#define __efi64_argmap_free_pages(addr, size) \ #define __efi64_argmap_free_pages(addr, size) \
((addr), 0, (size)) ((addr), 0, (size))
...@@ -288,6 +293,9 @@ static inline void *efi64_zero_upper(void *p) ...@@ -288,6 +293,9 @@ static inline void *efi64_zero_upper(void *p)
#define __efi64_argmap_locate_device_path(protocol, path, handle) \ #define __efi64_argmap_locate_device_path(protocol, path, handle) \
((protocol), (path), efi64_zero_upper(handle)) ((protocol), (path), efi64_zero_upper(handle))
#define __efi64_argmap_exit(handle, status, size, data) \
((handle), efi64_convert_status(status), (size), (data))
/* PCI I/O */ /* PCI I/O */
#define __efi64_argmap_get_location(protocol, seg, bus, dev, func) \ #define __efi64_argmap_get_location(protocol, seg, bus, dev, func) \
((protocol), efi64_zero_upper(seg), efi64_zero_upper(bus), \ ((protocol), efi64_zero_upper(seg), efi64_zero_upper(bus), \
......
...@@ -144,7 +144,10 @@ union efi_boot_services { ...@@ -144,7 +144,10 @@ union efi_boot_services {
void *); void *);
void *load_image; void *load_image;
void *start_image; void *start_image;
void *exit; efi_status_t __noreturn (__efiapi *exit)(efi_handle_t,
efi_status_t,
unsigned long,
efi_char16_t *);
void *unload_image; void *unload_image;
efi_status_t (__efiapi *exit_boot_services)(efi_handle_t, efi_status_t (__efiapi *exit_boot_services)(efi_handle_t,
unsigned long); unsigned long);
......
...@@ -340,6 +340,13 @@ static void setup_graphics(struct boot_params *boot_params) ...@@ -340,6 +340,13 @@ static void setup_graphics(struct boot_params *boot_params)
} }
} }
static void __noreturn efi_exit(efi_handle_t handle, efi_status_t status)
{
efi_bs_call(exit, handle, status, 0, NULL);
unreachable();
}
void startup_32(struct boot_params *boot_params); void startup_32(struct boot_params *boot_params);
void __noreturn efi_stub_entry(efi_handle_t handle, void __noreturn efi_stub_entry(efi_handle_t handle,
...@@ -369,12 +376,12 @@ efi_status_t __efiapi efi_pe_entry(efi_handle_t handle, ...@@ -369,12 +376,12 @@ efi_status_t __efiapi efi_pe_entry(efi_handle_t handle,
/* Check if we were booted by the EFI firmware */ /* Check if we were booted by the EFI firmware */
if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE) if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE)
return EFI_INVALID_PARAMETER; efi_exit(handle, EFI_INVALID_PARAMETER);
status = efi_bs_call(handle_protocol, handle, &proto, (void *)&image); status = efi_bs_call(handle_protocol, handle, &proto, (void *)&image);
if (status != EFI_SUCCESS) { if (status != EFI_SUCCESS) {
efi_printk("Failed to get handle for LOADED_IMAGE_PROTOCOL\n"); efi_printk("Failed to get handle for LOADED_IMAGE_PROTOCOL\n");
return status; efi_exit(handle, status);
} }
hdr = &((struct boot_params *)efi_table_attr(image, image_base))->hdr; hdr = &((struct boot_params *)efi_table_attr(image, image_base))->hdr;
...@@ -384,7 +391,7 @@ efi_status_t __efiapi efi_pe_entry(efi_handle_t handle, ...@@ -384,7 +391,7 @@ efi_status_t __efiapi efi_pe_entry(efi_handle_t handle,
above4g ? ULONG_MAX : UINT_MAX); above4g ? ULONG_MAX : UINT_MAX);
if (status != EFI_SUCCESS) { if (status != EFI_SUCCESS) {
efi_printk("Failed to allocate lowmem for boot params\n"); efi_printk("Failed to allocate lowmem for boot params\n");
return status; efi_exit(handle, status);
} }
memset(boot_params, 0x0, 0x4000); memset(boot_params, 0x0, 0x4000);
...@@ -442,7 +449,7 @@ efi_status_t __efiapi efi_pe_entry(efi_handle_t handle, ...@@ -442,7 +449,7 @@ efi_status_t __efiapi efi_pe_entry(efi_handle_t handle,
fail: fail:
efi_free(0x4000, (unsigned long)boot_params); efi_free(0x4000, (unsigned long)boot_params);
return status; efi_exit(handle, status);
} }
static void add_e820ext(struct boot_params *params, static void add_e820ext(struct boot_params *params,
...@@ -709,7 +716,7 @@ struct boot_params *efi_main(efi_handle_t handle, ...@@ -709,7 +716,7 @@ struct boot_params *efi_main(efi_handle_t handle,
/* Check if we were booted by the EFI firmware */ /* Check if we were booted by the EFI firmware */
if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE) if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE)
goto fail; efi_exit(handle, EFI_INVALID_PARAMETER);
/* /*
* If the kernel isn't already loaded at the preferred load * If the kernel isn't already loaded at the preferred load
...@@ -793,6 +800,5 @@ struct boot_params *efi_main(efi_handle_t handle, ...@@ -793,6 +800,5 @@ struct boot_params *efi_main(efi_handle_t handle,
fail: fail:
efi_printk("efi_main() failed!\n"); efi_printk("efi_main() failed!\n");
for (;;) efi_exit(handle, status);
asm("hlt");
} }
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