国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Golang > 正文

golang如何利用原始套接字構造UDP包詳解

2020-04-01 19:02:56
字體:
來源:轉載
供稿:網友

前言

本文主要給大家介紹了關于golang用原始套接字構造UDP包的相關內容,分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹吧。

RAW SOCKET 介紹

TCP/IP協議中,最常見的就是原始(SOCKET_RAW)、tcp(SOCKET_STREAM)、udp(SOCKET_DGRA)三種套接字。原始套接字能夠對底層傳輸進行控制,允許自行組裝數據包,比如修改本地IP,發送Ping包,進行網絡監聽。這里不做詳細介紹,要了解更多可以網上自己查詢。

實現

這里先看IP頭結構:

golang,udp,udp套接字編程,udp套接字

其中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頭結構:

golang,udp,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。

結語

這里只給出了UDP的實現,TCP的實現比較復雜,以后也會給出TCP實現的例子。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。

 

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 大渡口区| 古蔺县| 北宁市| 阿克| 新巴尔虎左旗| 临湘市| 合阳县| 望都县| 连云港市| 礼泉县| 芷江| 酒泉市| 宜君县| 铜陵市| 华亭县| 友谊县| 双牌县| 广元市| 津市市| 英山县| 阿克陶县| 启东市| 郎溪县| 政和县| 建宁县| 扎兰屯市| 四平市| 云南省| 延寿县| 达孜县| 明光市| 沙雅县| 图片| 宝应县| 建始县| 灵山县| 临澧县| 龙岩市| 广西| 永泰县| 萨嘎县|