/*****************************************************************
* gpio irq
*
* 一直以来都没了解过gpio的irq,如何使用,今天正好了解下。
*  本文从内核和应用层分析gpio的irq,并写验证程序。
*
* 本文部分程序内容源自以下链接,并自己更改进行验证。验证平台freescal imx6.
* http://blog.ittraining.com.tw/2015/05/raspberry-pi-b-pi2-linux-gpio-button.html
* http://m.blog.csdn.net/article/details?id=51187244
* http://blog.csdn.net/gqb_driver/article/details/8620809
* 参考文档:
* kernel/Documentation/gpio.txt
*
* Tony Liu, 2016-5-26, Shenzhen
*****************************************************************/
.内核中验证
1.1 程序 #include <linux/module.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/gpio.h>
#include <linux/interrupt.h> #define LED IMX_GPIO_NR(2, 3)
#define BUTTON IMX_GPIO_NR(3, 23)
#define INT_NAME "BUTTON_INT"
#define DEV_NAME "INTERRUPT" unsigned short int button_irq = ;
static unsigned long flags = ;
static int led_trigger = ; static irqreturn_t button_isr(int irq, void *data)
{
/* 使cpu不接受其他中断 */
local_irq_save(flags);
printk("button_isr !!!!\n");
gpio_set_value(LED, led_trigger);
led_trigger = led_trigger ? ():();
local_irq_restore(flags);
return IRQ_HANDLED;
} int __init gpio_irq_init(void)
{
int ret;
ret = gpio_request(LED, "led");
if (ret) {
printk("get led FAILED!\n");
return ret;
}
gpio_direction_output(LED, );
ret = gpio_request(BUTTON, "button");
if (ret) {
printk("get button FAILED!\n");
return ret;
}
button_irq = gpio_to_irq(BUTTON);
if (button_irq < ) {
printk("gpio_to_irq FAILED!\n");
return -;
}
printk("irq:%d\n", button_irq);
ret = request_irq(button_irq, button_isr, IRQF_TRIGGER_RISING, INT_NAME, DEV_NAME);
if (ret < )
return -; return ret;
} void __exit gpio_irq_exit(void)
{
gpio_free(LED);
gpio_free(BUTTON);
//需要释放申请的中断,否这下次不能使用,并且id要是申请使跳得内容相同,否则会出错。
free_irq(button_irq, DEV_NAME);
} module_init(gpio_irq_init);
module_exit(gpio_irq_exit); MODULE_LICENSE("GPL"); 1.2 分析
/*
irq:gpio_to_irq获得的中断号
handler: 中断处理函数
*devname: 会在/proc/intertupt中显示
*dev_id: request和free的时候必须相同,否则会出错。可以为NULL
*/
int request_irq(unsigned int irq,
irq_handler_t handler,
unsigned long flags, const char *devname, void *dev_id) //文件系统中查看缩写的dev_id,开头的343是中断号
root@freescale ~$ cat /proc/interrupts
CPU0 CPU1
......
: GPIO BUTTON_INT .应用层验证
2.1 验证代码(poll函数) #include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <poll.h> int main(int argc, char *argv[])
{
int fd, ret;
char value;
struct pollfd pfd;
//打开gpio87,对应imx6的IMX_GPIO_NR(3, 23),我连接了一个按键
system("echo \"87\" > /sys/class/gpio/export");
/*
* none 表示引脚为输入,不是中断引脚
* rising 表示引脚为中断输入,上升沿触发
* falling 表示引脚为中断输入,下降沿触发
* both 表示引脚为中断输入,边沿触发
*/
system("echo \"both\" > /sys/class/gpio/gpio87/edge"); fd = open("/sys/class/gpio/gpio87/value", O_RDWR);
if (fd < ) {
printf("open file fail\n");
return -;
}
/* poll要监听的文件描述符号 */
pfd.fd = fd;
/* 监听的文件的事件类型,当配置为中断'edge‘的时候.events必须设置为POLLPRI
   * 详情查看内核中的文档gpio.txt */
pfd.events = POLLPRI | POLLERR;
/* 由于value文件中本来就有数据需要读取,没有读取的话,会被当成一个中断而进行处理 */
//到文件开头读取
ret = lseek(fd, , SEEK_SET);
if (ret == -) {
printf("lseek error\n");
return -;
}
//读取1字节
ret = read(fd, &value, ); while () {
/* 监听个数1, 等待时间无限 */
ret = poll(&pfd, , -);
if (ret == -) {
printf("poll error\n");
return -;
}
/* 监听的时间会保存在revents成员中 */
if (pfd.revents & POLLPRI) {
//文件指针只想文件的开头
ret = lseek(fd, , SEEK_SET);
if (ret == -) {
printf("lseek error\n");
return -;
}
//读取,结果为字符'0'或者‘1’
read(fd, &value, );
printf("value:%c\n", value);
}
if (pfd.revents & POLLERR) {
printf("poll error\n");
}
} close(fd); return ;
} 2.1 验证代码(select函数) #include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/select.h> int main(int argc, char *argv[])
{
int fd[], ret;
int maxfd;
char value;
fd_set readfd;
fd_set exceptfd;
//打开gpio87,对应imx6的IMX_GPIO_NR(3, 23),我连接了一个按键
system("echo 87 > /sys/class/gpio/export");
//打开gpio94,对应imx6的IMX_GPIO_NR(3, 30),我连接了一个按键
system("echo 94 > /sys/class/gpio/export"); system("echo in > /sys/class/gpio/gpio94/direction");
system("echo in > /sys/class/gpio/gpio87/direction");
/*
* none 表示引脚为输入,不是中断引脚
* rising 表示引脚为中断输入,上升沿触发
* falling 表示引脚为中断输入,下降沿触发
* both 表示引脚为中断输入,边沿触发
*/
system("echo falling > /sys/class/gpio/gpio87/edge");
system("echo falling > /sys/class/gpio/gpio94/edge"); fd[] = open("/sys/class/gpio/gpio87/value", O_RDWR);
if (fd[0] < ) {
printf("open file fail\n");
return -;
} fd[] = open("/sys/class/gpio/gpio94/value", O_RDWR);
if (fd[1] < ) {
printf("open file fail\n");
return -;
} /* 由于value文件中本来就有数据需要读取,没有读取的话,会被当成一个中断而进行处理 */
//到文件开头读取
ret = lseek(fd[], , SEEK_SET);
if (ret == -) {
printf("lseek error\n");
return -;
}
//读取1字节
ret = read(fd[], &value, ); ret = lseek(fd[], , SEEK_SET);
if (ret == -) {
printf("lseek error\n");
return -;
}
//读取1字节
ret = read(fd[], &value, ); while () {
FD_ZERO(&exceptfd);
FD_SET(fd[], &exceptfd);
FD_SET(fd[], &exceptfd);
maxfd = (fd[] > fd[] ? fd[] : fd[]);
//使用select获取中断,必须将fd放在exceptfd的位置
ret = select(maxfd +, NULL, NULL, &exceptfd, NULL);
if (ret == -) {
printf("select operation fail\n");
return ret;
}
if (FD_ISSET(fd[], &exceptfd)) {
ret = lseek(fd[], , SEEK_SET);
if (ret == -) {
printf("lseek error\n");
return -;
}
read(fd[], &value, );
printf("0: value:%c\n", value);
}
if (FD_ISSET(fd[], &exceptfd)) {
ret = lseek(fd[], , SEEK_SET);
if (ret == -) {
printf("lseek error\n");
return -;
} read(fd[], &value, ); printf("1: value:%c\n", value); }
}
//关闭文件描述符
close(fd[]);
close(fd[]); //操作完毕,释放gpio
system("echo 87 > /sys/class/gpio/unexport");
system("echo 94 > /sys/class/gpio/unexport"); return ;
} .1分析
<>申请gpio, gpio_request
echo "" > /sys/class/gpio/export
会在/sys/class/gpio目录下边生成一个gpio87的目录 <>释放gpio, gpio_free
echo "" > /sys/class/gpio/unexport <>申请之后,gpio87目录的内容如下:
root@freescale /sys/class/gpio/gpio87$ ls -l
-rw-r--r-- root root Jan : active_low
-rw-r--r-- root root Jan : direction
-rw-r--r-- root root Jan : edge
drwxr-xr-x root root Jan : power
lrwxrwxrwx root root Jan : subsystem -> ../../../../class/gpio
-rw-r--r-- root root Jan : uevent
-rw-r--r-- root root Jan : value
<>设置输入输出
echo "in" > direction
echo "out" > direction
<>设置输出值
echo "" > value //非0值都会当作1
echo "" > value 详细参考内核的文档:
kernel/Documentation/gpio.txt
如下:
There are three kinds of entry in /sys/class/gpio: - Control interfaces used to get userspace control over GPIOs; - GPIOs themselves; and - GPIO controllers ("gpio_chip" instances). That's in addition to standard files including the "device" symlink. The control interfaces are write-only: /sys/class/gpio/ "export" ... Userspace may ask the kernel to export control of
a GPIO to userspace by writing its number to this file. Example: "echo 19 > export" will create a "gpio19" node
for GPIO #, if that's not requested by kernel code. "unexport" ... Reverses the effect of exporting to userspace. Example: "echo 19 > unexport" will remove a "gpio19"
node exported using the "export" file. GPIO signals have paths like /sys/class/gpio/gpio42/ (for GPIO #)
and have the following read/write attributes: /sys/class/gpio/gpioN/ "direction" ... reads as either "in" or "out". This value may
normally be written. Writing as "out" defaults to
initializing the value as low. To ensure glitch free
operation, values "low" and "high" may be written to
configure the GPIO as an output with that initial value. Note that this attribute *will not exist* if the kernel
doesn't support changing the direction of a GPIO, or
it was exported by kernel code that didn't explicitly
allow userspace to reconfigure this GPIO's direction. "value" ... reads as either (low) or (high). If the GPIO
is configured as an output, this value may be written;
any nonzero value is treated as high. If the pin can be configured as interrupt-generating interrupt
and if it has been configured to generate interrupts (see the
description of "edge"), you can poll() on that file and
poll() will return whenever the interrupt was triggered. If
you use poll(), set the events POLLPRI and POLLERR. If you
use select(), set the file descriptor in exceptfds. After
poll() returns, either lseek() to the beginning of the sysfs
file and read the new value or close the file and re-open it
to read the value. "edge" ... reads as either "none", "rising", "falling", or
"both". Write these strings to select the signal edge(s)
that will make poll() on the "value" file return. This file exists only if the pin can be configured as an
interrupt generating input pin. "active_low" ... reads as either (false) or (true). Write
any nonzero value to invert the value attribute both
for reading and writing. Existing and subsequent
poll() support configuration via the edge attribute
for "rising" and "falling" edges will follow this
setting. GPIO controllers have paths like /sys/class/gpio/gpiochip42/ (for the
controller implementing GPIOs starting at #) and have the following
read-only attributes: /sys/class/gpio/gpiochipN/ "base" ... same as N, the first GPIO managed by this chip "label" ... provided for diagnostics (not always unique) "ngpio" ... how many GPIOs this manges (N to N + ngpio - ) Board documentation should in most cases cover what GPIOs are used for
what purposes. However, those numbers are not always stable; GPIOs on
a daughtercard might be different depending on the base board being used,
or other cards in the stack. In such cases, you may need to use the
gpiochip nodes (possibly in conjunction with schematics) to determine
the correct GPIO number to use for a given signal.

gpio irq的更多相关文章

  1. imx6 gpio irq

    /***************************************************************** * gpio irq * * 一直以来都没了解过gpio的irq, ...

  2. 基於tiny4412的Linux內核移植--- 中斷和GPIO學習(3)

    作者 彭東林 pengdonglin137@163.com 平臺 tiny4412 ADK Linux-4.4.4 u-boot使用的U-Boot 2010.12,是友善自帶的,爲支持設備樹和uIma ...

  3. 基於tiny4412的Linux內核移植--- 中斷和GPIO學習(1)

    作者 彭東林 pengdonglin137@163.com 平臺 tiny4412 ADK Linux-4.4.4 u-boot使用的U-Boot 2010.12,是友善自帶的,爲支持設備樹和uIma ...

  4. A20 GPIO中断类型差别结果迥异的问题思考

    A20GPIO中断类型差别结果迥异的问题思考 最近在使用全志A20做开发时,发现在处理中断的时候,用电平触发模式,报中断比较乱,用边沿触发则很稳定,不会乱报.笔者感到比较困惑,笔者用电平触发写的cod ...

  5. Linux中的gpio口使用方法

    Linux中的IO使用方法 应该是新版本内核才有的方法.请参考:./Documentation/gpio.txt文件 提供的API:驱动需要包含 #include <linux/gpio.h&g ...

  6. linux设备驱动程序--gpio控制

    gpio驱动程序 上一章节linux设备驱动程序--创建设备节点章节主要介绍了linux字符设备驱动程序的框架,从这一章节开始我们讲解各种外设的控制,包括gpio,i2c,dma等等,既然是外设,那就 ...

  7. Linux 内核层和 用户层 配置 GPIO 引脚

    Linux BSP 开发的基础就是和GPIO打交道, 下面总结下这几天对某家开发板的GPIO控制的知识. 公司的开发板用的是 DTB  模式 ,首先,进入 dts,dtsi文件查看关于GPIO 的模块 ...

  8. 痞子衡嵌入式:利用GPIO模块来测量i.MXRT1xxx的系统中断延迟时间

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT1xxx的系统中断延迟时间. 在 <Cortex-M系统中断延迟及其测量方法简介> 一文里,痞子衡介绍了 Cor ...

  9. Android系统启动过程-uBoot+Kernel+Android

    摘要:本文是参考大量网上资源在结合自己查看源代码总结出来的,让自己同时也让大家加深对Android系统启动过程有一个更加深入的了解!再次强调,本文的大多数功劳应归功于那些原创者们,同时一些必要的参考链 ...

随机推荐

  1. vue v-model 与 组件化的表单组件如何沟通

    参考mint-ui的代码: https://github.com/ElemeFE/mint-ui/blob/master/packages/radio/src/radio.vue https://gi ...

  2. textureCache中的等价路径问题

    自己的引擎里做了个简单的TextueCache,每次新创建一个纹理,先到TextureCache里查找有没有路径相同的,如果有就直接返回纹理,如果没有加载图片创建纹理并将图片路径缓存起来.另外为了标准 ...

  3. haproxy有关session的问题

    在实验的时候遇到一个问题就是当我登录网站的时候,然后我再刷新一下,用户的状态就退出了 我现在的框架是这样的,前面有一台haproxy作为反向代理,后面有两台服务器跑的是java应用.后面两台服务器做的 ...

  4. Linux内存管理Swap和Buffer Cache机制

    Linux内存管理Swap和Buffer Cache机制 一个完整的Linux系统主要有存储管理,内存管理,文件系统和进程管理等几方面组成,贴出一些以前学习过的一个很好的文章.与大家共享!以下主要说明 ...

  5. jQuery.ajax() 如何设置 Headers 中的 Accept 内容

    其实很简单,首先如果是常见类型,则请直接设置 dataType 属性 $.ajax({ dataType: "json", type: "get", succe ...

  6. Scala之集合Collection

    概述 Scala的集合类能够从三个维度进行切分: 可变与不可变集合(Immutable and mutable collections) 静态与延迟载入集合 (Eager and delayed ev ...

  7. Solr 数字字符不能搜索的一个问题

    问题一: 测试人员告诉我数字不能被搜索.于是开始找原因: <fields> ***<field name="productName" type="tex ...

  8. Ribbon对于SocketTimeOutException重试的坑以及重试代码解析

    背景 本文基于Spring-Cloud, Daltson SR4 微服务一般多实例部署,在发布的时候,我们要做到无感知发布:微服务调用总会通过Ribbon,同时里面会实现一些重试的机制,相关配置是: ...

  9. 获取不到Repeater控件中的CheckBox选中状态

    写在前面的话:在做一个项目的时候,需要使用到Repeater控件,并且在Repeater控件内放置了CheckBox控件来标志需要删除的行,选中后,在后台取到的CheckBox的值总是为false.最 ...

  10. 常见电源品牌大揭密(转贴自pceva,作者royalk)

    常见电源品牌大揭密(转贴自pceva,作者royalk) 介绍电源品牌代工厂之前,必须介绍一下电源分类: 标准电源 标准电源就是电脑城装机用得最多的电源,性能正常,外观一般,原生接线,也没有什么风扇停 ...