首页 > 其他 > 详细

字符设备驱动之按键驱动(中断)

时间:2019-11-17 22:00:08      阅读:113      评论:0      收藏:0      [点我收藏+]

单片机下中断处理过程:

①按键按下

CPU发生中断,跳转到异常向量入口

③跳转到中断处理函数

a、保存被中断的现场

b、执行中断处理函数

c、恢复被中断现场

? ?

Linux下中断处理过程:

ARM架构CPU的异常向量地址可以是0x00000000,也可以是0xFFFF0000Linux内核使用的是0xFFFF00000xFFFF0000是一个虚拟地址)。因此,在建立虚拟地址映射之后,需要把异常向量复制到虚拟地址0xFFFF0000上去,Linux使用trap_init函数。

技术分享图片

技术分享图片

?

函数接口:

注册中断:

1 int request_irq(unsigned int irq, irq_handler_t handler,
2 ????????????????unsigned long irqflags, const char *devname, void *dev_id)

参数说明:

irq: 中断号

handle: 处理函数

irqflag: 触发方式

devname: 中断名字

dev_id: 设备号(用户自定义)

? ?

①分配了一个irqaction结构

setup_irq(irq, action);

? ?

卸载中断:

1 void free_irq(unsigned int irq, void *dev_id)

参数说明:

irq: 中断号

dev_id: 设备号

①根据dev_id,删除用户注册的中断处理函数(释放irqaction结构)

②当当前中断号下无irqaction结构时,禁止中断

? ?

一些常用的命令:

cat /proc/devices //查看设备

cat /proc/interrupts //查看中断

exec 5</dev/myKey //打开设备

exec 5<&- //关闭设备

? ?

实例:

driver.c

