设备树之GPIO和中断实例

概述

设备树不仅仅描述常规硬件信息,还可以描述中断,GPIO,DMA,PINCTRL,时钟,电源管理等内核基础设施信息及其使用情况,下面重点介绍中断,GPIO等结点属性

中断控制器

对于中断控制器结点,它提供如下属性:

interrupt-controller 表明自己的身份为中断控制器

interrupt-cells

该属性与#address-cells属性相似,它声明了设备结点interrupt cells的个数。对于ARM GIC中断控制器而言#interrupt-cells = <3>

具体含义如下:

第一个cell表示中断类型,0代表SPI interrupts,1代表PPI interrupts

第二个cell表示中断号码,SPI interrupts的中断号范围是0 ~ 987,而PPI interrupts的中断号范围是0 ~ 15

第三个cell表示中断触发方式,它有4个值,分别为1,2,4,8

1表示上升沿触发

2表示下降沿触发

4表示高电平触发

8表示低电平触发

                 图1

上图1为典型的中断控制器结点

从上图1可知:

该中断控制器包含4组寄存器,设备结点interrupt cells的个数为3

第一个cell表示中断类型,第二个cell表示中断号码,第三个cell表示中断触发方式

设备结点与中断相关的属性

interrupt-parent

指定设备结点所依附的中断控制器,当结点没有指定interrupt-parent时,则从父结点继承

interrupts

指定设备结点的中断号和触发方式,该属性cell个数,由它依附的中断控制器#interrupt-cells决定,另外某些设备还可以使用多个中断号

比如:某设备结点中断类型为SPI interrupts,中断号为168和169,高电平触发,该设备结点的中断属性定义为

interrupts-parents = <&gic>;

interrupts = <0 168 4>, <0 169 4>;

                   图2

上图2为watchdog结点信息,使用SPI interrupts方式,中断号为75,高电平触发

GPIO控制器

对于GPIO控制器结点,它提供如下属性:

gpio-controller 表明自己的身份为gpio控制器

gpio-cells

该属性与#address-cells相似,它声明了设备结点gpio cells的个数。

一般gpio控制器#gpio-cells = <2>

第一个cell表示gpio号,第二个cell表示gpio有效电平

                     图3

上图3为典型的gpio控制器结点

从上图3可知:

设备结点gpio cells的个数为2

第一个cell表示gpio号,第二个cell表示gpio有效电平

设备结点与gpio相关的属性

                     图4

上图4为mmc1结点信息,该结点定义了cd-gpios属性,该属性表示使用219号gpio引脚(32*6+27)作为热插拔探测引脚,有效电平为低电平。

在mmc控制器驱动中会通过如下方法获取cd-gpios值

of_get_named_gpio(np, “cd-gpios”, 0);

  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是一个使用epoll机制监测GPIO中断的C语言demo: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <fcntl.h> #include <poll.h> #include <errno.h> #include <string.h> #include <sys/epoll.h> #define GPIO_PIN_NUM 17 // 监测的GPIO引脚号 #define MAX_EVENTS 2 // 监测的最大事件数 int main() { int fd, epoll_fd, ret, count = 0; char buf[MAX_EVENTS]; struct epoll_event event, events[MAX_EVENTS]; struct pollfd pfd; // 打开GPIO文件 fd = open("/sys/class/gpio/gpio17/value", O_RDONLY); if(fd < 0) { printf("Failed to open GPIO file!\n"); return -1; } // 设置GPIO引脚为输入模式 fd = open("/sys/class/gpio/gpio17/direction", O_WRONLY); if(fd < 0) { printf("Failed to open GPIO direction file!\n"); return -1; } write(fd, "in", 2); close(fd); // 创建epoll实例 epoll_fd = epoll_create(1); if(epoll_fd < 0) { printf("Failed to create epoll instance!\n"); return -1; } // 添加GPIO文件描述符到epoll实例中 event.events = EPOLLIN | EPOLLET; event.data.fd = fd; ret = epoll_ctl(epoll_fd, EPOLL_CTL_ADD, fd, &event); if(ret < 0) { printf("Failed to add GPIO file descriptor to epoll instance!\n"); return -1; } // 等待GPIO中断事件发生 while(1) { ret = epoll_wait(epoll_fd, events, MAX_EVENTS, -1); if(ret < 0) { printf("Failed to wait for events!\n"); return -1; } else if(ret == 0) { printf("Timeout!\n"); } else { // 遍历所有就绪事件 for(int i = 0; i < ret; i++) { if(events[i].events & EPOLLIN) { // 读取GPIO文件内容 lseek(events[i].data.fd, 0, SEEK_SET); read(events[i].data.fd, buf, MAX_EVENTS); // 处理GPIO中断事件 if(buf[0] == '0') { printf("GPIO interrupt detected! Count: %d\n", ++count); } } } } } // 关闭文件描述符和epoll实例 close(fd); close(epoll_fd); return 0; } ``` 这个demo使用了poll()和epoll_wait()函数来监测GPIO中断事件。首先,程序打开GPIO文件并将GPIO引脚设置为输入模式。然后,创建epoll实例并将GPIO文件描述符添加到epoll实例中。最后,程序在一个无限循环中等待epoll事件的发生,当有事件发生时,程序读取GPIO文件内容并处理GPIO中断事件。需要注意的是,该程序只监测一个GPIO引脚的中断事件,如果需要监测多个GPIO引脚,需要在程序中添加相应的代码。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值