【Linux驱动开发100问】什么是模块?如何编写和使用模块?

news/2024/4/26 12:55:04/文章来源:https://blog.csdn.net/qq_45172832/article/details/129228731

在这里插入图片描述

🥇今日学习目标:什么是Linux内核?
🤵‍♂️ 创作者:JamesBin
⏰预计时间:10分钟
🎉个人主页:嵌入式悦翔园个人主页
🍁专栏介绍:Linux驱动开发100问

什么是模块?如何编写和使用模块?

    • 一、什么是模块
    • 二、如何编写模块
    • 三、如何编译模块
    • 四、如何挂载和卸载模块
    • 五、如何使用模块
    • 六、相关知识

一、什么是模块

在 Linux 内核中,模块是一种动态加载和卸载的可执行文件,它可以在运行时向内核添加新的功能和驱动。 与编译进内核的代码相比,模块的使用可以减少内核的体积和启动时间,并且可以让内核更加灵活,根据需要动态地添加或删除功能。

模块可以是设备驱动、文件系统、网络协议、安全模块等,通过内核提供的模块管理接口,可以动态地插入和删除模块。模块的代码通常是以源码形式提供,需要编译成二进制文件才能加载到内核中运行。模块也可以被其他模块所依赖,形成模块之间的依赖关系。

模块的使用可以帮助我们解决很多问题,例如:

  1. 节省内存空间:模块可以按需加载,不需要一直占用内存,从而节省内存空间。
  2. 扩展内核功能:模块可以添加新的设备驱动程序、文件系统、网络协议等内核功能。
  3. 提高系统安全性:模块可以对内核进行功能增强,从而提高系统的安全性。

总之,模块是Linux内核中一个非常重要的组成部分,可以让内核变得更加灵活、可扩展和可维护。

Linux模块可以是驱动,也可以是其他功能模块。模块和驱动之间存在一定的关系,因为驱动通常也是作为模块的形式存在于内核中的。

在Linux内核中,模块可以被动态地插入和卸载,因此模块通常被用来扩展内核的功能。而驱动则是一种特殊的模块,用于管理硬件设备,控制硬件设备的操作。在Linux中,驱动通常也以模块的形式存在于内核中,从而使得内核可以支持更多的硬件设备。因此,可以说驱动是模块的一种特殊形式。

二、如何编写模块

编写Linux内核模块需要遵循以下步骤:

1、包含必要的头文件

在C文件的开头,需要包含一些头文件,比如<linux/module.h><linux/init.h>等,这些头文件包含了模块开发所需的函数和宏等。

2、编写模块初始化和退出函数

模块初始化函数是模块载入时调用的函数,模块退出函数是模块被卸载时调用的函数。模块初始化函数需要使用宏module_init进行定义,模块退出函数需要使用宏module_exit进行定义。

3、定义模块许可证

Linux内核模块的代码需要遵循一定的许可证,这可以在模块代码中使用宏MODULE_LICENSE进行定义。

4、编写模块代码

这是编写模块最主要的部分。模块代码需要定义模块的功能,并提供接口以便其他程序可以使用这些功能。

下面是一个简单的Linux内核模块代码示例:

#include <linux/module.h>  // 模块头文件
#include <linux/init.h>    // 初始化函数头文件static int __init hello_init(void) // 初始化函数
{printk(KERN_ALERT "Hello, world!\n");  // 打印信息return 0;
}static void __exit hello_exit(void) // 退出函数
{printk(KERN_ALERT "Goodbye, cruel world!\n"); // 打印信息
}MODULE_LICENSE("Dual BSD/GPL"); // 定义许可证module_init(hello_init); // 定义初始化函数
module_exit(hello_exit); // 定义退出函数

三、如何编译模块

编译 Linux 模块可以使用 make 工具来完成,需要使用内核源码目录中的 Makefile 文件。下面是具体步骤:

进入 Linux 内核源码目录,并切换到要编译的内核版本分支。

运行命令 make modules_prepare,该命令会生成 Module.symvers 文件,该文件包含了内核中所有可导出符号的信息,包括符号名、地址和版本等。

