当前位置: 首页 > news >正文

初学者的第一个Linux驱动

软件环境:Ubuntu20.04

Linux内核源码:3.4.39

硬件环境:GEC6818

什么是驱动?简单来说就是让硬件工作起来的程序代码。

Linux驱动模块加载有两种方式

1、把写好的驱动代码直接编译进内核。

2、把写好的驱动代码编译成一个可加载的模块,然后再插入到内核中。

我们一般都是使用第二种方式,需要时加载,不需要时卸载,这样方便修改调试。

写驱动程序和应用程序不一样,应用程序出现了问题(像数组越界)会有系统报错该应用程序停止运行,而驱动程序是要加载到内核中出现问题可能会导致整个系统崩溃。

1、内核模块编程注意事项:

1)不能使用C语言库和C语言标准头文件

C语言库和C语言标准头文件(pirntf函数,stdio头文件等)是应用层才有的,而驱动作为底层是没有的。

2)没有内存保护机制

3)不能处理浮点运算

2、内核模块的编写:


#include <linux/module.h>       
#include <linux/kernel.h>        //加载函数
int hello_init(void)   
{printk("Hello World!\n");return 0;
}//卸载函数
void hello_exit(void)
{printk("Bye!\n");
}//声明模块的入口和出口
module_init(hello_init);
module_exit(hello_exit);//GPL模块许可证
MODULE_LICENSE("GPL");
//模块作者
MODULE_AUTHOR("xin");  
//版本号
MODULE_VERSION("1.0");
//描述信息
MODULE_DESCRIPTION("this is a test module!");

首先包含2个驱动必须要的头文件。

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

加载函数

加载函数是没有参数返回值为int的一个函数,其中函数名只要不和其他函数名起冲突随便起,此外返回0表示加载成功。加载模块时自动调用加载函数,当我们把驱动程序加载到内核中第一个就会进入加载函数,有点像应用程序中的main()函数相当于程序的入口。

//加载函数

int xxx(void)   
{
    return 0;
}

卸载函数

卸载函数是没有参数没有返回值的一个函数,函数名也是随便起。卸载模块时自动调用卸载函数,当我们把驱动程序从内核中卸载时,就会进入到卸载函数中。 

//卸载函数
void yyy(void)
{


}

使用宏来修饰加载函数和卸载函数

一个驱动程序中会有多个和加载函数和卸载函数相同结构的函数,怎么样区分呢?就需要使用宏来修饰加载和卸载函数了,只有经过宏修饰的函数才会被认作是加载函数和卸载函数。一个驱动程序中只能有一个加载和卸载函数。

//修饰加载函数

module_init(xxx);

//修饰卸载函数
module_exit(yyy);

注意: printk()是内核中的打印函数,不要和printf()等打印函数搞错了,但两者用法几乎差不多。

//GPL模块许可证

MODULE_LICENSE("GPL");

在编写内核模块时必须加上模块许可证,防止污染内核,造成某些功能无法使用。 "GPL" 是指明了 这是GNU General Public License的任意版本。

//模块作者

MODULE_AUTHOR("xin");  

//版本号

MODULE_VERSION("1.0");

//描述信息

MODULE_DESCRIPTION("this is a test module!");

 除了模块许可证以外,还可以加上模块作者,版本号,描述信息等信息就不一一列举了。

 3、内核模块的编译:

内核模块编译要是用它对应内核的编译方法来进行编译。

就是说要使用开发板中的Linux系统内核来编译,例如使用GEC6818开发板,想要把写好的驱动程序加载到该开发板的内核中,就必须使用GEC6818开发板中Linux系统的内核源码来编译,可以把驱动程序放入开发板中进行编译,也可以把相应的内核源码(必须是编译过的内核)放进Ubuntu中,然后在Ubuntu中进行编译,再把编译好的模块放进开发板中。

编写Makefile文件

为什么要写Makefile文件呢?Makefile是Make读入的唯一配置文件,而Make是一个工程管理器,所谓工程管理器就指用来管理较多文件的。可以想象一下,当有一上百个文件的代码构成的项目,如果其中只有一个或少数几个文件进行了修改,由于编译器不知道哪些文件是最近更新的,只知道需要包含这些文件才能把源代码编译成可执行文件,于是程序员就不得不重新输入数目如此庞大的文件名以完成最后的编译工作。

这样就有了Make工程管理器,实际上就是个”自动编译管理器“,这里的“自动”是指它能够根据文件时间戳自动发现更新过的文件而减少编译的工作量,同时通过读入Makefile文件的内容来执行大量的编译工作。用户只需编写一次简单的编译语句就可以了。他大大提高了实际项目的工作效率,而且几乎所有Linux下的项目编程均会涉及它。

以上大概讲述Make的由来和工作原理已经和Makefile的关系。我们暂时还不能接触到这么大的项目,所以我们的Makefile文件会简单很多。

