Linux驱动开发笔记(二) 基于字符设备驱动的GPIO操作

news2024/10/7 12:23:31

文章目录

  • 前言
  • 一、设备驱动的作用与本质
    • 1. 驱动的作用
    • 2. 有无操作系统的区别
  • 二、内存管理单元MMU
  • 三、相关函数
    • 1. ioremap( )
    • 2. iounmap( )
    • 3. class_create( )
    • 4. class_destroy( )
  • 四、GPIO的基本知识
    • 1. GPIO的寄存器进行读写操作流程
    • 2. 引脚复用
    • 2. 定义GPIO寄存器物理地址
  • 五、实验代码
    • 1. 宏定义出需要的地址
    • 2. 编写LED字符设备结构体且初始化
    • 3. container_of( )函数
    • 4. file_operations结构体成员函数的实现
    • 5. 实验效果


前言

  前段时间我们学习了字符驱动,并实现了字符的回环发送,这部分我们将进行I/O的操作学习,以万能的点亮LED为例。

一、设备驱动的作用与本质

  直接操作寄存器点亮LED和通过驱动程序点亮LED最本质的区别就是有无使用操作系统。 有操作系统的存在则大大降低了应用软件与硬件平台的耦合度,它充当了我们硬件与应用软件之间的纽带, 使得应用软件只需要调用驱动程序接口API就可以让硬件去完成要求的开发,而应用软件则不需要关心硬件到底是如何工作的。

1. 驱动的作用

  设备驱动与底层硬件直接打交道,按照硬件设备的具体工作方式读写设备寄存器, 完成设备的轮询、中断处理、DMA通信,进行物理内存向虚拟内存的映射,最终使通信设备能够收发数据, 使显示设备能够显示文字和画面,使存储设备能够记录文件和数据。

2. 有无操作系统的区别

  无操作系统(即裸机)时的设备驱动也就是直接操作寄存器的方式控制硬件,在这样的系统中,虽然不存在操作系统,但是设备驱动是必须存在的。 一般情况下,对每一种设备驱动都会定义为一个软件模块,包含.h文件和.c文件,前者定义该设备驱动的数据结构并声明外部函数, 后者进行设备驱动的具体实现。其他模块需要使用这个设备的时候,只需要包含设备驱动的头文件然后调用其中的外部接口函数即可。 比如我们在51或者STM32中直接看手册查找对应的寄存器,然后往寄存器相应的位写入数据0或1便可以实现LED的亮灭。
  有操作系统时的设备驱动反观有操作系统。首先,驱动硬件工作的的部分仍然是必不可少的,其次,我们还需要将设备驱动融入内核。 为了实现这种融合,必须在所有的设备驱动中设计面向操作系统内核的接口,这样的接口由操作系统规定,对一类设备而言结构一致,独立于具体的设备,还是以led为例,我们就要将LED灯引脚对应的数据寄存器(物理地址)映射到程序的虚拟地址空间当中,然后我们就可以像操作寄存器一样去操作我们的虚拟地址啦!

二、内存管理单元MMU

  MMU是一个实际的硬件,为编程提供了方便统一的内存空间抽象,MMU内部有一个专门存放页表的页表地址寄存器,该寄存器存放着页表的具体位置,这使得只要程序在被分配的虚拟地址范围内进行读写操作,实际上就是对设备(寄存器)的访问,如下图所示。他的主要作用是将虚拟地址翻译成真实的物理地址同时管理和保护内存, 不同的进程有各自的虚拟地址空间,某个进程中的程序不能修改另外一个进程所使用的物理地址,以此使得进程之间互不干扰,相互隔离。 总体而言MMU具有如下功能:

  • 保护内存: MMU给一些指定的内存块设置了读、写以及可执行的权限,这些权限存储在页表当中,MMU会检查CPU当前所处的是特权模式还是用户模式,如果和操作系统所设置的权限匹配则可以访问,如果CPU要访问一段虚拟地址,则将虚拟地址转换成物理地址,否则将产生异常,防止内存被恶意地修改。
  • 提供方便统一的内存空间抽象,实现虚拟地址到物理地址的转换: CPU可以运行在虚拟的内存当中,虚拟内存一般要比实际的物理内存大很多,使得CPU可以运行比较大的应用程序。

在这里插入图片描述
在这里插入图片描述

三、相关函数

  上面提到了物理地址到虚拟地址的转换函数。包括ioremap()地址映射和取消地址映射iounmap()函数。

1. ioremap( )

