嵌入式驱动学习第七周——GPIO子系统

2024-04-13 06:04

本文主要是介绍嵌入式驱动学习第七周——GPIO子系统,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

   GPIO子系统是用来统一便捷地访问实现输入输出中断等效果。

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

目录

  • 前言
  • GPIO子系统介绍
  • 设备树格式
  • API函数
  • pinctrl和gpio子系统示例
    • 设备树部分
    • 平台驱动框架
    • probe和remove函数实现
    • open,write等函数
  • 参考资料

GPIO子系统介绍

   之前介绍过pinctrl子系统是用来控制IO的复用功能以及一些电气属性的,那么配置完成之后,如果是GPIO的话就需要发挥功能,这些功能就包括输出、输入、触发中断

   gpio子系统顾名思义就是用于初始化GPIO并提供相应的API函数,用于设置GPIO为输入输出、读取GPIO的值。其目的是方便驱动开发者使用gpio,驱动开发者在设备树中添加gpio相关信息,然后就可以在驱动程序中使用gpio子系统提供的API函数来操作GPIO,Linux内核向驱动开发者屏蔽掉了GPIO的设置过程。

设备树格式

   查看原理图,发现LED是低电平触发,连接的GPIO是GPIO_3GPIO1_IO3。因此在GPIO子系统中,我们需要将GPIO1_IO3设置为低电平触发。

在这里插入图片描述

   在设备节点下,添加GPIO子系统,根据上面的原理图,我们设置GPIO1组的3口为低电平触发

gpioled {#address-cells = <1>;#size-cells = <1>;compatible = "atkmini-gpioled";pinctrl-names = "default";pinctrl-0 = <&pinctrl_led>;led-gpio = <&gpio1 3 GPIO_ACTIVE_LOW>;		// GPIO子系统status = "okay";
};

   其中必须指定&gpio1,其他可以不作要求。

   接下来打开imx6ul.dtsi文件,查看gpio1:

gpio1: gpio@0209c000 {compatible = "fsl,imx6ul-gpio", "fsl,imx35-gpio";reg = <0x0209c000 0x4000>;interrupts = <GIC_SPI 66 IRQ_TYPE_LEVEL_HIGH>,<GIC_SPI 67 IRQ_TYPE_LEVEL_HIGH>;gpio-controller;#gpio-cells = <2>;interrupt-controller;#interrupt-cells = <2>;
};

   上面代码中,gpio@0209c000表示gpio1的基地址,#gpio-cells 为 2,表示一共有两个 cell。第一个 cell 为 GPIO 编号,比如“&gpio1 3”就表示GPIO1_IO03。第二个 cell 表示 GPIO 极性 , 如果为0(GPIO_ACTIVE_HIGH) 的话表示高电平有效 , 如果为1(GPIO_ACTIVE_LOW)的话表示低电平有效。

API函数

/** @description  : 获取GPIO编号* @param-np     : 指定设备节点* @param-proname: GPIO属性名,与设备树中对应的属性名对应* @param-index	 : 引脚索引* @return       : 成功的话获取GPIO编号,失败的话返回负数*/
static inline int of_get_named_gpio(struct device_node *np, const char * proname, int index)
/** @description: GPIO申请函数* @param-gpio : 要申请的GPIO编号,该值是函数of_get_named_gpio的返回值* @param-label: 引脚名字,相当于为申请得到的引脚取了别名* @return     : 成功的话返回0,失败的话返回负数*/
static inline int gpio_request(unsigned gpio, const char *label)
/** @description: 释放GPIO* @param-gpio : 要释放的GPIO编号*/
static inline void gpio_free(unsigned gpio);
/** @description: GPIO输出设置函数* @param-gpio : 要设置的GPIO编号* @param-value: 输出值,1,表示高电平,0,表示低电平* @return     : 成功的话返回0,失败的话返回负数*/
static inline int gpio_direction_output(unsigned gpio , int value)
/** @description: GPIO输入设置函数* @param-gpio : 要设置的GPIO编号* @return     : 成功的话返回0,失败的话返回负数*/
static inline int gpio_direction_input(unsigned gpio)
/** @description: 获取GPIO引脚值* @param-gpio : 要设置的GPIO编号* @return     : 成功的话返回得到的引脚状态,失败的话返回负数*/
static inline int gpio_direction_input(unsigned gpio)
/** @description: 设置GPIO输出值* @param-gpio : 要设置的GPIO编号* @param-value: 设置的输出值,1为输出高电平,0为输出低电平* @return     : 成功的话返回0,失败的话返回负数*/
static inline int gpio_direction_output(unsigned gpio, int value)

pinctrl和gpio子系统示例

设备树部分

   在根目录下创建gpioled的子节点,其中compatible命名为"atkmini-gpioled"

gpioled {#address-cells = <1>;#size-cells = <1>;compatible = "atkmini-gpioled";pinctrl-names = "default";pinctrl-0 = <&pinctrl_led>;led-gpio = <&gpio1 3 GPIO_ACTIVE_LOW>;status = "okay";
};

   同时在其中使用pinctrl子系统和gpio子系统,pinctrl子系统的节点命名为pinctrl_led,具体如下所示。同时使用gpio子系统规定GPIO的状态,前面的硬件图可以看到,LED是GPIO1_IO3,并且是低电平点亮,因此设置为GPIO_ACTIVE_LOW

   接下来看pinctrl子系统中设置的节点,需要在&iomuxc中添加,因为配置复用和电器属性是iomuxc寄存器

