1. 开机后各可用内存和实际物理内存之间的关系
要计算开机之后剩余的总内存,首先必须弄清楚kernel在开机之后出现的几个可用内存变量之间的关系,今天以一个项目的log来分析开机之后memory的计算方式和各可用内存和总的实际内存之间的关系:
我们在开机之后kernel会有一句log如下:
[ 0.000000] <0>-(0)[0:swapper]Memory: 3919424K/4027408K available (12244K kernel code, 1856K rwdata, 5636K rodata, 1056K init, 13278K bss, 107984K reserved)
内存初始化信息的解读如下:
absent:表示不可用的物理内存大小。譬如有一些物理内存被BIOS保留、对kernel是不可用的,这部分物理内存被计入”absent”之中,这里没有出现。
107984k reserved:包括【initrd】和【内核代码及数据】等,其中内核代码和部分数据包含在下列统计值中:
12244k kernel code :表示kernel的代码,属于reserved memory;
1856k rwdata和5636K rodata :表示kernel的可读可写的数据和只读数据大小,属于reserved memory;
1056k init :表示init code和init data,属于reserved memory,但引导完成之后会释放给free memory;
13278k bss:表示kernel bss段的大小;
它们之间的关系如下:
reserved 包括 kernel code, data 和 init,由于它还包括initrd和其它更多的内容,所以reserved远远大于 kernel code + data + init 之和;
此时会有以下几个疑惑:1. 这里的4027408K是从哪里计算出来的,他与实际总内存4G之间差值包含了哪些内容?
2. 3919424K又是如何计算出来的?
经过查看kernel code可以回答上面的两个问题:
void __init mem_init_print_info(const char *str)
{
unsigned long physpages, codesize, datasize, rosize, bss_size;
unsigned long init_code_size, init_data_size;
physpages = get_num_physpages();
codesize = _etext - _stext;
datasize = _edata - _sdata;
rosize = __end_rodata - __start_rodata;
bss_size = __bss_stop - __bss_start;
init_data_size = __init_end - __init_begin;
init_code_size = _einittext - _sinittext;
……
printk("Memory: %luK/%luK available "
"(%luK kernel code, %luK rwdata, %luK rodata, "
"%luK init, %luK bss, %luK reserved"
#ifdef CONFIG_HIGHMEM
", %luK highmem"
#endif
"%s%s)\n",
nr_free_pages() << (PAGE_SHIFT-10), physpages << (PAGE_SHIFT-10),
codesize >> 10, datasize >> 10, rosize