| /* | 
 |  *  Copyright (c) 2017 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 "call/rtp_config.h" | 
 |  | 
 | #include <algorithm> | 
 | #include <cstddef> | 
 | #include <cstdint> | 
 | #include <iterator> | 
 | #include <optional> | 
 | #include <string> | 
 | #include <vector> | 
 |  | 
 | #include "absl/algorithm/container.h" | 
 | #include "api/array_view.h" | 
 | #include "api/rtp_headers.h" | 
 | #include "rtc_base/checks.h" | 
 | #include "rtc_base/strings/string_builder.h" | 
 |  | 
 | namespace webrtc { | 
 |  | 
 | namespace { | 
 |  | 
 | uint32_t FindAssociatedSsrc(uint32_t ssrc, | 
 |                             const std::vector<uint32_t>& ssrcs, | 
 |                             const std::vector<uint32_t>& associated_ssrcs) { | 
 |   RTC_DCHECK_EQ(ssrcs.size(), associated_ssrcs.size()); | 
 |   for (size_t i = 0; i < ssrcs.size(); ++i) { | 
 |     if (ssrcs[i] == ssrc) | 
 |       return associated_ssrcs[i]; | 
 |   } | 
 |   RTC_DCHECK_NOTREACHED(); | 
 |   return 0; | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | std::string LntfConfig::ToString() const { | 
 |   return enabled ? "{enabled: true}" : "{enabled: false}"; | 
 | } | 
 |  | 
 | std::string NackConfig::ToString() const { | 
 |   char buf[1024]; | 
 |   SimpleStringBuilder ss(buf); | 
 |   ss << "{rtp_history_ms: " << rtp_history_ms; | 
 |   ss << '}'; | 
 |   return ss.str(); | 
 | } | 
 |  | 
 | std::string UlpfecConfig::ToString() const { | 
 |   char buf[1024]; | 
 |   SimpleStringBuilder ss(buf); | 
 |   ss << "{ulpfec_payload_type: " << ulpfec_payload_type; | 
 |   ss << ", red_payload_type: " << red_payload_type; | 
 |   ss << ", red_rtx_payload_type: " << red_rtx_payload_type; | 
 |   ss << '}'; | 
 |   return ss.str(); | 
 | } | 
 |  | 
 | bool UlpfecConfig::operator==(const UlpfecConfig& other) const { | 
 |   return ulpfec_payload_type == other.ulpfec_payload_type && | 
 |          red_payload_type == other.red_payload_type && | 
 |          red_rtx_payload_type == other.red_rtx_payload_type; | 
 | } | 
 |  | 
 | std::string RtpStreamConfig::ToString() const { | 
 |   char buf[1024]; | 
 |   SimpleStringBuilder ss(buf); | 
 |   ss << "{ssrc: " << ssrc; | 
 |   ss << ", rid: " << rid; | 
 |   ss << ", payload_name: " << payload_name; | 
 |   ss << ", payload_type: " << payload_type; | 
 |   ss << ", raw_payload: " << (raw_payload ? "true" : "false"); | 
 |   if (rtx.has_value()) { | 
 |     ss << ", rtx: " << rtx->ToString(); | 
 |   } | 
 |   ss << '}'; | 
 |   return ss.str(); | 
 | } | 
 |  | 
 | std::string RtpStreamConfig::Rtx::ToString() const { | 
 |   char buf[1024]; | 
 |   SimpleStringBuilder ss(buf); | 
 |   ss << "{ssrc: " << ssrc; | 
 |   ss << ", payload_type: " << payload_type; | 
 |   ss << '}'; | 
 |   return ss.str(); | 
 | } | 
 |  | 
 | RtpConfig::RtpConfig() = default; | 
 | RtpConfig::RtpConfig(const RtpConfig&) = default; | 
 | RtpConfig::~RtpConfig() = default; | 
 |  | 
 | RtpConfig::Flexfec::Flexfec() = default; | 
 | RtpConfig::Flexfec::Flexfec(const Flexfec&) = default; | 
 | RtpConfig::Flexfec::~Flexfec() = default; | 
 |  | 
 | std::string RtpConfig::ToString() const { | 
 |   char buf[2 * 1024]; | 
 |   SimpleStringBuilder ss(buf); | 
 |   ss << "{ssrcs: ["; | 
 |   for (size_t i = 0; i < ssrcs.size(); ++i) { | 
 |     ss << ssrcs[i]; | 
 |     if (i != ssrcs.size() - 1) | 
 |       ss << ", "; | 
 |   } | 
 |   ss << "], rids: ["; | 
 |   for (size_t i = 0; i < rids.size(); ++i) { | 
 |     ss << rids[i]; | 
 |     if (i != rids.size() - 1) | 
 |       ss << ", "; | 
 |   } | 
 |   ss << "], mid: '" << mid << "'"; | 
 |   ss << ", rtcp_mode: " | 
 |      << (rtcp_mode == RtcpMode::kCompound ? "RtcpMode::kCompound" | 
 |                                           : "RtcpMode::kReducedSize"); | 
 |   ss << ", max_packet_size: " << max_packet_size; | 
 |   ss << ", extmap-allow-mixed: " << (extmap_allow_mixed ? "true" : "false"); | 
 |   ss << ", extensions: ["; | 
 |   for (size_t i = 0; i < extensions.size(); ++i) { | 
 |     ss << extensions[i].ToString(); | 
 |     if (i != extensions.size() - 1) | 
 |       ss << ", "; | 
 |   } | 
 |   ss << ']'; | 
 |  | 
 |   ss << ", lntf: " << lntf.ToString(); | 
 |   ss << ", nack: {rtp_history_ms: " << nack.rtp_history_ms << '}'; | 
 |   ss << ", ulpfec: " << ulpfec.ToString(); | 
 |   ss << ", payload_name: " << payload_name; | 
 |   ss << ", payload_type: " << payload_type; | 
 |   ss << ", raw_payload: " << (raw_payload ? "true" : "false"); | 
 |  | 
 |   ss << ", stream_configs: ["; | 
 |   for (size_t i = 0; i < stream_configs.size(); ++i) { | 
 |     ss << stream_configs[i].ToString(); | 
 |     if (i != stream_configs.size() - 1) | 
 |       ss << ", "; | 
 |   } | 
 |   ss << ']'; | 
 |  | 
 |   ss << ", flexfec: {payload_type: " << flexfec.payload_type; | 
 |   ss << ", ssrc: " << flexfec.ssrc; | 
 |   ss << ", protected_media_ssrcs: ["; | 
 |   for (size_t i = 0; i < flexfec.protected_media_ssrcs.size(); ++i) { | 
 |     ss << flexfec.protected_media_ssrcs[i]; | 
 |     if (i != flexfec.protected_media_ssrcs.size() - 1) | 
 |       ss << ", "; | 
 |   } | 
 |   ss << "]}"; | 
 |  | 
 |   ss << ", rtx: " << rtx.ToString(); | 
 |   ss << ", c_name: " << c_name; | 
 |   ss << '}'; | 
 |   return ss.str(); | 
 | } | 
 |  | 
 | RtpConfig::Rtx::Rtx() = default; | 
 | RtpConfig::Rtx::Rtx(const Rtx&) = default; | 
 | RtpConfig::Rtx::~Rtx() = default; | 
 |  | 
 | std::string RtpConfig::Rtx::ToString() const { | 
 |   char buf[1024]; | 
 |   SimpleStringBuilder ss(buf); | 
 |   ss << "{ssrcs: ["; | 
 |   for (size_t i = 0; i < ssrcs.size(); ++i) { | 
 |     ss << ssrcs[i]; | 
 |     if (i != ssrcs.size() - 1) | 
 |       ss << ", "; | 
 |   } | 
 |   ss << ']'; | 
 |  | 
 |   ss << ", payload_type: " << payload_type; | 
 |   ss << '}'; | 
 |   return ss.str(); | 
 | } | 
 |  | 
 | bool RtpConfig::IsMediaSsrc(uint32_t ssrc) const { | 
 |   return absl::c_linear_search(ssrcs, ssrc); | 
 | } | 
 |  | 
 | bool RtpConfig::IsRtxSsrc(uint32_t ssrc) const { | 
 |   return absl::c_linear_search(rtx.ssrcs, ssrc); | 
 | } | 
 |  | 
 | bool RtpConfig::IsFlexfecSsrc(uint32_t ssrc) const { | 
 |   return flexfec.payload_type != -1 && ssrc == flexfec.ssrc; | 
 | } | 
 |  | 
 | std::optional<uint32_t> RtpConfig::GetRtxSsrcAssociatedWithMediaSsrc( | 
 |     uint32_t media_ssrc) const { | 
 |   RTC_DCHECK(IsMediaSsrc(media_ssrc)); | 
 |   // If we don't use RTX there is no association. | 
 |   if (rtx.ssrcs.empty()) | 
 |     return std::nullopt; | 
 |   // If we use RTX there MUST be an association ssrcs[i] <-> rtx.ssrcs[i]. | 
 |   RTC_DCHECK_EQ(ssrcs.size(), rtx.ssrcs.size()); | 
 |   return FindAssociatedSsrc(media_ssrc, ssrcs, rtx.ssrcs); | 
 | } | 
 |  | 
 | uint32_t RtpConfig::GetMediaSsrcAssociatedWithRtxSsrc(uint32_t rtx_ssrc) const { | 
 |   RTC_DCHECK(IsRtxSsrc(rtx_ssrc)); | 
 |   // If we use RTX there MUST be an association ssrcs[i] <-> rtx.ssrcs[i]. | 
 |   RTC_DCHECK_EQ(ssrcs.size(), rtx.ssrcs.size()); | 
 |   return FindAssociatedSsrc(rtx_ssrc, rtx.ssrcs, ssrcs); | 
 | } | 
 |  | 
 | uint32_t RtpConfig::GetMediaSsrcAssociatedWithFlexfecSsrc( | 
 |     uint32_t flexfec_ssrc) const { | 
 |   RTC_DCHECK(IsFlexfecSsrc(flexfec_ssrc)); | 
 |   // If we use FlexFEC there MUST be an associated media ssrc. | 
 |   // | 
 |   // TODO(brandtr/hbos): The current implementation only supports an association | 
 |   // with a single media ssrc. If multiple ssrcs are to be supported in the | 
 |   // future, in order not to break GetStats()'s packet and byte counters, we | 
 |   // must be able to tell how many packets and bytes have contributed to which | 
 |   // SSRC. | 
 |   RTC_DCHECK_EQ(1u, flexfec.protected_media_ssrcs.size()); | 
 |   uint32_t media_ssrc = flexfec.protected_media_ssrcs[0]; | 
 |   RTC_DCHECK(IsMediaSsrc(media_ssrc)); | 
 |   return media_ssrc; | 
 | } | 
 |  | 
 | std::optional<std::string> RtpConfig::GetRidForSsrc(uint32_t ssrc) const { | 
 |   auto it = std::find(ssrcs.begin(), ssrcs.end(), ssrc); | 
 |   if (it != ssrcs.end()) { | 
 |     size_t ssrc_index = std::distance(ssrcs.begin(), it); | 
 |     if (ssrc_index < rids.size()) { | 
 |       return rids[ssrc_index]; | 
 |     } | 
 |   } | 
 |   return std::nullopt; | 
 | } | 
 |  | 
 | RtpStreamConfig RtpConfig::GetStreamConfig(size_t index) const { | 
 |   // GetStreamConfig function usually returns stream_configs[index], but if | 
 |   // stream_configs is not initialized (i.e., index >= stream_configs.size()), | 
 |   // it creates and returns an RtpStreamConfig using fields such as ssrcs, rids, | 
 |   // payload_name, and payload_type from RtpConfig. | 
 |   RTC_DCHECK_LT(index, ssrcs.size()); | 
 |   if (index < stream_configs.size()) { | 
 |     return stream_configs[index]; | 
 |   } | 
 |   RtpStreamConfig stream_config; | 
 |   stream_config.ssrc = ssrcs[index]; | 
 |   if (index < rids.size()) { | 
 |     stream_config.rid = rids[index]; | 
 |   } | 
 |   stream_config.payload_name = payload_name; | 
 |   stream_config.payload_type = payload_type; | 
 |   stream_config.raw_payload = raw_payload; | 
 |   if (!rtx.ssrcs.empty()) { | 
 |     RTC_DCHECK_EQ(ssrcs.size(), rtx.ssrcs.size()); | 
 |     auto& stream_config_rtx = stream_config.rtx.emplace(); | 
 |     stream_config_rtx.ssrc = rtx.ssrcs[index]; | 
 |     stream_config_rtx.payload_type = rtx.payload_type; | 
 |   } | 
 |  | 
 |   return stream_config; | 
 | } | 
 |  | 
 | }  // namespace webrtc |