百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

「技术干货」一文搞懂怎么使用Linux内核模块

ahcoder 2025-03-28 14:24 14 浏览

概述

嵌入式设备驱动开发中将驱动程序以模块的形式发布,更是极大地提高了设备使用的灵活性——用户只需要拿到相关驱动模块,再插入到用户的内核中,即可灵活地使用你的设备。

概述

嵌入式设备驱动开发中将驱动程序以模块的形式发布,更是极大地提高了设备使用的灵活性——用户只需要拿到相关驱动模块,再插入到用户的内核中,即可灵活地使用你的设备。

使用Linux模块的优点

  • 1. 用户可以随时扩展Linux系统的功能。
  • 2. 当要修改当前Linux系统的驱动时,只需要卸载旧模块,编译目标驱动模块,重新安装插入即可。
  • 3. 系统中如果需要使用新模块,不必重新编译内核,只要插入相应的模块即可。
  • 4. 减小Linux内核的体积,节省flash。

Linux模块命令详细介绍

1. 模块安装命令:insmod

insmod xxxx.ko

2. 查看当前已经安装模块:lsmod

lsmod 不需要参数

3. 模块卸载命令:rmmod

rmmod xxxxx.ko

4. 查看模块信息:modinfo

在X86上操作:

[root@zhifachen linux-3.5]# modinfo/root/work/rootfs/home/mod/tiny4412_hello_module.ko
filename: /root/work/rootfs/home/mod/tiny4412_hello_module.ko
license: GPL
depends:
intree: Y
vermagic: 3.5.0-FriendlyARM SMP preempt mod_unload ARMv7 p2v8
[root@zhifachen linux-3.5]#

Linux模块命令测试示例

注意:模块的编译是依赖具体一份源码,并且这份被编译过,并且没有使用mrproper,distclean清除的源码工程。

使用makemodules编译内核源码树中的模块

  • 1. 添加内核菜单
  • 2. make menuconfig 配置为 M
  • 3. 在顶层目录终端输入makemodules

直接make 也可生成模块文件(.ko),也会生成zImage,当你只想编译模块而不想编译zImage时候使用 make modules 命令会更加快速。

把内核源码树中的模块源文件编译到内核映像中

如果不需要配置菜单只想编译代码到内核zImage,Makefile可以这样写obj-y += xxxx.o

如果不需要配置菜单只想编译代码为外部模块,Makefile可以这样写 obj-m+= xxxx.o

在内核源码中添加自定义的模块并且编译1.2.5外部独立Makefile编译模块

在驱动开发阶段,接触到新驱动机率非常高,如果都要去修改内核源码Makefile, 太过于繁杂,也不利于移植,实际开发中会使用另外一种方法来编译模块文件。

Linux内核模块代码文件模板

可以直接复制内核源码中自带的示例模板来测试:
drivers/char/tiny4412_hello_module.c

Linux内核模块编译Makefile模板

编译模块Makefile文件:

obj-m += xyd_hello_module.o
all:
@make -C /root/work/linux-3.5/ M=$(PWD) modules
@rm -rf *.o *.mod.c modules.order Module.symvers *~ *.bak
clean:
@rm -rf *.o *.mod.c modules.order Module.symvers *.ko *~ *.bak
make -C $(KDIR)M=$(PWD) modules

M=选项让该Makefile在构造modules目标之前返回到模块源代码目录,然后modules目标指向obj-m变量中设定的模块;在上面的例子中,我们将该变量设置成了module.o。

上面的清除规则是自己写的,也可利用内核Makefile清除规则来清编译生成目标文件,改进后代码:

obj-m += xyd_hello_module.o
# x86 编译时候使用: uname -r 得到当前内核的版本号
#KDIR := /lib/modules/`uname -r`/build
#arm 编译时候使用
KDIR :=/root/work/linux-3.5
all:
@make -C $(KDIR) M=$(PWD) modules
clean:
@make -C $(KDIR) M=$(PWD) modules clean
@rm -f *.ko.unsigned *~

