小編給大家分享一下golang利用原始套接字構造UDP包的示例,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
創新互聯建站是一家從事企業網站建設、成都網站設計、成都網站建設、外貿網站建設、行業門戶網站建設、網頁設計制作的專業網站設計公司,擁有經驗豐富的網站建設工程師和網頁設計人員,具備各種規模與類型網站建設的實力,在網站建設領域樹立了自己獨特的設計風格。自公司成立以來曾獨立設計制作的站點千余家。
RAW SOCKET 介紹
TCP/IP協議中,最常見的就是原始(SOCKET_RAW)、tcp(SOCKET_STREAM)、udp(SOCKET_DGRA)三種套接字。原始套接字能夠對底層傳輸進行控制,允許自行組裝數據包,比如修改本地IP,發送Ping包,進行網絡監聽。這里不做詳細介紹,要了解更多可以網上自己查詢。
實現
這里先看IP頭結構:
其中16位總長度包括IP頭長度和數據的長度,8位協議填寫17,因為UDP協議類型為17。這里要說明一下IP頭中的首部校驗,這個值只校驗IP頭部,不包含數據。
這里給出校驗算法,IP頭和UDP頭中使用的校驗算法是一樣的。
func checkSum(msg []byte) uint16 { sum := 0 for n := 1; n < len(msg)-1; n += 2 { sum += int(msg[n])*256 + int(msg[n+1]) } sum = (sum >> 16) + (sum & 0xffff) sum += (sum >> 16) var ans = uint16(^sum) return ans }
下面開始填充IP頭,這里使用了golang.org/x/net下的ipv4包
//目的IP dst := net.IPv4(192, 168, 1, 2) //源IP src := net.IPv4(192, 168, 1, 3) //填充ip首部 iph := &ipv4.Header{ Version: ipv4.Version, //IP頭長一般是20 Len: ipv4.HeaderLen, TOS: 0x00, //buff為數據 TotalLen: ipv4.HeaderLen + len(buff), TTL: 64, Flags: ipv4.DontFragment, FragOff: 0, Protocol: 17, Checksum: 0, Src: src, Dst: dst, } h, err := iph.Marshal() if err != nil { log.Fatalln(err) } //計算IP頭部校驗值 iph.Checksum = int(checkSum(h))
下面開始處理UDP頭部,先來看UDP頭結構:
UDP頭結構就很簡單了,16位UDP校驗和涉及到一個UDP偽首部的東西,我們先來看下UDP偽首部的構成。
----------------------------------------- | 32bit Source IP address | ----------------------------------------- | 32bit Destination IP addr | ----------------------------------------- | 0 | 8bit Proto| 16bit header length| -----------------------------------------
偽首部包含了源IP,目的IP,協議號,16位的長度。這個偽首部僅僅參與校驗計算。
下面開始填充UDP頭:
//填充udp首部 //udp偽首部 udph := make([]byte, 20) //源ip地址 udph[0], udph[1], udph[2], udph[3] = src[12], src[13], src[14], src[15] //目的ip地址 udph[4], udph[5], udph[6], udph[7] = dst.IP[12], dst.IP[13], dst.IP[14], dst.IP[15] //協議類型 udph[8], udph[9] = 0x00, 0x11 //udp頭長度 udph[10], udph[11] = 0x00, byte(len(buff)+8) //下面開始就真正的udp頭部 //源端口號 udph[12], udph[13] = 0x27, 0x10 //目的端口號 udph[14], udph[15] = 0x17, 0x70 //udp頭長度 udph[16], udph[17] = 0x00, byte(len(buff)+8) //校驗和 udph[18], udph[19] = 0x00, 0x00 //計算校驗值 check := checkSum(append(udph, buff...)) udph[18], udph[19] = byte(check>>8&255), byte(check&255)
下面我們需要發送自己構造的UDP包,可以使用net下的ListenPacket。
listener, err := net.ListenPacket("ip4:udp", "192.168.1.104") if err != nil { log.Fatal(err) } defer listener.Close() //listener 實現了net.PacketConn接口 r, err := ipv4.NewRawConn(c) if err != nil { log.Fatal(err) } //發送自己構造的UDP包 if err = r.WriteTo(iph, append(udph[12:20], buff...), nil); err != nil { log.Fatal(err) }
這個實現只在linux和mac上測試過,windows上需要借助于第三方吧,比如winpcap。
以上是“golang利用原始套接字構造UDP包的示例”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注創新互聯行業資訊頻道!
分享名稱:golang利用原始套接字構造UDP包的示例
文章源于:http://vcdvsql.cn/article38/jhjdpp.html
成都網站建設公司_創新互聯,為您提供品牌網站建設、網站改版、建站公司、電子商務、虛擬主機、網站導航
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