?Linux是宏内核的代表;Windows是微内核的代表。
?内核模块是被单独编译的一段代码,可以理解为“应用商店”,其可以动态地加载或卸载。
/*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
?将模块卸载。
?上面的内核模块并非内核模块的一般形式,还有更多可供讨论的细节:
/*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"); //合法协议
?内核模块参数类似于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"
原文:https://www.cnblogs.com/hansenn/p/12737643.html