主頁 > 知識庫 > golang如何利用原始套接字構(gòu)造UDP包詳解

golang如何利用原始套接字構(gòu)造UDP包詳解

熱門標簽:濱州自動電銷機器人排名 惠州電銷防封電話卡 黃岡人工智能電銷機器人哪個好 浙江高頻外呼系統(tǒng)多少錢一個月 鄭州亮點科技用的什么外呼系統(tǒng) 建造者2地圖標注 阿里云ai電話機器人 汕頭小型外呼系統(tǒng) 釘釘有地圖標注功能嗎

前言

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

RAW SOCKET 介紹

TCP/IP協(xié)議中,最常見的就是原始(SOCKET_RAW)、tcp(SOCKET_STREAM)、udp(SOCKET_DGRA)三種套接字。原始套接字能夠?qū)Φ讓觽鬏斶M行控制,允許自行組裝數(shù)據(jù)包,比如修改本地IP,發(fā)送Ping包,進行網(wǎng)絡監(jiān)聽。這里不做詳細介紹,要了解更多可以網(wǎng)上自己查詢。

實現(xiàn)

這里先看IP頭結(jié)構(gòu):

其中16位總長度包括IP頭長度和數(shù)據(jù)的長度,8位協(xié)議填寫17,因為UDP協(xié)議類型為17。這里要說明一下IP頭中的首部校驗,這個值只校驗IP頭部,不包含數(shù)據(jù)。

這里給出校驗算法,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為數(shù)據(jù)
  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頭結(jié)構(gòu):

UDP頭結(jié)構(gòu)就很簡單了,16位UDP校驗和涉及到一個UDP偽首部的東西,我們先來看下UDP偽首部的構(gòu)成。

-----------------------------------------
|   32bit Source IP address  |
-----------------------------------------
|   32bit Destination IP addr  |
-----------------------------------------
| 0 | 8bit Proto| 16bit header length|
-----------------------------------------

偽首部包含了源IP,目的IP,協(xié)議號,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]
 //協(xié)議類型
 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>>8255), byte(check255)

下面我們需要發(fā)送自己構(gòu)造的UDP包,可以使用net下的ListenPacket。

 listener, err := net.ListenPacket("ip4:udp", "192.168.1.104")
 if err != nil {
  log.Fatal(err)
 }
 defer listener.Close()
 
 //listener 實現(xiàn)了net.PacketConn接口
 r, err := ipv4.NewRawConn(c)
 if err != nil {
  log.Fatal(err)
 }

 //發(fā)送自己構(gòu)造的UDP包
 if err = r.WriteTo(iph, append(udph[12:20], buff...), nil); err != nil {
  log.Fatal(err)
 }

這個實現(xiàn)只在linux和mac上測試過,windows上需要借助于第三方吧,比如winpcap。

結(jié)語

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

總結(jié)

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

標簽:東營 阿壩 駐馬店 滄州 瀘州 昭通 晉中 泰安

巨人網(wǎng)絡通訊聲明:本文標題《golang如何利用原始套接字構(gòu)造UDP包詳解》,本文關(guān)鍵詞  golang,如何,利用,原始,套接字,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《golang如何利用原始套接字構(gòu)造UDP包詳解》相關(guān)的同類信息!
  • 本頁收集關(guān)于golang如何利用原始套接字構(gòu)造UDP包詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章