关于两个队列

46323_all-1.jpeg

首先我们必须明白,处于“LISTENING”状态的TCP socket,有两个独立的队列:

SYN队列(SYN Queue)

Accept队列(Accept Queue)

这两个术语有时也被称为“reqsk_queue”,“ACK backlog”,“listen backlog”,甚至“TCP backlog”,但是这篇文章中我们使用上面两个术语以免造成混淆。

SYN队列

SYN队列存储了收到SYN包的连接(对应内核代码的结构体:struct inet_request_sock)。它的职责是回复SYN+ACK包,并且在没有收到ACK包时重传,直到超时。在Linux下,重传的次数为:

$ sysctl net.ipv4.tcp_synack_retries

net.ipv4.tcp_synack_retries = 5

tcp_synack_retries - int整型

对于一个被动TCP连接,重传SYNACKs的次数。该值不能超过255。

默认值为5,如果初始RTO是1秒,那么对应的最后一次重传是31秒。

对应的最后一次超时是63秒之后。

发送完SYN+ACK之后,SYN队列等待从客户端发出的ACK包(也即三次握手的最后一个包)。当收到ACK包时,首先找到对应的SYN队列,再在对应的SYN队列中检查相关的数据看是否匹配,如果匹配,内核将该连接相关的数据从SYN队列中移除,创建一个完整的连接(对应内核代码的结构体:struct inet_sock),并将这个连接加入Accept队列。

Accept队列

Accept队列中存放的是已建立好的连接,也即等待被上层应用程序取走的连接。当进程调用accept(),这个socket从队列中取出,传递给上层应用程序。

这就是Linux处理SYN包的一个简单描述。顺便一提,当socket开启了TCP_DEFER_ACCEPT和TCP_FASTOPEN时,工作方式将会有细微不同,本文不做介绍。

队列大小限制

应用程序通过调用系统调用listen(2),传入backlog参数,来设置SYN队列和Accept队列的最大大小。比如下面这样,将SYN队列和Accept队列的最大大小同时设置为1024:

listen(sfd, 1024)

注意,在4.3版本之前的内核,SYN队列的大小是用另一种方式计算。

SYN队列的最大大小以前是用net.ipv4.tcp_max_syn_backlog来配置,但是现在已经不再使用了。现在用net.core.somaxconn来同时表示SYN队列和Accept队列的最大大小。在我们的服务器上,我们将它设置为16k:

$ sysctl net.core.somaxconn

net.core.somaxconn = 16384

队列设置为多大合适

知道了上面这些信息后,你可能会问,队列设置为多大合适?

答案是:看情况。对于大多数的TCP服务来说,这并不太重要。比如,Go语言1.11版本之前,并没有提供设置队列大小的方法。

尽管如此,也存在一些合理的原因,需要增大队列的大小:

当建立连接的请求速度确实很大时,即使是对于一个高性能的服务来说,SYN队列也可能需要设置的大一些。

SYN队列的大小,换言之就是等待ACK包的连接数。也即与客户端的平均往返时间越大,堆积在SYN队列中的连接就越多。对于那些大部分客户端都距离服务器很远的场景,比如说往返时间几百毫秒以上,可以将队列大小设置的大一些。

TCP_DEFER_ACCEPT选项如果打开了,会导致socket在SYN-RECV状态下维持更长的时间,也即增大了处于SYN队列中的时间。

但是,将backlog设置的过大也会带来不好的影响:

SYN队列中的每一个槽位都需要占用一些内存。当遇到SYN Flood攻击时,我们没有必要为这些发起攻击的包浪费资源。SYN队列中的inet_request_sock结构体,在4.14内核下,每个将占用256字节的内存。

linux下,如果想查看SYN队列的当前状态,我们可以使用ss命令来查询SYN-RECV状态的socket。比如如下执行结果,表示80端口的SYN队列中当前有119个元素,443端口则为78。