//用于将物理内存地址映射到内核的虚拟地址空间
void __iomem *ioremap(phys_addr_t phys_addr, unsigned long size)

//定义寄存器物理地址
#define GPIO0_BASE (0xFDD60000)
#define GPIO0_DR (GPIO0_BASE+0x0000)

va_dr = ioremap(GPIO0_DR, 4);    // 将物理地址GPIO0_DR,映射给虚拟地址指针,这段地址大小为4个字节
val = ioread32(va_dr);			 //读取该地址的值,保存到临时变量,重新赋值
val |= (0x00400000);             // 设置GPIO0_A6引脚低电平
writel(val, va_dr);				 //把值重新写入到被映射后的虚拟地址当中,实际是往寄存器中写入了数据
  • 参数:
    • phys_addr:要映射的物理地址的起始地址
    • size:要映射的内存区域的大小(以字节为单位)
  • 返回值:
    • 如果成功,ioremap返回一个指向映射区域的虚拟地址的指针
    • 如果失败,返回NULL

  在使用ioremap函数将物理地址转换成虚拟地址之后,理论上我们便可以直接读写I/O内存,但是为了符合驱动的跨平台以及可移植性, 我们应该使用linux中指定的函数(如:iowrite8()、iowrite16()、iowrite32()、ioread8()、ioread16()、ioread32()等)去读写I/O内存,如下表所示:

函数名功能
unsigned int ioread8(void __iomem *addr)读取一个字节(8bit)
unsigned int ioread16(void __iomem *addr)读取一个字(16bit)
unsigned int ioread32(void __iomem *addr)读取一个双字(32bit)
void iowrite8(u8 data, void __iomem *addr)写入一个字节(8bit)
void iowrite16(u16 data, void __iomem *addr)写入一个字(16bit)
void iowrite32(u32 data, void __iomem *addr)写入一个双字(32bit)

2. iounmap( )

//取消地址映射
void iounmap(void *addr)

iounmap(va_dr);     //释放掉ioremap映射之后的起始地址(虚拟地址)
  • 参数
    • addr: 需要取消ioremap映射之后的起始地址(虚拟地址)。
  • 返回值: 无

3. class_create( )

//提交目录信息
#define class_create(owner, name) \
({
	static struct lock_class_key _key; \
	_class_create(owner, name, &_key); \
})
  • 参数
    • owner:THIS_MODULE (struct module结构体的首地址这个结构体存放了驱动的出口入口)
    • name:kobject对象名称
  • 返回值
    • 成功:返回结构体首地址
    • 失败:返回错误码指针

注:IS_ERR(cls); 判断是否为错误指针
  PTR_ERR(cls); 将错误码指针转换为错误码

4. class_destroy( )

//注销目录信息
void class_destroy(struct class *cls);
  • 参数
    • cls:结构体首地址
  • 返回值:无

四、GPIO的基本知识

1. GPIO的寄存器进行读写操作流程

  • 使能GPIO时钟(默认开启,不用设置)
  • 设置引脚复用为GPIO(复位默认为GPIO,不用配置)
  • 设置引脚属性(上下拉、速率、驱动能力,默认)
  • 控制GPIO引脚为输出,并输出高低电平

2. 引脚复用

  对于rockchip系类芯片,我们需要通过参考手册以及数据手册来确定引脚的复用功能。首先可以看到泰山派的小灯连接引脚,这里我们选择GPIO1_B0_d。
在这里插入图片描述
  通过查询rk3568官方资料,可以看到该引脚的复用功能如下所示。在这里插入图片描述
  再查找其复用功能存在于SYS_GRF寄存器,和复用相关的总共8个寄存器,如下图所示:
在这里插入图片描述

  查询 Rockchip_RK3568_TRM_Part1 手册,GRF_GPIO1B_IOMUX_L寄存器(由于GPIO1_b0是在低八位,下同),如下图所示:
在这里插入图片描述
  寄存器总共32位,高16位都是使能位,控制低16位的写使能,低16位对应4个引脚,每个引脚占用3bits,不同的值引脚复用为不同功能。与此同时由[14:12]进行具体功能的设定。
  我们可以查看到SYS_GRF寄存器的复用功能基地址为0xFDC60000。
在这里插入图片描述
  此时通过命令行输入可以查询到该寄存器的设置情况,可以看到默认为GPIO口模式。

//目标地址为Address Base(0xfdc60000)+offset(0x0008)
io -r -4 0xfdc60008

在这里插入图片描述

