SYSINIT是一个通用的调用排序与分别执行机制的框架。FreeBSD目前使用它来进行内核的动态初始化。SYSINIT使得FreeBSD的内核各子系统可以在内核或模块动态加载链接时被重整、添加、删除、替换,这样,内核和模块加载时就不必去修改一个静态的有序初始化安排表甚至重新编译内核。这个体系也使得内核模块(现在称为KLD可以与内核不同时编译、链接、在引导系统时加载,甚至在系统运行时加载。这些操作是通过"内核链接器"(kernel linker)和"链接器集合"(linker set)完成的。链接器集合(Linker Set)是一种链接方法。这种方法将整个程序源文件中静态申明的数据收集到一个可邻近寻址的数据单元中。 SYSINIT要依靠链接器获取遍布整个程序源代码多处申明的静态数据并把它们组成一个彼此相邻的数据块。这种链接方法被称为"链接器集合"(linker set)。SYSINIT使用两个链接器集合以维护两个数据集合,包含每个数据条目的调用顺序、函数、一个会被提交给该函数的数据指针。 
SYSINIT按照两类优先级标识对函数排序以便执行。第一类优先级的标识是子系统的标识,给出SYSINIT分别执行子系统的函数的全局顺序,定义在中的枚举sysinit_sub_id内。第二类优先级标识在子系统中的元素的顺序,定义在中的枚举sysinit_elem_order内。 有两种时刻需要使用SYSINIT:系统启动或内核模块加载时,系统析构或内核模块卸载时。内核子系统通常在系统启动时使用SYSINIT的定义项以初始化数据结构。例如,进程调度子系统使用一个SYSINIT定义项来初始化运行队列数据结构。设备驱动程序应避免直接使用SYSINIT(),对于总线结构上的物理真实设备应使用DRIVER_MODULE()调用的函数先侦测设备的存在,如果存在,再进行设备的初始化。这一系统过程中,会做一些专门针对设备的事情,然后调用SYSINIT()本身。对于非总线结构一部分的虚设备,应改用DEV_MODULE()。

使用SYSINIT

接口

头文件

<sys/kernel.h>

 

SYSINIT(uniquifier, subsystem, order, func, ident)

SYSUNINIT(uniquifier, subsystem, order, func, ident)

启动

宏SYSINIT()在SYSINIT启动数据集合中建立一个SYSINIT数据项,以便SYSINIT在系统启动或模块加载时排序并执行其中的函数。SYSINIT()有一个参数uniquifier,SYSINIT用它来标识数据项,随后是子系统顺序号、子系统元素顺序号、待调用函数、传递给函数的数据。所有的函数必须有一个恒量指针参数。
SYSINIT()的例子

#include <sys/kernel.h>

 

void foo_null(void *unused)

{

foo_doo();

}

SYSINIT(foo, SI_SUB_FOO, SI_ORDER_FOO, foo_null, NULL);

 

struct foo foo_voodoo = {

FOO_VOODOO;

}

 

void foo_arg(void *vdata)

{

struct foo *foo = (struct foo *)vdata;

foo_data(foo);

}

SYSINIT(bar, SI_SUB_FOO, SI_ORDER_FOO, foo_arg, &foo_voodoo);

 

注意,SI_SUB_FOO和SI_ORDER_FOO应当分别在上面提到的枚举sysinit_sub_id和sysinit_elem_order之中。既可以使用已有的枚举项,也可以将自己的枚举项添加到这两个枚举的定义之中。你可以使用数学表达式微调SYSINIT的执行顺序。以下的例子示例了一个需要刚好要在内核参数调整的SYSINIT之前执行的SYSINIT。

调整SYSINIT()顺序的例子

static void

mptable_register(void *dummy __unused)

{

 

apic_register_enumerator(&mptable_enumerator);

}

 

SYSINIT(mptable_register, SI_SUB_TUNABLES - 1, SI_ORDER_FIRST,

mptable_register, NULL);

析构

宏SYSUNINIT()的行为与SYSINIT()的相当,只是它将数据项填加至SYSINIT的析构数据集合。 

SYSUNINIT()的例子

#include <sys/kernel.h>

 

void foo_cleanup(void *unused)

{

foo_kill();

}

SYSUNINIT(foobar, SI_SUB_FOO, SI_ORDER_FOO, foo_cleanup, NULL);

 

struct foo_stack foo_stack = {

FOO_STACK_VOODOO;

}

 

void foo_flush(void *vdata)

{

}

SYSUNINIT(barfoo, SI_SUB_FOO, SI_ORDER_FOO, foo_flush, &foo_stack);

SRC=http://os.51cto.com/art/200511/11446.htm

最新文章

  1. android:使用RemoteView自定义Notification
  2. Linux下Java开发环境搭建—CentOS下Mysql安装教程
  3. intel simd 资料
  4. (DFS)hdoj1241-Oil Deposit
  5. 【Tsinghua OJ】隧道(Tunel)问题
  6. 用C语言实现评论系统设计 - 无数据库版
  7. Error Domain=kCLErrorDomain Code=0 &quot;The operation couldn’t be completed.
  8. On Memory Leaks in Java and in Android.
  9. PHP面向对象基础实例
  10. 枚举for/in
  11. -1-7 java 网络编程基本知识点 计算机网络 TCP/IP协议栈 通信必备 tcp udp
  12. Vagrant将下载好的镜像装载到本地中
  13. 在线客服兼容谷歌Chrome、苹果Safari、Opera浏览器的修改
  14. oracle 函数to_char(数据,&#39;FM999,999,999,999,990.00&#39;) 格式化数据(转)
  15. 谷歌重磅开源强化学习框架Dopamine吊打OpenAI
  16. 类中函数前、后、参数加const
  17. vue项目打包后图片路径问题
  18. nginx 域名(虚拟)部署nodejs项目
  19. python 使用csv 文件写入 出现多余空行数据解决方案
  20. 1.数据结构&amp;算法的引言+时间复杂度

热门文章

  1. C# 获取文件路径,读取项目中某程序集下文件
  2. (嵌入式开发)自己写bootloader之编写第二阶段
  3. ARCGIS动态画点
  4. php实现从尾到头打印列表
  5. [Angular] Test Directive
  6. Notepad++打开xml文件显示crlf的问题
  7. 线上java排查
  8. Make chrome extension
  9. winxp下安装mysql5.7提示mysqld.exe不是有效的win32文件
  10. PDF编译出现错误解决的方法————————【Badboy】