summaryrefslogtreecommitdiffstats
path: root/arch/x86/boot
diff options
context:
space:
mode:
authorBorislav Petkov <bp@suse.de>2019-02-11 12:19:45 +0100
committerBorislav Petkov <bp@suse.de>2019-02-13 12:19:05 +0100
commitf9d230e893e864f13ce5ded9a49990fd024bfed5 (patch)
tree050b669cf3e2bb57408ff6c69cbc32733b959cb5 /arch/x86/boot
parentccec81e4251f5a5421e02874e394338a897056ca (diff)
downloadlinux-0-day-f9d230e893e864f13ce5ded9a49990fd024bfed5.tar.gz
linux-0-day-f9d230e893e864f13ce5ded9a49990fd024bfed5.tar.xz
x86/boot: Correct RSDP parsing with 32-bit EFI
Guenter Roeck reported triple faults of a 64-bit VM using a 32-bit OVMF EFI image. After some singlestepping of the image in gdb, it turned out that some of the EFI config tables were at bogus addresses. Which, as Ard pointed out, results from using the wrong efi_config_table typedef. So switch all EFI table pointers to unsigned longs and convert them to the proper typedef only when accessing them. This way, the proper table type is being used. Shorten variable names, while at it. Fixes: 33f0df8d843d ("x86/boot: Search for RSDP in the EFI tables") Reported-by: Guenter Roeck <linux@roeck-us.net> Signed-off-by: Borislav Petkov <bp@suse.de> Tested-by: Chao Fan <fanc.fnst@cn.fujitsu.com> Cc: Ard Biesheuvel <ard.biesheuvel@linaro.org> Cc: bhe@redhat.com Cc: caoj.fnst@cn.fujitsu.com Cc: "H. Peter Anvin" <hpa@zytor.com> Cc: indou.takao@jp.fujitsu.com Cc: Ingo Molnar <mingo@redhat.com> Cc: kasong@redhat.com Cc: Kees Cook <keescook@chromium.org> Cc: msys.mizuma@gmail.com Cc: Thomas Gleixner <tglx@linutronix.de> Cc: x86-ml <x86@kernel.org> Link: https://lkml.kernel.org/r/20190208190248.GA10854@roeck-us.net
Diffstat (limited to 'arch/x86/boot')
-rw-r--r--arch/x86/boot/compressed/acpi.c50
1 files changed, 31 insertions, 19 deletions
diff --git a/arch/x86/boot/compressed/acpi.c b/arch/x86/boot/compressed/acpi.c
index c5a949335d8b4..0ef4ad55b29b2 100644
--- a/arch/x86/boot/compressed/acpi.c
+++ b/arch/x86/boot/compressed/acpi.c
@@ -50,7 +50,8 @@ static acpi_physical_address efi_get_rsdp_addr(void)
acpi_physical_address rsdp_addr = 0;
#ifdef CONFIG_EFI
- efi_system_table_t *systab;
+ unsigned long systab, systab_tables, config_tables;
+ unsigned int nr_tables;
struct efi_info *ei;
bool efi_64;
int size, i;
@@ -70,46 +71,57 @@ static acpi_physical_address efi_get_rsdp_addr(void)
/* Get systab from boot params. */
#ifdef CONFIG_X86_64
- systab = (efi_system_table_t *)(ei->efi_systab | ((__u64)ei->efi_systab_hi<<32));
+ systab = ei->efi_systab | ((__u64)ei->efi_systab_hi << 32);
#else
if (ei->efi_systab_hi || ei->efi_memmap_hi) {
debug_putstr("Error getting RSDP address: EFI system table located above 4GB.\n");
return 0;
}
- systab = (efi_system_table_t *)ei->efi_systab;
+ systab = ei->efi_systab;
#endif
if (!systab)
error("EFI system table not found.");
- /*
- * Get EFI tables from systab.
- */
- size = efi_64 ? sizeof(efi_config_table_64_t) :
- sizeof(efi_config_table_32_t);
+ /* Handle EFI bitness properly */
+ if (efi_64) {
+ efi_system_table_64_t *stbl = (efi_system_table_64_t *)systab;
+
+ config_tables = stbl->tables;
+ nr_tables = stbl->nr_tables;
+ size = sizeof(efi_config_table_64_t);
+ } else {
+ efi_system_table_32_t *stbl = (efi_system_table_32_t *)systab;
- for (i = 0; i < systab->nr_tables; i++) {
+ config_tables = stbl->tables;
+ nr_tables = stbl->nr_tables;
+ size = sizeof(efi_config_table_32_t);
+ }
+
+ if (!config_tables)
+ error("EFI config tables not found.");
+
+ /* Get EFI tables from systab. */
+ for (i = 0; i < nr_tables; i++) {
acpi_physical_address table;
- void *config_tables;
efi_guid_t guid;
- config_tables = (void *)(systab->tables + size * i);
+ config_tables += size;
+
if (efi_64) {
- efi_config_table_64_t *tmp_table;
+ efi_config_table_64_t *tbl = (efi_config_table_64_t *)config_tables;
- tmp_table = config_tables;
- guid = tmp_table->guid;
- table = tmp_table->table;
+ guid = tbl->guid;
+ table = tbl->table;
if (!IS_ENABLED(CONFIG_X86_64) && table >> 32) {
debug_putstr("Error getting RSDP address: EFI config table located above 4GB.\n");
return 0;
}
} else {
- efi_config_table_32_t *tmp_table;
+ efi_config_table_32_t *tbl = (efi_config_table_32_t *)config_tables;
- tmp_table = config_tables;
- guid = tmp_table->guid;
- table = tmp_table->table;
+ guid = tbl->guid;
+ table = tbl->table;
}
if (!(efi_guidcmp(guid, ACPI_TABLE_GUID)))