2017-07-23 12:24:45 +00:00
|
|
|
// Copyright 2015 The Go 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 icmp
|
|
|
|
|
2018-03-30 10:45:07 +00:00
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
|
|
|
|
"golang.org/x/net/ipv4"
|
|
|
|
"golang.org/x/net/ipv6"
|
|
|
|
)
|
2017-07-23 12:24:45 +00:00
|
|
|
|
|
|
|
// An Extension represents an ICMP extension.
|
|
|
|
type Extension interface {
|
|
|
|
// Len returns the length of ICMP extension.
|
|
|
|
// Proto must be either the ICMPv4 or ICMPv6 protocol number.
|
|
|
|
Len(proto int) int
|
|
|
|
|
|
|
|
// Marshal returns the binary encoding of ICMP extension.
|
|
|
|
// Proto must be either the ICMPv4 or ICMPv6 protocol number.
|
|
|
|
Marshal(proto int) ([]byte, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
const extensionVersion = 2
|
|
|
|
|
|
|
|
func validExtensionHeader(b []byte) bool {
|
|
|
|
v := int(b[0]&0xf0) >> 4
|
|
|
|
s := binary.BigEndian.Uint16(b[2:4])
|
|
|
|
if s != 0 {
|
|
|
|
s = checksum(b)
|
|
|
|
}
|
|
|
|
if v != extensionVersion || s != 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// parseExtensions parses b as a list of ICMP extensions.
|
|
|
|
// The length attribute l must be the length attribute field in
|
|
|
|
// received icmp messages.
|
|
|
|
//
|
|
|
|
// It will return a list of ICMP extensions and an adjusted length
|
|
|
|
// attribute that represents the length of the padded original
|
|
|
|
// datagram field. Otherwise, it returns an error.
|
2018-03-30 10:45:07 +00:00
|
|
|
func parseExtensions(typ Type, b []byte, l int) ([]Extension, int, error) {
|
2017-07-23 12:24:45 +00:00
|
|
|
// Still a lot of non-RFC 4884 compliant implementations are
|
|
|
|
// out there. Set the length attribute l to 128 when it looks
|
|
|
|
// inappropriate for backwards compatibility.
|
|
|
|
//
|
|
|
|
// A minimal extension at least requires 8 octets; 4 octets
|
|
|
|
// for an extension header, and 4 octets for a single object
|
|
|
|
// header.
|
|
|
|
//
|
|
|
|
// See RFC 4884 for further information.
|
2018-03-30 10:45:07 +00:00
|
|
|
switch typ {
|
|
|
|
case ipv4.ICMPTypeExtendedEchoRequest, ipv6.ICMPTypeExtendedEchoRequest:
|
|
|
|
if len(b) < 8 || !validExtensionHeader(b) {
|
2017-07-23 12:24:45 +00:00
|
|
|
return nil, -1, errNoExtension
|
|
|
|
}
|
2018-03-30 10:45:07 +00:00
|
|
|
l = 0
|
|
|
|
default:
|
|
|
|
if 128 > l || l+8 > len(b) {
|
|
|
|
l = 128
|
|
|
|
}
|
|
|
|
if l+8 > len(b) {
|
2017-07-23 12:24:45 +00:00
|
|
|
return nil, -1, errNoExtension
|
|
|
|
}
|
2018-03-30 10:45:07 +00:00
|
|
|
if !validExtensionHeader(b[l:]) {
|
|
|
|
if l == 128 {
|
|
|
|
return nil, -1, errNoExtension
|
|
|
|
}
|
|
|
|
l = 128
|
|
|
|
if !validExtensionHeader(b[l:]) {
|
|
|
|
return nil, -1, errNoExtension
|
|
|
|
}
|
|
|
|
}
|
2017-07-23 12:24:45 +00:00
|
|
|
}
|
|
|
|
var exts []Extension
|
|
|
|
for b = b[l+4:]; len(b) >= 4; {
|
|
|
|
ol := int(binary.BigEndian.Uint16(b[:2]))
|
|
|
|
if 4 > ol || ol > len(b) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
switch b[2] {
|
|
|
|
case classMPLSLabelStack:
|
|
|
|
ext, err := parseMPLSLabelStack(b[:ol])
|
|
|
|
if err != nil {
|
|
|
|
return nil, -1, err
|
|
|
|
}
|
|
|
|
exts = append(exts, ext)
|
|
|
|
case classInterfaceInfo:
|
|
|
|
ext, err := parseInterfaceInfo(b[:ol])
|
|
|
|
if err != nil {
|
|
|
|
return nil, -1, err
|
|
|
|
}
|
|
|
|
exts = append(exts, ext)
|
2018-03-30 10:45:07 +00:00
|
|
|
case classInterfaceIdent:
|
|
|
|
ext, err := parseInterfaceIdent(b[:ol])
|
|
|
|
if err != nil {
|
|
|
|
return nil, -1, err
|
|
|
|
}
|
|
|
|
exts = append(exts, ext)
|
2017-07-23 12:24:45 +00:00
|
|
|
}
|
|
|
|
b = b[ol:]
|
|
|
|
}
|
|
|
|
return exts, l, nil
|
|
|
|
}
|