本节继续在上一节中断按键程序里改进,添加poll机制.

那么我们为什么还需要poll机制呢。之前的测试程序是这样:

  1. while ()
  2. {
  3. read(fd, &key_val, );
  4. printf("key_val = 0x%x\n", key_val);
  5. }

在没有poll机制的情况下,大部分时间程序都处在read中休眠的那个位置。如果我们不想让程序停在这个位置,而是希望当有按键按下时,我们再去read,因此我们编写poll函数,测试程序调用poll函数根据返回值,来决定是否执行read函数。

poll机制作用:相当于定时器,设置一定时间使进程等待资源,如果时间到了中断还处于睡眠状态(等待队列),poll机制就会唤醒中断,获取一次资源

1.poll机制内核框架

如下图所示,在用户层上,使用poll或select函数时,和open、read那些函数一样,也要进入内核sys_poll函数里,接下来我们分析sys_poll函数来了解poll机制(位于/fs/select.c)

1.1 sys_poll代码如下:

  1. asmlinkage long sys_poll(struct pollfd __user *ufds, unsigned int nfds,long timeout_msecs)
  2. {
  3. if (timeout_msecs > ) //参数timeout>0
  4.     {
  5.    timeout_jiffies = msecs_to_jiffies(timeout_msecs); //通过频率来计算timeout时间需要多少计数值
  6.     }
  7.     else
  8.     {
  9. timeout_jiffies = timeout_msecs; //如果timeout时间为0,直接赋值
  10. }
  11. return do_sys_poll(ufds, nfds, &timeout_jiffies); //调用do_sys_poll。
  12. }

1.2 然后进入do_sys_poll(位于fs/select.c):

  1. int do_sys_poll(struct pollfd __user *ufds, unsigned int nfds, s64 *timeout)
  2. {
  3.   ... ...
  4.   /*初始化一个poll_wqueues变量table*/
  5.   poll_initwait(&table);
  6.   ... ...
  7.   fdcount = do_poll(nfds, head, &table, timeout);
  8.   ... ...
  9. }

1.3进入poll_initwait函数,发现主要实现以下一句,后面会分析这里:

  1. table ->pt-> qproc=__pollwait; //__pollwait将在驱动的poll函数里的poll_wait函数用到

1.4然后进入do_poll函数, (位于fs/select.c):

  1. static int do_poll(unsigned int nfds, struct poll_list *list, struct poll_wqueues *wait, s64 *timeout)
  2. {
  3.   ……
  4. for (;;)
  5.    {
  6.     ……
  7.     set_current_state(TASK_INTERRUPTIBLE); //设置为等待队列状态
  8.     ......
  9.    for (; pfd != pfd_end; pfd++) { //for循环运行多个poll机制
  10.       /*将pfd和pt参数代入我们驱动程序里注册的poll函数*/
  11. if (do_pollfd(pfd, pt)) //若返回非0,count++,后面并退出
  12.               { count++;
  13. pt = NULL; } }
  14.  
  15.     ……
  16.  
  17.     /*count非0(.poll函数返回非0),timeout超时计数到0,有信号在等待*/
  18.  
  19.   if (count || !*timeout || signal_pending(current))
  20.    break;
  21.     ……
  22.   
  23.     /*进入休眠状态,只有当timeout超时计数到0,或者被中断唤醒才退出,*/
  24.      __timeout = schedule_timeout(__timeout);
  25.  
  26.     ……
  27.  
  28.    }
  29.  
  30. __set_current_state(TASK_RUNNING); //开始运行
  31. return count;
  32.  
  33. }

1.4.1上面do_pollfd函数到底是怎么将pfd和pt参数代入的?代码如下(位于fs/select.c):

  1. static inline unsigned int do_pollfd(struct pollfd *pollfd, poll_table *pwait)
  2. {
  3.       ……
  4. if (file->f_op && file->f_op->poll)
  5. mask = file->f_op->poll(file, pwait);
  6.       ……
  7.  
  8. return mask;
  9. }

