0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

使用pinctrl和gpio子系统实现LED灯驱动

玩转单片机 来源:玩转单片机 2023-04-03 10:17 次阅读

前边已经学了两种点灯,本质依然还是通过配置寄存器;在学习STM32的时候除了学习配置一下寄存器,基本都是使用库来开发,那么在i.MX6ULL还使用寄存器开发明显是不太适合,那么i.MX6ULL有更方便的开发呢,这篇就来学习一下使用 pinctrl 和 gpio 子系统来完成 LED 灯驱动。

|修改设备树文件

添加 pinctrl 节点

开发板上的 LED 灯使用了 GPIO1_IO04这个 PIN,打开 imx6ull-14x14-evk.dts,在 iomuxc 节点的 imx6ul-evk 子节点下创建一个名为“pinctrl_led”的子节点,节点内容如下所示:

/* 添加的 */
pinctrl_led: ledgrp { 
  fsl,pins = < 
    MX6UL_PAD_GPIO1_IO04__GPIO1_IO04 0x10B0 /* LED0 */
  >;
};
MX6UL_PAD_GPIO1_IO04__GPIO1_IO04 表示将该io复用为GPIO。

0x10b0 表示对PAD寄存器的配置值,具体含义为如下:
/*寄存器SW_PAD_SNVS_TAMPER3设置IO属性
*bit 16:0 HYS关闭
*bit [15:14]: 00 默认下拉
*bit [13]: 0 kepper功能
*bit [12]: 1 pull/keeper使能
*bit [11]: 0 关闭开路输出
*bit [7:6]: 10 速度100Mhz
*bit [5:3]: 110 R0/6驱动能力
*bit [0]: 0 低转换率
*/

图示:

9d077a5c-d006-11ed-bfe3-dac502259ad0.png

添加 LED 设备节点

在根节点“/”下创建 LED 灯节点,节点名为“gpioled”,节点内容如下:

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

pinctrl-0 属性设置 LED 灯所使用的 PIN 对应的 pinctrl 节点。

led-gpio 属性指定了 LED 灯所使用的 GPIO,在这里就是 GPIO1 的 IO04,低电平有效。稍后编写驱动程序的时候会获取 led-gpio 属性的内容来得到 GPIO 编号,因为 gpio 子系统的 API 操作函数需要 GPIO 编号。

图示:

9d615fcc-d006-11ed-bfe3-dac502259ad0.png

检查 PIN 是否被其他外设使用

这一点非常重要!!!

很多初次接触设备树的驱动开发人员很容易因为这个小问题栽了大跟头!因为所使用的设备树基本都是在半导体厂商提供的设备树文件基础上修改而来的,而半导体厂商提供的设备树是根据自己官方开发板编写的,很多 PIN 的配置和实际所使用的开发板不一样。

比如 A 这个引脚在官方开发板接的是 I2CSDA,而实际所使用的硬件可能将 A 这个引脚接到了其他的外设,比如 LED 灯上,接不同的外设,A 这个引脚的配置就不同。一个引脚一次只能实现一个功能,如果 A 引脚在设备树中配置为了 I2C 的 SDA 信号,那么 A 引脚就不能再配置为 GPIO,否则的话驱动程序在申请 GPIO 的时候就会失败。检查 PIN 有没有被其他外设使用包括两个方面:

①、检查 pinctrl 设置。

②、如果这个 PIN 配置为 GPIO 的话,检查这个 GPIO 有没有被别的外设使用。

因为本章实验将 GPIO1_IO04这个 PIN 配置为了 GPIO,所以还需要查找一下有没有其他的外设使用了 GPIO1_IO04,在 可能使用到的设备树中搜索“gpio1 4”,看看是否被其他外设使用到:

9db89d5a-d006-11ed-bfe3-dac502259ad0.png

编译设备树和复制文件

编译没有问题:

9e031ab0-d006-11ed-bfe3-dac502259ad0.png

复制文件:

9e4144de-d006-11ed-bfe3-dac502259ad0.png

|编译驱动程序

复制一份新字符驱动,对应改下名称:

9e8b8206-d006-11ed-bfe3-dac502259ad0.png

简单提前了解:使用pinctrl 和 gpio 子系统来完成 LED 灯驱动最明显的变化就是不同操作寄存器,也就不用对物理地址映射成虚拟地址。完整的代码如下:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 


