在Linux下的中断方式读取按键驱动程序

// 在Linux下的中断方式读取按键驱动程序

//包含外部中断 休眠 加入poll机制

// 采用异步通知的方式

// 驱动程序发 ---> app接收 (通过kill_fasync()发送)

// 为了使设备支持异步通知机制,驱动程序中涉及以下3项工作:

// 1. 支持F_SETOWN命令,能在这个控制命令处理中设置filp->f_owner为对应进程ID。

// 不过此项工作已由内核完成,设备驱动无须处理。

// 2. 支持F_SETFL命令的处理,每当FASYNC标志改变时,驱动程序中的fasync()函数将得以执行。

// 驱动中应该实现fasync()函数。

// 3. 在设备资源可获得时,调用kill_fasync()函数激发相应的信号

// 应用程序:

// fcntl(fd, F_SETOWN, getpid()); // 告诉内核,发给谁

// Oflags = fcntl(fd, F_GETFL);

// fcntl(fd, F_SETFL, Oflags | FASYNC); // 改变fasync标记,最终会调用到驱动的faync > fasync_helper:初始化/释放fasync_struct

// 外部中断测试程序 包含poll机制 进程之间异步通信 加入原子操作

// 原子操作:指的是在执行过程中不会被别的代码路径所中断的操作。

// 信号量的实现

// 阻塞 :是指在执行设备操作时若不能获得资源则挂起进程,直到满足可操作的条件后再进行操作。

// 被挂起的进程进入休眠状态,被从调度器的运行队列移走,直到等待的条件被满足。

// 非阻塞:进程在不能进行设备操作时并不挂起,它或者放弃,或者不停地查询,直至可以进行操作为止。

// 加入定时器消抖动功能

#include <linux/module.h>

#include <linux/kernel.h>

#include <linux/fs.h>

#include <linux/init.h>

#include <linux/delay.h>

#include <linux/irq.h>

#include <asm/uaccess.h>

#include <asm/irq.h>

#include <asm/io.h>

#include <asm/arch/regs-gpio.h>

#include <asm/hardware.h>

#include <linux/poll.h>

#define usingatomic (0) // 0使用信号量 1使用的是原子操作

//设备类

static struct class *Eint_class;

// 设备节点

static struct class_device *Eint_class_devs;

// 地址映射

volatile unsigned long *gpfcon;

volatile unsigned long *gpfdat;

volatile unsigned long *gpgcon;

volatile unsigned long *gpgdat;

// 全局变量 存放中断读出的键值

static unsigned int key_val;

//创建一个休眠队列

static DECLARE_WAIT_QUEUE_HEAD(button_waitq);

/* 中断事件标志, 中断服务程序将它置1,third_drv_read将它清0 */

static volatile int ev_press = 0;

//信号量初始化结构体

static struct fasync_struct *button_async_queue;

// 定时器结构体

struct timer_list buttons_timer;

// 存储外部中断号和键值结构体变量

static struct pin_desc *irq_pd;

//定义结构体 存放按键 pin 端口 key_val键值

struct pin_desc

{

unsigned int pin;

unsigned int key_val;

};

#if usingatomic

//定义原子变量v并初始化为1

atomic_t canopen = ATOMIC_INIT(1);

#else

//定义互斥锁 信号量

static DECLARE_MUTEX(button_lock);

#endif

//定义结构体数组 存放中断端口和键值

struct pin_desc pins_desc[4]={ {S3C2410_GPF0,0x01},

{S3C2410_GPF2,0x02},

{S3C2410_GPG3,0x03},

{S3C2410_GPG11,0x04}};

//中断服务程序

//读取键值

static irqreturn_t buttons_irq(int irq, void *ignored)

{

irq_pd = ( struct pin_desc *)ignored;

mod_timer(&buttons_timer, jiffies+HZ/100); //10ms 产生中断

// return IRQ_RETVAL(IRQ_HANDLED);

return IRQ_HANDLED;

}

//定时器中断函数

static void buttons_timer_function(unsigned long data)

{

struct pin_desc *pins_desc= irq_pd;

unsigned int pinval;

if(!pins_desc)

return;

pinval=s3c2410_gpio_getpin(pins_desc->pin); //读取IO的值

if(pinval)

{

key_val =0x80|pins_desc->key_val;

}

else

{

key_val =pins_desc->key_val;

}

ev_press = 1; /* 表示中断发生了 */

wake_up_interruptible(&button_waitq); /* 唤醒休眠的进程 */

kill_fasync (&button_async_queue, SIGIO, POLL_IN);//发送信号给app

}

//打开设备调用

//初始化IO端口 配置为输入模式

//GPF0-->S2 GPF2-->S3 GPG3-->S4 GPG11-->S5

static int Eint_drv_open(struct inode *inode, struct file *file)

