Linux驱动开发

news/2024/11/15 14:40:01/

一、驱动分类

Linux中包含三大类驱动:字符设备驱动、块设备驱动和网络设备驱动。其中字符设备驱动是最大的一类驱动,因为字符设备最多,从led到I2C、SPI、音频等都属于字符设备驱动。块设备驱动和网络设备驱动都要比字符设备驱动复杂。因为其比较复杂,所以半导体厂商都已经帮我们写好了,大部分情况下都是可以直接使用的。所谓的块设备驱动就是存储器设备的驱动,比如EMMC、NAND、SD卡和U盘等存储设备

二、字符设备驱动开发

字符设备驱动是Linux驱动中最基本的一类设备驱动,字符设备就是一个一个字节,按照字节流进行读写操作的设备,读写数据是分先后顺序的。我们先来了解Linux下的应用程序是如何调用驱动程序的,如下图:

应用程序运行在用户空间,而Linux驱动属于内核的一部分,因此驱动运行于内核空间。open、close、write和read等这些函数是由C库提供的,在LInux系统中,系统调用作为C库的一部分。

三、Linux设备号

1、设备好的组成

为了方便管理,Linux中每个设备都有一个设备号,设备号由主设备号和次设备号两部分组成,主设备号表示某一个具体的驱动,次设备号表示使用这个驱动的各个设备。Linux提供了一个名为dev_t的数据类型表示设备号,dev_t定义在include/linux/types.h里面。dev_t其实是unsigned int类型,是一个32位的数据类型。其中高12位是主设备号,低20位是次设备号,因此Linux系统中主设备号范围是0~4095

设备号的操作函数

MAJOR用于从dev_t中获取主设备号,将dev_t右移20位即可
MINOR用于从dev_t中获取次设备号,取dev_t的低20位即可
MKDEV用于将给定的主设备号和次设备号的值组合成dev_t类型的设备号

2、设备号的分配

静态设备号:注册字符设备需要给设备指定一个设备号,这个设备号可以是驱动开发者静态指定的设备号,但是要注意该设备号没有被内核开发者分配掉。使用cat/proc/devices命令即可查看当前系统中已经使用的设备号

使用register_chrdev函数注册字符设备的时候只需要给定一个主设备号即可

/* 注册字符设备驱动 */
retvalue = register_chrdev(CHRDEVBASE_MAJOR, CHRDEVBASE_NAME, &chrdevbase_fops);

这是老版本字符设备注册函数,其0~1048575(2^20-1)这个区间的次设备号全部设置为0

动态分配设备号:使用设备号的时候向linux内核申请,需要几个就申请几个,由linux内核分配设备可以使用的设备号。在注册字符设备之前先申请一个设备号,系统会自动给你一个没有被使用的设备号,这样就避免了冲突,卸载驱动的时候释放掉这个设备号即可

int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name)

dev:保存申请到的设备号

baseminor:次设备号起始地址,alloc_chrdev_region可以申请一段连续的多个设备号,这些设备好的主设备号一样,但是次设备号不同,次设备号以baseminor为起始地址开始递增。一般baseminor为0,也就是次设备号从0开始

count:要申请的设备号数量

name:设别名字

如果给定了设备的主设备号和次设备号就使用如下所示函数来注册设备号

int register_chrdev_region(dev_t from, unsigned count, const char *name)
参数 from 是要申请的起始设备号,也就是给定的设备号
参数 count 是要申请的数量,一般都是一个
参数 name 是设备名字

一般采用动态分配设备号,模板如下

/* 注册字符设备驱动 */
/* 1、创建设备号 */
if (newchrled.major) 
{        /*  定义了设备号 */newchrled.devid = MKDEV(newchrled.major, 0);register_chrdev_region(newchrled.devid, NEWCHRLED_CNT, NEWCHRLED_NAME);
} 
else 
{        /* 没有定义设备号 */alloc_chrdev_region(&newchrled.devid, 0, NEWCHRLED_CNT, NEWCHRLED_NAME);    /* 申请设备号 */newchrled.major = MAJOR(newchrled.devid);    /* 获取分配号的主设备号 */newchrled.minor = MINOR(newchrled.devid);    /* 获取分配号的次设备号 */
}
printk("newcheled major=%d,minor=%d\r\n",newchrled.major, newchrled.minor);    

注销设备号

void unregister_chrdev_region(dev_t from, unsigned count)

四、模块注册与卸载

Linux驱动有两种运行方式,第一种是将驱动编译进Linux内核中,当Linux内核启动的时候就会自动运行驱动程序。第二种是将驱动编写成模块(Linux下模块扩展名为ko),在Linux内核启动以后使用相应命令加载驱动模块

模块有加载和卸载两种操作,模块的加载和卸载注册函数如下

module_init(xxx_init); //注册模块加载函数
module_exit(xxx_exit); //注册模块卸载函数

module_init函数来向Linux内核注册一个模块加载函数,参数xxx_init就是需要注册的具体函数,当使用insmod命令加载驱动的时候,xxx_init这个函数就会被调用

module_exit函数用来向Linux内核注册一个模块卸载函数,参数xxx_exit就是需要注册的函数,当使用rmmod命令卸载驱动的时候xxx_exit就会被调用

五、字符设备注册与注销

当驱动模块加载成功需要注册字符设备,卸载驱动模块的时候也需要注销掉字符设备