进入模块源码所在的目录,并创建一个名为 Makefile 的文件。

在 Makefile 文件中添加以下内容:

obj-m := module_name.o

其中 module_name 表示模块的名称,.o 表示编译后生成的目标文件格式。

运行命令

make -C /lib/modules/$(uname -r)/build M=$(pwd) modules

-C 选项指定内核源码目录,$(uname -r) 表示当前系统运行的内核版本号。
M 选项指定模块源码目录。

如果编译成功,将会生成一个名为 module_name.ko 的文件,该文件即为编译后的模块文件。

四、如何挂载和卸载模块

我们上一步骤通过make命令编译出来的.ko文件即为在最终的模块文件,如果想要使用该模块文件还需要使用 insmod 命令加载模块,例如:

$ sudo insmod module_name.ko

如果需要卸载模块,可以使用 rmmod 命令,例如:

$ sudo rmmod module_name

注意,在加载和卸载模块时需要拥有管理员权限。

为什么要挂载和卸载模块?

在Linux内核中,模块是可以被动态加载和卸载的。当内核启动时,并不会将所有的模块都加载进来,而是按需加载,这样可以提高系统的启动速度和节省内存空间。

当需要使用某个模块时,就需要将它挂载到内核中,使得内核能够调用模块中的功能。而当不再需要某个模块时,可以将其从内核中卸载,释放内存空间。因此,模块的挂载和卸载是为了方便动态管理模块的加载和卸载,提高系统的效率和稳定性。

五、如何使用模块

假设我们已经编写好了一个名为hello_module的内核模块,现在需要编写一个测试程序来调用该模块。下面是一个简单的测试程序代码示例:

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>int main()
{int fd = open("/dev/hello", O_RDWR);if (fd < 0) {perror("Failed to open device /dev/hello");exit(1);}char buf[32];int ret = read(fd, buf, sizeof(buf));if (ret < 0) {perror("Failed to read from device /dev/hello");close(fd);exit(1);}printf("Read from /dev/hello: %s\n", buf);close(fd);return 0;
}

该测试程序的作用是打开内核模块创建的设备文件/dev/hello,然后从设备文件中读取数据并打印到控制台。接下来,我们需要编译和运行这个测试程序。

首先,我们需要将测试程序的源代码保存为一个名为test.c的文件。我们可以使用gcc编译器来编译该程序:

$ gcc -o test test.c

这将生成一个名为test的可执行文件。接下来,需要将该可执行文件拷贝到Linux系统中运行。可以通过SSH等方式将文件传输到Linux系统中。在运行测试程序之前需要为测试程序添加可执行权限:

$ chmod +x test

现在,我们就可以运行测试程序了:

$ ./test

运行程序后,应该能够看到从/dev/hello设备文件中读取的数据被打印到控制台上。

模块内打印的信息可以通过dmesg命令进行打印,可以看到测试程序调用模块的过程。

六、相关知识

除了如何编写、编译和使用模块之外,还需要了解以下知识:

  1. 模块的依赖关系:在编写和使用模块时,需要考虑模块之间的依赖关系。如果一个模块依赖于另一个模块,则必须在前者之前将后者加载。否则,前者将无法正确运行。

  2. 模块的参数传递:有些模块需要在加载时传递参数。这些参数可以用于配置模块的行为或传递信息给模块。模块参数的传递方式有很多种,包括命令行参数、环境变量、配置文件等。

  3. 模块的版本管理:内核版本更新时,模块也需要相应更新。因此,需要了解如何为不同版本的内核编写适配的模块。

  4. 模块的调试:在开发模块时,可能会遇到各种各样的问题。因此,需要了解如何调试模块,例如使用 printk() 输出调试信息、使用 gdb 调试等。

  5. 模块的安全性:模块可以在内核空间执行,因此需要确保模块的安全性。内核提供了各种安全机制,例如模块签名验证、模块加载限制等。

👇点击下方公众号卡片获取资料👇

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_74395.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

分布式之PBFT算法