看一个简单的Makefile文件。

ifeq ($(KERNELRELEASE),)#内核源代码路径
KERNELDIR := /lib/modules/$(shell uname -r)/build
#模块源代码路径
PWD := $(shell pwd)default:$(MAKE) -C $(KERNELDIR) M=$(PWD) modules
clean:rm -rf *.o *.ko *.mod .*.cmd *.mod.* modules.order Module.symvers .tmp_versionselse
#obj-m表示编译生成可加载模块,obj-y表示直接将模块编译进内核。
obj-m := hello.oendif

ifeq ($(KERNELRELEASE),) 

判断变量KERNELRELEASE是否为空,KERNELRELEASE是在内核源码的顶层Makefile中定义的一个变量,在第一次读取执行此Makefile时,KERNELRELEASE没有被定义,所以执行下面的代码。

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

定义一个变量KERNELDIR 来存放内核源码的路径,其中$(shell uname -r)是使用shell命令来打印系统的内核版本号。

 这样就有完整的路径找到系统内核源码。

PWD := $(shell pwd) 

 定义一个变量PWD来存放当前模块代码的路径。

$(MAKE) -C $(KERNELDIR) M=$(PWD) modules

$(MAKE)就是make。

-C 参数告诉make把工作目录切换到 /lib/modules/$(shell uname -r)/build/目录,然后首先解析该目录下的顶层makefile。这保证了当前编译的模块与内核是适配的——使用相同的编译连接参数,同时KERNELRELEASE会被定义。

然后是M参数M=$(PWD),内核使用这个变量来确定要构建的外部模块的目录,完成内核的编译配置的读取后,在这个目录里完成模块的编译。

其实在指令中module表明的意思是把驱动编译成模块

rm -rf *.o *.ko *.mod .*.cmd *.mod.* modules.order Module.symvers .tmp_versions

这就很好理解了,就是把该目录里所有后缀名为o、ko、mod等文件删除。

obj-m := hello.o

obj-m表示以内核模块的形式单独编译,生成可加载模块,最终出现hello.ko的驱动文件。

当使用make命令后面不带参数是执行default语句,而make命令后面带有clean参数则是执行clean语句。Makefile还有许多的语法,大家感兴趣可自行百度。

执行Makefile流程(make):

第一次进入Makefile文件判断KERNELRELEASE为空,执行default语句,会跳转到内核源码的目录解析该目录下的顶层文件,同时KERNELRELEASE会被定义,然后跳转到之前的目录,第二次进入Makefile文件。此时KERNELRELEASE不为空,进入else语句,编译生成可加载模块.ko驱动文件。

执行Makefile流程(make clean):

进入Makefile文件判断KERNELRELEASE为空,执行clean语句,删除目录下带有指定后缀名的文件。

4、模块的使用和基本命令

make:编译模块

 make clean:删除指定后缀名文件

 insmod:加载模块

lsmod:列出内核已载入模块的状态 

 modinfo: 显示内核模块的信息

 dmesg: 显示内核的相关信息

 

 rmmod: 卸载内核中指定的模块

其中加载和卸载模块的指令需要root权限。

以上是最简单的驱动程序和Makefile的编写以及模块的使用和基本命令,以后复杂的驱动程序都是在此基础上增加修改,大家可以在自己Ubuntu上编写测试一下(把该模块加载到开发板上应该会出问题,因为使用的Ubuntu中的内核源码来编译,可以加载到Ubuntu中内核测试一下,后续会有加载到开发板上和驱动程序和Makefile文件的编写教程)。

相关文章:

初学者的第一个Linux驱动

软件环境&#xff1a;Ubuntu20.04 Linux内核源码&#xff1a;3.4.39 硬件环境&#xff1a;GEC6818 什么是驱动&#xff1f;简单来说就是让硬件工作起来的程序代码。 Linux驱动模块加载有两种方式&#xff1a; 1、把写好的驱动代码直接编译进内核。 2、把写好的驱动代码编…...

7. 拼数

1 题目描述 拼数成绩10开启时间2021年09月24日 星期五 18:00折扣0.8折扣时间2021年11月15日 星期一 00:00允许迟交否关闭时间2021年11月23日 星期二 00:00 设有 n个正整数 a[1]​…a[n]​&#xff0c;将它们联接成一排&#xff0c;相邻数字首尾相接&#xff0c;组成一个最大的整…...

Java每天15道面试题 | Redis

redis 和 和 memcached 什么区别&#xff1f;为什么高并发下有时单线程的 redis 比多线程的memcached 效率要高&#xff1f; 区别&#xff1a; 1.mc 可缓存图片和视频。rd 支持除 k/v 更多的数据结构; 2.rd 可以使用虚拟内存&#xff0c;rd 可持久化和 aof 灾难恢复&#xff0…...

