当前位置: 首页 > news >正文

pinctl 和 gpio子系统驱动

一.设备树中添加pinctl节点模板

1.创建对应的节点

同一个外设的 PIN 都放到一个节点里面,打开 imx6ull-14x14-evk.dts,在 iomuxc 节点
中的“imx6ul-evk”子节点下添加 “pinctrl_test” 节点。添加完成以后如下所示:

pinctrl_test:test_grp
{/* 具体的PIN信息 */
};

2.添加 “fsl,pins” 属性

设备树是通过属性来保存信息的,因此我们需要添加一个属性,属性名字一定要为“fsl,pins”,
因为对于 I.MX 系列 SOC 而言, pinctrl 驱动程序是通过读取“fsl,pins”属性值来获取 PIN 的配
置信息,完成以后如下所示:
 

pinctrl_test: testgrp 
{fsl,pins = </* 设备所使用的 PIN 配置信息 */>;
};

3.在 “fsl,pins” 属性中添加 PIN 配置信息

最后在“fsl,pins”属性中添加具体的 PIN 配置信息,完成以后如下所示:
 

pinctrl_test: testgrp 
{fsl,pins = <MX6UL_PAD_GPIO1_IO00__GPIO1_IO00 config /*config 是电器属性设置值*/>;
};

一.获取设备树中的 pin 信息

二.根据获取到的 pin 信息,来设置 pin 的复用功能

三.根据获取到的 pin 信息,设置 pin 的电器属性

二. gpio 子系统常用API函数

        对于驱动开发人员,设置好设备树以后就可以使用 gpio 子系统提供的 API 函数来操作指定
的 GPIO

1.申请 GPIO 管脚 - gpio_request

功能

        申请一个 GPIO 管脚

        成功时返回(0),失败时返回(其他值)

参数

        gpio:要申请的 gpio 标号,使用 of_get_named_gpio()从设备树获取该标号

        label:给 gpio 设置个名字

int gpio_request(unsigned gpio,const char *label);

2.释放 GPIO 管脚 - gpio_free

功能

        释放 GPIO 管脚

        无返回值

参数

        gpio:要释放的 gpio 编号

void gpio_free(unsigned gpio);

3.设置 GPIO 为输入 - gpio_direction_input

功能

        设置某个 GPIO 管脚为输入

        成功时返回(0),失败时返回(负值)

参数

        gpio:要设置的 gpio 编号

int gpio_direction_input(unsigned gpio);

4.设置 GPIO 为输出 - gpio_direction_output

功能

        设置 GPIO 为输出,并设置默认输出值

        成功时返回(0),失败时返回(负值)

参数

        gpio:要设置的 gpio 编号

        value:GPIO 默认输出值

int gpio_direction_output(unsigned gpio,int value);

5.获取 GPIO 的值 - gpio_get_value

功能

        获取某个 GPIO 的值,(0 或 1)

        成功时返回(读到的 GPIO 的值),失败时返回(负值)

参数

        gpio:要读的 gpio 编号

int gpio_get_value(unsigned gpio);

6.设置 GPIO  的值 - gpio_set_value

功能:

        设置 GPIO 的值

        无返回值

参数:

        gpio:要设置的 gpio 编号

        value:要设置的值(0 或 1)

void gpio_set_value(unsigned gpio,int value);

三.gpio子系统 OF 函数

1.获取某个属性的GPIO数量 - of_gpio_named_count

功能

        获取设备树某个属性里面定义了几个 GPIO 信息

        成功时返回(统计到的GPIO数量),失败时返回(负值)

参数

        np:设备节点

        propname:要统计的 GPIO 属性

int of_gpio_named_count(struct device_node *np,const char *propname);

2.获取某个节点的GPIO数量 - of_gpio_count

功能

        获取设备树中某节点的 GPIO 数量

        成功时返回(统计到的GPIO数量),失败时返回(负值)

