嵌入式驱动学习第三周——设备号与字符设备的注册、分配、释放

前言

   这一篇博客来谈谈字符设备的注册、分配与释放。

   嵌入式驱动学习专栏将详细记录博主学习驱动的详细过程,未来预计四个月将高强度更新本专栏,喜欢的可以关注本博主并订阅本专栏,一起讨论一起学习。现在关注就是老粉啦!

目录

  • 前言
  • 1. 设备号
    • 1.1 设备号组成
    • 1.2 设备号的获取
  • 2. 字符设备分配
    • 2.1 静态分配
    • 2.2 动态分配
    • 2.3 实际代码编写
  • 3. 字符设备注册
    • 3.1 cdev结构体
    • 3.2 cdev_init()函数
    • 3.3 cdev_add()函数
  • 4. 字符设备释放
  • 参考资料

1. 设备号

1.1 设备号组成

   Linux中,为了方便管理,每个设备都有一个设备号,由主设备号次设备号两部分组成。主设备号表示某一个具体的驱动,次设备号表示使用这个驱动的各个设备。

   Linux下提供了dev_t数据类型表示设备号,其是一个unsigned int 类型,32位的数据,其中高12位为主设备号,低20位为次设备号,因此主设备号的范围不要超过0~4095。

1.2 设备号的获取

   在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_t 中获取主设备号,将 dev_t 右移 20 位即可
  • MINOR 用于从 dev_t 中获取次设备号,取 dev_t 的低 20 位的值即可
  • MKDEV 用于将给定的主设备号和次设备号的值组合成 dev_t 类型的设备号

2. 字符设备分配

   设备号的分配有两种,一种是静态分配,一种是动态分配

2.1 静态分配

   所谓的静态分配就是直接在注册时指定设备号,这个设备号可以是驱动开发者静态的指定一个设备号,比如200。有一些常用的设备号已经被 Linux 内核开发者给分配掉了,具体分配的内容可以查看文档 Documentation/devices.txt。并不是说内核开发者已经分配掉的主设备号我们就不能用了,具体能不能用还得看我们的硬件平台运行过程中有没有使用这个主设备号。

   使用cat /proc/devices可以查看当前系统中使用了的设备号,字符设备、块设备、网络设备均可以看到。

在这里插入图片描述

   可以使用函数register_chrdev_region来用给定的主设备号进行注册

/* * @description: 使用静态分配注册设备号* @param-from : 要分配的设备编号范围的初始值(次设备号常设为0)* @param-count: 连续编号范围* @param-name : 编号相关联的设备名称* @return     : 当返回值小于0,表示注册失败*/
int register_chrdev_region(dev_t from, unsigned count, const char *name)

2.2 动态分配

   静态分配还是不够灵活,需要先查看已使用的设备号,因此常用的还是动态分配,等到运行时,让系统自动分配一个没有使用的设备号,这样就避免了冲突。

/** @description    : 向linux申请一个没有使用过的设备号* @param-dev      : 用于保存申请到的设备号* @param-baseminor: 次设备号起始地址,alloc_chrdev_region 可以申请一段连续的多个设备号,这些设备号的主设备号一样,但是次设备号不同,次设备号以 baseminor 为起始地址地址开始递增。一般 baseminor 为 0,也就是说次设备号从 0 开始* @param-count    : 要申请的设备号数量* @param-name     : 设备名字* @return         : 小于0,则错误,自动分配设备号错误。否则分配得到的设备号就被第一个参数带出来*/
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name)

   用alloc_chrdev_region()函数得到设备号后,需要利用MAJOR宏和MINOR宏获取一下主设备号和次设备号。例如下面:

	alloc_chrdev_region(&chrdevTest.devid, 0, chrdevTest_CNT, chrdevTest_NAME);     // 申请设备号chrdevTest.major = MAJOR(chrdevTest.devid);     // 获取主设备号chrdevTest.minor = MINOR(chrdevTest.devid);     // 获取次设备号

   最后卸载驱动的时候需要释放设备号:

/** @description: 释放设备号* @param-from : 要释放的设备号* @param-count: 从from开始,要释放的设备号数量* @return     : 无*/
void unregister_chrdev_region(dev_t from, unsigned count)

2.3 实际代码编写

   在实际操作中,我们可以通过判断是否已经给定主设备号来选择静态分配还是动态分配,具体写法如下:

if (chrdevTest.major) {chrdevTest.devid = MKDEV(chrdevTest.major, 0);register_chrdev_region(chrdevTest.devid, chrdevTest_CNT, chrdevTest_NAME);
} else {alloc_chrdev_region(&chrdevTest.devid, 0, chrdevTest_CNT, chrdevTest_NAME);     // 申请设备号chrdevTest.major = MAJOR(chrdevTest.devid);     // 获取主设备号chrdevTest.minor = MINOR(chrdevTest.devid);     // 获取次设备号
}

