在对文件(设备驱动)的访问上有分
同步,异步,阻塞,非阻塞,今天我学到是阻塞io的 poll---多个文件的阻塞。首先实现的是对设备文件的读(read)和写(write)。例如,这下面的是linux内核3.5版本中fs.h文件中file_operation结构体对read/write的声明:
keyword: 阻塞
poll();
poll函数调用了kernel里的SYS_POLL,SYS_POLL将每个文件描述符加入文件描述符集合,(这里的文件是在argv传参时传入)
------------------------------------
APP
------------------------------------
Kernel -- > SYS_POLL
------------------------------------
driver driver ......
No.1 No.2
| | ......
| | ......
poll poll ......
当添加完后,SYS_POLL会要求设备文件在读或写时(可以由程序员决定回应写操作或写操作),提醒SYS_POLL, 然后SYS_POLL阻塞等待被人提示。当设备文件提示SYS_POLL, 它不知道是谁叫他所以叫醒所有人轮询一遍,找出符合要求的文件。
预习了一下异步io:fasync_helper( ) 和 fcntl( )
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
keyword: 阻塞
poll();
poll函数调用了kernel里的SYS_POLL,SYS_POLL将每个文件描述符加入文件描述符集合,(这里的文件是在argv传参时传入)
------------------------------------
APP
------------------------------------
Kernel -- > SYS_POLL
------------------------------------
driver driver ......
No.1 No.2
| | ......
| | ......
poll poll ......
当添加完后,SYS_POLL会要求设备文件在读或写时(可以由程序员决定回应写操作或写操作),提醒SYS_POLL, 然后SYS_POLL阻塞等待被人提示。当设备文件提示SYS_POLL, 它不知道是谁叫他所以叫醒所有人轮询一遍,找出符合要求的文件。
//写入时提醒SYS_POLL
ssize_t test_write (struct file *filp, const char __user *buf, size_t size, loff_t *offset)
{
printk(KERN_DEBUG "test write\n");
empty = false;
wake_up_interruptible(&wq); //叫醒所有人;
return size;
}
预习了一下异步io:fasync_helper( ) 和 fcntl( )
#include <stdio.h>
#include <signal.h>
#include <sys/types.h>
#iinclude <sys/stat.h>
#include <fcntl.h>
#include <sys/select.h>
#include <sys/time.h>
#include <unistd.h>
int fd;
void func(int sig_num)
{
printf("sig_num: %d\n", sig_num);
char buf[1024] = {};
int ret; ret = read(fd, buf, 1024);
printf("ret: %d\n", ret);
}
int main(int argc, char ** argv)
{
if(argc != 2) return -1;
//在应用程序中捕捉SIGIO信号(由驱动程序发送)
signal(SIGIO, my_signal_fun);
fd = open(argv[1], O_RDWR); //argv[1] : 250 0 (设备文件) c (字符设备文件) -> cdev
if(-1 == fd)
{
perror("open");
return -1;
}
//将当前进程PID设置为fd文件所对应驱动程序将要发送SIGIO,SIGUSR信号进程PID
fcntl(fd, F_SETOWN, getpid());
//提取出`非阻塞`和`异步`,放回F_SETFL用与设置文件标识FLAG
fcntl(fd, F_SETFL, fcntl(fd, F_GETFL) | O_NONBLOCK | O_ASYNC);
close(fd);
return 0;
}