pinctl 和 gpio子系统驱动

news2025/1/15 19:40:59

一.设备树中添加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);        //删除cdev
    unregister_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

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

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

相关文章

8.21Qt作业

运用网络通信&#xff0c;实现简单聊天室 客户端主要代码 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget), socket(new QTcpSocket(this)) {ui->setupUi(this);//初始化界面ui-&g…

Postman之Newman命令以及常用参数

Newman介绍 Postman是专为接口测试而生&#xff0c;而Newman是专为Postman而生。因为服务器一般都是Linux系统&#xff0c;而前文提到的操作都离不开Postman的客户端&#xff0c;为解决这个问题&#xff0c;谷歌公司引入了 Newman工具。Newman是Postman的命令行&#xff0c;是…

STM32——I2C通信外设

软件只需要CR控制寄存器&#xff0c;DR数据寄存器&#xff0c;为实时监控状态&#xff0c;软件需要读取SR状态寄存器&#xff0c;好比&#xff1a;开车时CR是控制汽车方向&#xff0c;踩油门等&#xff0c;SR是仪表盘。 由于I2C是半双工&#xff0c;因此发送和接收数据都是在移…

又一家光伏企业终止,行业产能过剩竞争激烈,毛利率极低

一道新能终止原因如下&#xff1a;首先&#xff0c;一道新能从事光伏单晶电池和单晶组件业务&#xff0c;该行业竞争激烈&#xff0c;目前已经出现了产能过剩&#xff0c;甚至负毛利率的情况&#xff1b;其次&#xff0c;报告期内&#xff0c;一道新能营收高达227亿&#xff0c…

DC-DC 转换器中的压电谐振器:当前状态和限制

在小体积和高频下提高功率密度并减小电感器和变压器的尺寸是DC-DC转换器设计中的一大挑战。为了克服这些困难&#xff0c;压电谐振器&#xff08;PR&#xff09;通过利用潜在的压电效应&#xff0c;以振动模式而不是电模式存储能量。 即使 PR 的使用在效率和功率密度方面改进了…

关于开源许可协议

开源许可协议 引用开源代码之后是否可以保留知识产权&#xff0c;针对不同的开源协议要进行开源扫描。基于BSD、MIT和Apache三种是可以闭源。但是基于GPL、LGPL和Mozilla的开源方案必须同步开源。

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

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

【大模型从入门到精通32】开源库框架LangChain RAG 系统中的问答技术2

这里写目录标题 探索高级问答链类型MapReduce 和 Refine 技术 实用建议和最佳实践解决 RetrievalQA 限制结论进一步阅读和探索理论问题实践问题 探索高级问答链类型 MapReduce 和 Refine 技术 MapReduce 和 Refine 是设计用来规避由语言模型 (LM) 上下文窗口大小所导致的限制…

Pulsar官方文档学习笔记——消息机制

pulsar 基于3.x最新官方文档学习记录 概念与架构 典型的推送订阅模式。生产者发送消息&#xff0c;消费者订阅topic消费信息并回应ACK。订阅创建后&#xff0c;Pulsar会保留所有消息。仅消息被所有订阅 成功消费了才会丢弃&#xff08;可以配置消息保留机制保留一定量&#…

裸金属服务器和裸金属云服务器:区别、优势与选择

首先&#xff0c;必须肯定的是&#xff1a;裸金属服务器和裸金属云服务器是有区别的。 ‌ 二者的概述 裸金属服务器&#xff08;‌Bare Metal Server&#xff09;‌是一种物理服务器&#xff0c;‌它直接在硬件上运行&#xff0c;‌没有额外的虚拟化层。‌这意味着每个应用程…

封装通用第三方平台用户表(微信开放平台)

文章目录 一. 注册微信开放平台1.1 开发者资质认证1.2 应用申请1.3 配置应用 二.通用数据库表设计三.入库实体类四. 对接第三方平台4.1 微信开放平台VO对象4.2 通用方法 我们的系统可能要对接很多第三方系统&#xff0c;为了便利用户授权使用和对多平台账户的管理。有必要设计通…

2024 江苏省第二届数据安全技术应用职业技能竞赛 初赛 部分wp

文章目录 一、前言二、参考文章三、题目&#xff08;解析&#xff09;数据安全解题赛1、ds_0602&#xff08;30分&#xff09;2、333.file&#xff08;45分&#xff09;3、pf文件分析&#xff08;35分&#xff09;4、丢失的资料&#xff08;45分&#xff09;5、greatphp&#x…

基于SpringBoot的来访管理系统的设计与实现-

TOC springboot600基于SpringBoot的来访管理系统的设计与实现---论文 绪 论 1.1项目研究的背景 随着科学技术发展&#xff0c;计算机已成为人们生活中必不可少的生活办公工具&#xff0c;在这样的背景下&#xff0c;网络技术被应用到各个方面&#xff0c;为了提高办公生活效…

Android类加载机制简介

一、前言 随着 Android 技术的不断发展&#xff0c;对其内部机制的探索也日益深入。类加载机制作为 Android 运行时环境的核心组成部分之一&#xff0c;影响着应用的性能、安全性以及可扩展性。通过对 Android 类加载机制的研究&#xff0c;开发者可以更好地优化代码结构、提高…

requests请求库入门使用

requests 库是一个功能强大且易于使用的 HTTP 请求库&#xff0c;适用于各种网络请求任务。它简化了 HTTP 请求的发送过程&#xff0c;并提供了丰富的功能来处理各种网络请求和响应。 1.安装 首先&#xff0c;你需要安装 requests 库。如果你还没有安装&#xff0c;可以使用 …

网优学习干货:2.6G仿真操作(1)

2.6G工程建立——整体仿真过程 针对覆盖仿真、速率仿真及蒙特卡洛仿真的说明 覆盖仿真&#xff1a;覆盖仿真主要用于评估网络覆盖性能。基于MassiveMIMO天线、射线传模或经验传模进行覆盖预测&#xff0c;计算链路损耗后&#xff0c;基于栅格分析估算小区覆盖预测的各项指标&am…

前端不同项目使用不同的node版本(Volta管理切换)

前端不同项目使用不同的node版本(Volta管理切换) 使用volta自动切换前端项目的node版本&#xff0c; 每个不同的前端项目&#xff0c;可以使用不同的node版本。Volta这个工具&#xff0c;它允许用户方便地安装、切换和管理不同版本的Node.js&#xff0c;避免了为每个项目手动配…

QT6聊天室项目 核心类与主窗口设计逻辑分析

核心类 核心类设计逻辑 数据结构设计&#xff08;data.h&#xff09; 用户信息 用户ID用户网名用户个人签名用户手机号码用户头像聊天会话信息 会话编号会话名称&#xff08;单聊则是对方网名&#xff0c;群聊则是群名&#xff09;最新消息会话图标&#xff08;单聊对方头像&a…

sortable中el-table拖拽及点击箭头上下移动row

效果 安装 npm install sortablejs --save 引入 import Sortable from "sortablejs"; <el-table:data"tableBody"borderref"tableRef":stripe"true":key"tableKey"><el-table-column type"index" la…

driver.find_element 找不到元素的大坑

前端使用element框架。 登录进去使用开发人员工具能看到元素&#xff0c;复制xpath使用find_element死活找不到。 其中一次复制的xpath 注意红色部分: #先点击一下输入框 driver.find_element(By.XPATH,/html/body/div[1]/section/section/section/main/div/div[1]/div/form/…