分析TCP/IP協(xié)議棧代碼之IP & ICMP(STM32平臺)
1. IP介紹
IP是TCP/IP協(xié)議族中最為核心的協(xié)議。大家,如TCP、UDP、ICMP及IGMP數(shù)據(jù),都是在IP數(shù)據(jù)報格式基礎(chǔ)上再封裝一層再來傳輸?shù)模ㄒ妶D1 - 4)。
不可靠(unreliable)的意思是它不能保證 IP數(shù)據(jù)報能成功地到達(dá)目的地。 IP僅提供最好的傳輸服務(wù)。如果發(fā)生某種錯誤時,如某個路由器暫時用完了緩沖區(qū), IP有一個簡單的錯誤處理算法:丟棄該數(shù)據(jù)報,然后發(fā)送 ICMP消息報給信源端。任何要求的可靠性必須由上層來提供(如TCP) 。
無連接(connectionless)這個術(shù)語的意思是I P并不維護任何關(guān)于后續(xù)數(shù)據(jù)報的狀態(tài)信息。每個數(shù)據(jù)報的處理是相互獨立的。這也說明, IP數(shù)據(jù)報可以不按發(fā)送順序接收。如果一信源向相同的信宿發(fā)送兩個連續(xù)的數(shù)據(jù)報(先是 A,然后是B) ,每個數(shù)據(jù)報都是獨立地進行路由選擇,可能選擇不同的路線,因此B可能在A到達(dá)之前先到達(dá)。
2. IP首部
I P數(shù)據(jù)報的格式如圖3 - 1所示。普通的IP首部長為20個字節(jié),除非含有選項字段。
分析圖3 - 1中的首部。最高位在左邊,記為0 bit;最低位在右邊,記為31 bit。4個字節(jié)的32 bit值以下面的次序傳輸:首先是0~7 bit,其次8~15 bit,然后1 6~23 bit,最后是24~31 bit。這種傳輸次序稱作big endian字節(jié)序。由于TCP/IP首部中所有的二進制整數(shù)在網(wǎng)絡(luò)中傳輸時都要求以這種次序,因此它又稱作網(wǎng)絡(luò)字節(jié)序。
目前的協(xié)議版本號是4,因此IP有時也稱作IPv4。
服務(wù)類型(TOS)字段包括一個3 bit的優(yōu)先權(quán)子字段(現(xiàn)在已被忽略) ,4 bit的TO S子字段和1 bit未用位但必須置0。4 bit的TO S分別代表:最小時延、最大吞吐量、最高可靠性和最小費用。
總長度字段是指整個I P數(shù)據(jù)報的長度,以字節(jié)為單位。
標(biāo)識字段唯一地標(biāo)識主機發(fā)送的每一份數(shù)據(jù)報。通常每發(fā)送一份報文它的值就會加 1。在大多數(shù)從伯克利派生出來的系統(tǒng)中,每發(fā)送一個I P數(shù)據(jù)報,I P層都要把一個內(nèi)核變量的值加1,不管交給IP的數(shù)據(jù)來自哪一層。內(nèi)核變量的初始值根據(jù)系統(tǒng)引導(dǎo)時的時間來設(shè)置。
TTL(time-to-live)生存時間字段設(shè)置了數(shù)據(jù)報可以經(jīng)過的最多路由器數(shù)。TTL的初始值由源主機設(shè)置(通常為3 2或6 4) ,一旦經(jīng)過一個處理它的路由器,它的值就減去1。當(dāng)該字段的值為0時,數(shù)據(jù)報就被丟棄,并發(fā)送 ICMP報文通知源主機。
協(xié)議字段可以識別是哪個協(xié)議向I P傳送數(shù)據(jù)。
首部檢驗和字段是根據(jù)IP首部計算的檢驗和碼。它不對首部后面的數(shù)據(jù)進行計算。 ICMP、IGMP、UDP和TCP在它們各自的首部中均含有同時覆蓋首部和數(shù)據(jù)檢驗和碼。
目前,這些任選項定義如下:
? 安全和處理限制(用于軍事領(lǐng)域,詳細(xì)內(nèi)容參見 RFC 1108[Kent 1991])
? 記錄路徑(讓每個路由器都記下它的IP地址,見7 . 3節(jié))
? 時間戳(讓每個路由器都記下它的IP地址和時間,見7 . 4節(jié))
? 寬松的源站選路(為數(shù)據(jù)報指定一系列必須經(jīng)過的 IP地址,見8 . 5節(jié))
? 嚴(yán)格的源站選路(與寬松的源站選路類似,但是要求只能經(jīng)過指定的這些地址,不能經(jīng)過其他的地址) 。
------------------------------------------以上內(nèi)容整理于《TCP/IP協(xié)議詳解:卷1》--------------------------------------
------------------------------------------以下內(nèi)容產(chǎn)生于代碼及分析--------------------------------------
3. IP宏定義實現(xiàn)
C++ Code
4. IP函數(shù)實現(xiàn)
以太網(wǎng)的header在IP的header之前,很簡單的,介紹先。
配置以太網(wǎng)的頭,為14字節(jié):6字節(jié)目的mac地址+6字節(jié)源mac地址+2字節(jié)協(xié)議類型
展開之后如下所示,其在以太網(wǎng)幀中的位置與之前的宏定義是一一對應(yīng)的。
IP與ARP一樣,需要判定是不是發(fā)給本機的(eth_type_is_ip_and_my_ip函數(shù)),還有與填充make_eth 函數(shù)一樣需要填充函數(shù)(make_ip函數(shù)),此外還有填充其他雜七雜八和16位首部校驗和函數(shù)(fill_ip_hdr_checksum函數(shù))
C++ Code
5. IP校驗和實現(xiàn)
校驗和函數(shù)式如何得出校驗和值的呢?看《TCP/IP協(xié)議詳解:卷1》里面咋說的吧。
”為了計算一份數(shù)據(jù)報的 IP檢驗和,首先把檢驗和字段置為 0。然后,對首部中每個 16 bit進行二進制反碼求和(整個首部看成是由一串 16 bit的字組成) ,結(jié)果存在檢驗和字段中。當(dāng)收到一份I P數(shù)據(jù)報后,同樣對首部中每個16 bit進行二進制反碼的求和。由于接收方在計算過程中包含了發(fā)送方存在首部中的檢驗和,因此,如果首部在傳輸過程中