{

// *gpfcon &=~((3<<2*0)|(3<<2*2));

// *gpgcon &=~((3<<3*2)|(3<<11*2));

#if usingatomic

if(!atomic_dec_and_test(&canopen))// 原子操作

{

atomic_inc(&canopen);//自加1

printk("this a user in the use of\n");

return -EBUSY;//返回忙

}

#else

if (file->f_flags & O_NONBLOCK)

{

//非阻塞 立马返回

if (down_trylock(&button_lock))

return -EBUSY;

}

else

{

down(&button_lock);

}

#endif

printk("Eint_drv_open successed!\n");

request_irq(IRQ_EINT0,buttons_irq, IRQT_BOTHEDGE, "s2", &pins_desc[0]);//EINT0边沿触发方式

request_irq(IRQ_EINT2,buttons_irq, IRQT_BOTHEDGE, "s3", &pins_desc[1]);//EINT2边沿触发方式

request_irq(IRQ_EINT11,buttons_irq, IRQT_BOTHEDGE, "s4", &pins_desc[2]);//EINT11边沿触发方式

request_irq(IRQ_EINT19,buttons_irq, IRQT_BOTHEDGE, "s5", &pins_desc[3]);//EINT19边沿触发方式

return 0;

}

//write时候调用

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

{

}

//read时候调用

//读取按键值

ssize_t Eint_drv_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)

{

if(size != 1)

return -EINVAL;

if (file->f_flags & O_NONBLOCK)

{

//非阻塞 立马返回

if (!ev_press)

return -EAGAIN;

}

else

{

//如果没有按键按下 则休眠

wait_event_interruptible(button_waitq,ev_press);

}

//如果有按键动作发生的话 则返回

copy_to_user(buf, &key_val, 1);

ev_press = 0;//清中断标志位

// printk("key_val = 0x%x\n", key_val);

}

//关闭驱动时候调用

static int Eint_drv_colse(struct inode *inode, struct file *file)

{

#if usingatomic

atomic_inc(&canopen);//自加1

#else

up(&button_lock);

#endif

free_irq(IRQ_EINT0, &pins_desc[0]);//EINT0释放中断

free_irq(IRQ_EINT2, &pins_desc[1]);//EINT2释放中断

free_irq(IRQ_EINT11, &pins_desc[2]);//EINT11释放中断

free_irq(IRQ_EINT19, &pins_desc[3]);//EINT19释放中断

printk("Eint_drv_colse successed!\n");

}

//poll时候调用

// 在规定时间内没有按下按键 就返回超时

//中断没有发生 就return 0,在do_sys_poll中就会让系统休眠,唤醒休眠是chedule_timeout(__timeou)超时

//中断发生 return POLLIN | POLLRDNORM,在do_sys_poll退出休眠,唤醒进程

static unsigned int Eint_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;

}

//在应用程序中使用fcnt() 时候调用

static int Eint_drvl_fasync (int fd, struct file *filp, int on)

{

printk("\ndrivec:signal_fasync successed !\n");

return fasync_helper (fd, filp, on, &button_async_queue);

}

//告诉内核

static struct file_operations Eint_drv_fops = {

.owner = THIS_MODULE, // 这是一个宏,推向编译模块时自动创建的__this_module变量

.open = Eint_drv_open,

.write = Eint_drv_write,

.read = Eint_drv_read,

.release= Eint_drv_colse,

.poll = Eint_drv_poll,

.fasync = Eint_drvl_fasync,

};

int major;//自动分配主设备号

//安装驱动的时候调用

//注册驱动 创建设备类 创建设备节点 创建虚拟地址 创建定时器任务

int Eint_drv_init(void)

{

// 创建一个定时器

init_timer(&buttons_timer);

buttons_timer.function = buttons_timer_function;

add_timer(&buttons_timer);

major=register_chrdev( 0, "key_drv",&Eint_drv_fops);//告诉内核 注册驱动

Eint_class = class_create(THIS_MODULE, "key_drv");//获取一个设备信息类

if (IS_ERR(Eint_class))

return PTR_ERR(Eint_class);

Eint_class_devs = class_device_create(Eint_class, NULL, MKDEV(major, 0), NULL, "buttons");

if (unlikely(IS_ERR(Eint_class_devs)))

return PTR_ERR(Eint_class_devs);

//转换虚拟地址

gpfcon = (volatile unsigned long *)ioremap(0x56000050, 16);

gpfdat = gpfcon+1;

gpgcon =(volatile unsigned long *)ioremap(0x56000060,16);

gpgdat =gpgcon+1;

printk("Eint_drv_init successed!\n");

return 0;

}

//卸载驱动程序的时候调用

//卸载驱动 删除设备 删除设备节点 删除地址映射

void Eint_drv_exit(void)

{

unregister_chrdev( major, "key_drv");//卸载驱动

class_device_unregister(Eint_class_devs);//删除设备

class_destroy(Eint_class);//删除设备节点

iounmap(gpgcon);//删除地址映射

iounmap(gpfcon);

printk("\nEint_drv_exit successed!\n");

}

module_init(Eint_drv_init);

module_exit(Eint_drv_exit);

MODULE_LICENSE("GPL");

