Linux内核模块编程(2)


__init和__exit是Linux内核的一个宏定义,使系统在初始化完成后释放该函数,并释放其所占内存。 因此它的优点是显而易见的。所以建议大家啊在编写入口函数和出口函数时采用第二中方法。

3 >现在我们来看一下printk()函数

a.上面已经说了,我们在内核 编程时所用的库函数和在用户态下的是不一样的。 printk 是内核态信息打印函数,功能和比准 C 库的 printf 类似。 printk 还有信息打印级别。

b.现在我们来看一下printk()函数的原型:

int printk(const char *fmt, ...)


消息打印级别:

fmt----消息级别:

#define KERN_EMERG "<0>" /*紧急事件消息,系统崩溃之前提示,表示系统不可用 */

#define KERN_ALERT "<1>" /*报告消息,表示必须立即采取措施 */

#define KERN_CRIT "<2>" /*临界条件,通常涉及严重的硬件或软件操作失败 */

#define KERN_ERR "<3>" /*错误条件,驱动程序常用 KERN_ERR来报告硬件的错误 */

#define KERN_WARNING "<4>" /*警告条件,对可能出现问题的情况进行警告 */

#define KERN_NOTICE "<5>" /*正常但又重要的条件,用于提醒。常用于与安全相关的消息 */

#define KERN_INFO "<6>" /*提示信息,如驱动程序启动时,打印硬件信息 */

#define KERN_DEBUG "<7>" /*调试级别的消息 */

Tiger-John说明:

不同级别使用不同字符串表示,数字越小,级别越高。

c. 为什么内核态使用 printk() 函数,而在用户态使用 printf() 函数。

printk() 函数是直接使用了向终端写函数 tty_write() 。而 printf() 函数是调用 write() 系统调用函数向标准输出设备写。所以在用户态(如进程 0 )不能够直接使用 printk() 函数,而在内核态由于它已是特权级,所以无需系统调用来改变特权级,因而能够直接使用 printk() 函数。 printk 是内核输出,在终端是看不见的。我们可以看一下系统日志。

但是我们可以使用命令:cat  /var/log/syslog ,或者使用 dmesg 命令看一下输出的信息。

第三步: 加载模块和卸载模块 

1>module_init(hello_init)

a.告诉内核你编写模块程序从那里开始执行。

b.module_init()函数中的参数就是注册函数的函数名。


2>module_exit(hello_exit)

a. 告诉内核你编写模块程序从那里离开。

b.module_exit()中的参数名就是卸载函数的函数名。

Tiger-John说明:

我们一般在注册函数里进行一些初始化比如申请内存空间注册设备号等 。那么我们就要在卸载函数进行释放我们所占有的资源。

第四步: 许可权限的声明

1>函数实例:

MODULE_LICENSE("Dual BSD/GPL");

2> 此处可有可无,可以不加系统默认 ( 但是会报警 )

模块声明描述内核模块的许可权限,如果不声明 LICENSE ,模块被加载时,将收到内核的警告。

在 Linux2.6 内核中,可接受的 LICENSE 包括" GPL","GPL v2","GPL and additional rights","Dual BSD/GPL","Dual MPL/GPL","Proprietary"。

第五部: 模块的声明与描述(可加可不加)  

MODULE_AUTHOR(“author”);// 作者

MODULE_DESCRIPTION(“description”);// 描述

MODULE_VERSION(”version_string“);// 版本

MODULE_DEVICE_TABLE(“table_info”);// 设备表

MODULE_ALIAS(”alternate_name“);// 别名

Tiger-John:总结

经过以上五步(其实只要前四步)一个完整的模块编程就完成了。

但是,前面我们已经说过了。内核编程和用户层编程它们之间的编译

链接也不相同。那么我们 如何对模块程序进行编译,链接,运行呢?

现在我么继续深入来学习Makefile文件的编写:

三.   make 的使用以及 Makefile 的编写

1.什么是Makefile,make

1>Makefile是一种脚本,这种脚本主要是用于多文件的编译

2> make 程序可以维护具有相互依赖性的源文件,但某些文件发生改变时,它能自动识别出,

并只对相应文件进行自动编译

2.Makefile的写法

Makefile 文件由五部分组成:显示规则 含规则 变量定义 makefile 指示符和注释


一条Make 的规则原型为:

目标 ... :依赖 ..

命令

...

makefile 中可以使用Shell 命令,例如pwd ,uname

���单的makefile 文件:

obj-m := hello.o

kernel_path=/usr/src/linux-headers-$(shell uname -r)

all:

make -C $(kernel_path) M=$(PWD) modules

clean:

make -C $(kernel_path) M=$(PWD) clean

obj-m:= hello.o // 产生hello 模块的目标

kernel_path // 定义内核源文件目录

all :

make -C $(kernel_path) M=$(PWD) modules

// 生成内核模块参数为内核源代码目录以及模块所在目录

clean:

make -C $(kernel_path) M=$(PWD) clean

// 清除生成的模块文件以及中间文件

Tiger-John说明:

在all和clean下面的一行,即make之前必须用Table符隔开,不能用空

格隔开,否则编译错误。


3.函数实例:

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/wwszfj.html