上面file->f_op 就是我们驱动里的file_oprations结构体,如下图所示:

所以do_pollfd(pfd, pt)就执行了我们驱动程序里的.poll(pfd, pt)函数(第2小节开始分析.poll函数)

1.4.2当poll进入休眠状态后,又是谁来唤醒它?这就要分析我们的驱动程序.poll函数(第2小节开始分析.poll函数)

2写驱动程序.poll函数,并分析.poll函数:

在上一节驱动程序里添加以下代码:

  1. #include <linux/poll.h> //添加头文件

  2. /* .poll驱动函数: third_poll */
  3. static unsigned int third_poll(struct file *fp, poll_table * wait) //fp:文件 wait:
  4. {
  5. unsigned int mask =;
  6. poll_wait(fp, &button_wait, wait);
  7. if(even_press) //中断事件标志, 1:退出休眠状态 0:进入休眠状态
  8. mask |= POLLIN | POLLRDNORM ;
  9. return mask; //当超时,就返给应用层为0 ,被唤醒了就返回POLLIN | POLLRDNORM ;
  10.  
  11. }
  12.  
  13. static struct file_operations third_drv_fops={
  14. .owner = THIS_MODULE,
  15. .open = third_drv_open,
  16. .read = third_drv_read,
  17.    .release=third_drv_class,
  18.    .poll = third_poll, //创建.poll函数
  19. };

2.1 在我们1.4小节do_poll函数有一段以下代码:

  1. if (do_pollfd(pfd, pt)) //若返回非0,count++,后面并退出
  2. {
         count++;
  3. pt = NULL;
  4. }

且在1.4.1分析出: do_pollfd(pfd, pt)就是指向的驱动程序third_poll()函数,

所以当我们有按键按下时, 驱动函数third_poll()就会返回mask非0值,然后在内核函数do_poll里的count就++,poll机制并退出睡眠.

2.2分析在内核中poll机制如何被驱动里的中断唤醒的

在驱动函数third_poll()里有以下一句:

  1. poll_wait(fp, &button_wait, wait);

如上图所示,代入参数,poll_wait()就是执行了: p->qproc(filp, button_wait, p);

刚好对应了我们1.3小节的:

  1. table ->pt-> qproc=__pollwait;

所以poll_wait()函数就是调用了: __pollwait(filp, button_wait, p);

然后我们来分析__pollwait函数,pollwait的代码如下:

  1. static void __pollwait(struct file *filp, wait_queue_head_t *wait_address,poll_table *p)
  2. {
  3.   ... ...
  4.    //把current进程挂载到&entry->wait下
  5.    init_waitqueue_entry(&entry->wait, current);
  6.  
  7.    //再&entry->wait把添加到到button_wait中断下
  8.    add_wait_queue(wait_address, &entry->wait);
  9.  
  10. }

它是将poll进程添加到了button_wait中断队列里,这样,一有按键按下时,在中断服务函数里就会唤醒button_wait中断,同样也会唤醒poll机制,使poll机制重新进程休眠计数

2.3 驱动程序.poll函数返回值介绍

当中断休眠状态时,返回mask为0

当运行时返回:mask |= POLLIN | POLLRDNORM

其中参数意义如下:

常量

说明

POLLIN

普通或优先级带数据可读

POLLRDNORM

normal普通数据可读

POLLRDBAND

优先级带数据可读

POLLPRI

Priority高优先级数据可读

POLLOUT

普通数据可写

POLLWRNORM

normal普通数据可写

POLLWRBAND

band优先级带数据可写

POLLERR

发生错误

POLLHUP

发生挂起

POLLNVAL

描述字不是一个打开的文件

所以POLLIN | POLLRDNORM:普通数据可读|优先级带数据可读

mask就返回到应用层poll函数,

3.改进测试程序third_poll_text.c(添加poll函数)

