Linux驱动入门(6.2)按键驱动和LED驱动 --- 将逻辑电平与物理电平分离
前言
(1)在学习完Linux驱动入门(6)LED驱动—设备树之后,我们发现一个问题,设备树明明的gpios信息明明有三个元素gpios = <&gpio5 3 GPIO_ACTIVE_LOW>; &gpio5 3 用来确定控制那个引脚,而GPIO_ACTIVE_LOW究竟有什么用呢?
(2)通过前面的实验,我们发现,GPIO_ACTIVE_LOW似乎是没有使用上的。那么写上这个有什么用呢?
(3)Linux设备树中既然设置了这个元素,那么肯定是有意义的。接下来,我将讲解Linux中逻辑电平和物理电平之间的关系。
逻辑电平的意义
为什么需要逻辑电平
(1)在前面的代码里面,我们发现如果这个LED驱动硬件发生了改动,比如GND和VCC位置调整一下,代码就要进行比较多的改动。因为Linux的代码很多,很容易漏掉某个地方,导致硬件上的小改动,明明写好的软件又要做很多工作,调试,检查。
(2)Linux要与硬件进行强隔离,所以提出了逻辑电平的概念。

逻辑电平和物理电平的关系
(1)物理电平就是真实的电压值,物理电平1是指真实的高电平。例如TTL标准中,如果引脚接受到的是3.3V,那么就是1。如果引脚接受到的是0V,那么就是0。
(2)逻辑电平就是抽象出来的逻辑状态,逻辑电平的1是指有效电平。例如,上面左边的图,按键被按下,引脚为低电平,因此低电平是有效电平,对于这个按键的逻辑电平来说,1就是低电平。而右边这张图相反。
引入逻辑电平的好处
(1)引入逻辑电平之后,如果硬件只是有效电平发生了改变,驱动程序上就不再需要改动了。我们只需要将设备树中的gpios = <&gpio5 3 GPIO_ACTIVE_LOW>;改成gpios = <&gpio5 3 GPIO_ACTIVE_HIGH>;即可。
(2)这样做,能够实现软件和硬件上的强隔离作用,让同一个驱动程序对于各种类似的硬件上有更高适配能力。
编程中与上文的区别
(1)在讲解Linux驱动入门(4)LED驱动的时候,我提及了几个GPIO子系统函数进行了讲解介绍。而本文需要讲解的函数,其本质上也不过是这个几个函数的微调,底层调用的函数大差不差,感兴趣的可以自行阅读源码。既然只是微调,为什么不把老版本的删了?这当然是为了兼容老版本代码,所以没有删除他们。
of_get_gpio_flags()获取GPIO信息
函数介绍
(1)在上文中,我们获取gpio引脚号是调用的of_get_gpio()这个函数,而本文是使用的of_get_gpio_flags()函数。
(2)如果阅读源码会发现,of_get_gpio()就是调用的of_get_gpio_flags()函数,不过第三个参数是传入的一个空指针,并没有获取有效电平信息。