3. 字符设备注册

3.1 cdev结构体

   在Linux中使用cdev结构体表示一个字符设备,cdev在include/linux/cdev.h文件中的定义:

struct cdev {struct kobject kobj;struct module *owner;const struct file_operations *ops;struct list_head list;dev_t dev;unsigned int count;
};

   在 cdev 中有两个重要的成员变量:opsdev,这两个就是字符设备文件操作函数集合file_operations 以及设备号 dev_t。编写字符设备驱动之前需要定义一个 cdev 结构体变量,一般在设备结构体中定义,如下所示:

struct chrdevTest_dev {dev_t devid;struct cdev cdev;struct class *class;struct device *device;int major;int minor;struct device_node *nd;
};

3.2 cdev_init()函数

   使用cdev_init函数进行初始化:

/** @description: 初始化cdev结构体* @param-cdev : 要初始化的cdev结构体变量* @param-fops : 字符设备文件操作函数集合* @return     : 无*/
void cdev_init(struct cdev *cdev, const struct file_operations *fops)

3.3 cdev_add()函数

   cdev_add用于向Linux系统添加字符设备(cdev结构体变量)

/** @description: 向系统添加字符设备* @param-p    : 指向要添加的字符设备(cdev结构体变量)* @param-dev  : 设备所使用的设备号* @param-count: 要添加的设备数量* @return     : 无*/
int cdev_add(struct cdev *p, dev_t dev, unsigned count)

4. 字符设备释放

   卸载驱动时,一定要用cdev_del函数从Linux内核中删除相应的字符设备,即在出口函数中编写cdev_del。

/* * @description: 从Linux内核中删除相应的字符设备* @param-p    : 要删除的字符设备* @return     : 无*/
void cdev_del(struct cdev *p)

   删除字符设备后,再释放设备号,因此完整的流程是:

	cdev_del(&chrdevTest.cdev);unregister_chrdev_region(chrdevTest.devid, chrdevTest_CNT);

参考资料

[1] 【正点原子】I.MX6U嵌入式Linux驱区动开发指南 第四十章、第四十二章
[2] linux驱动2:设备号与字符设备注册与注销
[3] 字符设备驱动之register_chrdev_region()系列

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

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

相关文章

盲盒抽卡机小程序——开启神秘之旅!

亲爱的朋友们&#xff0c;欢迎来到盲盒抽卡机小程序&#xff01;这里&#xff0c;是一个充满神秘与惊喜的世界&#xff0c;让你随时随地体验抽卡的乐趣。在这里&#xff0c;你可以轻松尝试各种盲盒&#xff0c;发现隐藏的宝藏&#xff0c;感受心跳加速的刺激。 【丰富多样的盲…

Unity类银河恶魔城学习记录9-4 p92 Death of entity源代码

Alex教程每一P的教程原代码加上我自己的理解初步理解写的注释&#xff0c;可供学习Alex教程的人参考 此代码仅为较上一P有所改变的代码 【Unity教程】从0编程制作类银河恶魔城游戏_哔哩哔哩_bilibili PlayerStat using System.Collections; using System.Collections.Generi…

k8s 安全机制详解

目录 一、安全机制 三个主要安全机制 认证&#xff08;Authentication&#xff09;&#xff1a; 鉴权&#xff08;Authorization&#xff09;&#xff1a; 准入控制&#xff08;Admission Control&#xff09;&#xff1a; 二、 认证 认证方式 对比总结 认证机制框架和相关组件…

小程序学习 1

pages/goods/search/home.wxml首页功能设定 1. loading入场 2. 下拉刷新 3. 搜索栏 4. 分类切换 5. 商品列表 6. 规格弹层 7. 加载更多 <view style"text-align: center; color: #b9b9b9" wx:if"{{pageLoading}}"><t-loading theme"circula…

Java中的常见类“Math”(一)用法详解

Java 中的 Math 类包含了许多数学函数&#xff0c;可以进行各种数学计算。Math 类是 Java 标准库的一部分&#xff0c;不需要额外引入即可使用。它包含在 java.lang 包中&#xff0c;而 java.lang 包中的类会自动导入到每个 Java 源文件中&#xff0c;所以在编写 Java 程序时无…

uView Swiper 轮播图

该组件一般用于导航轮播&#xff0c;广告展示等场景,可开箱即用&#xff0c;具有如下特点&#xff1a; 自定义指示器模式&#xff0c;可配置指示器样式3D轮播图效果&#xff0c;满足不同的开发需求可配置显示标题&#xff0c;涵盖不同的应用场景具有设置加载状态和嵌入视频的能…

月份的天数

