linux 内核bootmem分配器
内核在启动期间,内存管理还没有初始化,bootmem内存分配器在启动期间分配内存
static void __init bootmem_init(void)
{
unsigned long reserved_end;
unsigned long mapstart = ~0UL;
unsigned long bootmap_size;
int i;
/* 全局变量reserved_end记录保留区后第一个空闲页的pfn
__end 是内核内核映像的最后的位置。具体的值可以参考System.map
在我的1b开发板中_end 为0x808567c0
__pa_symbol 为获取物理地址。
__pa_symbol(&_end) 为0x8567c0,其实就是x808567c0最高位清零 在mips体系中
地址0x80000000 ~ 0xA0000000地址,只要将最高位清空就会得到物理地址
Reserved_end 为 0x857
PFN_UP 的意思就是获取物理页帧号 UP 就是四舍五入的意思,在我1b开发板中页 的大小4KB=4*1024B = (1 << 12);而PFN_UP就是将地址左移12位,reserved_end
就是将 0x8567c0 << 12*/
reserved_end = max(init_initrd(),
(unsigned long) PFN_UP(__pa_symbol(&_end)));
/* 全局变量min_low_pfn和max_low_pfn为可用低端内存的最小、最大pfn */
min_low_pfn = ~0UL;
max_low_pfn = 0;
for (i = 0; i < boot_mem_map.nr_map; i++) {
unsigned long start, end;
/* 不是可用的内存区域,跳过 */
if (boot_mem_map.map[i].type != BOOT_MEM_RAM)
continue;
/* 获得这块内存区起止页面的pfn */
start = PFN_UP(boot_mem_map.map[i].addr);
end = PFN_DOWN(boot_mem_map.map[i].addr
+ boot_mem_map.map[i].size);
/*进入kdb可以查看boot_mem_map的值
kdb> md boot_mem_map
0x807c5110 00000