2. 定义GPIO寄存器物理地址

  需要设置的寄存器的地址为base+offset,由下图可以知道GPIO1的基地址为:0xFE740000
在这里插入图片描述
  接下来就是确定GPIO的是输入还是输出,我们这里需要的是GPIO_SWPORT_DDR_L。
在这里插入图片描述在这里插入图片描述
  可以看到GPIO_SWPORT_DDR_L的定义情况,这里我们可以重复上面提到的命令行,查看寄存器的设置情况,我们的b0应当是第1x7+1=8位。
在这里插入图片描述
  同样查看可以看到这里的值为0x00000700。
在这里插入图片描述
  数据寄存器选择GPIO_SWPORT_DR_L,大致流程和上面一样就不再赘述了。这里便完成了对GPIO的设置。

五、实验代码

在这里插入图片描述

1. 宏定义出需要的地址

#define GPIO1_BASE (0xFE740000)

//一个寄存器32位,其中高16位都是写使能位,控制低16位的写使能;低16位对应16个引脚,控制引脚的输出电平
#define GPIO1_DR_L (GPIO0_BASE + 0x0000)  // GPIO0的低十六位引脚的数据寄存器地址
#define GPIO1_DR_H (GPIO0_BASE + 0x0004)  // GPIO0的高十六位引脚的数据寄存器地址

//一个寄存器32位,其中高16位都是写使能位,控制低16位的写使能;低16位对应16个引脚,控制引脚的输入输出模式
#define GPIO1_DDR_L (GPIO0_BASE + 0x0008)   // GPIO0的低十六位引脚的数据方向寄存器地址
#define GPIO1_DDR_H (GPIO0_BASE + 0x000C)   // GPIO0的低十六位引脚的数据方向寄存器地址

2. 编写LED字符设备结构体且初始化

//led字符设备结构体
struct led_chrdev {
        struct cdev dev;
        unsigned int __iomem *va_dr;    // 数据寄存器虚拟地址保存变量
        unsigned int __iomem *va_ddr;   // 数据方向寄存器虚拟地址保存变量
        unsigned int led_pin; 			// 引脚
};

static struct led_chrdev led_cdev[DEV_CNT] = {
        {
        	.led_pin = 8				//CPIO1_B0的偏移为8+0=8
        },
};

3. container_of( )函数

  在Linux驱动编程当中我们会经常和container_of()这个函数打交道,其宏定义实现如下所示:

#define container_of(ptr, type, member) ({                      \
        const typeof( ((type *)0)->member ) *__mptr = (ptr);    \
        (type *)( (char *)__mptr - offsetof(type,member) );})
  • 参数:
    • ptr: 结构体变量中某个成员的地址
    • type: 结构体类型
    • member: 该结构体变量的具体名字
  • 返回值: 结构体type的首地址

  原理其实很简单,就是通过已知类型type的成员member的地址ptr,计算出结构体type的首地址。 type的首地址 = ptr - size ,需要注意的是它们的大小都是以字节为单位计算的,container_of( )函数的主要作用如下:

  • 判断ptr 与 member 是否为同一类型
  • 计算size大小,结构体的起始地址 = (type *)((char *)ptr - size) (注:强转为该结构体指针)

注:文件私有数据
  一般很多的linux驱动都会将文件的私有数据private_data指向设备结构体,其保存了用户自定义设备结构体的地址。 自定义结构体的地址被保存在private_data后,可以通过读、写等操作通过该私有数据去访问设备结构体中的成员, 这样做体现了linux中面向对象的程序设计思想。

4. file_operations结构体成员函数的实现

static int led_chrdev_open(struct inode *inode, struct file *filp)
{
    unsigned int val = 0;
    struct led_chrdev *led_cdev = (struct led_chrdev *)container_of(inode->i_cdev, struct led_chrdev, dev);
    filp->private_data = container_of(inode->i_cdev, struct led_chrdev, dev);

    printk("open\n");

    //读取数据方向寄存器
    val = ioread32(led_cdev->va_ddr);
    //设置数据方向寄存器为pin位可写
    val |= ((unsigned int)0x1 << (led_cdev->led_pin+16));	
    //设置数据方向寄存器为pin位输出
    val |= ((unsigned int)0X1 << (led_cdev->led_pin));
    //写入数据方向寄存器
    iowrite32(val,led_cdev->va_ddr);

    //读取数据寄存器
    val = ioread32(led_cdev->va_dr);
    //设置数据寄存器为pin位可写
    val |= ((unsigned int)0x1 << (led_cdev->led_pin+16));
   	//设置数据寄存器为pin位高电平
    val |= ((unsigned int)0x1 << (led_cdev->led_pin));
    //写入数据寄存器
    iowrite32(val, led_cdev->va_dr);

    return 0;
}

  这部分代码位open_operations结构体的设置,其中container_of()函数和寄存器设置部分需要联系前节4.2的介绍反复理解(笔者这里看了很久才顿悟)。

