tcp选项TCP_DEFER_ACCEPT

tcp选项TCP_DEFER_ACCEPT

http://blog.chinaunix.net/uid-23207633-id-274317.html

之前在项目测试的时候,如果第三次握手发完裸ack(没有数据)之后不发送数据的时候,连接状态一直为SYN_RCV,而且服务端重传synack,当时很不解,后来看了下源码,才发现些端倪。当时测试的内核是2.6.18-194(centos5.5)。

第三次握手会调用函数tcp_v4_hnd_req:

  1. static struct sock *tcp_v4_hnd_req(struct sock *sk, struct sk_buff *skb)
  2. {
  3. ......
  4. struct request_sock *req = inet_csk_search_req(sk, &prev, th->source,
  5. iph->saddr, iph->daddr);//查找半连接队列,返回req
  6. if (req)
  7. return tcp_check_req(sk, skb, req, prev);//ack的处理
  8. ......
  9. }

我们看函数tcp_check_req

  1. struct sock *tcp_check_req(struct sock *sk,struct sk_buff *skb,
  2. struct request_sock *req,
  3. struct request_sock **prev)
  4. {
  5. ......
  6. /* If TCP_DEFER_ACCEPT is set, drop bare ACK. */
  7. if (inet_csk(sk)->icsk_accept_queue.rskq_defer_accept &&
  8. TCP_SKB_CB(skb)->end_seq == tcp_rsk(req)->rcv_isn + 1) {//如果选项设置了,并且是裸
  9. ack,丢弃该ack;选项值得默
  10. 认为1
  11. inet_rsk(req)->acked = 1;
  12. return NULL;
  13. }
  14. child = inet_csk(sk)->icsk_af_ops->syn_recv_sock(sk, skb,
  15. req, NULL);//如果非裸ack或没设置选项则建立连接(req从半连接
  16. 队列到连接队列及tcp状态变为ESTABLISHED)
  17. ......
  18. }

我们在用户层写socket程序时,可以通过setsockopt来设置TCP_DEFER_ACCEPT选项:

  1. val = 5;
  2. setsockopt(srv_socket->fd, SOL_TCP, TCP_DEFER_ACCEPT, &val, sizeof(val)) ;
  3. 里面 val 的单位是秒,注意如果打开这个功能,kernel 在 val 秒之内还没有收到数据,不会继续唤醒进程,而是直接丢弃连接。

在内核空间会调用:

    1. static int do_tcp_setsockopt(struct sock *sk, int level,
    2. int optname, char __user *optval, int optlen)
    3. {
    4. struct tcp_sock *tp = tcp_sk(sk);
    5. struct inet_connection_sock *icsk = inet_csk(sk);
    6. int val;
          ......
    7. if (get_user(val, (int __user *)optval))//拷贝用户空间数据

      return -EFAULT;

  1. ......
  2. case TCP_DEFER_ACCEPT:
  3. icsk->icsk_accept_queue.rskq_defer_accept = 0;
  4. if (val > 0) {//如果setsockopt中设置val为0,则不开始TCP_DEFER_ACCEPT选项
  5. /* Translate value in seconds to number of
  6. * retransmits */
  7. while (icsk->icsk_accept_queue.rskq_defer_accept < 32 &&
  8. val > ((TCP_TIMEOUT_INIT / HZ) <<
  9. icsk->icsk_accept_queue.rskq_defer_accept))//根据设置的val决定重传次数,譬
  10. 如val=10,重传次数为3;后面我们可以看到,只有
  11. /proc/sys/net/ipv4/tcp_synack_retries的
  12. 值小于等于通过val算出的重传次数时,这个val才
  13. 起作用
  14. icsk->icsk_accept_queue.rskq_defer_accept++;
  15. icsk->icsk_accept_queue.rskq_defer_accept++;
  16. }
  17. break;
  18. ......
  19. }

