Linux内核网络系统结构图(TCP/IP)和源代码的布局

file: init.h
#define __init_call __attribute__ ((unused,__section__ (".initcall.init")))
#define __exit_call  __attribute__ ((unused,__section__ (".exitcall.exit")))
extern initcall_t __initcall_start, __initcall_end;
#define __initcall(fn)            \
static initcall_t __initcall_##fn __init_call = fn
#define __exitcall(fn)            \
static exitcall_t __exitcall_##fn __exit_call = fn

上面的代码在内核中经常使用到,其中“__attribute__”、“__section__”等就是GCC编译器的关键字。  上面这段程序提供了一种初始化模块的方法:

1、如果一个内核模块需要在系统启动的时候初始化,只需在模块的源程序中调用“module_init( function )”这样的代码,GCC编译器在编译代码时,就会将这个函数的指针放到一个特定的代码段中(“.initcall.init”代码段);

2、Linux启动时,在生成第一个内核线程后,会调用do_initcalls()函数,这个函数的作用就是从“.initcall.init”代码段的首地址开始挨个调用初始化函数(系统定义的初始化函数时没有参数的);
从上面两个步骤可以看出,Linux是借助了GCC编译器的强大功能,使得模块的初始化工作变得简单,不需要到一个专门的地方去注册,只需要在模块中调用“module_init( function )”这样的代码就可以完成工作,GCC能够根据编译选项决定是编译到内核还是作为可加载模块编译。

Linux内核网络系统结构图(TCP/IP)和源代码的布局