写点什么

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:451649

评论

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

TinyEngine低代码多人实时协作“原理”+“实操”全攻略解读

OpenTiny社区

开源 前端 低代码 OpenTiny

在线白板软件有哪些?8个主流工具盘点

职场工具箱

效率 在线白板 协同办公 办公软件 AI工具

从技术管理者到战略决策者,揭秘IT技术负责人的四个价值层次,看看您在第几层?

六边形架构

团队管理 技术战略 价值层次 价值提升

区块链 Web3 系统的开发方法

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

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

手把手教你搭建智能产品技术文档:从零到一的完整指南

百川云开发者

知识库工具 技术文档管理

Spring Data JPA 最佳实践【2/2】:存储库设计指南

郝培强

新加坡跃居全球人才竞争力榜首

财见

AI 智慧检查综合管理平台:让质检管理从 “碎片化” 变 “全链路可控”

上海拔俗

金融科技中网络安全的关键作用

qife122

区块链 网络安全

AI存储大战,京东云海跑出中国速度

脑极体

AI

众托帮风控逻辑:用规则留住信任

科技汇

从客服到“数字员工”:天润融通AI如何接管连锁门店的后台运营

天润融通

Proofpoint Satori威胁情报代理正式登陆Microsoft Security Copilot平台

qife122

网络安全 AI安全

史上最全各大AI模型对数据集成平台分析汇总

郝培强

极壳Hypershell完成7000万美元Pre-B及B轮融资,估值近4亿美元,引领消费级外骨骼新时代

财见

碎片化网络安全监管正增加移动运营商成本与风险

财见

护士响应快了,患者心里踏实了:开源鸿蒙智慧病房的一线实探

最新动态

MyEMS 在数据中心绿色演进中的角色:从 PUE 优化到余热回收与碳中和路径规划

开源能源管理系统

开源 能源管理系统

大数据-166 Apache Kylin 1.6 Streaming Cubing 实战:Kafka 到分钟级 OLAP

武子康

Java 大数据 kylin 分布式 Apache Kylin

VideoByte BD-DVD Ripper for Mac 蓝光DVD光盘提取工具

做梦万元户

当 MyEMS 遇见数字孪生:构建能源系统的虚拟镜像,实现模拟运行与优化策略预演

开源能源管理系统

开源 能源管理系统

从分钟到秒级,从 ClickHouse 到 StarRocks:哈啰的实时进化之路

StarRocks

数据湖 OLAP Clickhouse StarRocks 哈啰

集成亮数据代理IP到火狐浏览器实现数据采集

阿Q说代码

剖析大模型产生幻觉的三大根源

Baihai IDP

程序员 AI LLM 大模型幻觉

Web信息的物联网设备指纹如何生成

郑州埃文科技

分布式链路追踪实战:SkyWalking vs Zipkin 选型、部署与核心场景解析

郝培强

1688买家/卖家店铺订单API接口指南

Datafox(数据狐)

1688API 1688店铺订单API 1688店铺订单数据接口 1688店铺API

在 PowerPoint 中创建柱状图和折线图:用 Java 实现自动化数据可视化

郝培强

我用Gemini3pro 造了个手控全息太阳系

知识浅谈

视觉交互 谷歌Gemini3

VideoByte DVD Creator for Mac 蓝光光盘刻录软件

做梦万元户

当推荐系统真正“懂你”:快手团队在NeurIPS 2025提出新成果TagCF

快手技术

NeurIPS 快手技术

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