tcp option 结构体_TCP的SYN队列和Accept队列

370ae8d7e513ddaf0307460998196943.png

首先我们必须明白,处于“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_retriesnet.ipv4.tcp_synack_retries = 5

文档中对 tcp_synack_retries 的描述如下:

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.somaxconnnet.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 -l119$ ss -n state syn-recv sport = :443 | wc -l78

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

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

cce6907302a74fad1ae65cae7dfc1fd8.png

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

TcpExtListenOverflows / LINUX_MIB_LISTENOVERFLOWSTcpExtListenDrops / LINUX_MIB_LISTENDROPS

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

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

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

$ nstat -az TcpExtListenDropsTcpExtListenDrops 49199 0.0

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

$ ss -plnt sport = :6443|catState Recv-Q Send-Q Local Address:Port Peer Address:PortLISTEN 0 1024 *:6443 *:*

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

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

$ sudo stap -v acceptq.stptime (us) acceptq qmax local addr remote_addr1495634198449075 1025 1024 0.0.0.0:6443 10.0.1.92:285851495634198449253 1025 1024 0.0.0.0:6443 10.0.1.92:505001495634198450062 1025 1024 0.0.0.0:6443 10.0.1.92:65434...

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`tcp_accept` 是一个 TCP 协议栈中的函数,用于接受客户端连接。当 TCP 服务器收到客户端连接请求时,会调用 `tcp_accept` 函数,并将连接套接字(`struct tcp_pcb`)和一个回调函数作为参数传递给它。回调函数将在新连接建立时被调用。 在实现 TCP 服务器时,需要定义一个回调函数作为参数传递给 `tcp_accept` 函数。该回调函数将在新连接建立时被调用,用于处理新连接的请求。回调函数的类型定义如下: ```c typedef err_t (*tcp_accept_fn)(void *arg, struct tcp_pcb *newpcb, err_t err); ``` 其中,`arg` 是传递给 `tcp_accept` 函数的上下文信息,`newpcb` 是新连接的套接字,`err` 是错误码。回调函数应该返回一个错误码,表示新连接的建立是否成功。 以下是一个简单的 `tcp_accept` 回调函数的示例,用于接受新连接并将其加入到连接列表中: ```c err_t tcp_accept_callback(void *arg, struct tcp_pcb *newpcb, err_t err) { err_t ret_err; struct tcp_conn *conn; if (err != ERR_OK) { /* 处理错误 */ ret_err = err; goto out; } /* 创建连接结构 */ conn = (struct tcp_conn *)mem_malloc(sizeof(struct tcp_conn)); if (conn == NULL) { /* 处理错误 */ ret_err = ERR_MEM; goto out; } /* 初始化连接结构 */ conn->pcb = newpcb; conn->state = TCP_CONN_STATE_CONNECTED; /* 设置连接套接字回调函数 */ tcp_recv(newpcb, tcp_recv_callback); tcp_err(newpcb, tcp_err_callback); tcp_arg(newpcb, conn); /* 将连接结构添加到连接列表中 */ tcp_conn_add(conn); /* 返回成功 */ ret_err = ERR_OK; out: return ret_err; } ``` 在该回调函数中,首先判断是否有错误发生。如果有错误发生,应该进行相应的错误处理,并返回错误码。否则,应该分配一个新的连接结构,将连接套接字和连接状态等信息存储到结构中,并将该结构添加到连接列表中。最后,回调函数应该返回 `ERR_OK` 表示新连接的建立成功。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值