在linux中可以通过man poll 来查看poll函数如何使用

poll函数原型如下(#include <poll.h>):

  1. int poll(struct pollfd *fds, nfds_t nfds, int timeout);

参数介绍:

1) *fds:是一个poll描述符结构体数组(可以处理多个poll),结构体pollfd如下:

  1. struct pollfd {
  2. int fd; /* file descriptor 文件描述符*/
  3. short events; /* requested events 请求的事件*/
  4. short revents; /* returned events 返回的事件(函数返回值)*/
  5. };

其中events和revents值参数如下:

常量

说明

POLLIN

普通或优先级带数据可读

POLLRDNORM

normal普通数据可读

POLLRDBAND

优先级带数据可读

POLLPRI

Priority高优先级数据可读

POLLOUT

普通数据可写

POLLWRNORM

normal普通数据可写

POLLWRBAND

band优先级带数据可写

POLLERR

发生错误

POLLHUP

发生挂起

POLLNVAL

描述字不是一个打开的文件

2) nfds:表示多少个poll,如果1个,就填入1

3) timeout:定时多少ms

返回值介绍:

返回值为0:表示超时或者fd文件描述符无法打开

返回值为 -1:表示错误

返回值为>0时 :就是以下几个常量

常量

说明

POLLIN

普通或优先级带数据可读

POLLRDNORM

normal普通数据可读

POLLRDBAND

优先级带数据可读

POLLPRI

Priority高优先级数据可读

POLLOUT

普通数据可写

POLLWRNORM

normal普通数据可写

POLLWRBAND

band优先级带数据可写

POLLERR

发生错误

POLLHUP

发生挂起

POLLNVAL

描述字不是一个打开的文件

最终改进的测试代码如下:

  1. #include <sys/types.h>
  2. #include <sys/stat.h>
  3. #include <fcntl.h>
  4. #include <stdio.h>
  5. #include <string.h>
  6. #include <poll.h> //添加poll头文件
  7.  
  8. /*useg: thirdtext */
  9. int main(int argc,char **argv)
  10. {
  11. int fd,ret;
  12. unsigned int val=;
  13. struct pollfd fds; //定义poll文件描述结构体
  14. fd=open("/dev/buttons",O_RDWR);
  15. if(fd<)
  16. {printf("can't open!!!\n");
  17. return -;}
  18.  
  19. fds.fd=fd;
  20. fds.events= POLLIN; //请求类型是 普通或优先级带数据可读
  21.  
  22. while()
  23. {
  24.  
  25. ret=poll(&fds,,) ; //一个poll, 定时5000ms,进入休眠状态
  26. if(ret==) //超时
  27. {
  28. printf("time out \r\n");
  29. }
  30. else if(ret>) //poll机制被唤醒,表示有数据可读
  31. {
  32. read(fd,&val,); //读取一个值
  33. printf("key_val=0X%x\r\n",val);
  34. }
  35. }
  36. return ;
  37. }

 效果如下:

若5S没有数据,则打印time out

下节开始学习——使用异步通知来通知信号

