关于__initcall_start的研究(转贴!)

王朝other·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

这是我研究内核代码一个很初级的东西时,的经过,或许有些错误,希望大家能够指正

哈哈

作者PianoPan Email:PianoPan@beeship.com 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);

的代码,哈哈,这下明白了。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航