4、Linux驱动开发:设备-设备号设备号注册

目录

🍅点击这里查看所有博文

  随着自己工作的进行,接触到的技术栈也越来越多。给我一个很直观的感受就是,某一项技术/经验在刚开始接触的时候都记得很清楚。往往过了几个月都会忘记的差不多了,只有经常会用到的东西才有可能真正记下来。存在很多在特殊情况下有一点用处的技巧,用的不多的技巧可能一个星期就忘了。

  想了很久想通过一些手段把这些事情记录下来。也尝试过在书上记笔记,这也只是一时的,书不在手边的时候那些笔记就和没记一样,不是很方便。

  很多时候我们遇到了问题,一般情况下都是选择在搜索引擎检索相关内容,这样来的也更快一点,除非真的找不到才会去选择翻书。后来就想到了写博客,博客作为自己的一个笔记平台倒是挺合适的。随时可以查阅,不用随身携带。

  同时由于写博客是对外的,既然是对外的就不能随便写,任何人都可以看到。经验对于我来说那就只是经验而已,公布出来说不一定我的一些经验可以帮助到其他的人。遇到和我相同问题时可以少走一些弯路。

  既然决定了要写博客,那就只能认真去写。不管写的好不好,尽力就行。千里之行始于足下,一步一个脚印,慢慢来 ,写的多了慢慢也会变好的。权当是记录自己的成长的一个过程,等到以后再往回看时,就会发现自己以前原来这么菜😂。

  本系列博客所述资料均来自互联网资料,并不是本人原创(只有博客是自己写的)。出于热心,本人将自己的所学笔记整理并推出相对应的使用教程,方面其他人学习。为国内的物联网事业发展尽自己的一份绵薄之力,没有为自己谋取私利的想法。若出现侵权现象,请告知本人,本人会立即停止更新,并删除相应的文章和代码。

设备分类

  Linux系统中的设备可以分为字符设备、块设备和网络设备这三类。

字符设备

  字符设备是能够像字节流一样被访问的设备,通常不支持随机存取。当对字符设备发出读写请求,相应的IO操作立即发生。

  Linux系统中很多设备都是字符设备,我们常用的键盘、串口、I2C、SPI、音频都是字符设备。

  在嵌入式Linux开发中,接触最多的就是字符设备以及驱动。

块设备

  块设备是Linux系统中进行IO操作时必须以块为单位进行访问的设备,应用程序可以寻址磁盘上的任何位置,并由此读取数据(随机读取)。

  块设备驱动会利用一块系统内存作为缓冲区,因此对块设备发出读写访问,并不一定立即产生硬件I/O操作。

  块设备能够安装文件系统,Linux系统中常见的块设备有硬盘、EMMC、NAND、SD 卡、闪存等

网络设备

  网络设备既可以是网卡这样的硬件设备,也可以是一个纯软件设备如回环设备。
  网络设备由Linux的网络子系统驱动,负责数据包的发送和接收,而不是面向流设备,因此在Linux系统文件系统中网络设备没有节点。
  对网络设备的访问是通过socket调用产生,而不是普通的文件操作如open/closc和read/write等。

其他

  一个设备可以 属于多种设备驱动类型,比如 USB WIFI,其使用 USB 接口,所以属于字符设备。但是其又能上网,所以也属于网络设备驱动。

  这些设备中,有些设备是对实际存在的物理硬件的抽象,而有些设备则是内核自身提供的功能,不依赖于特定的物理硬件,又称为虚拟设备。

  • /dev/loop[x]
  • /dev/random
  • /dev/null
  • /dev/zero
  • /dev/full

内核结构

  linux内核结构如下图所示,Linux的设备经由内核统一管理。

  其中字符设备和块设备最终都挂载与文件子系统之下。两者都可通过虚拟文件系统找到对应的节设备点,并支持使用文件接口进行访问。

输入图片说明

