4,信号驱动I/O :让内核在描述字就绪时发送SIGIO信号通知我们。首先开启套接口的信号驱动1/O功能,sigaction系统调用安装一个信号处理函数,
当内核数据包准备好时,会为该进程产生一个SIGIO信号。应用可以在信号处理时接收数据。
具体步骤:1,建立SIGIO信号处理函数。
2,设置该套接口的属主,通常使用fcntl的F_SETOWN命令设置。
3,开启该套接口的信号驱动I/O,通常使用fcntl的F_SETFL命令打开O_ASYNC标志完成。
UDP套接口上的SIGIO信号:
UDP上使用信号驱动I/O是简单的。当下述事件发生时产生SIGIO信号:
1. 数据报到达套接口
2. 套接口上发生异步错误
TCP套接口上的SIGIO信号
不幸的是,信号驱动I/O对TCP套接口几乎是没用的,原因是该信号产生得过于频繁,并且该信号的出现并没有告诉我们发生了什么事情。
下列条件均可在TCP套接口上产生SIGIO信号(假设信号驱动I/O是使能的):
1. 在监听套接口上有一个连接请求已经完成
2. 发起了一个连接拆除请求
3. 一个连接拆除请求已经完成
4. 一个连接的一半已经关闭
5. 数据到达了套接口
6. 数据已从套接口上发出(即输出缓冲区有空闲时间)
7. 发生了一个异步错误
因此,当我们捕获到SIGIO信号时,我们调用recvfrom读取到达的数据报或者获取异步错误。
例如,如果一个进程既从一个TCP套接口读数据,又向其上写数据,当新数据到达或者以前所写数据得到确认后均会产生SIGIO信号,
进程无法在信号处理程序中区分这两种情况。如果在这种情况下使用SIGIO,TCP套接口应该被设置为非阻塞方式以防止read或write发生阻塞。
我们应该考虑只在监听TCP套接口上使用SIGIO,因为在监听套接口上产生SIGIO的唯一条件是一个新连接的完成。
这里找到的实际使用信号驱动I/O的程序是基于UDP的NTP(网络时间协议)服务器程序。NTP服务器的主循环从客户接收数据报并送回相应,但是
每个客户请求都要进行相当数量的处理(比我们简单的回射服务器多得多)。对于服务器来讲,很重要的一点是给每个收到的数据报记录精确的
时间戳,因为这个值要返回给客户,客户要用它来计算到达该服务器的来回时间。图22.1展示了构建这样一个UDP服务器的两种方法。
大多数UDP服务器都被设计成图中左边的方式。但是NTP服务器采用了图中右边的技术: 当一个新数据报到达时,SIGIO处理程序读得该数据报,
同时记录数据报到达的时刻,然后把它放入进程的另一个队列中,由主服务器循环移走和处理。虽然这种技巧使服务器代码变得复杂,但是它为
到达的数据报提供了精确的时间戳。
使用SIGIO的UDP回射服务器程序