go x/net/icmp:在Windows中监听ICMP时无法正常工作

1yjd4xko  于 4个月前  发布在  Go
关注(0)|答案(7)|浏览(45)

你正在使用的Go版本是什么( go version )?

$ go version
go version go1.13.8 darwin/amd64

这个问题在最新版本中是否会重现?

是的

你正在使用什么操作系统和处理器架构( go env )?

go env 输出

$ go env
GO111MODULE=""
GOARCH="amd64"
GOBIN=""
GOCACHE="/Users/censored/Library/Caches/go-build"
GOENV="/Users/censored/Library/Application Support/go/env"
GOEXE=""
GOFLAGS=""
GOHOSTARCH="amd64"
GOHOSTOS="darwin"
GONOPROXY=""
GONOSUMDB=""
GOOS="darwin"
GOPATH="/Users/censored/go"
GOPRIVATE=""
GOPROXY="https://proxy.golang.org,direct"
GOROOT="/usr/local/Cellar/go/1.13.8/libexec"
GOSUMDB="sum.golang.org"
GOTMPDIR=""
GOTOOLDIR="/usr/local/Cellar/go/1.13.8/libexec/pkg/tool/darwin_amd64"
GCCGO="gccgo"
AR="ar"
CC="clang"
CXX="clang++"
CGO_ENABLED="1"
GOMOD="/Users/censored/go/src/srt/go.mod"
CGO_CFLAGS="-g -O2"
CGO_CPPFLAGS=""
CGO_CXXFLAGS="-g -O2"
CGO_FFLAGS="-g -O2"
CGO_LDFLAGS="-g -O2"
PKG_CONFIG="pkg-config"
GOGCCFLAGS="-fPIC -m64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -fdebug-prefix-map=/var/folders/4s/1bq_2censoredw0000gn/T/go-build488770590=/tmp/go-build -gno-record-gcc-switches -fno-common"

你做了什么?

我编写了一个函数来检测谁在ping我的电脑

conn, err := icmp.ListenPacket("ip4:icmp", "0.0.0.0")
	if err != nil {
		log.Fatalf("listen err, %s", err)
	}
	defer conn.Close()

	if err != nil {
		 return
	}
	bytes := make([]byte, 512)
	for {
		fmt.Println("recv")
		n,_, err := conn.ReadFrom(bytes)
		if err != nil {
			fmt.Println(err.Error())
			continue
		}
		fmt.Println(n)
	}

你期望看到什么?

每收到一个ping,就打印出n。

你实际上看到了什么?

有时候它能工作,有时候不能工作。
PoC视频: [https://youtu.be/AyQDH9AQSRc](https://youtu.be/AyQDH9AQSRc)

92dk7w1h

92dk7w1h1#

你好,

看起来$SIO_RCVALL$标志没有设置,所以只有在网络适配器重新启用时才会接收。

uyto3xhc

uyto3xhc3#

@lochv 我不知道有关icmp的任何事情。我不想让你感到困惑。
Alex

fcwjkofz

fcwjkofz4#

@alexbrainman 问题不是 ICMP,
问题1:在Windows中,套接字需要设置SIO_RCVALL标志来监控原始套接字(Linux没有)。
问题2:在Windows中,将套接字绑定到0.0.0.0上无法设置SIO_RCVALL标志。
因此

icmp.ListenPacket("ip4:icmp", "0.0.0.0")

在Windows中无法工作。
需要将此示例代码从C语言移植到Go语言以实现ICMP监听。

np8igboo

np8igboo5#

问题不是icmp。
问题1:在Windows中,需要设置SIO_RCVALL标志来监控原始套接字(Linux没有)。
问题2:在Windows中,将套接字绑定到0.0.0.0上无法设置SIO_RCVALL标志。
@lochv 我对这些事情一无所知。如果你想修复代码,这里是如何贡献的
https://golang.org/doc/contribute.html
也许我们可以一起修复代码。
谢谢。
Alex

kxkpmulp

kxkpmulp6#

如果有人仍然对这个问题感兴趣,我找到了一个解决方法。
重要警告:从任何来源接收ICMP数据包似乎只有在禁用Windows防火墙时才能正常工作。🤷‍♂️
如果有人对此有更多了解,请指明更多信息。
以下是创建可以从任何来源发送和接收ICMP数据包的监听器的函数。
以下是我们使用它的特殊(部分)traceroute函数。

6l7fqoea

6l7fqoea7#

我认为问题在于如何修改"net"包:在Windows上监听ICMP并保持设计模式。

相关问题