2020-06-04 23:42:44 +01:00
|
|
|
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package packet
|
|
|
|
|
2020-11-11 03:09:47 +00:00
|
|
|
import "encoding/binary"
|
|
|
|
|
2020-11-11 06:26:00 +00:00
|
|
|
// icmp4HeaderLength is the size of the ICMPv4 packet header, not
|
|
|
|
// including the outer IP layer or the variable "response data"
|
|
|
|
// trailer.
|
|
|
|
const icmp4HeaderLength = 4
|
|
|
|
|
|
|
|
// ICMP4Type is an ICMPv4 type, as specified in
|
|
|
|
// https://www.iana.org/assignments/icmp-parameters/icmp-parameters.xhtml
|
2020-11-09 23:34:03 +00:00
|
|
|
type ICMP4Type uint8
|
2020-06-04 23:42:44 +01:00
|
|
|
|
|
|
|
const (
|
2020-11-09 23:34:03 +00:00
|
|
|
ICMP4EchoReply ICMP4Type = 0x00
|
|
|
|
ICMP4EchoRequest ICMP4Type = 0x08
|
|
|
|
ICMP4Unreachable ICMP4Type = 0x03
|
|
|
|
ICMP4TimeExceeded ICMP4Type = 0x0b
|
2020-06-04 23:42:44 +01:00
|
|
|
)
|
|
|
|
|
2020-11-09 23:34:03 +00:00
|
|
|
func (t ICMP4Type) String() string {
|
2020-06-04 23:42:44 +01:00
|
|
|
switch t {
|
2020-11-09 23:34:03 +00:00
|
|
|
case ICMP4EchoReply:
|
2020-06-04 23:42:44 +01:00
|
|
|
return "EchoReply"
|
2020-11-09 23:34:03 +00:00
|
|
|
case ICMP4EchoRequest:
|
2020-06-04 23:42:44 +01:00
|
|
|
return "EchoRequest"
|
2020-11-09 23:34:03 +00:00
|
|
|
case ICMP4Unreachable:
|
2020-06-04 23:42:44 +01:00
|
|
|
return "Unreachable"
|
2020-11-09 23:34:03 +00:00
|
|
|
case ICMP4TimeExceeded:
|
2020-06-04 23:42:44 +01:00
|
|
|
return "TimeExceeded"
|
|
|
|
default:
|
|
|
|
return "Unknown"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-11 06:26:00 +00:00
|
|
|
// ICMP4Code is an ICMPv4 code, as specified in
|
|
|
|
// https://www.iana.org/assignments/icmp-parameters/icmp-parameters.xhtml
|
2020-11-09 23:34:03 +00:00
|
|
|
type ICMP4Code uint8
|
2020-06-04 23:42:44 +01:00
|
|
|
|
|
|
|
const (
|
2020-11-09 23:34:03 +00:00
|
|
|
ICMP4NoCode ICMP4Code = 0
|
2020-06-04 23:42:44 +01:00
|
|
|
)
|
|
|
|
|
2020-11-11 06:26:00 +00:00
|
|
|
// ICMP4Header is an IPv4+ICMPv4 header.
|
2020-11-09 23:34:03 +00:00
|
|
|
type ICMP4Header struct {
|
|
|
|
IP4Header
|
|
|
|
Type ICMP4Type
|
|
|
|
Code ICMP4Code
|
2020-06-04 23:42:44 +01:00
|
|
|
}
|
|
|
|
|
2020-11-11 06:26:00 +00:00
|
|
|
// Len implements Header.
|
|
|
|
func (h ICMP4Header) Len() int {
|
|
|
|
return h.IP4Header.Len() + icmp4HeaderLength
|
2020-06-04 23:42:44 +01:00
|
|
|
}
|
|
|
|
|
2020-11-11 06:26:00 +00:00
|
|
|
// Marshal implements Header.
|
2020-11-09 23:34:03 +00:00
|
|
|
func (h ICMP4Header) Marshal(buf []byte) error {
|
2020-11-11 06:26:00 +00:00
|
|
|
if len(buf) < h.Len() {
|
2020-06-04 23:42:44 +01:00
|
|
|
return errSmallBuffer
|
|
|
|
}
|
|
|
|
if len(buf) > maxPacketLength {
|
|
|
|
return errLargePacket
|
|
|
|
}
|
|
|
|
// The caller does not need to set this.
|
2020-11-10 09:00:35 +00:00
|
|
|
h.IPProto = ICMPv4
|
2020-06-04 23:42:44 +01:00
|
|
|
|
|
|
|
buf[20] = uint8(h.Type)
|
|
|
|
buf[21] = uint8(h.Code)
|
|
|
|
|
2020-11-09 23:34:03 +00:00
|
|
|
h.IP4Header.Marshal(buf)
|
2020-06-04 23:42:44 +01:00
|
|
|
|
2020-11-11 06:26:00 +00:00
|
|
|
binary.BigEndian.PutUint16(buf[22:24], ip4Checksum(buf))
|
2020-06-04 23:42:44 +01:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-11-11 06:26:00 +00:00
|
|
|
// ToResponse implements Header. TODO: it doesn't implement it
|
|
|
|
// correctly, instead it statically generates an ICMP Echo Reply
|
|
|
|
// packet.
|
2020-11-09 23:34:03 +00:00
|
|
|
func (h *ICMP4Header) ToResponse() {
|
|
|
|
// TODO: this doesn't implement ToResponse correctly, as it
|
|
|
|
// assumes the ICMP request type.
|
|
|
|
h.Type = ICMP4EchoReply
|
|
|
|
h.Code = ICMP4NoCode
|
|
|
|
h.IP4Header.ToResponse()
|
2020-06-04 23:42:44 +01:00
|
|
|
}
|