您当前的位置:首页 > 电脑百科 > 软件技术 > 操作系统 > linux

教你编写你的第一个Linux 内核模块“hello_module”

时间:2020-07-16 10:14:11  来源:  作者:

前言:

linux 内 核 模 块 全 称 为 “ 动 态 可 加 载 内 核 模 块 (Loadable Kernel Module,LKM)”,是系统内核向外部提供的功能插口。作为宏内核结构,Linux 内核具有效率高的特点,但也有可扩展性和可维护性相对较差的不足,Linux 提供模块机制正是弥补这一缺陷。

模块是具有独立功能的程序,它可以被单独编译,但不能独立运行。模块在运行时被链接到内核作为内核的一部分在内核空间运行,这与运行在用户控件的进程是不同的。模块通常有一组函数和数据结构组成,用来实现某种文件系统、驱动程序或其它内核上层功能。

本文将介绍如何编写一个简单的内核模块以及如何传递参数给此模块。

一、 编写一个简单的内核模块

1.编写模块程序

编写如下简单代码,本示例中代码文件命名“hello_module.c”。

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

static int __init hello_init(void){
   printk("This is hello_module, welcome to Linux kernel n");
   return 0;
}

static void __exit hello_exit(void){
   printk("see you next time!n");
}

module_init(hello_init);
module_exit(hello_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Mr Q");
MODULE_DESCRIPTION("hello kernel module");
MODULE_ALIAS("hello");

以上代码解释如下:

(1) #include <linux/module.h>:必须。module.h 头文件包含了对模块的结构定义以及模块的版本控制,任何模块程序的编写都要包含这个头文件;

(2) #include <linux/kernel.h>:kernel.h 包含了常用的内核函数,如以上程序中的 printk()函数;

(3) #include <linux/init.h>:必须。init.h 包含了 module_init()和 module_exit()函数的声明;

(4) module_init():必须。模块加载函数,加载模块式该函数自动执行,进行初始化操作;

(5) module_exit():必须。模块卸载函数,卸载模块时函数自动执行,进行清理操作;

(6) MODULE_LICENSE():表示模块代码接受的软件许可协议。Linux 内核是使用 GPL V2 的开源项目,其要求所有使用和修改了 Linux 内核代码的个人或组织都有义务把修改后的源代码公开,这是一个强制的开源协议,所以一般编写驱动代码都需要显示的声明和遵循本协议,否则内核 UI 发出被污染的警告;

(7) MODULE_AUTHOR():描述模块的作者信息;

(8) MODULE_DESCRIPTION():简单描述模块的用途、功能介绍等;

(9) MODULE_ALIAS():为用户控件提供的别名;

(10) printk():内核输出函数,默认打印系统文件 “ /var/log/kern.log”的内容。

2. 编译内核模块

编写 Makefile 文件,文件名必须为“Makefile”

obj-m := hello_module.o

KERNELBUILD := /lib/modules/$(shell uname -r)/build
CURRENT_PATH := $(shell pwd)

all:
	make -C $(KERNELBUILD) M=$(CURRENT_PATH) modules

clean:
		make -C $(KERNELBUILD) M=$(CURRENT_PATH) clean

以上代码解释如下:

(1) obj-m := <模块名>.o:定义要生成的模块名称

(2) KERNELBUILD := /lib/modules/$(shell uname -r)/build :

KERNELBUILD 为自定义名称,用于指向正在运行 Linux 的内核编译目录,其中“uname -r”标识显示对应的内核版本;

(3) CURRENT_PATH := $(shell pwd):CURRENT_PATH 为自定义名称,用于指向当前当前目录;

(4) all:编译执行的动作

(5) clean:zhixing make clean 需要的动作。“make clean”用于清除上次的 make 命令所产生的 object 文件(后缀为“.o”的文件)及可执行文件。

3.编译

将以上两个文件(hello_module.c 和 Makefile)保存于同一目录下,将上文中代码存放在路径为“/code/hellomodule/”,编译需在文件保存目录中进行。

教你编写你的第一个Linux 内核模块“hello_module”

 

编译成功后,可看到生成的 hello_module.ko 目标文件

教你编写你的第一个Linux 内核模块“hello_module”

 

4.检查编译模块

可通过 file 命令检查编译的模块是否正确,可以看到 x86-64 架构的 elf文件,说明编译成功:

教你编写你的第一个Linux 内核模块“hello_module”

 

也可通过 modinfo 命令进一步检查 :

教你编写你的第一个Linux 内核模块“hello_module”

 

5.插入模块

通过 insmod 命令插入模块,完成插入后可通过 lsmod 命令查看当前模块是否已经被加载到系统中:

教你编写你的第一个Linux 内核模块“hello_module”

 

系统加载模块后,也会在“/sys/module”目录下新建以模块名命名的目录 :

教你编写你的第一个Linux 内核模块“hello_module”

 

6.查看输出

因 本 演示 中 prink()采 用 默认 输出 等级 ,可 通 过“ dmesg” 或“ tail /var/log/kern.log”命令查看输出结果。

教你编写你的第一个Linux 内核模块“hello_module”

 

“ tail /var/log/kern.log”命令查看输出结果:

教你编写你的第一个Linux 内核模块“hello_module”

 

7. 卸载模块

卸载模块,可通过“rmmod 模块名”实现,通 过“ tail /var/log/kern.log”命令查看输出结果。

教你编写你的第一个Linux 内核模块“hello_module”

 

二、 模块参数

1.说明

Linux 内核提供一个宏来实现模块的参数传递

#define module_param(name, type, perm) 
module_param_named(name, name, type, perm)

#define MODULE_PARM_DESC(_parm, desc) 
_MODULE_INFO(parm, _parm, #_parm ":" desc);

module_param()宏由 3 个参数组成,name 表示参数名,type 表示参数类型,perm 表示参数读写权限。MODULE_PARM_DESC()宏提供参数的简单说明,参数类型可为 byte、short、int、long、char、bool 等类型;perm 指定在 sysfs 中相应文件的访问权限,如设置为 0 则不会出现在 sysfs 文件系统中,设置为 0644 标识 root 用户可修改本参数。

static int debug = 1;
module_param(debug, int, 0644);
MODULE_PARM_DESC(debug, "enable debugging information");

#define dprintk(args...) if(debug){printk(KERN_DEBUG args);}

如上述实际代码所示(driver/misc/altera-stap1/altera.c),实际定义模块参数 debug,类型是 int,访问权限是 0644。参数用途是大概调试信息,实际内核编程中常用此方法进行内核调试。

2.开始动手

修改上文中的“hello_module.c”文件,改为以下内容:

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

static int debug = 1;
module_param(debug, int, 0644);
MODULE_PARM_DESC(debug, "debugging information");

#define dprintk(args...) if(debug){printk(KERN_DEBUG args);}
static int myparm = 10;
module_param(myparm, int, 0644);
MODULE_PARM_DESC(myparm, "kernel module parameter experiment.");


static int __init parm_init(void){
   dprintk("my linux kernel module init.n");
   dprintk("module parameter = %dn", myparm);

   return 0;
}

static void __exit parm_exit(void){
   printk("see you next time!n");
}

module_init(parm_init);
module_exit(parm_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Mr Q");
MODULE_DESCRIPTION("kernel module paramter experiment");
MODULE_ALIAS("myparm");

make编译,装载模块,并查看输出:

教你编写你的第一个Linux 内核模块“hello_module”

 

通过查看日志信息,可发现输出以上程序中 参数 的默认值。

卸载模块,赋值重新加载模块,修改参数 myparm 值为 116:

insmod parm_module.ko myparm=116  
教你编写你的第一个Linux 内核模块“hello_module”

 

通过查看日志信息,可发现 参数 值已经改变。



Tags:Linux   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
作用显示文件或目录所占用的磁盘空间使用命令格式du [option] 文件/目录命令功能显示文件或目录所占用的磁盘空间一些写法的区别du -sh xxx 显示总目录的大小,但是不会列出...【详细内容】
2021-12-23  Tags: Linux  点击:(12)  评论:(0)  加入收藏
以下是服务器安全加固的步骤,本文以腾讯云的CentOS7.7版本为例来介绍,如果你使用的是秘钥登录服务器1-5步骤可以跳过。1、设置复杂密码服务器设置大写、小写、特殊字符、数字...【详细内容】
2021-12-20  Tags: Linux  点击:(7)  评论:(0)  加入收藏
大数据技术AI Flink/Spark/Hadoop/数仓,数据分析、面试,源码解读等干货学习资料 98篇原创内容 -->公众号 Linux sed 命令是利用脚本来处理文本文件。sed 可依照脚本的指令来处...【详细内容】
2021-12-17  Tags: Linux  点击:(22)  评论:(0)  加入收藏
简介Netstat 命令用于显示各种网络相关信息,如网络连接,路由表,接口状态 (Interface Statistics),masquerade 连接,多播成员 (Multicast Memberships) 等等。输出信息含义执行net...【详细内容】
2021-12-13  Tags: Linux  点击:(28)  评论:(0)  加入收藏
今天,我们来了解下 Linux 系统的革命性通用执行引擎-eBPF,之所以聊着玩意,因为它确实牛逼,作为一项底层技术,在现在的云原生生态领域中起着举足轻重的作用。截至目前,业界使用范...【详细内容】
2021-12-10  Tags: Linux  点击:(29)  评论:(0)  加入收藏
近日,外媒 ZDNet 评选了适合专业人士使用的 Linux 桌面版操作系统,并排出了前五名。这里的专业人士主要指从事开发、运维、网络安全等行业的人员。开发者首选:Fedora 来自 Red...【详细内容】
2021-12-09  Tags: Linux  点击:(27)  评论:(0)  加入收藏
su和sudo这两个命令是Linux运维必须要会的,通常在生产环境中都是使用普通帐号来登录,再使用su或sudo来执行一些管理命令。su命令su:run a shell with substitute user and grou...【详细内容】
2021-12-07  Tags: Linux  点击:(23)  评论:(0)  加入收藏
0|无效端口,通常用于分析操作系统 1|传输控制协议端口服务多路开关选择器 2|管理实用程序 3|压缩进程 5|远程作业登录 7|回显 9|丢弃 11|在线用户 13|时间 17|每日引用 18|...【详细内容】
2021-12-07  Tags: Linux  点击:(32)  评论:(0)  加入收藏
虽然技术在进步,Linux各发行版本已经更新迭代的很多了,目前Centos、Redhat已经出到8了,Ubuntu已经到21.04了,但我相信一定有很多朋友跟我一样因为各种各样的问题还在使用老久的2...【详细内容】
2021-11-30  Tags: Linux  点击:(41)  评论:(0)  加入收藏
1. 前言本文主要介绍Rocky Linux 8.5的新特性以及如何把当前版本升级至最新版本。Rocky Linux在RHEL 8.5发布不到48小时的时间里,Rocky Linux开发团队成功地对iso镜像进行了...【详细内容】
2021-11-26  Tags: Linux  点击:(75)  评论:(0)  加入收藏
▌简易百科推荐
作用显示文件或目录所占用的磁盘空间使用命令格式du [option] 文件/目录命令功能显示文件或目录所占用的磁盘空间一些写法的区别du -sh xxx 显示总目录的大小,但是不会列出...【详细内容】
2021-12-23  mitsuhide1992    Tags:du命令   点击:(12)  评论:(0)  加入收藏
什么是linux内核linux就像是一个哲学的最佳实践。如果非要对它评价,我真的不知道该怎么赞叹,我只能自豪地说着:“linux的美丽简直让人沉醉。”我只能说是我处在linux学习的修炼...【详细内容】
2021-12-23  linux上的码农    Tags:linux内核   点击:(15)  评论:(0)  加入收藏
本文将比较 Linux 中 service 和 systemctl 命令,先分别简单介绍这两个命令的基础用法,然后进行比较。从 CentOS 7.x 开始,CentOS 开始使用 systemd 服务来代替 service服务(dae...【详细内容】
2021-12-23  软件架构    Tags:systemctl   点击:(14)  评论:(0)  加入收藏
mv是move的缩写,可以用来移动文件或者重命名文件名,经常用来备份文件或者目录。命令格式mv [选项] 源文件或者目录 目标文件或者目录命令功能mv命令中第二个参数类型的不同(...【详细内容】
2021-12-17  入门小站    Tags:mv命令   点击:(23)  评论:(0)  加入收藏
大数据技术AI Flink/Spark/Hadoop/数仓,数据分析、面试,源码解读等干货学习资料 98篇原创内容 -->公众号 Linux sed 命令是利用脚本来处理文本文件。sed 可依照脚本的指令来处...【详细内容】
2021-12-17  仙风道骨的宝石骑士    Tags:sed命令   点击:(22)  评论:(0)  加入收藏
Node是个啥?  写个东西还是尽量面面俱到吧,所以有关基本概念的东西我也从网上选择性地拿了下来,有些地方针对自己的理解有所改动,对这些概念性的东西有过了解的可选择跳过这段...【详细内容】
2021-12-15  linux上的码农    Tags:node   点击:(25)  评论:(0)  加入收藏
难道只有我一个人觉得Ubuntu的unity桌面非常好用吗?最近把台式机上面的Ubuntu 16.04格式化了,装了黑苹果用了一周,不得不说,MacOS确实很精美,软件生态比Linux丰富很多,比Windows简...【详细内容】
2021-12-14  地球末日村    Tags:ubuntu   点击:(40)  评论:(0)  加入收藏
简介Netstat 命令用于显示各种网络相关信息,如网络连接,路由表,接口状态 (Interface Statistics),masquerade 连接,多播成员 (Multicast Memberships) 等等。输出信息含义执行net...【详细内容】
2021-12-13  窥镜天    Tags:Linux netstat   点击:(28)  评论:(0)  加入收藏
对于较多数量的文件描述符的监听无论是select还是poll系统调用都显得捉襟见肘,poll每次都需要将所有的文件描述符复制到内核,内核本身不会对这些文件描述符加以保存,这样的设计...【详细内容】
2021-12-13  深度Linux    Tags:Linux   点击:(19)  评论:(0)  加入收藏
今天,我们来了解下 Linux 系统的革命性通用执行引擎-eBPF,之所以聊着玩意,因为它确实牛逼,作为一项底层技术,在现在的云原生生态领域中起着举足轻重的作用。截至目前,业界使用范...【详细内容】
2021-12-10  架构驿站    Tags:eBPF   点击:(29)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条