阻塞与非阻塞

阻塞和非阻塞

所谓阻塞,就是等待某件事情发生。比如调用read读取按键时,如果没有按键数据则read函数不会返回,它会让线程休眠等待。
使用poll时,如果传入的超时时间不为0,这种访问方法也是阻塞的。

APP调用open函数时,传入O_NONBLOCK,就表示要使用非阻塞方式;默认是阻塞方式。
注意:对于普通文件、块设备文件,O_NONBLOCK不起作用。
注意:对于字符设备文件,O_NONBLOCK起作用的前提是驱动程序针对O_NONBLOCK做了处理。

只能在open时表明O_NONBLOCK吗?在open之后,也可以通过fcntl修改为阻塞或非阻塞。
使用poll时,可以设置超时时间为0,这样即使没有数据它也会立刻返回,这就是非阻塞方式。

open时设置:
int fd = open(“/dev/xxx”, O_RDWR | O_NONBLOCK); /* 非阻塞方式 /
int fd = open(“/dev/xxx”, O_RDWR ); /
阻塞方式 /
open之后设置:
int flags = fcntl(fd, F_GETFL);
fcntl(fd, F_SETFL, flags | O_NONBLOCK); /
非阻塞方式 /
fcntl(fd, F_SETFL, flags & ~O_NONBLOCK); /
阻塞方式 */

1.应用程序打开驱动程序时会给驱动程序构造,在内核文件系统这一层会给驱动程序构造FILE 结构体,里面有flag,有f_flag保存打开驱动程序保存的flag
或者使用fcntl来修改是flag新的值都会保存成员里面,我们可以由此分辩是想阻塞还是非阻塞。
2.static ssize_t drv_read(struct file *fp, char __user *buf, size_t count, loff_t *ppos)

if (queue_empty(&as->queue) && fp->f_flags & O_NONBLOCK)
return -EAGAIN;
可以判断struct file *fp结构体中这一位O_NONBLOCK的话就是非阻塞假设没有数据的话立刻返回错误,如果这一位没有设置可以调用下面函数
wait_event_interruptible(apm_waitqueue, !queue_empty(&as->queue));
……
}有数据就返回数据没数据就休眠

1.应用程序

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

static int fd;

int main(int argc, char **argv)
{
int val;
int ret;
int flags;

int i;

if (argc != 2) 
{
	printf("Usage: %s <dev>\n", argv[0]);
	return -1;
}
fd = open(argv[1], O_RDWR | O_NONBLOCK);//以非阻塞打开
if (fd == -1)
{
	printf("can not open file %s\n", argv[1]);
	return -1;
}

for (i = 0; i < 10; i++) //读10次
{
	if (read(fd, &val, 4) == 4)
		printf("get button: 0x%x\n", val);
	else
		printf("get button: -1\n");
}

flags = fcntl(fd, F_GETFL);//取出flag
fcntl(fd, F_SETFL, flags & ~O_NONBLOCK);//把flags位清掉

while (1)
{
	if (read(fd, &val, 4) == 4)
		printf("get button: 0x%x\n", val);
	else
		printf("while get button: -1\n");
}
close(fd);
return 0;

}

驱动程序程序“只提供功能,不提供策略”。就是说驱动程序可以提供休眠唤醒、查询等等各种方式,,驱动程序只提供这些能力,怎么用由APP决定。

上一篇:移动端的长按事件的实现


下一篇:mybatis循环插入数据问题