内核是通过函数inet_csk_reqsk_queue_prune进行重传synack:

  1. void inet_csk_reqsk_queue_prune(struct sock *parent,
  2. const unsigned long interval,
  3. const unsigned long timeout,
  4. const unsigned long max_rto)
  5. {
  6. struct inet_connection_sock *icsk = inet_csk(parent);
  7. struct request_sock_queue *queue = &icsk->icsk_accept_queue;
  8. struct listen_sock *lopt = queue->listen_opt;
  9. int max_retries = icsk->icsk_syn_retries ? : sysctl_tcp_synack_retries;//默认synack
  10. 重传次数为5
  11. int thresh = max_retries;
  12. unsigned long now = jiffies;
  13. struct request_sock **reqp, *req;
  14. int i, budget;
  15. ......
  16. if (queue->rskq_defer_accept)
  17. max_retries = queue->rskq_defer_accept;//设定支持选项时候的重传次数
  18. budget = 2 * (lopt->nr_table_entries / (timeout / interval));
  19. i = lopt->clock_hand;
  20. do {
  21. reqp=&lopt->syn_table[i];
  22. while ((req = *reqp) != NULL) {
  23. if (time_after_eq(now, req->expires)) {
  24. if ((req->retrans < thresh ||
  25. (inet_rsk(req)->acked && req->retrans < max_retries))
  26. && !req->rsk_ops->rtx_syn_ack(parent, req, NULL)) {//如果重传次数小于设定
  27. 的重传次数,就重传synack;这里可以看出两个并列的判断条件:req->retrans < thres
  28. h和(inet_rsk(req)->acked && req->retrans < max_retries),第一个是当前req
  29. 的重传次数小于设定的最大重传次数,这里是5;第二个则是TCP_DEFER_ACCEPT;inet_rs
  30. k(req)->acked则是在函数tcp_check_req中设定的,上面讨论过了,而max_retries则
  31. 为通过val计算的值,默认为1。这个重传次数决定了synack包的重传次数及最长超时时间,
  32. 显然两者中较大者起到决定性的作用。譬如,默认重传为2,通过val计算出的max_retries
  33. 值为3,则将发送3次重传的synack及超时时间为12秒后,关闭连接
  34. unsigned long timeo;
  35. if (req->retrans++ == 0)
  36. lopt->qlen_young--;
  37. timeo = min((timeout << req->retrans), max_rto);
  38. req->expires = now + timeo;//每重传一次,超时值就按初始值
  39. timeout(TCP_TIMEOUT_INIT)比值为2的等比
  40. 数列增加,如3 6 12 24 48 96
  41. reqp = &req->dl_next;
  42. continue;//继续循环
  43. }
  44. /* Drop this request */
  45. 如果超时,如超过例子中的96秒,就将req从半连接队列里删除,丢弃连接
  46. inet_csk_reqsk_queue_unlink(parent, req, reqp);
  47. reqsk_queue_removed(queue, req);
  48. reqsk_free(req);
  49. continue;
  50. }
  51. reqp = &req->dl_next;
  52. }
  53. i = (i + 1) & (lopt->nr_table_entries - 1);
  54. } while (--budget > 0);
  55. lopt->clock_hand = i;
  56. if (lopt->qlen)
  57. inet_csk_reset_keepalive_timer(parent, interval);
  58. }

那么TCP_DEFER_ACCEPT选项有什么好处呢,我们知道服务端处于监听时,客户端connect;服务端会收到syn包,并发送
synack;当客户端收到synack并发送裸ack时,服务端accept创建一个新的句柄,这是不支持TCP_DEFER_ACCEPT选项下的流
程。如果支持TCP_DEFER_ACCEPT,收到裸ack时,不会建立连接,操作系统不会Accept,也不会创建IO句柄。操作系统应该在若干秒
后,会释放相关的链接;但没有同时关闭相应的端口,所以客户端会一直以为处于链接状态,如果Connect后面马上有后续的发送数据,那么服务器会调用
Accept接收这个连接。

函数inet_csk_reqsk_queue_prune是通过tcp_synack_timer,是它在定时器中起作用的

  1. static void tcp_synack_timer(struct sock *sk)
  2. {
  3. inet_csk_reqsk_queue_prune(sk, TCP_SYNQ_INTERVAL,
  4. TCP_TIMEOUT_INIT, TCP_RTO_MAX);
  5. }

关于定时器,在后续的分析中。

时间: 2024-11-13 19:55:47

tcp选项TCP_DEFER_ACCEPT的相关文章

TCP/IP详解--TCP首部选项中时间戳选项

