9.按键之使用异步通知

之前学的应用层都是:

1)查询方式:一直读

2)中断方式.同样一直读,直到中断进程唤醒

3)poll机制:一直在poll函数中睡眠,一定时间读一次

以上3种,我们都是让应用程序主动去读,本节我们学习异步通知,它的作用就是当驱动层有数据时,主动告诉应用程序,然后应用程序再来读, 这样,应用程序就可以干其它的事情,不必一直读

比如:kill -9 pid ,其实就是通过发信号杀死进程,kill发数据9给指定id号进程

1.怎么来收信号?

通过signal函数来实现获取信号,先来看看以下例子:

头函数:

sighandler_t signal(int signum, sighandler_t handler); 

函数说明:让一个信号与与一个函数对应,每当接收到这个信号就会调用相应的函数。

头文件: #include <signal.h>

参数1: 指明了所要处理的信号类型

信号有以下几种:

  • SIGINT    键盘中断(如break、ctrl+c键被按下)
  • SIGUSR1  用户自定义信号1,kill的USR1(10)信号
  • SIGUSR2  用户自定义信号2, kill的USR2(12)信号

参数2:  信号产生后需要处理的方式,可以是个函数

代码如下:

#include <stdio.h>
#include <signal.h>

void my_signal_run(int signum)       //信号处理函数
{
   static int run_cnt=0;
   printf("signal = %d, %d count\r\n",signum,++count);
}

int main(int argc,char **argv)
{
   signal(SIGUSR1,my_signal_run);  //调用signal函数,让指定的信号SIGUSR1与处理函数my_signal_run对应。
  while(1)
  {    sleep(1000);     //去做其它事,睡眠1s
  }

   return 0;
}

然后运行后,使用kill -10 802,可以看到产生单信号USR1(10)时就会调用my_signal_run()打印数据。

# kill -10 802

# signal = 10, 1 count

# kill -10 802

# signal = 10, 2 count

2. 来实现异步通知

要求:

  • 一、应用程序要实现有:注册信号处理函数,使用signal函数
  • 二、谁来发?驱动来发
  • 三、发给谁?驱动发给应用程序,但应用程序必须告诉驱动PID,
  • 四、怎么发?驱动程序调用kill_fasync函数

3先来写驱动程序,我们在之前的中断程序上修改 

3.1定义 异步信号结构体 变量:

static struct fasync_struct * button_async;

3.2在file_operations结构体添加成员.fasync函数,并写函数

static struct file_operations third_drv_fops={
         .owner = THIS_MODULE,
         .open = fourth_drv_open,
         .read = fourth _drv_read,
       .release= fourth _drv_class,
      .poll = fourth _poll,
      .fasync = fourth_fasync             //添加初始化异步信号函数
};

static int fourth_fasync (int fd, struct file *file, int on)
{
  return fasync_helper(fd, file, on, & button_async); //初始化button_async结构体,就能使用kill_fasync()了
}

成员.fasync函数又是什么情况下使用?

是被应用程序调用,在下面第4小节会见到。

3.3在buttons_irq中断服务函数里发送信号:

kill_fasync(&button_async, SIGIO, POLL_IN);

  //当有中断时,就发送SIGIO信号给应用层,应用层就会触发与SIGIO信号对应的函数

3.4 驱动程序代码如下:

#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/irq.h>
#include <asm/arch/regs-gpio.h>
#include <asm/hardware.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#include <linux/poll.h>                  

static struct class *fourthdrv_class;
static struct class_device   *fourthdrv_class_devs; 

/*    声明等待队列类型中断 button_wait      */
static DECLARE_WAIT_QUEUE_HEAD(button_wait);

/*    异步信号结构体变量    */
static struct fasync_struct * button_async;

/*
  * 定义中断事件标志
  * 0:进入等待队列        1:退出等待队列
  */
static int even_press=0;                          

/*
  * 定义全局变量key_val,保存key状态
  */
static int key_val=0;                          

/*
  *引脚描述结构体
  */
 struct pin_desc{

   unsigned int  pin;

   unsigned int  pin_status;

};

/*
  *key初始状态(没有按下): 0x01,0x02,0x03,0x04
  *key状态(按下):             0x81,0x82,0x83,0x84
  */

         struct pin_desc  pins_desc[4]={
                   {S3C2410_GPF0,0x01 },
                   {S3C2410_GPF2, 0x02 },
                   {S3C2410_GPG3, 0x03 },
                   {S3C2410_GPG11,0x04},

} ;

int  fourth_drv_class(struct inode *inode, struct file  *file)  //卸载中断
{
  free_irq(IRQ_EINT0,&pins_desc[0]);
  free_irq(IRQ_EINT2,&pins_desc[1]);
  free_irq(IRQ_EINT11,&pins_desc[2]);
  free_irq(IRQ_EINT19,&pins_desc[3]);
  return 0;
}

/*
  *   确定是上升沿还是下降沿
  */
