Commit 0a46fff2f9108c2c44218380a43a736cf4612541
Committed by
Borislav Petkov
1 parent
f897e60a12
x86/boot/compressed/64: Fix boot on machines with broken E820 table
BIOS on Samsung 500C Chromebook reports very rudimentary E820 table that consists of 2 entries: BIOS-e820: [mem 0x0000000000000000-0x0000000000000fff] usable BIOS-e820: [mem 0x00000000fffff000-0x00000000ffffffff] reserved It breaks logic in find_trampoline_placement(): bios_start lands on the end of the first 4k page and trampoline start gets placed below 0. Detect underflow and don't touch bios_start for such cases. It makes kernel ignore E820 table on machines that doesn't have two usable pages below BIOS_START_MAX. Fixes: 1b3a62643660 ("x86/boot/compressed/64: Validate trampoline placement against E820") Signed-off-by: Kirill A. Shutemov <kirill.shutemov@linux.intel.com> Signed-off-by: Borislav Petkov <bp@suse.de> Cc: "H. Peter Anvin" <hpa@zytor.com> Cc: Ingo Molnar <mingo@redhat.com> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: x86-ml <x86@kernel.org> Link: https://bugzilla.kernel.org/show_bug.cgi?id=203463 Link: https://lkml.kernel.org/r/20190813131654.24378-1-kirill.shutemov@linux.intel.com
Showing 1 changed file with 10 additions and 3 deletions Side-by-side Diff
arch/x86/boot/compressed/pgtable_64.c
... | ... | @@ -72,6 +72,8 @@ |
72 | 72 | |
73 | 73 | /* Find the first usable memory region under bios_start. */ |
74 | 74 | for (i = boot_params->e820_entries - 1; i >= 0; i--) { |
75 | + unsigned long new; | |
76 | + | |
75 | 77 | entry = &boot_params->e820_table[i]; |
76 | 78 | |
77 | 79 | /* Skip all entries above bios_start. */ |
78 | 80 | |
79 | 81 | |
80 | 82 | |
... | ... | @@ -84,15 +86,20 @@ |
84 | 86 | |
85 | 87 | /* Adjust bios_start to the end of the entry if needed. */ |
86 | 88 | if (bios_start > entry->addr + entry->size) |
87 | - bios_start = entry->addr + entry->size; | |
89 | + new = entry->addr + entry->size; | |
88 | 90 | |
89 | 91 | /* Keep bios_start page-aligned. */ |
90 | - bios_start = round_down(bios_start, PAGE_SIZE); | |
92 | + new = round_down(new, PAGE_SIZE); | |
91 | 93 | |
92 | 94 | /* Skip the entry if it's too small. */ |
93 | - if (bios_start - TRAMPOLINE_32BIT_SIZE < entry->addr) | |
95 | + if (new - TRAMPOLINE_32BIT_SIZE < entry->addr) | |
94 | 96 | continue; |
95 | 97 | |
98 | + /* Protect against underflow. */ | |
99 | + if (new - TRAMPOLINE_32BIT_SIZE > bios_start) | |
100 | + break; | |
101 | + | |
102 | + bios_start = new; | |
96 | 103 | break; |
97 | 104 | } |
98 | 105 |