Linux内核的early_param原理追踪
来源:互联网 发布:电脑工作备忘录软件 编辑:程序博客网 时间:2024/05/19 04:52
作者:gfree.wind@gmail.com
=========================================================================================================================================================
early_param用于注册内核选项解析的处理函数。与之类似的,__setup也是用于这个目的。在后文会慢慢看出这两者的区别。
先看它们的定义:
#define __setup_param(str, unique_id, fn,early) \
static
const
char
__setup_str_##unique_id[] __initconst \
__aligned(1)= str; \
static
struct
obs_kernel_param __setup_##unique_id \
__used__section(.init.setup) \
__attribute__((aligned((
sizeof
(
long
))))) \
={ __setup_str_##unique_id, fn, early }
#define __setup(str,fn) \
__setup_param(str,fn, fn, 0)
#define early_param(str,fn) \
__setup_param(str,fn, fn, 1)
看这个宏定义,很容易让人眼花缭乱,不如以一个实例来说明一下:
如early_param("debug",nf_debug_setup);
将其展开则为
static
const
char
__setup_str_nf_debug_setup[] __initconst __aligned(1) =
"debug"
;
static
struct
obs_kernel_param __setup_nf_debug_setup __used__section(.init.setup) __attribute__((aligned((
sizeof
(
long
)))))
={ __setup_str_nf_debug_setup, nf_debug_setup, 1 };
这里关键的变量就是__setup_nf_debug_setup,而关键中的关键是__section(.init.setup)。通过__section宏,编译器会将__setup_nf_debug_setup放置在.init.setupsection中。然后我们查看arch/x86/kernel/vmlinux.lds内核链接器的脚本文件。可以找到下面这几行语句
对于内核连接脚本文件,我们不需要十分明白,也可以看出,这里__setup_start指向了.init.setup开头的地址,而__setup_end指向了.init.setup的结束地址。——有兴趣的同学,可以自行查找vmlinux.lds的资料。我对其也是一知半解呵。
搜索__setup_start,可以发现有两个函数do_early_param和obsolete_checksetup会引用它。
其调用流程如下start_kernel->parse_early_param->parse_early_options->parse_args->do_early_param
static
int
__init do_early_param(
char
*param,
char
*val,
const
char
*unused)
{
const
struct
obs_kernel_param *p;
for
(p = __setup_start; p < __setup_end; p++) {
if
((p->early && parameq(param, p->str)) ||
(
strcmp
(param,
"console"
)== 0 &&
strcmp
(p->str,
"earlycon"
)== 0)
){
if
(p->setup_func(val) != 0)
printk(KERN_WARNING
"Malformedearly option '%s'\n"
,param);
}
}
return
0;
}
在do_early_param中,通过__setup_start和__setup_end,它遍历了.init.setup段中的structobs_kernel_param变量。如果p->early为真且为对应的选项字符串,则调用注册的处理函数p->setup_func。
到这里,我们已经明白了early_param的作用机制。
那么利用__setup注册的处理函数,何时会被调用呢?这里简单说明一下:
在start_kernel调用完parse_early_param,其会再次调用parse_args->unknown_bootoption->obsolete_checksetup,这里会调用__setup注册的处理函数。
这也就是Linux内核的两次解析
- Linux内核的early_param原理追踪
- Linux内核的early_param原理追踪
- Linux内核的early_param原理追踪
- Linux内核的early_param原理追踪
- Linux内核的early_param原理追踪
- Linux内核-----early_param原理追踪
- 追踪分析Linux内核的启动过程
- Linux内核追踪[4.3] Ext3的终结
- [内核文档]使用Linux内核的追踪点[Tracepoint]
- Linux kernel mips early_param
- Linux 内核 SMP 代码追踪
- linux内核启动过程追踪
- Linux内核追踪[4.13] AIO的非阻塞优化
- Linux内核追踪[4.13] Blk层的出错处理优化
- Linux内核追踪[4.14] X86的5级页表管理
- linux之early_param()和__setup
- Open()函数的内核追踪
- Linux内核追踪[4.13] Linux Kernel TLS
- 关于printk的分析
- 关于printk的分析
- Linux启动参数及实现 __setup…
- Linux启动参数及实现 __setup…
- Linux内核的early_param原理追踪
- Linux内核的early_param原理追踪
- 2013年12月15日
- 2013年12月15日
- 将博客搬至CSDN
- 将博客搬至CSDN
- IOS后台运行机制详解(二)
- 有限状态机(转)
- C语言创建单链表,输出单链表的内容。
- c语言练习题 模拟n个人参加选举的过程,四个 候选人A、B、C、D。若选举某人直接 输入其编号,最后按获得票数从高到 低排序并输出候选人编号和票数。