Linux Kernel是如何做到释放仅在启动时使用的内存的?
你在哪里
走遍千山万水,终于找到了你。
void free_initmem(void) { free_init_pages("unused kernel", (unsigned long)(&__init_begin), (unsigned long)(&__init_end)); }
啥也不说了,看这个函数名大家就知道是释放启动过程中使用的内存空间的。
只是因为在代码中多看了你一眼
static struct memblock_region memblock_memory_init_regions[INIT_MEMBLOCK_REGIONS] __initdata_memblock; static struct memblock_region memblock_reserved_init_regions[INIT_MEMBLOCK_REGIONS] __initdata_memblock;
在看内存管理的代码中,说memblock使用的内存在启动后会又内核回收,继续作为内核可以使用的内存。引发了我一点点的好奇心。 话说内核也是像我们贫下中农一样,挺省吃俭用的嘛。
从代码上看,和普通变量定义时的差异就是这个后缀修饰,__initdata_memblock。
#ifdef CONFIG_ARCH_DISCARD_MEMBLOCK #define __init_memblock __meminit #define __initdata_memblock __meminitdata #else #define __init_memblock #define __initdata_memblock #endif
看来这个还得在内核配置上有所配置才能做到省吃俭用。感觉和人生一样,有的娃出生的时候含着金钥匙,完全不在乎那么点内存;而有的就要默默执行hard模式咯。
作为穷苦的孩子,咱还是学习学习hard模式是怎么个玩儿的哈
#define __meminitdata __section(.meminit.data)
ok,原来就是把这个数据结构定义在了一个叫 .meminit.data的区域。
区域(section)这个高级货大家请自行搜索 elf section哈,因为我也不大记得了。还是不要误导了大家。
幸福像花儿一样
此处略过若干字,在include/asm-generic/vmlinux.lds.h文件中,有如下定义
#if defined(CONFIG_MEMORY_HOTPLUG) #define MEM_KEEP(sec) *(.mem##sec) #define MEM_DISCARD(sec) #else #define MEM_KEEP(sec) #define MEM_DISCARD(sec) *(.mem##sec) #endif /* init and exit section handling */ #define INIT_DATA \ *(.init.data) \ MEM_DISCARD(init.data) \ KERNEL_CTORS() \ ... #define INIT_DATA_SECTION(initsetup_align) \ .init.data : AT(ADDR(.init.data) - LOAD_OFFSET) { \ INIT_DATA \ INIT_SETUP(initsetup_align) \ INIT_CALLS \ CON_INITCALL \ SECURITY_INITCALL \ INIT_RAM_FS \ }
我猜你已经看出点什么来了,如果没用猜出来那再看上述代码一百遍。。。
哦,忘了说了,这里有个东西叫 “gcc链接脚本”,大家请先自行搜之 -_-!
踏破铁鞋无觅处
高兴得太早了,刚才一路顺溜得找到了INIT_DATA_SECTION(),然后,就不知道然后了。。。
关键时候注释起到了不可忽视的作用。所以说,写代码注释很重要。
* SECTIONS * { * . = START; * __init_begin = .; * HEAD_TEXT_SECTION * INIT_TEXT_SECTION(PAGE_SIZE) * INIT_DATA_SECTION(...) * PERCPU_SECTION(CACHELINE_SIZE) * __init_end = .; * * ... * }
原来启动阶段可以释放的内存不仅仅是内存初始化相关的部分,还包括了其他好多东西。嗯,又那么深入了一点点~
好了,你看这里和最开始的free_initmem()有没有点眼熟呢?
我相信你会发现的:)
好了,今天就到这里,希望对你有帮助~