写点什么

TCP 连接的 99 号和 110 号错误

  • 2019-11-25
  • 本文字数:2566 字

    阅读完需:约 8 分钟

TCP连接的99号和110号错误

当客户端频繁的采用短链接时候,经常会遇到[110][connection time out]和[99][could not assigned requested address]的错误。以下是对两种错误的分析以及优化建议,希望对大家有所帮助。

短链接常见的两种错误

当客户端频繁的采用短链接时候,经常会遇到[110][connection time out]和[99][could not assigned requested address]的错误。前段时间我们的存储服务就遇到了这样的一拨报警,经过调研分析,基本确定以上这两个错误与客户端端口的 TIME-WAIT 状态以及服务端的 listen 队列有关(当然也有其它可能的原因,这里只分析这两种)。


从客户端来看,在我们的应用场景中,因为频繁的使用端连接,而且在同一台机上的客户端的数量比较多,造成了大量的 TIME-WAIT 状态的端口,当 TIME-WAIT 状态端口的数量铺满了整个 port_range(由 ip_local_port_range 内核参数指定)范围后,就会产生 99 号错误;从服务端来看,因为频繁大量的 accept 短链接,到达一定量后,服务端口的 listen 队列会出现溢出,这个时候,新的连接请求会被丢弃,连接建立失败,客户端也就产生了 110 号错误。

两种错误产生的原因

[99][could not assigned requested address]

TIME-WAIT 状态是连接一端主动关闭并发送完最后一个 ACK 之后所处的状态,这个状态一般会存在 2MSL(Max Segment Lifttime,即一个包在传输过程中的最大生存时间)时间(所以又叫 2MSL 状态),之所以要有这个状态,是为了让前一个连接的包不影响后面的链接,并且可以被有效的应答,以保证 TCP 连接的可靠性。


为了避免混淆在 TIME-WAIT 状态连接上的处理的包是前一个连接迟到的包还是新连接的包,TCP 协议规定在整个 TIME-WAIT 状态下,不能再建立同样的连接(即四元组一样的连接,但是可以利用处于 TIME-WAIT 的端口建立四元组不一样的连接)Linux 的 TCP/IP 协议栈在判断一个处于 TIME-WAIT 状态的本地端口是否可以作为一个新连接的本地绑定端口时,需要对这个端口做一个是否可用的的判断(在四员组符合之后)。


下面是 linux.2.6.32.70 内核中这一部分逻辑的代码片段



在上面的逻辑中返回 1 表示可以用,返回 0 表示不可用,不可用后报的错是"EADDRNOTAVAIL", 也就是“[99][could not assigned requested address]”。从上面的代码逻辑中可以看出,当 tcp_tw_resuse 对能否重用处于 TIME-WAIT 状态的端口至关重要


1.若 tcp_tw_resuse 未打开,且没有空闲的窗口使用时,则会报“[99][could not assigned requested address]”的错误


2.若 tcp_tw_reuse 打开了,且处于 TIME_WAIT 状态端口的连续两次连接使用间隔要小于等于 1 秒,也会报“[99][could not assigned requested address]”的错误


验证:1 的情况好理解,现在验证 2 的情况,这也是我们线上的客户端的情况,设定系统的 port_range 只有一个元素


$cat "net.ipv4.ip_local_port_range = 1024 1024" >> /etc/sysctl.conf && sysctl -p
复制代码


然后客户端果然返回"[99][could not assigned requested address]"错误,验证成功。

[110][connection time out]

Linux 的服务端从 listen 的端口建立的连接要经过两个队列的过渡,分别是 SYN 队列和 ACCEPT 队列。服务端接受到 SYN 请求后,会发送 SYNACK,并把这个 request sock 存在 SYN 队列内;等到三次握手完成后,再存放到 ACCEPT 队列内;然后再由 accept 系统调用,从 ACCEPT 队列内拿出,交给用户使用。


SYN 队列和 ACCEPT 队列都是有长度限制的,这个长度限制与以下三个参数有关:


a. 调用 listen 接口,传递给 back_log 参数;


b. 内核参数 somaxconn; //与 ACCEPT 队列相关


c.内核参数 tcp_max_syn_backlog; //与 SYN 队列相关


我们线上的问题主要是 ACCEPT 队列出现溢出造成的,所以这里主要分析 ACCEPT 队列长度限制的情况 。


在调用 listen 接口的时候,内核会用系统的 somaxconn 参数去截断传递给 listen 的 back_log 参数。下面是 linux2.6.32-70 的相关代码片段



上面的 sk_max_ack_backlog 就是 listen 端口的 ACCEPT 队列的最大长度


当短链接的量太大,accept 系统调用接口处理来不及时,ACCEPT 队列就可能会阻塞溢出,这个时候,Linux 的 TCP/IP 协议栈的做法是把新来的 SYN 请求丢弃掉( Accept backlog is full. If we have already queued enough of warm entries in syn queue, drop request. It is better than clogging syn queue with openreqs with exponentially increasing timeout.),这样当客户端设定的连接超时不够发送第二次 SYN 请求时,就会收不到服务端 ack,连接建立失败,这个时候报的错误是 ETIMEDOUT,也就是“[110][connection time out]“。


下面是 linux.2.6.32-70 的相关代码片段