static irqreturn_t  buttons_irq (int irq, void *dev_id)       //中断服务函数
{
     struct pin_desc *pindesc=(struct pin_desc *)dev_id;     //获取引脚描述结构体
      unsigned int  pin_val=0;
      pin_val=s3c2410_gpio_getpin(pindesc->pin);
        if(pin_val)
        {
                   /*没有按下 (下降沿),清除0x80*/
                   key_val=pindesc->pin_status&0xef;
         }
         else
         {
                   /*按下(上升沿),加上0x80*/
                   key_val=pindesc->pin_status|0x80;
         }

             even_press=1;                                        //退出等待队列
            wake_up_interruptible(&button_wait);                  //唤醒 中断
            kill_fasync(&button_async, SIGIO, POLL_IN);        //发送SIGIO信号给应用层                        

         return IRQ_RETVAL(IRQ_HANDLED);
}

static int fourth_drv_open(struct inode *inode, struct file  *file)
{
   request_irq(IRQ_EINT0,buttons_irq,IRQT_BOTHEDGE,"S1",&pins_desc[0]);
   request_irq(IRQ_EINT2, buttons_irq,IRQT_BOTHEDGE, "S2", &pins_desc[1]);
   request_irq(IRQ_EINT11, buttons_irq,IRQT_BOTHEDGE, "S3", &pins_desc[2]);
   request_irq(IRQ_EINT19, buttons_irq,IRQT_BOTHEDGE, "S4", &pins_desc[3]);
   return 0;
}

static int fourth_drv_read(struct file *file, const char __user *buf, size_t count, loff_t * ppos)
{
         /*将中断 进入等待队列(休眠状态)*/
         wait_event_interruptible(button_wait, even_press);           

        /*有按键按下,退出等待队列,上传key_val 给用户层*/
         if(copy_to_user(buf,&key_val,sizeof(key_val)))
          return EFAULT;

         even_press=0;
  return 0;
}

static unsigned fourth_poll(struct file *file, poll_table *wait)
 {
                   unsigned int mask = 0;
                   poll_wait(file, &button_wait, wait); // 不会立即休眠
                   if (even_press)
                            mask |= POLLIN | POLLRDNORM;     

                   return mask;
         }

static int fourth_fasync (int fd, struct file *file, int on)
{
         return fasync_helper(fd, file, on, & button_async); //初始化button_async结构体,就能使用kill_fasync()了
}

static struct file_operations fourth_drv_fops={
         .owner = THIS_MODULE,
         .open = fourth_drv_open,
         .read = fourth_drv_read,
        .release=fourth_drv_class,    //里面添加free_irq函数,来释放中断服务函数
        .poll = fourth_poll,
        .fasync= fourth_fasync,     //初始化异步信号函数
};

volatile int fourth_major;
static int fourth_drv_init(void)
{
   fourth_major=register_chrdev(0,"fourth_drv",&fourth_drv_fops);  //创建驱动
   fourthdrv_class=class_create(THIS_MODULE,"fourth_dev");    //创建类名
   fourthdrv_class_devs=class_device_create(fourthdrv_class, NULL, MKDEV(fourth_major,0), NULL,"buttons");
 return 0;

}

static int fourth_drv_exit(void)
{
 unregister_chrdev(fourth_major,"fourth_drv");            //卸载驱动
class_device_unregister(fourthdrv_class_devs);         //卸载类设备
class_destroy(fourthdrv_class);                              //卸载类
return 0;
}

module_init(fourth_drv_init);
module_exit(fourth_drv_exit);
MODULE_LICENSE("GPL v2");
                    

4 写应用测试程序

步骤如下:

1) signal(SIGIO, my_signal_fun);  

调用signal函数,当接收到SIGIO信号就进入my_signal_fun函数,读取驱动层的数据

2) fcntl(fd,F_SETOWN,getpid());  

指定进程做为fd文件的”属主”,内核收到F_SETOWN命令,就会设置pid(驱动无需处理),这样fd驱动程序就知道发给哪个进程

3) oflags=fcntl(fd,F_GETFL);

获取fd的文件状态标志

4) fcntl(fd,F_SETFL, oflags| FASYNC );

添加FASYNC状态标志,会调用驱动中成员.fasync函数,执行fasync_helper()来初始化异步信号结构体

这4个步骤执行后,一旦有驱动层有SIGIO信号时,进程就会收到

应用层代码如下:

#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <string.h>
#include <poll.h>
#include <signal.h>
#include <unistd.h>
#include <fcntl.h>

int fd,ret;
void my_signal_fun(int signame)      //有信号来了
{
   read( fd, &ret, 1);              //读取驱动层数据
   printf("key_vale=0X%x\r\n",ret); 

}

/*useg:    fourthtext   */
int main(int argc,char **argv)
{
  int oflag;
  unsigned int val=0;
  fd=open("/dev/buttons",O_RDWR);
  if(fd<0)
        {printf("can‘t open!!!\n");
       return -1;}

   signal(SIGIO,my_signal_fun); //指定的信号SIGIO与处理函数my_signal_run对应
   fcntl( fd, F_SETOWN, getip());  //指定进程作为fd 的属主,发送pid给驱动
   oflag=fcntl( fd, F_GETFL);   //获取fd的文件标志状态
   fcntl( fd, F_SETFL, oflag|FASYNC);  //添加FASYNC状态标志,调用驱动层.fasync成员函数 

   while(1)
   {
         sleep(1000);                  //做其它的事情
   }
   return 0;

}