参数

        np:设备节点

int of_gpio_count(struct device_node *np);

3.获取GPIO编号 - of_get_named_gpio

功能

        获取 GPIO 编号,将设备树中类似<&gpio5 7 GPIO_ACTIVE_LOW>的属性信息转换为对应的 GPIO 编号;

        成功时返回(获取到的GPIO编号),失败时返回(负值)

参数

        np:设备节点

        propname:包含要获取GPIO信息的属性名

        index:GPIO索引,一个属性里面可能包含多个GPIO,此参数指定要获取哪个GPIO的编号,如果只有一个GPIO信息的话,此参数为0

int of_get_named_gpio(struct device_node *np,const char *propname,int index);

四.在设备树下添加 gpio 节点模板

1.在"/"根节点下创建 test 设备节点

test
{};

2.添加 pinctrl 信息

        在本章的 “一.设备树中添加pinctrl节点模板” 中,创建的pinctrl_test节点,此节点描述了test设备所使用的GPIO1_IO00这个pin的信息。我们要将这节点添加到test设备节点中。

test
{/* 添加pinctrl-names 属性,此属性描述pinctrl名字为"default" */pinctrl-names = "default";/* 添加pinctrl-0节点,引用了创建的pinctrl_test节点,表示test设备所使用的PIN信息保存在pinctrl_test节点中 */pinctrl-0 = <&pinctrl_test>;/* 其他节点内容 */
};

3.添加 GPIO 属性信息

test
{/* 添加pinctrl-names 属性,此属性描述pinctrl名字为"default" */pinctrl-names = "default";/* 添加pinctrl-0节点,引用了创建的pinctrl_test节点,表示test设备所使用的PIN信息保存在pinctrl_test节点中 */pinctrl-0 = <&pinctrl_test>;/* 添加GPIO属性信息,表面test所使用的是GPIO1_IO00  低电平有效 */gpio = <&gpio1 0 GPIO_ACTIVE_LOW>;
};

五.驱动程序示例代码

1.设备树程序

①.流程图

②.代码

(1).在设备树的iomuxc下添加pinctrl节点

将GPIO1_IO03的电器属性设置为 0X10B0

(2).在根节点下创建LED设备节点

2.驱动程序

①.流程图

