Linux设备驱动-内核如何管理设备号
ztj100 2024-11-09 15:19 12 浏览 0 评论
开篇
本文引用的内核代码参考来自版本 linux-5.15.4 。
在 Linux 系统中,每个注册到系统的设备都有一个编号,这个编号便是 Linux 系统中的设备号。
设备号作为一种系统资源,需要加以管理。否则,如果设备号与驱动程序对应关系错误,就会引起混乱或引起潜在的问题。
通过查看 /proc/devices 文件可以得到系统中注册的设备,第一列为主设备号,第二列为设备名称
$ cat /proc/devices
Character devices:
1 mem
4 /dev/vc/0
4 tty
4 ttyS
5 /dev/tty
5 /dev/console
5 /dev/ptmx
5 ttyprintk
6 lp
7 vcs
10 misc
13 input
21 sg
...
Block devices:
7 loop
8 sd
9 md
11 sr
65 sd
66 sd
...
设备号的构成
一个设备号由主设备号和次设备号构成。
主设备号对应设备驱动程序,同一类设备一般使用相同的主设备号。
次设备号由驱动程序使用,驱动程序用来描述使用该驱动的设备的序号,序号一般从 0 开始。
Linux 设备号用 dev_t 类型的变量进行标识,这是一个 32位 无符号整数,内核源码定义为:
/* <include/linux/types.h> */
typedef u32 __kernel_dev_t;
typedef __kernel_dev_t dev_t;
主设备号用 dev_t 的高 12 位表示,次设备号用 dev_t 低 20 位表示。
内核提供了几个宏定义,供驱动程序操作设备号时使用:
/* <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))
宏 MAJOR 从设备号 dev 中提取主设备号。宏 MINOR 用来从设备号 dev 中提取次设备号。宏 MKDEV 用来将主设备号 ma 和 次设备号 mi 组合成 dev_t 类型的设备号。
另外,内核也提供了从设备文件 i-节点结构(inode 结构体)中获取主次设备号的函数,如下:
/* <include/linux/fs.h> */
/* 获取次设备号 */
static inline unsigned iminor(const struct inode *inode)
{
return MINOR(inode->i_rdev);
}
/* 获取主设备号 */
static inline unsigned imajor(const struct inode *inode)
{
return MAJOR(inode->i_rdev);
}
通过函数源码可知,获取主设备号和次设备号最终是通过宏定义完成的。
内核管理设备号
以字符设备为例,向内核中注册设备号,内核是如何分配和管理设备号的呢?
在编写字符设备驱动时,可以通过如下两个系统调用向内核注册设备号:
- register_chrdev_region()
注册一系列连续的字符设备号,主设备号需要函数调用者指定。此函数的原型为:
int register_chrdev_region(dev_t from, unsigned count, const char *name)
参数 from 为设备编号,包含主设备号和次设备号。参数 count 用于指定连续设备号的个数,即当前驱动程序所管理的同类设备的个数。参数 name 为设备或驱动的名字。
执行成功,返回 0。失败,则返回一个负值的错误码。
- alloc_chrdev_region()
注册一系列连续的字符设备号,主设备号是由内核动态分配得到的。此函数的原型为:
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name)
参数 dev 为函数的传出参数,用于记录动态分配的设备号,如果申请多个设备号,则此参数记录这些连续设备号的起始值。
参数 baseminor 指定首个次设备号。参数 count 用于指定连续设备号的个数。参数 name 为设备或驱动的名字。
执行成功,返回 0。失败,则返回一个负值的错误码。
接下来,看看这两个函数的内部实现流程。
register_chrdev_region()
该函数的内核源码为,关键部分已加注释:
/* <fs.char_dev.c> */
int register_chrdev_region(dev_t from, unsigned count, const char *name)
{
struct char_device_struct *cd;
dev_t to = from + count;
dev_t n, next;
/* 循环,注册多个连续的设备号 */
for (n = from; n < to; n = next)
{
/* 计算得到下一个设备号 */
next = MKDEV(MAJOR(n)+1, 0);
/* 判断是否超限 */
if (next > to)
next = to;
/* 向内核注册指定的设备号 */
cd = __register_chrdev_region(MAJOR(n), MINOR(n), next - n, name);
if (IS_ERR(cd))
goto fail;
}
return 0;
fail:
/* 如果失败,则释放已申请的设备号资源 */
to = n;
for (n = from; n < to; n = next)
{
next = MKDEV(MAJOR(n)+1, 0);
kfree(__unregister_chrdev_region(MAJOR(n), MINOR(n), next - n));
}
return PTR_ERR(cd);
}
由代码内容可知,这个函数的核心处理流程是通过内部调用 __register_chrdev_region()实现的。
这个函数的主要功能是,将要使用的设备号注册到内核的设备号管理体系中,避免多个驱动程序使用相同的设备号,而引起的混乱。
如果注册设备号已经被使用,则会返回错误码告知调用者,即调用失败。如果成功,则函数返回 0。
struct char_device_struct
在调用过程中,会涉及到一个关键的数据结构 struct char_device_struct,其定义如下:
#define CHRDEV_MAJOR_HASH_SIZE 255
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];
定义结构体的同时,还定义了一个全局性的指针数组 chrdevs,是内核用来分配和管理设备号的。数组中的每一个元素都是指向 struct char_device_struct 类型的指针。
函数 register_chrdev_region() 的主要功能是将驱动程序要使用的设备号记录到 chrdevs 数组中。
__register_chrdev_region()
核心处理函数 __register_chrdev_region() 内部,首先会分配一个 struct char_device_struct 类型的指针 cd,然后对其进行初始化(已经去除无关代码):
static struct char_device_struct *
__register_chrdev_region(unsigned int major, unsigned int baseminor,
int minorct, const char *name)
{
...
cd = kzalloc(sizeof(struct char_device_struct), GFP_KERNEL);
if (cd == NULL)
return ERR_PTR(-ENOMEM);
/* 根据主设备号计算索引,搜索 chrdevs 数组,判断主设备号是否可用 */
i = major_to_index(major);
for (curr = chrdevs[i]; curr; prev = curr, curr = curr->next)
{
if (curr->major < major)
continue;
if (curr->major > major)
break;
if (curr->baseminor + curr->minorct <= baseminor)
continue;
if (curr->baseminor >= baseminor + minorct)
break;
goto out;
}
/* 初始化信息 */
cd->major = major;
cd->baseminor = baseminor;
cd->minorct = minorct;
strlcpy(cd->name, name, sizeof(cd->name));
/* 将分配的 cd 加入到 chrdevs[i] 中 */
if (!prev) {
cd->next = curr;
chrdevs[i] = cd;
} else {
cd->next = prev->next;
prev->next = cd;
}
...
}
函数申请完内存资源后,开始扫描 chrdevs 数组,确保当前注册的设备号可用。如果设备号占用,函数返回错误码,即调用失败。
如果设备号可用,则用设备号和名字信息初始化。初始化完成后,将 struct char_device_struct 加入到内核管理设备号的链表中。
alloc_chrdev_region()
此函数由内核动态分配设备号,该函数的内核源码如下,关键部分已加注释:
/* <fs.char_dev.c> */
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count,
const char *name)
{
struct char_device_struct *cd;
/* 向内核注册设备号 */
cd = __register_chrdev_region(0, baseminor, count, name);
if (IS_ERR(cd))
return PTR_ERR(cd);
/* 得到动态获取的首个设备号 */
*dev = MKDEV(cd->major, cd->baseminor);
return 0;
}
这个函数的核心处理也是由函数 __register_chrdev_region() 实现的。
与 register_chrdev_region() 相比,alloc_chrdev_region() 在调用 __register_chrdev_region() 时,第一个参数为 0。此时 __register_chrdev_region() 处理流程代码如下,
static struct char_device_struct *
__register_chrdev_region(unsigned int major, unsigned int baseminor,
int minorct, const char *name)
{
...
cd = kzalloc(sizeof(struct char_device_struct), GFP_KERNEL);
if (cd == NULL)
return ERR_PTR(-ENOMEM);
/* 查找可用的主设备号 */
f (major == 0) {
ret = find_dynamic_major();
if (ret < 0) {
pr_err("CHRDEV \"%s\" dynamic allocation region is full\n",
name);
goto out;
}
major = ret;
}
...
}
在分配完成 struct char_device_struct 内存资源之后,通过 find_dynamic_major() 查找可用的主设备号。后续处理与 register_chrdev_region() 函数调用处理相同。
设备号分配成功后,将 struct char_device_struct 类型指针返回给 alloc_chrdev_region() 函数。然后再通过如下代码将新分派的设备号返回给 alloc_chrdev_region() 调用者:
*dev = MKDEV(cd->major, cd->baseminor);
小结
本文主要介绍了以下几点内容:
- 设备号是如何构成的,以及对其操作的宏定义。
- register_chrdev_region() 和 alloc_chrdev_region() 实现细节。
- 记录设备号相关信息的关键数据结构 struct char_device_struct。
- 内核通过 chrdevs 数组来跟踪系统中设备号的使用情况。
相关推荐
- Vue 技术栈(全家桶)(vue technology)
-
Vue技术栈(全家桶)尚硅谷前端研究院第1章:Vue核心Vue简介官网英文官网:https://vuejs.org/中文官网:https://cn.vuejs.org/...
- vue 基础- nextTick 的使用场景(vue的nexttick这个方法有什么用)
-
前言《vue基础》系列是再次回炉vue记的笔记,除了官网那部分知识点外,还会加入自己的一些理解。(里面会有部分和官网相同的文案,有经验的同学择感兴趣的阅读)在开发时,是不是遇到过这样的场景,响应...
- vue3 组件初始化流程(vue组件初始化顺序)
-
学习完成响应式系统后,咋们来看看vue3组件的初始化流程既然是看vue组件的初始化流程,咋们先来创建基本的代码,跑跑流程(在app.vue中写入以下内容,来跑流程)...
- vue3优雅的设置element-plus的table自动滚动到底部
-
场景我是需要在table最后添加一行数据,然后把滚动条滚动到最后。查网上的解决方案都是读取html结构,暴力的去获取,虽能解决问题,但是不喜欢这种打补丁的解决方案,我想着官方应该有相关的定义,于是就去...
- Vue3为什么推荐使用ref而不是reactive
-
为什么推荐使用ref而不是reactivereactive本身具有很大局限性导致使用过程需要额外注意,如果忽视这些问题将对开发造成不小的麻烦;ref更像是vue2时代optionapi的data的替...
- 9、echarts 在 vue 中怎么引用?(必会)
-
首先我们初始化一个vue项目,执行vueinitwebpackechart,接着我们进入初始化的项目下。安装echarts,npminstallecharts-S//或...
- 无所不能,将 Vue 渲染到嵌入式液晶屏
-
该文章转载自公众号@前端时刻,https://mp.weixin.qq.com/s/WDHW36zhfNFVFVv4jO2vrA前言...
- vue-element-admin 增删改查(五)(vue-element-admin怎么用)
-
此篇幅比较长,涉及到的小知识点也比较多,一定要耐心看完,记住学东西没有耐心可不行!!!一、添加和修改注:添加和编辑用到了同一个组件,也就是此篇文章你能学会如何封装组件及引用组件;第二能学会async和...
- 最全的 Vue 面试题+详解答案(vue面试题知识点大全)
-
前言本文整理了...
- 基于 vue3.0 桌面端朋友圈/登录验证+60s倒计时
-
今天给大家分享的是Vue3聊天实例中的朋友圈的实现及登录验证和倒计时操作。先上效果图这个是最新开发的vue3.x网页端聊天项目中的朋友圈模块。用到了ElementPlus...
- 不来看看这些 VUE 的生命周期钩子函数?| 原力计划
-
作者|huangfuyk责编|王晓曼出品|CSDN博客VUE的生命周期钩子函数:就是指在一个组件从创建到销毁的过程自动执行的函数,包含组件的变化。可以分为:创建、挂载、更新、销毁四个模块...
- Vue3.5正式上线,父传子props用法更丝滑简洁
-
前言Vue3.5在2024-09-03正式上线,目前在Vue官网显最新版本已经是Vue3.5,其中主要包含了几个小改动,我留意到日常最常用的改动就是props了,肯定是用Vue3的人必用的,所以针对性...
- Vue 3 生命周期完整指南(vue生命周期及使用)
-
Vue2和Vue3中的生命周期钩子的工作方式非常相似,我们仍然可以访问相同的钩子,也希望将它们能用于相同的场景。...
- 救命!这 10 个 Vue3 技巧藏太深了!性能翻倍 + 摸鱼神器全揭秘
-
前端打工人集合!是不是经常遇到这些崩溃瞬间:Vue3项目越写越卡,组件通信像走迷宫,复杂逻辑写得脑壳疼?别慌!作为在一线摸爬滚打多年的老前端,今天直接甩出10个超实用的Vue3实战技巧,手把...
- 怎么在 vue 中使用 form 清除校验状态?
-
在Vue中使用表单验证时,经常需要清除表单的校验状态。下面我将介绍一些方法来清除表单的校验状态。1.使用this.$refs...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- Vue 技术栈(全家桶)(vue technology)
- vue 基础- nextTick 的使用场景(vue的nexttick这个方法有什么用)
- vue3 组件初始化流程(vue组件初始化顺序)
- vue3优雅的设置element-plus的table自动滚动到底部
- Vue3为什么推荐使用ref而不是reactive
- 9、echarts 在 vue 中怎么引用?(必会)
- 无所不能,将 Vue 渲染到嵌入式液晶屏
- vue-element-admin 增删改查(五)(vue-element-admin怎么用)
- 最全的 Vue 面试题+详解答案(vue面试题知识点大全)
- 基于 vue3.0 桌面端朋友圈/登录验证+60s倒计时
- 标签列表
-
- idea eval reset (50)
- vue dispatch (70)
- update canceled (42)
- order by asc (53)
- spring gateway (67)
- 简单代码编程 贪吃蛇 (40)
- transforms.resize (33)
- redisson trylock (35)
- 卸载node (35)
- np.reshape (33)
- torch.arange (34)
- node卸载 (33)
- npm 源 (35)
- vue3 deep (35)
- win10 ssh (35)
- exceptionininitializererror (33)
- vue foreach (34)
- idea设置编码为utf8 (35)
- vue 数组添加元素 (34)
- std find (34)
- tablefield注解用途 (35)
- python str转json (34)
- java websocket客户端 (34)
- tensor.view (34)
- java jackson (34)