diff options
Diffstat (limited to 'arch/x86/kernel/e820.c')
-rw-r--r-- | arch/x86/kernel/e820.c | 72 |
1 files changed, 72 insertions, 0 deletions
diff --git a/arch/x86/kernel/e820.c b/arch/x86/kernel/e820.c index 432c49178577..49477484a2fa 100644 --- a/arch/x86/kernel/e820.c +++ b/arch/x86/kernel/e820.c | |||
@@ -1029,4 +1029,76 @@ void __init e820_reserve_resources(void) | |||
1029 | } | 1029 | } |
1030 | } | 1030 | } |
1031 | 1031 | ||
1032 | char *__init __attribute__((weak)) machine_specific_memory_setup(void) | ||
1033 | { | ||
1034 | char *who = "BIOS-e820"; | ||
1035 | int new_nr; | ||
1036 | /* | ||
1037 | * Try to copy the BIOS-supplied E820-map. | ||
1038 | * | ||
1039 | * Otherwise fake a memory map; one section from 0k->640k, | ||
1040 | * the next section from 1mb->appropriate_mem_k | ||
1041 | */ | ||
1042 | new_nr = boot_params.e820_entries; | ||
1043 | sanitize_e820_map(boot_params.e820_map, | ||
1044 | ARRAY_SIZE(boot_params.e820_map), | ||
1045 | &new_nr); | ||
1046 | boot_params.e820_entries = new_nr; | ||
1047 | if (copy_e820_map(boot_params.e820_map, boot_params.e820_entries) < 0) { | ||
1048 | #ifdef CONFIG_X86_64 | ||
1049 | early_panic("Cannot find a valid memory map"); | ||
1050 | #else | ||
1051 | unsigned long mem_size; | ||
1052 | |||
1053 | /* compare results from other methods and take the greater */ | ||
1054 | if (boot_params.alt_mem_k | ||
1055 | < boot_params.screen_info.ext_mem_k) { | ||
1056 | mem_size = boot_params.screen_info.ext_mem_k; | ||
1057 | who = "BIOS-88"; | ||
1058 | } else { | ||
1059 | mem_size = boot_params.alt_mem_k; | ||
1060 | who = "BIOS-e801"; | ||
1061 | } | ||
1062 | |||
1063 | e820.nr_map = 0; | ||
1064 | e820_add_region(0, LOWMEMSIZE(), E820_RAM); | ||
1065 | e820_add_region(HIGH_MEMORY, mem_size << 10, E820_RAM); | ||
1066 | #endif | ||
1067 | } | ||
1068 | |||
1069 | /* In case someone cares... */ | ||
1070 | return who; | ||
1071 | } | ||
1072 | |||
1073 | /* Overridden in paravirt.c if CONFIG_PARAVIRT */ | ||
1074 | char * __init __attribute__((weak)) memory_setup(void) | ||
1075 | { | ||
1076 | return machine_specific_memory_setup(); | ||
1077 | } | ||
1078 | |||
1079 | void __init setup_memory_map(void) | ||
1080 | { | ||
1081 | printk(KERN_INFO "BIOS-provided physical RAM map:\n"); | ||
1082 | e820_print_map(memory_setup()); | ||
1083 | } | ||
1084 | |||
1085 | #ifdef CONFIG_X86_64 | ||
1086 | int __init arch_get_ram_range(int slot, u64 *addr, u64 *size) | ||
1087 | { | ||
1088 | int i; | ||
1032 | 1089 | ||
1090 | if (slot < 0 || slot >= e820.nr_map) | ||
1091 | return -1; | ||
1092 | for (i = slot; i < e820.nr_map; i++) { | ||
1093 | if (e820.map[i].type != E820_RAM) | ||
1094 | continue; | ||
1095 | break; | ||
1096 | } | ||
1097 | if (i == e820.nr_map || e820.map[i].addr > (max_pfn << PAGE_SHIFT)) | ||
1098 | return -1; | ||
1099 | *addr = e820.map[i].addr; | ||
1100 | *size = min_t(u64, e820.map[i].size + e820.map[i].addr, | ||
1101 | max_pfn << PAGE_SHIFT) - *addr; | ||
1102 | return i + 1; | ||
1103 | } | ||
1104 | #endif | ||