②.代码

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_gpio.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>#define GPIOLED_CNT     1               /* 设备号个数 */
#define GPIOLED_NAME    "gpioled"       /* 名字 */
#define LEDOFF          0               /* 关灯 */
#define LEDON           1               /* 开灯 *//* gpioled设备结构体 */
struct gpioled_dev
{dev_t devid;                /* 设备号 */struct cdev cdev;           /* cdev */struct class *class;        /* 类 */struct device *device;      /* 设备 */int major;                  /* 主设备号 */int minor;                  /* 次设备号 */struct device_node *nd;     /* 设备结点 */int led_gpio;               /* led所使用的 GPIO 编号 */
};/* led设备 */
struct gpioled_dev gpioled;/*** @description:            打开设备* @param - inode   :       传递给驱动的inode* @param - filp    :       设备文件* @return          :       0 成功,其他 失败      
*/
static int led_open(struct inode *inode,struct file *filp)
{/* 设置私有属性 */filp->private_data = &gpioled; return 0;
}/*** @description:            从设备读取数据* @param - filp    :       要打开的设备文件(文件描述符)* @param - buf     :       返回给用户空间的数据缓冲区* @param - cnt     :       要读取的数据长度* @param - offt    :       相对于文件首地址的偏移* @return          :       读取的字节数,如果为负值,则为失败
*/
static ssize_t led_read(struct file *filp,char __user *buf,size_t cnt,loff_t *offt)
{return 0;
}/*** @description:            向设备写数据* @param - filp    :       设备文件,表示打开的文件描述符* @param - buf     :       要写给设备的数据* @param - cnt     :       要写入的数据长度* @param - offt    :       写入的字节数,如果为负值,则为失败
*/
static ssize_t led_write(struct file *filp,const char __user *buf,size_t cnt,loff_t *offt)
{int retvalue;unsigned char databuf[1];unsigned char ledstat;struct gpioled_dev *dev = filp->private_data;retvalue = copy_from_user(databuf,buf,cnt);if(0 > retvalue){printk("kernel write failed!\r\n");return -EFAULT;}ledstat = databuf[0];if(ledstat == LEDON){gpio_set_value(dev->led_gpio,0);    //打开LED灯}else if(ledstat == LEDOFF){gpio_set_value(dev->led_gpio,1);    //关闭LED}return 0;
}/*** @description:        关闭/释放设备* @param - filp    :   要关闭的设备文件(文件描述符)* @return          :   0 成功,其他 失败  
*/
static int led_release(struct inode *inode,struct file *filp)
{return 0 ;
}/* 绑定设备操作函数 */
static struct file_operations gpioled_fops = 
{.owner = THIS_MODULE,.open = led_open,.read = led_read,.write = led_write,.release = led_release,
};/*** @description:        驱动入口函数* @param           :   无* @return          :   无
*/
static int __init led_init(void)
{int ret = 0;/* 设置LED所使用的GPIO *//* 1.从设备数中获取设备节点:gpioled */gpioled.nd = of_find_node_by_path("/gpioled");if(NULL == gpioled.nd){printk("gpioled node can not found!\r\n");}else{printk("gpioled node has been found!\r\n");}/* 2.获取设备数中的gpio属性,得到LED所使用的LED编号 */gpioled.led_gpio = of_get_named_gpio(gpioled.nd,"led-gpio",0);if(0 > gpioled.led_gpio){printk("can not get led-gpio");return -EINVAL;}printk("led-gpio num = %d\r\n",gpioled.led_gpio);/* 3.初始化GPIO,默认关闭LED */ret = gpio_direction_output(gpioled.led_gpio,1);if(0 > ret){printk("can not init gpio!\r\n");}/* 注册字符设备驱动 *//* 1.创建设备号 */if(gpioled.major)       //若定义了设备号{gpioled.devid = MKDEV(gpioled.major,0);register_chrdev_region(gpioled.devid,GPIOLED_CNT,GPIOLED_NAME);}else                    //没有定义设备号{alloc_chrdev_region(&gpioled.devid,0,GPIOLED_CNT,GPIOLED_NAME); //申请设备号gpioled.major = MAJOR(gpioled.devid);           //获取主设备号gpioled.minor = MINOR(gpioled.devid);           //获取次设备号}printk("gpioled major = %d,minor = %d",gpioled.major,gpioled.minor);/* 2.初始化cdev */gpioled.cdev.owner = THIS_MODULE;cdev_init(&gpioled.cdev,&gpioled_fops);/* 3.添加一个cdev */cdev_add(&gpioled.cdev,gpioled.devid,GPIOLED_CNT);/* 4.创建类 */gpioled.class = class_create(THIS_MODULE,GPIOLED_NAME);if(IS_ERR(gpioled.class)){return PTR_ERR(gpioled.class);}/* 5.创建设备 */gpioled.device = device_create(gpioled.class,NULL,gpioled.devid,NULL,GPIOLED_NAME);if(IS_ERR(gpioled.device)){return PTR_ERR(gpioled.device);}return 0;
}/*** @description:        驱动出口函数* @param       :       无* @return      :       无
*/
static void __exit led_exit(void)
{/* 注销字符设备驱动 */cdev_del(&gpioled.cdev);        //删除cdevunregister_chrdev_region(gpioled.devid,GPIOLED_CNT);        //注销device_destroy(gpioled.class,gpioled.devid);class_destroy(gpioled.class);
}module_init(led_init);
module_exit(led_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("kaneki");

Makefile:

KERNELDIR := /home/linux/IMX6ULL/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
CURRENT_PATH :=$(shell pwd)
obj-m := gpioled.o
build: kernel_modules
kernel_modules:$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) modules
clean:$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) clean