5 运行查看结果

下节开始学习——同步、互斥、阻塞

时间: 2024-10-05 05:31:39

9.按键之使用异步通知的相关文章

按键驱动程序(异步通知)

此驱动程序之前的按键驱动程序(中断方式)上加以优化.用到异步通知.对于内核来讲,既然用户想得到的是按键后的状态,那么自然不必时时都要read状态.当它检测到中断发生变主动通知用户,用户再来读.这样,用户空间.内核就可以着手干点其它的事情,而不必忙等按键按下或释放.那么就先从应用程序上面看. 怎么设置相关联到"异步通知"呢?flag = fcntl(fd, F_GETFL);fcntl(fd, F_SETFL, flag|FASYNC);这样的两句就可以将文件描述符fd与异步通知关联,到

字符设备驱动(七)按键异步通知

目录 按键驱动方式对比 进程间发信号 目标 如何让驱动通知应用 程序编写 驱动程序 应用程序 完整代码如下 测试 title: 字符设备驱动(七)按键异步通知 tags: linux date: 2018-11-24 16:39:47 toc: true --- 按键驱动方式对比 查询:耗资源 中断: 没有超时机制,当没有中断作为生产者,read函数一直休眠 poll机制,加入超时机制 上述三种都是app主动去获取按键,使用异步通知的形式可以使按键发生后,通知app去读取 进程间发信号 以前使用

按键驱动异步通知

在此以前,我们都是让应用程序主动去读按键的状态,有没有一种情况,当驱动程序有数据时,主动去告诉应用程序,告诉它,有数据了,你赶紧来读吧.这种情况在linux里的专业术语就叫异步通知. 在按键的例子中异步通知可以理解为:当按键按下时,驱动程序会提醒(即触发)应用程序(通过信号signal来实现). 举一个例子:进程之间发信号 原来我们常用  kill 这个命令 : kill       -9    pid   kill这个命令就是一个发信号 发送者  :   kill 接收者  :   pid 信

入门级的按键驱动——按键驱动笔记之poll机制-异步通知-同步互斥阻塞-定时器防抖

文章对应视频的第12课,第5.6.7.8节. 在这之前还有查询方式的驱动编写,中断方式的驱动编写,这篇文章中暂时没有这些类容.但这篇文章是以这些为基础写的,前面的内容有空补上. 按键驱动——按下按键,打印键值: 目录 概要 poll机制 异步通知 同步互斥阻塞 定时器防抖 概要: 查询方式: 12-3 缺点:占用CPU99%的资源.中断方式:12-4 缺点:调用read函数后如果没有按键按下,该函数永远不会结束,一直在等待按键按下. 优点:使用到了休眠机制,占用cpu资源极少.poll机制: 1

关于 exynos 4412 按键中断 异步通知

以下是驱动测试代码: //内核的驱动代码 #include <linux/init.h> #include <linux/module.h> //for module_init moudule_exit #include <linux/fs.h> //for MKDEV register_chrdev_region #include <linux/cdev.h> //字符设备头文件 #include <linux/device.h> #inclu

字符设备驱动程序之异步通知(韦大仙)

读取按键的方法: (1)查询的方式:极度耗费资源 (2)中断的方式:如果没有按键按下,read函数会一直的等待 (3)poll机制的引入:可以指定超时时间 上述三种方式有一个共同点:应用程序主动的去查询. 问题:有没有一种方式当有按键按下时,驱动程序通知应用程序去读取.这就是本节所说的异步通知,该方式用信号的方式来实现的. 进程间发信号,例如: kill  -9  PID kill这个程序是发送者,进程号为PID的进程为接受者,9就是发送的信号.接下来引入signal函数: signal 函数是

字符设备驱动程序之异步通知

异步通知: 驱动程序的所谓异步通知,就是说并不是应用程序来对驱动程序操作的,而是驱动程序查询到有事件发生或者有数据发生变化的时候通知应用程序.角色发生了变化,应用程序由主动改为被动执行. 比如按键驱动: 1.有不断进行查询引脚状态的,CPU资源消耗非常的打: 2.有中断操作的,发生按键事件后采取执行相关事件处理函数,需要应用程序不断执行read函数,使得不能去干其它事情: 3.poll机制,改善了中断方式操作,在应用程序上当没有事件发生时,会跳去read函数继续执行其它的任务,知道有事件发生才返

linux驱动的异步通知(kill_fasync,fasync)---- 驱动程序向应用程序发送信号

应用程序 [cpp] view plain copy #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <stdio.h> #include <poll.h> #include <signal.h> #include <sys/types.h> #include <unistd.h> #include <fcn

Linux之异步通知20160702

异步通知,主要说的是使用信号的方式,同时使用信号也是实现进程之间通信的一种方式. 多的不说,我们直接看代码: 首先应用程序的: #include <sys/types.h> #include <unistd.h> #include <fcntl.h> /* fifthdrvtest */ int fd; void my_signal_fun(int signum) { unsigned char key_val; read(fd, &key_val, 1); p