一步步写驱动--module_init/module_exit
2012-07-28 22:37
253 查看
在前面helloworld的编写里面,我们使用了两个宏分别是module_init和module_exit,这里分析下为什么使用这两个宏。
在写模块的时候有两个特殊的函数,分别是init_module和cleanup_module,这两个函数分别在insmod的时候和rmmod的时候调用,并且insmod和rmmod只识别这两个特殊的函数,可是我们前面的例子里面并没有这两个函数。怎么会这样呢,那就必须得说说module_init/module_exit了。
一个驱动可以作为一个模块动态的加载到内核里,也可以作为内核的一部分静态的编译进内核,module_init/module_exit也就有了两个含义:
一、动态编译成模块
在内核里有如下定义:/*Each module must use one module_init(). */
#define module_init(initfn) \
staticinline initcall_t __inittest(void) \
{return initfn; } \
intinit_module(void) __attribute__((alias(#initfn)));
/* This is only required if youwant to be unloadable. */
#define module_exit(exitfn) \
staticinline exitcall_t __exittest(void) \
{return exitfn; } \
voidcleanup_module(void) __attribute__((alias(#exitfn)));
首先我们可以发现发现module_init有两个含义:
1、验证加载函数的格式
static inline initcall_t__inittest(void) \
{ return initfn; }
这个函数的作用是验证我们穿过来的加载函数格式是否正确,linux内核规定加载函数的的原型是:
typedef int(*initcall_t)(void);
所以我们写加载函数的时候必须是返回值为int参数为void的函数,这个在内核里要求比较严格,所以我们写加载函数的时候必须按照这个约定。
2、定义别名
intinit_module(void) __attribute__((alias(#initfn)));
这段代码的作用是给我们的加载函数定义一个别名,别名就是我们前面提到的init_module,这样insmod就能够执行我们的加载函数了。
module_exit的作用和module_init一样,同样也是验证函数格式和定义别名。
二、静态编译
在静态编译的时候module_init的定义如下:#define module_init(x) __initcall(x);
#define __initcall(fn)device_initcall(fn)
#definedevice_initcall(fn) __define_initcall("6",fn,6)
#define__define_initcall(level,fn,id) \
static initcall_t __initcall_##fn##id __used \
__attribute__((__section__(".initcall" level".init"))) = fn
通过这些段代码,我们能够看出最终的结果是将我们的使用module_init修饰的函数指针链接到一个叫.initcall的段里,也就是说最终所以的使用module_init修饰的函数指针都被链接在这个段里,最终内核在启动的时候顺序调用所有链接在这个段里的函数,实现设备的初始化。
相关文章推荐
- 驱动之module_init/module_exit
- Linux内核驱动将多个C文件编译成一个ko文件的方法——每个C文件里都有module_init与module_exit,moduleinit
- Linux内核驱动将多个C文件编译成一个ko文件的方法——每个C文件里都有module_init与module_exit
- 驱动之module_init/module_exit
- 驱动之module_init/module_exit
- 驱动之module_init/module_exit
- 驱动之module_init/module_exit
- Linux内核驱动将多个C文件编译成一个ko文件的方法——每一个C文件中都有module_init与module_exit
- 驱动之module_init/module_exit
- ⭐驱动之module_init/module_exit与系统启动关系
- linux驱动 重点:module_init module_exit
- linux驱动的入口函数module_init的加载和释放
- Kernel启动时 驱动是如何加载的module_init,加载的次序如何;略见本文
- android底层驱动学习之 module_init的内核调用顺序
- linux驱动 之 module_init解析 (上)
- linux驱动的入口函数module_init的加载和释放
- Kernel启动时 驱动是如何加载的module_init,加载的次序如何
- module_init和module_exit,init_module和cleanup_module
- linux驱动的入口函数module_init的加载和释放【转】
- linux 驱动module_init()本质--->不同驱动加载顺序对应不同的优先级