日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

TCP queue 的一些问题

發(fā)布時間:2023/12/20 编程问答 26 豆豆
生活随笔 收集整理的這篇文章主要介紹了 TCP queue 的一些问题 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

轉(zhuǎn)自Jasey Wang的blog,原文地址

首先回顧下三次握手里面涉及到的問題:

  • 當 client 通過 connect 向 server 發(fā)出 SYN 包時,client 會維護一個 socket 等待隊列,而 server 會維護一個 SYN 隊列
  • 此時進入半鏈接的狀態(tài),如果 socket 等待隊列滿了,server 則會丟棄,而 client 也會由此返回 connection time out;只要是 client 沒有收到 SYN+ACK,3s 之后,client 會再次發(fā)送,如果依然沒有收到,9s 之后會繼續(xù)發(fā)送
  • 半連接 syn 隊列的長度為 max(64, /proc/sys/net/ipv4/tcp_max_syn_backlog) 決定
  • 當 server 收到 client 的 SYN 包后,會返回 SYN, ACK 的包加以確認,client 的 TCP 協(xié)議棧會喚醒 socket 等待隊列,發(fā)出 connect 調(diào)用.
  • client 返回 ACK 的包后,server 會進入一個新的叫 accept 的隊列,該隊列的長度為 min(backlog, somaxconn)+1,默認情況下,somaxconn 的值為 128,表示最多有 129 的 ESTAB 的連接等待 accept(),而 backlog 的值則由 int listen(int sockfd, int backlog) 中的第二個參數(shù)指定,listen 里面的 backlog 的含義請看這里。
  • 當 accept 隊列滿了之后,即使 client 繼續(xù)向 server 發(fā)送 ACK 的包,也會不被響應(yīng),此時,server 通過 /proc/sys/net/ipv4/tcp_abort_on_overflow 來決定如何返回,0 表示直接丟丟棄該 ACK,1 表示發(fā)送 RST 通知 client;相應(yīng)的,client 則會分別返回 read timeout 或者 connection reset by peer。上面說的只是些理論,如果服務(wù)器不及時的調(diào)用 accept(),當 queue 滿了之后,服務(wù)器并不會按照理論所述,不再對 SYN 進行應(yīng)答,返回 ETIMEDOUT。根據(jù)這篇文檔的描述,實際情況并非如此,服務(wù)器會隨機的忽略收到的 SYN,建立起來的連接數(shù)可以無限的增加,只不過客戶端會遇到延時以及超時的情況。
  • 可以看到,整個 TCP stack 有如下的兩個 queue:

  • 一個是 half open(syn queue) queue(max(tcp_max_syn_backlog, 64)),用來保存 SYN_SENT 以及 SYN_RECV 的信息。
  • 另外一個是 accept queue(min(somaxconn, backlog)+1),保存 ESTAB 的狀態(tài),但是調(diào)用 accept()。
  • LISTEN 狀態(tài): Recv-Q 表示的當前等待服務(wù)端調(diào)用 accept 完成三次握手的 listen backlog 數(shù)值,也就是說,當客戶端通過 connect() 去連接正在 listen() 的服務(wù)端時,這些連接會一直處于這個 queue 里面直到被服務(wù)端 accept();Send-Q 表示的則是最大的 listen backlog 數(shù)值,這就就是上面提到的 min(backlog, somaxconn) 的值。
    其余狀態(tài): 非 LISTEN 狀態(tài)之前理解的沒有問題。Recv-Q 表示 receive queue 中的 bytes 數(shù)量;Send-Q 表示 send queue 中的 bytes 數(shù)值。

    要理解上面總結(jié)的這些,可以參見下這兩個案例(1, 2)。

    通過 "SYNs to LISTEN sockets dropped" 以及 "times the listen queue of a socket overflowed" 這兩個 netstat -s 獲取到的 TCP 狀態(tài),可以很快的發(fā)現(xiàn)系統(tǒng)存在的一些問題。
    任何一個包含 "dropped" 或者 "overflowed" 并且數(shù)值一直居高不下的 metric 從字面含義理解來看,都不是一個好現(xiàn)象。

    對于 Nginx 來說,backlog 的默認值為 511,這個可以通過 ss/netstat 的 Send-Q 確認:
    State Recv-Q Send-Q Local Address:Port Peer Address:Port
    LISTEN 0 511 :80 :*

    可以通過適當?shù)脑龃?nginx 的 backlog 以及 somaxconn 來增大隊列:
    listen 80 backlog=1638

    上面說了這么多,其實就是為了引入下面這個問題。
    我們線上一個基于 Netty 的代碼,3.5.12 的版本,監(jiān)控顯示 "times the listen queue of a socket overflowed" 常年居高不下,動輒幾十 K,通過 ss,我們發(fā)現(xiàn)其 backlog 的值只有 50:
    Recv-Q Send-Q Local Address:Port Peer Address:Port
    0 50 :6928 :* users:(("java",454409,196))

    g 了一下,發(fā)現(xiàn)這個版本復(fù)用了 Java 默認的 50 這個值。將其增加到 1024 測試,監(jiān)控曲線一下子降低到了 0。

    除了上面這些,還有一個比較基礎(chǔ)的 net.core.netdev_max_backlog,如果內(nèi)核接受包的速度大于被 userspace 處理的速度,該值定義了可以在接口輸入最大的的包數(shù)量。

    chartbeat 分享了兩篇很精彩的文檔,其中涉及到了 queue 的一些問題。
    Lessons learned tuning TCP and Nginx in EC2 1
    Lessons learned tuning TCP and Nginx in EC2 2
    參考鏈接https://www.cnblogs.com/jcli/p/3911505.html

    轉(zhuǎn)載于:https://www.cnblogs.com/johnsonjie/p/10300987.html

    總結(jié)

    以上是生活随笔為你收集整理的TCP queue 的一些问题的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。