$ ss -n state syn-recv sport = :80 | wc -l

119

$ ss -n state syn-recv sport = :443 | wc -l

78

还可以通过我们的SystemTap脚本来观察这个数据:resq.stp

假如程序调用accept()不够快?

46323_full-accept-1.jpeg

如果程序调用accept()不够快会发生什么呢?

后续收到的SYN包,不会被SYN队列处理

后续收到的(用于建立连接的)ACK包,不会被SYN队列处理

TcpExtListenOverflows / LINUX_MIB_LISTENOVERFLOWS计数增加

TcpExtListenDrops / LINUX_MIB_LISTENDROPS计数增加

发生这种情况时,我们只能寄希望于程序的处理性能稍后能恢复正常,客户端重新发送被服务端丢弃的包。

内核的这种表现对于大部分服务来说是可接受的。顺便一提,可以通过调整net.ipv4.tcp_abort_on_overflow这个全局参数来修改这种表现,但是最好还是不要改这个参数。

可以通过查看nstat的计数来观察Accept队列溢出的状态:

$ nstat -az TcpExtListenDrops

TcpExtListenDrops 49199 0.0

但是这是一个全局的计数。观察起来不够直观,比如有时我们观察到它在增长,但是所有的服务程序看起来都是正常的。此时我们可以使用ss命令来观察单个监听端口的Accept队列大小:

$ ss -plnt sport = :6443|cat

State Recv-Q Send-Q Local Address:Port Peer Address:Port

LISTEN 0 1024 *:6443 *:*

Recv-Q这一列显示的是处于Accept队列中的socket数量,Send-Q显示的是队列的最大大小。在上面的例子中,我们发现并没有未被程序accept()的socket,但是我们依然发现ListenDrops计数在增长。

这是因为我们的程序只是周期性的短暂卡住不处理新的连接,而非永久性的不处理,过段时间程序又恢复了正常。这种情况下,用ss命令比较难观察这种现象,因此我们写了一个SystemTap脚本,它会hook进内核,把被丢弃的SYN包打印出来:

$ sudo stap -v acceptq.stp

time (us) acceptq qmax local addr remote_addr

1495634198449075 1025 1024 0.0.0.0:6443 10.0.1.92:28585

1495634198449253 1025 1024 0.0.0.0:6443 10.0.1.92:50500

1495634198450062 1025 1024 0.0.0.0:6443 10.0.1.92:65434

...

通过上面的操作,可以观察到哪些SYN包被ListenDrops影响了。从而我们也就可以知道哪些程序在丢失连接。

英文原文来自cloudflare的博客,地址如下:

SYN packet handling in the wild。

英文原文在后半部分还介绍了SYN Cookies对于SYN Flood的影响,我在本文中没有翻译,感兴趣的可以看看原文。

本文原始地址: https://pengrl.com/p/46323/

声明: 本文后续所有修改都会第一时间在原始地址更新。本文欢迎任何形式转载,转载时注明原始出处即可。

