第二章——内核模块

 Linux是宏内核的代表;Windows是微内核的代表。
 内核模块是被单独编译的一段代码,可以理解为“应用商店”,其可以动态地加载或卸载。

2.1、第一个内核模块程序

/*vser.c*/
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.h>

//模块初始化函数
int init_module(void) {
    printk("module init\n");
    return 0;
}

//模块清空函数
void cleanup_module(void) {
    printk("cleanup init\n");
}

 在对应的目录下写makefile即可将此程序生成对应的vser.ko文件。之后可以用:

#insmod vser.ko
#dmesg

 成功后可用dmesg查看控制台输出。

rmmod vser

 将模块卸载。

2.2、内核模块的一般形式

 上面的内核模块并非内核模块的一般形式,还有更多可供讨论的细节:

/*vser.c*/
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.h>

//模块初始化函数
static int __init vser_init(void) {
/*
static为了避免因重名带来的重定义问题,这么写让函数成为了内链接;
模块初始化的函数仅会调用一次,执行完成后内存应该释放,加__init有这个效果。
*/
    printk("vser init\n");
    return 0;
}

//模块清空函数
static void __exit vser_exit(void) {
    printk("vser_exit\n");
}

module_init(vser_init);
module_exit(vser_exit);
MODULE_LICENSE("GPL"); //合法协议

2.3、内核模块参数

 内核模块参数类似于argv的作用,希望对内核模块进行控制。
内核支持的参数类型有:bool, invbool, charp, short int long ushort uint ulong。比如说串口驱动想通过传参控制其波特率等:

#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.h>

static int baudrate = 9600; //波特率
static int port[4] = {0, 1, 2, 3}; //端口
static char *name = "vser";

/*这里将三种类型变量声明为了模块参数*/
module_param(baudrate, int, S_IRUGO);
module_param_array(port, int, NULL, S_IRUGO);
module_param(name, charp, S_IRUGO);

static int __init vser_init(void) {

    int i;
    printk("vser init\n");
	printk("baudrate=%d\n", baudrate);
	printk("port=");
	for(i=0; i<ARRAY_SIZE(port);i++) {
		printk("%d,\n", port[i]);
	}
	printk("name=%s\n", name);
    return 0;
}

static void __exit vser_exit(void) {
    printk("vser_exit\n");
}

module_init(vser_init);
module_exit(vser_exit);
MODULE_LICENSE("GPL");

 如果需要指定模块参数的值,可以用下面的命令:

# modprobe vser baudrate=115200 port=1,2,3,4 name="virtual"

2.4、内核模块和普通程序的差异

  1. 内核模块是操作系统的一部分,运行在内核空间;应用程序运行在用户空间;
  2. 内核模块的函数是被动调用的;应用程序则是顺序执行,通常进入一个循环反复调用一些函数;
  3. 内核模块处于C库之下,不能调用C库的函数,内核有类似的函数提供;
  4. 内核有些清除性的工作需要做;
  5. 内核模块中存在更多的并发,中断、多处理器等……
  6. 整个内核空间调用链上只有4kb或8kb的栈,相对于应用程序来说非常小。如果需要大的内存空间,通常应该动态分配;

猜你喜欢

转载自www.cnblogs.com/hansenn/p/12737643.html