5. 实验效果

#蓝灯亮
sudo sh -c 'echo 0 >/dev/led_chrdev0'
#蓝灯灭
sudo sh -c 'echo 1 >/dev/led_chrdev0'

在这里插入图片描述

在这里插入图片描述
需要源码可私聊笔者

免责声明:本程序参考了野火和北京讯为科技的部分视频资料,不作商用仅供学习,若有侵权和错误请联系笔者删除

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

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

相关文章

小红书图文笔记怎么做?纯干货!

小红书图文笔记的制作是一门艺术&#xff0c;它需要结合精美的图片和有价值的内容&#xff0c;以吸引和留住用户的注意力。伯乐网络传媒给大家分享制作小红书图文笔记的干货指南&#xff0c;包括准备、制作、发布和优化的各个环节。 一、准备阶段 确定目标受众&#xff1a;找到…

kubernetes-PV与PVC、存储卷

一、PV和PVC详解 当前&#xff0c;存储的方式和种类有很多&#xff0c;并且各种存储的参数也需要非常专业的技术人员才能够了解。在Kubernetes集群中&#xff0c;放了方便我们的使用和管理&#xff0c;Kubernetes提出了PV和PVC的概念&#xff0c;这样Kubernetes集群的管理人员就…

npm镜像源管理、nvm安装多版本node异常处理

查看当前使用的镜像源 npm config get registry --locationglobal 设置使用官方源 npm config set registry https://registry.npmjs.org/ --locationglobal 设置淘宝镜像源 npm config set registry https://registry.npm.taobao.org/ --locationglobal 需要更改淘宝镜像源地址…

uniapp登录成功后跳回原有页面+无感刷新token

uniapp登录成功后跳回原有页面 引言 在C端的页面场景中&#xff0c;我们经常会有几种情况到登录页&#xff1a; 区分需要登录和不用登录的页面&#xff0c;点击需要登录才能查看的页面 已经登录但是超时&#xff0c;用户凭证失效等原因 以上情况可以细分为两种&#xff0c;一…

自动化测试实践:揭秘WebSocket在接口测试中的应用

如何写接口自动化&#xff1f;这个问题&#xff0c;但凡涉足过自动化测试的人员都能娓娓道来。Requests、urlib、jmeter、curl等等&#xff0c;不在话下。那么&#xff0c;如何获取接口的url、参数、响应等信息呢&#xff1f;&#xff01;答案就更是随口而出&#xff1a;看接口…

必看项目|多维度揭示心力衰竭患者生存关键因素(生存分析、统计检验、随机森林)

1.项目背景 心力衰竭是一种严重的公共卫生问题,影响着全球数百万人的生活质量和寿命,心力衰竭的病因复杂多样,既有个体生理因素的影响,也受到环境和社会因素的制约,个体的生活方式、饮食结构和医疗状况在很大程度上决定了其心力衰竭的风险。在现代社会,随着生活水平的提…

利用英特尔 Gaudi 2 和至强 CPU 构建经济高效的企业级 RAG 应用

