IMX257实现GPIO-IRQ中断按键驱动程序

发布者:SparklingEyes最新更新时间:2024-08-16 来源: cnblogs关键字:GPIO  IRQ  中断按键 手机看文章 扫描二维码
随时随地手机看文章

    昨天我们已经实现了中断查询的方式实现GPIO按键驱动程序,但是,有一个缺点就是,当我们把应用程序放在后台执行时,即便没有按键,应用程序while循环中的read函数也不断的在运行,严重的导致了CPU资源的浪费。


    本文中,我们在前面按键查询驱动程序的基础上来修改。


    大概介绍一下设计思路吧:


    和前面的差不多,当我们加载驱动时,首先在init函数中,对GPIO功能进行模式设置,都设置为GPIO模式,然后申请GPIO引脚的内存,


接着,当我们应用程序使用ioctl函数的gpio_input命令时,将所有的GPIO引脚设置为输入,并且22K上拉模式,当应用程序使用read函数读取时,如果按键没有按下,则会在read函数中睡眠(可被中断唤醒),交出CPU的使用权,当按键按下时,首先会在中断程序中进行唤醒操作,在read函数中接着运行,读取GPIO引脚的电平,并且将数据从内核空间传递到用户空间中,并且打印出来。


    最后,当我们不要用驱动程序时,rsmod会调用exit函数,释放所有我们已经申请的GPIO资源。


    好了,大概的思路就是这样,和前面的查询驱动程序,差不多,只不过就加了一个中断,和睡眠而已。


附上驱动程序代码


  1 /******************************

  2     linux key_query

  3  *****************************/

  4 #include

  5 #include

  6 #include

  7 #include

  8 #include

  9 #include

 10 #include

 11 #include

 12 #include

 13 #include

 14 #include

 15 #include

 16 #include //error: 'TASK_INTERRUPTIBLE' undeclared 

 17 #include

 18 

 19 #include 'mx257_gpio.h'

 20 #include 'mx25_pins.h'

 21 #include 'iomux.h'

 22 

 23 #define Driver_NAME 'key_interrupt'

 24 #define DEVICE_NAME 'key_interrupt'

 25 

 26 #define GPIO2_21    MX25_PIN_CLKO

 27 #define GPIO3_15    MX25_PIN_EXT_ARMCLK

 28 #define GPIO2_10    MX25_PIN_A24

 29 #define GPIO2_11    MX25_PIN_A25

 30 #define GPIO2_8     MX25_PIN_A22

 31 #define GPIO2_9     MX25_PIN_A23

 32 #define GPIO2_6     MX25_PIN_A20

 33 #define GPIO2_7     MX25_PIN_A21

 34 //command

 35 #define key_input     0

 36 #define version        1

 37 

 38 //interrupt head

 39 static DECLARE_WAIT_QUEUE_HEAD(key_interrupt_wait);

 40 static volatile unsigned char ev_press;  

 41 

 42 static int major=0;

 43 

 44 //auto to create device node

 45 static struct class *drv_class = NULL;

 46 static struct class_device *drv_class_dev = NULL;

 47 

 48 

 49 /* 应用程序对设备文件/dev/key_query执行open(...)时,

 50  * 就会调用key_open函数*/

 51 static int key_open(struct inode *inode, struct file *file)

 52 {

 53     printk('<0>function open!nn');

 54     

 55     return 0;

 56 }

 57 

 58 /* 中断程序key_irq */

 59 static irqreturn_t key_irq(int irq, void *dev_id)

 60 {

 61     //发生了中断

 62     printk('<0>function interrupt key_irq!nn');

 63     ev_press = 1;

 64     wake_up_interruptible(&key_interrupt_wait);

 65 

 66     return IRQ_RETVAL(IRQ_HANDLED);

 67 }

 68 

 69 

 70 static int key_read(struct file *filp, char __user *buff, size_t count, loff_t *offp)

 71 {

 72     int ret;

 73     //nt cnt=0;

 74     unsigned char key_vals[8];

 75 

 76     //如果按键没有按下,没有中断,休眠

 77     wait_event_interruptible(key_interrupt_wait,ev_press);

 78 

 79     // reading the pins value

 80     key_vals[0] = gpio_get_value(IOMUX_TO_GPIO(GPIO2_6)) ? 1 : 0;

 81     key_vals[1] = gpio_get_value(IOMUX_TO_GPIO(GPIO2_7)) ? 1 : 0;

 82     key_vals[2] = gpio_get_value(IOMUX_TO_GPIO(GPIO2_8)) ? 1 : 0;

 83     key_vals[3] = gpio_get_value(IOMUX_TO_GPIO(GPIO2_9)) ? 1 : 0;

 84     key_vals[4] = gpio_get_value(IOMUX_TO_GPIO(GPIO2_10)) ? 1 : 0;

 85     key_vals[5] = gpio_get_value(IOMUX_TO_GPIO(GPIO2_11)) ? 1 : 0;

 86     key_vals[6] = gpio_get_value(IOMUX_TO_GPIO(GPIO2_21)) ? 1 : 0;

 87     key_vals[7] = gpio_get_value(IOMUX_TO_GPIO(GPIO3_15)) ? 1 : 0;

 88     

 89     ret = copy_to_user(buff,key_vals,sizeof(key_vals));

 90     if(ret){

 91         ;

 92     }

 93     ev_press = 0;

 94     //printk('<0>%04d key pressed: %d %d %d %d %d %d %d %dn',cnt++,key_vals[0],key_vals[1],key_vals[2],key_vals[3],key_vals[4],key_vals[5],key_vals[6],key_vals[7]); 

 95 

 96     return sizeof(key_vals);

 97 }

 98 

 99 static ssize_t key_write(struct file *file, const char __user *buf, size_t count, loff_t * ppos)