一.简介 TCP时间戳选项会在TCP包头增加12个字节,以一种比重发超时更精确的方法来启用对RTT 的计算.   二.作用 1) TCP时间戳位于TCP选项中,kind=8:lenth=10:data由timestamp和timestamp echo两个值组成,各4个字节的长度. 2) TCP时间戳理论作用有3个:序列号回绕,乱序的时间判断依据,避免确认二义性,以及计算RTT. 3) TCP时间戳工作方式:双方各自维护自己的时间戳,时间戳的值随时间单调递增(规定为1ms-1s/次,常见值为1ms

TCP KeepAlive的几个附加选项

TCP_KEEPALIVE选项只是一个开关,Linux中默认的Keepalive的选项如下: $sudo sysctl -a | grep keepalive net.ipv4.tcp_keepalive_time = 7200 net.ipv4.tcp_keepalive_probes = 9 net.ipv4.tcp_keepalive_intvl = 75 上文中的keepalive选项表示如果一个连接上7200s后没有任何数据发送,则设置了这个选项的本端向对端发送keepalive保活报

TCP协议选项

原文转自:http://blog.chinaunix.net/uid-20249205-id-1713871.html Kind   Meaning                               Reference----   -------------------------------   ---------  0       End of Option List                 [RFC793]  1       No-Operation           

TCP连接建立系列 — 客户端接收SYNACK和发送ACK

主要内容:客户端接收SYNACK.发送ACK,完成连接的建立. 内核版本:3.15.2 我的博客:http://blog.csdn.net/zhangskd 接收入口 tcp_v4_rcv |--> tcp_v4_do_rcv |-> tcp_rcv_state_process |-> tcp_rcv_synsent_state_process 1. 状态为ESTABLISHED时,用tcp_rcv_established()接收处理. 2. 状态为LISTEN时,说明这个sock处于监

TCP连接建立系列 — 服务端接收ACK段(一)

http://blog.csdn.net/zhangskd/article/details/17923917 分类: Linux TCP/IP Linux Kernel 2014-01-07 09:46 2311人阅读 评论(2) 收藏 举报 TCPIPlinux内核 目录(?)[+] 本文主要分析:三次握手中最后一个ACK段到达时,服务器端的处理路径. 内核版本:3.6 Author:zhangskd @ csdn blog 函数路径 以下是第三次握手时,服务端接收到ACK后的处理路径. 接收

TCP_DEFER_ACCEPT的坑

我实现了一个server,支持HTTP协议和内部私有协议,为了简化部署,我设计成一个端口同时兼容两种协议的客户端.根据连接后到达的消息头自动识别客户端协议.这种事情的传统做法是,accept后加入epoll,当fd第一次可读时,读出一些并解析,判断协议类型. 创建相应的上下文对象,开始服务.这样就引入了中间状态,为了省事,我用了TCP_DEFER_ACCEPT来简化这个过程. TCP_DEFER_ACCEPT,是Linux下的socket支持一个tcp选项,man这么说的: TCP_DEFER_

服务器重复发送SYN ACK 和 TCP_DEFER_ACCEPT设置

现象: 以下为其他网站提供,和我遇到的情况一样. 就是服务器老是重复发送 SYN, ACK. 4414.229553  client -> server TCP 62464 > http [SYN] Seq=0 Win=65535 Len=0 MSS=1452 WS=3 TSV=116730231 TSER=04414.229633 server -> client  TCP http > 62464 [SYN, ACK] Seq=0 Ack=1 Win=5792 Len=0 MS

TCP/IP协议族

TCP/IP协议 TCP/IP协议栈主要分为四层:应用层.传输层.网络层.数据链路层,每层都有相应的协议,现在几乎所有的操作系统都实现了TCP/IP协议栈.如下图 上图运输层应为传输层 所谓的协议就是双方进行数据传输的一种格式.整个网络中使用的协议有很多,所幸的是每一种协议都有RFC文档.在这里只对IP.TCP.UDP协议头做一个分析. 首先来看看在网络中,一帧 以太网数据包的格式: 在Linux 操作系统中,当我们想发送数据的时候,我们只需要在上层准备好数据,然后提交给内核协议栈 , 内核协议

《TCP/IP详解卷1:协议》第17、18章 TCP:传输控制协议(2)-读书笔记

章节回顾: <TCP/IP详解卷1:协议>第1章 概述-读书笔记 <TCP/IP详解卷1:协议>第2章 链路层-读书笔记 <TCP/IP详解卷1:协议>第3章 IP:网际协议(1)-读书笔记 <TCP/IP详解卷1:协议>第3章 IP:网际协议(2)-读书笔记 <TCP/IP详解卷1:协议>第4章 ARP:地址解析协议-读书笔记 <TCP/IP详解卷1:协议>第5章 RARP:逆地址解析协议-读书笔记 <TCP/IP详解卷1:协