8.中断按键驱动程序之poll机制的更多相关文章

  1. 8.中断按键驱动程序之poll机制(详解)

    本节继续在上一节中断按键程序里改进,添加poll机制. 那么我们为什么还需要poll机制呢.之前的测试程序是这样: ) { read(fd, &key_val, ); printf(" ...

  2. 韦东山驱动视频笔记——3.字符设备驱动程序之poll机制

    linux内核版本:linux-2.6.30.4 目的:我们在中断方式的按键应用程序中,如果没有按键按下,read就会永远在那等待,所以如果在这个程序里还想做其他事就不可能了.因此我们这次改进它,让它 ...

  3. 字符驱动程序之——poll机制

    关于这个韦老师给了一个简单的参考文档: poll机制分析 韦东山 2009.12.10 所有的系统调用,基本都可以在它的名字前加上“sys_”前缀,这就是它在内核中对应的函数.比如系统调用open.r ...

  4. 字符设备驱动程序之poll机制(韦大仙)

    明确为什么要引用poll机制? while(1) { read(fd,&key_val,1);//如果没有按键按下,它会一直在等待.现在想做这么一件事情:如果5s后,没有按键按下,它就会返回. ...

  5. 3.字符设备驱动------Poll机制

    1.poll情景描述 以之前的按键驱动为例进行说明,用阻塞的方式打开按键驱动文件/dev/buttons,应用程序使用read()函数来读取按键的键值. ) { read(fd, &key_v ...

  6. Linux学习 :中断处理机制 & poll机制

    中断是指在CPU正常运行期间,由于内外部事件或由程序预先安排的事件引起的CPU暂时停止正在运行的程序,转而为该内部或外部事件或预先安排的事件服务 的程序中去,服务完毕后再返回去继续运行被暂时中断的程序 ...

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

    // 在Linux下的中断方式读取按键驱动程序 //包含外部中断 休眠 加入poll机制 // 采用异步通知的方式 // 驱动程序发 ---> app接收 (通过kill_fasync()发送) ...

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

    文章对应视频的第12课,第5.6.7.8节. 在这之前还有查询方式的驱动编写,中断方式的驱动编写,这篇文章中暂时没有这些类容.但这篇文章是以这些为基础写的,前面的内容有空补上. 按键驱动——按下按键, ...

  9. 字符设备驱动(六)按键poll机制

    title: 字符设备驱动(六)按键poll机制 tags: linux date: 2018-11-23 18:57:40 toc: true --- 字符设备驱动(六)按键poll机制 引入 在字 ...

随机推荐

  1. oracle ORA-00604/ORA-01653

    问题描述: ORA-00604: error occurred at recursive SQL level 1ORA-01653: unable to extend table SYS.AUD$ b ...

  2. Redis-rdb持久化

    Redis实现快照的过程 redis调用fork,现在有了子进程和父进程 父进程继续处理client请求,子进程负责将内存内容写入到临时文. 由于os的写时复制机制(copy on write)父子进 ...

  3. 超超超简单的bfs——POJ-3278

    Catch That Cow Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 89836   Accepted: 28175 ...

  4. My first_leetcode_Rever Ingeter 数字翻转java实现(办法集合)

    7. Reverse Integer Reverse digits of an integer. Example1: x = 123, return 321  Example2: x = -123, ...

  5. Java入门——(2)面对对象(上)

      关键词:面对对象.类..构造方法.this.static.内部类   一.面对对象的概念:把解决的问题安装一定规则划分为多个独立的对象,然后通过调用对象的方法来解决问题.其特点可概括为封装性.继承 ...

  6. python学习之数字

    数字python中的数字比较典型,典型在可以表示很小的数,也可以表示很大的数,c语言的数字类型如果表示很大的数的话,可能会报溢出错误,但是python不会,python 数字类型的完整工具包括整数和浮 ...

  7. webpack打包体积优化

    优化: 1:外部引入模块(cdn)     如 jquery,zepto,d3, bootstrap这些固定的lib 使用cdn直接引用就可以,没有必要打包到build,有效利用302. 2:图标优化 ...

  8. Bug列表

    1.Space is not allowed after parameter prefix ': 这个问题主要原因是Hibernate不能识别SQL语句中的":="导致的网上有其他 ...

  9. C#操作EML邮件文件实例(含HTML格式化邮件正文和附件)

    使用QQ邮箱.163邮箱等导出的EML邮件,包含了邮件的发件人.主题.内容.附件等所有信息,该实例就如何解析这些信息,并在编辑后保存做个Demo. 如下图所示,EML文件是编码后的文本文件,可以使用正 ...

  10. python使用环境的设置

    virtualenv usage mkidr ~/pyenv cd ~/pyenv virtualenv pycaffe #it will setup a new python environtmen ...