wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 1 | /* |
kjellander | b24317b | 2016-02-10 15:54:43 | [diff] [blame] | 2 | * Copyright 2013 The WebRTC project authors. All Rights Reserved. |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 3 | * |
kjellander | b24317b | 2016-02-10 15:54:43 | [diff] [blame] | 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 9 | */ |
| 10 | |
Steve Anton | 10542f2 | 2019-01-11 17:11:00 | [diff] [blame] | 11 | #include "pc/sctp_utils.h" |
Yves Gerey | 3e70781 | 2018-11-28 15:47:49 | [diff] [blame] | 12 | |
| 13 | #include <stdint.h> |
| 14 | |
Tommi | 492296c | 2023-03-12 15:59:25 | [diff] [blame] | 15 | #include <limits> |
| 16 | |
Harald Alvestrand | c24a218 | 2022-02-23 13:44:59 | [diff] [blame] | 17 | #include "absl/types/optional.h" |
| 18 | #include "api/priority.h" |
Tommi | 492296c | 2023-03-12 15:59:25 | [diff] [blame] | 19 | #include "media/sctp/sctp_transport_internal.h" |
Steve Anton | 10542f2 | 2019-01-11 17:11:00 | [diff] [blame] | 20 | #include "rtc_base/byte_buffer.h" |
| 21 | #include "rtc_base/copy_on_write_buffer.h" |
Yves Gerey | 3e70781 | 2018-11-28 15:47:49 | [diff] [blame] | 22 | #include "test/gtest.h" |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 23 | |
Tommi | 492296c | 2023-03-12 15:59:25 | [diff] [blame] | 24 | using webrtc::StreamId; |
| 25 | |
Mirko Bonadei | 6a489f2 | 2019-04-09 13:11:12 | [diff] [blame] | 26 | class SctpUtilsTest : public ::testing::Test { |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 27 | public: |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 28 | void VerifyOpenMessageFormat(const rtc::CopyOnWriteBuffer& packet, |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 29 | const std::string& label, |
| 30 | const webrtc::DataChannelInit& config) { |
Peter Boström | 0c4e06b | 2015-10-07 10:23:21 | [diff] [blame] | 31 | uint8_t message_type; |
| 32 | uint8_t channel_type; |
| 33 | uint32_t reliability; |
| 34 | uint16_t priority; |
| 35 | uint16_t label_length; |
| 36 | uint16_t protocol_length; |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 37 | |
Harald Alvestrand | 572502c | 2023-11-22 10:19:06 | [diff] [blame] | 38 | rtc::ByteBufferReader buffer(packet); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 39 | ASSERT_TRUE(buffer.ReadUInt8(&message_type)); |
| 40 | EXPECT_EQ(0x03, message_type); |
| 41 | |
| 42 | ASSERT_TRUE(buffer.ReadUInt8(&channel_type)); |
| 43 | if (config.ordered) { |
Harald Alvestrand | f3736ed | 2019-04-08 11:09:30 | [diff] [blame] | 44 | EXPECT_EQ( |
| 45 | config.maxRetransmits ? 0x01 : (config.maxRetransmitTime ? 0x02 : 0), |
| 46 | channel_type); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 47 | } else { |
Harald Alvestrand | f3736ed | 2019-04-08 11:09:30 | [diff] [blame] | 48 | EXPECT_EQ(config.maxRetransmits |
Yves Gerey | 665174f | 2018-06-19 13:03:05 | [diff] [blame] | 49 | ? 0x81 |
Harald Alvestrand | f3736ed | 2019-04-08 11:09:30 | [diff] [blame] | 50 | : (config.maxRetransmitTime ? 0x82 : 0x80), |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 51 | channel_type); |
| 52 | } |
| 53 | |
wu@webrtc.org | 97077a3 | 2013-10-25 21:18:33 | [diff] [blame] | 54 | ASSERT_TRUE(buffer.ReadUInt16(&priority)); |
Harald Alvestrand | fd5ae7f | 2020-05-16 06:37:49 | [diff] [blame] | 55 | if (config.priority) { |
| 56 | // Exact values are checked by round-trip conversion, but |
| 57 | // all values defined are greater than zero. |
| 58 | EXPECT_GT(priority, 0); |
| 59 | } else { |
| 60 | EXPECT_EQ(priority, 0); |
| 61 | } |
wu@webrtc.org | 97077a3 | 2013-10-25 21:18:33 | [diff] [blame] | 62 | |
| 63 | ASSERT_TRUE(buffer.ReadUInt32(&reliability)); |
Harald Alvestrand | f3736ed | 2019-04-08 11:09:30 | [diff] [blame] | 64 | if (config.maxRetransmits || config.maxRetransmitTime) { |
| 65 | EXPECT_EQ(config.maxRetransmits ? *config.maxRetransmits |
| 66 | : *config.maxRetransmitTime, |
wu@webrtc.org | 97077a3 | 2013-10-25 21:18:33 | [diff] [blame] | 67 | static_cast<int>(reliability)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 68 | } |
| 69 | |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 70 | ASSERT_TRUE(buffer.ReadUInt16(&label_length)); |
| 71 | ASSERT_TRUE(buffer.ReadUInt16(&protocol_length)); |
| 72 | EXPECT_EQ(label.size(), label_length); |
| 73 | EXPECT_EQ(config.protocol.size(), protocol_length); |
| 74 | |
Tommi | eb4a314 | 2024-01-15 20:06:05 | [diff] [blame] | 75 | absl::string_view label_output; |
| 76 | ASSERT_TRUE(buffer.ReadStringView(&label_output, label_length)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 77 | EXPECT_EQ(label, label_output); |
Tommi | eb4a314 | 2024-01-15 20:06:05 | [diff] [blame] | 78 | absl::string_view protocol_output; |
| 79 | ASSERT_TRUE(buffer.ReadStringView(&protocol_output, protocol_length)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 80 | EXPECT_EQ(config.protocol, protocol_output); |
| 81 | } |
| 82 | }; |
| 83 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 84 | TEST_F(SctpUtilsTest, WriteParseOpenMessageWithOrderedReliable) { |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 85 | webrtc::DataChannelInit config; |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 86 | std::string label = "abc"; |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 87 | config.protocol = "y"; |
| 88 | |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 89 | rtc::CopyOnWriteBuffer packet; |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 90 | ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 91 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 92 | VerifyOpenMessageFormat(packet, label, config); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 93 | |
| 94 | std::string output_label; |
| 95 | webrtc::DataChannelInit output_config; |
Yves Gerey | 665174f | 2018-06-19 13:03:05 | [diff] [blame] | 96 | ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, |
| 97 | &output_config)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 98 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 99 | EXPECT_EQ(label, output_label); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 100 | EXPECT_EQ(config.protocol, output_config.protocol); |
| 101 | EXPECT_EQ(config.ordered, output_config.ordered); |
| 102 | EXPECT_EQ(config.maxRetransmitTime, output_config.maxRetransmitTime); |
| 103 | EXPECT_EQ(config.maxRetransmits, output_config.maxRetransmits); |
| 104 | } |
| 105 | |
| 106 | TEST_F(SctpUtilsTest, WriteParseOpenMessageWithMaxRetransmitTime) { |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 107 | webrtc::DataChannelInit config; |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 108 | std::string label = "abc"; |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 109 | config.ordered = false; |
| 110 | config.maxRetransmitTime = 10; |
| 111 | config.protocol = "y"; |
| 112 | |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 113 | rtc::CopyOnWriteBuffer packet; |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 114 | ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 115 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 116 | VerifyOpenMessageFormat(packet, label, config); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 117 | |
| 118 | std::string output_label; |
| 119 | webrtc::DataChannelInit output_config; |
Yves Gerey | 665174f | 2018-06-19 13:03:05 | [diff] [blame] | 120 | ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, |
| 121 | &output_config)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 122 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 123 | EXPECT_EQ(label, output_label); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 124 | EXPECT_EQ(config.protocol, output_config.protocol); |
| 125 | EXPECT_EQ(config.ordered, output_config.ordered); |
Harald Alvestrand | f3736ed | 2019-04-08 11:09:30 | [diff] [blame] | 126 | EXPECT_EQ(*config.maxRetransmitTime, *output_config.maxRetransmitTime); |
| 127 | EXPECT_FALSE(output_config.maxRetransmits); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 128 | } |
| 129 | |
| 130 | TEST_F(SctpUtilsTest, WriteParseOpenMessageWithMaxRetransmits) { |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 131 | webrtc::DataChannelInit config; |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 132 | std::string label = "abc"; |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 133 | config.maxRetransmits = 10; |
| 134 | config.protocol = "y"; |
| 135 | |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 136 | rtc::CopyOnWriteBuffer packet; |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 137 | ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 138 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 139 | VerifyOpenMessageFormat(packet, label, config); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 140 | |
| 141 | std::string output_label; |
| 142 | webrtc::DataChannelInit output_config; |
Yves Gerey | 665174f | 2018-06-19 13:03:05 | [diff] [blame] | 143 | ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, |
| 144 | &output_config)); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 145 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 146 | EXPECT_EQ(label, output_label); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 147 | EXPECT_EQ(config.protocol, output_config.protocol); |
| 148 | EXPECT_EQ(config.ordered, output_config.ordered); |
| 149 | EXPECT_EQ(config.maxRetransmits, output_config.maxRetransmits); |
Harald Alvestrand | f3736ed | 2019-04-08 11:09:30 | [diff] [blame] | 150 | EXPECT_FALSE(output_config.maxRetransmitTime); |
wu@webrtc.org | 1d1ffc9 | 2013-10-16 18:12:02 | [diff] [blame] | 151 | } |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 152 | |
Harald Alvestrand | fd5ae7f | 2020-05-16 06:37:49 | [diff] [blame] | 153 | TEST_F(SctpUtilsTest, WriteParseOpenMessageWithPriority) { |
| 154 | webrtc::DataChannelInit config; |
| 155 | std::string label = "abc"; |
| 156 | config.protocol = "y"; |
| 157 | config.priority = webrtc::Priority::kVeryLow; |
| 158 | |
| 159 | rtc::CopyOnWriteBuffer packet; |
| 160 | ASSERT_TRUE(webrtc::WriteDataChannelOpenMessage(label, config, &packet)); |
| 161 | |
| 162 | VerifyOpenMessageFormat(packet, label, config); |
| 163 | |
| 164 | std::string output_label; |
| 165 | webrtc::DataChannelInit output_config; |
| 166 | ASSERT_TRUE(webrtc::ParseDataChannelOpenMessage(packet, &output_label, |
| 167 | &output_config)); |
| 168 | |
| 169 | EXPECT_EQ(label, output_label); |
| 170 | ASSERT_TRUE(output_config.priority); |
| 171 | EXPECT_EQ(*config.priority, *output_config.priority); |
| 172 | } |
| 173 | |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 174 | TEST_F(SctpUtilsTest, WriteParseAckMessage) { |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 175 | rtc::CopyOnWriteBuffer packet; |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 176 | webrtc::WriteDataChannelOpenAckMessage(&packet); |
| 177 | |
Peter Boström | 0c4e06b | 2015-10-07 10:23:21 | [diff] [blame] | 178 | uint8_t message_type; |
Harald Alvestrand | 572502c | 2023-11-22 10:19:06 | [diff] [blame] | 179 | rtc::ByteBufferReader buffer(packet); |
henrika@webrtc.org | aebb1ad | 2014-01-14 10:00:58 | [diff] [blame] | 180 | ASSERT_TRUE(buffer.ReadUInt8(&message_type)); |
| 181 | EXPECT_EQ(0x02, message_type); |
| 182 | |
| 183 | EXPECT_TRUE(webrtc::ParseDataChannelOpenAckMessage(packet)); |
| 184 | } |
deadbeef | ab9b2d1 | 2015-10-14 18:33:11 | [diff] [blame] | 185 | |
| 186 | TEST_F(SctpUtilsTest, TestIsOpenMessage) { |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 187 | rtc::CopyOnWriteBuffer open(1); |
Danil Chapovalov | e15dc58 | 2021-01-07 14:24:05 | [diff] [blame] | 188 | open.MutableData()[0] = 0x03; |
deadbeef | ab9b2d1 | 2015-10-14 18:33:11 | [diff] [blame] | 189 | EXPECT_TRUE(webrtc::IsOpenMessage(open)); |
| 190 | |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 191 | rtc::CopyOnWriteBuffer openAck(1); |
Danil Chapovalov | e15dc58 | 2021-01-07 14:24:05 | [diff] [blame] | 192 | openAck.MutableData()[0] = 0x02; |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 193 | EXPECT_FALSE(webrtc::IsOpenMessage(openAck)); |
deadbeef | ab9b2d1 | 2015-10-14 18:33:11 | [diff] [blame] | 194 | |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 195 | rtc::CopyOnWriteBuffer invalid(1); |
Danil Chapovalov | e15dc58 | 2021-01-07 14:24:05 | [diff] [blame] | 196 | invalid.MutableData()[0] = 0x01; |
deadbeef | ab9b2d1 | 2015-10-14 18:33:11 | [diff] [blame] | 197 | EXPECT_FALSE(webrtc::IsOpenMessage(invalid)); |
| 198 | |
ossu | d4d2f60 | 2016-11-08 10:05:32 | [diff] [blame] | 199 | rtc::CopyOnWriteBuffer empty; |
deadbeef | ab9b2d1 | 2015-10-14 18:33:11 | [diff] [blame] | 200 | EXPECT_FALSE(webrtc::IsOpenMessage(empty)); |
| 201 | } |
Tommi | 492296c | 2023-03-12 15:59:25 | [diff] [blame] | 202 | |
| 203 | TEST(SctpSidTest, Basics) { |
| 204 | // These static asserts are mostly here to aid with readability (i.e. knowing |
| 205 | // what these constants represent). |
| 206 | static_assert(cricket::kMinSctpSid == 0, "Min stream id should be 0"); |
| 207 | static_assert(cricket::kMaxSctpSid <= cricket::kSpecMaxSctpSid, ""); |
| 208 | static_assert( |
| 209 | cricket::kSpecMaxSctpSid == std::numeric_limits<uint16_t>::max(), |
| 210 | "Max legal sctp stream value should be 0xffff"); |
Tommi | 492296c | 2023-03-12 15:59:25 | [diff] [blame] | 211 | } |