aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/e820.c
diff options
context:
space:
mode:
authorYinghai Lu <yinghai@kernel.org>2010-02-10 04:20:17 -0500
committerH. Peter Anvin <hpa@zytor.com>2010-02-10 20:47:18 -0500
commit28b1c57d3c1f8df69c958f2ae7b9e4b67538ff4d (patch)
treef4893e9e715a73d32e1f1ed6f27c689cd730e35e /arch/x86/kernel/e820.c
parent264ebb182e85f30aa473fa2189d5d5ea173ec3ab (diff)
x86: Dynamically increase early_res array size
Use early_res_count to track the num, and use find_e820 to get a new buffer, then copy from the old to the new one. Also, clear early_res to prevent later invalid usage. -v2 _check_and_double_early_res should take new start Signed-off-by: Yinghai Lu <yinghai@kernel.org> LKML-Reference: <1265793639-15071-14-git-send-email-yinghai@kernel.org> Signed-off-by: H. Peter Anvin <hpa@zytor.com>
Diffstat (limited to 'arch/x86/kernel/e820.c')
-rw-r--r--arch/x86/kernel/e820.c53
1 files changed, 53 insertions, 0 deletions
diff --git a/arch/x86/kernel/e820.c b/arch/x86/kernel/e820.c
index 7053f4adb8e..e09c18c8f3c 100644
--- a/arch/x86/kernel/e820.c
+++ b/arch/x86/kernel/e820.c
@@ -916,6 +916,48 @@ void __init reserve_early_overlap_ok(u64 start, u64 end, char *name)
916 __reserve_early(start, end, name, 1); 916 __reserve_early(start, end, name, 1);
917} 917}
918 918
919static void __init __check_and_double_early_res(u64 start)
920{
921 u64 end, size, mem;
922 struct early_res *new;
923
924 /* do we have enough slots left ? */
925 if ((max_early_res - early_res_count) > max(max_early_res/8, 2))
926 return;
927
928 /* double it */
929 end = max_pfn_mapped << PAGE_SHIFT;
930 size = sizeof(struct early_res) * max_early_res * 2;
931 mem = find_e820_area(start, end, size, sizeof(struct early_res));
932
933 if (mem == -1ULL)
934 panic("can not find more space for early_res array");
935
936 new = __va(mem);
937 /* save the first one for own */
938 new[0].start = mem;
939 new[0].end = mem + size;
940 new[0].overlap_ok = 0;
941 /* copy old to new */
942 if (early_res == early_res_x) {
943 memcpy(&new[1], &early_res[0],
944 sizeof(struct early_res) * max_early_res);
945 memset(&new[max_early_res+1], 0,
946 sizeof(struct early_res) * (max_early_res - 1));
947 early_res_count++;
948 } else {
949 memcpy(&new[1], &early_res[1],
950 sizeof(struct early_res) * (max_early_res - 1));
951 memset(&new[max_early_res], 0,
952 sizeof(struct early_res) * max_early_res);
953 }
954 memset(&early_res[0], 0, sizeof(struct early_res) * max_early_res);
955 early_res = new;
956 max_early_res *= 2;
957 printk(KERN_DEBUG "early_res array is doubled to %d at [%llx - %llx]\n",
958 max_early_res, mem, mem + size - 1);
959}
960
919/* 961/*
920 * Most early reservations come here. 962 * Most early reservations come here.
921 * 963 *
@@ -929,6 +971,8 @@ void __init reserve_early(u64 start, u64 end, char *name)
929 if (start >= end) 971 if (start >= end)
930 return; 972 return;
931 973
974 __check_and_double_early_res(end);
975
932 drop_overlaps_that_are_ok(start, end); 976 drop_overlaps_that_are_ok(start, end);
933 __reserve_early(start, end, name, 0); 977 __reserve_early(start, end, name, 0);
934} 978}
@@ -957,6 +1001,10 @@ void __init early_res_to_bootmem(u64 start, u64 end)
957 for (i = 0; i < max_early_res && early_res[i].end; i++) 1001 for (i = 0; i < max_early_res && early_res[i].end; i++)
958 count++; 1002 count++;
959 1003
1004 /* need to skip first one ?*/
1005 if (early_res != early_res_x)
1006 idx = 1;
1007
960 printk(KERN_INFO "(%d/%d early reservations) ==> bootmem [%010llx - %010llx]\n", 1008 printk(KERN_INFO "(%d/%d early reservations) ==> bootmem [%010llx - %010llx]\n",
961 count - idx, max_early_res, start, end); 1009 count - idx, max_early_res, start, end);
962 for (i = idx; i < count; i++) { 1010 for (i = idx; i < count; i++) {
@@ -974,6 +1022,11 @@ void __init early_res_to_bootmem(u64 start, u64 end)
974 reserve_bootmem_generic(final_start, final_end - final_start, 1022 reserve_bootmem_generic(final_start, final_end - final_start,
975 BOOTMEM_DEFAULT); 1023 BOOTMEM_DEFAULT);
976 } 1024 }
1025 /* clear them */
1026 memset(&early_res[0], 0, sizeof(struct early_res) * max_early_res);
1027 early_res = NULL;
1028 max_early_res = 0;
1029 early_res_count = 0;
977} 1030}
978 1031
979/* Check for already reserved areas */ 1032/* Check for already reserved areas */