对于客户端的sendto函数引发的错误,服务端会以一个port unreachable(端口不可达)的ICMP报文消息响应,遗憾的是这个错误并不会返回给客户端,我们称这个错误为异步错误(asyncchronous error),虽然这个错误是由sendto引起的,但是sendto却返回成功,而不是返回错误。
换句话说,udp套接字的sendto并不会返回这个异步错误,因为udp是无连接的。考虑这么一种情况:一个udp客户端给三个ip地址不同的udp服务器发送了三个数据报,然后客户端调用recvfrom等待读取udp服务器的应答数据报,数据报1和2正确到达目的服务器,但是有一个server3主机没有运行服务器,那么该主机收到数据报会以一个端口不可达的ICMP差错报文响应。
如图1所示,167为客户端,169为server3主机。客户端向server3主机发送了一个hello的udp数据包,由于server3主机并没有运行服务器程序,于是发送了一个ICMP差错报文进行响应。
我们可以在图2中看到ICMP差错报文中中Type字段和Code字段知道这是一个端口不可达错误,Data字段中是udp数据报中的数据部分,并且在这个ICMP差错报文中还封装了引发错误的数据报IP首部和UDP首部,这样udp客户端就可以通过数据报的目的ip地址和端口来区分哪一个数据报引发的错误。但有一个问题:recvfrom函数能返回的仅仅只有errno错误号而已,内核应该如何把错误信息返回给udp客户端。
如果客户端想要知道具体是哪个主机引发的错误应该怎么做呢?解决办法就是udp客户端套接字调用connect函数连接到对server3主机,这样错误就会返回给udp客户端进程。