Commit 9f3a5f52 authored by Yinghai Lu's avatar Yinghai Lu Committed by H. Peter Anvin

x86: Make e820_remove_range to handle all covered case

Rusty found on lguest with trim_bios_range, max_pfn is not right anymore, and
looks e820_remove_range does not work right.

[    0.000000] BIOS-provided physical RAM map:
[    0.000000]  LGUEST: 0000000000000000 - 0000000004000000 (usable)
[    0.000000] Notice: NX (Execute Disable) protection missing in CPU or disabled in BIOS!
[    0.000000] DMI not present or invalid.
[    0.000000] last_pfn = 0x3fa0 max_arch_pfn = 0x100000
[    0.000000] init_memory_mapping: 0000000000000000-0000000003fa0000

root cause is: the e820_remove_range doesn't handle the all covered
case.  e820_remove_range(BIOS_START, BIOS_END - BIOS_START, ...)
produces a bogus range as a result.

Make it match e820_update_range() by handling that case too.
Reported-by: default avatarRusty Russell <rusty@rustcorp.com.au>
Signed-off-by: default avatarYinghai Lu <yinghai@kernel.org>
Tested-by: default avatarRusty Russell <rusty@rustcorp.com.au>
LKML-Reference: <4BB18E55.6090903@kernel.org>
Signed-off-by: default avatarH. Peter Anvin <hpa@zytor.com>
parent 8ae06d22
...@@ -519,29 +519,45 @@ u64 __init e820_remove_range(u64 start, u64 size, unsigned old_type, ...@@ -519,29 +519,45 @@ u64 __init e820_remove_range(u64 start, u64 size, unsigned old_type,
printk(KERN_DEBUG "e820 remove range: %016Lx - %016Lx ", printk(KERN_DEBUG "e820 remove range: %016Lx - %016Lx ",
(unsigned long long) start, (unsigned long long) start,
(unsigned long long) end); (unsigned long long) end);
e820_print_type(old_type); if (checktype)
e820_print_type(old_type);
printk(KERN_CONT "\n"); printk(KERN_CONT "\n");
for (i = 0; i < e820.nr_map; i++) { for (i = 0; i < e820.nr_map; i++) {
struct e820entry *ei = &e820.map[i]; struct e820entry *ei = &e820.map[i];
u64 final_start, final_end; u64 final_start, final_end;
u64 ei_end;
if (checktype && ei->type != old_type) if (checktype && ei->type != old_type)
continue; continue;
ei_end = ei->addr + ei->size;
/* totally covered? */ /* totally covered? */
if (ei->addr >= start && if (ei->addr >= start && ei_end <= end) {
(ei->addr + ei->size) <= (start + size)) {
real_removed_size += ei->size; real_removed_size += ei->size;
memset(ei, 0, sizeof(struct e820entry)); memset(ei, 0, sizeof(struct e820entry));
continue; continue;
} }
/* new range is totally covered? */
if (ei->addr < start && ei_end > end) {
e820_add_region(end, ei_end - end, ei->type);
ei->size = start - ei->addr;
real_removed_size += size;
continue;
}
/* partially covered */ /* partially covered */
final_start = max(start, ei->addr); final_start = max(start, ei->addr);
final_end = min(start + size, ei->addr + ei->size); final_end = min(end, ei_end);
if (final_start >= final_end) if (final_start >= final_end)
continue; continue;
real_removed_size += final_end - final_start; real_removed_size += final_end - final_start;
/*
* left range could be head or tail, so need to update
* size at first.
*/
ei->size -= final_end - final_start; ei->size -= final_end - final_start;
if (ei->addr < final_start) if (ei->addr < final_start)
continue; continue;
......
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