/* 作用 : 从设备树中获取GPIO引脚的标志信息的函数* 传入参数 :* np :设备节点* index : 节点中的索引* flags : 存储有效电平的信息* 返回值 : GPIO的引脚号
*/
int of_get_gpio_flags(struct device_node *np, int index, enum of_gpio_flags *flags)
原来获取GPIO信息
gpios[i].gpio = of_get_gpio(np, i); //获得gpio信息
现在获取GPIO信息
gpios[i].gpio = of_get_gpio_flags(np, i, &flag);
devm_gpio_request_one()设置引脚初始化
函数介绍
(1)在上文中,我们是调用gpio_request()函数申请到GPIO,然后使用gpio_direction_output()函数将引脚设置成输出。
(2)现在这里只需要调用一个devm_gpio_request_one()函数即可。
(3)虽然这里调用的函数少了,但是需要进行的操作也变多了。
/* 作用 : 从设备树中获取GPIO引脚的标志信息的函数* 传入参数 :* dev :要申请GPIO的设备* gpio : 引脚号* flags : 有效电平,引脚的输入输出方向,默认输出电平(物理电平)信息* label : 注册GPIO时候的名字* 返回值 : 如果返回值小于0,表示申请失败
*/
int devm_gpio_request_one(struct device *dev, unsigned gpio,unsigned long flags, const char *label)
原来设置GPIO
/*------------GPIO设置成默认低电平输出------------*/
//申请指定GPIO引脚,申请的时候需要用到名字
err = gpio_request(gpios[i].gpio, gpios[i].name);
//如果返回值小于0,表示申请失败
if (err < 0)
{//如果GPIO申请失败,打印出是哪个LED申请出现问题printk("can not request gpio %s \n", gpios[i].name);return -ENODEV;
}
//如果GPIO申请成功,设置输出低电平
gpio_direction_output(gpios[i].gpio, 0);
/*------------GPIO设置成默认高电平输出------------*/
//申请指定GPIO引脚,申请的时候需要用到名字
err = gpio_request(gpios[i].gpio, gpios[i].name);
//如果返回值小于0,表示申请失败
if (err < 0)
{//如果GPIO申请失败,打印出是哪个LED申请出现问题printk("can not request gpio %s \n", gpios[i].name);return -ENODEV;
}
//如果GPIO申请成功,设置输出高电平
gpio_direction_output(gpios[i].gpio, 1);
/*------------GPIO设置成输入------------*/
//申请指定GPIO引脚,申请的时候需要用到名字
err = gpio_request(gpios[i].gpio, gpios[i].name);
//如果返回值小于0,表示申请失败
if (err < 0)
{//如果GPIO申请失败,打印出是哪个LED申请出现问题printk("can not request gpio %s \n", gpios[i].name);return -ENODEV;
}
//如果GPIO申请成功,设置为输入引脚
gpio_direction_input(gpios[i].gpio);
现在设置GPIO
/*------------GPIO设置成默认低电平输出(注意,这里是物理电平)------------*/
gpios[i].flag = GPIOF_OUT_INIT_LOW; //将GPIO设置成默认低电平输出(注意,这里是物理电平)
if (flag & OF_GPIO_ACTIVE_LOW) //判断有效电平是否为低电平
{gpios[i].flag |= GPIOF_ACTIVE_LOW;
}
printk("gpios[%d].flag is %d \r\n",i,gpios[i].flag);
err = devm_gpio_request_one(&pdev->dev, gpios[i].gpio, gpios[i].flag, gpios[i].name);
//如果返回值小于0,表示申请失败
if (err < 0)
{//如果GPIO申请失败,打印出是哪个LED申请出现问题printk("can not request gpio %s \n", gpios[i].name);return -ENODEV;
}
/*------------GPIO设置成默认高电平输出(注意,这里是物理电平)------------*/
gpios[i].flag = GPIOF_OUT_INIT_HIGH; //将GPIO设置成默认高电平输出(注意,这里是物理电平)
if (flag & OF_GPIO_ACTIVE_LOW) //判断有效电平是否为低电平
{gpios[i].flag |= GPIOF_ACTIVE_LOW;
}
printk("gpios[%d].flag is %d \r\n",i,gpios[i].flag);
err = devm_gpio_request_one(&pdev->dev, gpios[i].gpio, gpios[i].flag, gpios[i].name);
//如果返回值小于0,表示申请失败
if (err < 0)
{//如果GPIO申请失败,打印出是哪个LED申请出现问题printk("can not request gpio %s \n", gpios[i].name);return -ENODEV;
}
/*------------GPIO设置成输入------------*/
gpios[i].flag = GPIOF_IN; //将引脚设置成输入方向
if (flag & OF_GPIO_ACTIVE_LOW) //判断有效电平是否为低电平
{gpios[i].flag |= GPIOF_ACTIVE_LOW;
}
printk("gpios[%d].flag is %d \r\n",i,gpios[i].flag);
err = devm_gpio_request_one(&pdev->dev, gpios[i].gpio, gpios[i].flag, gpios[i].name);
//如果返回值小于0,表示申请失败
if (err < 0)
{//如果GPIO申请失败,打印出是哪个LED申请出现问题printk("can not request gpio %s \n", gpios[i].name);return -ENODEV;
}
gpiod_set_value()设置引脚输出电平(逻辑电平)
函数介绍
(1)上文我们调用gpio_set_value()函数是设置的物理电平,而本文将会使用gpiod_set_value()函数设置逻辑电平。
(2)感兴趣的朋友可以看看gpio_set_value()函数和gpiod_set_value()函数的底层实现,我们会发现他们都调用了gpiod_set_raw_value()函数。只不过gpio_set_value()是直接将自己的参数传递进去,而gpiod_set_value()函数会判断有效电平信息,然后根据有效电平信息翻转Value值。
(3)这里需要注意的一点是,gpiod_set_value()函数第一参数传入的是gpio_desc结构体类型指针,而gpio_set_value()传入的是引脚号。我们可以调用gpio_to_desc()函数利用引脚号获得gpio_desc结构体类型指针。