写在前面 在分布式之拜占庭问题 一文中我们分析了拜占庭问题&#xff0c;并一起看了支持拜占庭容错的口信消息性和签名消息性算法&#xff0c;但是这两种算法都有一个非常严重的问题&#xff0c;就是消息数量太多&#xff0c;通信的成本太大&#xff0c;消息数量复杂度为O(n ^…

HMM(隐马尔科夫模型)-理论补充2

目录 一.大数定理 二.监督学习方法 1.初始概率 2.转移概率 3.观测概率 三.Baum-Welch算法 1.EM算法整体框架 2. Baum-Welch算法 3.EM过程 4.极大化 5.初始状态概率 6.转移概率和观测概率 四.预测算法 1.预测的近似算法 2.Viterbi算法 1.定义 2. 递推&#xff1…

2023安装archlinux笔记

本文只是个笔记&#xff0c;不是详细教程&#xff0c;仅供参考。 安装过程基本与 《2021年vmware安装archlinux》 https://blog.csdn.net/lxyoucan/article/details/115226297 差不多。 无U盘安装 不想格式化U盘了&#xff0c;直接从硬盘安装。参考一下文章。 《没有U盘纯硬…

Laravel框架02:路由与控制器

Laravel框架02&#xff1a;路由与控制器一、路由配置文件二、路由参数三、路由别名四、路由群组五、控制器概述六、控制器路由七、接收用户输入一、路由配置文件 以web网页路由文件为例&#xff1a; 默认根路由 路由定义格式Route::请求方式(请求的URL, 匿名函数或控制响应的方…

CV学习笔记-MobileNet

MobileNet 文章目录MobileNet1. MobileNet概述2. 深度可分离卷积&#xff08;depthwise separable convolution&#xff09;2.1 深度可分离卷积通俗理解2.2 深度可分离卷积对于参数的优化3. MobileNet网络结构4. 代码实现4.1 卷积块4.2 深度可分离卷积块4.3 MobileNet定义4.4 完…

一步步教你电脑变成服务器,tomcat的花生壳设置(原创)

1&#xff0c;首先你去https://console.oray.com/这网站注册个帐号&#xff0c;如果注册成功它会送你一个免费域名&#xff0c;当然不记得也没关系&#xff0c;你记住你注册的 帐号跟密码&#xff0c;然后下载它的软件&#xff08;花生壳动态域名6.0正式版&#xff09;有xp跟li…

java基础系列(六) sleep()和wait() 区别

一.前言 关于并发编程这块, 线程的一些基础知识我们得搞明白, 本篇文章来说一下这两个方法的区别,对Android中的HandlerThread机制原理可以有更深的理解, HandlerThread源码理解,请查看笔者的这篇博客: HandlerThread源码理解_handlerthread 源码_broadview_java的博客-CSDN博…

安装kibana 报错/访问不了

安装kibana 报错1&#xff0c;elasticsearch.yaml 和kibana.yaml 配置问题2&#xff0c;elasticsearch 和kibana版本不一致3&#xff0c;索引问题1&#xff0c;elasticsearch.yaml 和kibana.yaml 配置问题 我的RPM安装的&#xff0c;配置文件都在/etc/ vim /etc/elasticsearc…

【Python基础】类

面向对象编程 面向对象编程是最有效的软件编写方法之一。面向对象是一种对现实世界理解和抽象的方法&#xff0c;是计算机编程技术发展到一定阶段后的产物。 面向对象和面向过程的区别 比如我想吃西红柿炒蛋&#xff0c;怎么运用面向过程的方法来解决这个问题呢&#xff1f;…

《高性能MySQL》——MySQL基准测试(笔记)

文章目录二、MySQL基准测试2.1 为什么需要基准测试2.2 基准测试的策略2.2.1 测试何种指标2.3 基准测试方法2.3.1设计和规划基准测试2.3.2 基准测试应该运行多长时间2.3.3 获取系统性能和状态2.3.4 获得准确的测试结果2.3.5 运行基准测试并分析结果2.3.6 绘图的重要性2.4 基准测…

yii-shopwind商城多数页面报错,修改mysql一个配置就解决!