/* 添加头文件 */
#include 
#include 
#include 
#include 


#define CHRDEVBASE_CNT      1    /* 设备号个数 */
#define CHRDEVBASE_NAME   "chrdevbase"  /* 名字 */


#define LEDOFF 0 /* 关灯 */
#define LEDON 1 /* 开灯 */


/* chrdevbase 设备结构体 */
struct newchr_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 编号 */
};


struct newchr_dev chrdevbase;/* 自定义字符设备 */


/*
* @description : LED 硬件初始化
* @param : 无
* @return : 无
*/
static int led_hal_init(void)
{
  int ret = 0;


  /* 设置 LED 所使用的 GPIO */
  /* 1、获取设备节点:gpioled */
  chrdevbase.nd = of_find_node_by_path("/gpioled");
  if(chrdevbase.nd == NULL) {
    printk("chrdevbase node cant not found!
");
    return -EINVAL;
  } else {
    printk("chrdevbase node has been found!
");
  }


  /* 2、 获取设备树中的 gpio 属性,得到 LED 所使用的 LED 编号 */
  chrdevbase.led_gpio = of_get_named_gpio(chrdevbase.nd, "led-gpio", 0);
  if(chrdevbase.led_gpio < 0) {
    printk("can't get led-gpio");
    return -EINVAL;
  }
  printk("led-gpio num = %d
", chrdevbase.led_gpio);


  /* 3、设置 GPIO1_IO03 为输出,并且输出高电平,默认关闭 LED 灯 */
  ret = gpio_direction_output(chrdevbase.led_gpio, 1);
  if(ret < 0) {
    printk("can't set gpio!
");
  }


  return 0;
}


/*
 * @description    : 打开设备
 * @param - inode   : 传递给驱动的inode
 * @param - filp   : 设备文件,file结构体有个叫做private_data的成员变量
 *             一般在open的时候将private_data指向设备结构体。
 * @return       : 0 成功;其他 失败
 */
static int chrdevbase_open(struct inode *inode, struct file *filp)
{
  printk("[BSP]chrdevbase open!
");
  filp->private_data = &chrdevbase; /* 设置私有数据 */
  return 0;
}


/*
 * @description    : 从设备读取数据 
 * @param - filp   : 要打开的设备文件(文件描述符)
 * @param - buf   : 返回给用户空间的数据缓冲区
 * @param - cnt   : 要读取的数据长度
 * @param - offt   : 相对于文件首地址的偏移
 * @return       : 读取的字节数,如果为负值,表示读取失败
 */
static ssize_t chrdevbase_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)
{
  printk("chrdevbase read!
");
  return 0;
}


/*
 * @description    : 向设备写数据 
 * @param - filp   : 设备文件,表示打开的文件描述符
 * @param - buf   : 要写给设备写入的数据
 * @param - cnt   : 要写入的数据长度
 * @param - offt   : 相对于文件首地址的偏移
 * @return       : 写入的字节数,如果为负值,表示写入失败
 */
static ssize_t chrdevbase_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)
{
  int retvalue = 0;
  char writebuf[1];
  struct newchr_dev *dev = filp->private_data;


  /* 接收用户空间传递给内核的数据并且打印出来 */
  retvalue = copy_from_user(writebuf, buf, cnt);
  printk("[BSP]kernel recevdata data:%d!
",writebuf[0]);


  if(writebuf[0] == LEDON) { 
    gpio_set_value(dev->led_gpio, 0); /* 打开 LED 灯 */
  } else if(writebuf[0] == LEDOFF) {
    gpio_set_value(dev->led_gpio, 1); /* 关闭 LED 灯 */
  }


  // printk("chrdevbase write!
");
  return 0;
}


/*
 * @description    : 关闭/释放设备
 * @param - filp   : 要关闭的设备文件(文件描述符)
 * @return       : 0 成功;其他 失败
 */
static int chrdevbase_release(struct inode *inode, struct file *filp)
{
  printk("[BSP]release!
");
  return 0;
}


/*
 * 设备操作函数结构体
 */
static struct file_operations chrdevbase_fops = {
  .owner = THIS_MODULE,  
  .open = chrdevbase_open,
  .read = chrdevbase_read,
  .write = chrdevbase_write,
  .release = chrdevbase_release,
};


/*
 * @description  : 驱动入口函数 
 * @param     : 无
 * @return     : 0 成功;其他 失败
 */
static int __init chrdevbase_init(void)
{
  /* 初始化硬件 */
  led_hal_init();


  /* 注册字符设备驱动 */
  /* 1、创建设备号 */
  if (chrdevbase.major) { /* 定义了设备号 */
    chrdevbase.devid = MKDEV(chrdevbase.major, 0);
    register_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT, CHRDEVBASE_NAME);
  } else { /* 没有定义设备号 */
    alloc_chrdev_region(&chrdevbase.devid, 0, CHRDEVBASE_CNT,CHRDEVBASE_NAME); /* 申请设备号 */
    chrdevbase.major = MAJOR(chrdevbase.devid); /* 获取主设备号 */
    chrdevbase.minor = MINOR(chrdevbase.devid); /* 获取次设备号 */
  }
  printk("newcheled major=%d,minor=%d
",chrdevbase.major,chrdevbase.minor);


  /* 2、初始化 cdev */
  chrdevbase.cdev.owner = THIS_MODULE;
  cdev_init(&chrdevbase.cdev, &chrdevbase_fops);


  /* 3、添加一个 cdev */
  cdev_add(&chrdevbase.cdev, chrdevbase.devid, CHRDEVBASE_CNT);


  /* 4、创建类 */
  chrdevbase.class = class_create(THIS_MODULE, CHRDEVBASE_NAME);
  if (IS_ERR(chrdevbase.class)) {
    return PTR_ERR(chrdevbase.class);
  }


  /* 5、创建设备 */
  chrdevbase.device = device_create(chrdevbase.class, NULL,chrdevbase.devid, NULL, CHRDEVBASE_NAME);
  if (IS_ERR(chrdevbase.device)) {
    return PTR_ERR(chrdevbase.device);
  }


  return 0;
}


/*
 * @description  : 驱动出口函数
 * @param     : 无
 * @return     : 无
 */
static void __exit chrdevbase_exit(void)
{
  /* 注销字符设备 */
  cdev_del(&chrdevbase.cdev);/* 删除 cdev */
  unregister_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT);/* 注销设备号 */


  device_destroy(chrdevbase.class, chrdevbase.devid);/* 销毁设备 */
  class_destroy(chrdevbase.class);/* 销毁类 */


  printk("[BSP]chrdevbase exit!
");
}


/* 
 * 将上面两个函数指定为驱动的入口和出口函数 
 */
module_init(chrdevbase_init);
module_exit(chrdevbase_exit);


/* 
 * LICENSE和作者信息
 */
MODULE_LICENSE("GPL");
MODULE_AUTHOR("zuozhongkai");

编译驱动:

9ebaa90a-d006-11ed-bfe3-dac502259ad0.png

复制驱动到对应位置:

9f0032ea-d006-11ed-bfe3-dac502259ad0.png

| 观察效果

1、观察设备树节点

9f364ab0-d006-11ed-bfe3-dac502259ad0.png

2、加载驱动

depmod //第一次加载驱动的时候需要运行此命令
modprobe gpioled.ko //加载驱动

9f7d83e4-d006-11ed-bfe3-dac502259ad0.png

3、操作GPIO

9fab6228-d006-11ed-bfe3-dac502259ad0.png

使用pinctrl 和 gpio 子系统可以很方便对GPIO进行操作,可以不去查寄存器的地址也能实现,提高了程序员对底层驱动开发的效率。

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • led
    led
    +关注

    关注

    237

    文章

    22450

    浏览量

    645890
  • 寄存器
    +关注

    关注

    30

    文章

    5032

    浏览量

    117741
  • STM32
    +关注

    关注

    2240

    文章

    10674

    浏览量

    348788
  • 子系统
    +关注

    关注

    0

    文章

    100

    浏览量

    12265
  • GPIO
    +关注

    关注

    16

    文章

    1135

    浏览量

    50578

原文标题:i.MX6ULL|pinctrl 和 gpio 子系统点灯

文章出处:【微信号:玩转单片机,微信公众号:玩转单片机】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    一文搞懂Linux pinctrl/gpio子系统

    GPIO的寄存器操作。分享给刚刚接触外设bsp的小伙伴们。当然后面有时间还会分享GPIO子系统框架和pinctrl子系统框架,先知道黑盒怎么
    发表于 06-09 09:52 1529次阅读

    「正点原子Linux连载」第四十五章 pinctrlgpio子系统实验(一)

    GPIO驱动基本都是必须的,而pinctrlgpio子系统又是GPIO
    发表于 03-19 14:58

    「正点原子Linux连载」第四十五章 pinctrlgpio子系统实验(二)

    =5};第4行,test设备所使用的gpio。关于pinctrl子系统gpio子系统就讲解到这里,接下来就使用
    发表于 03-19 14:59

    【正点原子FPGA连载】第二十六章gpio子系统简介-领航者ZYNQ之linux开发指南

    原子公众号,获取最新资料第二十六章gpio子系统简介上一章我们编写了基于设备树的LED驱动,但是驱动的本质还是没变,都是配置
    发表于 09-16 17:37

    基于GPIO子系统LED驱动程序分享

    Pinctrl 子系统把引脚的复用、配置抽出来,做成 Pinctrl 子系统,给 GPIO、I2C 等模块使用。让我们在使用某个引脚功能时不
    发表于 12-16 07:16

    怎样去使用linux下的pintcrl和gpio子系统

    pinctrlgpio内部的原理是如何实现的?怎样去使用linux下的pintcrl和gpio子系统呢?
    发表于 03-07 13:38

    RK3399开发板的pinctrlgpio子系统相关资料介绍

    驱动工程师只做驱动,应用工程师专注做应用。  linux下的pintcrl和gpio子系统就类似于ST的“BSP库”,但是linux的pinctr
    发表于 09-16 17:27

    RK3288是如何使用pinctrl子系统去控制LED编写的

    子系统的接口。其次在platform_driver里,需要配置设备树的结构,并且设置compatible和设备树中的myled一致,“myled,led_drv”。然后在platform的probe中
    发表于 09-16 17:29

    更新 | 持续开源 迅为RK3568驱动指南第十一篇-pinctrl子系统

    《iTOP-RK3568开发板驱动开发指南》更新,本次更新内容对应的是驱动(第十一期_pinctrl子系统-全新升级)视频,后续资料会不断更新,不断完善,帮助用户快速入门,大大提升研发
    发表于 10-18 11:12

    gpiopinctrl子系统的关系与区别

    gpiopinctrl 子系统在内核里的使用率非常高,和嵌入式产品的关联非常大。从这两个子系统开始学习驱动开发是个不错的入门选择。
    的头像 发表于 03-15 11:40 3795次阅读

    嵌入式驱动开发两大子系统的使用

    本文的关注点是 gpio driver --> gpio subsystem core -> gpio consumer 这一路径,读者如果想更深入地了解 pinctrl
    的头像 发表于 03-15 13:41 1553次阅读

    【i.MX6ULL】驱动开发6——GPIO子系统点亮LED

    本篇介绍了使用**Pinctrl子系统GPIO子系统**的方式来点亮LED,与之前的寄存器版点亮LED
    的头像 发表于 05-21 21:50 2852次阅读
    【i.MX6ULL】<b class='flag-5'>驱动</b>开发6——<b class='flag-5'>GPIO</b><b class='flag-5'>子系统</b>点亮<b class='flag-5'>LED</b>

    RK3568pinctrlgpio 子系统详解

    如果 pinctrl 子系统将 PIN 复用为 GPIO,那么接下来就要配置 gpio 子系统,且 gp
    的头像 发表于 12-20 10:22 2104次阅读
    RK3568<b class='flag-5'>pinctrl</b> 和 <b class='flag-5'>gpio</b> <b class='flag-5'>子系统</b>详解

    Linux中pinctrl操作GPIO只需要几步

    pinctrl 子系统 API pinctrl 子系统的 API 有很多,对于驱动工程师来说,pinct
    的头像 发表于 09-27 17:24 1225次阅读

    瑞芯微RK3568-iomuxc和pinctrl子系统初窥

    pinctrl子系统作用:从设备树中获取PIN的描述信息来设置PIN的复用和电气属性,PIN可复用为I2C、SPI、GPIOgpio子系统
    发表于 12-20 10:10 54次下载