原来设置GPIO输出电平(物理电平)
/* gpios[(int)tmp_buf[0]].gpio是引脚号* tmp_buf[1]是要设置的物理电平信息
*/
gpio_set_value(gpios[(int)tmp_buf[0]].gpio, tmp_buf[1]);
现在设置GPIO输出电平(逻辑电平)
(1)因为gpiod_set_value()函数第一个参数需要传入的是一个gpio_desc结构体。所以我在probe函数中获取,并且存入gpios结构体中。
(2)这里需要注意了,我们现在设置的是逻辑电平了。如果你的LED需要低电平点亮,你在设备树中设置了有效电平是低电平。那么现在gpiod_set_value()函数传入的第二个值,如果是1,输出的其实是低电平!!!
/*---- 在probe函数中我们使用了gpio_to_desc函数获得gpio_desc结构体 */
gpios[i].gpiod = gpio_to_desc(gpios[i].gpio);
/*---- 下面这段是在驱动程序中write函数中修改 */
/* gpios[(int)tmp_buf[0]].gpio是引脚号* tmp_buf[1]是要设置的物理电平信息
*/
gpiod_set_value(gpios[(int)tmp_buf[0]].gpiod, tmp_buf[1]);
需要注意gpio_desc结构体无法被访问
(1)这里有一个注意的点,gpio_desc结构体是Linux内核结构体。他与我们所编写的C文件编译环境是隔离的。所以我们没有访问gpio_desc结构体权限。这个问题卡了我很久,望各位了解。
(2)如果有头铁的兄弟说,哎,我就是牛逼,我就要访问,怎么滴?可以的,我也给头铁的兄弟们提供思路。
// 在对于的内核文件中写入下面这个宏,然后重新编译Linux内核。
EXPORT_SYMBOL(gpio_to_desc);
gpiod_get_value()获得引脚电平(逻辑电平)
函数介绍
(1)同样的道理,我们阅读gpiod_get_value()和gpio_get_value()函数源码会发现,他们都会调用gpiod_get_raw_value()函数获取真实的物理电平。
(2)但是,gpiod_get_value()会根据设备树中设置的有效电平,翻转gpiod_get_raw_value()函数返回值。而gpio_get_value()函数则是直接将gpiod_get_raw_value()函数返回值输出。
(3)因此gpiod_get_value()返回的是逻辑电平,gpio_get_value()返回的是物理电平。

