1
// Copyright 2012 The Go Authors. All rights reserved.
2
// Use of this source code is governed by a BSD-style
3
// license that can be found in the LICENSE file.
15
"golang.org/x/net/icmp"
16
"golang.org/x/net/internal/iana"
17
"golang.org/x/net/internal/nettest"
18
"golang.org/x/net/ipv4"
21
func TestPacketConnReadWriteUnicastUDP(t *testing.T) {
23
case "nacl", "plan9", "solaris", "windows":
24
t.Skipf("not supported on %s", runtime.GOOS)
26
ifi := nettest.RoutedInterface("ip4", net.FlagUp|net.FlagLoopback)
28
t.Skipf("not available on %s", runtime.GOOS)
31
c, err := net.ListenPacket("udp4", "127.0.0.1:0")
37
dst, err := net.ResolveUDPAddr("udp4", c.LocalAddr().String())
41
p := ipv4.NewPacketConn(c)
43
cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
44
wb := []byte("HELLO-R-U-THERE")
46
for i, toggle := range []bool{true, false, true} {
47
if err := p.SetControlMessage(cf, toggle); err != nil {
48
if nettest.ProtocolNotSupported(err) {
49
t.Logf("not supported on %s", runtime.GOOS)
55
if err := p.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)); err != nil {
58
if n, err := p.WriteTo(wb, nil, dst); err != nil {
60
} else if n != len(wb) {
61
t.Fatalf("got %v; want %v", n, len(wb))
63
rb := make([]byte, 128)
64
if err := p.SetReadDeadline(time.Now().Add(100 * time.Millisecond)); err != nil {
67
if n, cm, _, err := p.ReadFrom(rb); err != nil {
69
} else if !bytes.Equal(rb[:n], wb) {
70
t.Fatalf("got %v; want %v", rb[:n], wb)
72
t.Logf("rcvd cmsg: %v", cm)
77
func TestPacketConnReadWriteUnicastICMP(t *testing.T) {
79
case "nacl", "plan9", "solaris", "windows":
80
t.Skipf("not supported on %s", runtime.GOOS)
82
if m, ok := nettest.SupportsRawIPSocket(); !ok {
85
ifi := nettest.RoutedInterface("ip4", net.FlagUp|net.FlagLoopback)
87
t.Skipf("not available on %s", runtime.GOOS)
90
c, err := net.ListenPacket("ip4:icmp", "0.0.0.0")
96
dst, err := net.ResolveIPAddr("ip4", "127.0.0.1")
100
p := ipv4.NewPacketConn(c)
102
cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
104
for i, toggle := range []bool{true, false, true} {
105
wb, err := (&icmp.Message{
106
Type: ipv4.ICMPTypeEcho, Code: 0,
108
ID: os.Getpid() & 0xffff, Seq: i + 1,
109
Data: []byte("HELLO-R-U-THERE"),
115
if err := p.SetControlMessage(cf, toggle); err != nil {
116
if nettest.ProtocolNotSupported(err) {
117
t.Logf("not supported on %s", runtime.GOOS)
123
if err := p.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)); err != nil {
126
if n, err := p.WriteTo(wb, nil, dst); err != nil {
128
} else if n != len(wb) {
129
t.Fatalf("got %v; want %v", n, len(wb))
131
rb := make([]byte, 128)
133
if err := p.SetReadDeadline(time.Now().Add(100 * time.Millisecond)); err != nil {
136
if n, cm, _, err := p.ReadFrom(rb); err != nil {
137
switch runtime.GOOS {
138
case "darwin": // older darwin kernels have some limitation on receiving icmp packet through raw socket
139
t.Logf("not supported on %s", runtime.GOOS)
144
t.Logf("rcvd cmsg: %v", cm)
145
m, err := icmp.ParseMessage(iana.ProtocolICMP, rb[:n])
149
if runtime.GOOS == "linux" && m.Type == ipv4.ICMPTypeEcho {
150
// On Linux we must handle own sent packets.
153
if m.Type != ipv4.ICMPTypeEchoReply || m.Code != 0 {
154
t.Fatalf("got type=%v, code=%v; want type=%v, code=%v", m.Type, m.Code, ipv4.ICMPTypeEchoReply, 0)
160
func TestRawConnReadWriteUnicastICMP(t *testing.T) {
161
switch runtime.GOOS {
162
case "nacl", "plan9", "solaris", "windows":
163
t.Skipf("not supported on %s", runtime.GOOS)
165
if m, ok := nettest.SupportsRawIPSocket(); !ok {
168
ifi := nettest.RoutedInterface("ip4", net.FlagUp|net.FlagLoopback)
170
t.Skipf("not available on %s", runtime.GOOS)
173
c, err := net.ListenPacket("ip4:icmp", "0.0.0.0")
179
dst, err := net.ResolveIPAddr("ip4", "127.0.0.1")
183
r, err := ipv4.NewRawConn(c)
188
cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
190
for i, toggle := range []bool{true, false, true} {
191
wb, err := (&icmp.Message{
192
Type: ipv4.ICMPTypeEcho, Code: 0,
194
ID: os.Getpid() & 0xffff, Seq: i + 1,
195
Data: []byte("HELLO-R-U-THERE"),
202
Version: ipv4.Version,
205
TotalLen: ipv4.HeaderLen + len(wb),
210
if err := r.SetControlMessage(cf, toggle); err != nil {
211
if nettest.ProtocolNotSupported(err) {
212
t.Logf("not supported on %s", runtime.GOOS)
217
if err := r.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)); err != nil {
220
if err := r.WriteTo(wh, wb, nil); err != nil {
223
rb := make([]byte, ipv4.HeaderLen+128)
225
if err := r.SetReadDeadline(time.Now().Add(100 * time.Millisecond)); err != nil {
228
if _, b, cm, err := r.ReadFrom(rb); err != nil {
229
switch runtime.GOOS {
230
case "darwin": // older darwin kernels have some limitation on receiving icmp packet through raw socket
231
t.Logf("not supported on %s", runtime.GOOS)
236
t.Logf("rcvd cmsg: %v", cm)
237
m, err := icmp.ParseMessage(iana.ProtocolICMP, b)
241
if runtime.GOOS == "linux" && m.Type == ipv4.ICMPTypeEcho {
242
// On Linux we must handle own sent packets.
245
if m.Type != ipv4.ICMPTypeEchoReply || m.Code != 0 {
246
t.Fatalf("got type=%v, code=%v; want type=%v, code=%v", m.Type, m.Code, ipv4.ICMPTypeEchoReply, 0)