/* * Copyright 2013 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 "pc/sctp_utils.h" #include #include #include "absl/types/optional.h" #include "api/priority.h" #include "media/sctp/sctp_transport_internal.h" #include "rtc_base/byte_buffer.h" #include "rtc_base/copy_on_write_buffer.h" #include "test/gtest.h" using webrtc::StreamId; class SctpUtilsTest : public ::testing::Test { public: void VerifyOpenMessageFormat(const rtc::CopyOnWriteBuffer& packet, const std::string& label, const webrtc::DataChannelInit& config) { uint8_t message_type; uint8_t channel_type; uint32_t reliability; uint16_t priority; uint16_t label_length; uint16_t protocol_length; rtc::ByteBufferReader buffer(packet); ASSERT_TRUE(buffer.ReadUInt8(&message_type)); EXPECT_EQ(0x03, message_type); ASSERT_TRUE(buffer.ReadUInt8(&channel_type)); if (config.ordered) { EXPECT_EQ( config.maxRetransmits ? 0x01 : (config.maxRetransmitTime ? 0x02 : 0), channel_type); } else { EXPECT_EQ(config.maxRetransmits ? 0x81 : (config.maxRetransmitTime ? 0x82 : 0x80), channel_type); } ASSERT_TRUE(buffer.ReadUInt16(&priority)); if (config.priority) { // Exact values are checked by round-trip conversion, but // all values defined are greater than zero. EXPECT_GT(priority, 0); } else { EXPECT_EQ(priority, 0); } ASSERT_TRUE(buffer.ReadUInt32(&reliability)); if (config.maxRetransmits || config.maxRetransmitTime) { EXPECT_EQ(config.maxRetransmits ? *config.maxRetransmits : *config.maxRetransmitTime, static_cast(reliability)); } ASSERT_TRUE(buffer.ReadUInt16(&label_length)); ASSERT_TRUE(buffer.ReadUInt16(&protocol_length)); EXPECT_EQ(label.size(), label_length); EXPECT_EQ(config.protocol.size(), protocol_length); absl::string_view label_output; ASSERT_TRUE(buffer.ReadStringView(&label_output, label_length)); EXPECT_EQ(label, label_output); absl::string_view protocol_output; ASSERT_TRUE(buffer.ReadStringView(&protocol_output, protocol_length)); EXPECT_EQ(config.protocol, protocol_output); } }; TEST_F(SctpUtilsTest, WriteParseOpenMessageWithOrderedReliable) { webrtc::DataChannelInit config; std::string label = "abc"; config.protocol = "y"; rtc::CopyOnWriteBuffer packet; ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); VerifyOpenMessageFormat(packet, label, config); std::string output_label; webrtc::DataChannelInit output_config; ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, &output_config)); EXPECT_EQ(label, output_label); EXPECT_EQ(config.protocol, output_config.protocol); EXPECT_EQ(config.ordered, output_config.ordered); EXPECT_EQ(config.maxRetransmitTime, output_config.maxRetransmitTime); EXPECT_EQ(config.maxRetransmits, output_config.maxRetransmits); } TEST_F(SctpUtilsTest, WriteParseOpenMessageWithMaxRetransmitTime) { webrtc::DataChannelInit config; std::string label = "abc"; config.ordered = false; config.maxRetransmitTime = 10; config.protocol = "y"; rtc::CopyOnWriteBuffer packet; ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); VerifyOpenMessageFormat(packet, label, config); std::string output_label; webrtc::DataChannelInit output_config; ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, &output_config)); EXPECT_EQ(label, output_label); EXPECT_EQ(config.protocol, output_config.protocol); EXPECT_EQ(config.ordered, output_config.ordered); EXPECT_EQ(*config.maxRetransmitTime, *output_config.maxRetransmitTime); EXPECT_FALSE(output_config.maxRetransmits); } TEST_F(SctpUtilsTest, WriteParseOpenMessageWithMaxRetransmits) { webrtc::DataChannelInit config; std::string label = "abc"; config.maxRetransmits = 10; config.protocol = "y"; rtc::CopyOnWriteBuffer packet; ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); VerifyOpenMessageFormat(packet, label, config); std::string output_label; webrtc::DataChannelInit output_config; ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, &output_config)); EXPECT_EQ(label, output_label); EXPECT_EQ(config.protocol, output_config.protocol); EXPECT_EQ(config.ordered, output_config.ordered); EXPECT_EQ(config.maxRetransmits, output_config.maxRetransmits); EXPECT_FALSE(output_config.maxRetransmitTime); } TEST_F(SctpUtilsTest, WriteParseOpenMessageWithPriority) { webrtc::DataChannelInit config; std::string label = "abc"; config.protocol = "y"; config.priority = webrtc::Priority::kVeryLow; rtc::CopyOnWriteBuffer packet; ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); VerifyOpenMessageFormat(packet, label, config); std::string output_label; webrtc::DataChannelInit output_config; ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, &output_config)); EXPECT_EQ(label, output_label); ASSERT_TRUE(output_config.priority); EXPECT_EQ(*config.priority, *output_config.priority); } TEST_F(SctpUtilsTest, WriteParseAckMessage) { rtc::CopyOnWriteBuffer packet; webrtc::WriteDataChannelOpenAckMessage(&packet); uint8_t message_type; rtc::ByteBufferReader buffer(packet); ASSERT_TRUE(buffer.ReadUInt8(&message_type)); EXPECT_EQ(0x02, message_type); EXPECT_TRUE(webrtc::ParseDataChannelOpenAckMessage(packet)); } TEST_F(SctpUtilsTest, TestIsOpenMessage) { rtc::CopyOnWriteBuffer open(1); open.MutableData()[0] = 0x03; EXPECT_TRUE(webrtc::IsOpenMessage(open)); rtc::CopyOnWriteBuffer openAck(1); openAck.MutableData()[0] = 0x02; EXPECT_FALSE(webrtc::IsOpenMessage(openAck)); rtc::CopyOnWriteBuffer invalid(1); invalid.MutableData()[0] = 0x01; EXPECT_FALSE(webrtc::IsOpenMessage(invalid)); rtc::CopyOnWriteBuffer empty; EXPECT_FALSE(webrtc::IsOpenMessage(empty)); } TEST(SctpSidTest, Basics) { // These static asserts are mostly here to aid with readability (i.e. knowing // what these constants represent). static_assert(cricket::kMinSctpSid == 0, "Min stream id should be 0"); static_assert(cricket::kMaxSctpSid <= cricket::kSpecMaxSctpSid, ""); static_assert( cricket::kSpecMaxSctpSid == std::numeric_limits::max(), "Max legal sctp stream value should be 0xffff"); // cricket::kMaxSctpSid is a chosen value in the webrtc implementation, // the highest generated `sid` value chosen for resource reservation reasons. // It's one less than kMaxSctpStreams (1024) or 1023 since sid values are // zero based. EXPECT_TRUE(!StreamId(-1).HasValue()); EXPECT_TRUE(!StreamId(-2).HasValue()); EXPECT_TRUE(StreamId(cricket::kMinSctpSid).HasValue()); EXPECT_TRUE(StreamId(cricket::kMinSctpSid + 1).HasValue()); EXPECT_TRUE(StreamId(cricket::kSpecMaxSctpSid).HasValue()); EXPECT_TRUE(StreamId(cricket::kMaxSctpSid).HasValue()); // Two illegal values are equal (both not valid). EXPECT_EQ(StreamId(-1), StreamId(-2)); // Two different, but legal, values, are not equal. EXPECT_NE(StreamId(1), StreamId(2)); // Test operator<() for container compatibility. EXPECT_LT(StreamId(1), StreamId(2)); // Test assignment, value() and reset(). StreamId sid1; StreamId sid2(cricket::kMaxSctpSid); EXPECT_NE(sid1, sid2); sid1 = sid2; EXPECT_EQ(sid1, sid2); EXPECT_EQ(sid1.stream_id_int(), cricket::kMaxSctpSid); EXPECT_TRUE(sid1.HasValue()); sid1.reset(); EXPECT_FALSE(sid1.HasValue()); }