Linux内核printk函数

内核中不能使用printf函数输出信息,要使用printk函数,这个函数有输出等级控制的。

内核通过 printk() 输出的信息具有日志级别,日志级别是通过在 printk() 输出的字符串前加一个带尖括号的整数来控制的,如 printk("<6>Hello, world!/n");。内核中共提供了八种不同的日志级别,在 linux/kernel.h 中有相应的宏对应。

#defineKERN_EMERG "<0>" /* system is unusable 系统不可用*/
#defineKERN_ALERT "<1>" /* action must be takenimmediately警报,必须立即采取行动*/
#define KERN_CRIT "<2>" /*critical conditions临界状态 */
#defineKERN_ERR "<3>" /* error conditions 错误状态*/
#defineKERN_WARNING "<4>" /* warningconditions警告状态*/
#defineKERN_NOTICE "<5>" /* normal but significant正常的,但引人注目*/
#define KERN_INFO "<6>" /*informational 信息*/
#defineKERN_DEBUG "<7>" /* debug-level messages调试等级信息 */
可以通过查看/proc/sys/kernel/printk文件内容知道设置信息。
[root@ChenZhiFa/]# cat /proc/sys/kernel/printk

可以通过echo命令修改种个等级:

[root@ChenZhiFa/]# echo 4 4 1 7 > /proc/sys/kernel/printk

说明:直接使用printk( “kkkk”);这样没有指明等级,这种未明确指定。

只有等级比较当前终端等级高才会在终端上显示出来。

以下进行X86 系统上的测试(在纯字符界面下测试有效,在图形界面中是无效)

在图形界面系统中按Ctl + Alt + F2(F2~F6键其中一个,不同电脑可能不同),可以进入到纯字符界面,要返回按Ctl +Alt + F1(不同电脑可能不同)

printk --- 不支持浮点数,实际上驱动程序也不支持浮点运算。所有浮点运算应该放在用户空间。


使用Linux模块的优点

  • 1. 用户可以随时扩展Linux系统的功能。
  • 2. 当要修改当前Linux系统的驱动时,只需要卸载旧模块,编译目标驱动模块,重新安装插入即可。
  • 3. 系统中如果需要使用新模块,不必重新编译内核,只要插入相应的模块即可。
  • 4. 减小Linux内核的体积,节省flash。

Linux模块命令详细介绍

1. 模块安装命令:insmod

insmod xxxx.ko

2. 查看当前已经安装模块:lsmod

lsmod 不需要参数

3. 模块卸载命令:rmmod

rmmod xxxxx.ko

4. 查看模块信息:modinfo

在X86上操作:

[root@zhifachen linux-3.5]# modinfo/root/work/rootfs/home/mod/tiny4412_hello_module.ko
filename: /root/work/rootfs/home/mod/tiny4412_hello_module.ko
license: GPL
depends:
intree: Y
vermagic: 3.5.0-FriendlyARM SMP preempt mod_unload ARMv7 p2v8
[root@zhifachen linux-3.5]#

Linux模块命令测试示例

注意:模块的编译是依赖具体一份源码,并且这份被编译过,并且没有使用mrproper,distclean清除的源码工程。

使用makemodules编译内核源码树中的模块

  • 1. 添加内核菜单
  • 2. make menuconfig 配置为 M
  • 3. 在顶层目录终端输入makemodules

直接make 也可生成模块文件(.ko),也会生成zImage,当你只想编译模块而不想编译zImage时候使用 make modules 命令会更加快速。

把内核源码树中的模块源文件编译到内核映像中

如果不需要配置菜单只想编译代码到内核zImage,Makefile可以这样写obj-y += xxxx.o

如果不需要配置菜单只想编译代码为外部模块,Makefile可以这样写 obj-m+= xxxx.o

在内核源码中添加自定义的模块并且编译1.2.5外部独立Makefile编译模块

