Linux驱动函数register_chrdev_region

2023-12-16 10:32

本文主要是介绍Linux驱动函数register_chrdev_region,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Linux驱动注册有两种接口,一种是旧接口register_chrdev,另一种是新接口register_chrdev_region + cdev_init/cdev_add。
register_chrdev_region函数内容如下:

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);}

该函数有三个参数,第一个参数from表示要注册的主设备号(该主设备号必须使用MKDEV宏转换一下,例如主设备号为2,传参时需要使用MKDEV(2, 0)),第二个参数count表示要注册的设备共有多少个次设备号,最后一个参数用来传递设备名。该函数的实现机制其实就是一个for循环多次调用__register_chrdev_region这个函数来完成的,有人会问调用一次不久好了嘛,为什么要整个for循环?这个for循环其实是为了处理次设备号溢出的状况,设想一下如果Linux内核只支持256个次设备号,而我们给count参数257个,这时for循环就起作用了。
在具体分析前,我们还需要了解一下这三个宏:

#define MAJOR(dev) ((unsigned int) ((dev) >> MINORBITS))
#define MINOR(dev) ((unsigned int) ((dev) & MINORMASK))
#define MKDEV(ma,mi) (((ma) << MINORBITS) | (mi))

由于Linux内核中一个设备号是由主设备号和次设备号组成,而为了操作方便,我们就需要将这两个号通过MKDEV宏组合起来,同时内核也提供了MAJOR和MINOR宏用于从组合设备号中分别提取主设备号与次设备号。我们可以通过MKDEV这个宏清除的看到两个设备号是如何组合起来的,实际上就是定义一个32位二进制数,高位存放主设备号,低位存放次设备号,由于内核中MINORBITS宏定义位20,所以理论上次设备号可以有2^20个。
好了,我们开始正式分析这个函数,如果count小与2^20,主设备号(from)为MKDEV(t, 0),也就是(t << 20),那么to就是(t << 20 + count)。由于第一次(t << 20)必定小与to,所以for循环执行次数大于等于1。在进入for循环后,next被赋值((t + 1)<< 20)),第一次next必定大于to,所以next又等于了to(t << 20 + count),接着调用了__register_chrdev_region(n, 0,count, name)函数完成了设备号的注册,最后退出循环。
考虑第二种情况,如果count大于2^20,主设备号(from)为t << 20,to等于(t << 20 + count)。执行第一次for循环, next被赋值((t + 1)<< 20)),显然next小与to,故第一次for循环会调用了__register_chrdev_region(t, 0,1 << 20, name)函数完成了设备号的注册,然后执行下次循环,from变为((t + 1) << 20), 与之前相同若没有满足退出循环条件,调用__register_chrdev_region(t + 1, 0,1 << 20, name)注册设备号,直到最后一次满足条件,调用__register_chrdev_region(t + x, 0,to - (t + x) << 20, name)注册设备号,退出循环。
至此,我们可以分析出这个函数的主要作用是处理次设备号过多的情况,当次设备号超出内核设定值时,内核自动重新分配一个新的主设备号继续分配次设备号。

这篇关于Linux驱动函数register_chrdev_region的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/500147

相关文章

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

Linux中SSH服务配置的全面指南

《Linux中SSH服务配置的全面指南》作为网络安全工程师,SSH(SecureShell)服务的安全配置是我们日常工作中不可忽视的重要环节,本文将从基础配置到高级安全加固,全面解析SSH服务的各项参... 目录概述基础配置详解端口与监听设置主机密钥配置认证机制强化禁用密码认证禁止root直接登录实现双因素

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

在Linux终端中统计非二进制文件行数的实现方法

《在Linux终端中统计非二进制文件行数的实现方法》在Linux系统中,有时需要统计非二进制文件(如CSV、TXT文件)的行数,而不希望手动打开文件进行查看,例如,在处理大型日志文件、数据文件时,了解... 目录在linux终端中统计非二进制文件的行数技术背景实现步骤1. 使用wc命令2. 使用grep命令

Python get()函数用法案例详解

《Pythonget()函数用法案例详解》在Python中,get()是字典(dict)类型的内置方法,用于安全地获取字典中指定键对应的值,它的核心作用是避免因访问不存在的键而引发KeyError错... 目录简介基本语法一、用法二、案例:安全访问未知键三、案例:配置参数默认值简介python是一种高级编

python 常见数学公式函数使用详解(最新推荐)

《python常见数学公式函数使用详解(最新推荐)》文章介绍了Python的数学计算工具,涵盖内置函数、math/cmath标准库及numpy/scipy/sympy第三方库,支持从基础算术到复杂数... 目录python 数学公式与函数大全1. 基本数学运算1.1 算术运算1.2 分数与小数2. 数学函数

Linux如何快速检查服务器的硬件配置和性能指标

《Linux如何快速检查服务器的硬件配置和性能指标》在运维和开发工作中,我们经常需要快速检查Linux服务器的硬件配置和性能指标,本文将以CentOS为例,介绍如何通过命令行快速获取这些关键信息,... 目录引言一、查询CPU核心数编程(几C?)1. 使用 nproc(最简单)2. 使用 lscpu(详细信