• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            牽著老婆滿街逛

            嚴以律己,寬以待人. 三思而后行.
            GMail/GTalk: yanglinbo#google.com;
            MSN/Email: tx7do#yahoo.com.cn;
            QQ: 3 0 3 3 9 6 9 2 0 .

            libjingle源碼解析(3)-【PseudoTcp】建立UDP之上的TCP(1):連接和關閉

            轉載自:http://blog.csdn.net/leehark/article/details/7654183

            PseudoTcp - 建立UDP之上的TCP1):連接和關閉

            mail:lihe21327 [at] gmail [dot] com

            最近閱讀了LibjinglePseudoTcp.LibJingle很是下功夫做P2P了,在UDP之上做了可靠的傳輸協(xié)議PseudoTcp.

            了解PseudoTcp之前,我們需要了解一些TCP的特性。


            根據(jù)《TCP/IP詳解》卷1,可以總結如下:

            1.TCP是面相連接的,他需要3次握手和4次終止過程。

            2.TCP支持Nangle算法和經受時延的確認來控制報文段數(shù)目。

            3.TCP含有滑動窗口來控制接收方的流量。

            4.TCP支持超時與重傳。

            5.TCP支持擁塞避免算法。

            6.TCP具有堅持定時器和?;疃〞r器

            7.TCP要支持路徑MTU發(fā)現(xiàn)、長肥管道、時間戳選項。

            那我們一起剖析一下PseudoTcp實現(xiàn)了上面哪些功能。

            PseudoTcp(以后簡稱PTCP吧)的格式:

            通過結構Segment 定義此報文頭部:

            struct Segment {

                uint32 convseqack;

                uint8 flags;

                uint16 wnd;

                const char * data;

                uint32 len;

                uint32 tsvaltsecr;

              };

            各個字段的含義如下:

              A)Conversation Number : 流水號,是用來標識此次連接。即TCP里所謂的本地IP:本地端口-遠程IP:遠程端口,4組合為一個流水號。因為PTCPUDP之上的(當然也可以是其他協(xié)議之上),如果socket沒有綁定到本地端口,可能獲取的不是需要的數(shù)據(jù)。如果獲取的Conv Number不一樣,接收方會發(fā)送RST(不過PTCP里已經注釋了此段代碼)。此外,PTCP并不關心他的傳輸層是有一個連接還是多個連接,她只關心CONV Number是否一致。

              B)Seq Number:32位序號,即此數(shù)據(jù)表示的序列,不一定從0開始

              C)Ack Number:32確認序列號。確認已經獲取到的數(shù)據(jù)序列加1,即下一個需要接受的序列號。

              D)Control:現(xiàn)未使用

              E)URG:緊急指針,1bit

              F)ACK:確認序列號有效,1bit

              G)PSH:接收方盡可能將這個報文送給應用層,1bit

              H)RST:重置連接

              I)FIN:表示發(fā)送完所有數(shù)據(jù),斷開連接。

              J)Window:窗口大小

              K)TimeStamp Sending:本端發(fā)送包時間(采用以本端的時間計算方式)

              L)TimeStamp Receiving:對方最近接收包時間(采用以對方的時間計算方式)

              M)Data:數(shù)據(jù)

              注:上面的E-I的含義,在實現(xiàn)上完全不同。下面會提到。


            PTCP的狀態(tài):

              TCP_LISTEN:監(jiān)聽

              TCP_SYN_SENTSYN包已經發(fā)送

              TCP_SYN_RECEIVED:已經接收SYN

              TCP_ESTABLISHED:已經建立連接

              TCP_CLOSED:已經關閉連接


            PTCP的狀態(tài)轉移相對TCP來說簡單多了,TCP如下:

             

            3路握手:

            TCP建議連接時需要來回總共有3TCP包來做握手,即

              A)SYN[A]:

              B)ACK[B],SYN[A+1]

              C)ACK[B+1]


            PTCP握手過程如下:

              當開始時兩端都處于TCP_LISTEN狀態(tài)。

              當C端發(fā)送SYN包到S端時,C端處于TCP_SYN_SENT狀態(tài)

              當S端處于TCP_LISTEN時收到SYN包,S端轉為TCP_SYN_RECEIVED

              當S端處于TCP_SYN_RECEIVED時,發(fā)送ACK時狀態(tài)不變

              當C端處于TCP_SYN_SENT時,收到ACK,則轉為TCP_ESTABLISHED

              當S端處于TCP_SYN_RECEIVED,收到非控制包時轉為TCP_ESTABLISHED

              這里解釋一下控制包:上面PTCP協(xié)議頭結構里的第13個字節(jié)處(即URG,ACK等在的字節(jié))其實只取3個值之一:

                0:數(shù)據(jù)包

                0x02CTL包,當握手時使用。

                0x04RST包?,F(xiàn)在發(fā)此段包的代碼被注釋掉。

            所以控制包,指的是握手時才會發(fā)送,握手完之后都屬于數(shù)據(jù)包。

            可見PTCP的握手過程和TCP的握手過程有微小的差異。當C端轉為TCP_ESTABLISHED后,等到有數(shù)據(jù)才會發(fā)送給S端(而不是立即),S端直到只有等到有數(shù)據(jù)的包時,才把狀態(tài)改為TCP_ESTABLISHED。而TCP是,如果沒有數(shù)據(jù)會立即發(fā)送,S端只要收到ACK就改為ESTABLISHED狀態(tài)。


            連接建立時超時:

            C端發(fā)送完SYN包之后,一直沒有響應時,沒過3SC端會發(fā)送一個SYN請求。直到發(fā)送30次之后,還沒有收到回包,則停止發(fā)送并關閉連接。即等待時間為3*30=90S,而大多數(shù)TCP實現(xiàn)的超時時間為75S。


            最大報文段長度(MSS)

            TCP默認MSS536,即取MTU576 X.25 Networks),包括20個字節(jié)的IP頭和20個字節(jié)的TCP頭。

            對于PTCP,默認MTU取為65536,即UDP容納的最大長度,那么MSS取值為65536-116

            116的計算來自:

              PACKET_OVERHEAD = HEADER_SIZE + UDP_HEADER_SIZE + IP_HEADER_SIZE + JINGLE_HEADER_SIZE

              JINGLE_HEADER_SIZE用于Relay包,具體需要了解STUN協(xié)議和TURN協(xié)議。

            MTU的發(fā)現(xiàn)完全由調用方來決定,PTCP只提供了接口來更新MTU。

            Libjingle里,對于win32,枚舉下面數(shù)組PACKET_MAXIMUMS,然后通過WinPing來發(fā)現(xiàn)此次PTCP連接的MTU。如果沒有獲取到MTU,默認取值為1280(此時MSS1280-116=1164)。

            為什么MTU默認取值為1280呢,有什么數(shù)據(jù)依據(jù)呢?

            // Standard MTUs

              const uint16 PACKET_MAXIMUMS[] = {

                65535,    // Theoretical maximum, Hyperchannel

                32000,    // Nothing

                17914,    // 16Mb IBM Token Ring

                8166,   // IEEE 802.4

                //4464,   // IEEE 802.5 (4Mb max)

                4352,   // FDDI

                //2048,   // Wideband Network

                2002,   // IEEE 802.5 (4Mb recommended)

                //1536,   // Expermental Ethernet Networks

                //1500,   // Ethernet, Point-to-Point (default)

                1492,   // IEEE 802.3

                1006,   // SLIP, ARPANET

                //576,    // X.25 Networks

                //544,    // DEC IP Portal

                //512,    // NETBIOS

                508,    // IEEE 802/Source-Rt Bridge, ARCNET

                296,    // Point-to-Point (low delay)

                //68,     // Official minimum

                0,      // End of list marker

              };

            PTCP的關閉。

            TCP的關閉時由4步驟完成。


              1. FIN[A]

              2. ACK[A+1]

              3. FIN[B]

              4. ACK[B+1]

            然而,有時候可以做到3步,即上面的2,3步可以合成在一個TCP包里發(fā)送。對于上面只完成前兩步的狀態(tài)成為半關閉狀態(tài),此時發(fā)送FIN[A]的端表示自己不再有多余的數(shù)據(jù)要發(fā)送,但還能接收數(shù)據(jù)。

            當調用PTCPClose方法時,此端丟棄對方發(fā)過來的數(shù)據(jù),只做應答,即只發(fā)送對方發(fā)來數(shù)據(jù)的ACK。并且等到此方數(shù)據(jù)都發(fā)送完,需關閉整個連接。以此看來,PTCP沒有半關閉狀態(tài),并且PTCP也只是用來支持P2P用的,不需要半關閉狀態(tài)。

            2MSL等待狀態(tài)

            MSL是指一個數(shù)據(jù)包在網絡上存在的最長時間。而2MSL是指當主動關閉方發(fā)送被動關閉方發(fā)送的FIN對應的ACK時,如果這個ACK被丟失了,則被動關閉方超時重發(fā)最后的FIN,此時主動關閉方再次發(fā)ACK,當主動關閉方發(fā)送第一個FIN對應的ACK到,拿到最后的FIN之間的時間段最長為2MSL。那為什么主動關閉方處于2MSL等待狀態(tài)呢?是因為,如果主動關閉方發(fā)送了第一個FIN對應的ACK之后,放棄了此連接,那么下一個新建的連接有可能復用此連接(即同一個插口對),此時新建的連接有可能因為上一個丟失的ACK,而收到重發(fā)的FIN,導致連接被關閉。

            然而PTCP不存在半關閉的概念,故2MSL等待狀態(tài)也隨之沒有。此外,PTCP是用來做P2P的,兩者之間的連接時雙方協(xié)商定義的,并且PTCP在頭部給予了Conversation number的概念,以便協(xié)商中防止產生同一個連接的產生。

            復位報文段

            TCP存在如下情況時會產生復位報文段。

              A.當服務器沒有開啟指定的連接端口時,對于UDP來說產生端口不可達,而TCP產生RST報文

              B.當一端產生異常終止時,會發(fā)送RST報文。即當設置SO_LINGER套接口選項時,close套接口會產生RST報文。

              C.檢測到半打開連接,當接收方異常終止重啟后接收對方在舊的連接上傳過來的數(shù)據(jù)時,會發(fā)送RST報文。

            對于PTCP來說,現(xiàn)在沒有一個地方會發(fā)送RST報文(之前有過的被注釋了,當收到不是當前的CONV時會發(fā)送RST),但如果一旦收到了RST報文,則立即關閉此連接。

            同時打開

            TCP的同時打開情景是如下:當C用端口7777連接S的端口8888,同時S用端口8888連接C的端口7777,此時包的順序如下:

              1) SYN[A]

              2) SYN[B]

              3) ACK[A+1]

              4) ACK[B+1]

            顯然上面的握手從3次變?yōu)?/span>4次。

            PTCP的同時打開,也類似如上,由4個包來完成握手。

              1) C端發(fā)送SYN時,狀態(tài)變?yōu)?/span>TCP_SYN_SENT

              2) 同時S端發(fā)送SYNSC的狀態(tài)此時都為TCP_SYN_SENT

              3) C,S同時向對方(可以不是同時)發(fā)送ACK,此時CS狀態(tài)都變?yōu)?/span>TCP_ESTABLISHED。

            同時關閉

            TCP是支持C,S同時關閉的。

              1)C,S同時發(fā)送FIN,狀態(tài)變?yōu)?/span>FIN_WAIT_1

              2)C,S同時收到FIN,并發(fā)送ACK,狀態(tài)變?yōu)?/span>CLOSING

              3)C,S同時收到ACK,兩個狀態(tài)都變?yōu)?/span>TIME_WAIT

            對于PTCP,沒有像TCP,不存在FIN包,顯然對關閉狀態(tài)的維護不是很完美。也同樣,看不到同時關閉的情形,這些交給底層傳輸層(UDP)等之類來完成,由調用方來維護狀態(tài)。

            為什么PTCP沒有提供FIN報文以及對應的狀態(tài)呢?

            TCP選項

            TCP保留40個字節(jié)傳輸其他選項,主要有窗口擴大因子,時間戳選項,MSS長度等。

            PTCP也通過一種方式來增加其他選項,如MSS和窗口擴大因子。當傳輸?shù)氖强刂瓢矣袛?shù)據(jù)內容時,如果第一個字節(jié)為CTL_CONNECT,則會調用方法parseOptions來解析是否含有MSS,窗口擴大因子等等選項。這些選項的實現(xiàn)細節(jié)后續(xù)會提及(時間戳選項直接在報文頭里有,固這個選項很重要,后續(xù)會提到此選項的作用)。

            posted on 2013-09-01 14:05 楊粼波 閱讀(1069) 評論(0)  編輯 收藏 引用

            免费精品国产日韩热久久| 一级做a爰片久久毛片免费陪 | 午夜福利91久久福利| 久久亚洲精品国产精品婷婷| 亚洲午夜福利精品久久| 国产成人综合久久精品红| 色婷婷综合久久久久中文一区二区| 久久久黄色大片| 久久午夜无码鲁丝片| 青青草国产97免久久费观看| 热99RE久久精品这里都是精品免费 | 青青草国产精品久久| 国产亚洲精久久久久久无码77777 国产亚洲精品久久久久秋霞 | 久久久99精品成人片中文字幕 | 91亚洲国产成人久久精品| 久久免费视频1| 国产午夜精品久久久久九九| 热久久视久久精品18| 精品久久久无码中文字幕| 久久99精品国产自在现线小黄鸭| 久久久久综合国产欧美一区二区| 综合久久一区二区三区 | 久久久久人妻一区精品性色av| 97精品伊人久久久大香线蕉| 一日本道伊人久久综合影| 亚洲午夜久久久精品影院| 精品午夜久久福利大片| 97久久精品无码一区二区| 日韩人妻无码精品久久久不卡| 中文成人久久久久影院免费观看| 精品欧美一区二区三区久久久| 欧美精品一区二区精品久久| 久久免费高清视频| 国产99久久久久久免费看| 国产成人无码精品久久久免费| 国产成人精品久久一区二区三区av | 国产精品青草久久久久福利99| 精品国产福利久久久| 一本久久a久久精品vr综合| 日本精品久久久久影院日本| 亚洲va久久久噜噜噜久久|