原来获取GPIO电平(物理电平)
tmp_buf[1] = gpio_get_value(gpios[(int)tmp_buf[0]].gpio);
现在获取GPIO电平(逻辑电平)
(1)同输出电平一样,如果设备树中,LED驱动设置的有效电平是低电平。那么我们调用gpiod_get_value()函数发现LED的物理电平是高电平的时候,他返回的却是0!
tmp_buf[1] = gpiod_get_value(gpios[(int)tmp_buf[0]].gpiod);
总结
(1)将逻辑电平和物理电平隔离之后,驱动文件不再需要修改。如果硬件产生了更换,也只需要修改设备树。
(2)编写应用程序的程序员,也不需要管LED点亮到底是高电平还是低电平,在他眼里,输入1就是点亮,输入0就是熄灭。
相关文章:
Linux驱动入门(6.2)按键驱动和LED驱动 --- 将逻辑电平与物理电平分离
前言 (1)在学习完Linux驱动入门(6)LED驱动—设备树之后,我们发现一个问题,设备树明明的gpios信息明明有三个元素gpios <&gpio5 3 GPIO_ACTIVE_LOW>; &gpio5 3 用来确定控制那个引脚…...
CentOS系统环境搭建(十四)——CentOS7.9安装elasticsearch-head
centos系统环境搭建专栏🔗点击跳转 关于node的安装请看上一篇CentOS系统环境搭建(十三)——CentOS7安装nvm,🔗点击跳转。 CentOS7.9安装elasticsearch-head 文章目录 CentOS7.9安装elasticsearch-head1.下载2.解压3.修…...
设计HTML5图像和多媒体
在网页中的文本信息直观、明了,而多媒体信息更富内涵和视觉冲击力。恰当使用不同类型的多媒体可以展示个性,突出重点,吸引用户。在HTML5之前,需要借助插件为网页添加多媒体,如Adobe Flash Player、苹果的QuickTime等。…...
基于YOLOv8模型和Caltech数据集的行人检测系统(PyTorch+Pyside6+YOLOv8模型)
摘要 基于YOLOv8模型和Caltech数据集的行人检测系统可用于日常生活中检测与定位行人,利用深度学习算法可实现图片、视频、摄像头等方式的行人目标检测,另外本系统还支持图片、视频等格式的结果可视化与结果导出。本系统采用YOLOv8目标检测算法训练数据集…...
Flutter 宽高自适应
在Flutter开发中也需要宽高自适应,手动写一个工具类,集成之后在像素后面直接使用 px或者 rpx即可。 工具类代码如下: import dart:ui;class HYSizeFit {static double screenWidth 0.0;static double screenHeight 0.0;static double phys…...
LeetCode 0833. 字符串中的查找与替换
【LetMeFly】833.字符串中的查找与替换 力扣题目链接:https://leetcode.cn/problems/find-and-replace-in-string/ 你会得到一个字符串 s (索引从 0 开始),你必须对它执行 k 个替换操作。替换操作以三个长度均为 k 的并行数组给出:indices,…...
Redis对象和五种常用数据类型
Redisobject 对象 对象分为键对象和值对象 键对象一般是string类型 值对象可以是string,list,set,zset,hash q:redisobj的结构 typedef struct redisObject { //类型 unsigned type:4; //编码 unsigned encoding:4; //指向底层实现…...
常用的Elasticsearch查询DSL
1.基本查询 GET /index_name/_search {"query": {"match": {"dispatchClass": "1"}} }2.多条件查询 GET /index_name/_search {"query": {"bool": {"must": [{"match": {"createUser&…...
计算机网络笔记
TCP有连接可靠服务 TCP特点: 1.TCP是面向连接的传输层协议; 2.每条TCP连接只能有两个端点,每条TCP连接是一对一的; 3.TCP提供可靠交付,保证传送数据无差错,不丢失,不重复且有序; 4.…...
高效反编译luac文件
对于游戏开发人员,有时候希望从一些游戏apk中反编译出源代码,进行学习,但是如果你触碰到法律边缘,那么你要非常小心。 这篇文章,我针对一些用lua写客户端或者服务器的编译过的luac文件进行反编译,获取其源代码的过程。 这里我不赘述如何反编译解压apk包的过程了,只说重点…...
密码湘军,融合创新!麒麟信安参展2023商用密码大会,铸牢数据安全坚固堡垒
2023年8月9日至11日,商用密码大会在郑州国际会展中心正式开幕。本次大会由国家密码管理局指导,中国密码学会支持,郑州市人民政府、河南省密码管理局主办,以“密码赋能美好发展”为主题,旨在推进商用密码创新驱动、前沿…...
关于视频监控平台EasyCVR视频汇聚平台建设“明厨亮灶”具体实施方案以及应用
一、方案背景 近几年来,餐饮行业的食品安全、食品卫生等新闻频频发生,比如某火锅店、某网红奶茶,食材以次充好、后厨卫生被爆堪忧,种种问题引起大众关注和热议。这些负面新闻不仅让餐饮门店的品牌口碑暴跌,附带的连锁…...
区块链系统探索之路:私钥的压缩和WIF格式详解
在前面章节中,我们详细介绍了公钥的压缩,在比特币网络中,一个私钥可以对应两个地址,一个地址是由未压缩公钥所生成的地址,另一个就是由压缩公钥所创建的地址,从公钥到区块链地址的转换算法,我们…...
DiffusionDet: Diffusion Model for Object Detection
DiffusionDet: Diffusion Model for Object Detection 论文概述不同之处整体流程 论文题目:DiffusionDet: Diffusion Model for Object Detection 论文来源:arXiv preprint 2022 论文地址:https://arxiv.org/abs/2211.09788 论文代码…...
CH01_重构、第一个示例
概述 在这一章节,作者给出了一个戏剧演出团售票的示例:剧目有悲剧(tragedy)和喜剧(comedy);为了卖出更多的票,剧团则更具观众的数量来为下次演出打折扣(大致意思是这次的…...
学习篇之React Fiber概念及原理
什么是React Fibber? React Fiber 是 React 框架的一种底层架构,为了改进 React 的渲染引擎,使其更加高效、灵活和可扩展。 传统上,React 使用一种称为堆栈调和递归算法来处理虚拟 DOM 的更新,这种方法在大型应用或者…...
商城-学习整理-高级-全文检索-ES(九)
目录 一、ES简介1、网址2、基本概念1、Index(索引)2、Type(类型)3、Document(文档)4、倒排索引机制4.1 正向索引和倒排索引4.2 正向索引4.3 倒排索引 3、相关软件及下载地址3.1 Kibana简介3.2 logstash简介…...
无人机跟随一维高度避障场景--逻辑分析
无人机跟随一维高度避障场景--逻辑分析 1. 源由2. 视频3. 问题3.1 思维发散3.2 问题收敛 4. 图示4.1 水平模式4.2 下坡模式4.3 上坡模式4.4 碰撞分析 5. 总结5.1 一维高度避障场景5.2 业界跟随产品5.3 APM集成跟随 6. 参考资料7. 补充资料 - 大疆智能跟随7.1 炸机7.2 成功 1. 源…...
Android Studio Giraffe控制台乱码
这几天在使用Android Studio Giraffe进行一个App的开发,在项目构建的时候,控制台输出中文都是乱码,看着很不爽,进行了两项配置,中文就可以正常输出了,看起来就爽多了。 第一个配置:点击Help菜单…...
云原生 envoy xDS 动态配置 java控制平面开发 支持restful grpc实现 EDS 动态endpoint配置
envoy xDS 动态配置 java控制平面开发 支持restful grpc 动态endpoint配置 大纲 基础概念Envoy 动态配置API配置方式动静结合的配置方式纯动态配置方式实战 基础概念 Envoy 的强大功能之一是支持动态配置,当使用动态配置时,我们不需要重新启动 Envoy…...
Vim 调用外部命令学习笔记
Vim 外部命令集成完全指南 文章目录 Vim 外部命令集成完全指南核心概念理解命令语法解析语法对比 常用外部命令详解文本排序与去重文本筛选与搜索高级 grep 搜索技巧文本替换与编辑字符处理高级文本处理编程语言处理其他实用命令 范围操作示例指定行范围处理复合命令示例 实用技…...
零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?
一、核心优势:专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发,是一款收费低廉但功能全面的Windows NAS工具,主打“无学习成本部署” 。与其他NAS软件相比,其优势在于: 无需硬件改造:将任意W…...
React第五十七节 Router中RouterProvider使用详解及注意事项
前言 在 React Router v6.4 中,RouterProvider 是一个核心组件,用于提供基于数据路由(data routers)的新型路由方案。 它替代了传统的 <BrowserRouter>,支持更强大的数据加载和操作功能(如 loader 和…...
iPhone密码忘记了办?iPhoneUnlocker,iPhone解锁工具Aiseesoft iPhone Unlocker 高级注册版分享
平时用 iPhone 的时候,难免会碰到解锁的麻烦事。比如密码忘了、人脸识别 / 指纹识别突然不灵,或者买了二手 iPhone 却被原来的 iCloud 账号锁住,这时候就需要靠谱的解锁工具来帮忙了。Aiseesoft iPhone Unlocker 就是专门解决这些问题的软件&…...
UDP(Echoserver)
网络命令 Ping 命令 检测网络是否连通 使用方法: ping -c 次数 网址ping -c 3 www.baidu.comnetstat 命令 netstat 是一个用来查看网络状态的重要工具. 语法:netstat [选项] 功能:查看网络状态 常用选项: n 拒绝显示别名&#…...
2.Vue编写一个app
1.src中重要的组成 1.1main.ts // 引入createApp用于创建应用 import { createApp } from "vue"; // 引用App根组件 import App from ./App.vue;createApp(App).mount(#app)1.2 App.vue 其中要写三种标签 <template> <!--html--> </template>…...
屋顶变身“发电站” ,中天合创屋面分布式光伏发电项目顺利并网!
5月28日,中天合创屋面分布式光伏发电项目顺利并网发电,该项目位于内蒙古自治区鄂尔多斯市乌审旗,项目利用中天合创聚乙烯、聚丙烯仓库屋面作为场地建设光伏电站,总装机容量为9.96MWp。 项目投运后,每年可节约标煤3670…...
基于Docker Compose部署Java微服务项目
一. 创建根项目 根项目(父项目)主要用于依赖管理 一些需要注意的点: 打包方式需要为 pom<modules>里需要注册子模块不要引入maven的打包插件,否则打包时会出问题 <?xml version"1.0" encoding"UTF-8…...
安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲
文章目录 前言第一部分:体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分:体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...
vulnyx Blogger writeup
信息收集 arp-scan nmap 获取userFlag 上web看看 一个默认的页面,gobuster扫一下目录 可以看到扫出的目录中得到了一个有价值的目录/wordpress,说明目标所使用的cms是wordpress,访问http://192.168.43.213/wordpress/然后查看源码能看到 这…...