目录 题目描述 输入格式 输出格式 输入输出样例 说明/提示 题目描述 输入年份和月份&#xff0c;输出这一年的这一月有多少天。需要考虑闰年。 输入格式 输入两个正整数&#xff0c;分别表示年份 y 和月数 m&#xff0c;以空格隔开。 输出格式 输出一行一个正整数&…

【单调队列】

滑动窗口 给定一个大小为 n≤10^6 的数组。 有一个大小为 k 的滑动窗口&#xff0c;它从数组的最左边移动到最右边。 你只能在窗口中看到 k 个数字。 每次滑动窗口向右移动一个位置。 以下是一个例子&#xff1a; 该数组为 [1 3 -1 -3 5 3 6 7]&#xff0c;k 为 3。 窗口…

利用ffmpeg对两个音频文件进行混音处理

前言 最近&#xff0c;拿到了一个语音识别程序&#xff0c;想测试一下它识别的准确性。原本程序有一段自己的测试音频&#xff0c;准确性还可以&#xff0c;但是&#xff0c;自己想增加一下测试素材的复杂性。想到了在原本的测试音频中引入干扰数据&#xff08;噪点&#xff…

python中的**可以表示什么??

在Python中&#xff0c;** 有两个主要的用途&#xff1a; 作为幂运算符&#xff1a;a ** b 表示a的b次方。例如&#xff0c;2 ** 3 会返回 8&#xff0c;因为2的3次方等于8。 在函数调用或定义时作为关键字参数的解包&#xff1a; 当你有一个字典&#xff0c;并且你想将这个字…

C++从零开始(day44)——二叉搜索树

这是关于一个普通双非本科大一学生的C的学习记录贴 在此前&#xff0c;我学了一点点C语言还有简单的数据结构&#xff0c;如果有小伙伴想和我一起学习的&#xff0c;可以私信我交流分享学习资料 那么开启正题 今天分享的是关于二叉搜索树的知识点 1.二叉搜索树概念 二叉搜…

python+realsense

单目相机(RGB影像):分辨率&#xff1a;320180,320240,424240,640360,640480,848480,960540,1280720,19201080&#xff1b;帧率&#xff1a;6,15,30,60 按照博文Python实战之Realsense_realsense python-CSDN博客的代码显示如下&#xff08;我更改了分辨率和帧率&#xff0c;大…

java八股文 笔记(持续更新中~)

1 Redis 2Mysql 3JVM 4java基础底层 5 spring 6 微服务 7.......(持续更新) One:Redis篇 1.穿透 2&#xff1a;击穿 3&#xff1a;雪崩 3 33 4:双写一致 5.持久化

【网站项目】012医院住院管理系统

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

内网穿透的应用-如何在Linux系统Docker安装JSON Crack并实现远程访问本地数据

文章目录 1. 在Linux上使用Docker安装JSONCrack2. 安装Cpolar内网穿透工具3. 配置JSON Crack界面公网地址4. 远程访问 JSONCrack 界面5. 固定 JSONCrack公网地址 JSON Crack 是一款免费的开源数据可视化应用程序&#xff0c;能够将 JSON、YAML、XML、CSV 等数据格式可视化为交互…

mysql中 多表查询介绍

在 MySQL 中&#xff0c;多表查询是 SQL 语句的重要组成部分&#xff0c;用于从两个或多个表中检索数据。多表查询可以帮助我们更灵活地处理复杂的数据关系&#xff0c;并从中获取所需的信息。以下是 MySQL 中常见的多表查询及其特点、区别和应用场景。 常见多表查询 1. **内连…

《量子计算:下一个大风口,还是一个热炒概念?》

引言 量子计算,作为一项颠覆性的技术,一直以来备受关注。它被认为是未来计算领域的一次革命,可能改变我们对计算能力和数据处理的理解。然而,随着技术的不断进步和商业应用的探索,人们开始思考,量子计算到底是一个即将到来的大风口,还是一个被过度炒作的概念? 量子计…

归并排序 刷题笔记

归并排序的写法 归并排序 分治双指针 1.定义一个mid if(l>r)return ; 2.分治 sort(q,l,mid); sort(q,mid1,r); 3. 双指针 int il,jmid,k0; 将双序列扫入 缓存数组 条件 while(i<mid&&j<r) 两个数列比较大小 小的一方 进入缓存数组 4. 扫尾 while(…

基于USB数据采集器的电能质量监测系统

目录 摘要 1 Abstract 3 第一章 引言 5 1.1 研究背景 5 1.2 研究现状 5 1.3 研究工作 6 第二章 电能质量及其分析方法 8 2.1 电能质量定义 8 2.2 电力系统频率偏差 8 2.3 供电电压允许偏差 10 2.4 公用电网谐波 12 2.5 电压波动和闪变 13 2.6 三相电压不平衡度 15 2.7 常见的电…