一个没正常处理tcp对端关闭的bug

最近使用自研的http client时发现一个问题,对端在发送响应数据之后立即调用close关闭了连接,我这没有调用到响应的回调,而是调用到了连接关闭的回调。对端延迟一会再关闭连接就没问题,用curl去访问也是正常的。经过排查是没有正确处理对端关闭导致的。

根据gnu文档以及其他搜索结果,如果没通过SO_LINGER选项改变close的默认行为,调用close关闭socket时,会关闭两个方向的数据流。读方向上,内核会将套接字设置为不可读,任何读操作都会返回异常;写方向上,内核会尝试将发送缓冲区的数据发送,之后发送FIN包结束连接,这个过程中,往套接字写入数据都会返回异常;若对端还发送数据过来,会返回一个RST报文。

epoll触发EPOLLIN事件时,表示对应的连接上有数据可以读出。tcp连接的远端主动关闭连接,也相当于可读事件,因为需要处理发送来的FIN包。

有问题的接收代码如下,问题在于recv返回0时,没有处理已经收到的数据就直接关闭socket。

cpp 复制代码
int tcp_event::handle_read()
{
	octets& read_buf = _ses->get_read_buf();
	int ret_bytes = 0;
	while(true)
	{
		if(read_buf.full())
		{
			_ses->onrecv();
		}
		int bytes_cnt = recv(_fd, read_buf.end(), read_buf.free_space(), 0);
		if(bytes_cnt == 0)
		{
			_ses->close(CLOSE_REASON_ONRECV);
			return -1;
		}
		else if(bytes_cnt != -1)
		{
			ret_bytes += bytes_cnt;
			read_buf.resize(read_buf.size() + bytes_cnt);
		}
		else
		{
			if(errno == EAGAIN || errno == EWOULDBLOCK) break;
			if(errno == EINTR) continue;
			_ses->close(CLOSE_REASON_ONRESET);
			return -1;
		}
	}
	_ses->onrecv();
	return ret_bytes;
}
相关推荐
辉视官方33 分钟前
2026深圳校园安全新方案 | SIP网络广播对讲系统,提升应急响应与智能教学水平
网络·安全
振浩微433射频芯片34 分钟前
告别“遥控失灵”:如何评估国产433芯片在智能家居领域的可靠性?
网络·单片机·嵌入式硬件·物联网·智能家居
pengyi87101537 分钟前
HTTP代理抓包核心原理,全面读懂请求与响应数据逻辑
网络·网络协议·http
云边云科技_云网融合1 小时前
@WAN SASE 1.0 全新起航,重新定义企业网络安全边界
网络·安全·web安全
小辰记事本1 小时前
从零读懂网卡内部架构:一条数据包的硬件之旅
网络·网络协议·架构·rdma
智慧光迅AINOPOL1 小时前
如何实现全光网的简单运维
网络·全光网解决方案·全光网·校园全光网·校园全光网解决方案
这是谁的博客?1 小时前
Python 异步编程核心原理与实践深度解析
java·网络·python·协程·asyncio·异步编程
2401_873479401 小时前
跨境电商如何评估用户IP真实性?用高精度IP地址查询+IP离线库实现
网络·tcp/ip·ip
minji...1 小时前
Linux 网络基础之网络IP层(十二)路由、路由表,分片和组装
linux·网络·tcp/ip·智能路由器·路由表·ip分片
Chengbei112 小时前
对标PentestGPT!新一代去中心化集群式AI全自动渗透测试工具
网络·人工智能·网络安全·去中心化·区块链·系统安全