相关文章:

pinctl 和 gpio子系统驱动

一.设备树中添加pinctl节点模板 1.创建对应的节点 同一个外设的 PIN 都放到一个节点里面&#xff0c;打开 imx6ull-14x14-evk.dts&#xff0c;在 iomuxc 节点 中的“imx6ul-evk”子节点下添加 “pinctrl_test” 节点。添加完成以后如下所示&#xff1a; pinctrl_test:test_g…...

RocketMQ消息堆积了怎么解决?

RocketMQ 的消息堆积&#xff0c;一般都是因为客户端本地消费过程中&#xff0c;由于消费耗时过长或消费并发度较小等原因&#xff0c;导致客户端消费能力不足&#xff0c;出现消息堆积的问题。 当线上出现消息堆积的问题时&#xff0c;一般有以下几种方式来解决: 增加消费者…...

C++第十二弹 -- STL之list模拟实现

文章索引 前言模拟实现list1. ListNode节点类2. list的迭代器封装3. 反向迭代器4. list类的模拟实现测试代码 list的反向迭代器总结 前言 通过模拟实现可以让我们更加深刻的理解C底层STL的实现逻辑, 本篇就对list的底层进行模拟实现. 博客主页: 酷酷学!!! 点击关注 共同进步!…...

Destiny of Gods首轮测试正式开启,参与玩家数量突破10万

天神风云&#xff0c;波澜再兴&#xff0c;GameFi链游聚合平台Destiny of Gods首款同名数字卡牌回合制游戏首轮测试定档8月20日20:00&#xff08;GMT8&#xff09;&#xff0c;现已正式开启&#xff01; 这是一个由人、游灵和神灵共存的世界&#xff0c;历经蛮荒时期的纷争与信…...

QT聊天室基于Tcp

server.cpp #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget),server(new QTcpServer(this)) // 给服务器指针对象实例化空间{ui->setupUi(this); }Widget::~Widget() {delete ui; }…...

公开课观后感:密歇根大学python for everyone

从2024年1月17日到2024年8月20日&#xff0c;终于将密歇根大学的python for everyone的python公开课跟完。站在一月份规划的时刻来看&#xff0c;比我想象中花费的时间更多&#xff0c;我当时肯定没有想到要花上整整七个月的时间才能将这个公开课的内容看完&#xff0c;毕竟这个…...

goweb框架-gin

文章目录 Gin框架概览Gin框架的特点Gin框架的安装和基本使用安装基本使用 路由系统路由的基本概念Gin框架路由的特点 Radix Tree&#xff08;基数树&#xff09;基数树的定义和原理基数树在Gin框架中的应用节省空间的优化动态路由和通配符处理 路由树的构建注册路由的过程路由树…...

2024年接口测试高频面试题及答案

1. 什么是接口测试&#xff1f; •接口测试就是通过测试不同情况下的入参与之相应的出参信息来判断接口是否符合或满足相应的功能性、安全性要求 •测试的重点是要检查数据的交换&#xff0c;传递和控制管理过程&#xff0c;以及系统间的相互逻辑依赖关系 2. 为什么要做接口…...

ESP32-C3在MQTT访问时出现“transport_base: Poll timeout or error”问题的分析(8)

接前一篇文章:ESP32-C3在MQTT访问时出现“transport_base: Poll timeout or error”问题的分析(7) 前边几回分析了笔者在MQTT测试时所遇到的问题: 最终定位到了是由于components\components\tcp_transport\transport_ssl.c的base_poll_write函数中调用的select函数超时返回…...

Linux: 忘记密码的解决方法,passwd

