这是我研究内核代码一个很初级的东西时,的经过,或许有些错误,希望大家能够指正 哈哈 作者PianoPan Email:[email protected] www.beeship.com
1. 在看linux核心代码的时候看到/init/main.c 里面的do_initcalls函数 static void __init do_initcalls(void) { initcall_t *call;
call = &__initcall_start; do { (*call)(); call++; } while (call < &__initcall_end);
/* Make sure there is no pending stuff from the initcall sequence */ flush_scheduled_tasks(); } 当时就很诧异,在用source insight 查看的时候,没有发现__initcall_start的定义, 然后就很网站搜索,(估计是我搜索水平太菜),没有发现有相关的文章,于是在相关的linux 论坛提问,有人指引我看看/arch/i386/vmlinux.lds lds是什么?晕菜,不清楚,查了资料得知是ld script.看来的研究ld了
2. 察看/arch/i386/vmlinux.lds,发现一段代码 __initcall_start = .; .initcall.init : { *(.initcall.init) } __initcall_end = .; 跟我找的东西相关 使用info ld,察看相关资料,(最终发现太麻烦,到网上找了一个ld.pdf).发现有这么一 段介绍关于c++地联结构造器的使用,和这段用法相同 其含义时,是让__initcall_start指向代码节.initcall.init的节首,而__initcall_end 指向.initcall.init的节尾。 那么第一段代码从程序逻辑上得到了解释。
3。 因为do_initcalls所作的是系统中有关于选择的驱动部分的初始化工作,那么具体是这些 函数指针数据怎样放到了.initcall.init节。 想起来了,还没有使用grep哈哈,在 grep -rn .initcall.init * 发现在include/linux/init.h:83:有这样一个定义 #define __init_call __attribute__ ((unused,__section__ (".initcall.init"))) 娃哈哈哈 终于让我发现了 然后又发现了 #define __initcall(fn) \ static initcall_t __initcall_##fn __init_call = fn
4。 问题是什么是__attribute__??,查找man gcc,找到关于__attribute__的定义 `section ("section-name")' Normally, the compiler places the code it generates in the `text' section. Sometimes, however, you need additional sections, or you need certain particular functions to appear in special sections. The `section' attribute specifies that a function lives in a particular section. For example, the declaration:
extern void foobar (void) __attribute__ ((section ("bar")));
puts the function `foobar' in the `bar' section.
Some file formats do not support arbitrary sections so the `section' attribute is not available on all platforms. If you need to map the entire contents of a module to a particular section, consider using the facilities of the linker instead. 他的意思就是使它建造一个在.initcall.init节的指向初始函数的指针
5。 问题是##是什么意思? 查阅gcc的man page得知,她是用在可变参数使用宏定义的时候的 在这里也就是建立一个变量名称为所指向的函数的名称,并且前面加上 __initcall_.
6. 然后看看成果 在/include/linux/init.c中有发现 #define module_init(x) __initcall(x); 看看很多驱动中都有类似 module_init(usb_init); module_exit(usb_exit); 的代码,哈哈,这下明白了。 
|