Stefan Holmer | 1acbd68 | 2017-09-01 13:29:28 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2017 The WebRTC project authors. All Rights Reserved. |
| 3 | * |
| 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. |
| 9 | */ |
| 10 | |
Mirko Bonadei | 92ea95e | 2017-09-15 04:47:31 | [diff] [blame] | 11 | #include "call/rtp_config.h" |
Stefan Holmer | dbdb3a0 | 2018-07-17 14:03:46 | [diff] [blame] | 12 | |
Harald Alvestrand | 93c9aa1 | 2024-09-02 20:55:52 | [diff] [blame] | 13 | #include <algorithm> |
| 14 | #include <cstddef> |
Yves Gerey | 988cc08 | 2018-10-23 10:03:01 | [diff] [blame] | 15 | #include <cstdint> |
Harald Alvestrand | 93c9aa1 | 2024-09-02 20:55:52 | [diff] [blame] | 16 | #include <iterator> |
| 17 | #include <optional> |
| 18 | #include <string> |
| 19 | #include <vector> |
Yves Gerey | 988cc08 | 2018-10-23 10:03:01 | [diff] [blame] | 20 | |
Henrik Boström | f45ca37 | 2020-03-24 12:30:50 | [diff] [blame] | 21 | #include "absl/algorithm/container.h" |
Yves Gerey | 988cc08 | 2018-10-23 10:03:01 | [diff] [blame] | 22 | #include "api/array_view.h" |
Harald Alvestrand | 93c9aa1 | 2024-09-02 20:55:52 | [diff] [blame] | 23 | #include "api/rtp_headers.h" |
Henrik Boström | f45ca37 | 2020-03-24 12:30:50 | [diff] [blame] | 24 | #include "rtc_base/checks.h" |
Jonas Olsson | 0a713b6 | 2018-04-04 13:49:32 | [diff] [blame] | 25 | #include "rtc_base/strings/string_builder.h" |
Stefan Holmer | 1acbd68 | 2017-09-01 13:29:28 | [diff] [blame] | 26 | |
| 27 | namespace webrtc { |
| 28 | |
Henrik Boström | f45ca37 | 2020-03-24 12:30:50 | [diff] [blame] | 29 | namespace { |
| 30 | |
| 31 | uint32_t FindAssociatedSsrc(uint32_t ssrc, |
| 32 | const std::vector<uint32_t>& ssrcs, |
| 33 | const std::vector<uint32_t>& associated_ssrcs) { |
| 34 | RTC_DCHECK_EQ(ssrcs.size(), associated_ssrcs.size()); |
| 35 | for (size_t i = 0; i < ssrcs.size(); ++i) { |
| 36 | if (ssrcs[i] == ssrc) |
| 37 | return associated_ssrcs[i]; |
| 38 | } |
Artem Titov | d325196 | 2021-11-15 15:57:07 | [diff] [blame] | 39 | RTC_DCHECK_NOTREACHED(); |
Henrik Boström | f45ca37 | 2020-03-24 12:30:50 | [diff] [blame] | 40 | return 0; |
| 41 | } |
| 42 | |
| 43 | } // namespace |
| 44 | |
Elad Alon | fadb181 | 2019-05-24 11:40:02 | [diff] [blame] | 45 | std::string LntfConfig::ToString() const { |
| 46 | return enabled ? "{enabled: true}" : "{enabled: false}"; |
| 47 | } |
| 48 | |
Stefan Holmer | 1acbd68 | 2017-09-01 13:29:28 | [diff] [blame] | 49 | std::string NackConfig::ToString() const { |
Jonas Olsson | 0a713b6 | 2018-04-04 13:49:32 | [diff] [blame] | 50 | char buf[1024]; |
| 51 | rtc::SimpleStringBuilder ss(buf); |
Stefan Holmer | 1acbd68 | 2017-09-01 13:29:28 | [diff] [blame] | 52 | ss << "{rtp_history_ms: " << rtp_history_ms; |
| 53 | ss << '}'; |
| 54 | return ss.str(); |
| 55 | } |
| 56 | |
| 57 | std::string UlpfecConfig::ToString() const { |
Jonas Olsson | 0a713b6 | 2018-04-04 13:49:32 | [diff] [blame] | 58 | char buf[1024]; |
| 59 | rtc::SimpleStringBuilder ss(buf); |
Stefan Holmer | 1acbd68 | 2017-09-01 13:29:28 | [diff] [blame] | 60 | ss << "{ulpfec_payload_type: " << ulpfec_payload_type; |
| 61 | ss << ", red_payload_type: " << red_payload_type; |
| 62 | ss << ", red_rtx_payload_type: " << red_rtx_payload_type; |
| 63 | ss << '}'; |
| 64 | return ss.str(); |
| 65 | } |
| 66 | |
| 67 | bool UlpfecConfig::operator==(const UlpfecConfig& other) const { |
| 68 | return ulpfec_payload_type == other.ulpfec_payload_type && |
| 69 | red_payload_type == other.red_payload_type && |
| 70 | red_rtx_payload_type == other.red_rtx_payload_type; |
| 71 | } |
Stefan Holmer | dbdb3a0 | 2018-07-17 14:03:46 | [diff] [blame] | 72 | |
| 73 | RtpConfig::RtpConfig() = default; |
| 74 | RtpConfig::RtpConfig(const RtpConfig&) = default; |
| 75 | RtpConfig::~RtpConfig() = default; |
| 76 | |
| 77 | RtpConfig::Flexfec::Flexfec() = default; |
| 78 | RtpConfig::Flexfec::Flexfec(const Flexfec&) = default; |
| 79 | RtpConfig::Flexfec::~Flexfec() = default; |
| 80 | |
| 81 | std::string RtpConfig::ToString() const { |
| 82 | char buf[2 * 1024]; |
| 83 | rtc::SimpleStringBuilder ss(buf); |
| 84 | ss << "{ssrcs: ["; |
| 85 | for (size_t i = 0; i < ssrcs.size(); ++i) { |
| 86 | ss << ssrcs[i]; |
| 87 | if (i != ssrcs.size() - 1) |
| 88 | ss << ", "; |
| 89 | } |
Rasmus Brandt | 9df698c | 2020-03-09 12:53:26 | [diff] [blame] | 90 | ss << "], rids: ["; |
| 91 | for (size_t i = 0; i < rids.size(); ++i) { |
| 92 | ss << rids[i]; |
| 93 | if (i != rids.size() - 1) |
| 94 | ss << ", "; |
| 95 | } |
| 96 | ss << "], mid: '" << mid << "'"; |
Stefan Holmer | dbdb3a0 | 2018-07-17 14:03:46 | [diff] [blame] | 97 | ss << ", rtcp_mode: " |
| 98 | << (rtcp_mode == RtcpMode::kCompound ? "RtcpMode::kCompound" |
| 99 | : "RtcpMode::kReducedSize"); |
| 100 | ss << ", max_packet_size: " << max_packet_size; |
Johannes Kron | 9190b82 | 2018-10-29 10:22:05 | [diff] [blame] | 101 | ss << ", extmap-allow-mixed: " << (extmap_allow_mixed ? "true" : "false"); |
Stefan Holmer | dbdb3a0 | 2018-07-17 14:03:46 | [diff] [blame] | 102 | ss << ", extensions: ["; |
| 103 | for (size_t i = 0; i < extensions.size(); ++i) { |
| 104 | ss << extensions[i].ToString(); |
| 105 | if (i != extensions.size() - 1) |
| 106 | ss << ", "; |
| 107 | } |
| 108 | ss << ']'; |
| 109 | |
Elad Alon | fadb181 | 2019-05-24 11:40:02 | [diff] [blame] | 110 | ss << ", lntf: " << lntf.ToString(); |
Stefan Holmer | dbdb3a0 | 2018-07-17 14:03:46 | [diff] [blame] | 111 | ss << ", nack: {rtp_history_ms: " << nack.rtp_history_ms << '}'; |
| 112 | ss << ", ulpfec: " << ulpfec.ToString(); |
| 113 | ss << ", payload_name: " << payload_name; |
| 114 | ss << ", payload_type: " << payload_type; |
Mirta Dvornicic | fe68daa | 2019-05-23 11:21:12 | [diff] [blame] | 115 | ss << ", raw_payload: " << (raw_payload ? "true" : "false"); |
Stefan Holmer | dbdb3a0 | 2018-07-17 14:03:46 | [diff] [blame] | 116 | |
| 117 | ss << ", flexfec: {payload_type: " << flexfec.payload_type; |
| 118 | ss << ", ssrc: " << flexfec.ssrc; |
| 119 | ss << ", protected_media_ssrcs: ["; |
| 120 | for (size_t i = 0; i < flexfec.protected_media_ssrcs.size(); ++i) { |
| 121 | ss << flexfec.protected_media_ssrcs[i]; |
| 122 | if (i != flexfec.protected_media_ssrcs.size() - 1) |
| 123 | ss << ", "; |
| 124 | } |
| 125 | ss << "]}"; |
| 126 | |
| 127 | ss << ", rtx: " << rtx.ToString(); |
| 128 | ss << ", c_name: " << c_name; |
| 129 | ss << '}'; |
| 130 | return ss.str(); |
| 131 | } |
| 132 | |
| 133 | RtpConfig::Rtx::Rtx() = default; |
| 134 | RtpConfig::Rtx::Rtx(const Rtx&) = default; |
| 135 | RtpConfig::Rtx::~Rtx() = default; |
| 136 | |
| 137 | std::string RtpConfig::Rtx::ToString() const { |
| 138 | char buf[1024]; |
| 139 | rtc::SimpleStringBuilder ss(buf); |
| 140 | ss << "{ssrcs: ["; |
| 141 | for (size_t i = 0; i < ssrcs.size(); ++i) { |
| 142 | ss << ssrcs[i]; |
| 143 | if (i != ssrcs.size() - 1) |
| 144 | ss << ", "; |
| 145 | } |
| 146 | ss << ']'; |
| 147 | |
| 148 | ss << ", payload_type: " << payload_type; |
| 149 | ss << '}'; |
| 150 | return ss.str(); |
| 151 | } |
Henrik Boström | f45ca37 | 2020-03-24 12:30:50 | [diff] [blame] | 152 | |
| 153 | bool RtpConfig::IsMediaSsrc(uint32_t ssrc) const { |
| 154 | return absl::c_linear_search(ssrcs, ssrc); |
| 155 | } |
| 156 | |
| 157 | bool RtpConfig::IsRtxSsrc(uint32_t ssrc) const { |
| 158 | return absl::c_linear_search(rtx.ssrcs, ssrc); |
| 159 | } |
| 160 | |
| 161 | bool RtpConfig::IsFlexfecSsrc(uint32_t ssrc) const { |
| 162 | return flexfec.payload_type != -1 && ssrc == flexfec.ssrc; |
| 163 | } |
| 164 | |
Florent Castelli | 8037fc6 | 2024-08-29 13:00:40 | [diff] [blame] | 165 | std::optional<uint32_t> RtpConfig::GetRtxSsrcAssociatedWithMediaSsrc( |
Henrik Boström | f45ca37 | 2020-03-24 12:30:50 | [diff] [blame] | 166 | uint32_t media_ssrc) const { |
| 167 | RTC_DCHECK(IsMediaSsrc(media_ssrc)); |
| 168 | // If we don't use RTX there is no association. |
| 169 | if (rtx.ssrcs.empty()) |
Florent Castelli | 8037fc6 | 2024-08-29 13:00:40 | [diff] [blame] | 170 | return std::nullopt; |
Henrik Boström | f45ca37 | 2020-03-24 12:30:50 | [diff] [blame] | 171 | // If we use RTX there MUST be an association ssrcs[i] <-> rtx.ssrcs[i]. |
| 172 | RTC_DCHECK_EQ(ssrcs.size(), rtx.ssrcs.size()); |
| 173 | return FindAssociatedSsrc(media_ssrc, ssrcs, rtx.ssrcs); |
| 174 | } |
| 175 | |
| 176 | uint32_t RtpConfig::GetMediaSsrcAssociatedWithRtxSsrc(uint32_t rtx_ssrc) const { |
| 177 | RTC_DCHECK(IsRtxSsrc(rtx_ssrc)); |
| 178 | // If we use RTX there MUST be an association ssrcs[i] <-> rtx.ssrcs[i]. |
| 179 | RTC_DCHECK_EQ(ssrcs.size(), rtx.ssrcs.size()); |
| 180 | return FindAssociatedSsrc(rtx_ssrc, rtx.ssrcs, ssrcs); |
| 181 | } |
| 182 | |
| 183 | uint32_t RtpConfig::GetMediaSsrcAssociatedWithFlexfecSsrc( |
| 184 | uint32_t flexfec_ssrc) const { |
| 185 | RTC_DCHECK(IsFlexfecSsrc(flexfec_ssrc)); |
| 186 | // If we use FlexFEC there MUST be an associated media ssrc. |
| 187 | // |
| 188 | // TODO(brandtr/hbos): The current implementation only supports an association |
| 189 | // with a single media ssrc. If multiple ssrcs are to be supported in the |
| 190 | // future, in order not to break GetStats()'s packet and byte counters, we |
| 191 | // must be able to tell how many packets and bytes have contributed to which |
| 192 | // SSRC. |
| 193 | RTC_DCHECK_EQ(1u, flexfec.protected_media_ssrcs.size()); |
| 194 | uint32_t media_ssrc = flexfec.protected_media_ssrcs[0]; |
| 195 | RTC_DCHECK(IsMediaSsrc(media_ssrc)); |
| 196 | return media_ssrc; |
| 197 | } |
| 198 | |
Florent Castelli | 8037fc6 | 2024-08-29 13:00:40 | [diff] [blame] | 199 | std::optional<std::string> RtpConfig::GetRidForSsrc(uint32_t ssrc) const { |
Henrik Boström | a0ff50c | 2020-05-05 13:54:46 | [diff] [blame] | 200 | auto it = std::find(ssrcs.begin(), ssrcs.end(), ssrc); |
| 201 | if (it != ssrcs.end()) { |
| 202 | size_t ssrc_index = std::distance(ssrcs.begin(), it); |
| 203 | if (ssrc_index < rids.size()) { |
| 204 | return rids[ssrc_index]; |
| 205 | } |
| 206 | } |
Florent Castelli | 8037fc6 | 2024-08-29 13:00:40 | [diff] [blame] | 207 | return std::nullopt; |
Henrik Boström | a0ff50c | 2020-05-05 13:54:46 | [diff] [blame] | 208 | } |
| 209 | |
Stefan Holmer | 1acbd68 | 2017-09-01 13:29:28 | [diff] [blame] | 210 | } // namespace webrtc |