解决办法打开mysql配置文件&#xff0c;在[mysqld]下添加如下一行&#xff1a;sql_modeNO_ENGINE_SUBSTITUTION,STRICT_TRANS_TABLES成功解决!还以为是网站的代码问题, 惊讶到我了. 开源网站下载下来就报错 多不可思议. 终于是配置的问题!加油报错信息如下是其中一个界面的&…

[MySQL]MySQL数据类型

文章目录数据类型分类数值类型tinyint类型bit类型float类型decimal类型字符串类型char类型varchar类型char和varchar对比日期和时间类型enum和set类型数据类型分类 MySQL中&#xff0c;支持各种各样的类型&#xff0c;比如表示数值的整型浮点型&#xff0c;文本、二进制类型、…

RK3568镜像的拆包和打包

文章目录 前言一、window上分包和打包分包打包二、Linux上分包和打包分包打包总结前言 本文记录在win10上利用瑞芯微提供的工具进行分包和打包,同样也有Linux教程 提示:以下是本篇文章正文内容,下面案例可供参考 一、window上分包和打包 分包 window下一般直接利用工具即…

[oeasy]python0094_视频游戏_双人网球_pong_atari_mos_6502_雅达利_米洛华

编码进化 回忆上次内容 上次 我们回顾了 微软之前的 比尔盖茨和保罗艾伦 mits 迎来的 是帮手还是隐患&#xff1f; intel-8080 遇到了 mos-6502 底层硬件 驱动 游戏行业进化 不光是扑克牌和柏青哥了出现了双人网球 不过 目前的游戏 PDP-1 上的《太空大战》Donner Model 30 上…

信号类型(雷达)——脉冲雷达(三)

系列文章目录 《信号类型&#xff08;雷达通信&#xff09;》 《信号类型&#xff08;雷达&#xff09;——雷达波形认识&#xff08;一&#xff09;》 《信号类型&#xff08;雷达&#xff09;——连续波雷达&#xff08;二&#xff09;》 文章目录 前言 一、相参雷达 1…

从中国文化看面试挑人标准

文章目录标准一、面相1. 1 四白眼1.2 浓眉二、讲话2.1 言多与气虚总结本文结合中国面相&#xff0c;是个概率性问题&#xff0c;对于个体无效。 标准 正直&#xff0c;三观正&#xff0c;沟通好&#xff0c;技术。从概率上讲&#xff1a; 正直且三观正的人----有恒心&#x…

Android OTA 相关工具(四) 查看 payload 文件信息

文章目录1. payload_info.py 的使用1. 环境2. 帮助信息2. 查看 payload 文件信息1. 不带选项查看2. 使用 stats 选项查看3. 使用 signagures 选项4. 使用 list_ops 选项查看3. 其它一直以来&#xff0c;很多人都表达过很想去研究一下 Android OTA 的 payload 文件&#xff0c;看…

Guna Charts WinForm 1.0.8 Crack

Guna Charts 16 图表 在 16 种不同的图表类型中可视化您的数据。 Guna Charts 反应灵敏 轻松响应屏幕尺寸的变化。 Guna Charts 实时图表 创建实时数据仪表板现在非常容易。 Guna Charts 混合图表类型 混合多种图表类型&#xff0c;例如条形图和折线图/面积图。 Guna Charts…

26 openEuler管理网络-使用ip命令配置网络

文章目录26 openEuler管理网络-使用ip命令配置网络26.1 配置IP地址26.1.1 配置静态地址26.1.2 配置多个地址26.2 配置静态路由26 openEuler管理网络-使用ip命令配置网络 说明&#xff1a; 使用ip命令配置的网络配置可以立即生效但系统重启后配置会丢失。 26.1 配置IP地址 使用…

基于stm32电梯管理系统设计

基于stm32电梯管理系统设计这里记录一下以前自己做的嵌入式课程设计&#xff0c;报告中的图片和文字太多了&#xff0c;全部一个一个把搬过来太麻烦了,需要完整文本和代码自行q我963160156&#xff0c;也可在微信公众号 *高级嵌入式软件* 里回复 *电梯* 查看完整版文章摘要关键…