【Linux 驱动】第六章 高级字符驱动程序操作----poll,select,epoll

时间:2022-06-29 03:40:15
一,poll
        允许进程决定是否可以对一个或者多个打开的文件做非阻塞的读取或者写入(但是请注意select自身会阻塞进程知道某个描述符满足条件或者超时),常常用于那些要使用多个输入或者输出流而又不会阻塞于其中任何一个流的应用程序中,比如telnet程序,需要2个输出,2个输入流而又不希望阻塞。

       unsigned int (*poll)(struct  file *filp,poll_table  *wait);//把当前的文件指针挂到设备内部定义的等待   队列中。这里的参数table可以不考虑,是在select函数实现过程中的一个内部变量

该驱动方法一般分为两步:
          1) 在一个或者多个可指示poll状态变化的等待队列上调用poll_wait.如果当时没有文件描述符可来执行I/O,则内核将使进程在传递到该系统调用的所有文件描述符对应的等待队列上等待。
         void poll_wait(struct file *,wait_queue_head_t *,poll_table *);
         2) 返回一个用来描述操作是否可以立即无阻塞执行的位掩码. 
              可读取返回POLLIN | POLLRDNORM;
              可写入返回POLLOUT | POLLWRNORM;
使用非阻塞和阻塞操作的组合以及select方法可以有效地查询设备,但是对于某些情况却不是很适用,此时我们需要使用异步通知的方法。

二,select     //经常用于socket编程

        原型: int    select(int   maxfdp,fd_set   *readfds,fd_set   *writefds,fd_set   *errorfds,struct timeval   *timeout);
         
         1)struct   fd_set:可以理解为一个集合,这个集合中存放的是文件描述符(filedescriptor),即文件句柄,这可以是我们所说的普通意义的文件,当然Unix下任何设备、管道、FIFO等都是文件形式,全部包括在内,所以毫无疑问一个socket就是一个文件,socket句柄就是一个文件描述符。        fd_set集合可以通过一些宏由人为来操作,比如               FD_ZERO(fd_set *);//清空集合               FD_SET(int ,fd_set *);//将一个给定的文件描述符加入集合之中               FD_CLR(int ,fd_set*);//将一个给定的文件描述符从集合中删除               FD_ISSET(int ,fd_set* );//检查集合中指定的文件描述符是否可以读写   2)struct timeval:是一个大家常用的结构,用来代表时间值,有两个成员,一个是秒数,另一个是微妙数。   具体解释select的参数:           1>int maxfdp是一个整数值,是指集合中所有文件描述符的范围,即所有文件描述符的最大值加1,不能错          
                2>fd_set  *readfds是指向fd_set结构的指针,这个集合中应该包括文件描述符,我们是要监视这些文件描述符的读变化的,即我们关心是否可以从这些文件中读取数据了,如果这个集合中有一个文件可读,select就会返回一个大于0的值,表示有文件可读,如果没有可读的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。可以传入NULL值,表示不关心任何文件的读变化。        
              3>fd_set  *writefds是指向fd_set结构的指针,这个集合中应该包括文件描述符,我们是要监视这些文件描述符的写变化的,即我们关心是否可以向这些文件中写入数据了,如果这个集合中有一个文件可写,select就会返回一个大于0的值,表示有文件可写,如果没有可写的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。可以传入NULL值,表示不关心任何文件的写变化。        
             4>fd_set *errorfds同上面两个参数的意图,用来监视文件错误异常。       
             5>struct   timeval   *timeout是select的超时时间,这个参数至关重要,它可以使select处于三种状态,第一,若将NULL以形参传入,即不传入时间结构,就是将select置于阻塞状态,一定等到监视文件描述符集合中某个文件描述符发生变化为止;第二,若将时间值设为0秒0毫秒,就变成一个纯粹的非阻塞函数,不管文件描述符是否有变化,都立刻返回继续执行,文件无变化返回0,有变化返回一个正值;第三,timeout的值大于0,这就是等待的超时时间,即select在timeout时间内阻塞,超时时间之内有事件到来就返回了,否则在超时后不管怎样一定返回,返回值同上述。   
        返回值:          负值:select错误 正值:某些文件可读写或出错 0:等待超时,没有可读写或错误的文件   在有了select后可以写出像样的网络程序来!举个简单的例子,就是从网络上接受数据写入一个文件中。   例子:   
