2024-06-05 16:10:03 -04:00
|
|
|
// SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
2022-03-10 10:44:48 +01:00
|
|
|
package sctp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
"encoding/hex"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
2024-06-05 16:10:03 -04:00
|
|
|
type paramHeaderUnrecognizedAction byte
|
|
|
|
|
2022-03-10 10:44:48 +01:00
|
|
|
type paramHeader struct {
|
2024-06-05 16:10:03 -04:00
|
|
|
typ paramType
|
|
|
|
unrecognizedAction paramHeaderUnrecognizedAction
|
|
|
|
len int
|
|
|
|
raw []byte
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
|
|
|
|
2024-06-05 16:10:03 -04:00
|
|
|
/*
|
|
|
|
The Parameter Types are encoded such that the highest-order 2 bits specify
|
|
|
|
the action that is taken if the processing endpoint does not recognize the
|
|
|
|
Parameter Type.
|
|
|
|
|
|
|
|
00 - Stop processing this parameter and do not process any further parameters within this chunk.
|
|
|
|
|
|
|
|
01 - Stop processing this parameter, do not process any further parameters within this chunk, and
|
|
|
|
report the unrecognized parameter, as described in Section 3.2.2.
|
|
|
|
|
|
|
|
10 - Skip this parameter and continue processing.
|
|
|
|
|
|
|
|
11 - Skip this parameter and continue processing, but report the unrecognized
|
|
|
|
parameter, as described in Section 3.2.2.
|
|
|
|
|
|
|
|
https://www.rfc-editor.org/rfc/rfc9260.html#section-3.2.1
|
|
|
|
*/
|
|
|
|
|
2022-03-10 10:44:48 +01:00
|
|
|
const (
|
2024-06-05 16:10:03 -04:00
|
|
|
paramHeaderUnrecognizedActionMask = 0b11000000
|
|
|
|
paramHeaderUnrecognizedActionStop paramHeaderUnrecognizedAction = 0b00000000
|
|
|
|
paramHeaderUnrecognizedActionStopAndReport paramHeaderUnrecognizedAction = 0b01000000
|
|
|
|
paramHeaderUnrecognizedActionSkip paramHeaderUnrecognizedAction = 0b10000000
|
|
|
|
paramHeaderUnrecognizedActionSkipAndReport paramHeaderUnrecognizedAction = 0b11000000
|
|
|
|
|
2022-03-10 10:44:48 +01:00
|
|
|
paramHeaderLength = 4
|
|
|
|
)
|
|
|
|
|
2024-05-15 19:15:00 -04:00
|
|
|
// Parameter header parse errors
|
2022-03-10 10:44:48 +01:00
|
|
|
var (
|
2024-05-15 19:15:00 -04:00
|
|
|
ErrParamHeaderTooShort = errors.New("param header too short")
|
|
|
|
ErrParamHeaderSelfReportedLengthShorter = errors.New("param self reported length is shorter than header length")
|
|
|
|
ErrParamHeaderSelfReportedLengthLonger = errors.New("param self reported length is longer than header length")
|
|
|
|
ErrParamHeaderParseFailed = errors.New("failed to parse param type")
|
2022-03-10 10:44:48 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
func (p *paramHeader) marshal() ([]byte, error) {
|
|
|
|
paramLengthPlusHeader := paramHeaderLength + len(p.raw)
|
|
|
|
|
|
|
|
rawParam := make([]byte, paramLengthPlusHeader)
|
|
|
|
binary.BigEndian.PutUint16(rawParam[0:], uint16(p.typ))
|
|
|
|
binary.BigEndian.PutUint16(rawParam[2:], uint16(paramLengthPlusHeader))
|
|
|
|
copy(rawParam[paramHeaderLength:], p.raw)
|
|
|
|
|
|
|
|
return rawParam, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *paramHeader) unmarshal(raw []byte) error {
|
|
|
|
if len(raw) < paramHeaderLength {
|
2024-05-15 19:15:00 -04:00
|
|
|
return ErrParamHeaderTooShort
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
paramLengthPlusHeader := binary.BigEndian.Uint16(raw[2:])
|
|
|
|
if int(paramLengthPlusHeader) < paramHeaderLength {
|
2024-05-15 19:15:00 -04:00
|
|
|
return fmt.Errorf("%w: param self reported length (%d) shorter than header length (%d)", ErrParamHeaderSelfReportedLengthShorter, int(paramLengthPlusHeader), paramHeaderLength)
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
|
|
|
if len(raw) < int(paramLengthPlusHeader) {
|
2024-05-15 19:15:00 -04:00
|
|
|
return fmt.Errorf("%w: param length (%d) shorter than its self reported length (%d)", ErrParamHeaderSelfReportedLengthLonger, len(raw), int(paramLengthPlusHeader))
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
typ, err := parseParamType(raw[0:])
|
|
|
|
if err != nil {
|
2024-05-15 19:15:00 -04:00
|
|
|
return fmt.Errorf("%w: %v", ErrParamHeaderParseFailed, err) //nolint:errorlint
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
|
|
|
p.typ = typ
|
2024-06-05 16:10:03 -04:00
|
|
|
p.unrecognizedAction = paramHeaderUnrecognizedAction(raw[0] & paramHeaderUnrecognizedActionMask)
|
2022-03-10 10:44:48 +01:00
|
|
|
p.raw = raw[paramHeaderLength:paramLengthPlusHeader]
|
|
|
|
p.len = int(paramLengthPlusHeader)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *paramHeader) length() int {
|
|
|
|
return p.len
|
|
|
|
}
|
|
|
|
|
|
|
|
// String makes paramHeader printable
|
|
|
|
func (p paramHeader) String() string {
|
|
|
|
return fmt.Sprintf("%s (%d): %s", p.typ, p.len, hex.Dump(p.raw))
|
|
|
|
}
|