时间: 2025-01-01 08:46:09

在Linux下的中断方式读取按键驱动程序的相关文章

linux驱动之中断方式获取键值

linux驱动之中断方式获取键值 ------------------------------------------------------------------------------------------------------------------------------------------------------ 回想在单片机下的中断处理 分辨是哪个中断 调用处理函数 清中断 --------------------------------------------------

linux下c/c++方式访问curl的帮助手册

自:http://blog.chinaunix.net/u1/47395/showart_1768832.html 有个业务需求需要通过curl 代理的方式来访问外网 百度了一把,测试可以正常使用.记录下来方便后续查找 example:   1. http://curl.haxx.se/libcurl/c/example.html  2. http://www.libcurl.org/book:  1. http://www.linuxdevcenter.com/pub/a/linux/2005

Linux下用memory方式访问PCIE空间

测试环境:Ubuntu 14.04LTS 在Windows下,我们 用RW everything很容易可以看到PCIE所有的config space,但是我们最近想在Linux下dump PCIE config space,首先我们尝试用IO read的方式, 也就是通常我们会用CF8和CFC的方式,但是很遗憾这种方式只能读出来256个字节,那么后面的0x100~0x1FF怎么去读,就是下面我们要解决的问题. 1,首先我们需要获取PCIE在memory中的映射地址,我们看ACPI的手册,会发现有

在Linux下以pptp方式拨入VPN网络

如果你需要在Linux中拨入虚拟网管(http://www.vpnlife.com)中,那就需要安装Linux下相应VPN的客户端,本文将介绍以pptp方式拨入虚拟网管的VPN的方法. 本操作均在文本终端下完成,不涉及图形操作,本文假设你对Linux命令有一定的了解,这里不对所出现的命令进行解释.以下操作均在root用户下操作完成,并假设你的Linux系统已经安装了编译环境.1.下载pptp客户端wget http://nchc.dl.sourceforge.net/sourceforge/pp

Linux 下三种方式设置环境变量

1.在Windows 系统下,很多软件安装都需要配置环境变量,比如 安装 jdk ,如果不配置环境变量,在非软件安装的目录下运行javac 命令,将会报告找不到文件,类似的错误. 2.那么什么是环境变量?简单说,就是指定一个目录,运行软件的时候,相关的程序将会按照该目录寻找相关文件. 设置变量对于一般人最实用的功能就是: 不用拷贝某些dll文件到系统目录中了,而path 这一系统变量就是系统搜索dll文件的一系列路径 在Linux系统下,如果你下载并安装应用程序,很有可能在键入它的名称的时候出现

一种Linux下共享中断的处理方法

前段时间调试一款芯片的时候,碰到一个奇怪的问题:只要在板卡上插入一个PS2键盘,启动内核时系统就可能会进入串口中断函数去执行,过一会系统就panic不往下继续执行.后来经过分析出现问题时的panic的堆栈,借助EJTAG工具,读到这个时候的串口的中断状态位,竟然发现串口并没有真正产生中断.那么,串口本身没有中断,内核怎么又会跑到串口的中断服务函数去执行呢? 我们知道Linux的中断可以分为I/O 中断 .时钟中断和处理器核间中断.其中I/O中断是Linux 系统响应外部IO事件的重要方式.尽管不

10种linux下磁盘快照方式恢复系统

导读 大家都知道windows系统有一个磁盘快照的功能,在windows2003中系统恢复开始依赖于一个叫做硬盘快照服务(Volume Snapshot Service)的服务,他能够自动创建系统快照--包括正在使用的文件--然后将这些文件转换为可恢复的节点文件,在之后的文件系统NTFS这个格式的分区具有系统恢复快照功能快照可以保存,这样在磁盘误操作后就可以完成恢复系统了.linux有没有磁盘快照呢?他的系统误操作怎么进行恢复呢?今天小编带您用10种方式玩转linux磁盘快照的恢复. Linux

linux下如何用php读取word

在实际的工作中遇到到要导入word格式的文件,经过努力,终于成功了. 在linux上用PHP读取WORD文档,其实是使用了 antiword程序把word文档转化为txt文档. 再使用php执行系统命令调用而已. 具体操作如下: 1.安装antiword 官方站:http://www.winfield.demon.nl/ 下载地:http://www.winfield.demon.nl/linux/antiword-0.37.tar.gz 下载完,解压,进入目录 使用命令 make &&

Linux下基于源代码方式安装MySQL 5.6

版权声明:本文为博主原创文章,欢迎扩散,扩散请务必注明出处. https://blog.csdn.net/robinson_0612/article/details/26485851 ??? MySQL为开源数据库,因此能够基于源代码实现安装. 基于源代码安装有很多其它的灵活性.也就是说我们能够针对自己的硬件平台选用合适的编译器来优化编译后的二进制代码.依据不同的软件平台环境调整相关的编译參数,选择自身须要选择不同的安装组件,设定须要的字符集等等一些能够依据特定应用场景所作的各种调整.本文描写叙