blob: e4c9ea24d2cb3a45353cbb9f4503acf83d8429f5 [file] [log] [blame]
deadbeefe814a0d2017-02-25 18:15:09 -08001/*
2 * Copyright 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 Bonadei92ea95e2017-09-15 06:47:31 +020011#include "ortc/rtptransportcontrolleradapter.h"
deadbeefe814a0d2017-02-25 18:15:09 -080012
13#include <algorithm> // For "remove", "find".
deadbeefe814a0d2017-02-25 18:15:09 -080014#include <set>
zhihuangd3501ad2017-03-03 14:39:06 -080015#include <sstream>
deadbeefe814a0d2017-02-25 18:15:09 -080016#include <unordered_map>
17#include <utility> // For std::move.
18
Karl Wiberg918f50c2018-07-05 11:40:33 +020019#include "absl/memory/memory.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020020#include "api/proxy.h"
21#include "media/base/mediaconstants.h"
22#include "ortc/ortcrtpreceiveradapter.h"
23#include "ortc/ortcrtpsenderadapter.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020024#include "ortc/rtptransportadapter.h"
Steve Anton1d03a752017-11-27 14:30:09 -080025#include "pc/rtpmediautils.h"
Florent Castelli72b751a2018-06-28 14:09:33 +020026#include "pc/rtpparametersconversion.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020027#include "rtc_base/checks.h"
deadbeefe814a0d2017-02-25 18:15:09 -080028
29namespace webrtc {
30
31// Note: It's assumed that each individual list doesn't have conflicts, since
32// they should have been detected already by rtpparametersconversion.cc. This
33// only needs to detect conflicts *between* A and B.
34template <typename C1, typename C2>
35static RTCError CheckForIdConflicts(
36 const std::vector<C1>& codecs_a,
37 const cricket::RtpHeaderExtensions& extensions_a,
38 const cricket::StreamParamsVec& streams_a,
39 const std::vector<C2>& codecs_b,
40 const cricket::RtpHeaderExtensions& extensions_b,
41 const cricket::StreamParamsVec& streams_b) {
42 std::ostringstream oss;
43 // Since it's assumed that C1 and C2 are different types, codecs_a and
44 // codecs_b should never contain the same payload type, and thus we can just
45 // use a set.
46 std::set<int> seen_payload_types;
47 for (const C1& codec : codecs_a) {
48 seen_payload_types.insert(codec.id);
49 }
50 for (const C2& codec : codecs_b) {
51 if (!seen_payload_types.insert(codec.id).second) {
52 oss << "Same payload type used for audio and video codecs: " << codec.id;
53 LOG_AND_RETURN_ERROR(RTCErrorType::INVALID_PARAMETER, oss.str());
54 }
55 }
56 // Audio and video *may* use the same header extensions, so use a map.
57 std::unordered_map<int, std::string> seen_extensions;
58 for (const webrtc::RtpExtension& extension : extensions_a) {
59 seen_extensions[extension.id] = extension.uri;
60 }
61 for (const webrtc::RtpExtension& extension : extensions_b) {
62 if (seen_extensions.find(extension.id) != seen_extensions.end() &&
63 seen_extensions.at(extension.id) != extension.uri) {
64 oss << "Same ID used for different RTP header extensions: "
65 << extension.id;
66 LOG_AND_RETURN_ERROR(RTCErrorType::INVALID_PARAMETER, oss.str());
67 }
68 }
69 std::set<uint32_t> seen_ssrcs;
70 for (const cricket::StreamParams& stream : streams_a) {
71 seen_ssrcs.insert(stream.ssrcs.begin(), stream.ssrcs.end());
72 }
73 for (const cricket::StreamParams& stream : streams_b) {
74 for (uint32_t ssrc : stream.ssrcs) {
75 if (!seen_ssrcs.insert(ssrc).second) {
76 oss << "Same SSRC used for audio and video senders: " << ssrc;
77 LOG_AND_RETURN_ERROR(RTCErrorType::INVALID_PARAMETER, oss.str());
78 }
79 }
80 }
81 return RTCError::OK();
82}
83
84BEGIN_OWNED_PROXY_MAP(RtpTransportController)
85PROXY_SIGNALING_THREAD_DESTRUCTOR()
86PROXY_CONSTMETHOD0(std::vector<RtpTransportInterface*>, GetTransports)
87protected:
88RtpTransportControllerAdapter* GetInternal() override {
89 return internal();
90}
91END_PROXY_MAP()
92
93// static
94std::unique_ptr<RtpTransportControllerInterface>
95RtpTransportControllerAdapter::CreateProxied(
96 const cricket::MediaConfig& config,
97 cricket::ChannelManager* channel_manager,
98 webrtc::RtcEventLog* event_log,
99 rtc::Thread* signaling_thread,
Zhi Huange830e682018-03-30 10:48:35 -0700100 rtc::Thread* worker_thread,
101 rtc::Thread* network_thread) {
deadbeefe814a0d2017-02-25 18:15:09 -0800102 std::unique_ptr<RtpTransportControllerAdapter> wrapped(
103 new RtpTransportControllerAdapter(config, channel_manager, event_log,
Zhi Huange830e682018-03-30 10:48:35 -0700104 signaling_thread, worker_thread,
105 network_thread));
deadbeefe814a0d2017-02-25 18:15:09 -0800106 return RtpTransportControllerProxyWithInternal<
107 RtpTransportControllerAdapter>::Create(signaling_thread, worker_thread,
108 std::move(wrapped));
109}
110
111RtpTransportControllerAdapter::~RtpTransportControllerAdapter() {
112 RTC_DCHECK_RUN_ON(signaling_thread_);
113 if (!transport_proxies_.empty()) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100114 RTC_LOG(LS_ERROR)
deadbeefe814a0d2017-02-25 18:15:09 -0800115 << "Destroying RtpTransportControllerAdapter while RtpTransports "
116 "are still using it; this is unsafe.";
117 }
118 if (voice_channel_) {
119 // This would mean audio RTP senders/receivers that are using us haven't
120 // been destroyed. This isn't safe (see error log above).
121 DestroyVoiceChannel();
122 }
123 if (voice_channel_) {
124 // This would mean video RTP senders/receivers that are using us haven't
125 // been destroyed. This isn't safe (see error log above).
126 DestroyVideoChannel();
127 }
nisseeaabdf62017-05-05 02:23:02 -0700128 // Call must be destroyed on the worker thread.
129 worker_thread_->Invoke<void>(
Yves Gerey665174f2018-06-19 15:03:05 +0200130 RTC_FROM_HERE, rtc::Bind(&RtpTransportControllerAdapter::Close_w, this));
deadbeefe814a0d2017-02-25 18:15:09 -0800131}
132
133RTCErrorOr<std::unique_ptr<RtpTransportInterface>>
134RtpTransportControllerAdapter::CreateProxiedRtpTransport(
sprangdb2a9fc2017-08-09 06:42:32 -0700135 const RtpTransportParameters& parameters,
deadbeefe814a0d2017-02-25 18:15:09 -0800136 PacketTransportInterface* rtp,
137 PacketTransportInterface* rtcp) {
sprangdb2a9fc2017-08-09 06:42:32 -0700138 if (!transport_proxies_.empty() && (parameters.keepalive != keepalive_)) {
139 LOG_AND_RETURN_ERROR(RTCErrorType::INVALID_MODIFICATION,
140 "Cannot create RtpTransport with different keep-alive "
141 "from the RtpTransports already associated with this "
142 "transport controller.");
143 }
144 auto result = RtpTransportAdapter::CreateProxied(parameters, rtp, rtcp, this);
deadbeefe814a0d2017-02-25 18:15:09 -0800145 if (result.ok()) {
146 transport_proxies_.push_back(result.value().get());
147 transport_proxies_.back()->GetInternal()->SignalDestroyed.connect(
148 this, &RtpTransportControllerAdapter::OnRtpTransportDestroyed);
149 }
150 return result;
151}
152
zhihuangd3501ad2017-03-03 14:39:06 -0800153RTCErrorOr<std::unique_ptr<SrtpTransportInterface>>
154RtpTransportControllerAdapter::CreateProxiedSrtpTransport(
sprangdb2a9fc2017-08-09 06:42:32 -0700155 const RtpTransportParameters& parameters,
zhihuangd3501ad2017-03-03 14:39:06 -0800156 PacketTransportInterface* rtp,
157 PacketTransportInterface* rtcp) {
158 auto result =
sprangdb2a9fc2017-08-09 06:42:32 -0700159 RtpTransportAdapter::CreateSrtpProxied(parameters, rtp, rtcp, this);
zhihuangd3501ad2017-03-03 14:39:06 -0800160 if (result.ok()) {
161 transport_proxies_.push_back(result.value().get());
162 transport_proxies_.back()->GetInternal()->SignalDestroyed.connect(
163 this, &RtpTransportControllerAdapter::OnRtpTransportDestroyed);
164 }
165 return result;
166}
167
deadbeefe814a0d2017-02-25 18:15:09 -0800168RTCErrorOr<std::unique_ptr<OrtcRtpSenderInterface>>
169RtpTransportControllerAdapter::CreateProxiedRtpSender(
170 cricket::MediaType kind,
171 RtpTransportInterface* transport_proxy) {
172 RTC_DCHECK(transport_proxy);
173 RTC_DCHECK(std::find(transport_proxies_.begin(), transport_proxies_.end(),
174 transport_proxy) != transport_proxies_.end());
175 std::unique_ptr<OrtcRtpSenderAdapter> new_sender(
176 new OrtcRtpSenderAdapter(kind, transport_proxy, this));
177 RTCError err;
178 switch (kind) {
179 case cricket::MEDIA_TYPE_AUDIO:
180 err = AttachAudioSender(new_sender.get(), transport_proxy->GetInternal());
181 break;
182 case cricket::MEDIA_TYPE_VIDEO:
183 err = AttachVideoSender(new_sender.get(), transport_proxy->GetInternal());
184 break;
185 case cricket::MEDIA_TYPE_DATA:
186 RTC_NOTREACHED();
187 }
188 if (!err.ok()) {
189 return std::move(err);
190 }
191
192 return OrtcRtpSenderAdapter::CreateProxy(std::move(new_sender));
193}
194
195RTCErrorOr<std::unique_ptr<OrtcRtpReceiverInterface>>
196RtpTransportControllerAdapter::CreateProxiedRtpReceiver(
197 cricket::MediaType kind,
198 RtpTransportInterface* transport_proxy) {
199 RTC_DCHECK(transport_proxy);
200 RTC_DCHECK(std::find(transport_proxies_.begin(), transport_proxies_.end(),
201 transport_proxy) != transport_proxies_.end());
202 std::unique_ptr<OrtcRtpReceiverAdapter> new_receiver(
203 new OrtcRtpReceiverAdapter(kind, transport_proxy, this));
204 RTCError err;
205 switch (kind) {
206 case cricket::MEDIA_TYPE_AUDIO:
207 err = AttachAudioReceiver(new_receiver.get(),
208 transport_proxy->GetInternal());
209 break;
210 case cricket::MEDIA_TYPE_VIDEO:
211 err = AttachVideoReceiver(new_receiver.get(),
212 transport_proxy->GetInternal());
213 break;
214 case cricket::MEDIA_TYPE_DATA:
215 RTC_NOTREACHED();
216 }
217 if (!err.ok()) {
218 return std::move(err);
219 }
220
221 return OrtcRtpReceiverAdapter::CreateProxy(std::move(new_receiver));
222}
223
224std::vector<RtpTransportInterface*>
225RtpTransportControllerAdapter::GetTransports() const {
226 RTC_DCHECK_RUN_ON(signaling_thread_);
227 return transport_proxies_;
228}
229
sprangdb2a9fc2017-08-09 06:42:32 -0700230RTCError RtpTransportControllerAdapter::SetRtpTransportParameters(
231 const RtpTransportParameters& parameters,
deadbeefe814a0d2017-02-25 18:15:09 -0800232 RtpTransportInterface* inner_transport) {
sprangdb2a9fc2017-08-09 06:42:32 -0700233 if ((video_channel_ != nullptr || voice_channel_ != nullptr) &&
234 (parameters.keepalive != keepalive_)) {
235 LOG_AND_RETURN_ERROR(RTCErrorType::INVALID_MODIFICATION,
236 "Cannot change keep-alive settings after creating "
237 "media streams or additional transports for the same "
238 "transport controller.");
239 }
240 // Call must be configured on the worker thread.
241 worker_thread_->Invoke<void>(
242 RTC_FROM_HERE,
243 rtc::Bind(&RtpTransportControllerAdapter::SetRtpTransportParameters_w,
244 this, parameters));
245
deadbeefe814a0d2017-02-25 18:15:09 -0800246 do {
247 if (inner_transport == inner_audio_transport_) {
sprangdb2a9fc2017-08-09 06:42:32 -0700248 CopyRtcpParametersToDescriptions(parameters.rtcp,
249 &local_audio_description_,
deadbeefe814a0d2017-02-25 18:15:09 -0800250 &remote_audio_description_);
251 if (!voice_channel_->SetLocalContent(&local_audio_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800252 SdpType::kOffer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800253 break;
254 }
255 if (!voice_channel_->SetRemoteContent(&remote_audio_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800256 SdpType::kAnswer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800257 break;
258 }
259 } else if (inner_transport == inner_video_transport_) {
sprangdb2a9fc2017-08-09 06:42:32 -0700260 CopyRtcpParametersToDescriptions(parameters.rtcp,
261 &local_video_description_,
deadbeefe814a0d2017-02-25 18:15:09 -0800262 &remote_video_description_);
263 if (!video_channel_->SetLocalContent(&local_video_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800264 SdpType::kOffer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800265 break;
266 }
267 if (!video_channel_->SetRemoteContent(&remote_video_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800268 SdpType::kAnswer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800269 break;
270 }
271 }
272 return RTCError::OK();
273 } while (false);
274 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
275 "Failed to apply new RTCP parameters.");
276}
277
sprangdb2a9fc2017-08-09 06:42:32 -0700278void RtpTransportControllerAdapter::SetRtpTransportParameters_w(
279 const RtpTransportParameters& parameters) {
280 call_send_rtp_transport_controller_->SetKeepAliveConfig(parameters.keepalive);
281}
282
deadbeefe814a0d2017-02-25 18:15:09 -0800283RTCError RtpTransportControllerAdapter::ValidateAndApplyAudioSenderParameters(
284 const RtpParameters& parameters,
285 uint32_t* primary_ssrc) {
286 RTC_DCHECK(voice_channel_);
287 RTC_DCHECK(have_audio_sender_);
288
289 auto codecs_result = ToCricketCodecs<cricket::AudioCodec>(parameters.codecs);
290 if (!codecs_result.ok()) {
291 return codecs_result.MoveError();
292 }
293
294 auto extensions_result =
295 ToCricketRtpHeaderExtensions(parameters.header_extensions);
296 if (!extensions_result.ok()) {
297 return extensions_result.MoveError();
298 }
299
300 auto stream_params_result = MakeSendStreamParamsVec(
sprangdb2a9fc2017-08-09 06:42:32 -0700301 parameters.encodings, inner_audio_transport_->GetParameters().rtcp.cname,
deadbeefe814a0d2017-02-25 18:15:09 -0800302 local_audio_description_);
303 if (!stream_params_result.ok()) {
304 return stream_params_result.MoveError();
305 }
306
307 // Check that audio/video sender aren't using the same IDs to refer to
308 // different things, if they share the same transport.
309 if (inner_audio_transport_ == inner_video_transport_) {
310 RTCError err = CheckForIdConflicts(
311 codecs_result.value(), extensions_result.value(),
312 stream_params_result.value(), remote_video_description_.codecs(),
313 remote_video_description_.rtp_header_extensions(),
314 local_video_description_.streams());
315 if (!err.ok()) {
316 return err;
317 }
318 }
319
Steve Anton4e70a722017-11-28 14:57:10 -0800320 bool local_send = false;
deadbeefe814a0d2017-02-25 18:15:09 -0800321 int bandwidth = cricket::kAutoBandwidth;
322 if (parameters.encodings.size() == 1u) {
323 if (parameters.encodings[0].max_bitrate_bps) {
324 bandwidth = *parameters.encodings[0].max_bitrate_bps;
325 }
Steve Anton1d03a752017-11-27 14:30:09 -0800326 local_send = parameters.encodings[0].active;
deadbeefe814a0d2017-02-25 18:15:09 -0800327 }
Steve Anton4e70a722017-11-28 14:57:10 -0800328 const bool local_recv =
329 RtpTransceiverDirectionHasRecv(local_audio_description_.direction());
330 const auto local_direction =
331 RtpTransceiverDirectionFromSendRecv(local_send, local_recv);
deadbeefe814a0d2017-02-25 18:15:09 -0800332 if (primary_ssrc && !stream_params_result.value().empty()) {
333 *primary_ssrc = stream_params_result.value()[0].first_ssrc();
334 }
335
336 // Validation is done, so we can attempt applying the descriptions. Sent
337 // codecs and header extensions go in remote description, streams go in
338 // local.
339 //
340 // If there are no codecs or encodings, just leave the previous set of
341 // codecs. The media engine doesn't like an empty set of codecs.
342 if (local_audio_description_.streams().empty() &&
343 remote_audio_description_.codecs().empty()) {
344 } else {
345 remote_audio_description_.set_codecs(codecs_result.MoveValue());
346 }
347 remote_audio_description_.set_rtp_header_extensions(
348 extensions_result.MoveValue());
349 remote_audio_description_.set_bandwidth(bandwidth);
350 local_audio_description_.mutable_streams() = stream_params_result.MoveValue();
351 // Direction set based on encoding "active" flag.
Steve Anton4e70a722017-11-28 14:57:10 -0800352 local_audio_description_.set_direction(local_direction);
deadbeefe814a0d2017-02-25 18:15:09 -0800353 remote_audio_description_.set_direction(
Steve Anton4e70a722017-11-28 14:57:10 -0800354 RtpTransceiverDirectionReversed(local_direction));
deadbeefe814a0d2017-02-25 18:15:09 -0800355
356 // Set remote content first, to ensure the stream is created with the correct
357 // codec.
358 if (!voice_channel_->SetRemoteContent(&remote_audio_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800359 SdpType::kOffer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800360 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
361 "Failed to apply remote parameters to media channel.");
362 }
363 if (!voice_channel_->SetLocalContent(&local_audio_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800364 SdpType::kAnswer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800365 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
366 "Failed to apply local parameters to media channel.");
367 }
368 return RTCError::OK();
369}
370
371RTCError RtpTransportControllerAdapter::ValidateAndApplyVideoSenderParameters(
372 const RtpParameters& parameters,
373 uint32_t* primary_ssrc) {
374 RTC_DCHECK(video_channel_);
375 RTC_DCHECK(have_video_sender_);
376
377 auto codecs_result = ToCricketCodecs<cricket::VideoCodec>(parameters.codecs);
378 if (!codecs_result.ok()) {
379 return codecs_result.MoveError();
380 }
381
382 auto extensions_result =
383 ToCricketRtpHeaderExtensions(parameters.header_extensions);
384 if (!extensions_result.ok()) {
385 return extensions_result.MoveError();
386 }
387
388 auto stream_params_result = MakeSendStreamParamsVec(
sprangdb2a9fc2017-08-09 06:42:32 -0700389 parameters.encodings, inner_video_transport_->GetParameters().rtcp.cname,
deadbeefe814a0d2017-02-25 18:15:09 -0800390 local_video_description_);
391 if (!stream_params_result.ok()) {
392 return stream_params_result.MoveError();
393 }
394
395 // Check that audio/video sender aren't using the same IDs to refer to
396 // different things, if they share the same transport.
397 if (inner_audio_transport_ == inner_video_transport_) {
398 RTCError err = CheckForIdConflicts(
399 codecs_result.value(), extensions_result.value(),
400 stream_params_result.value(), remote_audio_description_.codecs(),
401 remote_audio_description_.rtp_header_extensions(),
402 local_audio_description_.streams());
403 if (!err.ok()) {
404 return err;
405 }
406 }
407
Steve Anton4e70a722017-11-28 14:57:10 -0800408 bool local_send = false;
deadbeefe814a0d2017-02-25 18:15:09 -0800409 int bandwidth = cricket::kAutoBandwidth;
410 if (parameters.encodings.size() == 1u) {
411 if (parameters.encodings[0].max_bitrate_bps) {
412 bandwidth = *parameters.encodings[0].max_bitrate_bps;
413 }
Steve Anton1d03a752017-11-27 14:30:09 -0800414 local_send = parameters.encodings[0].active;
deadbeefe814a0d2017-02-25 18:15:09 -0800415 }
Steve Anton4e70a722017-11-28 14:57:10 -0800416 const bool local_recv =
417 RtpTransceiverDirectionHasRecv(local_audio_description_.direction());
418 const auto local_direction =
419 RtpTransceiverDirectionFromSendRecv(local_send, local_recv);
deadbeefe814a0d2017-02-25 18:15:09 -0800420 if (primary_ssrc && !stream_params_result.value().empty()) {
421 *primary_ssrc = stream_params_result.value()[0].first_ssrc();
422 }
423
424 // Validation is done, so we can attempt applying the descriptions. Sent
425 // codecs and header extensions go in remote description, streams go in
426 // local.
427 //
428 // If there are no codecs or encodings, just leave the previous set of
429 // codecs. The media engine doesn't like an empty set of codecs.
430 if (local_video_description_.streams().empty() &&
431 remote_video_description_.codecs().empty()) {
432 } else {
433 remote_video_description_.set_codecs(codecs_result.MoveValue());
434 }
435 remote_video_description_.set_rtp_header_extensions(
436 extensions_result.MoveValue());
437 remote_video_description_.set_bandwidth(bandwidth);
438 local_video_description_.mutable_streams() = stream_params_result.MoveValue();
439 // Direction set based on encoding "active" flag.
Steve Anton4e70a722017-11-28 14:57:10 -0800440 local_video_description_.set_direction(local_direction);
deadbeefe814a0d2017-02-25 18:15:09 -0800441 remote_video_description_.set_direction(
Steve Anton4e70a722017-11-28 14:57:10 -0800442 RtpTransceiverDirectionReversed(local_direction));
deadbeefe814a0d2017-02-25 18:15:09 -0800443
444 // Set remote content first, to ensure the stream is created with the correct
445 // codec.
446 if (!video_channel_->SetRemoteContent(&remote_video_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800447 SdpType::kOffer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800448 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
449 "Failed to apply remote parameters to media channel.");
450 }
451 if (!video_channel_->SetLocalContent(&local_video_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800452 SdpType::kAnswer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800453 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
454 "Failed to apply local parameters to media channel.");
455 }
456 return RTCError::OK();
457}
458
459RTCError RtpTransportControllerAdapter::ValidateAndApplyAudioReceiverParameters(
460 const RtpParameters& parameters) {
461 RTC_DCHECK(voice_channel_);
462 RTC_DCHECK(have_audio_receiver_);
463
464 auto codecs_result = ToCricketCodecs<cricket::AudioCodec>(parameters.codecs);
465 if (!codecs_result.ok()) {
466 return codecs_result.MoveError();
467 }
468
469 auto extensions_result =
470 ToCricketRtpHeaderExtensions(parameters.header_extensions);
471 if (!extensions_result.ok()) {
472 return extensions_result.MoveError();
473 }
474
deadbeefe814a0d2017-02-25 18:15:09 -0800475 auto stream_params_result = ToCricketStreamParamsVec(parameters.encodings);
476 if (!stream_params_result.ok()) {
477 return stream_params_result.MoveError();
478 }
479
480 // Check that audio/video receive aren't using the same IDs to refer to
481 // different things, if they share the same transport.
482 if (inner_audio_transport_ == inner_video_transport_) {
483 RTCError err = CheckForIdConflicts(
484 codecs_result.value(), extensions_result.value(),
485 stream_params_result.value(), local_video_description_.codecs(),
486 local_video_description_.rtp_header_extensions(),
487 remote_video_description_.streams());
488 if (!err.ok()) {
489 return err;
490 }
491 }
492
Steve Anton4e70a722017-11-28 14:57:10 -0800493 const bool local_send =
494 RtpTransceiverDirectionHasSend(local_audio_description_.direction());
495 const bool local_recv =
deadbeefe814a0d2017-02-25 18:15:09 -0800496 !parameters.encodings.empty() && parameters.encodings[0].active;
Steve Anton4e70a722017-11-28 14:57:10 -0800497 const auto local_direction =
Steve Anton1d03a752017-11-27 14:30:09 -0800498 RtpTransceiverDirectionFromSendRecv(local_send, local_recv);
deadbeefe814a0d2017-02-25 18:15:09 -0800499
500 // Validation is done, so we can attempt applying the descriptions. Received
501 // codecs and header extensions go in local description, streams go in
502 // remote.
503 //
504 // If there are no codecs or encodings, just leave the previous set of
505 // codecs. The media engine doesn't like an empty set of codecs.
506 if (remote_audio_description_.streams().empty() &&
507 local_audio_description_.codecs().empty()) {
508 } else {
509 local_audio_description_.set_codecs(codecs_result.MoveValue());
510 }
511 local_audio_description_.set_rtp_header_extensions(
512 extensions_result.MoveValue());
513 remote_audio_description_.mutable_streams() =
514 stream_params_result.MoveValue();
515 // Direction set based on encoding "active" flag.
Steve Anton4e70a722017-11-28 14:57:10 -0800516 local_audio_description_.set_direction(local_direction);
deadbeefe814a0d2017-02-25 18:15:09 -0800517 remote_audio_description_.set_direction(
Steve Anton4e70a722017-11-28 14:57:10 -0800518 RtpTransceiverDirectionReversed(local_direction));
deadbeefe814a0d2017-02-25 18:15:09 -0800519
520 if (!voice_channel_->SetLocalContent(&local_audio_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800521 SdpType::kOffer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800522 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
523 "Failed to apply local parameters to media channel.");
524 }
525 if (!voice_channel_->SetRemoteContent(&remote_audio_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800526 SdpType::kAnswer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800527 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
528 "Failed to apply remote parameters to media channel.");
529 }
530 return RTCError::OK();
531}
532
533RTCError RtpTransportControllerAdapter::ValidateAndApplyVideoReceiverParameters(
534 const RtpParameters& parameters) {
535 RTC_DCHECK(video_channel_);
536 RTC_DCHECK(have_video_receiver_);
537
538 auto codecs_result = ToCricketCodecs<cricket::VideoCodec>(parameters.codecs);
539 if (!codecs_result.ok()) {
540 return codecs_result.MoveError();
541 }
542
543 auto extensions_result =
544 ToCricketRtpHeaderExtensions(parameters.header_extensions);
545 if (!extensions_result.ok()) {
546 return extensions_result.MoveError();
547 }
548
deadbeefe814a0d2017-02-25 18:15:09 -0800549 int bandwidth = cricket::kAutoBandwidth;
550 auto stream_params_result = ToCricketStreamParamsVec(parameters.encodings);
551 if (!stream_params_result.ok()) {
552 return stream_params_result.MoveError();
553 }
554
555 // Check that audio/video receiver aren't using the same IDs to refer to
556 // different things, if they share the same transport.
557 if (inner_audio_transport_ == inner_video_transport_) {
558 RTCError err = CheckForIdConflicts(
559 codecs_result.value(), extensions_result.value(),
560 stream_params_result.value(), local_audio_description_.codecs(),
561 local_audio_description_.rtp_header_extensions(),
562 remote_audio_description_.streams());
563 if (!err.ok()) {
564 return err;
565 }
566 }
567
Steve Anton4e70a722017-11-28 14:57:10 -0800568 const bool local_send =
569 RtpTransceiverDirectionHasSend(local_video_description_.direction());
570 const bool local_recv =
deadbeefe814a0d2017-02-25 18:15:09 -0800571 !parameters.encodings.empty() && parameters.encodings[0].active;
Steve Anton4e70a722017-11-28 14:57:10 -0800572 const auto local_direction =
Steve Anton1d03a752017-11-27 14:30:09 -0800573 RtpTransceiverDirectionFromSendRecv(local_send, local_recv);
deadbeefe814a0d2017-02-25 18:15:09 -0800574
575 // Validation is done, so we can attempt applying the descriptions. Received
576 // codecs and header extensions go in local description, streams go in
577 // remote.
578 //
579 // If there are no codecs or encodings, just leave the previous set of
580 // codecs. The media engine doesn't like an empty set of codecs.
581 if (remote_video_description_.streams().empty() &&
582 local_video_description_.codecs().empty()) {
583 } else {
584 local_video_description_.set_codecs(codecs_result.MoveValue());
585 }
586 local_video_description_.set_rtp_header_extensions(
587 extensions_result.MoveValue());
588 local_video_description_.set_bandwidth(bandwidth);
589 remote_video_description_.mutable_streams() =
590 stream_params_result.MoveValue();
591 // Direction set based on encoding "active" flag.
Steve Anton4e70a722017-11-28 14:57:10 -0800592 local_video_description_.set_direction(local_direction);
deadbeefe814a0d2017-02-25 18:15:09 -0800593 remote_video_description_.set_direction(
Steve Anton4e70a722017-11-28 14:57:10 -0800594 RtpTransceiverDirectionReversed(local_direction));
deadbeefe814a0d2017-02-25 18:15:09 -0800595
596 if (!video_channel_->SetLocalContent(&local_video_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800597 SdpType::kOffer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800598 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
599 "Failed to apply local parameters to media channel.");
600 }
601 if (!video_channel_->SetRemoteContent(&remote_video_description_,
Steve Anton3828c062017-12-06 10:34:51 -0800602 SdpType::kAnswer, nullptr)) {
deadbeefe814a0d2017-02-25 18:15:09 -0800603 LOG_AND_RETURN_ERROR(RTCErrorType::INTERNAL_ERROR,
604 "Failed to apply remote parameters to media channel.");
605 }
606 return RTCError::OK();
607}
608
609RtpTransportControllerAdapter::RtpTransportControllerAdapter(
610 const cricket::MediaConfig& config,
611 cricket::ChannelManager* channel_manager,
612 webrtc::RtcEventLog* event_log,
613 rtc::Thread* signaling_thread,
Zhi Huange830e682018-03-30 10:48:35 -0700614 rtc::Thread* worker_thread,
615 rtc::Thread* network_thread)
deadbeefe814a0d2017-02-25 18:15:09 -0800616 : signaling_thread_(signaling_thread),
617 worker_thread_(worker_thread),
Zhi Huange830e682018-03-30 10:48:35 -0700618 network_thread_(network_thread),
nisseeaabdf62017-05-05 02:23:02 -0700619 media_config_(config),
620 channel_manager_(channel_manager),
sprangdb2a9fc2017-08-09 06:42:32 -0700621 event_log_(event_log),
622 call_send_rtp_transport_controller_(nullptr) {
deadbeefe814a0d2017-02-25 18:15:09 -0800623 RTC_DCHECK_RUN_ON(signaling_thread_);
nisseeaabdf62017-05-05 02:23:02 -0700624 RTC_DCHECK(channel_manager_);
deadbeefe814a0d2017-02-25 18:15:09 -0800625 // Add "dummy" codecs to the descriptions, because the media engines
626 // currently reject empty lists of codecs. Note that these codecs will never
627 // actually be used, because when parameters are set, the dummy codecs will
628 // be replaced by actual codecs before any send/receive streams are created.
629 static const cricket::AudioCodec dummy_audio(0, cricket::kPcmuCodecName, 8000,
630 0, 1);
631 static const cricket::VideoCodec dummy_video(96, cricket::kVp8CodecName);
632 local_audio_description_.AddCodec(dummy_audio);
633 remote_audio_description_.AddCodec(dummy_audio);
634 local_video_description_.AddCodec(dummy_video);
635 remote_video_description_.AddCodec(dummy_video);
nisseeaabdf62017-05-05 02:23:02 -0700636
637 worker_thread_->Invoke<void>(
Yves Gerey665174f2018-06-19 15:03:05 +0200638 RTC_FROM_HERE, rtc::Bind(&RtpTransportControllerAdapter::Init_w, this));
nisseeaabdf62017-05-05 02:23:02 -0700639}
640
641// TODO(nisse): Duplicates corresponding method in PeerConnection (used
642// to be in MediaController).
643void RtpTransportControllerAdapter::Init_w() {
644 RTC_DCHECK(worker_thread_->IsCurrent());
645 RTC_DCHECK(!call_);
646
647 const int kMinBandwidthBps = 30000;
648 const int kStartBandwidthBps = 300000;
649 const int kMaxBandwidthBps = 2000000;
650
651 webrtc::Call::Config call_config(event_log_);
652 call_config.audio_state = channel_manager_->media_engine()->GetAudioState();
653 call_config.bitrate_config.min_bitrate_bps = kMinBandwidthBps;
654 call_config.bitrate_config.start_bitrate_bps = kStartBandwidthBps;
655 call_config.bitrate_config.max_bitrate_bps = kMaxBandwidthBps;
Sebastian Jansson9a03dd82018-02-22 10:31:14 +0100656 std::unique_ptr<RtpTransportControllerSend> controller_send =
Karl Wiberg918f50c2018-07-05 11:40:33 +0200657 absl::make_unique<RtpTransportControllerSend>(
Sebastian Janssondfce03a2018-05-18 18:05:10 +0200658 Clock::GetRealTimeClock(), event_log_,
659 call_config.network_controller_factory, call_config.bitrate_config);
Sebastian Jansson9a03dd82018-02-22 10:31:14 +0100660 call_send_rtp_transport_controller_ = controller_send.get();
661 call_.reset(webrtc::Call::Create(call_config, std::move(controller_send)));
nisseeaabdf62017-05-05 02:23:02 -0700662}
663
664void RtpTransportControllerAdapter::Close_w() {
665 call_.reset();
sprangdb2a9fc2017-08-09 06:42:32 -0700666 call_send_rtp_transport_controller_ = nullptr;
deadbeefe814a0d2017-02-25 18:15:09 -0800667}
668
669RTCError RtpTransportControllerAdapter::AttachAudioSender(
670 OrtcRtpSenderAdapter* sender,
671 RtpTransportInterface* inner_transport) {
672 if (have_audio_sender_) {
673 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
674 "Using two audio RtpSenders with the same "
675 "RtpTransportControllerAdapter is not currently "
676 "supported.");
677 }
678 if (inner_audio_transport_ && inner_audio_transport_ != inner_transport) {
679 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
680 "Using different transports for the audio "
681 "RtpSender and RtpReceiver is not currently "
682 "supported.");
683 }
Zhi Huange830e682018-03-30 10:48:35 -0700684
deadbeefe814a0d2017-02-25 18:15:09 -0800685 // If setting new transport, extract its RTCP parameters and create voice
686 // channel.
687 if (!inner_audio_transport_) {
sprangdb2a9fc2017-08-09 06:42:32 -0700688 CopyRtcpParametersToDescriptions(inner_transport->GetParameters().rtcp,
deadbeefe814a0d2017-02-25 18:15:09 -0800689 &local_audio_description_,
690 &remote_audio_description_);
691 inner_audio_transport_ = inner_transport;
692 CreateVoiceChannel();
693 }
694 have_audio_sender_ = true;
695 sender->SignalDestroyed.connect(
696 this, &RtpTransportControllerAdapter::OnAudioSenderDestroyed);
697 return RTCError::OK();
698}
699
700RTCError RtpTransportControllerAdapter::AttachVideoSender(
701 OrtcRtpSenderAdapter* sender,
702 RtpTransportInterface* inner_transport) {
703 if (have_video_sender_) {
704 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
705 "Using two video RtpSenders with the same "
706 "RtpTransportControllerAdapter is not currently "
707 "supported.");
708 }
709 if (inner_video_transport_ && inner_video_transport_ != inner_transport) {
710 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
711 "Using different transports for the video "
712 "RtpSender and RtpReceiver is not currently "
713 "supported.");
714 }
Zhi Huange830e682018-03-30 10:48:35 -0700715
deadbeefe814a0d2017-02-25 18:15:09 -0800716 // If setting new transport, extract its RTCP parameters and create video
717 // channel.
718 if (!inner_video_transport_) {
sprangdb2a9fc2017-08-09 06:42:32 -0700719 CopyRtcpParametersToDescriptions(inner_transport->GetParameters().rtcp,
deadbeefe814a0d2017-02-25 18:15:09 -0800720 &local_video_description_,
721 &remote_video_description_);
722 inner_video_transport_ = inner_transport;
723 CreateVideoChannel();
724 }
725 have_video_sender_ = true;
726 sender->SignalDestroyed.connect(
727 this, &RtpTransportControllerAdapter::OnVideoSenderDestroyed);
728 return RTCError::OK();
729}
730
731RTCError RtpTransportControllerAdapter::AttachAudioReceiver(
732 OrtcRtpReceiverAdapter* receiver,
733 RtpTransportInterface* inner_transport) {
734 if (have_audio_receiver_) {
735 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
736 "Using two audio RtpReceivers with the same "
737 "RtpTransportControllerAdapter is not currently "
738 "supported.");
739 }
740 if (inner_audio_transport_ && inner_audio_transport_ != inner_transport) {
741 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
742 "Using different transports for the audio "
743 "RtpReceiver and RtpReceiver is not currently "
744 "supported.");
745 }
Zhi Huange830e682018-03-30 10:48:35 -0700746
deadbeefe814a0d2017-02-25 18:15:09 -0800747 // If setting new transport, extract its RTCP parameters and create voice
748 // channel.
749 if (!inner_audio_transport_) {
sprangdb2a9fc2017-08-09 06:42:32 -0700750 CopyRtcpParametersToDescriptions(inner_transport->GetParameters().rtcp,
deadbeefe814a0d2017-02-25 18:15:09 -0800751 &local_audio_description_,
752 &remote_audio_description_);
753 inner_audio_transport_ = inner_transport;
754 CreateVoiceChannel();
755 }
756 have_audio_receiver_ = true;
757 receiver->SignalDestroyed.connect(
758 this, &RtpTransportControllerAdapter::OnAudioReceiverDestroyed);
759 return RTCError::OK();
760}
761
762RTCError RtpTransportControllerAdapter::AttachVideoReceiver(
763 OrtcRtpReceiverAdapter* receiver,
764 RtpTransportInterface* inner_transport) {
765 if (have_video_receiver_) {
766 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
767 "Using two video RtpReceivers with the same "
768 "RtpTransportControllerAdapter is not currently "
769 "supported.");
770 }
771 if (inner_video_transport_ && inner_video_transport_ != inner_transport) {
772 LOG_AND_RETURN_ERROR(RTCErrorType::UNSUPPORTED_OPERATION,
773 "Using different transports for the video "
774 "RtpReceiver and RtpReceiver is not currently "
775 "supported.");
776 }
777 // If setting new transport, extract its RTCP parameters and create video
778 // channel.
779 if (!inner_video_transport_) {
sprangdb2a9fc2017-08-09 06:42:32 -0700780 CopyRtcpParametersToDescriptions(inner_transport->GetParameters().rtcp,
deadbeefe814a0d2017-02-25 18:15:09 -0800781 &local_video_description_,
782 &remote_video_description_);
783 inner_video_transport_ = inner_transport;
784 CreateVideoChannel();
785 }
786 have_video_receiver_ = true;
787 receiver->SignalDestroyed.connect(
788 this, &RtpTransportControllerAdapter::OnVideoReceiverDestroyed);
789 return RTCError::OK();
790}
791
792void RtpTransportControllerAdapter::OnRtpTransportDestroyed(
793 RtpTransportAdapter* transport) {
794 RTC_DCHECK_RUN_ON(signaling_thread_);
795 auto it = std::find_if(transport_proxies_.begin(), transport_proxies_.end(),
796 [transport](RtpTransportInterface* proxy) {
797 return proxy->GetInternal() == transport;
798 });
799 if (it == transport_proxies_.end()) {
800 RTC_NOTREACHED();
801 return;
802 }
803 transport_proxies_.erase(it);
804}
805
806void RtpTransportControllerAdapter::OnAudioSenderDestroyed() {
807 if (!have_audio_sender_) {
808 RTC_NOTREACHED();
809 return;
810 }
811 // Empty parameters should result in sending being stopped.
812 RTCError err =
813 ValidateAndApplyAudioSenderParameters(RtpParameters(), nullptr);
814 RTC_DCHECK(err.ok());
815 have_audio_sender_ = false;
816 if (!have_audio_receiver_) {
817 DestroyVoiceChannel();
818 }
819}
820
821void RtpTransportControllerAdapter::OnVideoSenderDestroyed() {
822 if (!have_video_sender_) {
823 RTC_NOTREACHED();
824 return;
825 }
826 // Empty parameters should result in sending being stopped.
827 RTCError err =
828 ValidateAndApplyVideoSenderParameters(RtpParameters(), nullptr);
829 RTC_DCHECK(err.ok());
830 have_video_sender_ = false;
831 if (!have_video_receiver_) {
832 DestroyVideoChannel();
833 }
834}
835
836void RtpTransportControllerAdapter::OnAudioReceiverDestroyed() {
837 if (!have_audio_receiver_) {
838 RTC_NOTREACHED();
839 return;
840 }
841 // Empty parameters should result in receiving being stopped.
842 RTCError err = ValidateAndApplyAudioReceiverParameters(RtpParameters());
843 RTC_DCHECK(err.ok());
844 have_audio_receiver_ = false;
845 if (!have_audio_sender_) {
846 DestroyVoiceChannel();
847 }
848}
849
850void RtpTransportControllerAdapter::OnVideoReceiverDestroyed() {
851 if (!have_video_receiver_) {
852 RTC_NOTREACHED();
853 return;
854 }
855 // Empty parameters should result in receiving being stopped.
856 RTCError err = ValidateAndApplyVideoReceiverParameters(RtpParameters());
857 RTC_DCHECK(err.ok());
858 have_video_receiver_ = false;
859 if (!have_video_sender_) {
860 DestroyVideoChannel();
861 }
862}
863
864void RtpTransportControllerAdapter::CreateVoiceChannel() {
nisseeaabdf62017-05-05 02:23:02 -0700865 voice_channel_ = channel_manager_->CreateVoiceChannel(
Zhi Huange830e682018-03-30 10:48:35 -0700866 call_.get(), media_config_, inner_audio_transport_->GetInternal(),
867 signaling_thread_, "audio", false, rtc::CryptoOptions(),
868 cricket::AudioOptions());
deadbeefe814a0d2017-02-25 18:15:09 -0800869 RTC_DCHECK(voice_channel_);
870 voice_channel_->Enable(true);
871}
872
873void RtpTransportControllerAdapter::CreateVideoChannel() {
nisseeaabdf62017-05-05 02:23:02 -0700874 video_channel_ = channel_manager_->CreateVideoChannel(
Zhi Huange830e682018-03-30 10:48:35 -0700875 call_.get(), media_config_, inner_video_transport_->GetInternal(),
876 signaling_thread_, "video", false, rtc::CryptoOptions(),
877 cricket::VideoOptions());
deadbeefe814a0d2017-02-25 18:15:09 -0800878 RTC_DCHECK(video_channel_);
879 video_channel_->Enable(true);
880}
881
882void RtpTransportControllerAdapter::DestroyVoiceChannel() {
883 RTC_DCHECK(voice_channel_);
nisseeaabdf62017-05-05 02:23:02 -0700884 channel_manager_->DestroyVoiceChannel(voice_channel_);
deadbeefe814a0d2017-02-25 18:15:09 -0800885 voice_channel_ = nullptr;
886 inner_audio_transport_ = nullptr;
887}
888
889void RtpTransportControllerAdapter::DestroyVideoChannel() {
890 RTC_DCHECK(video_channel_);
nisseeaabdf62017-05-05 02:23:02 -0700891 channel_manager_->DestroyVideoChannel(video_channel_);
deadbeefe814a0d2017-02-25 18:15:09 -0800892 video_channel_ = nullptr;
893 inner_video_transport_ = nullptr;
894}
895
896void RtpTransportControllerAdapter::CopyRtcpParametersToDescriptions(
897 const RtcpParameters& params,
898 cricket::MediaContentDescription* local,
899 cricket::MediaContentDescription* remote) {
900 local->set_rtcp_mux(params.mux);
901 remote->set_rtcp_mux(params.mux);
902 local->set_rtcp_reduced_size(params.reduced_size);
903 remote->set_rtcp_reduced_size(params.reduced_size);
904 for (cricket::StreamParams& stream_params : local->mutable_streams()) {
905 stream_params.cname = params.cname;
906 }
907}
908
909uint32_t RtpTransportControllerAdapter::GenerateUnusedSsrc(
910 std::set<uint32_t>* new_ssrcs) const {
911 uint32_t ssrc;
912 do {
913 ssrc = rtc::CreateRandomNonZeroId();
914 } while (
915 cricket::GetStreamBySsrc(local_audio_description_.streams(), ssrc) ||
916 cricket::GetStreamBySsrc(remote_audio_description_.streams(), ssrc) ||
917 cricket::GetStreamBySsrc(local_video_description_.streams(), ssrc) ||
918 cricket::GetStreamBySsrc(remote_video_description_.streams(), ssrc) ||
919 !new_ssrcs->insert(ssrc).second);
920 return ssrc;
921}
922
923RTCErrorOr<cricket::StreamParamsVec>
924RtpTransportControllerAdapter::MakeSendStreamParamsVec(
925 std::vector<RtpEncodingParameters> encodings,
926 const std::string& cname,
927 const cricket::MediaContentDescription& description) const {
928 if (encodings.size() > 1u) {
929 LOG_AND_RETURN_ERROR(webrtc::RTCErrorType::UNSUPPORTED_PARAMETER,
930 "ORTC API implementation doesn't currently "
931 "support simulcast or layered encodings.");
932 } else if (encodings.empty()) {
933 return cricket::StreamParamsVec();
934 }
935 RtpEncodingParameters& encoding = encodings[0];
936 std::set<uint32_t> new_ssrcs;
937 if (encoding.ssrc) {
938 new_ssrcs.insert(*encoding.ssrc);
939 }
940 if (encoding.rtx && encoding.rtx->ssrc) {
941 new_ssrcs.insert(*encoding.rtx->ssrc);
942 }
943 // May need to fill missing SSRCs with generated ones.
944 if (!encoding.ssrc) {
945 if (!description.streams().empty()) {
946 encoding.ssrc.emplace(description.streams()[0].first_ssrc());
947 } else {
948 encoding.ssrc.emplace(GenerateUnusedSsrc(&new_ssrcs));
949 }
950 }
951 if (encoding.rtx && !encoding.rtx->ssrc) {
952 uint32_t existing_rtx_ssrc;
953 if (!description.streams().empty() &&
954 description.streams()[0].GetFidSsrc(
955 description.streams()[0].first_ssrc(), &existing_rtx_ssrc)) {
956 encoding.rtx->ssrc.emplace(existing_rtx_ssrc);
957 } else {
958 encoding.rtx->ssrc.emplace(GenerateUnusedSsrc(&new_ssrcs));
959 }
960 }
961
962 auto result = ToCricketStreamParamsVec(encodings);
963 if (!result.ok()) {
964 return result.MoveError();
965 }
966 // If conversion was successful, there should be one StreamParams.
967 RTC_DCHECK_EQ(1u, result.value().size());
968 result.value()[0].cname = cname;
969 return result;
970}
971
deadbeefe814a0d2017-02-25 18:15:09 -0800972} // namespace webrtc