在用户程序中,使用非阻塞I/O的应用程序通常会使用select()和poll()系统调用查询是否可对设备进行无阻塞的访问。select()和poll()系统调用最终会引发设备驱动中的poll()函数被执行。select()和poll()系统调用的本质一样,前者在 BSD UNIX 中引入的,后者在 System V中引入的。应用程序中最广泛用到的是BSD UNIX中引入的select()系统调用,其原型如下: - int select(int numfds, fd _ set *readfds, fd _ set *writefds, fd _ set *exceptfds,struct timeval *timeout);
复制代码其中readfds、writefds、exceptfds分别是被select()监视的读、写和异常处理的文件描述符集合,numfds的值是文件描述符加1。timeout参数是一个指向struct timeval类型的指针,它可以使select()在等待timeout时间后若没有文件描述符准备好则返回。 下列操作用来设置、清除、判断文件描述符集合。 清除一个文件描述符集。 - FD_SET(int fd,fd_set *set)
复制代码将一个文件描述符加入文件描述符集中。 - FD_CLR(int fd,fd_set *set)
复制代码将一个文件描述符从文件描述符集中清除。 - FD_ISSET(int fd,fd_set *set)
复制代码判断文件描述符是否被置位。 设备驱动中 poll()函数的原型如下: - unsigned int(*poll)(struct file * filp, struct poll_table* wait);
复制代码第一个参数为file结构体指针,第二个参数为轮询表指针。这个函数应该进行以下两项工作。 l 对可能引起设备文件状态变化的等待队列调用poll_wait()函数,将对应的等待队列头添加到poll_table。 l 返回表示是否能对设备进行无阻塞读、写访问的掩码。 驱动程序poll()函数应该返回设备资源的可获取状态,即POLLIN、POLLOUT、POLLPRI、POLLERR、POLLNVAL等宏的位“或”结果。每个宏的含义都表明设备的一种状态,如POLLIN(定义为 0x0001)意味着设备可以无阻塞地读,POLLOUT(定义为0x0004)意味着设备可以无阻塞地写。 将上一小节的FIFO驱动修改为使用轮询操作,在kernel/drivers/char/x4412目录下新建x4412-globalfifo-poll.c,编辑内容如下: - #include <linux/module.h>
- #include <linux/types.h>
- #include <linux/fs.h>
- #include <linux/errno.h>
- #include <linux/sched.h>
- #include <linux/init.h>
- #include <linux/cdev.h>
- #include <asm/uaccess.h>
- #include <linux/poll.h>
- #include <linux/slab.h>
- #include <linux/device.h>
-
- #define GLOBALFIFO_SIZE 0x1000 /*全局fifo最大4K字节*/
- #define FIFO_CLEAR 0x1 /*清0全局内存的长度*/
- DEFINE_SEMAPHORE(sem);//初始化并声明一个信号量
- DECLARE_WAIT_QUEUE_HEAD(r_wait);//初始化并声明一个等待队列r_wait
- DECLARE_WAIT_QUEUE_HEAD(w_wait);//初始化并声明一个等待队列w_wait
- static int globalfifo_major;
- static struct class *cdev_class;
- struct globalfifo_dev
- {
- struct cdev cdev; /*cdev结构体*/
- unsigned int current_len; /*fifo有效数据长度*/
- unsigned char mem[GLOBALFIFO_SIZE]; /*全局内存*/
- };
- struct globalfifo_dev *globalfifo_devp; /*设备结构体指针*/
- int globalfifo_open(struct inode *inode, struct file *filp)
- {
- /*将设备结构体指针赋值给文件私有数据指针*/
- filp->private_data = globalfifo_devp;
- return 0;
- }
- int globalfifo_release(struct inode *inode, struct file *filp)
- {
- return 0;
- }
- static long globalfifo_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
- {
- struct globalfifo_dev *dev = filp->private_data;/*获得设备结构体指针*/
- switch (cmd)
- {
- case FIFO_CLEAR:
- down(&sem); //获得信号量
- dev->current_len = 0;
- memset(dev->mem,0,GLOBALFIFO_SIZE);
- up(&sem); //释放信号量
- printk(KERN_INFO "globalfifo is set to zero\n");
- break;
- default:
- return - EINVAL;
- }
- return 0;
- }
- static unsigned int globalfifo_poll(struct file *filp, poll_table *wait)
- {
- unsigned int mask = 0;
- struct globalfifo_dev *dev = filp->private_data; /*获得设备结构体指针*/
- down(&sem);
- poll_wait(filp, &r_wait, wait);//添加读等待队列头
- poll_wait(filp, &w_wait, wait);//添加写等待队列头
- if(dev->current_len != 0)/*fifo非空*/
- {
- mask |= POLLIN | POLLRDNORM; /*标示数据可获得*/
- }
- if (dev->current_len != GLOBALFIFO_SIZE)/*fifo非满*/
- {
- mask |= POLLOUT | POLLWRNORM; /*标示数据可写入*/
- }
- up(&sem);
- return mask;
- }
- static ssize_t globalfifo_read(struct file *filp, char __user *buf, size_t count,loff_t *ppos)
- {
- int ret;
- struct globalfifo_dev *dev = filp->private_data; //获得设备结构体指针
- DECLARE_WAITQUEUE(wait, current); //定义等待队列
- down(&sem); //获得信号量
- add_wait_queue(&r_wait, &wait); //进入读等待队列头
- if (dev->current_len == 0)/* 如果FIFO为空,则进程进入睡眠等待 */
- {
- if (filp->f_flags &O_NONBLOCK)//非阻塞
- {
- ret = - EAGAIN;
- goto out;
- }
- __set_current_state(TASK_INTERRUPTIBLE); //改变进程状态为睡眠
- up(&sem);
- schedule(); //调度其他进程执行
- if (signal_pending(current))//如果是因为信号唤醒
- {
- ret = - ERESTARTSYS;
- goto out2;
- }
- down(&sem);
- }
- if (count > dev->current_len)/* 拷贝到用户空间 */
- count = dev->current_len;
- if (copy_to_user(buf, dev->mem, count)) //从内核空间拷贝到用户空间
- {
- ret = - EFAULT;
- goto out;
- }
- else
- {
- memcpy(dev->mem, dev->mem + count, dev->current_len - count); //fifo(管道)数据前移
- dev->current_len -= count; //有效数据长度减少
- printk(KERN_INFO "read %d bytes(s),current_len:%d\n", count, dev->current_len);
- wake_up_interruptible(&w_wait); //唤醒写等待队列
- ret = count;
- }
- out:
- up(&sem); //释放信号量
- out2:
- remove_wait_queue(&w_wait, &wait); //从附属的等待队列头移除
- set_current_state(TASK_RUNNING);
- return ret;
- }
- /*globalfifo写操作*/
- static ssize_t globalfifo_write(struct file *filp, const char __user *buf, size_t count, loff_t *ppos)
- {
- struct globalfifo_dev *dev = filp->private_data; //获得设备结构体指针
- int ret;
- DECLARE_WAITQUEUE(wait, current); //定义等待队列
- down(&sem); //获取信号量
- add_wait_queue(&w_wait, &wait); //进入写等待队列头
- if (dev->current_len == GLOBALFIFO_SIZE)/* 如果FIFO已满则进入睡眠等待 */
- {
- if (filp->f_flags &O_NONBLOCK)//如果是非阻塞访问
- {
- ret = - EAGAIN;
- goto out;
- }
- __set_current_state(TASK_INTERRUPTIBLE); //改变进程状态为睡眠
- up(&sem);
- schedule(); //调度其他进程执行
- if (signal_pending(current))//如果是因为信号唤醒
- {
- ret = - ERESTARTSYS;
- goto out2;
- }
- down(&sem); //获得信号量
- }
- /*从用户空间拷贝到内核空间*/
- if (count > GLOBALFIFO_SIZE - dev->current_len)
- count = GLOBALFIFO_SIZE - dev->current_len;
- if (copy_from_user(dev->mem + dev->current_len, buf, count))
- {
- ret = - EFAULT;
- goto out;
- }
- else
- {
- dev->current_len += count;
- printk(KERN_INFO "written %d bytes(s),current_len:%d\n", count, dev->current_len);
- wake_up_interruptible(&r_wait); //唤醒读等待队列
- ret = count;
- }
- out:
- up(&sem); //释放信号量
- out2:
- remove_wait_queue(&w_wait, &wait); //从附属的等待队列头移除
- set_current_state(TASK_RUNNING);
- return ret;
- }
- /*文件操作结构体*/
- static const struct file_operations globalfifo_fops =
- {
- .owner = THIS_MODULE,
- .read = globalfifo_read,
- .write = globalfifo_write,
- .unlocked_ioctl = globalfifo_ioctl,
- .poll = globalfifo_poll,
- .open = globalfifo_open,
- .release = globalfifo_release,
- };
- /*初始化并注册cdev*/
- static void globalfifo_setup_cdev(struct globalfifo_dev *dev, int index)
- {
- int err, devno = MKDEV(globalfifo_major, index);
- cdev_init(&dev->cdev, &globalfifo_fops);
- dev->cdev.owner = THIS_MODULE;
- dev->cdev.ops = &globalfifo_fops;
- err = cdev_add(&dev->cdev, devno, 1);
- if (err)
- printk(KERN_NOTICE "Error %d adding LED%d", err, index);
- }
- /*设备驱动模块加载函数*/
- int globalfifo_init(void)
- {
- int ret;
- dev_t devno;
- ret = alloc_chrdev_region(&devno, 0, 1, "x4412-globalfifo-poll");
- globalfifo_major = MAJOR(devno);
- if (ret < 0)
- return ret;
- /* 动态申请设备结构体的内存*/
- globalfifo_devp = kmalloc(sizeof(struct globalfifo_dev), GFP_KERNEL);
- if (!globalfifo_devp) /*申请失败*/
- {
- ret = - ENOMEM;
- goto fail_malloc;
- }
- memset(globalfifo_devp, 0, sizeof(struct globalfifo_dev));
- globalfifo_setup_cdev(globalfifo_devp, 0);
- cdev_class = class_create(THIS_MODULE,"x4412-globalfifo-poll");
- if(IS_ERR(cdev_class))
- goto fail_class;
- device_create(cdev_class,NULL,devno,NULL,"x4412-globalfifo-poll");
- return 0;
- fail_class:
- class_destroy(cdev_class);
- fail_malloc:
- unregister_chrdev_region(devno, 1);
- return ret;
- }
- /*模块卸载函数*/
- void globalfifo_exit(void)
- {
- device_destroy(cdev_class,MKDEV(globalfifo_major,0));
- class_destroy(cdev_class);
- cdev_del(&globalfifo_devp->cdev); /*注销cdev*/
- kfree(globalfifo_devp); /*释放设备结构体内存*/
- unregister_chrdev_region(MKDEV(globalfifo_major, 0), 1); /*释放设备号*/
- }
- MODULE_LICENSE("Dual BSD/GPL");
- MODULE_AUTHOR("www.9tripod.com");
- module_init(globalfifo_init);
- module_exit(globalfifo_exit);
复制代码 程序中声明了两个等待队列,在读和写函数中实现数据同步,避免读写数据错乱。同时,在file_operations结构体中增加了轮询成员poll,在对应的函数globalfifo_poll中对读写等待队列调用poll_wait函数,返回可对设备操作权限的掩码。 编辑kernel/drivers/char/x4412/Kconfig文件,添加如下内容: - config X4412_GLOBALFIFO_POLL_DRIVER
- tristate "x4412 globalfifo by poll driver"
- default m
- help
- compile for x4412 globalfifo by poll driver,y for kernel,m for module.
复制代码 编辑kernel/drivers/char/x4412/Makefile文件,添加如下内容: - obj-$(CONFIG_X4412_GLOBALFIFO_DRIVER) += x4412-globalfifo-poll.o
复制代码 编译内核,在kernel/drivers/char/x4412目录将会生成目标映像x4412-globalfifo-poll.ko文件。 在ubuntu用户目录或samba目录下新建应用程序x4412-globalfifo-poll-app.c文件,编辑内容如下: - #include <sys/types.h>
- #include <sys/stat.h>
- #include <stdio.h>
- #include <fcntl.h>
- #include <unistd.h>
- #include <sys/time.h>
-
- main()
- {
- int fd;
- fd_set rfds,wfds;//读写文件描述符集
- /*以非阻塞方式打开设备文件*/
- fd = open("/dev/x4412-globalfifo-poll", O_RDONLY | O_NONBLOCK);
- if (fd != - 1)
- {
- while (1)
- {
- FD_ZERO(&rfds);//清除读文件描述符
- FD_ZERO(&wfds);//清除写文件描述符
- FD_SET(fd, &rfds);//将rfds加入文件描述符集
- FD_SET(fd, &wfds);//将wfds加入文件描述符集
- //监控读写文件描述符
- select(fd + 1, &rfds, &wfds, NULL, NULL);
- /*数据可获得*/
- if (FD_ISSET(fd, &rfds))
- {
- printf("Poll monitor:can be read\n");
- }
- /*数据可写入*/
- if (FD_ISSET(fd, &wfds))
- {
- printf("Poll monitor:can be written\n");
- }
- }
- }
- else
- {
- printf("Device open failure\n");
- }
- }
复制代码 应用程序首先定义了两个文件描述符rfds和wfds,再以非阻塞的方式打开驱动模块,紧接着清除读写文件描述符后,将他们加入文件描述符集,再通过select函数监控这两个文件描述符。在整个while循环中不断使用FD_ISSET函数判断数据是否可读可写,并打印相应的提示信息。 加载驱动模块后,在不写入任何内存的情况下运行应用程序,将不断打印“Pollmonitor:can be written”。 - [root@x4412 mnt]# insmod x4412-globalfifo-poll.ko
- [root@x4412 mnt]# ./x4412-globalfifo-poll
- Poll monitor:can be written
- Poll monitor:can be written
复制代码 通过echo指令向/dev/x4412-globalfifo-poll写入一些指令后,将不断打印“Poll monitor:can be read”和“Poll monitor:can be written”。 - [root@x4412 mnt]# echo 'www.9tripod.com' > /dev/x4412-globalfifo-poll
- [ 1121.931135] written 16 bytes(s),current_len:16
- [root@x4412 mnt]# ./x4412-globalfifo-poll
- Poll monitor:can be read
- Poll monitor:can be written
- Poll monitor:can be read
- Poll monitor:can be written
复制代码 反复通过echo指令向/dev/x4412-globalfifo-poll写指令,直到FIFO被写满4096bytes,再使用应用程序测试,将不断打印“Poll monitor:can beread”。 - [root@x4412 mnt]# echo 'www.9tripod.com' > /dev/x4412-globalfifo-poll
- [ 1419.245771] written 16 bytes(s),current_len:4096
- [root@x4412 mnt]#
- [root@x4412 mnt]# ./x4412-globalfifo-poll
- Poll monitor:can be read
- Poll monitor:can be read
复制代码 |