https://www.redhat.com/sysadmin/recover-root-passwd 这里的方法很简单&#xff0c;就是通过console进去&#xff0c;添加一个启动参数&#xff0c;加载sysroot&#xff0c;然后用passwd命令修改密码。这个是RHEL7适用。 https://access.redhat.com/solutions/1192 这个是提…...

36. 有效的数独【 力扣(LeetCode) 】

一、题目描述 请你判断一个 9 x 9 的数独是否有效。只需要 根据以下规则 &#xff0c;验证已经填入的数字是否有效即可。 数字 1-9 在每一行只能出现一次。 数字 1-9 在每一列只能出现一次。 数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。&#xff08;请参考示例图…...

机器学习中的没有免费午餐定理

嘿&#xff0c;各位机器学习的爱好者们&#xff01;今天&#xff0c;让我们一起深入探讨机器学习中那个神秘而又重要的概念——没有免费午餐定理。 一、定理引入&#xff1a;探索算法森林的钥匙 在广阔无垠的机器学习领域中&#xff0c;免费午餐定理就如同一把神奇的钥匙&…...

高级java每日一道面试题-2024年8月21日-框架篇[Spring篇]-使用IOC容器应该注意哪些?

如果有遗漏,评论区告诉我进行补充 面试官: 使用IOC容器应该注意哪些? 我回答: 1. 理解IOC的基本概念 控制反转&#xff1a;在传统的编程模式中&#xff0c;程序会主动控制依赖关系的创建和管理。而在IoC容器中&#xff0c;这种控制权被反转给了容器本身。程序员只需要声明…...

LLM训练推理相关概念

1. 有监督微调&#xff08;Supervised Fine-Tuning&#xff09;与指令微调&#xff08;Instruction Fine-Tuning&#xff09;对模型参数的影响 **有监督微调&#xff08;Supervised Fine-Tuning, SFT&#xff09;和指令微调&#xff08;Instruction Fine-Tuning, Instruct-Tun…...

IP in IP 协议

IP in IP 是一种多重IP协议&#xff0c;即&#xff1a;客户机可以发送一个IP协议内部在嵌套一个IP协议到某个特定的主机上&#xff0c;在由具体的主机作为路由进行转发的协议。 例如&#xff1a; IP in IP帧协议结构为&#xff0c;第一层为发送到IP in IP 路由主机的报文&…...

DAY2: HTTP请求报文和响应报文是怎样的,有哪些常见的字段?| HTTP有哪些请求方式?| GET请求和POST请求的区别

目录 HTTP请求报文和响应报文是怎样的&#xff0c;有哪些常见的字段&#xff1f; 请求报文 响应报文 HTTP有哪些请求方式&#xff1f; GET请求和POST请求的区别 HTTP请求报文和响应报文是怎样的&#xff0c;有哪些常见的字段&#xff1f; HTTP报文分为请求报文和响应报文…...

线性代数:每日一题1/特征值与相似对角化

设A, B 为二阶矩阵&#xff0c;且 AB BA , 则“A有两个不相等的特征值”是“B可对角化"的&#xff08;&#xff09; A. 充分必要条件 B. 充分不必要条件 C.必要不充分条件 D.既不充分也不必要条件 知识点&#xff1a; 特征向量与特征值的关系 相似矩阵的定义和性质 n阶…...

Android UI:PopupWindow:API

文章目录 类操作 对PopupWindow的操作 创建PopupWindow对象的操作添加并显示PopupWindow的操作移除PopupWindow的操作更新PopupWindow的操作显示内容的相关操作 布局的相关操作进入退出动画的相关操作 Transition设置进入动画的相关操作Transition设置退出动画的相关操作XML设置…...

什么是DevUI?

DevUI是面向企业中后台产品的开源前端解决方案&#xff0c;其设计价值观基于"高效、开放、可信、乐趣"四种自然与人文相结合的理念&#xff0c;旨在为设计师、前端开发者提供标准的设计体系&#xff0c;并满足各类落地场景&#xff0c;是一款企业级开箱即用的产品。 …...