设备号

  linux中设备号是用来标记一类设备以及区分这类设备中具体个体的一组号码。

  Linux提供了一个名为dev_t的数据类型表示设备号,dev_t定义在文件include/linux/types.h里面,定义如下:

typedef __u32 __kernel_dev_t; 
typedef __kernel_dev_t dev_t; 
//include/uapi/asm-generic/int-ll64.h
typedef unsigned int __u32;

  dev_t其实就是unsigned int类型,是一个32位的数据类型。主设备号(高12位)用来标记设备的类型,次设备号(低20位)用来区分在这类设备中具体的个体设备。不同的主设备号之间,次设备号编码可以出现重复现象。

输入图片说明
  因此Linux系统中主设备号范围为0~4095,所以大家在选择主设备号的时候一定不要超过这个范围。

  内核为了保证在主次设备号位宽发生变化时,现在的程序依然可以工作,内核提供了如下的几个宏,编写设备驱动代码是,最好使用宏来计算dev_t参数,而不要使用常数去计算。

<include  /linux/kdev_t.h>
#define MINORBITS   20
#define MINORMASK   ((1U << MINORBITS) - 1)
#define MAJOR(dev)  ((unsigned int) ((dev) >> MINORBITS))
#define MINOR(dev)  ((unsigned int) ((dev) & MINORMASK))
#define MKDEV(ma,mi)    (((ma) << MINORBITS) | (mi))

设备管理

  前面说到主设备号(高12位)的范围为0~4095。不代表Linux内核可以同时管理这么多主设备。

  linux内核为了方便管理设备,在fs/char_dev.c中维护了一个char_device_struct结构的结构体。

/*fs/char_dev.c*/
static struct char_device_struct {struct char_device_struct *next;unsigned int major;unsigned int baseminor;int minorct;char name[64];struct cdev *cdev;      /* will die */
} *chrdevs[CHRDEV_MAJOR_HASH_SIZE];/* fs.h */
#define CHRDEV_MAJOR_HASH_SIZE 255

  内核代码中chrdevs结构体的每一个成员,都是一个以char_device_struct结构体指针为元素的单项链的头指针。

  每条链表中的所有元素拥有相同的主设备号,及该链表所在chrdevs数组的下标。

输入图片说明

  从上面可以得知结构体的长度为255。内核通过对255取模(major%255)来实现将0~4095个主设备号,分配到255个哈希表项上。

  需要注意的是,只有255个哈希表不代表说内核最多只有255个主设备。很多人都会陷入误区,认为最多只有255个主设备,这是错误的。

  举个栗子,设备号256与设备号1处于同一表项位置,但是它们并不冲突。也就是说设备号1和设备号256是可以同时存在的。

root@ubuntu:# cat /proc/devices 
Character devices:1 mem4 /dev/vc/0
.......
254 gpiochip
256 hello_register_chrdev

字符设备注册

  Linux内核中字符设备号的分配有两个函数,分别是手动分配register_chrdev_region还有自动分配alloc_chrdev_region

register_chrdev_region

  此函数用于静态注册设备号,该函数第一个参数from表示是一个设备号,第二个参数count是连续设备编号的个数,代表当前驱动所管理的同类设备的个数,第三个参数name表示设备或者驱动的名称。

/*** register_chrdev_region() - register a range of device numbers* @from: the first in the desired range of device numbers; must include*        the major number.* @count: the number of consecutive device numbers required* @name: the name of the device or driver.** Return value is zero on success, a negative error code on failure.*/
int register_chrdev_region(dev_t from, unsigned count, const char *name)

  优点是可以在注册的时候就知道其设备号,缺点是可能会与系统中已经注册的设备号冲突导致注册失败。

示例

  下方示例中,注册了主设备为256的字符设备。

static int major = 256;
static int minor = 0;
static dev_t register_devno;
static int hello_register_chrdev(void)
{int result;printk("hello_register_chrdev \n");register_devno = MKDEV(major, minor);result = register_chrdev_region(register_devno, 1, "hello_register_chrdev");if (result < 0){printk("register_chrdev_region fail \n");return result;}return result;
}

