/* * Copyright (c) 2021 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "net/dcsctp/packet/parameter/parameter.h" #include #include #include #include #include "api/array_view.h" #include "net/dcsctp/common/math.h" #include "net/dcsctp/packet/bounded_byte_reader.h" #include "rtc_base/logging.h" namespace dcsctp { constexpr size_t kParameterHeaderSize = 4; Parameters::Builder& Parameters::Builder::Add(const Parameter& p) { // https://tools.ietf.org/html/rfc4960#section-3.2.1 // "If the length of the parameter is not a multiple of 4 bytes, the sender // pads the parameter at the end (i.e., after the Parameter Value field) with // all zero bytes." if (data_.size() % 4 != 0) { data_.resize(RoundUpTo4(data_.size())); } p.SerializeTo(data_); return *this; } std::vector Parameters::descriptors() const { webrtc::ArrayView span(data_); std::vector result; while (!span.empty()) { BoundedByteReader header(span); uint16_t type = header.Load16<0>(); uint16_t length = header.Load16<2>(); result.emplace_back(type, span.subview(0, length)); size_t length_with_padding = RoundUpTo4(length); if (length_with_padding > span.size()) { break; } span = span.subview(length_with_padding); } return result; } std::optional Parameters::Parse( webrtc::ArrayView data) { // Validate the parameter descriptors webrtc::ArrayView span(data); while (!span.empty()) { if (span.size() < kParameterHeaderSize) { RTC_DLOG(LS_WARNING) << "Insufficient parameter length"; return std::nullopt; } BoundedByteReader header(span); uint16_t length = header.Load16<2>(); if (length < kParameterHeaderSize || length > span.size()) { RTC_DLOG(LS_WARNING) << "Invalid parameter length field"; return std::nullopt; } size_t length_with_padding = RoundUpTo4(length); if (length_with_padding > span.size()) { break; } span = span.subview(length_with_padding); } return Parameters(std::vector(data.begin(), data.end())); } } // namespace dcsctp