在驱动开发阶段,接触到新驱动机率非常高,如果都要去修改内核源码Makefile, 太过于繁杂,也不利于移植,实际开发中会使用另外一种方法来编译模块文件。

Linux内核模块代码文件模板

可以直接复制内核源码中自带的示例模板来测试:
drivers/char/tiny4412_hello_module.c

Linux内核模块编译Makefile模板

编译模块Makefile文件:

obj-m += xyd_hello_module.o
all:
@make -C /root/work/linux-3.5/ M=$(PWD) modules
@rm -rf *.o *.mod.c modules.order Module.symvers *~ *.bak
clean:
@rm -rf *.o *.mod.c modules.order Module.symvers *.ko *~ *.bak
make -C $(KDIR)M=$(PWD) modules

M=选项让该Makefile在构造modules目标之前返回到模块源代码目录,然后modules目标指向obj-m变量中设定的模块;在上面的例子中,我们将该变量设置成了module.o。

上面的清除规则是自己写的,也可利用内核Makefile清除规则来清编译生成目标文件,改进后代码:

obj-m += xyd_hello_module.o
# x86 编译时候使用: uname -r 得到当前内核的版本号
#KDIR := /lib/modules/`uname -r`/build
#arm 编译时候使用
KDIR :=/root/work/linux-3.5
all:
@make -C $(KDIR) M=$(PWD) modules
clean:
@make -C $(KDIR) M=$(PWD) modules clean
@rm -f *.ko.unsigned *~

Linux内核printk函数

内核中不能使用printf函数输出信息,要使用printk函数,这个函数有输出等级控制的。

内核通过 printk() 输出的信息具有日志级别,日志级别是通过在 printk() 输出的字符串前加一个带尖括号的整数来控制的,如 printk("<6>Hello, world!/n");。内核中共提供了八种不同的日志级别,在 linux/kernel.h 中有相应的宏对应。

#defineKERN_EMERG "<0>" /* system is unusable 系统不可用*/
#defineKERN_ALERT "<1>" /* action must be takenimmediately警报,必须立即采取行动*/
#define KERN_CRIT "<2>" /*critical conditions临界状态 */
#defineKERN_ERR "<3>" /* error conditions 错误状态*/
#defineKERN_WARNING "<4>" /* warningconditions警告状态*/
#defineKERN_NOTICE "<5>" /* normal but significant正常的,但引人注目*/
#define KERN_INFO "<6>" /*informational 信息*/
#defineKERN_DEBUG "<7>" /* debug-level messages调试等级信息 */
可以通过查看/proc/sys/kernel/printk文件内容知道设置信息。
[root@ChenZhiFa/]# cat /proc/sys/kernel/printk

可以通过echo命令修改种个等级:

[root@ChenZhiFa/]# echo 4 4 1 7 > /proc/sys/kernel/printk

说明:直接使用printk( “kkkk”);这样没有指明等级,这种未明确指定。

只有等级比较当前终端等级高才会在终端上显示出来。

以下进行X86 系统上的测试(在纯字符界面下测试有效,在图形界面中是无效)

在图形界面系统中按Ctl + Alt + F2(F2~F6键其中一个,不同电脑可能不同),可以进入到纯字符界面,要返回按Ctl +Alt + F1(不同电脑可能不同)

printk --- 不支持浮点数,实际上驱动程序也不支持浮点运算。所有浮点运算应该放在用户空间。

相关推荐

安装 Debian 10 后要做的30件事(安装debian sid)

1.修复CD-ROM错误仅从DVD安装Debian的用户才需要此修复程序,因为从DVD安装Debian之后,每次尝试更新系统时都会出现错误“存储库cdrom没有发行文件”:要解决此错误,请打开“软...

Linux 的 18 个装 B 命令,记得全部搂一遍

1、sl命令你会看到一辆火车从屏幕右边开往左边……安装$sudoapt-getinstallsl运行$sl命令有-alFe几个选项,-aAnaccidentseemsto...

Fedora 39提高虚拟机max_map_count以满足一些Steam游戏