1 #include <linux/module.h>
2 #include <linux/kernel.h>
3 #include <linux/fs.h>
4 #include <linux/init.h>
5 #include <linux/delay.h>
6 #include <linux/irq.h>
7 #include <asm/uaccess.h>
8 #include <asm/irq.h>
9 #include <asm/io.h>
10 #include <asm/arch/regs-gpio.h>
11 #include <asm/hardware.h>
12
13
14 static int major;
15
16 static struct class *myKey_class;
17 static struct class_device *myKey_class_dev;
18
19 volatile unsigned long *gpfcon;
20 volatile unsigned long *gpfdat;
21
22 volatile unsigned long *gpgcon;
23 volatile unsigned long *gpgdat;
24
25 static DECLARE_WAIT_QUEUE_HEAD(button_waitq);
26 static volatile int ev_press = 0;
27
28
29 static unsigned char keyVal = 0;
30
31 struct pin_desc {
32 ????????unsigned int pin;
33 ????????unsigned int key_val;
34 };
35
36 /*
37 * 按键按下键值为0x01,...; 松开键值为0x81,...
38 */
39 struct pin_desc pins_desc[3] = {
40 ????????{S3C2410_GPF0, 0x01},
41 ????????{S3C2410_GPF2, 0x02},????????
42 ????????{S3C2410_GPG11, 0x03},????????
43 };
44
45
46 static int myKey_open(struct inode *inode, struct file *file);
47 static int myKey_close(struct inode *, struct file *);
48 static ssize_t myKey_read(struct file *file, char __user *buf, size_t size, loff_t *ppos);
49
50
51 static struct file_operations myKey_fops = {
52 ????????.open = myKey_open,
53 ????????.read = myKey_read,
54 ????????.owner = THIS_MODULE,
55 ????????.release = myKey_close,
56 };
57
58
59 static irqreturn_t handle_buttons(int irq, void *dev)
60 {
61 ????????unsigned int kval;
62 ????????struct pin_desc *pinDesc = dev;
63
64 ????????kval = s3c2410_gpio_getpin(pinDesc->pin);
65 ????????if (kval)???????? //松开
66 ????????{
67 ????????????????keyVal = 0x80 | pinDesc->key_val;
68 ????????}
69 ????????else { //按下
70 ????????????????keyVal = pinDesc->key_val;
71 ????????}
72
73 ????????//唤醒休眠进程
74 ????????ev_press = 1;????????//中断发生标志
75 ????????wake_up_interruptible(&button_waitq);
76
77 ????????return IRQ_RETVAL(IRQ_HANDLED);
78 }
79
80
81 static int myKey_open(struct inode *inode, struct file *file)
82 {
83 ????????request_irq(IRQ_EINT0, handle_buttons, IRQT_BOTHEDGE, "S2", &pins_desc[0]);
84 ????????request_irq(IRQ_EINT2, handle_buttons, IRQT_BOTHEDGE, "S3", &pins_desc[1]);
85 ????????request_irq(IRQ_EINT19, handle_buttons, IRQT_BOTHEDGE, "S5", &pins_desc[2]);
86 ????????
87 ????????return 0;
88 }
89
90
91 static int myKey_close(struct inode *inode, struct file *file)
92 {
93 ????????free_irq(IRQ_EINT0, &pins_desc[0]);
94 ????????free_irq(IRQ_EINT2, &pins_desc[1]);
95 ????????free_irq(IRQ_EINT19, &pins_desc[2]);
96
97 ????????return 0;
98 }
99
100
101 static ssize_t myKey_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
102 {
103 ????????//无中断进入休眠
104 ????????wait_event_interruptible(button_waitq, ev_press);
105
106 ????????ev_press = 0;????????//清除中断发生标志
107 ????????copy_to_user(buf, &keyVal, 1);
108 ????????return 0;
109 }
110
111 static int __init myKey_init(void)
112 {
113 ????????/* 物理地址映射成虚拟地址 */
114 ????????gpfcon = (volatile unsigned long*)ioremap(0x56000050, 16);
115 ????????gpfdat = gpfcon + 1;
116
117 ????????gpgcon = (volatile unsigned long*)ioremap(0x56000060, 16);
118 ????????gpgdat = gpgcon + 1;
119
120 ????????major = register_chrdev(0, "myKey", &myKey_fops);
121 ????????
122 ????????myKey_class = class_create(THIS_MODULE, "myKeyclass");
123 ????????myKey_class_dev = class_device_create(myKey_class, NULL, MKDEV(major, 0), NULL, "myKey");
124
125 ????????return 0;
126 }
127
128 static void __exit myKey_exit(void)
129 {
130 ????????/* 释放虚拟地址映射 */
131 ????????iounmap(0x56000050);
132 ????????iounmap(0x56000060);
133
134 ????????unregister_chrdev(major, "myKey");
135
136 ????????class_device_unregister(myKey_class_dev);
137 ????????class_destroy(myKey_class);
138 ????????return;
139 }
140
141 module_init(myKey_init);
142 module_exit(myKey_exit);
143
144 MODULE_LICENSE("GPL");

? ?

app.c

1 #include <sys/types.h>
2 #include <sys/stat.h>
3 #include <fcntl.h>
4 #include <stdio.h>
5
6 int main (void)
7 {
8 ????????int fd;
9 ????????unsigned char keyVal;
10 ????????ssize_t rst;
11 ????????
12 ????????printf("test app!\n");
13
14 ????????fd = open("/dev/myKey", O_RDWR);
15 ????????if(fd < 0)
16 ????????{
17 ????????????????printf("open failed! %d\n", fd);
18 ????????????????return -1;
19 ????????}
20 ????????
21 ????????while(1)
22 ????????{
23 ????????????????rst = read(fd, &keyVal, 1);
24 ????????????????if (rst < 0)
25 ????????????????{????????
26 ????????????????????????printf("read failed!\n");
27 ????????????????????????continue;
28 ????????????????}
29 ????????????????printf("keyVal 0x%x\n", keyVal);
30 ????????}
31 ????????return 0;????????
32 }

? ?

Makefile

1 KERN_DIR = /work/system/linux-2.6.22.6
2
3 all:
4 ????????make -C $(KERN_DIR) M=`pwd` modules
5

6 clean:
7 ????????make -C $(
KERN_DIR) M=`pwd` modules clean
8 ????????rm -rf modules.order

9

10 obj-m += myKey_irq.o

字符设备驱动之按键驱动(中断)

原文:https://www.cnblogs.com/lilto/p/11878157.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!