13_pinctrl子系统

总结 pinctrl作为驱动 iomuxc节点在设备树里面 存储全部所需的引脚配置信息 iomux节点匹配pinctrl子系统 控制硬件外设的时候 要知道有哪些gpio 再看gpio有哪些服用寄存器 接着在程序配置gpio相关寄存器 这样搞效率很低 所以用iomux节点保存所有的引脚组 pinctrl驱动起来的时…...

Linux系统对于实施人员的价值

Linux系统对于实施人员的价值 随着互联网的发展&#xff0c;linux系统越来越突显了巨大的作用&#xff0c;很多互联网公司&#xff0c;政府企业&#xff0c;只要用到服务器的地方几乎都能看到linux系统的身影&#xff0c;可以说服务是不是在linux系统跑的代表了企业的技术水平&…...

ForkJoin 和 Stream并行流

还在用 for 循环计算两个数之间所有数的和吗&#xff1f;下面提供两种新方法&#xff01; 1. ForkJoin 1.1 背景 要知道&#xff0c;在一个方法中&#xff0c;如果没有做特殊的处理&#xff0c;那么在方法开始到结束使用的都是同一个线程&#xff0c;无论你的业务有多复杂 那…...

逻辑优化-cofactor

1. 简介 逻辑综合中的Cofactor优化方法是一种重要的逻辑优化技术。它通过提取逻辑电路中的共同部分&#xff0c;从而简化电路、减小面积和延迟。该方法广泛应用于电子设计自动化&#xff08;EDA&#xff09;领域中的逻辑综合、等价转换和优化等方面。 Cofactor优化方法最早由…...

车道线检测CondLaneNet论文和源码解读

CondLaneNet: a Top-to-down Lane Detection Framework Based on Conditional Convolution Paper&#xff1a;https://arxiv.org/pdf/2105.05003.pdf code&#xff1a;GitHub - aliyun/conditional-lane-detection 论文解读&#xff1a; 一、摘要 这项工作作为车道线检测任…...

vue3的插槽slots

文章目录普通插槽Test.vueFancyButton.vue具名插槽Test.vueBaseLayout.vue作用域插槽默认插槽Test.vueBaseLayout.vue具名作用域插槽Test.vueBaseLayout.vue普通插槽 父组件使用子组件时&#xff0c;在子组件闭合标签中提供内容模板&#xff0c;插入到子组件定义的出口的地方 …...

docker学校服务器管理

docker 学校服务器管理使用docker&#xff0c;docker使用go语言编写。对于docker的理解&#xff0c;需要知道几个关键字docker, scp&#xff0c;images, container。 docker-码头工人scp-传输命令images/repository-镜像container-容器 docker是码头工人&#xff0c;scp相当…...

pv和pvc

一、PV和PVC详解当前&#xff0c;存储的方式和种类有很多&#xff0c;并且各种存储的参数也需要非常专业的技术人员才能够了解。在Kubernetes集群中&#xff0c;放了方便我们的使用和管理&#xff0c;Kubernetes提出了PV和PVC的概念&#xff0c;这样Kubernetes集群的管理人员就…...

k8s篇之Pod 干预与 PDB

文章目录自愿干预和非自愿干预PDBPDB 示例分离集群所有者和应用程序所有者角色如何在集群上执行中断操作自愿干预和非自愿干预 Pod 不会消失&#xff0c;除非有人&#xff08;用户或控制器&#xff09;将其销毁&#xff0c;或者出现了不可避免的硬件或软件系统错误。 我们把这…...

Django学习17 -- ManytoManyField

1. ManyToManyField &#xff08;参考&#xff1a;Django Documentation Release 4.1.4&#xff09; 类定义 class ManyToManyField(to, **options)使用说明 A many-to-many relationship. Requires a positional argument: the class to which the model is related, which w…...

既然有MySQL了,为什么还要有Redis?

目录专栏导读一、同样是缓存&#xff0c;用map不行吗&#xff1f;二、Redis为什么是单线程的&#xff1f;三、Redis真的是单线程的吗&#xff1f;四、Redis优缺点1、优点2、缺点五、Redis常见业务场景六、Redis常见数据类型1、String2、List3、Hash4、Set5、Zset6、BitMap7、Bi…...

RSTP基础要点(上)

RSTP基础RSTP引入背景STP所存在的问题RSTP对于STP的改进端口角色重新划分端口状态重新划分快速收敛机制&#xff1a;PA机制端口快速切换边缘端口的引入RSTP引入背景 STP协议虽然能够解决环路问题&#xff0c;但是由于网络拓扑收敛较慢&#xff0c;影响了用户通信质量&#xff…...

Linux操作系统学习(信号处理)