有一个关于Fedora39提升其默认vm的建议正在进行中。max_map_count,以满足通过Valve的SteamPlay在Linux上运行的一些Windows游戏。Fedora工程和指导委员...

Linux面板,宝塔Linux正式版 7.9.4发布

【增加】增加日志审计功能【增加】增加对7z格式压缩包的支持【增加】数据库增加PgSQL管理模型【增加】通知设置-消息通道增加微信公众号模块【增加】创建网站时增加一键部署【增加】完善首页【安全风险】的扫...

真榨干Switch!NS被刷Linux系统超频成功运行鬼泣5和战神等游戏

NS的性能已经是被玩家吐槽的不想再吐槽了,都希望任天堂快出新机。而任天堂高管之前则表示switch的性能完全是够用的,是玩家和开发者要求太高了!那么具体情况是如何呢?用实际例子告诉大家switch的真...

Slimbook Manjaro游戏本发布:RTX 4060显卡,Linux定制系统

IT之家2月16日消息,新款SlimbookManjaro游戏笔记本电脑于近日发布,配备英特尔酷睿i7-13620H处理器、英伟达GeForceRTX4060显卡、15.6...

Alienware高层认为Steam游戏机将会为Linux游戏销量带来30倍提升

Diana认为,随着Steam游戏机的推出,将会有越来越多的开发者在Linux上使用OpenGL来开发游戏。Diana预计将有700多部游戏将在SteamOS上的Steam游戏机上发行,其他平台上于...

通过将PS4注入自编译Linux系统,开发者成功运行Steam游戏

本世代的家用机和上世代的一个重要的区别就是架构十分接近x86,而不是PS3/Xbox360世代的的自有架构Cell、Xenon。自从出世以来关于运行PC游戏的讨论一直没有停息,各路好手也不断的尝试更...

在国产优麒麟(Linux)上使用 Steam 畅玩大型游戏

Steam平台是目前全球最大的综合性数字发行平台之一,由Counter-Strike(CS)的开发公司Valve聘请出的BT软件编写者Bram.Cohen亲自开发创建。原先只是作为...

大神给PS4装了“Linux”系统 竟然还能玩Steam游戏!

PS4游戏虽然挺多,但是对比起Steam平台来说,就有点相形见绌了。这不日前,就有一位国外大神努力研究之后,竟然给PS4装上了Linux系统,这还不是最关键的,关键的是他竟然还成功的在Steam平台玩...

Linux系统之安装Ninvaders太空入侵者小游戏

Linux系统之安装Ninvaders太空入侵者小游戏1.1Ninvaders小游戏简介1.2项目预览2.1本地环境规划2.2本次实践介绍3.1检查系统镜像源3.2更新软件列表4.1安装...

还未正式发售:《文明7》Linux版已经被破解!

在万众瞩目的期待中,《文明7》这一备受期待的策略游戏巨作即将于2月11日(国区则为2月12日)正式发售。对于许多热爱这一系列的玩家而言,这无疑是一个振奋人心的消息。预购了豪华版和奠基者版的幸运儿们,更...

在国产系统(Linux)上,安装运行Steam游戏详解

前言经过多年发展,Linux已经相当成熟!只是一直以来,使用者相对不多,软件、游戏等厂商不够重视。因此,能直接在Linux上玩的游戏不多!但是随着国产化的推进,越来越多的人开始接触和使用Linux,比...

在 V 社和 NVIDIA 的联手下,Linux 游戏体验已经接近 Windows 了

出于一些原因,越来越多的用户不再喜欢Windows系统。但如果你是一名PC玩家,似乎只有Windows一个选择。其实不然,在Windows的阴影下,一个操作系统正在游戏方面暗自发力,等待...

Linux 游戏中间层 Proton 7.0-4 发布:新增《祖玛的复仇》等

IT之家8月21日消息,Proton是一个基于Wine修改版的工具分发,由Valve设计和提供资金,可兼容Windows平台游戏。近日,Proton7.0-4已在GitHu...