TCP ACK响应延迟10ms

时间:2013-07-24 21:19:01

标签: linux networking tcp delay throughput

我正在编写一个应该从服务器接收文件的客户端应用程序。通信使用通过TCP设计的非常简单的专有协议:服务器发送一大块16KB的数据,等待来自我的客户端的简单响应,然后进入下一个块;这样做直到文件完全传输完毕。

服务器正在Windows上运行。如果我在Windows上运行我的客户端(通过Cygwin),我的吞吐量达到70Mbits / s。但是,如果我在Linux上运行它,我可以获得的最大吞吐量是5Mbits / s(两个测试使用相同的100Mb以太网链路)。

在对我收集的tcpdump跟踪进行一些调查之后,我发现Linux需要大约10ms才能回答每个TCP ACK。我每次调用recv()后都试图禁用sockoption TCP_QUICKACK,但没有成功。我没有找到任何可以改变这种行为的其他TCP选项(在bsd套接字下,甚至在Linux下)。

任何人都可以帮我弄清楚发生了什么吗?

  1. 我的客户端是用C ++编写的,使用BSD套接字。在其上设置了TCP_NODELAY sockoption。
  2. 我的linux内核是2.6.19
  3. 专有协议和服务器不是我的,所以我不能以任何方式改变它们。
  4. 下面是我分析的tcpdump跟踪示例。这是一个块传输。服务器IP为10.200.252.1,客户端为10.200.252.2。以**开头的行是延迟10ms的TCP ACK数据包:

       5184 28.060969   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4736738 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5185 28.060978   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4738182 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5186 28.060981   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4739626 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5187 28.060987   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4741070 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5188 28.060990   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4742514 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5189 28.060994   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4743958 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5190 28.060997   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4745402 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5191 28.061000   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4746846 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5192 28.061003   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4748290 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5193 28.061007   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4749734 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5194 28.061038   10.200.252.1          10.200.252.2          TCP      1510   search-agent > 58358 [ACK] Seq=4751178 Ack=1741 Win=66048 Len=1444 TSval=2951472 TSecr=2921060
       5195 28.061042   10.200.252.1          10.200.252.2          TCP      572    search-agent > 58358 [PSH, ACK] Seq=4752622 Ack=1741 Win=66048 Len=506 TSval=2951472 TSecr=2921060
     **5196 28.068422   10.200.252.2          10.200.252.1          TCP      66     58358 > search-agent [ACK] Seq=1741 Ack=4742514 Win=64512 Len=0 TSval=2921061 TSecr=2951472
     **5197 28.078479   10.200.252.2          10.200.252.1          TCP      66     58358 > search-agent [ACK] Seq=1741 Ack=4753128 Win=64512 Len=0 TSval=2921062 TSecr=2951472
       5198 28.082418   10.200.252.2          10.200.252.1          TCP      70     58358 > search-agent [PSH, ACK] Seq=1741 Ack=4753128 Win=64512 Len=4 TSval=2921062 TSecr=2951472
       5199 28.082675   10.200.252.2          10.200.252.1          TCP      68     58358 > search-agent [PSH, ACK] Seq=1745 Ack=4753128 Win=64512 Len=2 TSval=2921062 TSecr=2951472
       5200 28.082714   10.200.252.1          10.200.252.2          TCP      66     search-agent > 58358 [ACK] Seq=4753128 Ack=1747 Win=66048 Len=0 TSval=2951474 
    

1 个答案:

答案 0 :(得分:0)

完全摆脱分块和应用程序ACK。如果必须的话,只需在文件末尾有一个最后一个。 TCP已经进行了分块和确认,你正在疯狂,例如它必须确认ACK。不需要所有这些复杂性。这是一个流媒体协议:像一个一样使用它。