linux accept 队列,[译] TCP的SYN队列和Accept队列相关推荐

  1. socketmq 设置队列大小_[译] TCP的SYN队列和Accept队列

    关于两个队列 首先我们必须明白,处于"LISTENING"状态的TCP socket,有两个独立的队列: SYN队列(SYN Queue) Accept队列(Accept Queu ...

  2. Linux Kernel TCP/IP Stack — L1 Layer — 多队列网卡

    目录 文章目录 目录 多队列网卡 Intel 82575 的多队列硬件实现 Intel 82575 的多队列软件驱动实现 多队列网卡识别 多队列网卡 多队列网卡,是一种用来解决网络 I/O QoS 问 ...

  3. 【Linux网络编程】TCP网络编程中connect listen和accept三者之间的关系

    00. 目录 文章目录 00. 目录 01. TCP服务端和客户端流程 02. connect函数 03. listen函数 04. 三次握手 05. accept函数 06. 附录 01. TCP服 ...

  4. 【Linux网络编程】TCP网络编程中connect()、listen()和accept()三者之间的关系

    基于 TCP 的网络编程开发分为服务器端和客户端两部分,常见的核心步骤和流程如下: connect()函数 对于客户端的 connect() 函数,该函数的功能为客户端主动连接服务器,建立连接是通过三 ...

  5. c语言检查tcp连接_TCP/IP的底层队列

     自从上次学习了TCP/IP的拥塞控制算法后,我越发想要更加深入的了解TCP/IP的一些底层原理,搜索了很多网络上的资料,看到了陶辉大神关于高性能网络编程的专栏,收益颇多.今天就总结一下,并且加上自己 ...

  6. Linux tcp防止syn 攻击,Linux运维知识之linux 防御SYN攻击

    本文主要向大家介绍了Linux运维知识之linux 防御SYN攻击,通过具体的内容向大家展现,希望对大家学习Linux运维知识有所帮助. 一.默认syn配置 sysctl -a | grep _syn ...

  7. 基于数组的无锁队列(译)

    2019独角兽企业重金招聘Python工程师标准>>> 1 引言 最近对于注重性能的应用程序,我们有了一种能显著提高程序性能的选择:多线程.线程的概念实际上已经存在了很长时间.在过去 ...

  8. 重学TCP协议(9) 半连接队列、全连接队列

    1. 半连接队列.全连接队列基本概念 三次握手中,在第一步server收到client的syn后,把相关信息放到半连接队列中,同时回复syn+ack给client(第二步),同时开启一个定时器,如果超 ...

  9. java高性能阻塞队列,Linux c/c   后台开发组建之:高性能阻塞队列

    Linux c/c   后台开发组建之:高性能阻塞队列 (2015-12-01 06:01:47) 标签: Linux c/c 杂谈 分类: c/c 阻塞队列是后台开发中多线程异步架构的基本数据结构, ...

最新文章

  1. 南方科技大学唐圆圆组招聘环境相关领域科研人才(年薪33~50万)
  2. 长春8中2021年高考 成绩查询,长春八中2018高考喜报成绩、本科重本上线人数情况...
  3. Dynpro程序抬头信息要求多值输入的解决方法
  4. html实现 页面禁止右键 禁止复制 禁止图片拖动 禁止复制和剪切
  5. CPU 可以跑多快?地球到火星的距离告诉你!
  6. python将excel导入oracle,使用Python将Excel文件导入到Oracle数据库里
  7. Atitit.软件GUI按钮与仪表盘(01)--报警系统--
  8. Db4o数据库:快速入门
  9. python如何抓取网页里面的文字_python如何爬取网页中的文字
  10. 各种框架性能分析,和语言性能分析
  11. centOS7 防火墙关闭 远程端口无法访问问题
  12. 计算机连不了无线网络,如何解决电脑连接不上家里路由器的无线网
  13. sqlserver查询时间范围
  14. 老码农教你学英语(学习素材)
  15. phpstorm快速编辑模板技巧
  16. web端引入高德地图
  17. pythonprotect_Python models.PROTECT属性代码示例
  18. 初识Python(小白必看)
  19. 小程序JS-处理英语单词,获取英语意思,注意事项
  20. 数据结构与算法基础 基本概念

热门文章

  1. 预测未来 盘点大数据分析领域五大趋势
  2. ❀❀ selenium 学习网站 ★★★★★
  3. 数据库和MySQL相关面试题目
  4. Java反射_JDBC操作数据
  5. jq第一天(1.83里面的属性)属性-》属性
  6. 【Swift】在Swift中获取当前的wifi SSID
  7. IDC:中国IT安全硬件、软件和服务全景图2014–2018 预测与分析
  8. 技术人生:新的生活计划
  9. centos6.4 搭建vsftpd
  10. 在MOSS 2007中查看文档库和站点的大小