/* * 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/chunk/sack_chunk.h" #include #include #include #include #include #include "absl/types/optional.h" #include "api/array_view.h" #include "net/dcsctp/packet/bounded_byte_reader.h" #include "net/dcsctp/packet/bounded_byte_writer.h" #include "net/dcsctp/packet/tlv_trait.h" #include "rtc_base/logging.h" #include "rtc_base/strings/str_join.h" #include "rtc_base/strings/string_builder.h" namespace dcsctp { // https://tools.ietf.org/html/rfc4960#section-3.3.4 // 0 1 2 3 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Type = 3 |Chunk Flags | Chunk Length | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Cumulative TSN Ack | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Advertised Receiver Window Credit (a_rwnd) | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Number of Gap Ack Blocks = N | Number of Duplicate TSNs = X | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Gap Ack Block #1 Start | Gap Ack Block #1 End | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // / / // \ ... \ // / / // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Gap Ack Block #N Start | Gap Ack Block #N End | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Duplicate TSN 1 | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // / / // \ ... \ // / / // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Duplicate TSN X | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ constexpr int SackChunk::kType; absl::optional SackChunk::Parse(rtc::ArrayView data) { absl::optional> reader = ParseTLV(data); if (!reader.has_value()) { return absl::nullopt; } TSN tsn_ack(reader->Load32<4>()); uint32_t a_rwnd = reader->Load32<8>(); uint16_t nbr_of_gap_blocks = reader->Load16<12>(); uint16_t nbr_of_dup_tsns = reader->Load16<14>(); if (reader->variable_data_size() != nbr_of_gap_blocks * kGapAckBlockSize + nbr_of_dup_tsns * kDupTsnBlockSize) { RTC_DLOG(LS_WARNING) << "Invalid number of gap blocks or duplicate TSNs"; return absl::nullopt; } std::vector gap_ack_blocks; gap_ack_blocks.reserve(nbr_of_gap_blocks); size_t offset = 0; for (int i = 0; i < nbr_of_gap_blocks; ++i) { BoundedByteReader sub_reader = reader->sub_reader(offset); uint16_t start = sub_reader.Load16<0>(); uint16_t end = sub_reader.Load16<2>(); gap_ack_blocks.emplace_back(start, end); offset += kGapAckBlockSize; } std::set duplicate_tsns; for (int i = 0; i < nbr_of_dup_tsns; ++i) { BoundedByteReader sub_reader = reader->sub_reader(offset); duplicate_tsns.insert(TSN(sub_reader.Load32<0>())); offset += kDupTsnBlockSize; } RTC_DCHECK(offset == reader->variable_data_size()); return SackChunk(tsn_ack, a_rwnd, gap_ack_blocks, duplicate_tsns); } void SackChunk::SerializeTo(std::vector& out) const { int nbr_of_gap_blocks = gap_ack_blocks_.size(); int nbr_of_dup_tsns = duplicate_tsns_.size(); size_t variable_size = nbr_of_gap_blocks * kGapAckBlockSize + nbr_of_dup_tsns * kDupTsnBlockSize; BoundedByteWriter writer = AllocateTLV(out, variable_size); writer.Store32<4>(*cumulative_tsn_ack_); writer.Store32<8>(a_rwnd_); writer.Store16<12>(nbr_of_gap_blocks); writer.Store16<14>(nbr_of_dup_tsns); size_t offset = 0; for (int i = 0; i < nbr_of_gap_blocks; ++i) { BoundedByteWriter sub_writer = writer.sub_writer(offset); sub_writer.Store16<0>(gap_ack_blocks_[i].start); sub_writer.Store16<2>(gap_ack_blocks_[i].end); offset += kGapAckBlockSize; } for (TSN tsn : duplicate_tsns_) { BoundedByteWriter sub_writer = writer.sub_writer(offset); sub_writer.Store32<0>(*tsn); offset += kDupTsnBlockSize; } RTC_DCHECK(offset == variable_size); } std::string SackChunk::ToString() const { rtc::StringBuilder sb; sb << "SACK, cum_ack_tsn=" << *cumulative_tsn_ack() << ", a_rwnd=" << a_rwnd(); for (const GapAckBlock& gap : gap_ack_blocks_) { uint32_t first = *cumulative_tsn_ack_ + gap.start; uint32_t last = *cumulative_tsn_ack_ + gap.end; sb << ", gap=" << first << "--" << last; } if (!duplicate_tsns_.empty()) { sb << ", dup_tsns=" << StrJoin(duplicate_tsns(), ",", [](rtc::StringBuilder& sb, TSN tsn) { sb << *tsn; }); } return sb.Release(); } } // namespace dcsctp