9

我想使用 gopacket 制作自定义 TCP 数据包,然后使用原始套接字发送它们。

这是一个简短易读的示例 go 程序,演示了我想做的事情:

package main

import (
    "code.google.com/p/gopacket"
    "code.google.com/p/gopacket/examples/util"
    "code.google.com/p/gopacket/layers"
    "log"
    "net"
)

func main() {
    defer util.Run()()

    // XXX create tcp/ip packet
    srcIP := net.ParseIP("127.0.0.1")
    dstIP := net.ParseIP("192.168.0.1")
    //srcIPaddr := net.IPAddr{
    //  IP: srcIP,
    //}
    dstIPaddr := net.IPAddr{
        IP: dstIP,
    }
    ipLayer := layers.IPv4{
        SrcIP:    srcIP,
        DstIP:    dstIP,
        Protocol: layers.IPProtocolTCP,
    }
    tcpLayer := layers.TCP{
        SrcPort: layers.TCPPort(666),
        DstPort: layers.TCPPort(22),
        SYN:     true,
    }
    tcpLayer.SetNetworkLayerForChecksum(&ipLayer)
    buf := gopacket.NewSerializeBuffer()
    opts := gopacket.SerializeOptions{
        FixLengths:       true,
        ComputeChecksums: true,
    }
    err := gopacket.SerializeLayers(buf, opts, &ipLayer, &tcpLayer)
    if err != nil {
        panic(err)
    }
    // XXX end of packet creation

    // XXX send packet
    ipConn, err := net.ListenPacket("ip4:tcp", "0.0.0.0")
    if err != nil {
        panic(err)
    }
    _, err = ipConn.WriteTo(buf.Bytes(), &dstIPaddr)
    if err != nil {
        panic(err)
    }
    log.Print("packet sent!\n")
}

但是运行这个程序不起作用...... SerializeLayer 失败。这是恐慌:

恐慌:src IP 127.0.0.1 无效

goroutine 16 [运行]: runtime.panic(0x5bb020, 0xc2090723e0) /home/human/golang-empire/go/src/pkg/runtime/panic.c:279 +0xf5 main.main() /home/human/golang-帝国/gopkg/src/github.com/david415/HoneyBadger/packetSendTest.go:41 +0x464

goroutine 19 [finalizer wait]: runtime.park(0x413cc0, 0x7bc6c0, 0x7bb189) /home/human/golang-empire/go/src/pkg/runtime/proc.c:1369 +0x89 runtime.parkunlock(0x7bc6c0, 0x7bb189) / home/human/golang-empire/go/src/pkg/runtime/proc.c:1385 +0x3b runfinq() /home/human/golang-empire/go/src/pkg/runtime/mgc0.c:2644 +0xcf runtime.goexit() /home/human/golang-empire/go/src/pkg/runtime/proc.c:1445

4

1 回答 1

11

您的问题是“制作客户 [原文如此] TCP 数据包”,但您的代码清楚地表明您还想制作自定义 IP 第 3 层标头,两者之间存在差异。此外,您没有提到 IPv4 与 IPv6,但您的代码再次似乎是 IPv4 特定的。

鉴于您的示例代码,我假设您想要设置完整的 IPv4 标头。

从 Go 1.3.3 和即将发布的 Go 1.4 开始,你无法使用 Core Go 包做你想做的事。为了实现你想要的,你需要做两件事:

  1. 您需要在 Go 中创建一个原始套接字。net.ListenPacket与本网站上的其他错误答案相反,您可以使用、net.DialIP或之一在 Go 中创建原始套接字net.ListenIP

例如:

conn, err := net.ListenIP("ip4:tcp", netaddr)
if err != nil {
    log.Fatalf("ListenIP: %s\n", err)
}

创建一个原始套接字。

  1. 如果要设置自己的 IPv4 第 3 层标头,则需要设置套接字选项以启用功能。

您的问题没有说明您使用的是什么操作系统和架构。在我运行 Mac OS X 的笔记本电脑上:

% man ip
. . .
Outgoing packets automatically have an IP header prepended to them
(based on the destination address and the protocol number the socket is created with),
unless the IP_HDRINCL option has been set.

IP_HDRINCL在 Linux 上也可以使用。不幸的是,Core Go 没有办法设置IP_HDRINCL套接字选项,也没有办法设置其他 IP 套接字选项,例如IP_TTL. 我有一组私有补丁可以在 Core Go 中启用此功能,但这对您没有帮助。

我相信以下软件包具有您想要的所有功能ipv4。请注意,它是一个大包装,我自己没有使用过。我做了 grep,它支持IP_HDRINCL多个平台。你想调用NewRawConn来创建一个原始连接,这个函数创建一个原始套接字并设置IP_HDRINCL套接字选项。

另请参见此处:raw-sockets-in-go和他在此处编写的代码延迟,以了解一种更简单的方法,如果您只想设置 TCP 标头,可能会满足您的需求。但是,请注意此代码不允许您在 IPv4 IP 标头中设置 IP 地址,我怀疑您想要这样做。

于 2014-12-01T08:00:19.323 回答