TCP3次握手和backlog溢出

TCP3次握手,实际上可分为4步:

客户端发起connect(),发送SYN j

服务器从SYN queue中建立条目,响应SYN k, ACK J+1

客户端connect()成功返回,响应ACK K+1

服务器将socketSYN queue移入accept queueaccept()成功返回

注:SYN/FIN各占一个序列号,ACK/RST不占序列号



SYN queue长度由
tcp_max_syn_backlog指定,accept queue则由net.core.somaxconn决定,listen(fd, backlog)backlog上限由somaxconn决定;

Backlog解释

Now it specifies the queue length for completely established sockets waiting to be accepted, instead of the number of incomplete connection requests.

The maximum length of the queue for incomplete sockets can be set using the tcp_max_syn_backlog sysctl.

When syncookies are enabled there is no logical maximum length and this sysctl setting is ignored.

If the socket is of type AF_INET, and the backlog argument is greater than the constant SOMAXCONN (128 default), it is silently truncated to SOMAXCONN.

客户端connect()返回不代表TCP连接建立成功,有可能此时服务器accept queue已满,OS会直接丢弃后续ACK请求;

客户端以为连接已建立,开始后续调用(譬如send)等待直至超时;

服务器则等待ACK超时,会重传SYN k, ACK J+1给客户端(重传次数受限net.ipv4.tcp_synack_retries)

注:accept queue溢出,即便SYN queue没有溢出,新连接请求的SYN也可能被drop

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.

clog:


如何查看accept queue溢出

Netstat -s | grep LISTEN 返回***SYNS to LISTEN sockets ignored



以下案例全部来自维信公众号基础架构快报”TCP三次握手之backlog”,感兴趣的可以加微信阅读原文。


案例1

Nginx作为7层反向代理,客户端HTTP请求 – NGINX – 透明代理,透明代理接口存在大量慢请求;


思路

抓包,客户端同nginx通信,nginx立即返回ACK(1ms),但是3s后才返回响应数据;

Nginx同后端通信,发送SYN请求等待3s后端才响应;


结论

Backlog设置过小,导致accept queue溢出,SYN被丢弃导致3s重传;

backlog50增加到512somaxconn=512

案例2

Testserver随机生成RAR/ZIP文件,testclient访问testserver获取生成文件,所有调用采用block方式;

运行一段时间后程序永久阻塞,strace发现testclient阻塞在recvmsg


思路

抓包观察3次握手协议,服务器返回SYN+ACK,客户端响应ACK,可服务器再次发送同样的SYN+ACK

客户端响应的ACK丢包,而net.ipv4.tcp_synack_retries = 1


结论

三次握手最后一步失败,server保持SYN_RECV状态等待接收ACKclient发送ACK状态变为ESTABLISHED其认为connect()成功故接着调用recvmsg()

Syn+ack被设置为只重传一次,若这次重传仍失败,则客户端永久阻塞

其他案例1

Backlog过大,连接积压在accept queuenginx由于连接超时而断开PHP accept返回时连接已被客户端close,故报告PHP write Broken pipe


其他案例2

Backlog过小,accept queue溢出,握手第3步的ACK被丢弃,但client认为连接成功并发送数据,造成所谓慢请求


http://jaseywang.me/2014/07/20/tcp-queue-%E7%9A%84%E4%B8%80%E4%BA%9B%E9%97%AE%E9%A2%98/


http://blog.itpub.net/15480802/viewspace-1399303/


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值