100 {

101     printk('<0>function write!nn');

102     

103     return 1;

104 }

105 

106 static int  key_release(struct inode *inode, struct file *filp)

107 {

108     printk('<0>function release!nn');

109     //释放中断

110     free_irq(IOMUX_TO_IRQ(GPIO2_21), (void *)1);

111     //free_irq(IOMUX_TO_IRQ(GPIO3_15), (void *)1);

112     free_irq(IOMUX_TO_IRQ(GPIO2_11), (void *)1);

113     free_irq(IOMUX_TO_IRQ(GPIO2_10), (void *)1);

114     free_irq(IOMUX_TO_IRQ(GPIO2_9), (void *)1);

115     //free_irq(IOMUX_TO_IRQ(GPIO2_8), (void *)1);

116     free_irq(IOMUX_TO_IRQ(GPIO2_7), (void *)1);

117     free_irq(IOMUX_TO_IRQ(GPIO2_6), (void *)1);

118     return 0;

119 }

120 

121 static int key_ioctl(struct inode *inode,struct file *flip,unsigned int command,unsigned long arg)

122 {

123     int ret;

124     printk('<0>function ioctl!nn');

125     switch (command) {

126         case key_input:    

127             //设置所有的引脚为输入

128             gpio_direction_input(IOMUX_TO_GPIO(GPIO2_21));

129             gpio_direction_input(IOMUX_TO_GPIO(GPIO3_15));

130             gpio_direction_input(IOMUX_TO_GPIO(GPIO2_10));

131             gpio_direction_input(IOMUX_TO_GPIO(GPIO2_11));

132             gpio_direction_input(IOMUX_TO_GPIO(GPIO2_8));

133             gpio_direction_input(IOMUX_TO_GPIO(GPIO2_9));

134             gpio_direction_input(IOMUX_TO_GPIO(GPIO2_6));

135             gpio_direction_input(IOMUX_TO_GPIO(GPIO2_7));

136             printk('<0>have setting all pins to gpio input mod !n');

137             //设置GPIO引脚为上拉模式

138             mxc_iomux_set_pad(GPIO2_6, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

139             mxc_iomux_set_pad(GPIO2_7, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

140             //mxc_iomux_set_pad(GPIO2_8, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

141             mxc_iomux_set_pad(GPIO2_9, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

142             mxc_iomux_set_pad(GPIO2_10, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

143             mxc_iomux_set_pad(GPIO2_11, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

144             mxc_iomux_set_pad(GPIO2_21, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

145             //mxc_iomux_set_pad(GPIO3_15, PAD_CTL_HYS_SCHMITZ | PAD_CTL_PKE_ENABLE | PAD_CTL_PUE_PULL | PAD_CTL_22K_PU);

146 

147             //设置GPIO引脚中断  ,下降沿触发

148             request_irq(IOMUX_TO_IRQ(GPIO2_7), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO2_7', (void *)1);

149             request_irq(IOMUX_TO_IRQ(GPIO2_6), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO2_6', (void *)1);

150             request_irq(IOMUX_TO_IRQ(GPIO2_9), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO2_9', (void *)1);

151             request_irq(IOMUX_TO_IRQ(GPIO2_10), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO2_10', (void *)1);

152             request_irq(IOMUX_TO_IRQ(GPIO2_11), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO2_11', (void *)1);

153             request_irq(IOMUX_TO_IRQ(GPIO2_21), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO2_21', (void *)1);

154             //request_irq(IOMUX_TO_IRQ(GPIO3_15), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO3_15', (void *)1);

155             //request_irq(IOMUX_TO_IRQ(GPIO2_8), key_irq, IRQF_TRIGGER_FALLING, 'key_GPIO2_8', (void *)1);

156             printk('<0>have setting all pins to gpio interrupt mod by IRQF_TRIGGER_FALLING !n');

157 

158             break;

159         case version:

160             printk('<0>hello,the version is 0.1.0nn');

[1] [2]
关键字:GPIO  IRQ  中断按键 引用地址:IMX257实现GPIO-IRQ中断按键驱动程序

上一篇:【改进】IMX257实现GPIO-IRQ中断按键获取键值驱动程序
下一篇:IMX257实现GPIO-查询按键驱动程序

推荐阅读最新更新时间:2026-03-21 10:35

【改进】IMX257实现GPIO-IRQ中断按键获取键值驱动程序
一、使用struct pin_desc 管理按键的值 1.定义结构体 2.将前面我们申请中断时写的(void *)1修改为 &pins_desc 在ioctl中,设置中断中修改 在key_release中释放中修改 3.在中断程序中利用我们定义的struc pins_desc判断并得到按键的值 4.得到按键键值后,唤醒程序,在read函数中返回键值 附上驱动源程序: 1 /****************************** 2 linux key_query 3 *****************************/ 4 #include linux/module.h 5
[单片机]
【改进】<font color='red'>IMX257</font>实现<font color='red'>GPIO</font>-<font color='red'>IRQ</font><font color='red'>中断</font><font color='red'>按键</font>获取键值驱动程序
【改进Poll定时查询】IMX257实现GPIO-IRQ中断按键获取键值驱动程序
按键驱动程序中,如果不使用read函数中使程序休眠的,而是还是使用查询方式的话,可以使用Poll函数,来控制一定时间内,如果有按键发生,则立即返回键值。 同时,poll也可以同时监控多个(比如说按键,鼠标,等)一旦发生事件则立即返回。 我们在linux查看帮助: 从帮助中的说明得知, poll, ppoll - wait for some event on a file descriptor poll就是监控某个设备的事件。 修改驱动程序 1.增加头文件 #include linux/poll.h 2.增加key_poll 方法 static unsigned key_poll(struct file *fil
[单片机]
【改进Poll定时查询】<font color='red'>IMX257</font>实现<font color='red'>GPIO</font>-<font color='red'>IRQ</font><font color='red'>中断</font><font color='red'>按键</font>获取键值驱动程序
msp432快速入门第六节之按键扫描和GPIO中断操作
(一)按键扫描 (1)配置引脚 在按键扫描这一节,仅需要配置好引脚即可,打开TI的GPIO驱动库,可以看到GPIO设置函数H文件如下: 在这个函数上方有对函数的说明: 解释为:该函数配置MSP432的GPIO引脚为输入模式,没有指明被配置的引脚是否被上拉或者下拉,当然如果需要配置输入上拉或者输入下拉需要使用此函数配置: 在这里我们直接使用普通配置,配置引脚为输入即可。 如下进行引脚初始化:(板载两个按键,一左一右均可以使用,注意复位键是在上面的) //key pin init void Key_Init(void) { GPIO_setAsInputPinWithPullUpResistor(KEY_LEFT
[单片机]
msp432快速入门第六节之<font color='red'>按键</font>扫描和<font color='red'>GPIO</font><font color='red'>中断</font>操作
STM32CubeMX学习笔记2:GPIO按键外部中断控制LED状态
MCU:STM32F103ZET6 IDE: MDK-ARM V5 +STM32CubeMX5.0.0 本次实现的功能是通过按键的外部中断,来改变对应LED的状态,需要将按键的I/O端口映射到外部中断线上。 需要初始化的按键I/O口分别对应PA0PE2PE3PE4,LEDI/O口分别为PC0PC1PC2PC3PC4。 一. 在 Pinout&Configuration---System Core中: 1. 首先设置时钟RCC的HSE(外部高速时钟)为晶振模式:Crystal/ceramic Resonator 2. 设置系统SYS的Debug为Serial Wire: 3. 设置GPIO中的LED
[单片机]
STM32CubeMX学习笔记2:<font color='red'>GPIO</font><font color='red'>按键</font>外部<font color='red'>中断</font>控制LED状态
基于linux操作系统下s5pv210板子的按键中断实验
首先,在做这个实验之前有三件事是要明确的: 1. linux下的中断实验不需要像裸机实验一样要自己写取消屏蔽,清除中断标志位的操作,系统会自动帮你完成; 2.中断号的申请我使用的是gpio_to_irq()这个宏,它会帮我们自动分配中断号,返回值为中断号; 3. 在每个板子配套来的内核代码大部分都是已经包含了按键中断驱动,如果想另自己写的按键中断驱动不与内核本身带的发生中断号上的冲突,应先找到内核代码下的arch/arm/mach-s5pv210/mach-mini210.c中的gpio_bottons中的对按键初始化的代码注释掉(其他板子也类似),如下代码所示: static struct gpio_keys_
[单片机]
6.6410和210的按键中断编程
首先是打开6410底板原理图: 可以看到OK6410有六个按键: 可以看到OK6410的六个按键对应的引脚是KEYINT1、KEYINT2、KEYINT3、KEYINT4、KEYINT5和KEYINT6。接着在核心板的原理图里搜索这个词: 可以看到按键中断与GPN系列寄存器的引脚是互用的,接着就是在芯片手册里查看有关GPN系列寄存器的信息,重点是关注GPN控制寄存器:GPNCON: 这里我们使用到的按键对应的中断,需要配置成中断的方式,对应的GPN位设置为10,中断功能,即需要将KEYINT1-6设置为中断功能的,即是将GPNCON对应的前12位设置为101010101010即可。这样就把对应的GPN0-5设置
[单片机]
6.6410和210的<font color='red'>按键</font><font color='red'>中断</font>编程
专题2-通过按键中断\2440按键中断编程lesson2
1、程序优化 修改Makefile 把main.c里面的mmu代码复制到mmu.c并修改如下 main.c的修改 由于在bootloader当中一般不会使用MMU,所以 main.c 加入led.c文件 makefile 2440中断源初始化 打开开发板底板与核心板原理图 底板 比如K1对应EINT1,然后到核心板去搜EINT1 然后去2440芯片手册去找到GPF这一组IO口 创建一个button.c,加入到makefile里面去button.o 按键初始化 编写button.c文件 #define GPFCON (volatile unsigned long *)0x56000050
[单片机]
专题2-通过<font color='red'>按键</font>玩<font color='red'>中断</font>\2440<font color='red'>按键</font><font color='red'>中断</font>编程lesson2
(linux自学笔记)linux按键中断驱动
通常开发板自带按键中断的驱动,中断已被注册至内核。重新编译linux内核去掉自带驱动才能使用自己编写的驱动。 linux中断程序可分解为顶半部与底半部机制。顶半部完成尽可能少的紧急功能,底半部可以被新的中断打断。 驱动程序 #include linux/module.h #include linux/kernel.h #include linux/fs.h #include linux/init.h #include linux/delay.h #include linux/poll.h #include linux/irq.h #include asm/irq.h #include linux/inter
[单片机]
(linux自学笔记)linux<font color='red'>按键</font><font color='red'>中断</font>驱动
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2026 EEWORLD.com.cn, Inc. All rights reserved