使用信号实现异步通知机制的例子
include <sys/types.h> #include <sys/stat.h> #include <stdio.h> #include <fcntl.h> #include <signal.h> #include <unistd.h> #define MAX_LEN 100 //信号SIGIO的处理函数 void input_handler(int num) { char data[MAX_LEN]; int len; /* 读取并输出STDIN_FILENO上的输入 */ len = read(STDIN_FILENO, &data, MAX_LEN); data[len] = 0; printf("input available: %s\n", data); } void main() { int oflags; /* 启动信号驱动机制 */ signal(SIGIO, input_handler); //设置本进程为STDIN_FILENO文件的拥有者,没有这一步内核不会知道应该将信号发送给哪个进程 fcntl(STDIN_FILENO, F_SETOWN, getpid()); //获得文件的状态标志 oflags = fcntl(STDIN_FILENO, F_GETFL); //启动异步通知机制 fcntl(STDIN_FILENO, F_SETFL, oflags | FASYNC); while(1); }
由此可见,为了在用户空间中能处理一个设备释放的信号,它必须完成3项工作
(1)通过F_SETOWN IO控制命令设置设备文件的拥有者为本进程,这样从设备驱动发出的信号才能被本进程接收到。
(2)通过F_SETFL IO控制命令设置设备文件支持FASYNC,即异步通知机制。
(3)通过signal()函数连接信号和信号处理函数。
注:【fcntl系统调用】
功能描述:根据文件描述词来操作文件的特性。
用法:
int fcntl(int fd, int cmd);
int fcntl(int fd, int cmd, long arg);
int fcntl(int fd, int cmd, struct flock *lock);
参数:
fd:文件描述词。
cmd:操作命令。
arg:供命令使用的参数。
lock:同上。
有以下操作命令可供使用
F_GETFL :读取文件状态标志。
F_SETFL :设置文件状态标志。
F_GETOWN:获取当前在文件描述词 fd上接收到SIGIO 或 SIGURG事件信号的进程或进程组标识 。
F_SETOWN:设置将要在文件描述词fd上接收SIGIO 或 SIGURG事件信号的进程或进程组标识