你在哪里
走遍千山万水,终于找到了你。
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()有没有点眼熟呢?
我相信你会发现的。
就到这里,希望对你有帮助。
本文永久更新地址://m.ajphoenix.com/linux/24882.html