在上面的代码段中,sk_acceptq_is_full(sk)是判断 ACCEPT 队列是否满了(队列长度限制已经在 listen 系统调用中被截断了,这也是为什么我们修改内核 somaxconn 内核参数,对当前应用程序的已经 listen 的端口的 ACCEPT 队列长度限制不产生影响的原因,需要重起,才能够使用新的内核参数),如果满了,而且 SYN 队列中又有新的没有完成握手的连接请求,则丢弃当前这个链接请求,这个时候的如果客户端设置的链接超时只够它发送一次 SYN 请求,则链接失败,发生“[110][connection time out]“报错。


验证:


1.按照线上情况,设置 somaxconn 为 128,listen 接口的 back_log 为 8192 运行一定数量的客户端,频繁的向服务端建立 TCP 链接,然后释放,观察情况 。


2.设置 somaxconn 为 8192, 同时设置 listen 的接口的 back_log 参数也为 8192,重复 1 的步骤。



上面是单个客户端的代码逻辑,很简单。

somaxconn 为 128

客户端大量报错



服务端



从上面的结果可以看出,被丢弃的 SYNs 在不断的增加

somaxconn 为 8192

客户端没有报错


服务端



可以看出,这段时间内没有被丢弃的 SYNs

总结

验证的结果和内核代码以及我们的预想是吻合的

解决办法

1.提高客户端的链接超时限制。当前是 300ms,比如可以提升到 3s 等;


2.提高服务端的 somaxconn 限制,这是个指标不治本的方法,只能是一定程度的缓解。(修改内核的其他的网络参数也是一样,只能是缓解,并不能解决根本问题)。


3.在客户端使用连接缓冲池,将短链接转换成长链接来使用(个人认为这个才是更好的办法,一劳永逸)。


本文转载自公众号 360 云计算(ID:hulktalk)。


原文链接:


https://mp.weixin.qq.com/s/umQTPgYC7UVtOGCbVjRnyA


2019-11-25 10:451599

评论

发布
暂无评论
发现更多内容

基于Terraform构建AI会议摘要系统

qife122

人工智能 语音识别

Conc,一个神奇的Go语言并发利器!

左诗右码

基于开发者空间编写ODBC应用程序操作OpenGauss数据库

华为云开发者联盟

GaussDB(for openGauss) ODBC C++ API 接口 华为开发者空间

区块链Web3项目开发的进度管理

北京木奇移动技术有限公司

区块链开发 软件外包公司 web3开发

使用QDeveloper账号登录Kiro IDE

IT蜗壳-Tango

合合信息亮相WAIC大会—推动AI鉴伪技术革新,打造可信数字世界【人工智能大会最新AI安全技术成果】

申公豹

人工智能

星云低代码:低代码不再是平台,而是“中间件”

星云低代码中间件

低代码 开发工具 企业管理系统 Vue 3 #java

华为开发者空间使用CodeArts IDE调用API实现电商平台问答助手

华为云开发者联盟

AI+ 华为云CodeArts Qwen1.5 华为开发者空间

Altair HyperWorks仿真案例:通过设计优化最大化发挥增材制造的潜力

智造软件

仿真 Hypermesh hyperworks

基于多代理协作的智能电子取证解决方案

qife122

人工智能 多代理系统

三级等保测评流程优化实践

等保测评

小白也可以只用 2 轮 prompt 生成你的“哈利波特测试器”

北京中暄互动广告传媒有限公司

2025年最佳加密软件精选

qife122

数据保护 加密技术

本地IntelliJ IDEA连接开发者空间-云开发环境,进行小游戏开发

华为云开发者联盟

云主机 IntelliJ IDEA 华为开发者空间

CST怎样安装补丁?CST Studio Suite 2022 SP2补丁包

思茂信息

电磁仿真 CST软件 CST Studio Suite

播放器音频后处理实践(一)

百度Geek说

TRAE cue 体验提升之 Latency 篇

北京中暄互动广告传媒有限公司

零代码生成 3D 游戏:基于 Amazon Q Developer CLI 和 Three.js 的实践

亚马逊云科技 (Amazon Web Services)

哈尔滨二级等保设备选型指南

等保测评

闭源精密战与开源敏捷性:能源管理系统的双线引领

开源能源管理系统

开源 能源管理系统

工厂工地如何利用MyEMS实现安全绿色生成,助力企业节能减排,降本增效

开源能源管理系统

开源 能源管理系统

物理信息随机投影神经网络的线性稳定性分析

qife122

神经网络 数值分析

Gunra勒索软件集团推出高效Linux变种

qife122

加密技术 勒索软件

MyEMS开源能源管理系统实际案例分析

开源能源管理系统

开源 开源能源管理系统 MyEMS

大数据-60 Kafka 消息消费与心跳机制详解:原理、流程及参数调优

武子康

Java 大数据 kafka 分布式

三级等保在哈尔滨智慧城市建设中的战略价值

等保测评

大模型训练故障恢复效率提升方案

qife122

机器学习 容错机制

“代码跑着跑着,就变快了?”——揭秘Java性能幕后引擎:即时编译器

poemyang

编译原理 Java虚拟机 即时编译器 JIT编译器 #java

如何选择合适的LED显示屏像素间距?

Dylan

LED LED display LED显示屏 LED屏幕

苹果电脑怎么在虚拟机里安装office办公软件?Mac怎么运行Windows版的office365?

凡凡不烦

Office Parallels Desktop下载 虚拟机软件 Mac效率办公软件 如何在Mac运行win

TCP连接的99号和110号错误_文化 & 方法_吴晓飞_InfoQ精选文章