Linux 驱动——Button驱动3(poll机制)
button_drv.c驱动文件:
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <asm/io.h>
#include <asm/uaccess.h>
#include <linux/device.h>
#include <asm/arch/regs-gpio.h>
#include <linux/irq.h>
#include <asm-arm/irq.h>
#include <linux/interrupt.h>
#include <linux/delay.h>
#include <asm/hardware.h>
#include <linux/poll.h>
#define DRIVER_NAME "button_drv"
#define DEVICE_NAME "button_dev"
int major;
volatile unsigned long *gpfcon;
volatile unsigned long *gpfdat;
volatile unsigned long *gpgcon;
volatile unsigned long *gpgdat;
struct class *buttondrv_class;
struct class_device *buttondrv_class_device;
unsigned int ev_press;
DECLARE_WAIT_QUEUE_HEAD(button_waitq);
struct pin_desc{
unsigned int pin;
unsigned int key_val;
};
unsigned int key_val;
struct pin_desc pins_desc[4] = {
{S3C2410_GPF0, 0x01},
{S3C2410_GPF2, 0x02},
{S3C2410_GPG3, 0x03},
{S3C2410_GPG11, 0x04},
};
irqreturn_t buttons_irq(int irq, void *dev_id)
{
unsigned int pin_val;
struct pin_desc *pin_desc = (struct pin_desc *)dev_id;
pin_val = s3c2410_gpio_getpin(pin_desc->pin);
if(pin_val)
{
key_val = 0x80 | pin_desc->key_val;
}
else
{
key_val = pin_desc->key_val;
}
wake_up_interruptible(&button_waitq);
ev_press = 1;
return IRQ_HANDLED;
}
int button_drv_open(struct inode *inode, struct file *file)
{
int ret;
ret = request_irq(IRQ_EINT0, buttons_irq, IRQT_BOTHEDGE, "S1", (void*)&pins_desc[0]);
if(ret)
{
printk("open failed 1 \n");
return -1;
}
ret = request_irq(IRQ_EINT2, buttons_irq, IRQT_BOTHEDGE, "S2", (void*)&pins_desc[1]);
if(ret)
{
printk("open failed 2 \n");
return -1;
}
ret = request_irq(IRQ_EINT11, buttons_irq, IRQT_BOTHEDGE, "S3", (void*)&pins_desc[2]);
if(ret)
{
printk("open fail 3 \n");
return -1;
}
ret = request_irq(IRQ_EINT19, buttons_irq, IRQT_BOTHEDGE, "S4", (void*)&pins_desc[3]);
if(ret)
{
printk("open fail 4 \n");
return -1;
}
return 0;
}
ssize_t button_drv_read(struct file *file, char __user *userbuf, size_t count, loff_t *off)
{
int ret;
ret = copy_to_user(userbuf, &key_val, 1);
if(ret)
{
printk("copy eror \n");
return -1;
}
ev_press = 0;
return 1;
}
int button_drv_close(struct inode *inode, struct file *file)
{
free_irq(IRQ_EINT0, (void*)&pins_desc[0]);
free_irq(IRQ_EINT2, (void*)&pins_desc[1]);
free_irq(IRQ_EINT11, (void*)&pins_desc[2]);
free_irq(IRQ_EINT19, (void*)&pins_desc[3]);
return 0;
}
unsigned int button_drv_poll(struct file *file, poll_table *wait)
{
unsigned int mask = 0;
poll_wait(file, &button_waitq, wait);
if(ev_press)
{
mask |= POLLIN | POLLRDNORM;
}
return mask;
}
struct file_operations button_drv_fops = {
.owner = THIS_MODULE,
.open = button_drv_open,
.read = button_drv_read,
.release = button_drv_close,
.poll = button_drv_poll,
};
int __init button_drv_init(void)
{
major = register_chrdev(0, DRIVER_NAME, &button_drv_fops);
if(major<0)
{
printk("fail 1 button_drv \n");
}
buttondrv_class = class_create(THIS_MODULE, DEVICE_NAME);
if(buttondrv_class<0)
{
printk("fail 2 button_dev \n");
}
buttondrv_class_device = class_device_create(buttondrv_class, NULL, MKDEV(major,0), NULL, DEVICE_NAME);
if(buttondrv_class_device<0)
{
printk("fail 3 button_dev \n");
}
gpfcon = (volatile unsigned long *)ioremap(0x56000050, 16);
gpfdat = gpfcon + 1;
gpgcon = (volatile unsigned long *)ioremap(0x56000060, 16);
gpgdat = gpgcon + 1;
printk("register button_drv \n");
return 0;
}
void __exit button_drv_exit(void)
{
unregister_chrdev(major, DEVICE_NAME);
class_device_unregister(buttondrv_class_device);
class_destroy(buttondrv_class);
iounmap(gpfcon);
iounmap(gpgcon);
printk("unregister button_drv \n");
}
module_init(button_drv_init);
module_exit(button_drv_exit);
MODULE_LICENSE("GPL");
Makefile文件:
obj-m += button_drv.o
KERN_DIR = /work/system/linux-2.6.22.6
all:
make -C $(KERN_DIR) M=`pwd` modules
clean:
rm -rf *.o *.ko *.order *.symvers *.mod.c
button_app.c文件:
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <poll.h>
int main(int argc, char **argv)
{
int fd;
int ret;
char *filename;
struct pollfd fds;
unsigned char keyVal;
filename = argv[1];
fd = open(filename, O_RDWR);
if(fd<0)
{
printf("can not open \n");
}
fds.fd = fd;
fds.events = POLLIN;
while(1)
{
ret = poll(&fds, 1, 5000); //linux系统再调用poll函数时候,如果没有发生需要的事件,那么进程进入休眠。
//如果在限定的时间内得到需要的事件,那么成功返回,如果没有则返回超时错误信息。
if(ret<0)
{
printf("time out \n");
}
else
{
read(fd, &keyVal, 1);
printf("keyVal: %x \n",keyVal);
}
}
close(fd);
return 0;
}
编译生成button_drv.ko和button_app文件,运行./button_app /dev/button_dev
Linux 驱动——Button驱动3(poll机制)的更多相关文章
- Linux 驱动——Button驱动7(Timer)消抖
button_drv.c驱动文件: #include <linux/module.h>#include <linux/kernel.h>#include <linux/f ...
- Linux 驱动——Button驱动5(atomic)原子量
button_drv.c驱动文件: #include <linux/module.h>#include <linux/kernel.h>#include <linux/f ...
- Linux 驱动——Button驱动4(fasync)异步通知
button_drv.c驱动文件: #include <linux/module.h>#include <linux/kernel.h>#include <linux/f ...
- Linux 驱动——Button驱动6(mutex、NBLOCK、O_NONBLOCK)互斥信号量、阻塞、非阻塞
button_drv.c驱动文件: #include <linux/module.h>#include <linux/kernel.h>#include <linux/f ...
- Linux 驱动——Button驱动2
button_drv.c驱动文件: #include <linux/module.h>#include <linux/kernel.h>#include <linux/f ...
- Linux 驱动——Button驱动1
button_drv.c驱动文件: #include <linux/module.h>#include <linux/kernel.h>#include <linux/i ...
- Linux驱动之poll机制的理解与简单使用
之前在Linux驱动之按键驱动编写(中断方式)中编写的驱动程序,如果没有按键按下.read函数是永远没有返回值的,现在想要做到即使没有按键按下,在一定时间之后也会有返回值.要做到这种功能,可以使用po ...
- poll机制实例参考
poll机制:为了减少CPU资源的占用率,在编写驱动函数中添加poll机制 select,poll,epoll都是IO多路复用的机制.I/O多路复用就通过一种机制,可以监视多个描述符,一旦某个描述符就 ...
- 嵌入式Linux驱动学习之路(十二)按键驱动-poll机制
实现的功能是在读取按键信息的时候,如果没有产生按键,则程序休眠在read函数中,利用poll机制,可以在没有退出的情况下让程序自动退出. 下面的程序就是在读取按键信息的时候,如果5000ms内没有按键 ...
随机推荐
- C语言缓冲区
定义 缓冲区是内存空间的一部分,用于缓冲输入或输出的数据.根据其对应的是输入设备还是输出设备,分为输入缓冲区和输出缓冲区. 类型 缓冲区分为三种类型:全缓冲.行缓冲和不带缓冲. 1.全缓冲 在这种情况 ...
- 多重if-else语句
C语言自学之多重if-else语句 Dome : 某游戏对不同等级的积分的玩家赋予不同的荣誉称号,其对应关系如下: 积分>=10000分为钻石玩家 积分>=5000并且<10000为 ...
- C# deep copy List
https://stackoverflow.com/questions/14007405/how-create-a-new-deep-copy-clone-of-a-listt 1. copy lis ...
- 搭建SDN网络——mininet
好久没来写不博客,感觉自己不写写博客,就完全想不起当天干了什么.前几天参加了小米的面试,整个人被打击的不行,还是要好好学习,不然工作都找不到. 这学期选了SDN的课,只有六周,学完有点懵,这这这等于什 ...
- 【调试基础】Part 1 寄存器
01 寄存器体系 02 16/32/64位寄存器
- INSERT INTO SELECT
今天遇到一个问题在写把查询的数据添加到一个表里时出现问题 sql INSERT INTO db1_name(field1,field2) VALUES SELECT field1,field2 FR ...
- 用Tensorflow搭建神经网络的一般步骤
用Tensorflow搭建神经网络的一般步骤如下: ① 导入模块 ② 创建模型变量和占位符 ③ 建立模型 ④ 定义loss函数 ⑤ 定义优化器(optimizer), 使 loss 达到最小 ⑥ 引入 ...
- python基础数据类型练习2
1,写代码,有如下列表,按照要求实现每一个功能li = ['alex', 'wusir', 'eric', 'rain', 'alex'] 计算列表的长度并输出print(len(li))答:结果为5 ...
- 命令提示符出现-bash-4.1$如何解决
故障原因:用户模式/etc/skel内容丢失 解决方法 复制etc/skel里内容到当前用户 [root@web01 ~]# su - test -bash-4.1$ cp /etc/skel/.ba ...
- HYPERSPACE
Windows中,不管是应用程序还是内核程序,都不能直接访问物理内存,所有非IO指令都只能访问虚拟内存地址,如Mov eax, DWORD PTR[虚拟地址]形式,但是,有时候,我们明明已经知道了某个 ...