文章目录进程信号信号的产生方式&#xff08;信号产生前&#xff09;1. 硬件产生2.调用系统函数向进程发信号3.软件产生4.定位进程崩溃的代码&#xff08;进程异常退出产生信号&#xff09;信号保存的方式&#xff08;信号产生中&#xff09;获取pending表&&修改block表…...

CopyOnWriteArrayList 源码解读

一、CopyOnWriteArrayList 源码解读 在 JUC 中&#xff0c;对于 ArrayList 的线程安全用法&#xff0c;比较推崇于使用 CopyOnWriteArrayList &#xff0c;那 CopyOnWriteArrayList是怎么解决线程安全问题的呢&#xff0c;本文带领大家一起解读下 CopyOnWriteArrayList 的源码…...

方法

方法方法&#xff08;函数&#xff09;一、课前问答二、方法和函数三、方法的参数3.1 单个参数3.2 多个参数四、方法的返回值五、方法的多级调用六、递归方法&#xff08;函数&#xff09; 一、课前问答 1、break和continue的区别 2、嵌套循环的执行流程 3、二进制有哪些运算&…...

C/C++实现发送邮件功能(附源码)

C++常用功能源码系列 本文是C/C++常用功能代码封装专栏的导航贴。部分来源于实战项目中的部分功能提炼,希望能够达到你在自己的项目中拿来就用的效果,这样更好的服务于工作实践。 专栏介绍:专栏讲本人近10年后端开发常用的案例,以高质量的代码提取出来,并对其进行了介绍。…...

Java虚拟机JVM-运行时数据区域说明

及时编译器 HotSpot虚拟机中含有两个即时编译器&#xff0c;分别是编译耗时短但输出代码优化程度较低的客户端编译器&#xff08;简称为C1&#xff09;以及编译耗时长但输出代码优化质量也更高的服务端编译器&#xff08;简称为C2&#xff09;&#xff0c;通常它们会在分层编译…...

设计模式和设计原则回顾

设计模式和设计原则回顾 23种设计模式是设计原则的完美体现,设计原则设计原则是设计模式的理论基石, 设计模式 在经典的设计模式分类中(如《设计模式:可复用面向对象软件的基础》一书中),总共有23种设计模式,分为三大类: 一、创建型模式(5种) 1. 单例模式(Sing…...

条件运算符

C中的三目运算符&#xff08;也称条件运算符&#xff0c;英文&#xff1a;ternary operator&#xff09;是一种简洁的条件选择语句&#xff0c;语法如下&#xff1a; 条件表达式 ? 表达式1 : 表达式2• 如果“条件表达式”为true&#xff0c;则整个表达式的结果为“表达式1”…...

系统设计 --- MongoDB亿级数据查询优化策略

系统设计 --- MongoDB亿级数据查询分表策略 背景Solution --- 分表 背景 使用audit log实现Audi Trail功能 Audit Trail范围: 六个月数据量: 每秒5-7条audi log&#xff0c;共计7千万 – 1亿条数据需要实现全文检索按照时间倒序因为license问题&#xff0c;不能使用ELK只能使用…...

Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; &#x1f680; AI篇持续更新中&#xff01;&#xff08;长期更新&#xff09; 目前2025年06月05日更新到&#xff1a; AI炼丹日志-28 - Aud…...

从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)

设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile&#xff0c;新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...

智能仓储的未来:自动化、AI与数据分析如何重塑物流中心

当仓库学会“思考”&#xff0c;物流的终极形态正在诞生 想象这样的场景&#xff1a; 凌晨3点&#xff0c;某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径&#xff1b;AI视觉系统在0.1秒内扫描包裹信息&#xff1b;数字孪生平台正模拟次日峰值流量压力…...

USB Over IP专用硬件的5个特点

USB over IP技术通过将USB协议数据封装在标准TCP/IP网络数据包中&#xff0c;从根本上改变了USB连接。这允许客户端通过局域网或广域网远程访问和控制物理连接到服务器的USB设备&#xff08;如专用硬件设备&#xff09;&#xff0c;从而消除了直接物理连接的需要。USB over IP的…...

推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材)

推荐 github 项目:GeminiImageApp(图片生成方向&#xff0c;可以做一定的素材) 这个项目能干嘛? 使用 gemini 2.0 的 api 和 google 其他的 api 来做衍生处理 简化和优化了文生图和图生图的行为(我的最主要) 并且有一些目标检测和切割(我用不到) 视频和 imagefx 因为没 a…...

音视频——I2S 协议详解

I2S 协议详解 I2S (Inter-IC Sound) 协议是一种串行总线协议&#xff0c;专门用于在数字音频设备之间传输数字音频数据。它由飞利浦&#xff08;Philips&#xff09;公司开发&#xff0c;以其简单、高效和广泛的兼容性而闻名。 1. 信号线 I2S 协议通常使用三根或四根信号线&a…...

C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)

名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...