int main()
{
  int sock; int fd;
  fd_set fds;
  struct timeval timeout={3,0}; //select等待3秒,3秒轮询,要非阻塞就置0
  char buffer[256]={0}; //256字节的接收缓冲区
  /* 假定已经建立UDP连接,具体过程不写,简单,当然TCP也同理,主机ip和port都已经给定,要写的文件已经打开
  sock=socket(...);
  bind(...);
  fp=fopen(...); */
  while(1)
  {
     FD_ZERO(&fds); //每次循环都要清空集合,否则不能检测描述符变化
     FD_SET(sock,&fds); //添加描述符
     FD_SET(fp,&fds); //同上
     maxfdp=sock>fp?sock+1:fp+1; //描述符最大值加1
     switch(select(maxfdp,&fds,&fds,NULL,&timeout)) //select使用
     {
         case -1: exit(-1);break; //select错误,退出程序
         case 0:break; //再次轮询
         default:
         if(FD_ISSET(sock,&fds)) //测试sock是否可读,即是否网络上有数据
         {
            recvfrom(sock,buffer,256,.....);//接受网络数据if(FD_ISSET(fp,&fds)) //测试文件是否可写
            fwrite(fp,buffer...);//写入文件
            buffer清空;
          }
      }
  }
}

select()函数与Linux驱动程序的关系

  当用户调用select系统调用时,select系统调用会先调用poll_initwait(&table);,然后调用驱动程序中 struct file_operations下的fop->poll函数,在这个函数里应该调用poll_wait(),将current加到某个等待队列(这里调用poll_wait()),并检查是否有效,如果无效就调用schedule_timeout();去睡眠。事件发生后,schedule_timeout()回来,,调用fop->poll();,检查到可以运行,就调用poll_freewait(&table);从而完成select系统调用。重要的是fop->poll()里面要检查是否就绪,如果是,要返回相应标志。

三,epoll
       

         epoll是Linux内核为处理大批量句柄而作了改进的poll,是Linux下多路复用IO接口select/poll的增强版本,它能显著减少程序在大量并发连接中只有少量活跃的情况下的系统CPU利用率。

        epoll的接口非常简单,一共就三个函数:
         1) int    epoll_create(int size);
               创建一个epoll的句柄,size用来告诉内核这个监听的数目一共有多大。这个参数不同于select()中的第一个参数,给出最大监听的fd+1的值。需要注意的是,当创建好epoll句柄后,它就是会占用一个fd值,在linux下如果查看/proc/进程id/fd/,是能够看到这个fd的,所以在使用完epoll后,必须调用close()关闭,否则可能导致fd被耗尽。

         2) int epoll_ctl(int  epfd, int  op, int  fd, struct  epoll_event  *event);
epoll的事件注册函数,它不同与select()是在监听事件时告诉内核要监听什么类型的事件,而是在这里先注册要监听的事件类型。

              epfd:  是epoll_create()的返回值

              op:表示动作,用三个宏来表示:

EPOLL_CTL_ADD:注册新的fd到epfd中;

                                                                           EPOLL_CTL_MOD:修改已经注册的fd的监听事件;
                                                                           EPOLL_CTL_DEL:从epfd中删除一个fd;
              fd:是需要监听的fd

              event:告诉内核需要监听什么事,struct epoll_event结构如下:

 struct epoll_event {
__uint32_t events; /* Epoll events */
epoll_data_t data; /* User data variable */
};

  events可以是以下几个宏的集合:
       EPOLLIN :表示对应的文件描述符可以读(包括对端SOCKET正常关闭);
       EPOLLOUT:表示对应的文件描述符可以写;
       EPOLLPRI:表示对应的文件描述符有紧急的数据可读(这里应该表示有带外数据到来);
       EPOLLERR:表示对应的文件描述符发生错误;
       EPOLLHUP:表示对应的文件描述符被挂断;
       EPOLLET: 将EPOLL设为边缘触发(Edge Triggered)模式,这是相对于水平触发(Level Triggered)来说的。
       EPOLLONESHOT:只监听一次事件,当监听完这次事件之后,如果还需要继续监听这个socket的话,需要再次把这个socket加入到EPOLL队列里

          3) int  epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);
               等待事件的产生,类似于select()调用。参数events用来从内核得到事件的集合,maxevents告之内核这个events有多大,这个maxevents的值不能大于创建epoll_create()时的size,参数timeout是超时时间(毫秒,0会立即返回,-1将不确定,也有说法说是永久阻塞)。该函数返回需要处理的事件数目,如返回0表示已超时。