/* * 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/idata_chunk.h" #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/chunk/data_common.h" #include "rtc_base/strings/string_builder.h" namespace dcsctp { // https://tools.ietf.org/html/rfc8260#section-2.1 // 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 = 64 | Res |I|U|B|E| Length = Variable | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | TSN | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Stream Identifier | Reserved | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Message Identifier | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // | Payload Protocol Identifier / Fragment Sequence Number | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // \ \ // / User Data / // \ \ // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ constexpr int IDataChunk::kType; absl::optional IDataChunk::Parse( rtc::ArrayView data) { absl::optional> reader = ParseTLV(data); if (!reader.has_value()) { return absl::nullopt; } uint8_t flags = reader->Load8<1>(); TSN tsn(reader->Load32<4>()); StreamID stream_identifier(reader->Load16<8>()); MID message_id(reader->Load32<12>()); uint32_t ppid_or_fsn = reader->Load32<16>(); Options options; options.is_end = Data::IsEnd((flags & (1 << kFlagsBitEnd)) != 0); options.is_beginning = Data::IsBeginning((flags & (1 << kFlagsBitBeginning)) != 0); options.is_unordered = IsUnordered((flags & (1 << kFlagsBitUnordered)) != 0); options.immediate_ack = ImmediateAckFlag((flags & (1 << kFlagsBitImmediateAck)) != 0); return IDataChunk(tsn, stream_identifier, message_id, PPID(options.is_beginning ? ppid_or_fsn : 0), FSN(options.is_beginning ? 0 : ppid_or_fsn), std::vector(reader->variable_data().begin(), reader->variable_data().end()), options); } void IDataChunk::SerializeTo(std::vector& out) const { BoundedByteWriter writer = AllocateTLV(out, payload().size()); writer.Store8<1>( (*options().is_end ? (1 << kFlagsBitEnd) : 0) | (*options().is_beginning ? (1 << kFlagsBitBeginning) : 0) | (*options().is_unordered ? (1 << kFlagsBitUnordered) : 0) | (*options().immediate_ack ? (1 << kFlagsBitImmediateAck) : 0)); writer.Store32<4>(*tsn()); writer.Store16<8>(*stream_id()); writer.Store32<12>(*message_id()); writer.Store32<16>(options().is_beginning ? *ppid() : *fsn()); writer.CopyToVariableData(payload()); } std::string IDataChunk::ToString() const { rtc::StringBuilder sb; sb << "I-DATA, type=" << (options().is_unordered ? "unordered" : "ordered") << "::" << (*options().is_beginning && *options().is_end ? "complete" : *options().is_beginning ? "first" : *options().is_end ? "last" : "middle") << ", tsn=" << *tsn() << ", stream_id=" << *stream_id() << ", message_id=" << *message_id(); if (*options().is_beginning) { sb << ", ppid=" << *ppid(); } else { sb << ", fsn=" << *fsn(); } sb << ", length=" << payload().size(); return sb.Release(); } } // namespace dcsctp