static inline int register_chrdev(unsigned int major, const char *name,const struct file_operations *fops)
static inline void unregister_chrdev(unsigned int major, const char *name)

register_chrdev函数用于注册字符设备,unregister_chrdev注销掉字符设备

一般字符设备的注册在驱动模块的入口函数xxx_init中进行,字符设备的注销在驱动模块的出口函数xxx_exit进行

六、LICENSE和作者信息

在驱动中加如LICENSE和作者信息,其中LICENSE是必须添加的,否则的话编译会报错,作者信息可以不添加

MODULE_LICENSE("GPL") //添加模块 LICENSE 信息 ,LICENSE 采用 GPL 协议
MODULE_AUTHOR("mingfei") //添加模块作者信息

七、测试指令

1、加载驱动模块

驱动模块一般挂载在lib/modules/4.1.15目录下,所以需要编写的驱动模块和测试软件复制到根文件系统rootfs/lib/modules/4.1.15目录下,通过tftfp和nfs启动后,在开发板的lib/modules/4.1.15目录下存在驱动模块和测试软件

驱动编译完成以后扩展名是.ko,有两种命令可以加载模块:insmod和modprobe

加载驱动模块

insmod xxx.ko
modprobe xxx.ko

insmod不能解决模块的依赖关系,比如drv.ko依赖first.ko这个模块,就必须先使用insmod命令加载first.ko这个模块,然后再加载drv.ko这个模块

但是modprobe就不会存在这样的问题,modprobe会分析模块的依赖关系,然后会将所有的依赖模块都加载到内核中

若modprobe提示无法打开modules.dep,驱动挂载失败,输入depmod即可自动生成modules.dep

2、创建设备节点文件

驱动加载成功需要在/dev目录下创建一个与之对应设备节点文件,应用程序就是通过操作设备节点文件来完成对具体设备的操作

创建/dev/chrdev设备节点文件
mknod /dev/chrdev c 200 0
mknod:创建节点命令
/dev/chrdev:要创建的节点文件
c:字符设备
200 0:设备的主设备号和次设备号

创建完成以后就会存在/dev/chrdev文件,可以使用ls/dev/chrdev -l命令查看

3、设备测试

运行测试程序

./xxx(程序) 参数1 参数2... (参数传给应用层的main函数)
./chrdevbaseApp /dev/chrdevbase 1 //例如测试对chrdevbase设备的写操作

4、卸载驱动模块

rmmod xxx.ko

卸载后可以使用lsmod查看卸载模块是否成功


http://www.ppmy.cn/news/32790.html

相关文章

C实现带头循环双向链表(pushback pushfront popback popfront insert erase find destroy等)

带头循环双向链表是链表中效率最高的,但是由于他里面有两个指针节点,所以也会更浪费空间一些,但是他在任意位置的插入删除的效率很高,所以也就弥补了顺序表的不足。 首先我们来看一下他的逻辑结构是什么样子的 下面我们看一下如何…

【面试题】Python软件工程师能力评估试题(一)

文章目录前言应试者需知(一)Python 语言基础能力评估1、理解问题并完成代码:2、阅读理解代码,并在空白处补充完整代码:3、编写一个装饰器:exposer4、阅读代码并在空白处补充完整代码:5、自行用P…

5 全面认识java的控制流程

全面认识java控制流程1.块作用域2.条件语句3.迭代语句3.1while语句3.2do-while语句3.3for语句3.4 for-in语法4.中断控制流程的语句4.1 return4.2 break和continue4.2.1 不带标签的break语句4.2.2 带标签的break语句4.2.3 continue语句4.3 goto()5.多重选择:switch语句1.块作用域…

小米10s格机修复 nv报错案例解析 关于基带分区的一些常识

前面分享过几期关于基带 diag端口与qcn相关的几篇帖子。其中一位粉丝朋友联系我。他的机型因为误格机导致手机进不去系统,反复进入官方rec报错nv损坏。进不去系统。 有兴趣的朋友可以参阅我的几个帖子,只是个人的一些片面理解。 基带相关贴; 安卓玩机…

进程跟线程的区别

进程跟线程的区别 文章目录进程跟线程的区别前言一.什么线程二.线程与进程的联系三.线程与进程有什么不同前言 现代所有计算机都能同时做几件事情,当一个用户程序正在运行时,计算机还能同时读取磁盘,并向屏幕打印输出正文.在一个多道操作程序中,cpu由一道程序向另外一道程的切…

【HTML系列】前章 · HTML前序知识

写在前面 Hello大家好, 我是【麟-小白】,一位软件工程专业的学生,喜好计算机知识。希望大家能够一起学习进步呀!本人是一名在读大学生,专业水平有限,如发现错误或不足之处,请多多指正&#xff0…

L2-022 重排链表 L2-002 链表去重

给定一个单链表 L1 →L2→⋯→L n−1 →L n ,请编写程序将链表重新排列为 L n →L 1 →L n−1 →L 2 →⋯。例如:给定L为1→2→3→4→5→6,则输出应该为6→1→5→2→4→3。 输入格式: 每个输入包含1个测试用例。每个测试用例第1行…

33个非常实用的JavaScript一行代码

一、日期处理 1. 检察日期是否有效 该方法用于检测给出的日期是否有效: const isDateValid (...val) > !Number.isNaN(new Date(...val).valueOf());isDateValid("December 17, 1995 03:24:00"); // true2. 计算两个日期之间的间隔 该方法用于计…