DAY53

作业&#xff1a; 运行1个服务器和2个客户端 实现效果&#xff1a; 服务器和2个客户端互相聊天&#xff0c;服务器和客户端都需要使用select模型去实现 服务器要监视2个客户端是否连接&#xff0c;2个客户端是否发来消息以及服务器自己的标准输入流 客户端要监视服务器是否发来…...

Java 8 Stream API 入门到实践详解

一、告别 for 循环&#xff01; 传统痛点&#xff1a; Java 8 之前&#xff0c;集合操作离不开冗长的 for 循环和匿名类。例如&#xff0c;过滤列表中的偶数&#xff1a; List<Integer> list Arrays.asList(1, 2, 3, 4, 5); List<Integer> evens new ArrayList…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)

概述 在 Swift 开发语言中&#xff0c;各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过&#xff0c;在涉及到多个子类派生于基类进行多态模拟的场景下&#xff0c;…...

【2025年】解决Burpsuite抓不到https包的问题

环境&#xff1a;windows11 burpsuite:2025.5 在抓取https网站时&#xff0c;burpsuite抓取不到https数据包&#xff0c;只显示&#xff1a; 解决该问题只需如下三个步骤&#xff1a; 1、浏览器中访问 http://burp 2、下载 CA certificate 证书 3、在设置--隐私与安全--…...

汇编常见指令

汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX&#xff08;不访问内存&#xff09;XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...

Spring数据访问模块设计

前面我们已经完成了IoC和web模块的设计&#xff0c;聪明的码友立马就知道了&#xff0c;该到数据访问模块了&#xff0c;要不就这俩玩个6啊&#xff0c;查库势在必行&#xff0c;至此&#xff0c;它来了。 一、核心设计理念 1、痛点在哪 应用离不开数据&#xff08;数据库、No…...

Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习)

Aspose.PDF 限制绕过方案&#xff1a;Java 字节码技术实战分享&#xff08;仅供学习&#xff09; 一、Aspose.PDF 简介二、说明&#xff08;⚠️仅供学习与研究使用&#xff09;三、技术流程总览四、准备工作1. 下载 Jar 包2. Maven 项目依赖配置 五、字节码修改实现代码&#…...

LRU 缓存机制详解与实现(Java版) + 力扣解决

&#x1f4cc; LRU 缓存机制详解与实现&#xff08;Java版&#xff09; 一、&#x1f4d6; 问题背景 在日常开发中&#xff0c;我们经常会使用 缓存&#xff08;Cache&#xff09; 来提升性能。但由于内存有限&#xff0c;缓存不可能无限增长&#xff0c;于是需要策略决定&am…...

【SpringBoot自动化部署】

SpringBoot自动化部署方法 使用Jenkins进行持续集成与部署 Jenkins是最常用的自动化部署工具之一&#xff0c;能够实现代码拉取、构建、测试和部署的全流程自动化。 配置Jenkins任务时&#xff0c;需要添加Git仓库地址和凭证&#xff0c;设置构建触发器&#xff08;如GitHub…...

Unity中的transform.up

2025年6月8日&#xff0c;周日下午 在Unity中&#xff0c;transform.up是Transform组件的一个属性&#xff0c;表示游戏对象在世界空间中的“上”方向&#xff08;Y轴正方向&#xff09;&#xff0c;且会随对象旋转动态变化。以下是关键点解析&#xff1a; 基本定义 transfor…...

[拓扑优化] 1.概述

常见的拓扑优化方法有&#xff1a;均匀化法、变密度法、渐进结构优化法、水平集法、移动可变形组件法等。 常见的数值计算方法有&#xff1a;有限元法、有限差分法、边界元法、离散元法、无网格法、扩展有限元法、等几何分析等。 将上述数值计算方法与拓扑优化方法结合&#…...