Mikio Hara | d2e5a12 | 2012-09-26 21:03:09 +0900 | [diff] [blame] | 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. |
| 4 | |
| 5 | package ipv4_test |
| 6 | |
| 7 | import ( |
Mikio Hara | 8108b4b | 2013-06-07 14:52:58 +0900 | [diff] [blame] | 8 | "code.google.com/p/go.net/ipv4" |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 9 | "errors" |
Mikio Hara | d2e5a12 | 2012-09-26 21:03:09 +0900 | [diff] [blame] | 10 | "flag" |
| 11 | ) |
| 12 | |
| 13 | var testExternal = flag.Bool("external", true, "allow use of external networks during long test") |
| 14 | |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 15 | // icmpMessage represents an ICMP message. |
| 16 | type icmpMessage struct { |
Mikio Hara | 8108b4b | 2013-06-07 14:52:58 +0900 | [diff] [blame] | 17 | Type ipv4.ICMPType // type |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 18 | Code int // code |
| 19 | Checksum int // checksum |
| 20 | Body icmpMessageBody // body |
Mikio Hara | d2e5a12 | 2012-09-26 21:03:09 +0900 | [diff] [blame] | 21 | } |
| 22 | |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 23 | // icmpMessageBody represents an ICMP message body. |
| 24 | type 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. |
| 31 | func (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 Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 40 | 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 Hara | 2487781 | 2013-07-22 20:09:44 +0900 | [diff] [blame] | 52 | b[2] ^= byte(^s) |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 53 | b[3] ^= byte(^s >> 8) |
| 54 | return b, nil |
| 55 | } |
| 56 | |
| 57 | // parseICMPMessage parses b as an ICMP message. |
| 58 | func parseICMPMessage(b []byte) (*icmpMessage, error) { |
| 59 | msglen := len(b) |
| 60 | if msglen < 4 { |
| 61 | return nil, errors.New("message too short") |
| 62 | } |
Mikio Hara | 8108b4b | 2013-06-07 14:52:58 +0900 | [diff] [blame] | 63 | m := &icmpMessage{Type: ipv4.ICMPType(b[0]), Code: int(b[1]), Checksum: int(b[2])<<8 | int(b[3])} |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 64 | if msglen > 4 { |
| 65 | var err error |
| 66 | switch m.Type { |
Mikio Hara | 8108b4b | 2013-06-07 14:52:58 +0900 | [diff] [blame] | 67 | case ipv4.ICMPTypeEcho, ipv4.ICMPTypeEchoReply: |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 68 | 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. |
| 78 | type icmpEcho struct { |
| 79 | ID int // identifier |
| 80 | Seq int // sequence number |
| 81 | Data []byte // data |
| 82 | } |
| 83 | |
| 84 | func (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. |
| 93 | func (p *icmpEcho) Marshal() ([]byte, error) { |
| 94 | b := make([]byte, 4+len(p.Data)) |
Mikio Hara | 2487781 | 2013-07-22 20:09:44 +0900 | [diff] [blame] | 95 | b[0], b[1] = byte(p.ID>>8), byte(p.ID) |
| 96 | b[2], b[3] = byte(p.Seq>>8), byte(p.Seq) |
Mikio Hara | 6b91bf2 | 2013-02-16 13:02:07 +0900 | [diff] [blame] | 97 | 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. |
| 103 | func 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 Hara | d2e5a12 | 2012-09-26 21:03:09 +0900 | [diff] [blame] | 111 | } |