Select函数实现
int select(int nfds,fd_set *restrict readfds,fd_set *restrict writefds,fd_set *restrict errorfds,struct timeval *restrict timeout);
SYSCALL_DEFINE5(select, int, n,fd_set __user *, inp,fd_set __user *, outp,fd_set __user *, exp,struct timeval __user *, tvp){ret = core_sys_select(n, inp, outp, exp, to);ret = poll_select_copy_remaining(&end_time, tvp, 1, ret);return ret;}
core_sys_select 主要工作:
- 初始化读写还有异常的bitmap
- 调用
do_select实现核心的轮询工作。 - 把结果拷贝会用户空间
int core_sys_select(int n,fd_set __user *inp,fd_set __user *outp,fd_set __user *exp,struct timespec *end_time){fd_set_bits fds;// …if ((ret = get_fd_set(n, inp, fds.in)) ||(ret = get_fd_set(n, outp, fds.out)) ||(ret = get_fd_set(n, exp, fds.ex))) //*get_fd_set仅仅调用copy_from_user从用户空间拷贝了fd_set*/goto out;zero_fd_set(n, fds.res_in);zero_fd_set(n, fds.res_out);zero_fd_set(n, fds.res_ex);//发现do_select函数ret = do_select(n, &fds, end_time);/*把结果集,拷贝回用户空间*/- if (set_fd_set(n, inp, fds.res_in) ||
set_fd_set(n, outp, fds.res_out) ||set_fd_set(n, exp, fds.res_ex))ret = -EFAULT;}
int do_select(int n, fd_set_bits *fds, struct timespec *end_time){struct poll_wqueues table;poll_table *wait;poll_initwait(&table);//这个函数实现很关键,其内部的init_poll_funcptr初始化回调函数为__pollwait, 后面轮询会回调这个函数,然后通过这个函数把进程添加到对应的监听文件等待队列,当有事件到来时,就会唤醒这个进程。for (;;) {//一次大循环for (i = 0; i < n; ++rinp, ++routp, ++rexp) {// …struct fd f;f = fdget(i);if (f.file) {const struct file_operations *f_op; //每个设备拥有一个struct file_operations结构体f_op = f.file->f_op;mask = DEFAULT_POLLMASK;if (f_op->poll) { //轮询函数不为空,每当设备模块加载就自动会加载设备轮询函数,等于将轮回函数统一付给poll这个指针,以便调用wait_key_set(wait, in, out,bit, busy_flag);//检查集合// 对每个fd进行I/O事件检测 (*f_op->poll)返回当前设备fd的状态(可读可写)mask = (*f_op->poll)(f.file, wait);//将会调用poll_wait函数,检测文件设备的状态,并且将当前进程加入到设备等待队列中。并且返回掩码}fdput(f);}}// 退出循环体if (retval || timed_out || signal_pending(current))break;// 轮询一遍没有发现就绪。那就休眠if (!poll_schedule_timeout(&table, TASK_INTERRUPTIBLE,to, slack))timed_out = 1;}}
这个函数实现很关键,这里init_poll_funcptr初始化回调函数为__pollwait, 后面轮询会回调这个函数,然后通过这个函数把进程添加到对应的监听文件等待队列,当有事件到来时,就会唤醒这个进程。poll_initwait(&table);void poll_initwait(struct poll_wqueues *pwq){//这里p->_qproc实际就是__pollwait函数,因为p->qproc在init_poll_funcptr中被赋值为__pollwait函数指针init_poll_funcptr(&pwq->pt, __pollwait); //初始化函数指针,设置为__pollwaitpwq->error = 0;pwq->table = NULL;pwq->inline_index = 0;}static inline voidinit_poll_funcptr(poll_table *pt, poll_queue_proc qproc){pt->qproc = qproc;}
struct file {struct path f_path;//路径struct inode *f_inode; //inodeconst struct file_operations *f_op; //包含各种用于操作设备的函数指针- } __attribute__((aligned(4))); /* lest something weird decides that 2
struct file_operations {struct module *owner;loff_t (*llseek) (struct file *, loff_t, int);ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);// select()轮询设备fd的操作函数,对应一个file 跟poll_table_struct *unsigned int (*poll) (struct file *, struct poll_table_struct *); //驱动加载。一般就挂到这个地方轮询函数};
struct scull_pipe {wait_queue_head_t inq, outq; //可读可写队列};
static unsigned int scull_p_poll(struct file *filp, poll_table *wait){struct scull_pipe *dev = filp->private_data;unsigned int mask = 0;mutex_lock(&dev->mutex);poll_wait(filp, &dev->inq, wait);//pollwait函数包含了__pollwait.这函数就是把当前进程添加到设备队列中poll_wait(filp, &dev->outq, wait);//等待if (dev->rp != dev->wp)mask |= POLLIN | POLLRDNORM; //可读if (spacefree(dev))mask |= POLLOUT | POLLWRNORM; //可写mutex_unlock(&dev->mutex);return mask;//返回该设备的掩码,是否就绪可读可写}
static inline void poll_wait(struct file * filp, wait_queue_head_t * wait_address, poll_table *p){if (p && p->_qproc && wait_address)p->_qproc(filp, wait_address, p);//这里p->_qproc实际就是__pollwait函数,因为p->qproc在do_select中被赋值为__pollwait函数指针}
typedef void (*poll_queue_proc)(struct file *, wait_queue_head_t *, struct poll_table_struct *);
poll_table里的函数指针,是在do_select()初始化的。
int do_select(int n, fd_set_bits *fds, struct timespec *end_time){struct poll_wqueues table;poll_table *wait;poll_initwait(&table);//初始化}void poll_initwait(struct poll_wqueues *pwq){// 初始化poll_table里的函数指针init_poll_funcptr(&pwq->pt, __pollwait);}EXPORT_SYMBOL(poll_initwait);static inline void init_poll_funcptr(poll_table *pt, poll_queue_proc qproc){pt->_qproc = qproc;//将poll_table的函数指针设置为__pollwait完成初始化工作pt->_key = ~0UL; /* all events enabled */}
static void __pollwait(struct file *filp, wait_queue_head_t *wait_address,poll_table *p){// 把当前进程装到设备的等待队列add_wait_queue(wait_address, &entry->wait);}
static ssize_t scull_p_write(struct file *filp, const char __user *buf, size_t count,loff_t *f_pos){wake_up_interruptible(&dev->inq); //唤醒当前进程}
- select慢的原因
- 从上面看,在第一次所有监听都没有事件时,调用 select 都需要把进程挂到所有监听的文件描述符一次。
- 有事件到来时,不知道是哪些文件描述符有数据可以读写,需要把所有的文件描述符都轮询一遍才能知道。
- 通知事件到来给用户进程,需要把整个 bitmap 拷到用户空间,让用户空间去查询。
- select返回时,会将该进程从全部监听的fd的等待队列里移除掉,这样就需要select每次都要重新传入全部监听的fd,然后重现将本进程挂载到全部的监测fd的等待队列
Select函数实现的更多相关文章
- (十二)select()函数以及FD_ZERO、FD_SET、FD_CLR、FD_ISSET
select函数用于在非阻塞中,当一个套接字或一组套接字有信号时通知你,系统提供select函数来实现多路复用输入/输出模型,原型:int select(int maxfd,fd_set *rdset ...
- select 函数1
Select在Socket编程中还是比较重要的,可是对于初学Socket的人来说都不太爱用Select写程序,他们只是习惯写诸如connect.accept.recv或recvfrom这样的阻塞程序( ...
- select()函数以及FD_ZERO、FD_SET、FD_CLR、FD_ISSET
http://hi.baidu.com/%B1%D5%C4%BF%B3%C9%B7%F0/blog/item/e7284ef16bcec3c70a46e05e.html select函数用于在非阻塞中 ...
- PHP Socket实现websocket(四)Select函数
int select(int maxfdp,fd_set *readfds,fd_set *writefds,fd_set *errorfds,struct timeval *timeout); /* ...
- IO复用与select函数
socket select函数的详细讲解 select函数详细用法解析 http://blog.chinaunix.net/uid-21411227-id-1826874.html linu ...
- I/O多路复用——select函数与poll函数
1 区别 同:(1)机制类似,本质上没有多大差别,管理多个描述符也是进行轮询,根据描述符的状态进行处理.(2)包含大量文件描述符的数组被整体复制于用户态和内核的地址空间之间,而不论这些文件描述符是否就 ...
- select函数
select函数: http://baike.baidu.com/view/3421856.htm select函数 目录 概况 操作程序 宏解释 socket读写 概况 select()的机制中 ...
- select()函数以及FD_ZERO、FD_SET、FD_CLR、FD_ISSET(转)
select函数用于在非阻塞中,当一个套接字或一组套接字有信号时通知你,系统提供select函数来实现多路复用输入/输出模型, 原型: int select(int maxfd,fd_set *rds ...
- 阻塞、非阻塞的概念和select函数的阻塞功能
其它文档: http://www.cnitblog.com/zouzheng/archive/2010/11/25/71711.html (1)阻塞block 所谓阻塞方式block,顾名思义 ...
- select()函数以及FD_ZERO、FD_SET、FD_CLR、FD_ISSET (转)
select函数用于在非阻塞中,当一个套接字或一组套接字有信号时通知你,系统提供select函数来实现多路复用输入/输出模型,原型: #include <sys/time.h> ...
随机推荐
- Hexo + Github Pages搭建的个人博客
这个不算是新手的搭建流程,如果你恰巧看见这篇文章,希望你已经安装好node.git等软件,因为第一步的环境搭建准备并没有详写,默认都会了.希望能解决你的问题. 步骤: 一. 搭建环境准备 二.安装he ...
- destoon 自定义session丢失
destoon 在使用session之前 应该实例化 $session = new dsession(); destoon通过配置文件加载了不同session存储方式.如果你使用session的页面 ...
- 2.什么是composer与packgist,composer的安装
目录 学习地址: composer与packgist关系图片 composer的安装; 配置composer 修改国内镜像 用composer安装与卸载插件 composer插件升级后报错 学习地址: ...
- day14-推导式和生成器表达式
1.推导式规则 [每一个元素或者是和元素相关的操作 for 元素 in 可迭代数据类型] ----------遍历之后挨个处理[满足条件的元素相关的操作 for 元素 in 可迭代数据类型 if 元素 ...
- Python基础-函数参数
Python基础-函数参数 写在前面 如非特别说明,下文均基于Python3 摘要 本文详细介绍了函数的各种形参类型,包括位置参数,默认参数值,关键字参数,任意参数列表,强制关键字参数:也介绍了调用函 ...
- 1.python中的变量
什么是变量 1.在任何语言中都有变量的概念,在python中变量是用一个变量名表示,变量名必须是用大小写英文字母,数字,下滑写(_)组成.不能用数字开头.(但用中文做变量名也可以,不要这样做) 例: ...
- Python 使用multiprocessing 特别耗内存
采用multiprocessing多进程进行数据计算的时候内存飚升,这总体可以说是multiprocessing的一个「bug」导致: 大致原因如下: multiprocessing.Process ...
- OverflowError:django signed integer is greater than maximum
在学习一对一查询的时候,打印作者的电话时报了这个错 alex = Author.objects.filter(name='alex').first() print(alex.authordetail. ...
- JdbcTemplate实验
实验1:测试数据源 @Test public void test() throws SQLException { ApplicationContext ioc = new ClassPathXmlAp ...
- Linux学习-SELinux 初探
什么是 SELinux 什么是 SELinux 呢?其实他是『 Security Enhanced Linux 』的缩写,字面上的意义就是安全强化的 Linux 之意! 当初设计的目标:避免资源的误用 ...