http://blog.youkuaiyun.com/wh_19910525/article/details/16370863
http://blog.youkuaiyun.com/thl789/article/details/6581146
在内核源代码中,platform 设备的初始化(注册)用<strong>arch_initcall()调用,它的initcall 的level为3</strong>; 而驱动的注册用<strong>module_init()调用,即device_initcall(),它的initcall 的level为6</strong>。 #define module_init(x) __initcall(x); #define __initcall(fn) device_initcall(fn) kernel 初始化时(kernel_init@init/main.c),按照内核链接文件中(arm系统:kernel/arch/arm/vmlinux.lds)的<a target=_blank target="_blank" class="inner-link decor-none" href="http://zhidao.baidu.com/search?word=__init&fr=qb_search_exp&ie=utf8" rel="nofollow" style="color: rgb(51, 102, 153); text-decoration: none;">__init</a> call_start段的序列依次执行, 这样<strong>level小的初始化函数先于level大的初始化函数被调用</strong>。 所以platform设备先被注册,驱动加载时会调用驱动程序中的probe(),扫描系统中已注册的设备,找到匹配设备后将驱动和设备绑定。
比如I2C,适配器先需要 设备注册;
=============================
LINUX内核中有很多的初始化指示标志postcore_initcall(), arch_initcall(), subsys_initcall(), device_initcall(), etc. 这些起什么作用呢?
- 初始化标号
先看这些宏的定义(定义在文件include/linux/init.h中)
2、__define_initcall
这些宏都用到了__define_initcall(),再看看它的定义(同样定义在文件include/linux/init.h中)
这其中initcall_t是函数指针,原型如下,
而属性 __attribute__((__section__())) 则表示把对象放在一个这个由括号中的名称所指代的section中。
所以__define_initcall的含义是:
1) 声明一个名称为__initcall_##fn的函数指针;
2) 将这个函数指针初始化为fn;
3) 编译的时候需要把这个函数指针变量放置到名称为 ".initcall" level ".init"的section中。
3.放置.initcall.init SECTION
明确了__define_initcall的含义,就知道了是分别将这些初始化标号修饰的函数指针放到各自的section中的。
SECTION“.initcall”level”.init”被放入INITCALLS(include/asm-generic/vmlinux.lds.h)
__initcall_start和__initcall_end以及INITCALLS中定义的SECTION都是在arch/xxx/kernel/vmlinux.lds.S中放在.init段的。
4. 初始化.initcallxx.init里的函数
而这些SECTION里的函数在初始化时被顺序执行(init内核线程->do_basic_setup()[main.c#778]->do_initcalls())。
程序(init/main.c文件do_initcalls()函数)如下,do_initcalls()把.initcallXX.init中的函数按顺序都执行一遍。
=======================
http://blog.youkuaiyun.com/fenzhikeji/article/details/6860143