blob: 59f0869431ae59aef1e26b3dbc555b0c897e7e14 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
kjellander65c7f672016-02-12 00:05:01 -08002 * Copyright 2004 The WebRTC project authors. All Rights Reserved.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00003 *
kjellander65c7f672016-02-12 00:05:01 -08004 * 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.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00009 */
10
jbauch5869f502017-06-29 12:31:36 -070011#include <algorithm>
12#include <iterator>
kwiberg0eb15ed2015-12-17 03:04:15 -080013#include <utility>
14
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010015#include "webrtc/pc/channel.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000016
kjellandera69d9732016-08-31 07:33:05 -070017#include "webrtc/api/call/audio_sink.h"
kjellanderf4752772016-03-02 05:42:30 -080018#include "webrtc/media/base/mediaconstants.h"
kjellandera96e2d72016-02-04 23:52:28 -080019#include "webrtc/media/base/rtputils.h"
Edward Lemurc20978e2017-07-06 19:44:34 +020020#include "webrtc/rtc_base/bind.h"
21#include "webrtc/rtc_base/byteorder.h"
22#include "webrtc/rtc_base/checks.h"
23#include "webrtc/rtc_base/copyonwritebuffer.h"
24#include "webrtc/rtc_base/dscp.h"
25#include "webrtc/rtc_base/logging.h"
26#include "webrtc/rtc_base/networkroute.h"
zsteine8ab5432017-07-12 11:48:11 -070027#include "webrtc/rtc_base/ptr_util.h"
Edward Lemurc20978e2017-07-06 19:44:34 +020028#include "webrtc/rtc_base/trace_event.h"
zhihuang38ede132017-06-15 12:52:32 -070029// Adding 'nogncheck' to disable the gn include headers check to support modular
30// WebRTC build targets.
31#include "webrtc/media/engine/webrtcvoiceengine.h" // nogncheck
deadbeef5bd5ca32017-02-10 11:31:50 -080032#include "webrtc/p2p/base/packettransportinternal.h"
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010033#include "webrtc/pc/channelmanager.h"
zstein398c3fd2017-07-19 13:38:02 -070034#include "webrtc/pc/rtptransport.h"
35#include "webrtc/pc/srtptransport.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000036
37namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000038using rtc::Bind;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000039
deadbeef2d110be2016-01-13 12:00:26 -080040namespace {
kwiberg31022942016-03-11 14:18:21 -080041// See comment below for why we need to use a pointer to a unique_ptr.
deadbeef2d110be2016-01-13 12:00:26 -080042bool SetRawAudioSink_w(VoiceMediaChannel* channel,
43 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -080044 std::unique_ptr<webrtc::AudioSinkInterface>* sink) {
45 channel->SetRawAudioSink(ssrc, std::move(*sink));
deadbeef2d110be2016-01-13 12:00:26 -080046 return true;
47}
Danil Chapovalov33b01f22016-05-11 19:55:27 +020048
49struct SendPacketMessageData : public rtc::MessageData {
50 rtc::CopyOnWriteBuffer packet;
51 rtc::PacketOptions options;
52};
53
deadbeef2d110be2016-01-13 12:00:26 -080054} // namespace
55
henrike@webrtc.org28e20752013-07-10 00:45:36 +000056enum {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000057 MSG_EARLYMEDIATIMEOUT = 1,
Danil Chapovalov33b01f22016-05-11 19:55:27 +020058 MSG_SEND_RTP_PACKET,
59 MSG_SEND_RTCP_PACKET,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000060 MSG_CHANNEL_ERROR,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000061 MSG_READYTOSENDDATA,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000062 MSG_DATARECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000063 MSG_FIRSTPACKETRECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000064};
65
66// Value specified in RFC 5764.
67static const char kDtlsSrtpExporterLabel[] = "EXTRACTOR-dtls_srtp";
68
69static const int kAgcMinus10db = -10;
70
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000071static void SafeSetError(const std::string& message, std::string* error_desc) {
72 if (error_desc) {
73 *error_desc = message;
74 }
75}
76
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000077struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020078 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000079 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020080 : ssrc(in_ssrc), error(in_error) {}
81 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000082 VoiceMediaChannel::Error error;
83};
84
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000085struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020086 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000087 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020088 : ssrc(in_ssrc), error(in_error) {}
89 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000090 VideoMediaChannel::Error error;
91};
92
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000093struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020094 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000095 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020096 : ssrc(in_ssrc), error(in_error) {}
97 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000098 DataMediaChannel::Error error;
99};
100
jbaucheec21bd2016-03-20 06:15:43 -0700101static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000102 // Check the packet size. We could check the header too if needed.
zstein3dcf0e92017-06-01 13:22:42 -0700103 return packet && IsValidRtpRtcpPacketSize(rtcp, packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000104}
105
106static bool IsReceiveContentDirection(MediaContentDirection direction) {
107 return direction == MD_SENDRECV || direction == MD_RECVONLY;
108}
109
110static bool IsSendContentDirection(MediaContentDirection direction) {
111 return direction == MD_SENDRECV || direction == MD_SENDONLY;
112}
113
114static const MediaContentDescription* GetContentDescription(
115 const ContentInfo* cinfo) {
116 if (cinfo == NULL)
117 return NULL;
118 return static_cast<const MediaContentDescription*>(cinfo->description);
119}
120
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700121template <class Codec>
122void RtpParametersFromMediaDescription(
123 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700124 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700125 RtpParameters<Codec>* params) {
126 // TODO(pthatcher): Remove this once we're sure no one will give us
127 // a description without codecs (currently a CA_UPDATE with just
128 // streams can).
129 if (desc->has_codecs()) {
130 params->codecs = desc->codecs();
131 }
132 // TODO(pthatcher): See if we really need
133 // rtp_header_extensions_set() and remove it if we don't.
134 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -0700135 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700136 }
deadbeef13871492015-12-09 12:37:51 -0800137 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700138}
139
nisse05103312016-03-16 02:22:50 -0700140template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700141void RtpSendParametersFromMediaDescription(
142 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700143 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -0700144 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -0700145 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700146 send_params->max_bandwidth_bps = desc->bandwidth();
147}
148
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200149BaseChannel::BaseChannel(rtc::Thread* worker_thread,
150 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800151 rtc::Thread* signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -0700152 MediaChannel* media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700153 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800154 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800155 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200156 : worker_thread_(worker_thread),
157 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800158 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000159 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700160 rtcp_mux_required_(rtcp_mux_required),
zstein398c3fd2017-07-19 13:38:02 -0700161 rtp_transport_(
162 srtp_required
163 ? rtc::WrapUnique<webrtc::RtpTransportInternal>(
164 new webrtc::SrtpTransport(rtcp_mux_required, content_name))
165 : rtc::MakeUnique<webrtc::RtpTransport>(rtcp_mux_required)),
deadbeef7af91dd2016-12-13 11:29:11 -0800166 srtp_required_(srtp_required),
michaelt79e05882016-11-08 02:50:09 -0800167 media_channel_(media_channel),
168 selected_candidate_pair_(nullptr) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700169 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
jbauchdfcab722017-03-06 00:14:10 -0800170#if defined(ENABLE_EXTERNAL_AUTH)
171 srtp_filter_.EnableExternalAuth();
172#endif
zsteine8ab5432017-07-12 11:48:11 -0700173 rtp_transport_->SignalReadyToSend.connect(
zstein56162b92017-04-24 16:54:35 -0700174 this, &BaseChannel::OnTransportReadyToSend);
zstein3dcf0e92017-06-01 13:22:42 -0700175 // TODO(zstein): RtpTransport::SignalPacketReceived will probably be replaced
176 // with a callback interface later so that the demuxer can select which
177 // channel to signal.
zsteine8ab5432017-07-12 11:48:11 -0700178 rtp_transport_->SignalPacketReceived.connect(this,
zstein398c3fd2017-07-19 13:38:02 -0700179 &BaseChannel::OnPacketReceived);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000180 LOG(LS_INFO) << "Created channel for " << content_name;
181}
182
183BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800184 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700185 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
wu@webrtc.org78187522013-10-07 23:32:02 +0000186 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000187 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200188 // Eats any outstanding messages or packets.
189 worker_thread_->Clear(&invoker_);
190 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000191 // We must destroy the media channel before the transport channel, otherwise
192 // the media channel may try to send on the dead transport channel. NULLing
193 // is not an effective strategy since the sends will come on another thread.
194 delete media_channel_;
zhihuangf5b251b2017-01-12 19:37:48 -0800195 LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200196}
197
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200198void BaseChannel::DisconnectTransportChannels_n() {
199 // Send any outstanding RTCP packets.
200 FlushRtcpMessages_n();
201
202 // Stop signals from transport channels, but keep them alive because
203 // media_channel may use them from a different thread.
zhihuangb2cdd932017-01-19 16:54:25 -0800204 if (rtp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800205 DisconnectFromDtlsTransport(rtp_dtls_transport_);
zsteine8ab5432017-07-12 11:48:11 -0700206 } else if (rtp_transport_->rtp_packet_transport()) {
207 DisconnectFromPacketTransport(rtp_transport_->rtp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200208 }
zhihuangb2cdd932017-01-19 16:54:25 -0800209 if (rtcp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800210 DisconnectFromDtlsTransport(rtcp_dtls_transport_);
zsteine8ab5432017-07-12 11:48:11 -0700211 } else if (rtp_transport_->rtcp_packet_transport()) {
212 DisconnectFromPacketTransport(rtp_transport_->rtcp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200213 }
214
zsteine8ab5432017-07-12 11:48:11 -0700215 rtp_transport_->SetRtpPacketTransport(nullptr);
216 rtp_transport_->SetRtcpPacketTransport(nullptr);
zstein3dcf0e92017-06-01 13:22:42 -0700217
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200218 // Clear pending read packets/messages.
219 network_thread_->Clear(&invoker_);
220 network_thread_->Clear(this);
221}
222
zhihuangb2cdd932017-01-19 16:54:25 -0800223bool BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800224 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800225 rtc::PacketTransportInternal* rtp_packet_transport,
226 rtc::PacketTransportInternal* rtcp_packet_transport) {
skvlad6c87a672016-05-17 17:49:52 -0700227 if (!network_thread_->Invoke<bool>(
zhihuangb2cdd932017-01-19 16:54:25 -0800228 RTC_FROM_HERE, Bind(&BaseChannel::InitNetwork_n, this,
deadbeeff5346592017-01-24 21:51:21 -0800229 rtp_dtls_transport, rtcp_dtls_transport,
230 rtp_packet_transport, rtcp_packet_transport))) {
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000231 return false;
232 }
deadbeeff5346592017-01-24 21:51:21 -0800233 // Both RTP and RTCP channels should be set, we can call SetInterface on
234 // the media channel and it can set network options.
235 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.orgde305012013-10-31 15:40:38 +0000236 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000237 return true;
238}
239
deadbeeff5346592017-01-24 21:51:21 -0800240bool BaseChannel::InitNetwork_n(
241 DtlsTransportInternal* rtp_dtls_transport,
242 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800243 rtc::PacketTransportInternal* rtp_packet_transport,
244 rtc::PacketTransportInternal* rtcp_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200245 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800246 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport, rtp_packet_transport,
247 rtcp_packet_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200248
zstein56162b92017-04-24 16:54:35 -0700249 if (rtcp_mux_required_) {
deadbeefac22f702017-01-12 21:59:29 -0800250 rtcp_mux_filter_.SetActive();
251 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200252 return true;
253}
254
wu@webrtc.org78187522013-10-07 23:32:02 +0000255void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200256 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000257 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200258 // Packets arrive on the network thread, processing packets calls virtual
259 // functions, so need to stop this process in Deinit that is called in
260 // derived classes destructor.
261 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700262 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000263}
264
zhihuangb2cdd932017-01-19 16:54:25 -0800265void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
266 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800267 network_thread_->Invoke<void>(
268 RTC_FROM_HERE,
269 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
270 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000271}
272
deadbeeff5346592017-01-24 21:51:21 -0800273void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800274 rtc::PacketTransportInternal* rtp_packet_transport,
275 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800276 network_thread_->Invoke<void>(
277 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
278 rtp_packet_transport, rtcp_packet_transport));
279}
zhihuangf5b251b2017-01-12 19:37:48 -0800280
deadbeeff5346592017-01-24 21:51:21 -0800281void BaseChannel::SetTransports_n(
282 DtlsTransportInternal* rtp_dtls_transport,
283 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800284 rtc::PacketTransportInternal* rtp_packet_transport,
285 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800286 RTC_DCHECK(network_thread_->IsCurrent());
287 // Validate some assertions about the input.
288 RTC_DCHECK(rtp_packet_transport);
289 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
290 if (rtp_dtls_transport || rtcp_dtls_transport) {
291 // DTLS/non-DTLS pointers should be to the same object.
292 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
293 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
294 // Can't go from non-DTLS to DTLS.
zsteine8ab5432017-07-12 11:48:11 -0700295 RTC_DCHECK(!rtp_transport_->rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800296 } else {
297 // Can't go from DTLS to non-DTLS.
298 RTC_DCHECK(!rtp_dtls_transport_);
299 }
300 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800301 if (rtp_dtls_transport && rtcp_dtls_transport) {
302 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
303 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800304 }
deadbeeff5346592017-01-24 21:51:21 -0800305 std::string debug_name;
306 if (rtp_dtls_transport) {
307 transport_name_ = rtp_dtls_transport->transport_name();
308 debug_name = transport_name_;
309 } else {
310 debug_name = rtp_packet_transport->debug_name();
311 }
zsteine8ab5432017-07-12 11:48:11 -0700312 if (rtp_packet_transport == rtp_transport_->rtp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -0800313 // Nothing to do if transport isn't changing.
deadbeefbad5dad2017-01-17 18:32:35 -0800314 return;
deadbeefcbecd352015-09-23 11:50:27 -0700315 }
316
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800317 // When using DTLS-SRTP, we must reset the SrtpFilter every time the transport
318 // changes and wait until the DTLS handshake is complete to set the newly
319 // negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200320 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800321 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700322 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800323 writable_ = false;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800324 srtp_filter_.ResetParams();
325 }
326
deadbeefac22f702017-01-12 21:59:29 -0800327 // If this BaseChannel doesn't require RTCP mux and we haven't fully
328 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800329 if (rtcp_packet_transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800330 LOG(LS_INFO) << "Setting RTCP Transport for " << content_name() << " on "
deadbeeff5346592017-01-24 21:51:21 -0800331 << debug_name << " transport " << rtcp_packet_transport;
332 SetTransport_n(true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000333 }
334
deadbeeff5346592017-01-24 21:51:21 -0800335 LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
336 << debug_name << " transport " << rtp_packet_transport;
337 SetTransport_n(false, rtp_dtls_transport, rtp_packet_transport);
guoweis46383312015-12-17 16:45:59 -0800338
deadbeefcbecd352015-09-23 11:50:27 -0700339 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700340 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200341 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000342}
343
deadbeeff5346592017-01-24 21:51:21 -0800344void BaseChannel::SetTransport_n(
345 bool rtcp,
346 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800347 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200348 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800349 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800350 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700351 rtc::PacketTransportInternal* old_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700352 rtcp ? rtp_transport_->rtcp_packet_transport()
353 : rtp_transport_->rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800354
deadbeeff5346592017-01-24 21:51:21 -0800355 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700356 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000357 return;
358 }
zhihuangb2cdd932017-01-19 16:54:25 -0800359
deadbeeff5346592017-01-24 21:51:21 -0800360 RTC_DCHECK(old_packet_transport != new_packet_transport);
361 if (old_dtls_transport) {
362 DisconnectFromDtlsTransport(old_dtls_transport);
363 } else if (old_packet_transport) {
364 DisconnectFromPacketTransport(old_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000365 }
366
zsteind48dbda2017-04-04 19:45:57 -0700367 if (rtcp) {
zsteine8ab5432017-07-12 11:48:11 -0700368 rtp_transport_->SetRtcpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700369 } else {
zsteine8ab5432017-07-12 11:48:11 -0700370 rtp_transport_->SetRtpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700371 }
deadbeeff5346592017-01-24 21:51:21 -0800372 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000373
deadbeeff5346592017-01-24 21:51:21 -0800374 // If there's no new transport, we're done after disconnecting from old one.
375 if (!new_packet_transport) {
376 return;
377 }
378
379 if (rtcp && new_dtls_transport) {
380 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_filter_.IsActive()))
381 << "Setting RTCP for DTLS/SRTP after SrtpFilter is active "
382 << "should never happen.";
383 }
zstein56162b92017-04-24 16:54:35 -0700384
deadbeeff5346592017-01-24 21:51:21 -0800385 if (new_dtls_transport) {
386 ConnectToDtlsTransport(new_dtls_transport);
387 } else {
388 ConnectToPacketTransport(new_packet_transport);
389 }
390 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
391 for (const auto& pair : socket_options) {
392 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800393 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000394}
395
deadbeeff5346592017-01-24 21:51:21 -0800396void BaseChannel::ConnectToDtlsTransport(DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200397 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000398
zstein56162b92017-04-24 16:54:35 -0700399 // TODO(zstein): de-dup with ConnectToPacketTransport
zhihuangb2cdd932017-01-19 16:54:25 -0800400 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
zhihuangb2cdd932017-01-19 16:54:25 -0800401 transport->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
402 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
403 transport->ice_transport()->SignalSelectedCandidatePairChanged.connect(
Honghai Zhangcc411c02016-03-29 17:27:21 -0700404 this, &BaseChannel::OnSelectedCandidatePairChanged);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000405}
406
deadbeeff5346592017-01-24 21:51:21 -0800407void BaseChannel::DisconnectFromDtlsTransport(
408 DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200409 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangb2cdd932017-01-19 16:54:25 -0800410 OnSelectedCandidatePairChanged(transport->ice_transport(), nullptr, -1,
411 false);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000412
zhihuangb2cdd932017-01-19 16:54:25 -0800413 transport->SignalWritableState.disconnect(this);
zhihuangb2cdd932017-01-19 16:54:25 -0800414 transport->SignalDtlsState.disconnect(this);
415 transport->SignalSentPacket.disconnect(this);
416 transport->ice_transport()->SignalSelectedCandidatePairChanged.disconnect(
417 this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000418}
419
deadbeeff5346592017-01-24 21:51:21 -0800420void BaseChannel::ConnectToPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800421 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800422 RTC_DCHECK_RUN_ON(network_thread_);
423 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
deadbeeff5346592017-01-24 21:51:21 -0800424 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
425}
426
427void BaseChannel::DisconnectFromPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800428 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800429 RTC_DCHECK_RUN_ON(network_thread_);
430 transport->SignalWritableState.disconnect(this);
deadbeeff5346592017-01-24 21:51:21 -0800431 transport->SignalSentPacket.disconnect(this);
432}
433
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000434bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700435 worker_thread_->Invoke<void>(
436 RTC_FROM_HERE,
437 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
438 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000439 return true;
440}
441
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000442bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700443 return InvokeOnWorker<bool>(RTC_FROM_HERE,
444 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000445}
446
Peter Boström0c4e06b2015-10-07 12:23:21 +0200447bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700448 return InvokeOnWorker<bool>(
449 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000450}
451
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000452bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700453 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700454 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000455}
456
Peter Boström0c4e06b2015-10-07 12:23:21 +0200457bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700458 return InvokeOnWorker<bool>(
459 RTC_FROM_HERE,
460 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000461}
462
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000463bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000464 ContentAction action,
465 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100466 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700467 return InvokeOnWorker<bool>(
468 RTC_FROM_HERE,
469 Bind(&BaseChannel::SetLocalContent_w, this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000470}
471
472bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000473 ContentAction action,
474 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100475 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700476 return InvokeOnWorker<bool>(
477 RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w, this, content,
478 action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000479}
480
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000481void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800482 // We pass in the BaseChannel instead of the rtp_dtls_transport_
483 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000484 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200485 // We pass in the network thread because on that thread connection monitor
486 // will call BaseChannel::GetConnectionStats which must be called on the
487 // network thread.
488 connection_monitor_.reset(
489 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000490 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000491 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000492 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000493}
494
495void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000496 if (connection_monitor_) {
497 connection_monitor_->Stop();
498 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000499 }
500}
501
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000502bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200503 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800504 if (!rtp_dtls_transport_) {
505 return false;
506 }
zhihuangb2cdd932017-01-19 16:54:25 -0800507 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800508}
509
510bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800511 // If this BaseChannel doesn't require RTCP mux and we haven't fully
512 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700513 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000514}
515
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700516bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000517 // Receive data if we are enabled and have local content,
518 return enabled() && IsReceiveContentDirection(local_content_direction_);
519}
520
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700521bool BaseChannel::IsReadyToSendMedia_w() const {
522 // Need to access some state updated on the network thread.
523 return network_thread_->Invoke<bool>(
524 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
525}
526
527bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000528 // Send outgoing data if we are enabled, have local and remote content,
529 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800530 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000531 IsSendContentDirection(local_content_direction_) &&
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700532 was_ever_writable() &&
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200533 (srtp_filter_.IsActive() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000534}
535
jbaucheec21bd2016-03-20 06:15:43 -0700536bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700537 const rtc::PacketOptions& options) {
538 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000539}
540
jbaucheec21bd2016-03-20 06:15:43 -0700541bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700542 const rtc::PacketOptions& options) {
543 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000544}
545
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000546int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000547 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200548 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700549 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200550}
551
552int BaseChannel::SetOption_n(SocketType type,
553 rtc::Socket::Option opt,
554 int value) {
555 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800556 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000557 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000558 case ST_RTP:
zsteine8ab5432017-07-12 11:48:11 -0700559 transport = rtp_transport_->rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700560 socket_options_.push_back(
561 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000562 break;
563 case ST_RTCP:
zsteine8ab5432017-07-12 11:48:11 -0700564 transport = rtp_transport_->rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700565 rtcp_socket_options_.push_back(
566 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000567 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000568 }
deadbeeff5346592017-01-24 21:51:21 -0800569 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000570}
571
deadbeef5bd5ca32017-02-10 11:31:50 -0800572void BaseChannel::OnWritableState(rtc::PacketTransportInternal* transport) {
zsteine8ab5432017-07-12 11:48:11 -0700573 RTC_DCHECK(transport == rtp_transport_->rtp_packet_transport() ||
574 transport == rtp_transport_->rtcp_packet_transport());
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200575 RTC_DCHECK(network_thread_->IsCurrent());
576 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000577}
578
zhihuangb2cdd932017-01-19 16:54:25 -0800579void BaseChannel::OnDtlsState(DtlsTransportInternal* transport,
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800580 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200581 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800582 return;
583 }
584
585 // Reset the srtp filter if it's not the CONNECTED state. For the CONNECTED
586 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
zhihuangb2cdd932017-01-19 16:54:25 -0800587 // cover other scenarios like the whole transport is writable (not just this
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800588 // TransportChannel) or when TransportChannel is attached after DTLS is
589 // negotiated.
590 if (state != DTLS_TRANSPORT_CONNECTED) {
591 srtp_filter_.ResetParams();
592 }
593}
594
Honghai Zhangcc411c02016-03-29 17:27:21 -0700595void BaseChannel::OnSelectedCandidatePairChanged(
zhihuangb2cdd932017-01-19 16:54:25 -0800596 IceTransportInternal* ice_transport,
Honghai Zhang52dce732016-03-31 12:37:31 -0700597 CandidatePairInterface* selected_candidate_pair,
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700598 int last_sent_packet_id,
599 bool ready_to_send) {
deadbeeff5346592017-01-24 21:51:21 -0800600 RTC_DCHECK((rtp_dtls_transport_ &&
601 ice_transport == rtp_dtls_transport_->ice_transport()) ||
602 (rtcp_dtls_transport_ &&
603 ice_transport == rtcp_dtls_transport_->ice_transport()));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200604 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800605 selected_candidate_pair_ = selected_candidate_pair;
zhihuangb2cdd932017-01-19 16:54:25 -0800606 std::string transport_name = ice_transport->transport_name();
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700607 rtc::NetworkRoute network_route;
Honghai Zhangcc411c02016-03-29 17:27:21 -0700608 if (selected_candidate_pair) {
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700609 network_route = rtc::NetworkRoute(
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700610 ready_to_send, selected_candidate_pair->local_candidate().network_id(),
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700611 selected_candidate_pair->remote_candidate().network_id(),
612 last_sent_packet_id);
michaelt79e05882016-11-08 02:50:09 -0800613
614 UpdateTransportOverhead();
Honghai Zhangcc411c02016-03-29 17:27:21 -0700615 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200616 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700617 RTC_FROM_HERE, worker_thread_,
618 Bind(&MediaChannel::OnNetworkRouteChanged, media_channel_, transport_name,
619 network_route));
Honghai Zhangcc411c02016-03-29 17:27:21 -0700620}
621
zstein56162b92017-04-24 16:54:35 -0700622void BaseChannel::OnTransportReadyToSend(bool ready) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200623 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700624 RTC_FROM_HERE, worker_thread_,
zstein56162b92017-04-24 16:54:35 -0700625 Bind(&MediaChannel::OnReadyToSend, media_channel_, ready));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000626}
627
stefanc1aeaf02015-10-15 07:26:07 -0700628bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700629 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700630 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200631 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
632 // If the thread is not our network thread, we will post to our network
633 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000634 // synchronize access to all the pieces of the send path, including
635 // SRTP and the inner workings of the transport channels.
636 // The only downside is that we can't return a proper failure code if
637 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200638 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000639 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200640 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
641 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800642 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700643 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700644 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000645 return true;
646 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200647 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000648
649 // Now that we are on the correct thread, ensure we have a place to send this
650 // packet before doing anything. (We might get RTCP packets that we don't
651 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
652 // transport.
zsteine8ab5432017-07-12 11:48:11 -0700653 if (!rtp_transport_->IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000654 return false;
655 }
656
657 // Protect ourselves against crazy data.
658 if (!ValidPacket(rtcp, packet)) {
659 LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
zstein3dcf0e92017-06-01 13:22:42 -0700660 << RtpRtcpStringLiteral(rtcp)
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000661 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000662 return false;
663 }
664
stefanc1aeaf02015-10-15 07:26:07 -0700665 rtc::PacketOptions updated_options;
666 updated_options = options;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000667 // Protect if needed.
668 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200669 TRACE_EVENT0("webrtc", "SRTP Encode");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000670 bool res;
Karl Wibergc56ac1e2015-05-04 14:54:55 +0200671 uint8_t* data = packet->data();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000672 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000673 if (!rtcp) {
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000674 // If ENABLE_EXTERNAL_AUTH flag is on then packet authentication is not done
675 // inside libsrtp for a RTP packet. A external HMAC module will be writing
676 // a fake HMAC value. This is ONLY done for a RTP packet.
677 // Socket layer will update rtp sendtime extension header if present in
678 // packet with current time before updating the HMAC.
679#if !defined(ENABLE_EXTERNAL_AUTH)
680 res = srtp_filter_.ProtectRtp(
681 data, len, static_cast<int>(packet->capacity()), &len);
682#else
jbauchd48f4882017-03-01 15:34:36 -0800683 if (!srtp_filter_.IsExternalAuthActive()) {
684 res = srtp_filter_.ProtectRtp(
685 data, len, static_cast<int>(packet->capacity()), &len);
686 } else {
687 updated_options.packet_time_params.rtp_sendtime_extension_id =
688 rtp_abs_sendtime_extn_id_;
689 res = srtp_filter_.ProtectRtp(
690 data, len, static_cast<int>(packet->capacity()), &len,
691 &updated_options.packet_time_params.srtp_packet_index);
692 // If protection succeeds, let's get auth params from srtp.
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000693 if (res) {
jbauchd48f4882017-03-01 15:34:36 -0800694 uint8_t* auth_key = NULL;
695 int key_len;
696 res = srtp_filter_.GetRtpAuthParams(
697 &auth_key, &key_len,
698 &updated_options.packet_time_params.srtp_auth_tag_len);
699 if (res) {
700 updated_options.packet_time_params.srtp_auth_key.resize(key_len);
701 updated_options.packet_time_params.srtp_auth_key.assign(
702 auth_key, auth_key + key_len);
703 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000704 }
705 }
706#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000707 if (!res) {
708 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200709 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000710 GetRtpSeqNum(data, len, &seq_num);
711 GetRtpSsrc(data, len, &ssrc);
712 LOG(LS_ERROR) << "Failed to protect " << content_name_
713 << " RTP packet: size=" << len
714 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
715 return false;
716 }
717 } else {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000718 res = srtp_filter_.ProtectRtcp(data, len,
719 static_cast<int>(packet->capacity()),
720 &len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000721 if (!res) {
722 int type = -1;
723 GetRtcpType(data, len, &type);
724 LOG(LS_ERROR) << "Failed to protect " << content_name_
725 << " RTCP packet: size=" << len << ", type=" << type;
726 return false;
727 }
728 }
729
730 // Update the length of the packet now that we've added the auth tag.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000731 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800732 } else if (srtp_required_) {
deadbeef8f425f92016-12-01 12:26:27 -0800733 // The audio/video engines may attempt to send RTCP packets as soon as the
734 // streams are created, so don't treat this as an error for RTCP.
735 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
736 if (rtcp) {
737 return false;
738 }
739 // However, there shouldn't be any RTP packets sent before SRTP is set up
740 // (and SetSend(true) is called).
741 LOG(LS_ERROR) << "Can't send outgoing RTP packet when SRTP is inactive"
742 << " and crypto is required";
nisseeb4ca4e2017-01-12 02:24:27 -0800743 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000744 return false;
745 }
746
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000747 // Bon voyage.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200748 int flags = (secure() && secure_dtls()) ? PF_SRTP_BYPASS : PF_NORMAL;
zsteine8ab5432017-07-12 11:48:11 -0700749 return rtp_transport_->SendPacket(rtcp, packet, updated_options, flags);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000750}
751
zstein3dcf0e92017-06-01 13:22:42 -0700752bool BaseChannel::HandlesPayloadType(int packet_type) const {
zsteine8ab5432017-07-12 11:48:11 -0700753 return rtp_transport_->HandlesPayloadType(packet_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000754}
755
zstein3dcf0e92017-06-01 13:22:42 -0700756void BaseChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -0700757 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -0700758 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000759 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000760 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700761 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000762 }
763
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000764 // Unprotect the packet, if needed.
765 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200766 TRACE_EVENT0("webrtc", "SRTP Decode");
zstein634977b2017-07-14 12:30:04 -0700767 char* data = packet->data<char>();
768 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000769 bool res;
770 if (!rtcp) {
771 res = srtp_filter_.UnprotectRtp(data, len, &len);
772 if (!res) {
773 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200774 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000775 GetRtpSeqNum(data, len, &seq_num);
776 GetRtpSsrc(data, len, &ssrc);
777 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
zstein3dcf0e92017-06-01 13:22:42 -0700778 << " RTP packet: size=" << len << ", seqnum=" << seq_num
779 << ", SSRC=" << ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000780 return;
781 }
782 } else {
783 res = srtp_filter_.UnprotectRtcp(data, len, &len);
784 if (!res) {
785 int type = -1;
786 GetRtcpType(data, len, &type);
787 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
788 << " RTCP packet: size=" << len << ", type=" << type;
789 return;
790 }
791 }
792
zstein634977b2017-07-14 12:30:04 -0700793 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800794 } else if (srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000795 // Our session description indicates that SRTP is required, but we got a
796 // packet before our SRTP filter is active. This means either that
797 // a) we got SRTP packets before we received the SDES keys, in which case
798 // we can't decrypt it anyway, or
799 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800800 // transports, so we haven't yet extracted keys, even if DTLS did
801 // complete on the transport that the packets are being sent on. It's
802 // really good practice to wait for both RTP and RTCP to be good to go
803 // before sending media, to prevent weird failure modes, so it's fine
804 // for us to just eat packets here. This is all sidestepped if RTCP mux
805 // is used anyway.
zstein3dcf0e92017-06-01 13:22:42 -0700806 LOG(LS_WARNING) << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000807 << " packet when SRTP is inactive and crypto is required";
808 return;
809 }
810
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200811 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700812 RTC_FROM_HERE, worker_thread_,
zstein634977b2017-07-14 12:30:04 -0700813 Bind(&BaseChannel::ProcessPacket, this, rtcp, *packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200814}
815
zstein3dcf0e92017-06-01 13:22:42 -0700816void BaseChannel::ProcessPacket(bool rtcp,
817 const rtc::CopyOnWriteBuffer& packet,
818 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200819 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700820
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200821 // Need to copy variable because OnRtcpReceived/OnPacketReceived
822 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
823 rtc::CopyOnWriteBuffer data(packet);
824 if (rtcp) {
825 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000826 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200827 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000828 }
829}
830
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000831bool BaseChannel::PushdownLocalDescription(
832 const SessionDescription* local_desc, ContentAction action,
833 std::string* error_desc) {
834 const ContentInfo* content_info = GetFirstContent(local_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000835 const MediaContentDescription* content_desc =
836 GetContentDescription(content_info);
837 if (content_desc && content_info && !content_info->rejected &&
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000838 !SetLocalContent(content_desc, action, error_desc)) {
839 LOG(LS_ERROR) << "Failure in SetLocalContent with action " << action;
840 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000841 }
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000842 return true;
843}
844
845bool BaseChannel::PushdownRemoteDescription(
846 const SessionDescription* remote_desc, ContentAction action,
847 std::string* error_desc) {
848 const ContentInfo* content_info = GetFirstContent(remote_desc);
849 const MediaContentDescription* content_desc =
850 GetContentDescription(content_info);
851 if (content_desc && content_info && !content_info->rejected &&
852 !SetRemoteContent(content_desc, action, error_desc)) {
853 LOG(LS_ERROR) << "Failure in SetRemoteContent with action " << action;
854 return false;
855 }
856 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000857}
858
859void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700860 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000861 if (enabled_)
862 return;
863
864 LOG(LS_INFO) << "Channel enabled";
865 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700866 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000867}
868
869void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700870 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000871 if (!enabled_)
872 return;
873
874 LOG(LS_INFO) << "Channel disabled";
875 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700876 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000877}
878
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200879void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700880 rtc::PacketTransportInternal* rtp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700881 rtp_transport_->rtp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700882 rtc::PacketTransportInternal* rtcp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700883 rtp_transport_->rtcp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700884 if (rtp_packet_transport && rtp_packet_transport->writable() &&
885 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200886 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700887 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200888 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700889 }
890}
891
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200892void BaseChannel::ChannelWritable_n() {
893 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800894 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000895 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800896 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000897
deadbeefcbecd352015-09-23 11:50:27 -0700898 LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000899 << (was_ever_writable_ ? "" : " for the first time");
900
michaelt79e05882016-11-08 02:50:09 -0800901 if (selected_candidate_pair_)
902 LOG(LS_INFO)
903 << "Using "
904 << selected_candidate_pair_->local_candidate().ToSensitiveString()
905 << "->"
906 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000907
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000908 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200909 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000910 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700911 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000912}
913
deadbeef953c2ce2017-01-09 14:53:41 -0800914void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200915 RTC_DCHECK(network_thread_->IsCurrent());
916 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700917 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -0800918 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000919}
920
deadbeef953c2ce2017-01-09 14:53:41 -0800921void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700922 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -0800923 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000924}
925
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200926bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800927 // Since DTLS is applied to all transports, checking RTP should be enough.
928 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000929}
930
931// This function returns true if either DTLS-SRTP is not in use
932// *or* DTLS-SRTP is successfully set up.
zhihuangb2cdd932017-01-19 16:54:25 -0800933bool BaseChannel::SetupDtlsSrtp_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200934 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000935 bool ret = false;
936
zhihuangb2cdd932017-01-19 16:54:25 -0800937 DtlsTransportInternal* transport =
938 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
deadbeeff5346592017-01-24 21:51:21 -0800939 RTC_DCHECK(transport);
zhihuangb2cdd932017-01-19 16:54:25 -0800940 RTC_DCHECK(transport->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000941
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800942 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000943
zhihuangb2cdd932017-01-19 16:54:25 -0800944 if (!transport->GetSrtpCryptoSuite(&selected_crypto_suite)) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800945 LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000946 return false;
947 }
948
zhihuangb2cdd932017-01-19 16:54:25 -0800949 LOG(LS_INFO) << "Installing keys from DTLS-SRTP on " << content_name() << " "
zstein3dcf0e92017-06-01 13:22:42 -0700950 << RtpRtcpStringLiteral(rtcp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000951
jbauchcb560652016-08-04 05:20:32 -0700952 int key_len;
953 int salt_len;
954 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
955 &salt_len)) {
956 LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite" << selected_crypto_suite;
957 return false;
958 }
959
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000960 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -0700961 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000962
963 // RFC 5705 exporter using the RFC 5764 parameters
zhihuangb2cdd932017-01-19 16:54:25 -0800964 if (!transport->ExportKeyingMaterial(kDtlsSrtpExporterLabel, NULL, 0, false,
965 &dtls_buffer[0], dtls_buffer.size())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000966 LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -0800967 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000968 return false;
969 }
970
971 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -0700972 std::vector<unsigned char> client_write_key(key_len + salt_len);
973 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000974 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -0700975 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
976 offset += key_len;
977 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
978 offset += key_len;
979 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
980 offset += salt_len;
981 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000982
983 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000984 rtc::SSLRole role;
zhihuangb2cdd932017-01-19 16:54:25 -0800985 if (!transport->GetSslRole(&role)) {
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000986 LOG(LS_WARNING) << "GetSslRole failed";
987 return false;
988 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000989
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000990 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000991 send_key = &server_write_key;
992 recv_key = &client_write_key;
993 } else {
994 send_key = &client_write_key;
995 recv_key = &server_write_key;
996 }
997
jbauch5869f502017-06-29 12:31:36 -0700998 if (!srtp_filter_.IsActive()) {
999 if (rtcp) {
1000 ret = srtp_filter_.SetRtcpParams(selected_crypto_suite, &(*send_key)[0],
1001 static_cast<int>(send_key->size()),
1002 selected_crypto_suite, &(*recv_key)[0],
1003 static_cast<int>(recv_key->size()));
1004 } else {
1005 ret = srtp_filter_.SetRtpParams(selected_crypto_suite, &(*send_key)[0],
1006 static_cast<int>(send_key->size()),
1007 selected_crypto_suite, &(*recv_key)[0],
1008 static_cast<int>(recv_key->size()));
1009 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001010 } else {
jbauch5869f502017-06-29 12:31:36 -07001011 if (rtcp) {
1012 // RTCP doesn't need to be updated because UpdateRtpParams is only used
1013 // to update the set of encrypted RTP header extension IDs.
1014 ret = true;
1015 } else {
1016 ret = srtp_filter_.UpdateRtpParams(
1017 selected_crypto_suite,
1018 &(*send_key)[0], static_cast<int>(send_key->size()),
1019 selected_crypto_suite,
1020 &(*recv_key)[0], static_cast<int>(recv_key->size()));
1021 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001022 }
1023
michaelt79e05882016-11-08 02:50:09 -08001024 if (!ret) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001025 LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -08001026 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001027 dtls_keyed_ = true;
michaelt79e05882016-11-08 02:50:09 -08001028 UpdateTransportOverhead();
1029 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001030 return ret;
1031}
1032
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001033void BaseChannel::MaybeSetupDtlsSrtp_n() {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001034 if (srtp_filter_.IsActive()) {
1035 return;
1036 }
1037
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001038 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001039 return;
1040 }
1041
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001042 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001043 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001044 return;
1045 }
1046
zhihuangb2cdd932017-01-19 16:54:25 -08001047 if (rtcp_dtls_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001048 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001049 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001050 return;
1051 }
1052 }
1053}
1054
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001055void BaseChannel::ChannelNotWritable_n() {
1056 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001057 if (!writable_)
1058 return;
1059
deadbeefcbecd352015-09-23 11:50:27 -07001060 LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001061 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001062 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001063}
1064
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001065bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001066 const MediaContentDescription* content,
1067 ContentAction action,
1068 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001069 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001070 std::string* error_desc) {
1071 if (action == CA_UPDATE) {
1072 // These parameters never get changed by a CA_UDPATE.
1073 return true;
1074 }
1075
jbauch5869f502017-06-29 12:31:36 -07001076 std::vector<int> encrypted_extension_ids;
1077 for (const webrtc::RtpExtension& extension : extensions) {
1078 if (extension.encrypt) {
1079 LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
1080 << " encrypted extension: " << extension.ToString();
1081 encrypted_extension_ids.push_back(extension.id);
1082 }
1083 }
1084
deadbeef7af91dd2016-12-13 11:29:11 -08001085 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001086 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001087 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
jbauch5869f502017-06-29 12:31:36 -07001088 content, action, src, encrypted_extension_ids,
1089 error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001090}
1091
1092bool BaseChannel::SetRtpTransportParameters_n(
1093 const MediaContentDescription* content,
1094 ContentAction action,
1095 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001096 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001097 std::string* error_desc) {
1098 RTC_DCHECK(network_thread_->IsCurrent());
1099
jbauch5869f502017-06-29 12:31:36 -07001100 if (!SetSrtp_n(content->cryptos(), action, src, encrypted_extension_ids,
1101 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001102 return false;
1103 }
1104
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001105 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001106 return false;
1107 }
1108
1109 return true;
1110}
1111
zhihuangb2cdd932017-01-19 16:54:25 -08001112// |dtls| will be set to true if DTLS is active for transport and crypto is
1113// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001114bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
1115 bool* dtls,
1116 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -08001117 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001118 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001119 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001120 return false;
1121 }
1122 return true;
1123}
1124
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001125bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001126 ContentAction action,
1127 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001128 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001129 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001130 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001131 if (action == CA_UPDATE) {
1132 // no crypto params.
1133 return true;
1134 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001135 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001136 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001137 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001138 if (!ret) {
1139 return false;
1140 }
jbauch5869f502017-06-29 12:31:36 -07001141 srtp_filter_.SetEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001142 switch (action) {
1143 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001144 // If DTLS is already active on the channel, we could be renegotiating
1145 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001146 if (!dtls) {
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001147 ret = srtp_filter_.SetOffer(cryptos, src);
1148 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001149 break;
1150 case CA_PRANSWER:
1151 // If we're doing DTLS-SRTP, we don't want to update the filter
1152 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001153 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001154 ret = srtp_filter_.SetProvisionalAnswer(cryptos, src);
1155 }
1156 break;
1157 case CA_ANSWER:
1158 // If we're doing DTLS-SRTP, we don't want to update the filter
1159 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001160 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001161 ret = srtp_filter_.SetAnswer(cryptos, src);
1162 }
1163 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001164 default:
1165 break;
1166 }
jbauch5869f502017-06-29 12:31:36 -07001167 // Only update SRTP filter if using DTLS. SDES is handled internally
1168 // by the SRTP filter.
1169 // TODO(jbauch): Only update if encrypted extension ids have changed.
1170 if (ret && dtls_keyed_ && rtp_dtls_transport_ &&
1171 rtp_dtls_transport_->dtls_state() == DTLS_TRANSPORT_CONNECTED) {
1172 bool rtcp = false;
1173 ret = SetupDtlsSrtp_n(rtcp);
1174 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001175 if (!ret) {
1176 SafeSetError("Failed to setup SRTP filter.", error_desc);
1177 return false;
1178 }
1179 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001180}
1181
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001182bool BaseChannel::SetRtcpMux_n(bool enable,
1183 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001184 ContentSource src,
1185 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -08001186 // Provide a more specific error message for the RTCP mux "require" policy
1187 // case.
zstein56162b92017-04-24 16:54:35 -07001188 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -08001189 SafeSetError(
1190 "rtcpMuxPolicy is 'require', but media description does not "
1191 "contain 'a=rtcp-mux'.",
1192 error_desc);
1193 return false;
1194 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001195 bool ret = false;
1196 switch (action) {
1197 case CA_OFFER:
1198 ret = rtcp_mux_filter_.SetOffer(enable, src);
1199 break;
1200 case CA_PRANSWER:
zhihuangb2cdd932017-01-19 16:54:25 -08001201 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001202 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001203 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1204 break;
1205 case CA_ANSWER:
1206 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1207 if (ret && rtcp_mux_filter_.IsActive()) {
deadbeefe814a0d2017-02-25 18:15:09 -08001208 // We permanently activated RTCP muxing; signal that we no longer need
1209 // the RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -07001210 std::string debug_name =
1211 transport_name_.empty()
zsteine8ab5432017-07-12 11:48:11 -07001212 ? rtp_transport_->rtp_packet_transport()->debug_name()
zsteind48dbda2017-04-04 19:45:57 -07001213 : transport_name_;
deadbeefe814a0d2017-02-25 18:15:09 -08001214 ;
deadbeefcbecd352015-09-23 11:50:27 -07001215 LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
deadbeefe814a0d2017-02-25 18:15:09 -08001216 << "; no longer need RTCP transport for " << debug_name;
zsteine8ab5432017-07-12 11:48:11 -07001217 if (rtp_transport_->rtcp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -08001218 SetTransport_n(true, nullptr, nullptr);
1219 SignalRtcpMuxFullyActive(transport_name_);
zhihuangf5b251b2017-01-12 19:37:48 -08001220 }
deadbeef062ce9f2016-08-26 21:42:15 -07001221 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001222 }
1223 break;
1224 case CA_UPDATE:
1225 // No RTCP mux info.
1226 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001227 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001228 default:
1229 break;
1230 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001231 if (!ret) {
1232 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1233 return false;
1234 }
zsteine8ab5432017-07-12 11:48:11 -07001235 rtp_transport_->SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001236 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
zhihuangb2cdd932017-01-19 16:54:25 -08001237 // CA_ANSWER, but we only want to tear down the RTCP transport if we received
1238 // a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001239 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001240 // If the RTP transport is already writable, then so are we.
zsteine8ab5432017-07-12 11:48:11 -07001241 if (rtp_transport_->rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001242 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001243 }
1244 }
1245
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001246 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001247}
1248
1249bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001250 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001251 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001252}
1253
Peter Boström0c4e06b2015-10-07 12:23:21 +02001254bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001255 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001256 return media_channel()->RemoveRecvStream(ssrc);
1257}
1258
1259bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001260 ContentAction action,
1261 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001262 if (!(action == CA_OFFER || action == CA_ANSWER ||
1263 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001264 return false;
1265
1266 // If this is an update, streams only contain streams that have changed.
1267 if (action == CA_UPDATE) {
1268 for (StreamParamsVec::const_iterator it = streams.begin();
1269 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001270 const StreamParams* existing_stream =
1271 GetStreamByIds(local_streams_, it->groupid, it->id);
1272 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001273 if (media_channel()->AddSendStream(*it)) {
1274 local_streams_.push_back(*it);
1275 LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
1276 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001277 std::ostringstream desc;
1278 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1279 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001280 return false;
1281 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001282 } else if (existing_stream && !it->has_ssrcs()) {
1283 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001284 std::ostringstream desc;
1285 desc << "Failed to remove send stream with ssrc "
1286 << it->first_ssrc() << ".";
1287 SafeSetError(desc.str(), error_desc);
1288 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001289 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001290 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001291 } else {
1292 LOG(LS_WARNING) << "Ignore unsupported stream update";
1293 }
1294 }
1295 return true;
1296 }
1297 // Else streams are all the streams we want to send.
1298
1299 // Check for streams that have been removed.
1300 bool ret = true;
1301 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1302 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001303 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001304 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001305 std::ostringstream desc;
1306 desc << "Failed to remove send stream with ssrc "
1307 << it->first_ssrc() << ".";
1308 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001309 ret = false;
1310 }
1311 }
1312 }
1313 // Check for new streams.
1314 for (StreamParamsVec::const_iterator it = streams.begin();
1315 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001316 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001317 if (media_channel()->AddSendStream(*it)) {
stefanc1aeaf02015-10-15 07:26:07 -07001318 LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001319 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001320 std::ostringstream desc;
1321 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1322 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001323 ret = false;
1324 }
1325 }
1326 }
1327 local_streams_ = streams;
1328 return ret;
1329}
1330
1331bool BaseChannel::UpdateRemoteStreams_w(
1332 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001333 ContentAction action,
1334 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001335 if (!(action == CA_OFFER || action == CA_ANSWER ||
1336 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001337 return false;
1338
1339 // If this is an update, streams only contain streams that have changed.
1340 if (action == CA_UPDATE) {
1341 for (StreamParamsVec::const_iterator it = streams.begin();
1342 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001343 const StreamParams* existing_stream =
1344 GetStreamByIds(remote_streams_, it->groupid, it->id);
1345 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001346 if (AddRecvStream_w(*it)) {
1347 remote_streams_.push_back(*it);
1348 LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
1349 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001350 std::ostringstream desc;
1351 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1352 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001353 return false;
1354 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001355 } else if (existing_stream && !it->has_ssrcs()) {
1356 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001357 std::ostringstream desc;
1358 desc << "Failed to remove remote stream with ssrc "
1359 << it->first_ssrc() << ".";
1360 SafeSetError(desc.str(), error_desc);
1361 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001362 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001363 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001364 } else {
1365 LOG(LS_WARNING) << "Ignore unsupported stream update."
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001366 << " Stream exists? " << (existing_stream != nullptr)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001367 << " new stream = " << it->ToString();
1368 }
1369 }
1370 return true;
1371 }
1372 // Else streams are all the streams we want to receive.
1373
1374 // Check for streams that have been removed.
1375 bool ret = true;
1376 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1377 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001378 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001379 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001380 std::ostringstream desc;
1381 desc << "Failed to remove remote stream with ssrc "
1382 << it->first_ssrc() << ".";
1383 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001384 ret = false;
1385 }
1386 }
1387 }
1388 // Check for new streams.
1389 for (StreamParamsVec::const_iterator it = streams.begin();
1390 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001391 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001392 if (AddRecvStream_w(*it)) {
1393 LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
1394 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001395 std::ostringstream desc;
1396 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1397 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001398 ret = false;
1399 }
1400 }
1401 }
1402 remote_streams_ = streams;
1403 return ret;
1404}
1405
jbauch5869f502017-06-29 12:31:36 -07001406RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1407 const RtpHeaderExtensions& extensions) {
1408 if (!rtp_dtls_transport_ ||
1409 !rtp_dtls_transport_->crypto_options()
1410 .enable_encrypted_rtp_header_extensions) {
1411 RtpHeaderExtensions filtered;
1412 auto pred = [](const webrtc::RtpExtension& extension) {
1413 return !extension.encrypt;
1414 };
1415 std::copy_if(extensions.begin(), extensions.end(),
1416 std::back_inserter(filtered), pred);
1417 return filtered;
1418 }
1419
1420 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1421}
1422
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001423void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001424 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001425// Absolute Send Time extension id is used only with external auth,
1426// so do not bother searching for it and making asyncronious call to set
1427// something that is not used.
1428#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001429 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001430 webrtc::RtpExtension::FindHeaderExtensionByUri(
1431 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001432 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001433 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001434 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001435 RTC_FROM_HERE, network_thread_,
1436 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1437 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001438#endif
1439}
1440
1441void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1442 int rtp_abs_sendtime_extn_id) {
1443 rtp_abs_sendtime_extn_id_ = rtp_abs_sendtime_extn_id;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001444}
1445
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001446void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001447 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001448 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001449 case MSG_SEND_RTP_PACKET:
1450 case MSG_SEND_RTCP_PACKET: {
1451 RTC_DCHECK(network_thread_->IsCurrent());
1452 SendPacketMessageData* data =
1453 static_cast<SendPacketMessageData*>(pmsg->pdata);
1454 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1455 SendPacket(rtcp, &data->packet, data->options);
1456 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001457 break;
1458 }
1459 case MSG_FIRSTPACKETRECEIVED: {
1460 SignalFirstPacketReceived(this);
1461 break;
1462 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001463 }
1464}
1465
zstein3dcf0e92017-06-01 13:22:42 -07001466void BaseChannel::AddHandledPayloadType(int payload_type) {
zsteine8ab5432017-07-12 11:48:11 -07001467 rtp_transport_->AddHandledPayloadType(payload_type);
zstein3dcf0e92017-06-01 13:22:42 -07001468}
1469
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001470void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001471 // Flush all remaining RTCP messages. This should only be called in
1472 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001473 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001474 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001475 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1476 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001477 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1478 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001479 }
1480}
1481
johand89ab142016-10-25 10:50:32 -07001482void BaseChannel::SignalSentPacket_n(
deadbeef5bd5ca32017-02-10 11:31:50 -08001483 rtc::PacketTransportInternal* /* transport */,
johand89ab142016-10-25 10:50:32 -07001484 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001485 RTC_DCHECK(network_thread_->IsCurrent());
1486 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001487 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001488 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1489}
1490
1491void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1492 RTC_DCHECK(worker_thread_->IsCurrent());
1493 SignalSentPacket(sent_packet);
1494}
1495
1496VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1497 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001498 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001499 MediaEngineInterface* media_engine,
1500 VoiceMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001501 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001502 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001503 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001504 : BaseChannel(worker_thread,
1505 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001506 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001507 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001508 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001509 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001510 srtp_required),
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001511 media_engine_(media_engine),
deadbeefcbecd352015-09-23 11:50:27 -07001512 received_media_(false) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001513
1514VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001515 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001516 StopAudioMonitor();
1517 StopMediaMonitor();
1518 // this can't be done in the base class, since it calls a virtual
1519 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001520 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001521}
1522
Peter Boström0c4e06b2015-10-07 12:23:21 +02001523bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001524 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001525 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001526 AudioSource* source) {
stefanf79ade12017-06-02 06:44:03 -07001527 return InvokeOnWorker<bool>(
1528 RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
1529 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001530}
1531
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001532// TODO(juberti): Handle early media the right way. We should get an explicit
1533// ringing message telling us to start playing local ringback, which we cancel
1534// if any early media actually arrives. For now, we do the opposite, which is
1535// to wait 1 second for early media, and start playing local ringback if none
1536// arrives.
1537void VoiceChannel::SetEarlyMedia(bool enable) {
1538 if (enable) {
1539 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001540 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1541 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001542 } else {
1543 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001544 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001545 }
1546}
1547
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001548bool VoiceChannel::CanInsertDtmf() {
stefanf79ade12017-06-02 06:44:03 -07001549 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001550 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001551}
1552
Peter Boström0c4e06b2015-10-07 12:23:21 +02001553bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1554 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001555 int duration) {
stefanf79ade12017-06-02 06:44:03 -07001556 return InvokeOnWorker<bool>(
1557 RTC_FROM_HERE,
1558 Bind(&VoiceChannel::InsertDtmf_w, this, ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001559}
1560
solenberg4bac9c52015-10-09 02:32:53 -07001561bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
stefanf79ade12017-06-02 06:44:03 -07001562 return InvokeOnWorker<bool>(
1563 RTC_FROM_HERE,
1564 Bind(&VoiceMediaChannel::SetOutputVolume, media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001565}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001566
Tommif888bb52015-12-12 01:37:01 +01001567void VoiceChannel::SetRawAudioSink(
1568 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001569 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1570 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001571 // passing. So we invoke to our own little routine that gets a pointer to
1572 // our local variable. This is OK since we're synchronously invoking.
stefanf79ade12017-06-02 06:44:03 -07001573 InvokeOnWorker<bool>(RTC_FROM_HERE,
1574 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001575}
1576
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001577webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001578 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001579 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001580}
1581
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001582webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1583 uint32_t ssrc) const {
1584 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001585}
1586
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001587bool VoiceChannel::SetRtpSendParameters(
1588 uint32_t ssrc,
1589 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001590 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001591 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001592 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001593}
1594
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001595bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1596 webrtc::RtpParameters parameters) {
1597 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1598}
1599
1600webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1601 uint32_t ssrc) const {
1602 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001603 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001604 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1605}
1606
1607webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1608 uint32_t ssrc) const {
1609 return media_channel()->GetRtpReceiveParameters(ssrc);
1610}
1611
1612bool VoiceChannel::SetRtpReceiveParameters(
1613 uint32_t ssrc,
1614 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001615 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001616 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001617 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1618}
1619
1620bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1621 webrtc::RtpParameters parameters) {
1622 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001623}
1624
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001625bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001626 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1627 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001628}
1629
hbos8d609f62017-04-10 07:39:05 -07001630std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1631 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
zhihuang38ede132017-06-15 12:52:32 -07001632 RTC_FROM_HERE, Bind(&VoiceChannel::GetSources_w, this, ssrc));
1633}
1634
1635std::vector<webrtc::RtpSource> VoiceChannel::GetSources_w(uint32_t ssrc) const {
1636 RTC_DCHECK(worker_thread()->IsCurrent());
1637 return media_channel()->GetSources(ssrc);
hbos8d609f62017-04-10 07:39:05 -07001638}
1639
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001640void VoiceChannel::StartMediaMonitor(int cms) {
1641 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001642 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001643 media_monitor_->SignalUpdate.connect(
1644 this, &VoiceChannel::OnMediaMonitorUpdate);
1645 media_monitor_->Start(cms);
1646}
1647
1648void VoiceChannel::StopMediaMonitor() {
1649 if (media_monitor_) {
1650 media_monitor_->Stop();
1651 media_monitor_->SignalUpdate.disconnect(this);
1652 media_monitor_.reset();
1653 }
1654}
1655
1656void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001657 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001658 audio_monitor_
1659 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1660 audio_monitor_->Start(cms);
1661}
1662
1663void VoiceChannel::StopAudioMonitor() {
1664 if (audio_monitor_) {
1665 audio_monitor_->Stop();
1666 audio_monitor_.reset();
1667 }
1668}
1669
1670bool VoiceChannel::IsAudioMonitorRunning() const {
1671 return (audio_monitor_.get() != NULL);
1672}
1673
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001674int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001675 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001676}
1677
1678int VoiceChannel::GetOutputLevel_w() {
1679 return media_channel()->GetOutputLevel();
1680}
1681
1682void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1683 media_channel()->GetActiveStreams(actives);
1684}
1685
zstein3dcf0e92017-06-01 13:22:42 -07001686void VoiceChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -07001687 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -07001688 const rtc::PacketTime& packet_time) {
1689 BaseChannel::OnPacketReceived(rtcp, packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001690 // Set a flag when we've received an RTP packet. If we're waiting for early
1691 // media, this will disable the timeout.
zstein3dcf0e92017-06-01 13:22:42 -07001692 if (!received_media_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001693 received_media_ = true;
1694 }
1695}
1696
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001697void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001698 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001699 invoker_.AsyncInvoke<void>(
1700 RTC_FROM_HERE, worker_thread_,
1701 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001702}
1703
michaelt79e05882016-11-08 02:50:09 -08001704int BaseChannel::GetTransportOverheadPerPacket() const {
1705 RTC_DCHECK(network_thread_->IsCurrent());
1706
1707 if (!selected_candidate_pair_)
1708 return 0;
1709
1710 int transport_overhead_per_packet = 0;
1711
1712 constexpr int kIpv4Overhaed = 20;
1713 constexpr int kIpv6Overhaed = 40;
1714 transport_overhead_per_packet +=
1715 selected_candidate_pair_->local_candidate().address().family() == AF_INET
1716 ? kIpv4Overhaed
1717 : kIpv6Overhaed;
1718
1719 constexpr int kUdpOverhaed = 8;
1720 constexpr int kTcpOverhaed = 20;
1721 transport_overhead_per_packet +=
1722 selected_candidate_pair_->local_candidate().protocol() ==
1723 TCP_PROTOCOL_NAME
1724 ? kTcpOverhaed
1725 : kUdpOverhaed;
1726
1727 if (secure()) {
1728 int srtp_overhead = 0;
1729 if (srtp_filter_.GetSrtpOverhead(&srtp_overhead))
1730 transport_overhead_per_packet += srtp_overhead;
1731 }
1732
1733 return transport_overhead_per_packet;
1734}
1735
1736void BaseChannel::UpdateTransportOverhead() {
1737 int transport_overhead_per_packet = GetTransportOverheadPerPacket();
1738 if (transport_overhead_per_packet)
1739 invoker_.AsyncInvoke<void>(
1740 RTC_FROM_HERE, worker_thread_,
1741 Bind(&MediaChannel::OnTransportOverheadChanged, media_channel_,
1742 transport_overhead_per_packet));
1743}
1744
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001745void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001746 // Render incoming data if we're the active call, and we have the local
1747 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001748 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001749 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001750
1751 // Send outgoing data if we're the active call, we have the remote content,
1752 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001753 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001754 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001755
1756 LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
1757}
1758
1759const ContentInfo* VoiceChannel::GetFirstContent(
1760 const SessionDescription* sdesc) {
1761 return GetFirstAudioContent(sdesc);
1762}
1763
1764bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001765 ContentAction action,
1766 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001767 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001768 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001769 LOG(LS_INFO) << "Setting local voice description";
1770
1771 const AudioContentDescription* audio =
1772 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001773 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001774 if (!audio) {
1775 SafeSetError("Can't find audio content in local description.", error_desc);
1776 return false;
1777 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001778
jbauch5869f502017-06-29 12:31:36 -07001779 RtpHeaderExtensions rtp_header_extensions =
1780 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1781
1782 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1783 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001784 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001785 }
1786
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001787 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001788 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001789 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001790 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001791 error_desc);
1792 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001793 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001794 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001795 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001796 }
1797 last_recv_params_ = recv_params;
1798
1799 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1800 // only give it to the media channel once we have a remote
1801 // description too (without a remote description, we won't be able
1802 // to send them anyway).
1803 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1804 SafeSetError("Failed to set local audio description streams.", error_desc);
1805 return false;
1806 }
1807
1808 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001809 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001810 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001811}
1812
1813bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001814 ContentAction action,
1815 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001816 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001817 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001818 LOG(LS_INFO) << "Setting remote voice description";
1819
1820 const AudioContentDescription* audio =
1821 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001822 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001823 if (!audio) {
1824 SafeSetError("Can't find audio content in remote description.", error_desc);
1825 return false;
1826 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001827
jbauch5869f502017-06-29 12:31:36 -07001828 RtpHeaderExtensions rtp_header_extensions =
1829 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1830
1831 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
1832 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001833 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001834 }
1835
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001836 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001837 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1838 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001839 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001840 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001841 }
skvladdc1c62c2016-03-16 19:07:43 -07001842
1843 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1844 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001845 SafeSetError("Failed to set remote audio description send parameters.",
1846 error_desc);
1847 return false;
1848 }
1849 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001850
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001851 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1852 // and only give it to the media channel once we have a local
1853 // description too (without a local description, we won't be able to
1854 // recv them anyway).
1855 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1856 SafeSetError("Failed to set remote audio description streams.", error_desc);
1857 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001858 }
1859
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001860 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001861 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001862 }
1863
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001864 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001865 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001866 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001867}
1868
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001869void VoiceChannel::HandleEarlyMediaTimeout() {
1870 // This occurs on the main thread, not the worker thread.
1871 if (!received_media_) {
1872 LOG(LS_INFO) << "No early media received before timeout";
1873 SignalEarlyMediaTimeout(this);
1874 }
1875}
1876
Peter Boström0c4e06b2015-10-07 12:23:21 +02001877bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1878 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001879 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001880 if (!enabled()) {
1881 return false;
1882 }
solenberg1d63dd02015-12-02 12:35:09 -08001883 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001884}
1885
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001886void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001887 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001888 case MSG_EARLYMEDIATIMEOUT:
1889 HandleEarlyMediaTimeout();
1890 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001891 case MSG_CHANNEL_ERROR: {
1892 VoiceChannelErrorMessageData* data =
1893 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001894 delete data;
1895 break;
1896 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001897 default:
1898 BaseChannel::OnMessage(pmsg);
1899 break;
1900 }
1901}
1902
1903void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001904 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001905 SignalConnectionMonitor(this, infos);
1906}
1907
1908void VoiceChannel::OnMediaMonitorUpdate(
1909 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001910 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001911 SignalMediaMonitor(this, info);
1912}
1913
1914void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1915 const AudioInfo& info) {
1916 SignalAudioMonitor(this, info);
1917}
1918
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001919VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1920 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001921 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001922 VideoMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001923 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001924 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001925 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001926 : BaseChannel(worker_thread,
1927 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001928 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001929 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001930 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001931 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001932 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001933
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001934VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001935 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001936 StopMediaMonitor();
1937 // this can't be done in the base class, since it calls a virtual
1938 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001939
1940 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001941}
1942
nisse08582ff2016-02-04 01:24:52 -08001943bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001944 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001945 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001946 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001947 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001948 return true;
1949}
1950
deadbeef5a4a75a2016-06-02 16:23:38 -07001951bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001952 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001953 bool mute,
1954 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001955 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
stefanf79ade12017-06-02 06:44:03 -07001956 return InvokeOnWorker<bool>(
1957 RTC_FROM_HERE, Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
1958 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001959}
1960
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001961webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001962 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001963 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001964}
1965
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001966webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1967 uint32_t ssrc) const {
1968 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001969}
1970
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001971bool VideoChannel::SetRtpSendParameters(
1972 uint32_t ssrc,
1973 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001974 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001975 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001976 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001977}
1978
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001979bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1980 webrtc::RtpParameters parameters) {
1981 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1982}
1983
1984webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1985 uint32_t ssrc) const {
1986 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001987 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001988 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1989}
1990
1991webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1992 uint32_t ssrc) const {
1993 return media_channel()->GetRtpReceiveParameters(ssrc);
1994}
1995
1996bool VideoChannel::SetRtpReceiveParameters(
1997 uint32_t ssrc,
1998 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001999 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002000 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002001 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
2002}
2003
2004bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
2005 webrtc::RtpParameters parameters) {
2006 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07002007}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002008
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002009void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002010 // Send outgoing data if we're the active call, we have the remote content,
2011 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002012 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002013 if (!media_channel()->SetSend(send)) {
2014 LOG(LS_ERROR) << "Failed to SetSend on video channel";
2015 // TODO(gangji): Report error back to server.
2016 }
2017
Peter Boström34fbfff2015-09-24 19:20:30 +02002018 LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002019}
2020
stefanf79ade12017-06-02 06:44:03 -07002021void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
2022 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
2023 media_channel(), bwe_info));
2024}
2025
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00002026bool VideoChannel::GetStats(VideoMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07002027 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
2028 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002029}
2030
2031void VideoChannel::StartMediaMonitor(int cms) {
2032 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002033 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002034 media_monitor_->SignalUpdate.connect(
2035 this, &VideoChannel::OnMediaMonitorUpdate);
2036 media_monitor_->Start(cms);
2037}
2038
2039void VideoChannel::StopMediaMonitor() {
2040 if (media_monitor_) {
2041 media_monitor_->Stop();
2042 media_monitor_.reset();
2043 }
2044}
2045
2046const ContentInfo* VideoChannel::GetFirstContent(
2047 const SessionDescription* sdesc) {
2048 return GetFirstVideoContent(sdesc);
2049}
2050
2051bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002052 ContentAction action,
2053 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002054 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002055 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002056 LOG(LS_INFO) << "Setting local video description";
2057
2058 const VideoContentDescription* video =
2059 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002060 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002061 if (!video) {
2062 SafeSetError("Can't find video content in local description.", error_desc);
2063 return false;
2064 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002065
jbauch5869f502017-06-29 12:31:36 -07002066 RtpHeaderExtensions rtp_header_extensions =
2067 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
2068
2069 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
2070 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002071 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002072 }
2073
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002074 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07002075 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002076 if (!media_channel()->SetRecvParameters(recv_params)) {
2077 SafeSetError("Failed to set local video description recv parameters.",
2078 error_desc);
2079 return false;
2080 }
2081 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07002082 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002083 }
2084 last_recv_params_ = recv_params;
2085
2086 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2087 // only give it to the media channel once we have a remote
2088 // description too (without a remote description, we won't be able
2089 // to send them anyway).
2090 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2091 SafeSetError("Failed to set local video description streams.", error_desc);
2092 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002093 }
2094
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002095 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002096 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002097 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002098}
2099
2100bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002101 ContentAction action,
2102 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002103 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002104 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002105 LOG(LS_INFO) << "Setting remote video description";
2106
2107 const VideoContentDescription* video =
2108 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002109 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002110 if (!video) {
2111 SafeSetError("Can't find video content in remote description.", error_desc);
2112 return false;
2113 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002114
jbauch5869f502017-06-29 12:31:36 -07002115 RtpHeaderExtensions rtp_header_extensions =
2116 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
2117
2118 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2119 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002120 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002121 }
2122
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002123 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002124 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
2125 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002126 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002127 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002128 }
skvladdc1c62c2016-03-16 19:07:43 -07002129
2130 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2131
2132 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002133 SafeSetError("Failed to set remote video description send parameters.",
2134 error_desc);
2135 return false;
2136 }
2137 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002138
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002139 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2140 // and only give it to the media channel once we have a local
2141 // description too (without a local description, we won't be able to
2142 // recv them anyway).
2143 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2144 SafeSetError("Failed to set remote video description streams.", error_desc);
2145 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002146 }
2147
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002148 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07002149 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002150 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002151
2152 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002153 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002154 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002155}
2156
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002157void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002158 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002159 case MSG_CHANNEL_ERROR: {
2160 const VideoChannelErrorMessageData* data =
2161 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002162 delete data;
2163 break;
2164 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002165 default:
2166 BaseChannel::OnMessage(pmsg);
2167 break;
2168 }
2169}
2170
2171void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002172 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002173 SignalConnectionMonitor(this, infos);
2174}
2175
2176// TODO(pthatcher): Look into removing duplicate code between
2177// audio, video, and data, perhaps by using templates.
2178void VideoChannel::OnMediaMonitorUpdate(
2179 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002180 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002181 SignalMediaMonitor(this, info);
2182}
2183
deadbeef953c2ce2017-01-09 14:53:41 -08002184RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2185 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002186 rtc::Thread* signaling_thread,
deadbeef953c2ce2017-01-09 14:53:41 -08002187 DataMediaChannel* media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002188 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002189 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002190 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002191 : BaseChannel(worker_thread,
2192 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002193 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07002194 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07002195 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002196 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002197 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002198
deadbeef953c2ce2017-01-09 14:53:41 -08002199RtpDataChannel::~RtpDataChannel() {
2200 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002201 StopMediaMonitor();
2202 // this can't be done in the base class, since it calls a virtual
2203 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002204
2205 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002206}
2207
deadbeeff5346592017-01-24 21:51:21 -08002208bool RtpDataChannel::Init_w(
2209 DtlsTransportInternal* rtp_dtls_transport,
2210 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08002211 rtc::PacketTransportInternal* rtp_packet_transport,
2212 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -08002213 if (!BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
2214 rtp_packet_transport, rtcp_packet_transport)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002215 return false;
2216 }
deadbeef953c2ce2017-01-09 14:53:41 -08002217 media_channel()->SignalDataReceived.connect(this,
2218 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002219 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002220 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002221 return true;
2222}
2223
deadbeef953c2ce2017-01-09 14:53:41 -08002224bool RtpDataChannel::SendData(const SendDataParams& params,
2225 const rtc::CopyOnWriteBuffer& payload,
2226 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07002227 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002228 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2229 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002230}
2231
deadbeef953c2ce2017-01-09 14:53:41 -08002232const ContentInfo* RtpDataChannel::GetFirstContent(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002233 const SessionDescription* sdesc) {
2234 return GetFirstDataContent(sdesc);
2235}
2236
deadbeef953c2ce2017-01-09 14:53:41 -08002237bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002238 const DataContentDescription* content,
2239 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002240 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2241 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002242 // It's been set before, but doesn't match. That's bad.
2243 if (is_sctp) {
2244 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2245 error_desc);
2246 return false;
2247 }
2248 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002249}
2250
deadbeef953c2ce2017-01-09 14:53:41 -08002251bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2252 ContentAction action,
2253 std::string* error_desc) {
2254 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002255 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002256 LOG(LS_INFO) << "Setting local data description";
2257
2258 const DataContentDescription* data =
2259 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002260 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002261 if (!data) {
2262 SafeSetError("Can't find data content in local description.", error_desc);
2263 return false;
2264 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002265
deadbeef953c2ce2017-01-09 14:53:41 -08002266 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002267 return false;
2268 }
2269
jbauch5869f502017-06-29 12:31:36 -07002270 RtpHeaderExtensions rtp_header_extensions =
2271 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2272
2273 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
2274 rtp_header_extensions, error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08002275 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002276 }
2277
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002278 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07002279 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002280 if (!media_channel()->SetRecvParameters(recv_params)) {
2281 SafeSetError("Failed to set remote data description recv parameters.",
2282 error_desc);
2283 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002284 }
deadbeef953c2ce2017-01-09 14:53:41 -08002285 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07002286 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002287 }
2288 last_recv_params_ = recv_params;
2289
2290 // TODO(pthatcher): Move local streams into DataSendParameters, and
2291 // only give it to the media channel once we have a remote
2292 // description too (without a remote description, we won't be able
2293 // to send them anyway).
2294 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2295 SafeSetError("Failed to set local data description streams.", error_desc);
2296 return false;
2297 }
2298
2299 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002300 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002301 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002302}
2303
deadbeef953c2ce2017-01-09 14:53:41 -08002304bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2305 ContentAction action,
2306 std::string* error_desc) {
2307 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002308 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002309
2310 const DataContentDescription* data =
2311 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002312 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002313 if (!data) {
2314 SafeSetError("Can't find data content in remote description.", error_desc);
2315 return false;
2316 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002317
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002318 // If the remote data doesn't have codecs and isn't an update, it
2319 // must be empty, so ignore it.
2320 if (!data->has_codecs() && action != CA_UPDATE) {
2321 return true;
2322 }
2323
deadbeef953c2ce2017-01-09 14:53:41 -08002324 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002325 return false;
2326 }
2327
jbauch5869f502017-06-29 12:31:36 -07002328 RtpHeaderExtensions rtp_header_extensions =
2329 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2330
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002331 LOG(LS_INFO) << "Setting remote data description";
jbauch5869f502017-06-29 12:31:36 -07002332 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2333 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002334 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002335 }
2336
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002337 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002338 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
2339 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002340 if (!media_channel()->SetSendParameters(send_params)) {
2341 SafeSetError("Failed to set remote data description send parameters.",
2342 error_desc);
2343 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002344 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002345 last_send_params_ = send_params;
2346
2347 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2348 // and only give it to the media channel once we have a local
2349 // description too (without a local description, we won't be able to
2350 // recv them anyway).
2351 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2352 SafeSetError("Failed to set remote data description streams.",
2353 error_desc);
2354 return false;
2355 }
2356
2357 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002358 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002359 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002360}
2361
deadbeef953c2ce2017-01-09 14:53:41 -08002362void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002363 // Render incoming data if we're the active call, and we have the local
2364 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002365 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002366 if (!media_channel()->SetReceive(recv)) {
2367 LOG(LS_ERROR) << "Failed to SetReceive on data channel";
2368 }
2369
2370 // Send outgoing data if we're the active call, we have the remote content,
2371 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002372 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002373 if (!media_channel()->SetSend(send)) {
2374 LOG(LS_ERROR) << "Failed to SetSend on data channel";
2375 }
2376
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002377 // Trigger SignalReadyToSendData asynchronously.
2378 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002379
2380 LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
2381}
2382
deadbeef953c2ce2017-01-09 14:53:41 -08002383void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002384 switch (pmsg->message_id) {
2385 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002386 DataChannelReadyToSendMessageData* data =
2387 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002388 ready_to_send_data_ = data->data();
2389 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002390 delete data;
2391 break;
2392 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002393 case MSG_DATARECEIVED: {
2394 DataReceivedMessageData* data =
2395 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002396 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002397 delete data;
2398 break;
2399 }
2400 case MSG_CHANNEL_ERROR: {
2401 const DataChannelErrorMessageData* data =
2402 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002403 delete data;
2404 break;
2405 }
2406 default:
2407 BaseChannel::OnMessage(pmsg);
2408 break;
2409 }
2410}
2411
deadbeef953c2ce2017-01-09 14:53:41 -08002412void RtpDataChannel::OnConnectionMonitorUpdate(
2413 ConnectionMonitor* monitor,
2414 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002415 SignalConnectionMonitor(this, infos);
2416}
2417
deadbeef953c2ce2017-01-09 14:53:41 -08002418void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002419 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002420 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002421 media_monitor_->SignalUpdate.connect(this,
2422 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002423 media_monitor_->Start(cms);
2424}
2425
deadbeef953c2ce2017-01-09 14:53:41 -08002426void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002427 if (media_monitor_) {
2428 media_monitor_->Stop();
2429 media_monitor_->SignalUpdate.disconnect(this);
2430 media_monitor_.reset();
2431 }
2432}
2433
deadbeef953c2ce2017-01-09 14:53:41 -08002434void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2435 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002436 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002437 SignalMediaMonitor(this, info);
2438}
2439
deadbeef953c2ce2017-01-09 14:53:41 -08002440void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2441 const char* data,
2442 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002443 DataReceivedMessageData* msg = new DataReceivedMessageData(
2444 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002445 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002446}
2447
deadbeef953c2ce2017-01-09 14:53:41 -08002448void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2449 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002450 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2451 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002452 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002453}
2454
deadbeef953c2ce2017-01-09 14:53:41 -08002455void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002456 // This is usded for congestion control to indicate that the stream is ready
2457 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2458 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002459 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002460 new DataChannelReadyToSendMessageData(writable));
2461}
2462
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002463} // namespace cricket