检索增强生成 (Retrieval Augmented Generation&#xff0c;RAG) 可将存储在外部数据库中的新鲜领域知识纳入大语言模型以增强其文本生成能力。其提供了一种将公司数据与训练期间语言模型学到的知识分开的方式&#xff0c;有助于我们在性能、准确性及安全隐私之间进行有效折衷。…

计算机网络-Traffic-Filter流量过滤策略

一、概述 为提高网络安全性&#xff0c;管理人员需要控制进入网络的流量&#xff0c;将不信任的报文丢弃在网络边界。所谓的不信任报文是指对用户来说存在安全隐患或者不愿意接收的报文。同时保证数据访问安全性&#xff0c;企业网络中经常会要求一些部门之间不能相互访问。 背…

金融行业专题|超融合对国密卡和国产加密技术的支持能力如何?

目前&#xff0c;不少金融机构都使用国密卡&#xff08;满足国密算法要求的加密卡&#xff09;和国产密码解决方案保障金融信息安全。而在传统虚拟化架构下&#xff0c;单块加密卡通常只能服务一个系统&#xff0c;经常会出现资源利用率低、加密处理性能不足等问题&#xff0c;…

神经网络与深度学习——第14章 深度强化学习

本文讨论的内容参考自《神经网络与深度学习》https://nndl.github.io/ 第14章 深度强化学习 深度强化学习 强化学习&#xff08;Reinforcement Learning&#xff0c;RL&#xff09;&#xff0c;也叫增强学习&#xff0c;是指一类从与环境交互中不断学习的问题以及解决这类问题…

最简单的安卓模拟器抓包?

安装模拟器抓包似乎是有个绕不开的话题&#xff0c;但是现在普遍的安卓模拟器抓包会遇到以下问题&#xff1a; 1.证书配置繁琐 2.模拟器不兼容软件 3.系统设置繁琐。 前几天写过一次微信小程序如何抓包&#xff0c;现在来讲一下模拟器怎么抓包吧。首先使用的工具还是TangGo测…

开源集运wms系统

集运WMS系统是一种专为集运业务设计的仓库管理系统&#xff0c;它能够高效地处理来自多个来源的货物&#xff0c;优化存储和发货流程。 经过长时间的开发和测试&#xff0c;推出了我的集运WMS系统。它不仅具备传统WMS系统的所有功能&#xff0c;还针对集运业务的特点进行了特别…

Python轻量级的插件框架库之pluginbase使用详解

概要 在软件开发中,插件系统是一个常见的需求。插件系统允许开发者动态加载和卸载功能模块,从而提高应用程序的灵活性和可扩展性。Python的pluginbase库是一个轻量级的插件框架,旨在简化插件系统的构建过程。pluginbase库提供了一套简单易用的API,使开发者能够快速集成插件…

初步研究Pose_300W_LP datasets.py

mat文件参数解读 Color_para&#xff1a;颜色参数&#xff0c;用于描述图像的颜色属性&#xff0c;比如图像的亮度、对比度等信息。 亮度属性、对比度属性、饱和度属性&#xff08;颜色越鲜艳&#xff09;、色调属性&#xff08;色调越偏向蓝色&#xff09;、色温属性&#xf…

结构体(自定义类型)

1.结构体 结构体这种自定义的数据类型&#xff0c;让程序员可以自己创造适合的类型 结构是一些值的集合&#xff0c;这些值称为成员变量&#xff0c;结构的每个成员可以是不同类型的变量&#xff0c;可以是标量&#xff0c;数组&#xff0c;指针甚至是其他结构体 1.1.1 结构…

六西格玛培训:企业逆袭的秘密武器!——张驰咨询

为了提升企业的运营效率、产品质量和客户满意度&#xff0c;六西格玛培训成为了一个不可或缺的环节。以下是企业成功实施六西格玛培训的关键步骤&#xff1a; 一、清晰设定培训目标 首先&#xff0c;企业应明确六西格玛培训的具体目标&#xff0c;如提升产品质量、降低成本、…

武汉城投城更公司与竹云科技签署战略协议,携手构建智慧城市新未来!

2024年5月16日&#xff0c;武汉城投城更公司与深圳竹云科技股份有限公司&#xff08;以下简称“竹云”&#xff09;签订战略合作协议&#xff0c;双方将深入推进产业项目合作。 签约现场&#xff0c;双方围绕产业项目合作方向、路径和内容等进行了全面深入交流。城投城更公司党…

Windows和Linux系统部署Docker(2)

目录 一、Linux系统部署docker 前置环境&#xff1a; 1.安装需要的软件包&#xff0c; yum-util 提供yum-config-manager功能 2.添加阿里云 docker-ce 仓库 3.安装docker软件包 4.启动 docker并设置开机自启 5.查看版本&#xff1a; 二、windows系统部署docker 1.查看…

如何用unittest帮你快速生成自动化测试报告?

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 一直以来很多使用 pythonunittest 做自动化测试的的小伙伴都在想&#xff0c;unittest 这个官方库…

MFC工控项目实例之一主菜单制作

1、本项目用在WIN10下安装的vc6.0兼容版实现。创建项目名为SEAL_PRESSURE的MFC对话框。在项目res文件下添加相关256色ico格式图片。 2、项目名称&#xff1a;密封压力试验机 主菜单名称&#xff1a; 系统参数 SYS_DATA 系统测试 SYS_TEST 选择型号 TYP_CHOICE 开始试验 TES_STA…