&iomuxc {...pinctrl_led: ledgrp {fsl,pins = <MX6UL_PAD_GPIO1_IO03__GPIO1_IO03	0x10B0>;};...
}

平台驱动框架

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/module.h>
#include <linux/init.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 <linux/of_irq.h>
#include <linux/semaphore.h>
#include <linux/timer.h>
#include <linux/irq.h>
#include <linux/wait.h>
#include <linux/poll.h>
#include <linux/fs.h>
#include <linux/fcntl.h>
#include <linux/platform_device.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>#define LEDDEV_CNT      1
#define LEDDEV_NAME     "dtsplatled"
#define LEDOFF          0
#define LEDON           1struct leddev_dev {dev_t devid;struct cdev cdev;struct class *class;struct device *device;int major;int minor;struct device_node *nd;int led0;
};struct leddev_dev leddev;static const struct of_device_id led_of_match[] = {{.compatible = "atkmini-gpioled"},				// 与设备树中的节点对应{}
};static struct platform_driver led_driver = {.driver   = {.name = "imx6ul-led",.of_match_table = led_of_match,},.probe    = led_probe,.remove   = led_remove,
};// 驱动入口函数,注册platform
static int __init leddriver_init(void)
{return platform_driver_register(&led_driver);
}// 驱动出口函数,释放platform
static void __exit leddriver_exit(void)
{platform_driver_unregister(&led_driver);
}module_init(leddriver_init);
module_exit(leddriver_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("wp");

probe和remove函数实现

    probe函数中,使用of_find_node_by_path函数找到并获取gpioled在设备树中的设备节点。之后使用of_get_named_gpio函数获取GPIO号,读取成功则返回读取得到的GPIO号。同时用gpio_direction_output函数设置gpio为输出模式,默认输出高电平。

static int led_probe(struct platform_device *dev)
{printk("led driver and device was matched!\r\n");if (leddev.major) {leddev.devid = MKDEV(leddev.major, 0);register_chrdev_region(leddev.devid, LEDDEV_CNT, LEDDEV_NAME);} else {alloc_chrdev_region(&leddev.devid, 0, LEDDEV_CNT, LEDDEV_NAME);leddev.major = MAJOR(leddev.devid);leddev.minor = MINOR(leddev.minor);}cdev_init(&leddev.cdev, &led_fops);cdev_add(&leddev.cdev, leddev.devid, LEDDEV_CNT);leddev.class = class_create(THIS_MODULE, LEDDEV_NAME);if (IS_ERR(leddev.class)) {return PTR_ERR(leddev.class);}leddev.device = device_create(leddev.class, NULL, leddev.devid, NULL, LEDDEV_NAME);if (IS_ERR(leddev.device)) {return PTR_ERR(leddev.device);}leddev.nd = of_find_node_by_path("/gpioled");if (leddev.nd == NULL) {printk("gpioled node not find!\r\n");return -EINVAL;}leddev.led0 = of_get_named_gpio(leddev.nd, "led-gpio", 0);			// 获取GPIO的IO标号if (leddev.led0 < 0) {printk("can't get led-gpio\r\n");return -EINVAL;}gpio_request(leddev.led0, "led0");									// 注册IO口gpio_direction_output(leddev.led0, 1);								// 设置GPIO输出默认为1						return 0;
}static int led_remove(struct platform_device *dev)
{gpio_free(leddev.led0);										// 释放GPIO				cdev_del(&leddev.cdev);unregister_chrdev_region(leddev.devid, LEDDEV_CNT);device_destroy(leddev.class, leddev.devid);class_destroy(leddev.class);return 0;
}

open,write等函数

    首先写一个led状态翻转函数,用gpio_set_value函数设置led的输出逻辑值

void led0_switch(u8 sta)
{if (sta == LEDON)gpio_set_value(leddev.led0, 0);				// 设置gpio,第二个参数是逻辑值,0表示关,1表示开else if (sta == LEDOFF)gpio_set_value(leddev.led0, 1);
}

    然后定义open,write函数,write函数中是从用户态读取数据,然后按照当前状态翻转LED状态。

static int led_open(struct inode *inode, struct file *filp)
{filp->private_data = &leddev;return 0;
}static ssize_t led_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)
{int retvalue;unsigned char databuf[2];unsigned char ledstat;retvalue = copy_from_user(databuf, buf, cnt);if (retvalue < 0) {printk("kernel write failed!\r\n");return -EFAULT;}ledstat = databuf[0];if (ledstat == LEDON) {led0_switch(LEDON);} else if (ledstat == LEDOFF) {led0_switch(LEDOFF);}return 0;
}static struct file_operations led_fops = {.owner = THIS_MODULE,.open  = led_open,.write = led_write,
};

参考资料

[1] 【正点原子】I.MX6U嵌入式Linux驱区动开发指南 第四十五章
[2] 【Linux驱动开发】011 gpio子系统
[3] pinctrl子系统和gpio子系统-led实验

这篇关于嵌入式驱动学习第七周——GPIO子系统的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识