结果

  驱动模块挂载后可看到256号主设备被成功注册,且设备名与预期一致。

root@ubuntu:# cat /proc/devices 
Character devices:1 mem4 /dev/vc/0
.......
254 gpiochip
256 hello_register_chrdev

alloc_chrdev_region

  该函数的第一个参数*dev是返回的设备号,第二个参数baseminor是次设备号起始号,第三个参数count是连续设备编号的个数,第四个参数name表示设备或者驱动的名称。

/*** alloc_chrdev_region() - register a range of char device numbers* @dev: output parameter for first assigned number* @baseminor: first of the requested range of minor numbers* @count: the number of minor numbers required* @name: the name of the associated device or driver** Allocates a range of char device numbers.  The major number will be* chosen dynamically, and returned (along with the first minor number)* in @dev.  Returns zero or a negative error code.*/
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count,const char *name)

  该函数是由系统协助动态分配设备号,分配的主设备号的范围在1-254之间,系统会从chrdevs数组尾部(也就是第254项)依次往前找未使用的主设备号。

    /* temporary */if (major == 0) {for (i = ARRAY_SIZE(chrdevs)-1; i > 0; i--) {if (chrdevs[i] == NULL)break;}if (i == 0) {ret = -EBUSY;goto out;}major = i;}

示例

  使用自动分配注册一个字符设备。

static dev_t alloc_devno;
static int hello_alloc_chrdev(void)
{int result;printk("hello_alloc_chrdev \n");result = alloc_chrdev_region(&alloc_devno, 0, 1, "hello_alloc_chrdev");if (result < 0){printk("alloc_chrdev_region fail \n");return result;}return result;
}

结果

  驱动模块挂载后,可看到名字为hello_alloc_chrdev的字符设备被成功注册,设备号为243为自动分配。

root@ubuntu:# cat /proc/devices 
226 drm
243 hello_alloc_chrdev
244 hidraw
245 aux
246 bsg

字符设备释放

  设备号作为一种系统资源,当所对应的设备驱动程序被卸载时,就应该把设备号归还给系统,以便分配给其他内核模块使用。无论是静态分配还是动态分配,都是通过调用unregister_chrdev_region函数释放设备号。

/*** unregister_chrdev_region() - return a range of device numbers* @from: the first in the range of numbers to unregister* @count: the number of device numbers to unregister** This function will unregister a range of @count device numbers,* starting with @from.  The caller should normally be the one who* allocated those numbers in the first place...*/
void unregister_chrdev_region(dev_t from, unsigned count)

  函数在chrdevs数组中查找参数from和count所对应的struct char_device_struct 对象节点,找到以后将其从链表中删除并释放该节点所占用的内存,从而将对应的设备号释放以供其他设备驱动模块使用。

示例

  销毁前方示例创建的两个字符设备。

static void hello_exit(void)
{printk("hello_exit \n");unregister_chrdev_region(register_devno, 1);unregister_chrdev_region(alloc_devno, 1);
}

结果

  可以看到主设备编号为256和243的设备全部被释放。

root@ubuntu:# cat /proc/devices 
226 drm
244 hidraw
245 aux
246 bsg
247 hmm_device
248 watchdog
249 rtc
250 dax
251 dimmctl
252 ndctl
253 tpm
254 gpiochip

  那么本篇博客就到此结束了,这里只是记录了一些我个人的学习笔记,其中存在大量我自己的理解。文中所述不一定是完全正确的,可能有的地方我自己也理解错了。如果有些错的地方,欢迎大家批评指正。如有问题直接在对应的博客评论区指出即可,不需要私聊我。我们交流的内容留下来也有助于其他人查看,说不一定也有其他人遇到了同样的问题呢😂。

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

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

相关文章

Verilog语法学习——LV2_异步复位的串联T触发器

