blob: 3950c03a101d15a2ef768cc513246feaff5e1661 [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020015#include "pc/channel.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000016
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "api/call/audio_sink.h"
18#include "media/base/mediaconstants.h"
19#include "media/base/rtputils.h"
20#include "rtc_base/bind.h"
21#include "rtc_base/byteorder.h"
22#include "rtc_base/checks.h"
23#include "rtc_base/copyonwritebuffer.h"
24#include "rtc_base/dscp.h"
25#include "rtc_base/logging.h"
26#include "rtc_base/networkroute.h"
27#include "rtc_base/ptr_util.h"
28#include "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.
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020031#include "media/engine/webrtcvoiceengine.h" // nogncheck
32#include "p2p/base/packettransportinternal.h"
33#include "pc/channelmanager.h"
34#include "pc/rtptransport.h"
35#include "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
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700114template <class Codec>
115void RtpParametersFromMediaDescription(
116 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700117 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700118 RtpParameters<Codec>* params) {
119 // TODO(pthatcher): Remove this once we're sure no one will give us
120 // a description without codecs (currently a CA_UPDATE with just
121 // streams can).
122 if (desc->has_codecs()) {
123 params->codecs = desc->codecs();
124 }
125 // TODO(pthatcher): See if we really need
126 // rtp_header_extensions_set() and remove it if we don't.
127 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -0700128 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700129 }
deadbeef13871492015-12-09 12:37:51 -0800130 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700131}
132
nisse05103312016-03-16 02:22:50 -0700133template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700134void RtpSendParametersFromMediaDescription(
135 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700136 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -0700137 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -0700138 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700139 send_params->max_bandwidth_bps = desc->bandwidth();
140}
141
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200142BaseChannel::BaseChannel(rtc::Thread* worker_thread,
143 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800144 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -0800145 std::unique_ptr<MediaChannel> media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700146 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800147 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800148 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200149 : worker_thread_(worker_thread),
150 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800151 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000152 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700153 rtcp_mux_required_(rtcp_mux_required),
deadbeef7af91dd2016-12-13 11:29:11 -0800154 srtp_required_(srtp_required),
Steve Anton8699a322017-11-06 15:53:33 -0800155 media_channel_(std::move(media_channel)),
michaelt79e05882016-11-08 02:50:09 -0800156 selected_candidate_pair_(nullptr) {
Steve Anton8699a322017-11-06 15:53:33 -0800157 RTC_DCHECK_RUN_ON(worker_thread_);
Zhi Huangcf990f52017-09-22 12:12:30 -0700158 if (srtp_required) {
159 auto transport =
160 rtc::MakeUnique<webrtc::SrtpTransport>(rtcp_mux_required, content_name);
161 srtp_transport_ = transport.get();
162 rtp_transport_ = std::move(transport);
jbauchdfcab722017-03-06 00:14:10 -0800163#if defined(ENABLE_EXTERNAL_AUTH)
Zhi Huangcf990f52017-09-22 12:12:30 -0700164 srtp_transport_->EnableExternalAuth();
jbauchdfcab722017-03-06 00:14:10 -0800165#endif
Zhi Huangcf990f52017-09-22 12:12:30 -0700166 } else {
167 rtp_transport_ = rtc::MakeUnique<webrtc::RtpTransport>(rtcp_mux_required);
168 srtp_transport_ = nullptr;
169 }
zsteine8ab5432017-07-12 11:48:11 -0700170 rtp_transport_->SignalReadyToSend.connect(
zstein56162b92017-04-24 16:54:35 -0700171 this, &BaseChannel::OnTransportReadyToSend);
zstein3dcf0e92017-06-01 13:22:42 -0700172 // TODO(zstein): RtpTransport::SignalPacketReceived will probably be replaced
173 // with a callback interface later so that the demuxer can select which
174 // channel to signal.
zsteine8ab5432017-07-12 11:48:11 -0700175 rtp_transport_->SignalPacketReceived.connect(this,
zstein398c3fd2017-07-19 13:38:02 -0700176 &BaseChannel::OnPacketReceived);
Mirko Bonadei675513b2017-11-09 11:09:25 +0100177 RTC_LOG(LS_INFO) << "Created channel for " << content_name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000178}
179
180BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800181 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Steve Anton8699a322017-11-06 15:53:33 -0800182 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.org78187522013-10-07 23:32:02 +0000183 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000184 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200185 // Eats any outstanding messages or packets.
186 worker_thread_->Clear(&invoker_);
187 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000188 // We must destroy the media channel before the transport channel, otherwise
189 // the media channel may try to send on the dead transport channel. NULLing
190 // is not an effective strategy since the sends will come on another thread.
Steve Anton8699a322017-11-06 15:53:33 -0800191 media_channel_.reset();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100192 RTC_LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200193}
194
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200195void BaseChannel::DisconnectTransportChannels_n() {
196 // Send any outstanding RTCP packets.
197 FlushRtcpMessages_n();
198
199 // Stop signals from transport channels, but keep them alive because
200 // media_channel may use them from a different thread.
zhihuangb2cdd932017-01-19 16:54:25 -0800201 if (rtp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800202 DisconnectFromDtlsTransport(rtp_dtls_transport_);
zsteine8ab5432017-07-12 11:48:11 -0700203 } else if (rtp_transport_->rtp_packet_transport()) {
204 DisconnectFromPacketTransport(rtp_transport_->rtp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200205 }
zhihuangb2cdd932017-01-19 16:54:25 -0800206 if (rtcp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800207 DisconnectFromDtlsTransport(rtcp_dtls_transport_);
zsteine8ab5432017-07-12 11:48:11 -0700208 } else if (rtp_transport_->rtcp_packet_transport()) {
209 DisconnectFromPacketTransport(rtp_transport_->rtcp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200210 }
211
zsteine8ab5432017-07-12 11:48:11 -0700212 rtp_transport_->SetRtpPacketTransport(nullptr);
213 rtp_transport_->SetRtcpPacketTransport(nullptr);
zstein3dcf0e92017-06-01 13:22:42 -0700214
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200215 // Clear pending read packets/messages.
216 network_thread_->Clear(&invoker_);
217 network_thread_->Clear(this);
218}
219
Steve Anton8699a322017-11-06 15:53:33 -0800220void BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800221 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800222 rtc::PacketTransportInternal* rtp_packet_transport,
223 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -0800224 RTC_DCHECK_RUN_ON(worker_thread_);
225 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
226 return InitNetwork_n(rtp_dtls_transport, rtcp_dtls_transport,
227 rtp_packet_transport, rtcp_packet_transport);
228 });
229
deadbeeff5346592017-01-24 21:51:21 -0800230 // Both RTP and RTCP channels should be set, we can call SetInterface on
231 // the media channel and it can set network options.
wu@webrtc.orgde305012013-10-31 15:40:38 +0000232 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000233}
234
Steve Anton8699a322017-11-06 15:53:33 -0800235void BaseChannel::InitNetwork_n(
deadbeeff5346592017-01-24 21:51:21 -0800236 DtlsTransportInternal* rtp_dtls_transport,
237 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800238 rtc::PacketTransportInternal* rtp_packet_transport,
239 rtc::PacketTransportInternal* rtcp_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200240 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800241 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport, rtp_packet_transport,
242 rtcp_packet_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200243
zstein56162b92017-04-24 16:54:35 -0700244 if (rtcp_mux_required_) {
deadbeefac22f702017-01-12 21:59:29 -0800245 rtcp_mux_filter_.SetActive();
246 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200247}
248
wu@webrtc.org78187522013-10-07 23:32:02 +0000249void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200250 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000251 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200252 // Packets arrive on the network thread, processing packets calls virtual
253 // functions, so need to stop this process in Deinit that is called in
254 // derived classes destructor.
255 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700256 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000257}
258
zhihuangb2cdd932017-01-19 16:54:25 -0800259void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
260 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800261 network_thread_->Invoke<void>(
262 RTC_FROM_HERE,
263 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
264 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000265}
266
deadbeeff5346592017-01-24 21:51:21 -0800267void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800268 rtc::PacketTransportInternal* rtp_packet_transport,
269 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800270 network_thread_->Invoke<void>(
271 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
272 rtp_packet_transport, rtcp_packet_transport));
273}
zhihuangf5b251b2017-01-12 19:37:48 -0800274
deadbeeff5346592017-01-24 21:51:21 -0800275void BaseChannel::SetTransports_n(
276 DtlsTransportInternal* rtp_dtls_transport,
277 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800278 rtc::PacketTransportInternal* rtp_packet_transport,
279 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800280 RTC_DCHECK(network_thread_->IsCurrent());
281 // Validate some assertions about the input.
282 RTC_DCHECK(rtp_packet_transport);
283 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
284 if (rtp_dtls_transport || rtcp_dtls_transport) {
285 // DTLS/non-DTLS pointers should be to the same object.
286 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
287 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
288 // Can't go from non-DTLS to DTLS.
zsteine8ab5432017-07-12 11:48:11 -0700289 RTC_DCHECK(!rtp_transport_->rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800290 } else {
291 // Can't go from DTLS to non-DTLS.
292 RTC_DCHECK(!rtp_dtls_transport_);
293 }
294 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800295 if (rtp_dtls_transport && rtcp_dtls_transport) {
296 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
297 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800298 }
deadbeeff5346592017-01-24 21:51:21 -0800299 std::string debug_name;
300 if (rtp_dtls_transport) {
301 transport_name_ = rtp_dtls_transport->transport_name();
302 debug_name = transport_name_;
303 } else {
304 debug_name = rtp_packet_transport->debug_name();
305 }
zsteine8ab5432017-07-12 11:48:11 -0700306 if (rtp_packet_transport == rtp_transport_->rtp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -0800307 // Nothing to do if transport isn't changing.
deadbeefbad5dad2017-01-17 18:32:35 -0800308 return;
deadbeefcbecd352015-09-23 11:50:27 -0700309 }
310
Zhi Huangcf990f52017-09-22 12:12:30 -0700311 // When using DTLS-SRTP, we must reset the SrtpTransport every time the
312 // DtlsTransport changes and wait until the DTLS handshake is complete to set
313 // the newly negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200314 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800315 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700316 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800317 writable_ = false;
Zhi Huangcf990f52017-09-22 12:12:30 -0700318 dtls_active_ = false;
319 if (srtp_transport_) {
320 srtp_transport_->ResetParams();
321 }
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800322 }
323
deadbeefac22f702017-01-12 21:59:29 -0800324 // If this BaseChannel doesn't require RTCP mux and we haven't fully
325 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800326 if (rtcp_packet_transport) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100327 RTC_LOG(LS_INFO) << "Setting RTCP Transport for " << content_name()
328 << " on " << debug_name << " transport "
329 << rtcp_packet_transport;
deadbeeff5346592017-01-24 21:51:21 -0800330 SetTransport_n(true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000331 }
332
Mirko Bonadei675513b2017-11-09 11:09:25 +0100333 RTC_LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
334 << debug_name << " transport " << rtp_packet_transport;
deadbeeff5346592017-01-24 21:51:21 -0800335 SetTransport_n(false, rtp_dtls_transport, rtp_packet_transport);
guoweis46383312015-12-17 16:45:59 -0800336
deadbeefcbecd352015-09-23 11:50:27 -0700337 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700338 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200339 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000340}
341
deadbeeff5346592017-01-24 21:51:21 -0800342void BaseChannel::SetTransport_n(
343 bool rtcp,
344 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800345 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200346 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800347 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800348 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700349 rtc::PacketTransportInternal* old_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700350 rtcp ? rtp_transport_->rtcp_packet_transport()
351 : rtp_transport_->rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800352
deadbeeff5346592017-01-24 21:51:21 -0800353 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700354 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000355 return;
356 }
zhihuangb2cdd932017-01-19 16:54:25 -0800357
deadbeeff5346592017-01-24 21:51:21 -0800358 RTC_DCHECK(old_packet_transport != new_packet_transport);
359 if (old_dtls_transport) {
360 DisconnectFromDtlsTransport(old_dtls_transport);
361 } else if (old_packet_transport) {
362 DisconnectFromPacketTransport(old_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000363 }
364
zsteind48dbda2017-04-04 19:45:57 -0700365 if (rtcp) {
zsteine8ab5432017-07-12 11:48:11 -0700366 rtp_transport_->SetRtcpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700367 } else {
zsteine8ab5432017-07-12 11:48:11 -0700368 rtp_transport_->SetRtpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700369 }
deadbeeff5346592017-01-24 21:51:21 -0800370 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000371
deadbeeff5346592017-01-24 21:51:21 -0800372 // If there's no new transport, we're done after disconnecting from old one.
373 if (!new_packet_transport) {
374 return;
375 }
376
377 if (rtcp && new_dtls_transport) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700378 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_active()))
379 << "Setting RTCP for DTLS/SRTP after the DTLS is active "
deadbeeff5346592017-01-24 21:51:21 -0800380 << "should never happen.";
381 }
zstein56162b92017-04-24 16:54:35 -0700382
deadbeeff5346592017-01-24 21:51:21 -0800383 if (new_dtls_transport) {
384 ConnectToDtlsTransport(new_dtls_transport);
385 } else {
386 ConnectToPacketTransport(new_packet_transport);
387 }
388 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
389 for (const auto& pair : socket_options) {
390 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800391 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000392}
393
deadbeeff5346592017-01-24 21:51:21 -0800394void BaseChannel::ConnectToDtlsTransport(DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200395 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000396
zstein56162b92017-04-24 16:54:35 -0700397 // TODO(zstein): de-dup with ConnectToPacketTransport
zhihuangb2cdd932017-01-19 16:54:25 -0800398 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
zhihuangb2cdd932017-01-19 16:54:25 -0800399 transport->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
400 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
401 transport->ice_transport()->SignalSelectedCandidatePairChanged.connect(
Honghai Zhangcc411c02016-03-29 17:27:21 -0700402 this, &BaseChannel::OnSelectedCandidatePairChanged);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000403}
404
deadbeeff5346592017-01-24 21:51:21 -0800405void BaseChannel::DisconnectFromDtlsTransport(
406 DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200407 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangb2cdd932017-01-19 16:54:25 -0800408 OnSelectedCandidatePairChanged(transport->ice_transport(), nullptr, -1,
409 false);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000410
zhihuangb2cdd932017-01-19 16:54:25 -0800411 transport->SignalWritableState.disconnect(this);
zhihuangb2cdd932017-01-19 16:54:25 -0800412 transport->SignalDtlsState.disconnect(this);
413 transport->SignalSentPacket.disconnect(this);
414 transport->ice_transport()->SignalSelectedCandidatePairChanged.disconnect(
415 this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000416}
417
deadbeeff5346592017-01-24 21:51:21 -0800418void BaseChannel::ConnectToPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800419 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800420 RTC_DCHECK_RUN_ON(network_thread_);
421 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
deadbeeff5346592017-01-24 21:51:21 -0800422 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
423}
424
425void BaseChannel::DisconnectFromPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800426 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800427 RTC_DCHECK_RUN_ON(network_thread_);
428 transport->SignalWritableState.disconnect(this);
deadbeeff5346592017-01-24 21:51:21 -0800429 transport->SignalSentPacket.disconnect(this);
430}
431
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000432bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700433 worker_thread_->Invoke<void>(
434 RTC_FROM_HERE,
435 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
436 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000437 return true;
438}
439
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000440bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700441 return InvokeOnWorker<bool>(RTC_FROM_HERE,
442 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000443}
444
Peter Boström0c4e06b2015-10-07 12:23:21 +0200445bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700446 return InvokeOnWorker<bool>(
447 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000448}
449
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000450bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700451 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700452 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000453}
454
Peter Boström0c4e06b2015-10-07 12:23:21 +0200455bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700456 return InvokeOnWorker<bool>(
457 RTC_FROM_HERE,
458 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000459}
460
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000461bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000462 ContentAction action,
463 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100464 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700465 return InvokeOnWorker<bool>(
466 RTC_FROM_HERE,
467 Bind(&BaseChannel::SetLocalContent_w, this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000468}
469
470bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000471 ContentAction action,
472 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100473 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700474 return InvokeOnWorker<bool>(
475 RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w, this, content,
476 action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000477}
478
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000479void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800480 // We pass in the BaseChannel instead of the rtp_dtls_transport_
481 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000482 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200483 // We pass in the network thread because on that thread connection monitor
484 // will call BaseChannel::GetConnectionStats which must be called on the
485 // network thread.
486 connection_monitor_.reset(
487 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000488 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000489 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000490 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000491}
492
493void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000494 if (connection_monitor_) {
495 connection_monitor_->Stop();
496 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000497 }
498}
499
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000500bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200501 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800502 if (!rtp_dtls_transport_) {
503 return false;
504 }
zhihuangb2cdd932017-01-19 16:54:25 -0800505 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800506}
507
508bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800509 // If this BaseChannel doesn't require RTCP mux and we haven't fully
510 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700511 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000512}
513
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700514bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000515 // Receive data if we are enabled and have local content,
516 return enabled() && IsReceiveContentDirection(local_content_direction_);
517}
518
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700519bool BaseChannel::IsReadyToSendMedia_w() const {
520 // Need to access some state updated on the network thread.
521 return network_thread_->Invoke<bool>(
522 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
523}
524
525bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000526 // Send outgoing data if we are enabled, have local and remote content,
527 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800528 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000529 IsSendContentDirection(local_content_direction_) &&
Zhi Huangcf990f52017-09-22 12:12:30 -0700530 was_ever_writable() && (srtp_active() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000531}
532
jbaucheec21bd2016-03-20 06:15:43 -0700533bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700534 const rtc::PacketOptions& options) {
535 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000536}
537
jbaucheec21bd2016-03-20 06:15:43 -0700538bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700539 const rtc::PacketOptions& options) {
540 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000541}
542
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000543int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000544 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200545 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700546 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200547}
548
549int BaseChannel::SetOption_n(SocketType type,
550 rtc::Socket::Option opt,
551 int value) {
552 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800553 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000554 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000555 case ST_RTP:
zsteine8ab5432017-07-12 11:48:11 -0700556 transport = rtp_transport_->rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700557 socket_options_.push_back(
558 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000559 break;
560 case ST_RTCP:
zsteine8ab5432017-07-12 11:48:11 -0700561 transport = rtp_transport_->rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700562 rtcp_socket_options_.push_back(
563 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000564 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000565 }
deadbeeff5346592017-01-24 21:51:21 -0800566 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000567}
568
deadbeef5bd5ca32017-02-10 11:31:50 -0800569void BaseChannel::OnWritableState(rtc::PacketTransportInternal* transport) {
zsteine8ab5432017-07-12 11:48:11 -0700570 RTC_DCHECK(transport == rtp_transport_->rtp_packet_transport() ||
571 transport == rtp_transport_->rtcp_packet_transport());
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200572 RTC_DCHECK(network_thread_->IsCurrent());
573 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000574}
575
zhihuangb2cdd932017-01-19 16:54:25 -0800576void BaseChannel::OnDtlsState(DtlsTransportInternal* transport,
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800577 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200578 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800579 return;
580 }
581
Zhi Huangcf990f52017-09-22 12:12:30 -0700582 // Reset the SrtpTransport if it's not the CONNECTED state. For the CONNECTED
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800583 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
zhihuangb2cdd932017-01-19 16:54:25 -0800584 // cover other scenarios like the whole transport is writable (not just this
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800585 // TransportChannel) or when TransportChannel is attached after DTLS is
586 // negotiated.
587 if (state != DTLS_TRANSPORT_CONNECTED) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700588 dtls_active_ = false;
589 if (srtp_transport_) {
590 srtp_transport_->ResetParams();
591 }
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800592 }
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 }
Steve Anton8699a322017-11-06 15:53:33 -0800616 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_, [=] {
617 media_channel_->OnNetworkRouteChanged(transport_name, network_route);
618 });
Honghai Zhangcc411c02016-03-29 17:27:21 -0700619}
620
zstein56162b92017-04-24 16:54:35 -0700621void BaseChannel::OnTransportReadyToSend(bool ready) {
Steve Anton8699a322017-11-06 15:53:33 -0800622 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_,
623 [=] { media_channel_->OnReadyToSend(ready); });
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000624}
625
stefanc1aeaf02015-10-15 07:26:07 -0700626bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700627 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700628 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200629 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
630 // If the thread is not our network thread, we will post to our network
631 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000632 // synchronize access to all the pieces of the send path, including
633 // SRTP and the inner workings of the transport channels.
634 // The only downside is that we can't return a proper failure code if
635 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200636 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000637 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200638 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
639 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800640 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700641 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700642 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000643 return true;
644 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200645 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000646
647 // Now that we are on the correct thread, ensure we have a place to send this
648 // packet before doing anything. (We might get RTCP packets that we don't
649 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
650 // transport.
zsteine8ab5432017-07-12 11:48:11 -0700651 if (!rtp_transport_->IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000652 return false;
653 }
654
655 // Protect ourselves against crazy data.
656 if (!ValidPacket(rtcp, packet)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100657 RTC_LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
658 << RtpRtcpStringLiteral(rtcp)
659 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000660 return false;
661 }
662
Zhi Huangcf990f52017-09-22 12:12:30 -0700663 if (!srtp_active()) {
664 if (srtp_required_) {
665 // The audio/video engines may attempt to send RTCP packets as soon as the
666 // streams are created, so don't treat this as an error for RTCP.
667 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
668 if (rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000669 return false;
670 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700671 // However, there shouldn't be any RTP packets sent before SRTP is set up
672 // (and SetSend(true) is called).
Mirko Bonadei675513b2017-11-09 11:09:25 +0100673 RTC_LOG(LS_ERROR)
674 << "Can't send outgoing RTP packet when SRTP is inactive"
675 << " and crypto is required";
Zhi Huangcf990f52017-09-22 12:12:30 -0700676 RTC_NOTREACHED();
deadbeef8f425f92016-12-01 12:26:27 -0800677 return false;
678 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700679 // Bon voyage.
Zhi Huang04eaa152017-10-04 14:08:30 -0700680 return rtcp
681 ? rtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
682 : rtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000683 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700684 RTC_DCHECK(srtp_transport_);
685 RTC_DCHECK(srtp_transport_->IsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000686 // Bon voyage.
Zhi Huangcf990f52017-09-22 12:12:30 -0700687 return rtcp ? srtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
688 : srtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000689}
690
zstein3dcf0e92017-06-01 13:22:42 -0700691bool BaseChannel::HandlesPayloadType(int packet_type) const {
zsteine8ab5432017-07-12 11:48:11 -0700692 return rtp_transport_->HandlesPayloadType(packet_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000693}
694
zstein3dcf0e92017-06-01 13:22:42 -0700695void BaseChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -0700696 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -0700697 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000698 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000699 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700700 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000701 }
702
Zhi Huangcf990f52017-09-22 12:12:30 -0700703 if (!srtp_active() && srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000704 // Our session description indicates that SRTP is required, but we got a
705 // packet before our SRTP filter is active. This means either that
706 // a) we got SRTP packets before we received the SDES keys, in which case
707 // we can't decrypt it anyway, or
708 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800709 // transports, so we haven't yet extracted keys, even if DTLS did
710 // complete on the transport that the packets are being sent on. It's
711 // really good practice to wait for both RTP and RTCP to be good to go
712 // before sending media, to prevent weird failure modes, so it's fine
713 // for us to just eat packets here. This is all sidestepped if RTCP mux
714 // is used anyway.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100715 RTC_LOG(LS_WARNING)
716 << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
717 << " packet when SRTP is inactive and crypto is required";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000718 return;
719 }
720
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200721 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700722 RTC_FROM_HERE, worker_thread_,
zstein634977b2017-07-14 12:30:04 -0700723 Bind(&BaseChannel::ProcessPacket, this, rtcp, *packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200724}
725
zstein3dcf0e92017-06-01 13:22:42 -0700726void BaseChannel::ProcessPacket(bool rtcp,
727 const rtc::CopyOnWriteBuffer& packet,
728 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200729 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700730
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200731 // Need to copy variable because OnRtcpReceived/OnPacketReceived
732 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
733 rtc::CopyOnWriteBuffer data(packet);
734 if (rtcp) {
735 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000736 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200737 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000738 }
739}
740
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000741void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700742 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000743 if (enabled_)
744 return;
745
Mirko Bonadei675513b2017-11-09 11:09:25 +0100746 RTC_LOG(LS_INFO) << "Channel enabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000747 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700748 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000749}
750
751void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700752 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000753 if (!enabled_)
754 return;
755
Mirko Bonadei675513b2017-11-09 11:09:25 +0100756 RTC_LOG(LS_INFO) << "Channel disabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000757 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700758 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000759}
760
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200761void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700762 rtc::PacketTransportInternal* rtp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700763 rtp_transport_->rtp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700764 rtc::PacketTransportInternal* rtcp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700765 rtp_transport_->rtcp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700766 if (rtp_packet_transport && rtp_packet_transport->writable() &&
767 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200768 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700769 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200770 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700771 }
772}
773
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200774void BaseChannel::ChannelWritable_n() {
775 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800776 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000777 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800778 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000779
Mirko Bonadei675513b2017-11-09 11:09:25 +0100780 RTC_LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
781 << (was_ever_writable_ ? "" : " for the first time");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000782
michaelt79e05882016-11-08 02:50:09 -0800783 if (selected_candidate_pair_)
Mirko Bonadei675513b2017-11-09 11:09:25 +0100784 RTC_LOG(LS_INFO)
michaelt79e05882016-11-08 02:50:09 -0800785 << "Using "
786 << selected_candidate_pair_->local_candidate().ToSensitiveString()
787 << "->"
788 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000789
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000790 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200791 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000792 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700793 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000794}
795
deadbeef953c2ce2017-01-09 14:53:41 -0800796void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200797 RTC_DCHECK(network_thread_->IsCurrent());
798 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700799 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -0800800 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000801}
802
deadbeef953c2ce2017-01-09 14:53:41 -0800803void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700804 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -0800805 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000806}
807
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200808bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800809 // Since DTLS is applied to all transports, checking RTP should be enough.
810 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000811}
812
813// This function returns true if either DTLS-SRTP is not in use
814// *or* DTLS-SRTP is successfully set up.
zhihuangb2cdd932017-01-19 16:54:25 -0800815bool BaseChannel::SetupDtlsSrtp_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200816 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000817 bool ret = false;
818
zhihuangb2cdd932017-01-19 16:54:25 -0800819 DtlsTransportInternal* transport =
820 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
deadbeeff5346592017-01-24 21:51:21 -0800821 RTC_DCHECK(transport);
zhihuangb2cdd932017-01-19 16:54:25 -0800822 RTC_DCHECK(transport->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000823
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800824 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000825
zhihuangb2cdd932017-01-19 16:54:25 -0800826 if (!transport->GetSrtpCryptoSuite(&selected_crypto_suite)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100827 RTC_LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000828 return false;
829 }
830
Mirko Bonadei675513b2017-11-09 11:09:25 +0100831 RTC_LOG(LS_INFO) << "Installing keys from DTLS-SRTP on " << content_name()
832 << " " << RtpRtcpStringLiteral(rtcp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000833
jbauchcb560652016-08-04 05:20:32 -0700834 int key_len;
835 int salt_len;
836 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
837 &salt_len)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100838 RTC_LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite"
839 << selected_crypto_suite;
jbauchcb560652016-08-04 05:20:32 -0700840 return false;
841 }
842
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000843 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -0700844 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000845
846 // RFC 5705 exporter using the RFC 5764 parameters
zhihuangb2cdd932017-01-19 16:54:25 -0800847 if (!transport->ExportKeyingMaterial(kDtlsSrtpExporterLabel, NULL, 0, false,
848 &dtls_buffer[0], dtls_buffer.size())) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100849 RTC_LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -0800850 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000851 return false;
852 }
853
854 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -0700855 std::vector<unsigned char> client_write_key(key_len + salt_len);
856 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000857 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -0700858 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
859 offset += key_len;
860 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
861 offset += key_len;
862 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
863 offset += salt_len;
864 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000865
866 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000867 rtc::SSLRole role;
zhihuangb2cdd932017-01-19 16:54:25 -0800868 if (!transport->GetSslRole(&role)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100869 RTC_LOG(LS_WARNING) << "GetSslRole failed";
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000870 return false;
871 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000872
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000873 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000874 send_key = &server_write_key;
875 recv_key = &client_write_key;
876 } else {
877 send_key = &client_write_key;
878 recv_key = &server_write_key;
879 }
880
Zhi Huangc99b6c72017-11-10 16:44:46 -0800881 // Use an empty encrypted header extension ID vector if not set. This could
882 // happen when the DTLS handshake is completed before processing the
883 // Offer/Answer which contains the encrypted header extension IDs.
884 std::vector<int> send_extension_ids;
885 std::vector<int> recv_extension_ids;
886 if (catched_send_extension_ids_) {
887 send_extension_ids = *catched_send_extension_ids_;
888 }
889 if (catched_recv_extension_ids_) {
890 recv_extension_ids = *catched_recv_extension_ids_;
891 }
892
Zhi Huangcf990f52017-09-22 12:12:30 -0700893 if (rtcp) {
894 if (!dtls_active()) {
895 RTC_DCHECK(srtp_transport_);
896 ret = srtp_transport_->SetRtcpParams(
897 selected_crypto_suite, &(*send_key)[0],
Zhi Huangc99b6c72017-11-10 16:44:46 -0800898 static_cast<int>(send_key->size()), send_extension_ids,
899 selected_crypto_suite, &(*recv_key)[0],
900 static_cast<int>(recv_key->size()), recv_extension_ids);
jbauch5869f502017-06-29 12:31:36 -0700901 } else {
Zhi Huangcf990f52017-09-22 12:12:30 -0700902 // RTCP doesn't need to call SetRtpParam because it is only used
903 // to make the updated encrypted RTP header extension IDs take effect.
904 ret = true;
jbauch5869f502017-06-29 12:31:36 -0700905 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000906 } else {
Zhi Huangcf990f52017-09-22 12:12:30 -0700907 RTC_DCHECK(srtp_transport_);
Zhi Huangc99b6c72017-11-10 16:44:46 -0800908 ret = srtp_transport_->SetRtpParams(
909 selected_crypto_suite, &(*send_key)[0],
910 static_cast<int>(send_key->size()), send_extension_ids,
911 selected_crypto_suite, &(*recv_key)[0],
912 static_cast<int>(recv_key->size()), recv_extension_ids);
Zhi Huangcf990f52017-09-22 12:12:30 -0700913 dtls_active_ = ret;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000914 }
915
michaelt79e05882016-11-08 02:50:09 -0800916 if (!ret) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100917 RTC_LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -0800918 } else {
michaelt79e05882016-11-08 02:50:09 -0800919 UpdateTransportOverhead();
920 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000921 return ret;
922}
923
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200924void BaseChannel::MaybeSetupDtlsSrtp_n() {
Zhi Huangcf990f52017-09-22 12:12:30 -0700925 if (dtls_active()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800926 return;
927 }
928
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200929 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800930 return;
931 }
932
Zhi Huangcf990f52017-09-22 12:12:30 -0700933 if (!srtp_transport_) {
934 EnableSrtpTransport_n();
935 }
936
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200937 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -0800938 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800939 return;
940 }
941
zhihuangb2cdd932017-01-19 16:54:25 -0800942 if (rtcp_dtls_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200943 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -0800944 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800945 return;
946 }
947 }
948}
949
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200950void BaseChannel::ChannelNotWritable_n() {
951 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000952 if (!writable_)
953 return;
954
Mirko Bonadei675513b2017-11-09 11:09:25 +0100955 RTC_LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000956 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700957 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000958}
959
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200960bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700961 const MediaContentDescription* content,
962 ContentAction action,
963 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700964 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700965 std::string* error_desc) {
966 if (action == CA_UPDATE) {
967 // These parameters never get changed by a CA_UDPATE.
968 return true;
969 }
970
jbauch5869f502017-06-29 12:31:36 -0700971 std::vector<int> encrypted_extension_ids;
972 for (const webrtc::RtpExtension& extension : extensions) {
973 if (extension.encrypt) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100974 RTC_LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
975 << " encrypted extension: " << extension.ToString();
jbauch5869f502017-06-29 12:31:36 -0700976 encrypted_extension_ids.push_back(extension.id);
977 }
978 }
979
deadbeef7af91dd2016-12-13 11:29:11 -0800980 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200981 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700982 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
jbauch5869f502017-06-29 12:31:36 -0700983 content, action, src, encrypted_extension_ids,
984 error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200985}
986
987bool BaseChannel::SetRtpTransportParameters_n(
988 const MediaContentDescription* content,
989 ContentAction action,
990 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700991 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200992 std::string* error_desc) {
993 RTC_DCHECK(network_thread_->IsCurrent());
994
jbauch5869f502017-06-29 12:31:36 -0700995 if (!SetSrtp_n(content->cryptos(), action, src, encrypted_extension_ids,
996 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700997 return false;
998 }
999
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001000 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001001 return false;
1002 }
1003
1004 return true;
1005}
1006
zhihuangb2cdd932017-01-19 16:54:25 -08001007// |dtls| will be set to true if DTLS is active for transport and crypto is
1008// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001009bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
1010 bool* dtls,
1011 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -08001012 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001013 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001014 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001015 return false;
1016 }
1017 return true;
1018}
1019
Zhi Huangcf990f52017-09-22 12:12:30 -07001020void BaseChannel::EnableSrtpTransport_n() {
1021 if (srtp_transport_ == nullptr) {
1022 rtp_transport_->SignalReadyToSend.disconnect(this);
1023 rtp_transport_->SignalPacketReceived.disconnect(this);
1024
1025 auto transport = rtc::MakeUnique<webrtc::SrtpTransport>(
1026 std::move(rtp_transport_), content_name_);
1027 srtp_transport_ = transport.get();
1028 rtp_transport_ = std::move(transport);
1029
1030 rtp_transport_->SignalReadyToSend.connect(
1031 this, &BaseChannel::OnTransportReadyToSend);
1032 rtp_transport_->SignalPacketReceived.connect(
1033 this, &BaseChannel::OnPacketReceived);
Mirko Bonadei675513b2017-11-09 11:09:25 +01001034 RTC_LOG(LS_INFO) << "Wrapping RtpTransport in SrtpTransport.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001035 }
1036}
1037
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001038bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001039 ContentAction action,
1040 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001041 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001042 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001043 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001044 if (action == CA_UPDATE) {
1045 // no crypto params.
1046 return true;
1047 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001048 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001049 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001050 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001051 if (!ret) {
1052 return false;
1053 }
Zhi Huangcf990f52017-09-22 12:12:30 -07001054
1055 // If SRTP was not required, but we're setting a description that uses SDES,
1056 // we need to upgrade to an SrtpTransport.
1057 if (!srtp_transport_ && !dtls && !cryptos.empty()) {
1058 EnableSrtpTransport_n();
1059 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001060
1061 bool encrypted_header_extensions_id_changed =
1062 EncryptedHeaderExtensionIdsChanged(src, encrypted_extension_ids);
1063 CacheEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
1064
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001065 switch (action) {
1066 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001067 // If DTLS is already active on the channel, we could be renegotiating
1068 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001069 if (!dtls) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001070 ret = sdes_negotiator_.SetOffer(cryptos, src);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001071 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001072 break;
1073 case CA_PRANSWER:
1074 // If we're doing DTLS-SRTP, we don't want to update the filter
1075 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001076 if (!dtls) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001077 ret = sdes_negotiator_.SetProvisionalAnswer(cryptos, src);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001078 }
1079 break;
1080 case CA_ANSWER:
1081 // If we're doing DTLS-SRTP, we don't want to update the filter
1082 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001083 if (!dtls) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001084 ret = sdes_negotiator_.SetAnswer(cryptos, src);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001085 }
1086 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001087 default:
1088 break;
1089 }
Zhi Huangcf990f52017-09-22 12:12:30 -07001090
1091 // If setting an SDES answer succeeded, apply the negotiated parameters
1092 // to the SRTP transport.
1093 if ((action == CA_PRANSWER || action == CA_ANSWER) && !dtls && ret) {
1094 if (sdes_negotiator_.send_cipher_suite() &&
1095 sdes_negotiator_.recv_cipher_suite()) {
Zhi Huangc99b6c72017-11-10 16:44:46 -08001096 RTC_DCHECK(catched_send_extension_ids_);
1097 RTC_DCHECK(catched_recv_extension_ids_);
Zhi Huangcf990f52017-09-22 12:12:30 -07001098 ret = srtp_transport_->SetRtpParams(
1099 *(sdes_negotiator_.send_cipher_suite()),
1100 sdes_negotiator_.send_key().data(),
1101 static_cast<int>(sdes_negotiator_.send_key().size()),
Zhi Huangc99b6c72017-11-10 16:44:46 -08001102 *(catched_send_extension_ids_),
Zhi Huangcf990f52017-09-22 12:12:30 -07001103 *(sdes_negotiator_.recv_cipher_suite()),
1104 sdes_negotiator_.recv_key().data(),
Zhi Huangc99b6c72017-11-10 16:44:46 -08001105 static_cast<int>(sdes_negotiator_.recv_key().size()),
1106 *(catched_recv_extension_ids_));
Zhi Huangcf990f52017-09-22 12:12:30 -07001107 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001108 RTC_LOG(LS_INFO) << "No crypto keys are provided for SDES.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001109 if (action == CA_ANSWER && srtp_transport_) {
1110 // Explicitly reset the |srtp_transport_| if no crypto param is
1111 // provided in the answer. No need to call |ResetParams()| for
1112 // |sdes_negotiator_| because it resets the params inside |SetAnswer|.
1113 srtp_transport_->ResetParams();
1114 }
1115 }
1116 }
1117
Zhi Huangc99b6c72017-11-10 16:44:46 -08001118 // Only update SRTP transport if using DTLS. SDES is handled internally
jbauch5869f502017-06-29 12:31:36 -07001119 // by the SRTP filter.
Zhi Huangcf990f52017-09-22 12:12:30 -07001120 if (ret && dtls_active() && rtp_dtls_transport_ &&
Zhi Huangc99b6c72017-11-10 16:44:46 -08001121 rtp_dtls_transport_->dtls_state() == DTLS_TRANSPORT_CONNECTED &&
1122 encrypted_header_extensions_id_changed) {
1123 ret = SetupDtlsSrtp_n(/*rtcp=*/false);
jbauch5869f502017-06-29 12:31:36 -07001124 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001125
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001126 if (!ret) {
Zhi Huangc99b6c72017-11-10 16:44:46 -08001127 SafeSetError("Failed to setup SRTP.", error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001128 return false;
1129 }
1130 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001131}
1132
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001133bool BaseChannel::SetRtcpMux_n(bool enable,
1134 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001135 ContentSource src,
1136 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -08001137 // Provide a more specific error message for the RTCP mux "require" policy
1138 // case.
zstein56162b92017-04-24 16:54:35 -07001139 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -08001140 SafeSetError(
1141 "rtcpMuxPolicy is 'require', but media description does not "
1142 "contain 'a=rtcp-mux'.",
1143 error_desc);
1144 return false;
1145 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001146 bool ret = false;
1147 switch (action) {
1148 case CA_OFFER:
1149 ret = rtcp_mux_filter_.SetOffer(enable, src);
1150 break;
1151 case CA_PRANSWER:
zhihuangb2cdd932017-01-19 16:54:25 -08001152 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001153 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001154 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1155 break;
1156 case CA_ANSWER:
1157 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1158 if (ret && rtcp_mux_filter_.IsActive()) {
deadbeefe814a0d2017-02-25 18:15:09 -08001159 // We permanently activated RTCP muxing; signal that we no longer need
1160 // the RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -07001161 std::string debug_name =
1162 transport_name_.empty()
zsteine8ab5432017-07-12 11:48:11 -07001163 ? rtp_transport_->rtp_packet_transport()->debug_name()
zsteind48dbda2017-04-04 19:45:57 -07001164 : transport_name_;
Mirko Bonadei675513b2017-11-09 11:09:25 +01001165 RTC_LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1166 << "; no longer need RTCP transport for "
1167 << debug_name;
zsteine8ab5432017-07-12 11:48:11 -07001168 if (rtp_transport_->rtcp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -08001169 SetTransport_n(true, nullptr, nullptr);
1170 SignalRtcpMuxFullyActive(transport_name_);
zhihuangf5b251b2017-01-12 19:37:48 -08001171 }
deadbeef062ce9f2016-08-26 21:42:15 -07001172 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001173 }
1174 break;
1175 case CA_UPDATE:
1176 // No RTCP mux info.
1177 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001178 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001179 default:
1180 break;
1181 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001182 if (!ret) {
1183 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1184 return false;
1185 }
zsteine8ab5432017-07-12 11:48:11 -07001186 rtp_transport_->SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001187 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
zhihuangb2cdd932017-01-19 16:54:25 -08001188 // CA_ANSWER, but we only want to tear down the RTCP transport if we received
1189 // a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001190 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001191 // If the RTP transport is already writable, then so are we.
zsteine8ab5432017-07-12 11:48:11 -07001192 if (rtp_transport_->rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001193 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001194 }
1195 }
1196
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001197 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001198}
1199
1200bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001201 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001202 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001203}
1204
Peter Boström0c4e06b2015-10-07 12:23:21 +02001205bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001206 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001207 return media_channel()->RemoveRecvStream(ssrc);
1208}
1209
1210bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001211 ContentAction action,
1212 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001213 if (!(action == CA_OFFER || action == CA_ANSWER ||
1214 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001215 return false;
1216
1217 // If this is an update, streams only contain streams that have changed.
1218 if (action == CA_UPDATE) {
1219 for (StreamParamsVec::const_iterator it = streams.begin();
1220 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001221 const StreamParams* existing_stream =
1222 GetStreamByIds(local_streams_, it->groupid, it->id);
1223 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001224 if (media_channel()->AddSendStream(*it)) {
1225 local_streams_.push_back(*it);
Mirko Bonadei675513b2017-11-09 11:09:25 +01001226 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001227 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001228 std::ostringstream desc;
1229 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1230 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001231 return false;
1232 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001233 } else if (existing_stream && !it->has_ssrcs()) {
1234 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001235 std::ostringstream desc;
1236 desc << "Failed to remove send stream with ssrc "
1237 << it->first_ssrc() << ".";
1238 SafeSetError(desc.str(), error_desc);
1239 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001240 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001241 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001242 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001243 RTC_LOG(LS_WARNING) << "Ignore unsupported stream update";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001244 }
1245 }
1246 return true;
1247 }
1248 // Else streams are all the streams we want to send.
1249
1250 // Check for streams that have been removed.
1251 bool ret = true;
1252 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1253 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001254 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001255 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001256 std::ostringstream desc;
1257 desc << "Failed to remove send stream with ssrc "
1258 << it->first_ssrc() << ".";
1259 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001260 ret = false;
1261 }
1262 }
1263 }
1264 // Check for new streams.
1265 for (StreamParamsVec::const_iterator it = streams.begin();
1266 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001267 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001268 if (media_channel()->AddSendStream(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001269 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001270 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001271 std::ostringstream desc;
1272 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1273 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001274 ret = false;
1275 }
1276 }
1277 }
1278 local_streams_ = streams;
1279 return ret;
1280}
1281
1282bool BaseChannel::UpdateRemoteStreams_w(
1283 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001284 ContentAction action,
1285 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001286 if (!(action == CA_OFFER || action == CA_ANSWER ||
1287 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001288 return false;
1289
1290 // If this is an update, streams only contain streams that have changed.
1291 if (action == CA_UPDATE) {
1292 for (StreamParamsVec::const_iterator it = streams.begin();
1293 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001294 const StreamParams* existing_stream =
1295 GetStreamByIds(remote_streams_, it->groupid, it->id);
1296 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001297 if (AddRecvStream_w(*it)) {
1298 remote_streams_.push_back(*it);
Mirko Bonadei675513b2017-11-09 11:09:25 +01001299 RTC_LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001300 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001301 std::ostringstream desc;
1302 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1303 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001304 return false;
1305 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001306 } else if (existing_stream && !it->has_ssrcs()) {
1307 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001308 std::ostringstream desc;
1309 desc << "Failed to remove remote stream with ssrc "
1310 << it->first_ssrc() << ".";
1311 SafeSetError(desc.str(), error_desc);
1312 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001313 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001314 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001315 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001316 RTC_LOG(LS_WARNING)
1317 << "Ignore unsupported stream update."
1318 << " Stream exists? " << (existing_stream != nullptr)
1319 << " new stream = " << it->ToString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001320 }
1321 }
1322 return true;
1323 }
1324 // Else streams are all the streams we want to receive.
1325
1326 // Check for streams that have been removed.
1327 bool ret = true;
1328 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1329 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001330 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001331 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001332 std::ostringstream desc;
1333 desc << "Failed to remove remote stream with ssrc "
1334 << it->first_ssrc() << ".";
1335 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001336 ret = false;
1337 }
1338 }
1339 }
1340 // Check for new streams.
1341 for (StreamParamsVec::const_iterator it = streams.begin();
1342 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001343 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001344 if (AddRecvStream_w(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001345 RTC_LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001346 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001347 std::ostringstream desc;
1348 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1349 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001350 ret = false;
1351 }
1352 }
1353 }
1354 remote_streams_ = streams;
1355 return ret;
1356}
1357
jbauch5869f502017-06-29 12:31:36 -07001358RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1359 const RtpHeaderExtensions& extensions) {
1360 if (!rtp_dtls_transport_ ||
1361 !rtp_dtls_transport_->crypto_options()
1362 .enable_encrypted_rtp_header_extensions) {
1363 RtpHeaderExtensions filtered;
1364 auto pred = [](const webrtc::RtpExtension& extension) {
1365 return !extension.encrypt;
1366 };
1367 std::copy_if(extensions.begin(), extensions.end(),
1368 std::back_inserter(filtered), pred);
1369 return filtered;
1370 }
1371
1372 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1373}
1374
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001375void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001376 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001377// Absolute Send Time extension id is used only with external auth,
1378// so do not bother searching for it and making asyncronious call to set
1379// something that is not used.
1380#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001381 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001382 webrtc::RtpExtension::FindHeaderExtensionByUri(
1383 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001384 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001385 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001386 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001387 RTC_FROM_HERE, network_thread_,
1388 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1389 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001390#endif
1391}
1392
1393void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1394 int rtp_abs_sendtime_extn_id) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001395 if (srtp_transport_) {
1396 srtp_transport_->CacheRtpAbsSendTimeHeaderExtension(
1397 rtp_abs_sendtime_extn_id);
1398 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001399 RTC_LOG(LS_WARNING)
1400 << "Trying to cache the Absolute Send Time extension id "
1401 "but the SRTP is not active.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001402 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001403}
1404
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001405void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001406 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001407 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001408 case MSG_SEND_RTP_PACKET:
1409 case MSG_SEND_RTCP_PACKET: {
1410 RTC_DCHECK(network_thread_->IsCurrent());
1411 SendPacketMessageData* data =
1412 static_cast<SendPacketMessageData*>(pmsg->pdata);
1413 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1414 SendPacket(rtcp, &data->packet, data->options);
1415 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001416 break;
1417 }
1418 case MSG_FIRSTPACKETRECEIVED: {
1419 SignalFirstPacketReceived(this);
1420 break;
1421 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001422 }
1423}
1424
zstein3dcf0e92017-06-01 13:22:42 -07001425void BaseChannel::AddHandledPayloadType(int payload_type) {
zsteine8ab5432017-07-12 11:48:11 -07001426 rtp_transport_->AddHandledPayloadType(payload_type);
zstein3dcf0e92017-06-01 13:22:42 -07001427}
1428
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001429void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001430 // Flush all remaining RTCP messages. This should only be called in
1431 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001432 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001433 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001434 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1435 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001436 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1437 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001438 }
1439}
1440
johand89ab142016-10-25 10:50:32 -07001441void BaseChannel::SignalSentPacket_n(
deadbeef5bd5ca32017-02-10 11:31:50 -08001442 rtc::PacketTransportInternal* /* transport */,
johand89ab142016-10-25 10:50:32 -07001443 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001444 RTC_DCHECK(network_thread_->IsCurrent());
1445 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001446 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001447 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1448}
1449
1450void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1451 RTC_DCHECK(worker_thread_->IsCurrent());
1452 SignalSentPacket(sent_packet);
1453}
1454
Zhi Huangc99b6c72017-11-10 16:44:46 -08001455void BaseChannel::CacheEncryptedHeaderExtensionIds(
1456 cricket::ContentSource source,
1457 const std::vector<int>& extension_ids) {
1458 source == ContentSource::CS_LOCAL
1459 ? catched_recv_extension_ids_.emplace(extension_ids)
1460 : catched_send_extension_ids_.emplace(extension_ids);
1461}
1462
1463bool BaseChannel::EncryptedHeaderExtensionIdsChanged(
1464 cricket::ContentSource source,
1465 const std::vector<int>& new_extension_ids) {
1466 if (source == ContentSource::CS_LOCAL) {
1467 return !catched_recv_extension_ids_ ||
1468 (*catched_recv_extension_ids_) != new_extension_ids;
1469 } else {
1470 return !catched_send_extension_ids_ ||
1471 (*catched_send_extension_ids_) != new_extension_ids;
1472 }
1473}
1474
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001475VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1476 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001477 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001478 MediaEngineInterface* media_engine,
Steve Anton8699a322017-11-06 15:53:33 -08001479 std::unique_ptr<VoiceMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001480 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001481 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001482 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001483 : BaseChannel(worker_thread,
1484 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001485 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001486 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001487 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001488 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001489 srtp_required),
Steve Anton8699a322017-11-06 15:53:33 -08001490 media_engine_(media_engine) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001491
1492VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001493 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001494 StopAudioMonitor();
1495 StopMediaMonitor();
1496 // this can't be done in the base class, since it calls a virtual
1497 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001498 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001499}
1500
Peter Boström0c4e06b2015-10-07 12:23:21 +02001501bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001502 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001503 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001504 AudioSource* source) {
stefanf79ade12017-06-02 06:44:03 -07001505 return InvokeOnWorker<bool>(
1506 RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
1507 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001508}
1509
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001510// TODO(juberti): Handle early media the right way. We should get an explicit
1511// ringing message telling us to start playing local ringback, which we cancel
1512// if any early media actually arrives. For now, we do the opposite, which is
1513// to wait 1 second for early media, and start playing local ringback if none
1514// arrives.
1515void VoiceChannel::SetEarlyMedia(bool enable) {
1516 if (enable) {
1517 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001518 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1519 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001520 } else {
1521 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001522 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001523 }
1524}
1525
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001526bool VoiceChannel::CanInsertDtmf() {
stefanf79ade12017-06-02 06:44:03 -07001527 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001528 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001529}
1530
Peter Boström0c4e06b2015-10-07 12:23:21 +02001531bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1532 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001533 int duration) {
stefanf79ade12017-06-02 06:44:03 -07001534 return InvokeOnWorker<bool>(
1535 RTC_FROM_HERE,
1536 Bind(&VoiceChannel::InsertDtmf_w, this, ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001537}
1538
solenberg4bac9c52015-10-09 02:32:53 -07001539bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
stefanf79ade12017-06-02 06:44:03 -07001540 return InvokeOnWorker<bool>(
1541 RTC_FROM_HERE,
1542 Bind(&VoiceMediaChannel::SetOutputVolume, media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001543}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001544
Tommif888bb52015-12-12 01:37:01 +01001545void VoiceChannel::SetRawAudioSink(
1546 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001547 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1548 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001549 // passing. So we invoke to our own little routine that gets a pointer to
1550 // our local variable. This is OK since we're synchronously invoking.
stefanf79ade12017-06-02 06:44:03 -07001551 InvokeOnWorker<bool>(RTC_FROM_HERE,
1552 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001553}
1554
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001555webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001556 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001557 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001558}
1559
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001560webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1561 uint32_t ssrc) const {
1562 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001563}
1564
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001565bool VoiceChannel::SetRtpSendParameters(
1566 uint32_t ssrc,
1567 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001568 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001569 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001570 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001571}
1572
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001573bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1574 webrtc::RtpParameters parameters) {
1575 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1576}
1577
1578webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1579 uint32_t ssrc) const {
1580 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001581 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001582 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1583}
1584
1585webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1586 uint32_t ssrc) const {
1587 return media_channel()->GetRtpReceiveParameters(ssrc);
1588}
1589
1590bool VoiceChannel::SetRtpReceiveParameters(
1591 uint32_t ssrc,
1592 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001593 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001594 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001595 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1596}
1597
1598bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1599 webrtc::RtpParameters parameters) {
1600 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001601}
1602
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001603bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001604 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1605 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001606}
1607
hbos8d609f62017-04-10 07:39:05 -07001608std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1609 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
zhihuang38ede132017-06-15 12:52:32 -07001610 RTC_FROM_HERE, Bind(&VoiceChannel::GetSources_w, this, ssrc));
1611}
1612
1613std::vector<webrtc::RtpSource> VoiceChannel::GetSources_w(uint32_t ssrc) const {
1614 RTC_DCHECK(worker_thread()->IsCurrent());
1615 return media_channel()->GetSources(ssrc);
hbos8d609f62017-04-10 07:39:05 -07001616}
1617
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001618void VoiceChannel::StartMediaMonitor(int cms) {
1619 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001620 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001621 media_monitor_->SignalUpdate.connect(
1622 this, &VoiceChannel::OnMediaMonitorUpdate);
1623 media_monitor_->Start(cms);
1624}
1625
1626void VoiceChannel::StopMediaMonitor() {
1627 if (media_monitor_) {
1628 media_monitor_->Stop();
1629 media_monitor_->SignalUpdate.disconnect(this);
1630 media_monitor_.reset();
1631 }
1632}
1633
1634void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001635 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001636 audio_monitor_
1637 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1638 audio_monitor_->Start(cms);
1639}
1640
1641void VoiceChannel::StopAudioMonitor() {
1642 if (audio_monitor_) {
1643 audio_monitor_->Stop();
1644 audio_monitor_.reset();
1645 }
1646}
1647
1648bool VoiceChannel::IsAudioMonitorRunning() const {
1649 return (audio_monitor_.get() != NULL);
1650}
1651
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001652int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001653 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001654}
1655
1656int VoiceChannel::GetOutputLevel_w() {
1657 return media_channel()->GetOutputLevel();
1658}
1659
1660void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1661 media_channel()->GetActiveStreams(actives);
1662}
1663
zstein3dcf0e92017-06-01 13:22:42 -07001664void VoiceChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -07001665 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -07001666 const rtc::PacketTime& packet_time) {
1667 BaseChannel::OnPacketReceived(rtcp, packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001668 // Set a flag when we've received an RTP packet. If we're waiting for early
1669 // media, this will disable the timeout.
zstein3dcf0e92017-06-01 13:22:42 -07001670 if (!received_media_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001671 received_media_ = true;
1672 }
1673}
1674
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001675void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001676 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001677 invoker_.AsyncInvoke<void>(
1678 RTC_FROM_HERE, worker_thread_,
1679 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001680}
1681
michaelt79e05882016-11-08 02:50:09 -08001682int BaseChannel::GetTransportOverheadPerPacket() const {
1683 RTC_DCHECK(network_thread_->IsCurrent());
1684
1685 if (!selected_candidate_pair_)
1686 return 0;
1687
1688 int transport_overhead_per_packet = 0;
1689
1690 constexpr int kIpv4Overhaed = 20;
1691 constexpr int kIpv6Overhaed = 40;
1692 transport_overhead_per_packet +=
1693 selected_candidate_pair_->local_candidate().address().family() == AF_INET
1694 ? kIpv4Overhaed
1695 : kIpv6Overhaed;
1696
1697 constexpr int kUdpOverhaed = 8;
1698 constexpr int kTcpOverhaed = 20;
1699 transport_overhead_per_packet +=
1700 selected_candidate_pair_->local_candidate().protocol() ==
1701 TCP_PROTOCOL_NAME
1702 ? kTcpOverhaed
1703 : kUdpOverhaed;
1704
Zhi Huang04eaa152017-10-04 14:08:30 -07001705 if (srtp_active()) {
michaelt79e05882016-11-08 02:50:09 -08001706 int srtp_overhead = 0;
Zhi Huangcf990f52017-09-22 12:12:30 -07001707 if (srtp_transport_->GetSrtpOverhead(&srtp_overhead))
michaelt79e05882016-11-08 02:50:09 -08001708 transport_overhead_per_packet += srtp_overhead;
1709 }
1710
1711 return transport_overhead_per_packet;
1712}
1713
1714void BaseChannel::UpdateTransportOverhead() {
1715 int transport_overhead_per_packet = GetTransportOverheadPerPacket();
1716 if (transport_overhead_per_packet)
1717 invoker_.AsyncInvoke<void>(
1718 RTC_FROM_HERE, worker_thread_,
Steve Anton8699a322017-11-06 15:53:33 -08001719 Bind(&MediaChannel::OnTransportOverheadChanged, media_channel_.get(),
michaelt79e05882016-11-08 02:50:09 -08001720 transport_overhead_per_packet));
1721}
1722
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001723void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001724 // Render incoming data if we're the active call, and we have the local
1725 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001726 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001727 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001728
1729 // Send outgoing data if we're the active call, we have the remote content,
1730 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001731 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001732 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001733
Mirko Bonadei675513b2017-11-09 11:09:25 +01001734 RTC_LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001735}
1736
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001737bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001738 ContentAction action,
1739 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001740 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001741 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001742 RTC_LOG(LS_INFO) << "Setting local voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001743
1744 const AudioContentDescription* audio =
1745 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001746 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001747 if (!audio) {
1748 SafeSetError("Can't find audio content in local description.", error_desc);
1749 return false;
1750 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001751
jbauch5869f502017-06-29 12:31:36 -07001752 RtpHeaderExtensions rtp_header_extensions =
1753 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1754
1755 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1756 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001757 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001758 }
1759
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001760 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001761 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001762 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001763 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001764 error_desc);
1765 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001766 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001767 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001768 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001769 }
1770 last_recv_params_ = recv_params;
1771
1772 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1773 // only give it to the media channel once we have a remote
1774 // description too (without a remote description, we won't be able
1775 // to send them anyway).
1776 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1777 SafeSetError("Failed to set local audio description streams.", error_desc);
1778 return false;
1779 }
1780
1781 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001782 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001783 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001784}
1785
1786bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001787 ContentAction action,
1788 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001789 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001790 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001791 RTC_LOG(LS_INFO) << "Setting remote voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001792
1793 const AudioContentDescription* audio =
1794 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001795 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001796 if (!audio) {
1797 SafeSetError("Can't find audio content in remote description.", error_desc);
1798 return false;
1799 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001800
jbauch5869f502017-06-29 12:31:36 -07001801 RtpHeaderExtensions rtp_header_extensions =
1802 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1803
1804 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
1805 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001806 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001807 }
1808
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001809 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001810 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1811 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001812 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001813 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001814 }
skvladdc1c62c2016-03-16 19:07:43 -07001815
1816 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1817 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001818 SafeSetError("Failed to set remote audio description send parameters.",
1819 error_desc);
1820 return false;
1821 }
1822 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001823
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001824 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1825 // and only give it to the media channel once we have a local
1826 // description too (without a local description, we won't be able to
1827 // recv them anyway).
1828 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1829 SafeSetError("Failed to set remote audio description streams.", error_desc);
1830 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001831 }
1832
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001833 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001834 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001835 }
1836
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001837 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001838 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001839 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001840}
1841
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001842void VoiceChannel::HandleEarlyMediaTimeout() {
1843 // This occurs on the main thread, not the worker thread.
1844 if (!received_media_) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001845 RTC_LOG(LS_INFO) << "No early media received before timeout";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001846 SignalEarlyMediaTimeout(this);
1847 }
1848}
1849
Peter Boström0c4e06b2015-10-07 12:23:21 +02001850bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1851 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001852 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001853 if (!enabled()) {
1854 return false;
1855 }
solenberg1d63dd02015-12-02 12:35:09 -08001856 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001857}
1858
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001859void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001860 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001861 case MSG_EARLYMEDIATIMEOUT:
1862 HandleEarlyMediaTimeout();
1863 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001864 case MSG_CHANNEL_ERROR: {
1865 VoiceChannelErrorMessageData* data =
1866 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001867 delete data;
1868 break;
1869 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001870 default:
1871 BaseChannel::OnMessage(pmsg);
1872 break;
1873 }
1874}
1875
1876void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001877 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001878 SignalConnectionMonitor(this, infos);
1879}
1880
1881void VoiceChannel::OnMediaMonitorUpdate(
1882 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001883 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001884 SignalMediaMonitor(this, info);
1885}
1886
1887void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1888 const AudioInfo& info) {
1889 SignalAudioMonitor(this, info);
1890}
1891
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001892VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1893 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001894 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001895 std::unique_ptr<VideoMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001896 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001897 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001898 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001899 : BaseChannel(worker_thread,
1900 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001901 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001902 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001903 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001904 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001905 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001906
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001907VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001908 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001909 StopMediaMonitor();
1910 // this can't be done in the base class, since it calls a virtual
1911 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001912
1913 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001914}
1915
nisse08582ff2016-02-04 01:24:52 -08001916bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001917 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001918 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001919 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001920 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001921 return true;
1922}
1923
deadbeef5a4a75a2016-06-02 16:23:38 -07001924bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001925 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001926 bool mute,
1927 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001928 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
stefanf79ade12017-06-02 06:44:03 -07001929 return InvokeOnWorker<bool>(
1930 RTC_FROM_HERE, Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
1931 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001932}
1933
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001934webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001935 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001936 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001937}
1938
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001939webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1940 uint32_t ssrc) const {
1941 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001942}
1943
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001944bool VideoChannel::SetRtpSendParameters(
1945 uint32_t ssrc,
1946 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001947 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001948 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001949 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001950}
1951
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001952bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1953 webrtc::RtpParameters parameters) {
1954 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1955}
1956
1957webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1958 uint32_t ssrc) const {
1959 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001960 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001961 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1962}
1963
1964webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1965 uint32_t ssrc) const {
1966 return media_channel()->GetRtpReceiveParameters(ssrc);
1967}
1968
1969bool VideoChannel::SetRtpReceiveParameters(
1970 uint32_t ssrc,
1971 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001972 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001973 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001974 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1975}
1976
1977bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1978 webrtc::RtpParameters parameters) {
1979 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001980}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001981
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001982void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001983 // Send outgoing data if we're the active call, we have the remote content,
1984 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001985 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001986 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001987 RTC_LOG(LS_ERROR) << "Failed to SetSend on video channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001988 // TODO(gangji): Report error back to server.
1989 }
1990
Mirko Bonadei675513b2017-11-09 11:09:25 +01001991 RTC_LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001992}
1993
stefanf79ade12017-06-02 06:44:03 -07001994void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
1995 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
1996 media_channel(), bwe_info));
1997}
1998
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00001999bool VideoChannel::GetStats(VideoMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07002000 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
2001 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002002}
2003
2004void VideoChannel::StartMediaMonitor(int cms) {
2005 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002006 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002007 media_monitor_->SignalUpdate.connect(
2008 this, &VideoChannel::OnMediaMonitorUpdate);
2009 media_monitor_->Start(cms);
2010}
2011
2012void VideoChannel::StopMediaMonitor() {
2013 if (media_monitor_) {
2014 media_monitor_->Stop();
2015 media_monitor_.reset();
2016 }
2017}
2018
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002019bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002020 ContentAction action,
2021 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002022 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002023 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01002024 RTC_LOG(LS_INFO) << "Setting local video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002025
2026 const VideoContentDescription* video =
2027 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002028 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002029 if (!video) {
2030 SafeSetError("Can't find video content in local description.", error_desc);
2031 return false;
2032 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002033
jbauch5869f502017-06-29 12:31:36 -07002034 RtpHeaderExtensions rtp_header_extensions =
2035 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
2036
2037 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
2038 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002039 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002040 }
2041
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002042 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07002043 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002044 if (!media_channel()->SetRecvParameters(recv_params)) {
2045 SafeSetError("Failed to set local video description recv parameters.",
2046 error_desc);
2047 return false;
2048 }
2049 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07002050 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002051 }
2052 last_recv_params_ = recv_params;
2053
2054 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2055 // only give it to the media channel once we have a remote
2056 // description too (without a remote description, we won't be able
2057 // to send them anyway).
2058 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2059 SafeSetError("Failed to set local video description streams.", error_desc);
2060 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002061 }
2062
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002063 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002064 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002065 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002066}
2067
2068bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002069 ContentAction action,
2070 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002071 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002072 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01002073 RTC_LOG(LS_INFO) << "Setting remote video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002074
2075 const VideoContentDescription* video =
2076 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002077 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002078 if (!video) {
2079 SafeSetError("Can't find video content in remote description.", error_desc);
2080 return false;
2081 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002082
jbauch5869f502017-06-29 12:31:36 -07002083 RtpHeaderExtensions rtp_header_extensions =
2084 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
2085
2086 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2087 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002088 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002089 }
2090
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002091 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002092 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
2093 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002094 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002095 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002096 }
skvladdc1c62c2016-03-16 19:07:43 -07002097
2098 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2099
2100 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002101 SafeSetError("Failed to set remote video description send parameters.",
2102 error_desc);
2103 return false;
2104 }
2105 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002106
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002107 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2108 // and only give it to the media channel once we have a local
2109 // description too (without a local description, we won't be able to
2110 // recv them anyway).
2111 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2112 SafeSetError("Failed to set remote video description streams.", error_desc);
2113 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002114 }
2115
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002116 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07002117 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002118 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002119
2120 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002121 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002122 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002123}
2124
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002125void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002126 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002127 case MSG_CHANNEL_ERROR: {
2128 const VideoChannelErrorMessageData* data =
2129 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002130 delete data;
2131 break;
2132 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002133 default:
2134 BaseChannel::OnMessage(pmsg);
2135 break;
2136 }
2137}
2138
2139void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002140 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002141 SignalConnectionMonitor(this, infos);
2142}
2143
2144// TODO(pthatcher): Look into removing duplicate code between
2145// audio, video, and data, perhaps by using templates.
2146void VideoChannel::OnMediaMonitorUpdate(
2147 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002148 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002149 SignalMediaMonitor(this, info);
2150}
2151
deadbeef953c2ce2017-01-09 14:53:41 -08002152RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2153 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002154 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08002155 std::unique_ptr<DataMediaChannel> media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002156 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002157 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002158 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002159 : BaseChannel(worker_thread,
2160 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002161 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08002162 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07002163 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002164 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002165 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002166
deadbeef953c2ce2017-01-09 14:53:41 -08002167RtpDataChannel::~RtpDataChannel() {
2168 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002169 StopMediaMonitor();
2170 // this can't be done in the base class, since it calls a virtual
2171 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002172
2173 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002174}
2175
Steve Anton8699a322017-11-06 15:53:33 -08002176void RtpDataChannel::Init_w(
deadbeeff5346592017-01-24 21:51:21 -08002177 DtlsTransportInternal* rtp_dtls_transport,
2178 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08002179 rtc::PacketTransportInternal* rtp_packet_transport,
2180 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -08002181 BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
2182 rtp_packet_transport, rtcp_packet_transport);
2183
deadbeef953c2ce2017-01-09 14:53:41 -08002184 media_channel()->SignalDataReceived.connect(this,
2185 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002186 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002187 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002188}
2189
deadbeef953c2ce2017-01-09 14:53:41 -08002190bool RtpDataChannel::SendData(const SendDataParams& params,
2191 const rtc::CopyOnWriteBuffer& payload,
2192 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07002193 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002194 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2195 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002196}
2197
deadbeef953c2ce2017-01-09 14:53:41 -08002198bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002199 const DataContentDescription* content,
2200 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002201 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2202 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002203 // It's been set before, but doesn't match. That's bad.
2204 if (is_sctp) {
2205 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2206 error_desc);
2207 return false;
2208 }
2209 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002210}
2211
deadbeef953c2ce2017-01-09 14:53:41 -08002212bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2213 ContentAction action,
2214 std::string* error_desc) {
2215 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002216 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01002217 RTC_LOG(LS_INFO) << "Setting local data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002218
2219 const DataContentDescription* data =
2220 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002221 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002222 if (!data) {
2223 SafeSetError("Can't find data content in local description.", error_desc);
2224 return false;
2225 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002226
deadbeef953c2ce2017-01-09 14:53:41 -08002227 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002228 return false;
2229 }
2230
jbauch5869f502017-06-29 12:31:36 -07002231 RtpHeaderExtensions rtp_header_extensions =
2232 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2233
2234 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
2235 rtp_header_extensions, error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08002236 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002237 }
2238
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002239 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07002240 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002241 if (!media_channel()->SetRecvParameters(recv_params)) {
2242 SafeSetError("Failed to set remote data description recv parameters.",
2243 error_desc);
2244 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002245 }
deadbeef953c2ce2017-01-09 14:53:41 -08002246 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07002247 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002248 }
2249 last_recv_params_ = recv_params;
2250
2251 // TODO(pthatcher): Move local streams into DataSendParameters, and
2252 // only give it to the media channel once we have a remote
2253 // description too (without a remote description, we won't be able
2254 // to send them anyway).
2255 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2256 SafeSetError("Failed to set local data description streams.", error_desc);
2257 return false;
2258 }
2259
2260 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002261 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002262 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002263}
2264
deadbeef953c2ce2017-01-09 14:53:41 -08002265bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2266 ContentAction action,
2267 std::string* error_desc) {
2268 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002269 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002270
2271 const DataContentDescription* data =
2272 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002273 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002274 if (!data) {
2275 SafeSetError("Can't find data content in remote description.", error_desc);
2276 return false;
2277 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002278
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002279 // If the remote data doesn't have codecs and isn't an update, it
2280 // must be empty, so ignore it.
2281 if (!data->has_codecs() && action != CA_UPDATE) {
2282 return true;
2283 }
2284
deadbeef953c2ce2017-01-09 14:53:41 -08002285 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002286 return false;
2287 }
2288
jbauch5869f502017-06-29 12:31:36 -07002289 RtpHeaderExtensions rtp_header_extensions =
2290 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2291
Mirko Bonadei675513b2017-11-09 11:09:25 +01002292 RTC_LOG(LS_INFO) << "Setting remote data description";
jbauch5869f502017-06-29 12:31:36 -07002293 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2294 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002295 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002296 }
2297
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002298 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002299 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
2300 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002301 if (!media_channel()->SetSendParameters(send_params)) {
2302 SafeSetError("Failed to set remote data description send parameters.",
2303 error_desc);
2304 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002305 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002306 last_send_params_ = send_params;
2307
2308 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2309 // and only give it to the media channel once we have a local
2310 // description too (without a local description, we won't be able to
2311 // recv them anyway).
2312 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2313 SafeSetError("Failed to set remote data description streams.",
2314 error_desc);
2315 return false;
2316 }
2317
2318 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002319 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002320 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002321}
2322
deadbeef953c2ce2017-01-09 14:53:41 -08002323void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002324 // Render incoming data if we're the active call, and we have the local
2325 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002326 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002327 if (!media_channel()->SetReceive(recv)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002328 RTC_LOG(LS_ERROR) << "Failed to SetReceive on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002329 }
2330
2331 // Send outgoing data if we're the active call, we have the remote content,
2332 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002333 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002334 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002335 RTC_LOG(LS_ERROR) << "Failed to SetSend on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002336 }
2337
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002338 // Trigger SignalReadyToSendData asynchronously.
2339 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002340
Mirko Bonadei675513b2017-11-09 11:09:25 +01002341 RTC_LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002342}
2343
deadbeef953c2ce2017-01-09 14:53:41 -08002344void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002345 switch (pmsg->message_id) {
2346 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002347 DataChannelReadyToSendMessageData* data =
2348 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002349 ready_to_send_data_ = data->data();
2350 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002351 delete data;
2352 break;
2353 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002354 case MSG_DATARECEIVED: {
2355 DataReceivedMessageData* data =
2356 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002357 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002358 delete data;
2359 break;
2360 }
2361 case MSG_CHANNEL_ERROR: {
2362 const DataChannelErrorMessageData* data =
2363 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002364 delete data;
2365 break;
2366 }
2367 default:
2368 BaseChannel::OnMessage(pmsg);
2369 break;
2370 }
2371}
2372
deadbeef953c2ce2017-01-09 14:53:41 -08002373void RtpDataChannel::OnConnectionMonitorUpdate(
2374 ConnectionMonitor* monitor,
2375 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002376 SignalConnectionMonitor(this, infos);
2377}
2378
deadbeef953c2ce2017-01-09 14:53:41 -08002379void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002380 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002381 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002382 media_monitor_->SignalUpdate.connect(this,
2383 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002384 media_monitor_->Start(cms);
2385}
2386
deadbeef953c2ce2017-01-09 14:53:41 -08002387void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002388 if (media_monitor_) {
2389 media_monitor_->Stop();
2390 media_monitor_->SignalUpdate.disconnect(this);
2391 media_monitor_.reset();
2392 }
2393}
2394
deadbeef953c2ce2017-01-09 14:53:41 -08002395void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2396 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002397 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002398 SignalMediaMonitor(this, info);
2399}
2400
deadbeef953c2ce2017-01-09 14:53:41 -08002401void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2402 const char* data,
2403 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002404 DataReceivedMessageData* msg = new DataReceivedMessageData(
2405 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002406 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002407}
2408
deadbeef953c2ce2017-01-09 14:53:41 -08002409void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2410 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002411 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2412 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002413 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002414}
2415
deadbeef953c2ce2017-01-09 14:53:41 -08002416void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002417 // This is usded for congestion control to indicate that the stream is ready
2418 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2419 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002420 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002421 new DataChannelReadyToSendMessageData(writable));
2422}
2423
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002424} // namespace cricket