linux网络编程学习笔记之二 -----错误异常处理和各种碎碎(更新中)

errno

在unix系统中对大部分系统调用非正常返回时,通常返回值为-1,并设置全局变量errno(errno.h),如socket(), bind(), accept(), listen()。erron存放一个正整数来保存上次出错的错误值。

对线程而言,每个线程都有专用的errno变量,不必考虑同步问题。

strerror converts to English (Note: use strerror_r for thread safety)

perror is simplified strerror/fprintf


慢系统调用

指可能永远阻塞而无法返回的系统调用,通常是一些读写的例子,如pipe,终端设备,网络连接,典型的accept(), read(), write(), open(),select(),epoll()等。适用于慢系统调用的基本规则是:当阻塞于某个慢系统调用的进程捕获到某个信号且相应的信号处理函数返回时,系统调用可能会返回一个EINTR错误,即将errno的值置为该值。虽然有些内核会对该系统调用重启,但从可移植性的角度说,对EINTR处理是必须的。注意对connect()不能如此处理。(UNP 5.9)


一个信号被信号处理函数响应,在处理过程中,该信号被屏蔽。标准的信号实现没有排队的功能,所以信号可能会被丢失,多个连续的信号来不及处理。用waidpid()枚举检查不失为一种好的解决方法,见UNP 5.10


关于TOE(TCP offload engine)

本地抓包时一个小细节:


出现checksum incorrect。是由网卡上的TOE引擎造成的。TOE wiki

用TCP/IP协议处理网络流量,要占用大量服务器资源。为了减轻服务器的压力,一种称为TCP减负引擎(TCP Offload Engine :TOE)的技术应运而生。TCP减负引擎一般由软硬两部分组件构成,将传统的TIP/IP协议栈的功能进行延伸,把网络数据流量的处理工作全部转到网卡 上的集成硬件中进行,服务器只承担TCP/IP控制信息的处理任务。
一般由操作系统的TCP/IP协议栈完成TCP/UDP/IP校验和的计算工作,在网卡集成TOE的功能会包括计算checksum。设置Rx Checksum Offload/Tx Checksum Offload为Enable之后,协议栈不再进行校验和的计算,而是由网卡自己完成。
把网卡的属性修改一下就可以避免checksum incorrect,禁用 Checksum Offload。可修正checksum incorrect,代价是网络性能降低。

这个校验和的问题似乎对应用层的程序没有什么影响。。。(个人感觉)


Socket上的I/O处理

前文(标准I/O小结)提到的标准I/O库是ANSI C定义的一组高级输入输出函数,比起直接使用UNIX的系统I/O更加方便。
然而,标准I/O库没有提供读取文件元数据的方式,也不适合处理socket一类的特殊文件。


不足值(short count)
    定义,简而言之,我想处理10个字节,只处理了6个,则不足值是6 。注意不是4
    由于内核中套接口的缓冲区大小有限和网络时延的原因,造成read和write实际处理的字节比预期的要少。其他如遇到EOF或从终端读取,也可能遇到不足值
读写磁盘文件时不会出现不足值


处理方式:
 UNP 3.9和CSAPP 10.9(RIO) 均给出了解决方法,借鉴之,如下代码:

int simon_send(int fd, char* buf, unsigned int n)
{
	int left = n;	
	char *bufptr = buf;
	int send_bytes;
	
	while (left >= 0)
	{
		if ((send_bytes = send(fd, bufptr, MAX_BUF_SIZE, 0)) < 0)  
		{
			if (errno == EINTR)  //iterrupted by signal , send again
				send_bytes = 0;
			else
				break;
		}
		else if (!send_bytes)  // EOF or socket shutdown by peer
			break;

		left -= send_bytes;
		bufptr += send_bytes;
	}

	return n - left;
}

int simon_recv(int fd, char* buf, unsigned int n)
{
	int left = n;	
	char *bufptr = buf;
	int recv_bytes;
	
	while (left >= 0)
	{
		if ((recv_bytes = recv(fd, bufptr, MAX_BUF_SIZE, 0)) < 0)
		{
			if (errno == EINTR)  
				recv_bytes = 0;
			else
				return -1;
		}
		else if (!recv_bytes) 
			break;

		left -= recv_bytes;
		bufptr += recv_bytes;
	}

	return n - left;
}
另,使用带标志MSG_WAITALL的recv也能一定程度上处理。


关于I/O的选择和比较,可见下图(摘自某ppt):




linux网络编程学习笔记之二 -----错误异常处理和各种碎碎(更新中),古老的榕树,5-wow.com

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。