blob: be328ca5c8e00f312d6e48eac092b88be1ae1b63 [file] [log] [blame]
Mikio Harad2e5a122012-09-26 21:03:09 +09001// 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.
4
5package ipv4_test
6
7import (
Mikio Hara8108b4b2013-06-07 14:52:58 +09008 "code.google.com/p/go.net/ipv4"
Mikio Hara6b91bf22013-02-16 13:02:07 +09009 "errors"
Mikio Harad2e5a122012-09-26 21:03:09 +090010 "flag"
11)
12
13var testExternal = flag.Bool("external", true, "allow use of external networks during long test")
14
Mikio Hara6b91bf22013-02-16 13:02:07 +090015// icmpMessage represents an ICMP message.
16type icmpMessage struct {
Mikio Hara8108b4b2013-06-07 14:52:58 +090017 Type ipv4.ICMPType // type
Mikio Hara6b91bf22013-02-16 13:02:07 +090018 Code int // code
19 Checksum int // checksum
20 Body icmpMessageBody // body
Mikio Harad2e5a122012-09-26 21:03:09 +090021}
22
Mikio Hara6b91bf22013-02-16 13:02:07 +090023// icmpMessageBody represents an ICMP message body.
24type icmpMessageBody interface {
25 Len() int
26 Marshal() ([]byte, error)
27}
28
29// Marshal returns the binary enconding of the ICMP echo request or
30// reply message m.
31func (m *icmpMessage) Marshal() ([]byte, error) {
32 b := []byte{byte(m.Type), byte(m.Code), 0, 0}
33 if m.Body != nil && m.Body.Len() != 0 {
34 mb, err := m.Body.Marshal()
35 if err != nil {
36 return nil, err
37 }
38 b = append(b, mb...)
39 }
Mikio Hara6b91bf22013-02-16 13:02:07 +090040 csumcv := len(b) - 1 // checksum coverage
41 s := uint32(0)
42 for i := 0; i < csumcv; i += 2 {
43 s += uint32(b[i+1])<<8 | uint32(b[i])
44 }
45 if csumcv&1 == 0 {
46 s += uint32(b[csumcv])
47 }
48 s = s>>16 + s&0xffff
49 s = s + s>>16
50 // Place checksum back in header; using ^= avoids the
51 // assumption the checksum bytes are zero.
Mikio Hara24877812013-07-22 20:09:44 +090052 b[2] ^= byte(^s)
Mikio Hara6b91bf22013-02-16 13:02:07 +090053 b[3] ^= byte(^s >> 8)
54 return b, nil
55}
56
57// parseICMPMessage parses b as an ICMP message.
58func parseICMPMessage(b []byte) (*icmpMessage, error) {
59 msglen := len(b)
60 if msglen < 4 {
61 return nil, errors.New("message too short")
62 }
Mikio Hara8108b4b2013-06-07 14:52:58 +090063 m := &icmpMessage{Type: ipv4.ICMPType(b[0]), Code: int(b[1]), Checksum: int(b[2])<<8 | int(b[3])}
Mikio Hara6b91bf22013-02-16 13:02:07 +090064 if msglen > 4 {
65 var err error
66 switch m.Type {
Mikio Hara8108b4b2013-06-07 14:52:58 +090067 case ipv4.ICMPTypeEcho, ipv4.ICMPTypeEchoReply:
Mikio Hara6b91bf22013-02-16 13:02:07 +090068 m.Body, err = parseICMPEcho(b[4:])
69 if err != nil {
70 return nil, err
71 }
72 }
73 }
74 return m, nil
75}
76
77// imcpEcho represenets an ICMP echo request or reply message body.
78type icmpEcho struct {
79 ID int // identifier
80 Seq int // sequence number
81 Data []byte // data
82}
83
84func (p *icmpEcho) Len() int {
85 if p == nil {
86 return 0
87 }
88 return 4 + len(p.Data)
89}
90
91// Marshal returns the binary enconding of the ICMP echo request or
92// reply message body p.
93func (p *icmpEcho) Marshal() ([]byte, error) {
94 b := make([]byte, 4+len(p.Data))
Mikio Hara24877812013-07-22 20:09:44 +090095 b[0], b[1] = byte(p.ID>>8), byte(p.ID)
96 b[2], b[3] = byte(p.Seq>>8), byte(p.Seq)
Mikio Hara6b91bf22013-02-16 13:02:07 +090097 copy(b[4:], p.Data)
98 return b, nil
99}
100
101// parseICMPEcho parses b as an ICMP echo request or reply message
102// body.
103func parseICMPEcho(b []byte) (*icmpEcho, error) {
104 bodylen := len(b)
105 p := &icmpEcho{ID: int(b[0])<<8 | int(b[1]), Seq: int(b[2])<<8 | int(b[3])}
106 if bodylen > 4 {
107 p.Data = make([]byte, bodylen-4)
108 copy(p.Data, b[4:])
109 }
110 return p, nil
Mikio Harad2e5a122012-09-26 21:03:09 +0900111}