LV2_异步复位的串联T触发器 题目来源于牛客网 [牛客网在线编程_Verilog篇_Verilog快速入门 (nowcoder.com)](https://www.nowcoder.com/exam/oj?page1&tabVerilog篇&topicId301) 题目 题目描述&#xff1a; 用verilog实现两个串联的异步复位的T触发器的逻辑&#x…

【LeetCode】141.环形链表

题目 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置&#…

opencv-22 图像几何变换01-缩放-cv2.resize()(图像增强,图像变形,图像拼接)

什么是几何变换&#xff1f; 几何变换是计算机图形学中的一种图像处理技术&#xff0c;用于对图像进行空间上的变换&#xff0c;而不改变图像的内容。这些变换可以通过对图像中的像素位置进行调整来实现。 常见的几何变换包括&#xff1a; 平移&#xff08;Translation&#x…

STM32MP157驱动开发——按键驱动(tasklet)

文章目录 “tasklet”机制&#xff1a;内核函数定义 tasklet使能/ 禁止 tasklet调度 tasklet删除 tasklet tasklet软中断方式的按键驱动程序(stm32mp157)tasklet使用方法&#xff1a;button_test.cgpio_key_drv.cMakefile修改设备树文件编译测试 “tasklet”机制&#xff1a; …

【Ansible】Ansible自动化运维工具之playbook剧本

playbook 一、playbook 的概述1. playbook 的概念2. playbook 的构成 二、playbook 的应用1. 安装 httpd 并启动2. 定义、引用变量3. 指定远程主机 sudo 切换用户4. when条件判断5. 迭代6. Templates 模块6.1 添加模板文件6.2 修改主机清单文件6.3 编写 playbook 7. tags 模块 …

vue权限按钮的实现

鉴权函数 由于下面几种方式都需要用到鉴权函数&#xff0c;所以将其放置在组件外面&#xff0c;供组件或其他文件调用。 // src/utils/hasPermission.jsimport { usePermissionStore } from /stores import array from lodash/array export const hasPermission (value, def…

EXCEL,如何比较2个表里的数据差异(使用数据透视表)

目录 1 问题: 需要比较如下2个表的内容差异 1.1 原始数据喝问题 1.2 提前总结 2 使用EXCEL公式方法 2.1 新增辅助列&#xff1a; 辅助index 2.2 具体公式 配合条件格式 使用 3 数据透视表方法 3.1 新增辅助列&#xff1a; 辅助index 3.2 需要先打开 数据透视表向导 …

Ubuntu 20.04 Ubuntu18.04安装录屏软件Kazam

1.在Ubuntu Software里面输入Kazam&#xff0c;就可以找不到这个软件&#xff0c;直接点击install就可以了 2.使用方法&#xff1a; 选择Screencast&#xff08;录屏&#xff09; Fullscreen&#xff08;全屏&#xff09;-----Windows&#xff08;窗口&#xff09;--------Ar…

20.3 HTML表格

1. table表格 table标签是HTML中用来创建表格的元素. table标签通常包含以下子标签: - th标签: 表示表格的表头单元格(table header), 用于描述列的标题. - tr标签: 表示表格的行(table row). - td标签: 表示表格的单元格(table data), 通常位于tr标签内, 用于放置单元格中的…

数据结构之动态顺序表(附带完整程序)

&#x1f388;基本概念 &#x1f308;一.线性表、顺序表的定义 ☀️&#xff08;1&#xff09;线性表&#xff1a; 是n个具有相同特性的数据元素的有限序列。线性表在逻辑上是线性结构&#xff0c;但在物理上存储时&#xff0c;通常以数组和链式结构的形式存储。 ☀️&…

c# 此程序集中已使用了资源标识符

严重性 代码 说明 项目 文件 行 禁止显示状态 错误 CS1508 此程序集中已使用了资源标识符“BMap.NET.WindowsForm.BMapControl.resources” BMap.NET.WindowsForm D:\MySource\Decompile\BMap.NET.WindowsForm\CSC 1 活动 运行程序时&a…

Mock-MOCO使用过程

一、jar包下载&#xff1a;https://github.com/dreamhead/moco 二、准备mock的json文件 data.json内容&#xff1a; ####GET请求 [{"description": "response使用Content-Type为charsetGBK编码格式来查看返回信息为中文的内容","request": {&q…

《Elasticsearch 源码解析与优化实战》第5章:选主流程

《Elasticsearch 源码解析与优化实战》第5章&#xff1a;选主流程 - 墨天轮 一、简介 Discovery 模块负责发现集群中的节点&#xff0c;以及选择主节点。ES 支持多种不同 Discovery 类型选择&#xff0c;内置的实现称为Zen Discovery ,其他的包括公有云平台亚马逊的EC2、谷歌…

Ansible单yaml文件部署Zabbix5.0监控平台

文章目录 Ansible单yaml文件部署Zabbix5.0监控平台节点规划案例实施基础环境准备编写剧本文件ZabbixWeb界面(1)改中文(2)添加监控主机 Ansible单yaml文件部署Zabbix5.0监控平台 节点规划 IP主机名节点192.168.200.10ansibleAnsible节点192.168.200.20zabbix-serverZabbix-ser…

深度学习入门(一):神经网络基础

一、深度学习概念 1、定义 通过训练多层网络结构对位置数据进行分类或回归&#xff0c;深度学习解决特征工程问题。 2、深度学习应用 图像处理语言识别自然语言处理 在移动端不太好&#xff0c;计算量太大了&#xff0c;速度可能会慢 eg.医学应用、自动上色 3、例子 使用…

Effective Java 案例分享(八)

39、使用注解而不是通过命名规则分类 如果需要对定义class&#xff0c;property&#xff0c;或者method进行分类管理&#xff0c;推荐的做法是使用注解对其添加类别&#xff0c;而不是通过命名规则分类。这里以JUnit为例&#xff1a; 在JUnit 3中&#xff0c;如果要写测试的方…

linux环境安装mysql数据库

一&#xff1a;查看是否自带mariadb数据库 命令&#xff1a;rpm -qa | grep mariadb 如果自带数据库则卸载掉重新安装 命令&#xff1a;yum remove mariadb-connector-c-3.1.11-2.el8_3.x86_64 二&#xff1a;将压缩文件上传到/user/local/mysql文件夹 或者直接下载 命令&a…

基于ssm+mysql+html道路养护管理系统

基于ssmmysqlhtml道路养护管理系统 一、系统介绍二、功能展示1.道路信息管理2.损害类型信息管理3.损害类型信息管理4.评定等级信息管理5.日常巡查信息管理6.定期检查信息管理 四、获取源码 一、系统介绍 系统主要功能&#xff1a;道路信息管理、损害类型信息管理、评定等级信息…

【网络原理】 (1) (应用层 传输层 UDP协议 TCP协议 TCP协议段格式 TCP内部工作机制 确认应答 超时重传 连接管理)

文章目录 应用层传输层UDP协议TCP协议TCP协议段格式TCP内部工作机制确认应答超时重传 网络原理部分我们主要学习TCP/IP协议栈这里的关键协议(TCP 和 IP),按照四层分别介绍.(物理层,我们不涉及). 应用层 我们需要学会自定义一个应用层协议. 自定义协议的原因? 当前的软件(应用…

【JAVASE】顺序和选择结构

⭐ 作者&#xff1a;小胡_不糊涂 &#x1f331; 作者主页&#xff1a;小胡_不糊涂的个人主页 &#x1f4c0; 收录专栏&#xff1a;浅谈Java &#x1f496; 持续更文&#xff0c;关注博主少走弯路&#xff0c;谢谢大家支持 &#x1f496; 顺序和选择 1. 顺序结构2. 分支结构2.1 …