blob: 283c624059eb57fce8de765cc27a5fd0b4234ca8 [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"
Steve Anton4e70a722017-11-28 14:57:10 -080034#include "pc/rtpmediautils.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000035
36namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000037using rtc::Bind;
Steve Anton3828c062017-12-06 10:34:51 -080038using webrtc::SdpType;
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
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000066static void SafeSetError(const std::string& message, std::string* error_desc) {
67 if (error_desc) {
68 *error_desc = message;
69 }
70}
71
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000072struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020073 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000074 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020075 : ssrc(in_ssrc), error(in_error) {}
76 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000077 VoiceMediaChannel::Error error;
78};
79
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000080struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020081 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000082 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020083 : ssrc(in_ssrc), error(in_error) {}
84 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000085 VideoMediaChannel::Error error;
86};
87
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000088struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020089 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000090 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020091 : ssrc(in_ssrc), error(in_error) {}
92 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000093 DataMediaChannel::Error error;
94};
95
jbaucheec21bd2016-03-20 06:15:43 -070096static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000097 // Check the packet size. We could check the header too if needed.
zstein3dcf0e92017-06-01 13:22:42 -070098 return packet && IsValidRtpRtcpPacketSize(rtcp, packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000099}
100
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700101template <class Codec>
102void RtpParametersFromMediaDescription(
103 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700104 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700105 RtpParameters<Codec>* params) {
106 // TODO(pthatcher): Remove this once we're sure no one will give us
Zhi Huang801b8682017-11-15 11:36:43 -0800107 // a description without codecs. Currently the ORTC implementation is relying
108 // on this.
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700109 if (desc->has_codecs()) {
110 params->codecs = desc->codecs();
111 }
112 // TODO(pthatcher): See if we really need
113 // rtp_header_extensions_set() and remove it if we don't.
114 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -0700115 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700116 }
deadbeef13871492015-12-09 12:37:51 -0800117 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700118}
119
nisse05103312016-03-16 02:22:50 -0700120template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700121void RtpSendParametersFromMediaDescription(
122 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700123 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -0700124 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -0700125 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700126 send_params->max_bandwidth_bps = desc->bandwidth();
127}
128
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200129BaseChannel::BaseChannel(rtc::Thread* worker_thread,
130 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800131 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -0800132 std::unique_ptr<MediaChannel> media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700133 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800134 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800135 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200136 : worker_thread_(worker_thread),
137 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800138 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000139 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700140 rtcp_mux_required_(rtcp_mux_required),
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800141 unencrypted_rtp_transport_(
142 rtc::MakeUnique<webrtc::RtpTransport>(rtcp_mux_required)),
deadbeef7af91dd2016-12-13 11:29:11 -0800143 srtp_required_(srtp_required),
Zhi Huang1d88d742017-11-15 15:58:49 -0800144 media_channel_(std::move(media_channel)) {
Steve Anton8699a322017-11-06 15:53:33 -0800145 RTC_DCHECK_RUN_ON(worker_thread_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800146 rtp_transport_ = unencrypted_rtp_transport_.get();
147 ConnectToRtpTransport();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100148 RTC_LOG(LS_INFO) << "Created channel for " << content_name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000149}
150
151BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800152 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Steve Anton8699a322017-11-06 15:53:33 -0800153 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.org78187522013-10-07 23:32:02 +0000154 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000155 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200156 // Eats any outstanding messages or packets.
157 worker_thread_->Clear(&invoker_);
158 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000159 // We must destroy the media channel before the transport channel, otherwise
160 // the media channel may try to send on the dead transport channel. NULLing
161 // is not an effective strategy since the sends will come on another thread.
Steve Anton8699a322017-11-06 15:53:33 -0800162 media_channel_.reset();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100163 RTC_LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200164}
165
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800166void BaseChannel::ConnectToRtpTransport() {
167 RTC_DCHECK(rtp_transport_);
168 rtp_transport_->SignalReadyToSend.connect(
169 this, &BaseChannel::OnTransportReadyToSend);
170 // TODO(zstein): RtpTransport::SignalPacketReceived will probably be replaced
171 // with a callback interface later so that the demuxer can select which
172 // channel to signal.
173 rtp_transport_->SignalPacketReceived.connect(this,
174 &BaseChannel::OnPacketReceived);
175 rtp_transport_->SignalNetworkRouteChanged.connect(
176 this, &BaseChannel::OnNetworkRouteChanged);
177 rtp_transport_->SignalWritableState.connect(this,
178 &BaseChannel::OnWritableState);
179 rtp_transport_->SignalSentPacket.connect(this,
180 &BaseChannel::SignalSentPacket_n);
181}
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200182
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800183void BaseChannel::DisconnectFromRtpTransport() {
184 RTC_DCHECK(rtp_transport_);
185 rtp_transport_->SignalReadyToSend.disconnect(this);
186 rtp_transport_->SignalPacketReceived.disconnect(this);
187 rtp_transport_->SignalNetworkRouteChanged.disconnect(this);
188 rtp_transport_->SignalWritableState.disconnect(this);
189 rtp_transport_->SignalSentPacket.disconnect(this);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200190}
191
Steve Anton8699a322017-11-06 15:53:33 -0800192void BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800193 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800194 rtc::PacketTransportInternal* rtp_packet_transport,
195 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -0800196 RTC_DCHECK_RUN_ON(worker_thread_);
197 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800198 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport,
199 rtp_packet_transport, rtcp_packet_transport);
200
201 if (rtcp_mux_required_) {
202 rtcp_mux_filter_.SetActive();
203 }
Steve Anton8699a322017-11-06 15:53:33 -0800204 });
205
deadbeeff5346592017-01-24 21:51:21 -0800206 // Both RTP and RTCP channels should be set, we can call SetInterface on
207 // the media channel and it can set network options.
wu@webrtc.orgde305012013-10-31 15:40:38 +0000208 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000209}
210
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800211void BaseChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
212 RTC_DCHECK_RUN_ON(worker_thread_);
213 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
214 SetRtpTransport(rtp_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200215
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800216 if (rtcp_mux_required_) {
217 rtcp_mux_filter_.SetActive();
218 }
219 });
220
221 // Both RTP and RTCP channels should be set, we can call SetInterface on
222 // the media channel and it can set network options.
223 media_channel_->SetInterface(this);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200224}
225
wu@webrtc.org78187522013-10-07 23:32:02 +0000226void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200227 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000228 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200229 // Packets arrive on the network thread, processing packets calls virtual
230 // functions, so need to stop this process in Deinit that is called in
231 // derived classes destructor.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800232 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
233 FlushRtcpMessages_n();
234
235 if (dtls_srtp_transport_) {
236 dtls_srtp_transport_->SetDtlsTransports(nullptr, nullptr);
237 } else {
238 rtp_transport_->SetRtpPacketTransport(nullptr);
239 rtp_transport_->SetRtcpPacketTransport(nullptr);
240 }
241 // Clear pending read packets/messages.
242 network_thread_->Clear(&invoker_);
243 network_thread_->Clear(this);
244 });
wu@webrtc.org78187522013-10-07 23:32:02 +0000245}
246
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800247void BaseChannel::SetRtpTransport(webrtc::RtpTransportInternal* rtp_transport) {
248 if (!network_thread_->IsCurrent()) {
249 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
250 SetRtpTransport(rtp_transport);
251 return;
252 });
253 }
254
255 RTC_DCHECK(rtp_transport);
256
257 if (rtp_transport_) {
258 DisconnectFromRtpTransport();
259 }
260 rtp_transport_ = rtp_transport;
261 RTC_LOG(LS_INFO) << "Setting the RtpTransport for " << content_name();
262 ConnectToRtpTransport();
263
264 UpdateWritableState_n();
265}
266
zhihuangb2cdd932017-01-19 16:54:25 -0800267void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
268 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800269 network_thread_->Invoke<void>(
270 RTC_FROM_HERE,
271 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
272 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000273}
274
deadbeeff5346592017-01-24 21:51:21 -0800275void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800276 rtc::PacketTransportInternal* rtp_packet_transport,
277 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800278 network_thread_->Invoke<void>(
279 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
280 rtp_packet_transport, rtcp_packet_transport));
281}
zhihuangf5b251b2017-01-12 19:37:48 -0800282
deadbeeff5346592017-01-24 21:51:21 -0800283void BaseChannel::SetTransports_n(
284 DtlsTransportInternal* rtp_dtls_transport,
285 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800286 rtc::PacketTransportInternal* rtp_packet_transport,
287 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800288 RTC_DCHECK(network_thread_->IsCurrent());
289 // Validate some assertions about the input.
290 RTC_DCHECK(rtp_packet_transport);
291 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
292 if (rtp_dtls_transport || rtcp_dtls_transport) {
293 // DTLS/non-DTLS pointers should be to the same object.
294 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
295 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
296 // Can't go from non-DTLS to DTLS.
zsteine8ab5432017-07-12 11:48:11 -0700297 RTC_DCHECK(!rtp_transport_->rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800298 } else {
299 // Can't go from DTLS to non-DTLS.
300 RTC_DCHECK(!rtp_dtls_transport_);
301 }
302 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800303 if (rtp_dtls_transport && rtcp_dtls_transport) {
304 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
305 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800306 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800307
308 if (rtp_packet_transport == rtp_transport_->rtp_packet_transport()) {
309 // Nothing to do if transport isn't changing.
310 return;
311 }
312
deadbeeff5346592017-01-24 21:51:21 -0800313 std::string debug_name;
314 if (rtp_dtls_transport) {
315 transport_name_ = rtp_dtls_transport->transport_name();
316 debug_name = transport_name_;
317 } else {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800318 debug_name = rtp_packet_transport->transport_name();
deadbeeff5346592017-01-24 21:51:21 -0800319 }
deadbeefac22f702017-01-12 21:59:29 -0800320 // If this BaseChannel doesn't require RTCP mux and we haven't fully
321 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800322 if (rtcp_packet_transport) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100323 RTC_LOG(LS_INFO) << "Setting RTCP Transport for " << content_name()
324 << " on " << debug_name << " transport "
325 << rtcp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800326 SetTransport_n(/*rtcp=*/true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000327 }
328
Mirko Bonadei675513b2017-11-09 11:09:25 +0100329 RTC_LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
330 << debug_name << " transport " << rtp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800331 SetTransport_n(/*rtcp=*/false, rtp_dtls_transport, rtp_packet_transport);
332
333 // Set DtlsTransport/PacketTransport for RTP-level transport.
334 if ((rtp_dtls_transport_ || rtcp_dtls_transport_) && dtls_srtp_transport_) {
335 // When setting the transport with non-null |dtls_srtp_transport_|, we are
336 // using DTLS-SRTP. This could happen for bundling. If the
337 // |dtls_srtp_transport| is null, we cannot tell if it doing DTLS-SRTP or
338 // SDES until the description is set. So don't call |EnableDtlsSrtp_n| here.
339 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport,
340 rtcp_dtls_transport);
341 } else {
342 rtp_transport_->SetRtpPacketTransport(rtp_packet_transport);
343 rtp_transport_->SetRtcpPacketTransport(rtcp_packet_transport);
344 }
guoweis46383312015-12-17 16:45:59 -0800345
deadbeefcbecd352015-09-23 11:50:27 -0700346 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700347 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200348 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000349}
350
deadbeeff5346592017-01-24 21:51:21 -0800351void BaseChannel::SetTransport_n(
352 bool rtcp,
353 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800354 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200355 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800356 if (new_dtls_transport) {
357 RTC_DCHECK(new_dtls_transport == new_packet_transport);
358 }
deadbeeff5346592017-01-24 21:51:21 -0800359 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800360 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700361 rtc::PacketTransportInternal* old_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700362 rtcp ? rtp_transport_->rtcp_packet_transport()
363 : rtp_transport_->rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800364
deadbeeff5346592017-01-24 21:51:21 -0800365 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700366 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000367 return;
368 }
zhihuangb2cdd932017-01-19 16:54:25 -0800369
deadbeeff5346592017-01-24 21:51:21 -0800370 RTC_DCHECK(old_packet_transport != new_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000371
deadbeeff5346592017-01-24 21:51:21 -0800372 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000373
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800374 // If there's no new transport, we're done.
deadbeeff5346592017-01-24 21:51:21 -0800375 if (!new_packet_transport) {
376 return;
377 }
378
379 if (rtcp && new_dtls_transport) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700380 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_active()))
381 << "Setting RTCP for DTLS/SRTP after the DTLS is active "
deadbeeff5346592017-01-24 21:51:21 -0800382 << "should never happen.";
383 }
zstein56162b92017-04-24 16:54:35 -0700384
deadbeeff5346592017-01-24 21:51:21 -0800385 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
386 for (const auto& pair : socket_options) {
387 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800388 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000389}
390
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000391bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700392 worker_thread_->Invoke<void>(
393 RTC_FROM_HERE,
394 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
395 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000396 return true;
397}
398
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000399bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700400 return InvokeOnWorker<bool>(RTC_FROM_HERE,
401 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000402}
403
Peter Boström0c4e06b2015-10-07 12:23:21 +0200404bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700405 return InvokeOnWorker<bool>(
406 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000407}
408
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000409bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700410 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700411 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000412}
413
Peter Boström0c4e06b2015-10-07 12:23:21 +0200414bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700415 return InvokeOnWorker<bool>(
416 RTC_FROM_HERE,
417 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000418}
419
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000420bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800421 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000422 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100423 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700424 return InvokeOnWorker<bool>(
425 RTC_FROM_HERE,
Steve Anton3828c062017-12-06 10:34:51 -0800426 Bind(&BaseChannel::SetLocalContent_w, this, content, type, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000427}
428
429bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800430 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000431 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100432 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700433 return InvokeOnWorker<bool>(
Steve Anton3828c062017-12-06 10:34:51 -0800434 RTC_FROM_HERE,
435 Bind(&BaseChannel::SetRemoteContent_w, this, content, type, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000436}
437
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000438void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800439 // We pass in the BaseChannel instead of the rtp_dtls_transport_
440 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000441 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200442 // We pass in the network thread because on that thread connection monitor
443 // will call BaseChannel::GetConnectionStats which must be called on the
444 // network thread.
445 connection_monitor_.reset(
446 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000447 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000448 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000449 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000450}
451
452void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000453 if (connection_monitor_) {
454 connection_monitor_->Stop();
455 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000456 }
457}
458
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000459bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200460 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800461 if (!rtp_dtls_transport_) {
462 return false;
463 }
zhihuangb2cdd932017-01-19 16:54:25 -0800464 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800465}
466
467bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800468 // If this BaseChannel doesn't require RTCP mux and we haven't fully
469 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700470 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000471}
472
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700473bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000474 // Receive data if we are enabled and have local content,
Steve Anton4e70a722017-11-28 14:57:10 -0800475 return enabled() &&
476 webrtc::RtpTransceiverDirectionHasRecv(local_content_direction_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000477}
478
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700479bool BaseChannel::IsReadyToSendMedia_w() const {
480 // Need to access some state updated on the network thread.
481 return network_thread_->Invoke<bool>(
482 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
483}
484
485bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000486 // Send outgoing data if we are enabled, have local and remote content,
487 // and we have had some form of connectivity.
Steve Anton4e70a722017-11-28 14:57:10 -0800488 return enabled() &&
489 webrtc::RtpTransceiverDirectionHasRecv(remote_content_direction_) &&
490 webrtc::RtpTransceiverDirectionHasSend(local_content_direction_) &&
Zhi Huangcf990f52017-09-22 12:12:30 -0700491 was_ever_writable() && (srtp_active() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000492}
493
jbaucheec21bd2016-03-20 06:15:43 -0700494bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700495 const rtc::PacketOptions& options) {
496 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000497}
498
jbaucheec21bd2016-03-20 06:15:43 -0700499bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700500 const rtc::PacketOptions& options) {
501 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000502}
503
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000504int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000505 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200506 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700507 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200508}
509
510int BaseChannel::SetOption_n(SocketType type,
511 rtc::Socket::Option opt,
512 int value) {
513 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800514 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000515 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000516 case ST_RTP:
zsteine8ab5432017-07-12 11:48:11 -0700517 transport = rtp_transport_->rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700518 socket_options_.push_back(
519 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000520 break;
521 case ST_RTCP:
zsteine8ab5432017-07-12 11:48:11 -0700522 transport = rtp_transport_->rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700523 rtcp_socket_options_.push_back(
524 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000525 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000526 }
deadbeeff5346592017-01-24 21:51:21 -0800527 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000528}
529
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800530void BaseChannel::OnWritableState(bool writable) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200531 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800532 if (writable) {
533 // This is used to cover the scenario when the DTLS handshake is completed
534 // and DtlsTransport becomes writable before the remote description is set.
535 if (ShouldSetupDtlsSrtp_n()) {
536 EnableDtlsSrtp_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700537 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800538 ChannelWritable_n();
539 } else {
540 ChannelNotWritable_n();
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800541 }
542}
543
Zhi Huang942bc2e2017-11-13 13:26:07 -0800544void BaseChannel::OnNetworkRouteChanged(
545 rtc::Optional<rtc::NetworkRoute> network_route) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200546 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800547 rtc::NetworkRoute new_route;
548 if (network_route) {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800549 new_route = *(network_route);
Zhi Huang8c316c12017-11-13 21:13:45 +0000550 }
Zhi Huang942bc2e2017-11-13 13:26:07 -0800551 // Note: When the RTCP-muxing is not enabled, RTCP transport and RTP transport
552 // use the same transport name and MediaChannel::OnNetworkRouteChanged cannot
553 // work correctly. Intentionally leave it broken to simplify the code and
554 // encourage the users to stop using non-muxing RTCP.
Steve Anton8699a322017-11-06 15:53:33 -0800555 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_, [=] {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800556 media_channel_->OnNetworkRouteChanged(transport_name_, new_route);
Steve Anton8699a322017-11-06 15:53:33 -0800557 });
Honghai Zhangcc411c02016-03-29 17:27:21 -0700558}
559
zstein56162b92017-04-24 16:54:35 -0700560void BaseChannel::OnTransportReadyToSend(bool ready) {
Steve Anton8699a322017-11-06 15:53:33 -0800561 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_,
562 [=] { media_channel_->OnReadyToSend(ready); });
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000563}
564
stefanc1aeaf02015-10-15 07:26:07 -0700565bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700566 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700567 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200568 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
569 // If the thread is not our network thread, we will post to our network
570 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000571 // synchronize access to all the pieces of the send path, including
572 // SRTP and the inner workings of the transport channels.
573 // The only downside is that we can't return a proper failure code if
574 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200575 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000576 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200577 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
578 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800579 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700580 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700581 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000582 return true;
583 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200584 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000585
586 // Now that we are on the correct thread, ensure we have a place to send this
587 // packet before doing anything. (We might get RTCP packets that we don't
588 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
589 // transport.
zsteine8ab5432017-07-12 11:48:11 -0700590 if (!rtp_transport_->IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000591 return false;
592 }
593
594 // Protect ourselves against crazy data.
595 if (!ValidPacket(rtcp, packet)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100596 RTC_LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
597 << RtpRtcpStringLiteral(rtcp)
598 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000599 return false;
600 }
601
Zhi Huangcf990f52017-09-22 12:12:30 -0700602 if (!srtp_active()) {
603 if (srtp_required_) {
604 // The audio/video engines may attempt to send RTCP packets as soon as the
605 // streams are created, so don't treat this as an error for RTCP.
606 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
607 if (rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000608 return false;
609 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700610 // However, there shouldn't be any RTP packets sent before SRTP is set up
611 // (and SetSend(true) is called).
Mirko Bonadei675513b2017-11-09 11:09:25 +0100612 RTC_LOG(LS_ERROR)
613 << "Can't send outgoing RTP packet when SRTP is inactive"
614 << " and crypto is required";
Zhi Huangcf990f52017-09-22 12:12:30 -0700615 RTC_NOTREACHED();
deadbeef8f425f92016-12-01 12:26:27 -0800616 return false;
617 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800618
619 std::string packet_type = rtcp ? "RTCP" : "RTP";
620 RTC_LOG(LS_WARNING) << "Sending an " << packet_type
621 << " packet without encryption.";
622 } else {
623 // Make sure we didn't accidentally send any packets without encryption.
624 RTC_DCHECK(rtp_transport_ == sdes_transport_.get() ||
625 rtp_transport_ == dtls_srtp_transport_.get());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000626 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000627 // Bon voyage.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800628 return rtcp ? rtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
629 : rtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000630}
631
zstein3dcf0e92017-06-01 13:22:42 -0700632bool BaseChannel::HandlesPayloadType(int packet_type) const {
zsteine8ab5432017-07-12 11:48:11 -0700633 return rtp_transport_->HandlesPayloadType(packet_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000634}
635
zstein3dcf0e92017-06-01 13:22:42 -0700636void BaseChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -0700637 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -0700638 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000639 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000640 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700641 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000642 }
643
Zhi Huangcf990f52017-09-22 12:12:30 -0700644 if (!srtp_active() && srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000645 // Our session description indicates that SRTP is required, but we got a
646 // packet before our SRTP filter is active. This means either that
647 // a) we got SRTP packets before we received the SDES keys, in which case
648 // we can't decrypt it anyway, or
649 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800650 // transports, so we haven't yet extracted keys, even if DTLS did
651 // complete on the transport that the packets are being sent on. It's
652 // really good practice to wait for both RTP and RTCP to be good to go
653 // before sending media, to prevent weird failure modes, so it's fine
654 // for us to just eat packets here. This is all sidestepped if RTCP mux
655 // is used anyway.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100656 RTC_LOG(LS_WARNING)
657 << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
658 << " packet when SRTP is inactive and crypto is required";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000659 return;
660 }
661
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200662 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700663 RTC_FROM_HERE, worker_thread_,
zstein634977b2017-07-14 12:30:04 -0700664 Bind(&BaseChannel::ProcessPacket, this, rtcp, *packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200665}
666
zstein3dcf0e92017-06-01 13:22:42 -0700667void BaseChannel::ProcessPacket(bool rtcp,
668 const rtc::CopyOnWriteBuffer& packet,
669 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200670 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700671
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200672 // Need to copy variable because OnRtcpReceived/OnPacketReceived
673 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
674 rtc::CopyOnWriteBuffer data(packet);
675 if (rtcp) {
676 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000677 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200678 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000679 }
680}
681
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000682void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700683 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000684 if (enabled_)
685 return;
686
Mirko Bonadei675513b2017-11-09 11:09:25 +0100687 RTC_LOG(LS_INFO) << "Channel enabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000688 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700689 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000690}
691
692void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700693 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000694 if (!enabled_)
695 return;
696
Mirko Bonadei675513b2017-11-09 11:09:25 +0100697 RTC_LOG(LS_INFO) << "Channel disabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000698 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700699 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000700}
701
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200702void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700703 rtc::PacketTransportInternal* rtp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700704 rtp_transport_->rtp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700705 rtc::PacketTransportInternal* rtcp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700706 rtp_transport_->rtcp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700707 if (rtp_packet_transport && rtp_packet_transport->writable() &&
708 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200709 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700710 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200711 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700712 }
713}
714
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200715void BaseChannel::ChannelWritable_n() {
716 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800717 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000718 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800719 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000720
Mirko Bonadei675513b2017-11-09 11:09:25 +0100721 RTC_LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
722 << (was_ever_writable_ ? "" : " for the first time");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000723
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000724 was_ever_writable_ = true;
725 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700726 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000727}
728
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200729bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800730 // Since DTLS is applied to all transports, checking RTP should be enough.
731 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000732}
733
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200734void BaseChannel::ChannelNotWritable_n() {
735 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000736 if (!writable_)
737 return;
738
Mirko Bonadei675513b2017-11-09 11:09:25 +0100739 RTC_LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000740 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700741 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000742}
743
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200744bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700745 const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800746 SdpType type,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700747 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700748 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700749 std::string* error_desc) {
jbauch5869f502017-06-29 12:31:36 -0700750 std::vector<int> encrypted_extension_ids;
751 for (const webrtc::RtpExtension& extension : extensions) {
752 if (extension.encrypt) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100753 RTC_LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
754 << " encrypted extension: " << extension.ToString();
jbauch5869f502017-06-29 12:31:36 -0700755 encrypted_extension_ids.push_back(extension.id);
756 }
757 }
758
deadbeef7af91dd2016-12-13 11:29:11 -0800759 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200760 return network_thread_->Invoke<bool>(
Steve Anton3828c062017-12-06 10:34:51 -0800761 RTC_FROM_HERE,
762 Bind(&BaseChannel::SetRtpTransportParameters_n, this, content, type, src,
763 encrypted_extension_ids, error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200764}
765
766bool BaseChannel::SetRtpTransportParameters_n(
767 const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800768 SdpType type,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200769 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700770 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200771 std::string* error_desc) {
772 RTC_DCHECK(network_thread_->IsCurrent());
773
Steve Anton3828c062017-12-06 10:34:51 -0800774 if (!SetSrtp_n(content->cryptos(), type, src, encrypted_extension_ids,
775 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700776 return false;
777 }
778
Steve Anton3828c062017-12-06 10:34:51 -0800779 if (!SetRtcpMux_n(content->rtcp_mux(), type, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700780 return false;
781 }
782
783 return true;
784}
785
zhihuangb2cdd932017-01-19 16:54:25 -0800786// |dtls| will be set to true if DTLS is active for transport and crypto is
787// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200788bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
789 bool* dtls,
790 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -0800791 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000792 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200793 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000794 return false;
795 }
796 return true;
797}
798
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800799void BaseChannel::EnableSdes_n() {
800 if (sdes_transport_) {
801 return;
Zhi Huangcf990f52017-09-22 12:12:30 -0700802 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800803 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
804 // time.
805 RTC_DCHECK(!dtls_srtp_transport_);
806 RTC_DCHECK(unencrypted_rtp_transport_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800807 sdes_transport_ = rtc::MakeUnique<webrtc::SrtpTransport>(
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800808 std::move(unencrypted_rtp_transport_));
Zhi Huangd7455782017-11-30 14:50:52 -0800809#if defined(ENABLE_EXTERNAL_AUTH)
810 sdes_transport_->EnableExternalAuth();
811#endif
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800812 SetRtpTransport(sdes_transport_.get());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800813 RTC_LOG(LS_INFO) << "Wrapping RtpTransport in SrtpTransport.";
814}
815
816void BaseChannel::EnableDtlsSrtp_n() {
817 if (dtls_srtp_transport_) {
818 return;
819 }
820 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
821 // time.
822 RTC_DCHECK(!sdes_transport_);
823 RTC_DCHECK(unencrypted_rtp_transport_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800824
825 auto srtp_transport = rtc::MakeUnique<webrtc::SrtpTransport>(
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800826 std::move(unencrypted_rtp_transport_));
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800827#if defined(ENABLE_EXTERNAL_AUTH)
828 srtp_transport->EnableExternalAuth();
829#endif
830 dtls_srtp_transport_ =
831 rtc::MakeUnique<webrtc::DtlsSrtpTransport>(std::move(srtp_transport));
832
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800833 SetRtpTransport(dtls_srtp_transport_.get());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800834 if (cached_send_extension_ids_) {
835 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
836 *cached_send_extension_ids_);
837 }
838 if (cached_recv_extension_ids_) {
839 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
840 *cached_recv_extension_ids_);
841 }
842 // Set the DtlsTransport and the |dtls_srtp_transport_| will handle the DTLS
843 // relate signal internally.
844 RTC_DCHECK(rtp_dtls_transport_);
845 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
846 rtcp_dtls_transport_);
847
848 RTC_LOG(LS_INFO) << "Wrapping SrtpTransport in DtlsSrtpTransport.";
Zhi Huangcf990f52017-09-22 12:12:30 -0700849}
850
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200851bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
Steve Anton3828c062017-12-06 10:34:51 -0800852 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000853 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700854 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000855 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -0800856 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000857 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000858 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200859 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000860 if (!ret) {
861 return false;
862 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700863
864 // If SRTP was not required, but we're setting a description that uses SDES,
865 // we need to upgrade to an SrtpTransport.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800866 if (!sdes_transport_ && !dtls && !cryptos.empty()) {
867 EnableSdes_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700868 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800869
Steve Anton3828c062017-12-06 10:34:51 -0800870 if ((type == SdpType::kAnswer || type == SdpType::kPrAnswer) && dtls) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800871 EnableDtlsSrtp_n();
872 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800873
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800874 UpdateEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
875
876 if (!dtls) {
Steve Anton3828c062017-12-06 10:34:51 -0800877 switch (type) {
878 case SdpType::kOffer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700879 ret = sdes_negotiator_.SetOffer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800880 break;
Steve Anton3828c062017-12-06 10:34:51 -0800881 case SdpType::kPrAnswer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700882 ret = sdes_negotiator_.SetProvisionalAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800883 break;
Steve Anton3828c062017-12-06 10:34:51 -0800884 case SdpType::kAnswer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700885 ret = sdes_negotiator_.SetAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800886 break;
887 default:
888 break;
889 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700890
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800891 // If setting an SDES answer succeeded, apply the negotiated parameters
892 // to the SRTP transport.
Steve Anton3828c062017-12-06 10:34:51 -0800893 if ((type == SdpType::kPrAnswer || type == SdpType::kAnswer) && ret) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800894 if (sdes_negotiator_.send_cipher_suite() &&
895 sdes_negotiator_.recv_cipher_suite()) {
896 RTC_DCHECK(cached_send_extension_ids_);
897 RTC_DCHECK(cached_recv_extension_ids_);
898 ret = sdes_transport_->SetRtpParams(
899 *(sdes_negotiator_.send_cipher_suite()),
900 sdes_negotiator_.send_key().data(),
901 static_cast<int>(sdes_negotiator_.send_key().size()),
902 *(cached_send_extension_ids_),
903 *(sdes_negotiator_.recv_cipher_suite()),
904 sdes_negotiator_.recv_key().data(),
905 static_cast<int>(sdes_negotiator_.recv_key().size()),
906 *(cached_recv_extension_ids_));
907 } else {
908 RTC_LOG(LS_INFO) << "No crypto keys are provided for SDES.";
Steve Anton3828c062017-12-06 10:34:51 -0800909 if (type == SdpType::kAnswer && sdes_transport_) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800910 // Explicitly reset the |sdes_transport_| if no crypto param is
911 // provided in the answer. No need to call |ResetParams()| for
912 // |sdes_negotiator_| because it resets the params inside |SetAnswer|.
913 sdes_transport_->ResetParams();
914 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700915 }
916 }
917 }
918
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000919 if (!ret) {
Zhi Huangc99b6c72017-11-10 16:44:46 -0800920 SafeSetError("Failed to setup SRTP.", error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000921 return false;
922 }
923 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000924}
925
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200926bool BaseChannel::SetRtcpMux_n(bool enable,
Steve Anton3828c062017-12-06 10:34:51 -0800927 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000928 ContentSource src,
929 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -0800930 // Provide a more specific error message for the RTCP mux "require" policy
931 // case.
zstein56162b92017-04-24 16:54:35 -0700932 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -0800933 SafeSetError(
934 "rtcpMuxPolicy is 'require', but media description does not "
935 "contain 'a=rtcp-mux'.",
936 error_desc);
937 return false;
938 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000939 bool ret = false;
Steve Anton3828c062017-12-06 10:34:51 -0800940 switch (type) {
941 case SdpType::kOffer:
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000942 ret = rtcp_mux_filter_.SetOffer(enable, src);
943 break;
Steve Anton3828c062017-12-06 10:34:51 -0800944 case SdpType::kPrAnswer:
zhihuangb2cdd932017-01-19 16:54:25 -0800945 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700946 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000947 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
948 break;
Steve Anton3828c062017-12-06 10:34:51 -0800949 case SdpType::kAnswer:
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000950 ret = rtcp_mux_filter_.SetAnswer(enable, src);
951 if (ret && rtcp_mux_filter_.IsActive()) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800952 ActivateRtcpMux();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000953 }
954 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000955 default:
956 break;
957 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000958 if (!ret) {
959 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
960 return false;
961 }
zsteine8ab5432017-07-12 11:48:11 -0700962 rtp_transport_->SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
Steve Anton3828c062017-12-06 10:34:51 -0800963 // |rtcp_mux_filter_| can be active if |action| is SdpType::kPrAnswer or
964 // SdpType::kAnswer, but we only want to tear down the RTCP transport if we
965 // received a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000966 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000967 // If the RTP transport is already writable, then so are we.
zsteine8ab5432017-07-12 11:48:11 -0700968 if (rtp_transport_->rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200969 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000970 }
971 }
972
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000973 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000974}
975
976bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700977 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -0800978 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000979}
980
Peter Boström0c4e06b2015-10-07 12:23:21 +0200981bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700982 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000983 return media_channel()->RemoveRecvStream(ssrc);
984}
985
986bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
Steve Anton3828c062017-12-06 10:34:51 -0800987 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000988 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000989 // Check for streams that have been removed.
990 bool ret = true;
991 for (StreamParamsVec::const_iterator it = local_streams_.begin();
992 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +0000993 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000994 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000995 std::ostringstream desc;
996 desc << "Failed to remove send stream with ssrc "
997 << it->first_ssrc() << ".";
998 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000999 ret = false;
1000 }
1001 }
1002 }
1003 // Check for new streams.
1004 for (StreamParamsVec::const_iterator it = streams.begin();
1005 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001006 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001007 if (media_channel()->AddSendStream(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001008 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001009 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001010 std::ostringstream desc;
1011 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1012 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001013 ret = false;
1014 }
1015 }
1016 }
1017 local_streams_ = streams;
1018 return ret;
1019}
1020
1021bool BaseChannel::UpdateRemoteStreams_w(
1022 const std::vector<StreamParams>& streams,
Steve Anton3828c062017-12-06 10:34:51 -08001023 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001024 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001025 // Check for streams that have been removed.
1026 bool ret = true;
1027 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1028 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001029 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001030 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001031 std::ostringstream desc;
1032 desc << "Failed to remove remote stream with ssrc "
1033 << it->first_ssrc() << ".";
1034 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001035 ret = false;
1036 }
1037 }
1038 }
1039 // Check for new streams.
1040 for (StreamParamsVec::const_iterator it = streams.begin();
1041 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001042 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001043 if (AddRecvStream_w(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001044 RTC_LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001045 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001046 std::ostringstream desc;
1047 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1048 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001049 ret = false;
1050 }
1051 }
1052 }
1053 remote_streams_ = streams;
1054 return ret;
1055}
1056
jbauch5869f502017-06-29 12:31:36 -07001057RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1058 const RtpHeaderExtensions& extensions) {
1059 if (!rtp_dtls_transport_ ||
1060 !rtp_dtls_transport_->crypto_options()
1061 .enable_encrypted_rtp_header_extensions) {
1062 RtpHeaderExtensions filtered;
1063 auto pred = [](const webrtc::RtpExtension& extension) {
1064 return !extension.encrypt;
1065 };
1066 std::copy_if(extensions.begin(), extensions.end(),
1067 std::back_inserter(filtered), pred);
1068 return filtered;
1069 }
1070
1071 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1072}
1073
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001074void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001075 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001076// Absolute Send Time extension id is used only with external auth,
1077// so do not bother searching for it and making asyncronious call to set
1078// something that is not used.
1079#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001080 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001081 webrtc::RtpExtension::FindHeaderExtensionByUri(
1082 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001083 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001084 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001085 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001086 RTC_FROM_HERE, network_thread_,
1087 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1088 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001089#endif
1090}
1091
1092void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1093 int rtp_abs_sendtime_extn_id) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001094 if (sdes_transport_) {
1095 sdes_transport_->CacheRtpAbsSendTimeHeaderExtension(
Zhi Huangcf990f52017-09-22 12:12:30 -07001096 rtp_abs_sendtime_extn_id);
Zhi Huang2a4d70c2017-11-29 15:41:59 -08001097 } else if (dtls_srtp_transport_) {
1098 dtls_srtp_transport_->CacheRtpAbsSendTimeHeaderExtension(
1099 rtp_abs_sendtime_extn_id);
Zhi Huangcf990f52017-09-22 12:12:30 -07001100 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001101 RTC_LOG(LS_WARNING)
1102 << "Trying to cache the Absolute Send Time extension id "
1103 "but the SRTP is not active.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001104 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001105}
1106
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001107void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001108 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001109 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001110 case MSG_SEND_RTP_PACKET:
1111 case MSG_SEND_RTCP_PACKET: {
1112 RTC_DCHECK(network_thread_->IsCurrent());
1113 SendPacketMessageData* data =
1114 static_cast<SendPacketMessageData*>(pmsg->pdata);
1115 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1116 SendPacket(rtcp, &data->packet, data->options);
1117 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001118 break;
1119 }
1120 case MSG_FIRSTPACKETRECEIVED: {
1121 SignalFirstPacketReceived(this);
1122 break;
1123 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001124 }
1125}
1126
zstein3dcf0e92017-06-01 13:22:42 -07001127void BaseChannel::AddHandledPayloadType(int payload_type) {
zsteine8ab5432017-07-12 11:48:11 -07001128 rtp_transport_->AddHandledPayloadType(payload_type);
zstein3dcf0e92017-06-01 13:22:42 -07001129}
1130
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001131void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001132 // Flush all remaining RTCP messages. This should only be called in
1133 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001134 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001135 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001136 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1137 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001138 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1139 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001140 }
1141}
1142
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001143void BaseChannel::SignalSentPacket_n(const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001144 RTC_DCHECK(network_thread_->IsCurrent());
1145 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001146 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001147 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1148}
1149
1150void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1151 RTC_DCHECK(worker_thread_->IsCurrent());
1152 SignalSentPacket(sent_packet);
1153}
1154
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001155void BaseChannel::UpdateEncryptedHeaderExtensionIds(
Zhi Huangc99b6c72017-11-10 16:44:46 -08001156 cricket::ContentSource source,
1157 const std::vector<int>& extension_ids) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001158 if (source == ContentSource::CS_LOCAL) {
1159 cached_recv_extension_ids_ = std::move(extension_ids);
1160 if (dtls_srtp_transport_) {
1161 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
1162 extension_ids);
1163 }
1164 } else {
1165 cached_send_extension_ids_ = std::move(extension_ids);
1166 if (dtls_srtp_transport_) {
1167 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
1168 extension_ids);
1169 }
1170 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001171}
1172
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001173void BaseChannel::ActivateRtcpMux() {
1174 // We permanently activated RTCP muxing; signal that we no longer need
1175 // the RTCP transport.
1176 std::string debug_name =
1177 transport_name_.empty()
1178 ? rtp_transport_->rtp_packet_transport()->transport_name()
1179 : transport_name_;
1180 RTC_LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1181 << "; no longer need RTCP transport for " << debug_name;
1182 if (rtp_transport_->rtcp_packet_transport()) {
1183 SetTransport_n(/*rtcp=*/true, nullptr, nullptr);
1184 if (dtls_srtp_transport_) {
1185 RTC_DCHECK(rtp_dtls_transport_);
1186 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
1187 /*rtcp_dtls_transport_=*/nullptr);
1188 } else {
1189 rtp_transport_->SetRtcpPacketTransport(nullptr);
1190 }
1191 SignalRtcpMuxFullyActive(transport_name_);
Zhi Huangc99b6c72017-11-10 16:44:46 -08001192 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001193 UpdateWritableState_n();
Zhi Huangc99b6c72017-11-10 16:44:46 -08001194}
1195
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001196VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1197 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001198 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001199 MediaEngineInterface* media_engine,
Steve Anton8699a322017-11-06 15:53:33 -08001200 std::unique_ptr<VoiceMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001201 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001202 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001203 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001204 : BaseChannel(worker_thread,
1205 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001206 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001207 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001208 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001209 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001210 srtp_required),
Steve Anton8699a322017-11-06 15:53:33 -08001211 media_engine_(media_engine) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001212
1213VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001214 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001215 StopAudioMonitor();
1216 StopMediaMonitor();
1217 // this can't be done in the base class, since it calls a virtual
1218 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001219 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001220}
1221
Peter Boström0c4e06b2015-10-07 12:23:21 +02001222bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001223 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001224 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001225 AudioSource* source) {
stefanf79ade12017-06-02 06:44:03 -07001226 return InvokeOnWorker<bool>(
1227 RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
1228 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001229}
1230
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001231// TODO(juberti): Handle early media the right way. We should get an explicit
1232// ringing message telling us to start playing local ringback, which we cancel
1233// if any early media actually arrives. For now, we do the opposite, which is
1234// to wait 1 second for early media, and start playing local ringback if none
1235// arrives.
1236void VoiceChannel::SetEarlyMedia(bool enable) {
1237 if (enable) {
1238 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001239 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1240 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001241 } else {
1242 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001243 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001244 }
1245}
1246
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001247bool VoiceChannel::CanInsertDtmf() {
stefanf79ade12017-06-02 06:44:03 -07001248 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001249 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001250}
1251
Peter Boström0c4e06b2015-10-07 12:23:21 +02001252bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1253 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001254 int duration) {
stefanf79ade12017-06-02 06:44:03 -07001255 return InvokeOnWorker<bool>(
1256 RTC_FROM_HERE,
1257 Bind(&VoiceChannel::InsertDtmf_w, this, ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001258}
1259
solenberg4bac9c52015-10-09 02:32:53 -07001260bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
stefanf79ade12017-06-02 06:44:03 -07001261 return InvokeOnWorker<bool>(
1262 RTC_FROM_HERE,
1263 Bind(&VoiceMediaChannel::SetOutputVolume, media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001264}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001265
Tommif888bb52015-12-12 01:37:01 +01001266void VoiceChannel::SetRawAudioSink(
1267 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001268 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1269 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001270 // passing. So we invoke to our own little routine that gets a pointer to
1271 // our local variable. This is OK since we're synchronously invoking.
stefanf79ade12017-06-02 06:44:03 -07001272 InvokeOnWorker<bool>(RTC_FROM_HERE,
1273 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001274}
1275
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001276webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001277 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001278 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001279}
1280
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001281webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1282 uint32_t ssrc) const {
1283 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001284}
1285
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001286bool VoiceChannel::SetRtpSendParameters(
1287 uint32_t ssrc,
1288 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001289 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001290 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001291 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001292}
1293
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001294bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1295 webrtc::RtpParameters parameters) {
1296 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1297}
1298
1299webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1300 uint32_t ssrc) const {
1301 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001302 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001303 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1304}
1305
1306webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1307 uint32_t ssrc) const {
1308 return media_channel()->GetRtpReceiveParameters(ssrc);
1309}
1310
1311bool VoiceChannel::SetRtpReceiveParameters(
1312 uint32_t ssrc,
1313 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001314 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001315 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001316 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1317}
1318
1319bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1320 webrtc::RtpParameters parameters) {
1321 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001322}
1323
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001324bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001325 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1326 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001327}
1328
hbos8d609f62017-04-10 07:39:05 -07001329std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1330 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
zhihuang38ede132017-06-15 12:52:32 -07001331 RTC_FROM_HERE, Bind(&VoiceChannel::GetSources_w, this, ssrc));
1332}
1333
1334std::vector<webrtc::RtpSource> VoiceChannel::GetSources_w(uint32_t ssrc) const {
1335 RTC_DCHECK(worker_thread()->IsCurrent());
1336 return media_channel()->GetSources(ssrc);
hbos8d609f62017-04-10 07:39:05 -07001337}
1338
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001339void VoiceChannel::StartMediaMonitor(int cms) {
1340 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001341 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001342 media_monitor_->SignalUpdate.connect(
1343 this, &VoiceChannel::OnMediaMonitorUpdate);
1344 media_monitor_->Start(cms);
1345}
1346
1347void VoiceChannel::StopMediaMonitor() {
1348 if (media_monitor_) {
1349 media_monitor_->Stop();
1350 media_monitor_->SignalUpdate.disconnect(this);
1351 media_monitor_.reset();
1352 }
1353}
1354
1355void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001356 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001357 audio_monitor_
1358 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1359 audio_monitor_->Start(cms);
1360}
1361
1362void VoiceChannel::StopAudioMonitor() {
1363 if (audio_monitor_) {
1364 audio_monitor_->Stop();
1365 audio_monitor_.reset();
1366 }
1367}
1368
1369bool VoiceChannel::IsAudioMonitorRunning() const {
1370 return (audio_monitor_.get() != NULL);
1371}
1372
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001373int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001374 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001375}
1376
1377int VoiceChannel::GetOutputLevel_w() {
1378 return media_channel()->GetOutputLevel();
1379}
1380
1381void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1382 media_channel()->GetActiveStreams(actives);
1383}
1384
zstein3dcf0e92017-06-01 13:22:42 -07001385void VoiceChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -07001386 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -07001387 const rtc::PacketTime& packet_time) {
1388 BaseChannel::OnPacketReceived(rtcp, packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001389 // Set a flag when we've received an RTP packet. If we're waiting for early
1390 // media, this will disable the timeout.
zstein3dcf0e92017-06-01 13:22:42 -07001391 if (!received_media_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001392 received_media_ = true;
1393 }
1394}
1395
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001396void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001397 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001398 invoker_.AsyncInvoke<void>(
1399 RTC_FROM_HERE, worker_thread_,
1400 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001401}
1402
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001403void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001404 // Render incoming data if we're the active call, and we have the local
1405 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001406 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001407 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001408
1409 // Send outgoing data if we're the active call, we have the remote content,
1410 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001411 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001412 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001413
Mirko Bonadei675513b2017-11-09 11:09:25 +01001414 RTC_LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001415}
1416
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001417bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001418 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001419 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001420 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001421 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001422 RTC_LOG(LS_INFO) << "Setting local voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001423
Steve Antonb1c1de12017-12-21 15:14:30 -08001424 RTC_DCHECK(content);
1425 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001426 SafeSetError("Can't find audio content in local description.", error_desc);
1427 return false;
1428 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001429
Steve Antonb1c1de12017-12-21 15:14:30 -08001430 const AudioContentDescription* audio = content->as_audio();
1431
jbauch5869f502017-06-29 12:31:36 -07001432 RtpHeaderExtensions rtp_header_extensions =
1433 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1434
Steve Anton3828c062017-12-06 10:34:51 -08001435 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1436 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001437 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001438 }
1439
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001440 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001441 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001442 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001443 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001444 error_desc);
1445 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001446 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001447 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001448 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001449 }
1450 last_recv_params_ = recv_params;
1451
1452 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1453 // only give it to the media channel once we have a remote
1454 // description too (without a remote description, we won't be able
1455 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001456 if (!UpdateLocalStreams_w(audio->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001457 SafeSetError("Failed to set local audio description streams.", error_desc);
1458 return false;
1459 }
1460
1461 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001462 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001463 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001464}
1465
1466bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001467 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001468 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001469 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001470 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001471 RTC_LOG(LS_INFO) << "Setting remote voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001472
Steve Antonb1c1de12017-12-21 15:14:30 -08001473 RTC_DCHECK(content);
1474 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001475 SafeSetError("Can't find audio content in remote description.", error_desc);
1476 return false;
1477 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001478
Steve Antonb1c1de12017-12-21 15:14:30 -08001479 const AudioContentDescription* audio = content->as_audio();
1480
jbauch5869f502017-06-29 12:31:36 -07001481 RtpHeaderExtensions rtp_header_extensions =
1482 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1483
Steve Anton3828c062017-12-06 10:34:51 -08001484 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1485 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001486 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001487 }
1488
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001489 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001490 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1491 &send_params);
skvladdc1c62c2016-03-16 19:07:43 -07001492
1493 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1494 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001495 SafeSetError("Failed to set remote audio description send parameters.",
1496 error_desc);
1497 return false;
1498 }
1499 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001500
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001501 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1502 // and only give it to the media channel once we have a local
1503 // description too (without a local description, we won't be able to
1504 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001505 if (!UpdateRemoteStreams_w(audio->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001506 SafeSetError("Failed to set remote audio description streams.", error_desc);
1507 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001508 }
1509
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001510 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001511 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001512 }
1513
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001514 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001515 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001516 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001517}
1518
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001519void VoiceChannel::HandleEarlyMediaTimeout() {
1520 // This occurs on the main thread, not the worker thread.
1521 if (!received_media_) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001522 RTC_LOG(LS_INFO) << "No early media received before timeout";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001523 SignalEarlyMediaTimeout(this);
1524 }
1525}
1526
Peter Boström0c4e06b2015-10-07 12:23:21 +02001527bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1528 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001529 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001530 if (!enabled()) {
1531 return false;
1532 }
solenberg1d63dd02015-12-02 12:35:09 -08001533 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001534}
1535
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001536void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001537 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001538 case MSG_EARLYMEDIATIMEOUT:
1539 HandleEarlyMediaTimeout();
1540 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001541 case MSG_CHANNEL_ERROR: {
1542 VoiceChannelErrorMessageData* data =
1543 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001544 delete data;
1545 break;
1546 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001547 default:
1548 BaseChannel::OnMessage(pmsg);
1549 break;
1550 }
1551}
1552
1553void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001554 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001555 SignalConnectionMonitor(this, infos);
1556}
1557
1558void VoiceChannel::OnMediaMonitorUpdate(
1559 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001560 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001561 SignalMediaMonitor(this, info);
1562}
1563
1564void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1565 const AudioInfo& info) {
1566 SignalAudioMonitor(this, info);
1567}
1568
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001569VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1570 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001571 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001572 std::unique_ptr<VideoMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001573 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001574 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001575 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001576 : BaseChannel(worker_thread,
1577 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001578 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001579 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001580 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001581 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001582 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001583
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001584VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001585 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001586 StopMediaMonitor();
1587 // this can't be done in the base class, since it calls a virtual
1588 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001589
1590 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001591}
1592
nisse08582ff2016-02-04 01:24:52 -08001593bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001594 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001595 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001596 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001597 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001598 return true;
1599}
1600
deadbeef5a4a75a2016-06-02 16:23:38 -07001601bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001602 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001603 bool mute,
1604 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001605 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
stefanf79ade12017-06-02 06:44:03 -07001606 return InvokeOnWorker<bool>(
1607 RTC_FROM_HERE, Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
1608 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001609}
1610
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001611webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001612 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001613 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001614}
1615
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001616webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1617 uint32_t ssrc) const {
1618 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001619}
1620
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001621bool VideoChannel::SetRtpSendParameters(
1622 uint32_t ssrc,
1623 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001624 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001625 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001626 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001627}
1628
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001629bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1630 webrtc::RtpParameters parameters) {
1631 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1632}
1633
1634webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1635 uint32_t ssrc) const {
1636 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001637 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001638 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1639}
1640
1641webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1642 uint32_t ssrc) const {
1643 return media_channel()->GetRtpReceiveParameters(ssrc);
1644}
1645
1646bool VideoChannel::SetRtpReceiveParameters(
1647 uint32_t ssrc,
1648 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001649 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001650 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001651 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1652}
1653
1654bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1655 webrtc::RtpParameters parameters) {
1656 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001657}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001658
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001659void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001660 // Send outgoing data if we're the active call, we have the remote content,
1661 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001662 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001663 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001664 RTC_LOG(LS_ERROR) << "Failed to SetSend on video channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001665 // TODO(gangji): Report error back to server.
1666 }
1667
Mirko Bonadei675513b2017-11-09 11:09:25 +01001668 RTC_LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001669}
1670
stefanf79ade12017-06-02 06:44:03 -07001671void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
1672 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
1673 media_channel(), bwe_info));
1674}
1675
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00001676bool VideoChannel::GetStats(VideoMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001677 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
1678 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001679}
1680
1681void VideoChannel::StartMediaMonitor(int cms) {
1682 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001683 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001684 media_monitor_->SignalUpdate.connect(
1685 this, &VideoChannel::OnMediaMonitorUpdate);
1686 media_monitor_->Start(cms);
1687}
1688
1689void VideoChannel::StopMediaMonitor() {
1690 if (media_monitor_) {
1691 media_monitor_->Stop();
1692 media_monitor_.reset();
1693 }
1694}
1695
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001696bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001697 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001698 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001699 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001700 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001701 RTC_LOG(LS_INFO) << "Setting local video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001702
Steve Antonb1c1de12017-12-21 15:14:30 -08001703 RTC_DCHECK(content);
1704 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001705 SafeSetError("Can't find video content in local description.", error_desc);
1706 return false;
1707 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001708
Steve Antonb1c1de12017-12-21 15:14:30 -08001709 const VideoContentDescription* video = content->as_video();
1710
jbauch5869f502017-06-29 12:31:36 -07001711 RtpHeaderExtensions rtp_header_extensions =
1712 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1713
Steve Anton3828c062017-12-06 10:34:51 -08001714 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1715 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001716 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001717 }
1718
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001719 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001720 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001721 if (!media_channel()->SetRecvParameters(recv_params)) {
1722 SafeSetError("Failed to set local video description recv parameters.",
1723 error_desc);
1724 return false;
1725 }
1726 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001727 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001728 }
1729 last_recv_params_ = recv_params;
1730
1731 // TODO(pthatcher): Move local streams into VideoSendParameters, and
1732 // only give it to the media channel once we have a remote
1733 // description too (without a remote description, we won't be able
1734 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001735 if (!UpdateLocalStreams_w(video->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001736 SafeSetError("Failed to set local video description streams.", error_desc);
1737 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001738 }
1739
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001740 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001741 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001742 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001743}
1744
1745bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001746 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001747 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001748 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001749 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001750 RTC_LOG(LS_INFO) << "Setting remote video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001751
Steve Antonb1c1de12017-12-21 15:14:30 -08001752 RTC_DCHECK(content);
1753 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001754 SafeSetError("Can't find video content in remote description.", error_desc);
1755 return false;
1756 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001757
Steve Antonb1c1de12017-12-21 15:14:30 -08001758 const VideoContentDescription* video = content->as_video();
1759
jbauch5869f502017-06-29 12:31:36 -07001760 RtpHeaderExtensions rtp_header_extensions =
1761 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1762
Steve Anton3828c062017-12-06 10:34:51 -08001763 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1764 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001765 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001766 }
1767
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001768 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001769 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
1770 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001771 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08001772 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001773 }
skvladdc1c62c2016-03-16 19:07:43 -07001774
1775 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1776
1777 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001778 SafeSetError("Failed to set remote video description send parameters.",
1779 error_desc);
1780 return false;
1781 }
1782 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001783
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001784 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
1785 // and only give it to the media channel once we have a local
1786 // description too (without a local description, we won't be able to
1787 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001788 if (!UpdateRemoteStreams_w(video->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001789 SafeSetError("Failed to set remote video description streams.", error_desc);
1790 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001791 }
1792
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001793 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001794 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001795 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001796
1797 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001798 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001799 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001800}
1801
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001802void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001803 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001804 case MSG_CHANNEL_ERROR: {
1805 const VideoChannelErrorMessageData* data =
1806 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001807 delete data;
1808 break;
1809 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001810 default:
1811 BaseChannel::OnMessage(pmsg);
1812 break;
1813 }
1814}
1815
1816void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001817 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001818 SignalConnectionMonitor(this, infos);
1819}
1820
1821// TODO(pthatcher): Look into removing duplicate code between
1822// audio, video, and data, perhaps by using templates.
1823void VideoChannel::OnMediaMonitorUpdate(
1824 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001825 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001826 SignalMediaMonitor(this, info);
1827}
1828
deadbeef953c2ce2017-01-09 14:53:41 -08001829RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
1830 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001831 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001832 std::unique_ptr<DataMediaChannel> media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08001833 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001834 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001835 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001836 : BaseChannel(worker_thread,
1837 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001838 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001839 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001840 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001841 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001842 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001843
deadbeef953c2ce2017-01-09 14:53:41 -08001844RtpDataChannel::~RtpDataChannel() {
1845 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001846 StopMediaMonitor();
1847 // this can't be done in the base class, since it calls a virtual
1848 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001849
1850 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001851}
1852
Steve Anton8699a322017-11-06 15:53:33 -08001853void RtpDataChannel::Init_w(
deadbeeff5346592017-01-24 21:51:21 -08001854 DtlsTransportInternal* rtp_dtls_transport,
1855 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08001856 rtc::PacketTransportInternal* rtp_packet_transport,
1857 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -08001858 BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
1859 rtp_packet_transport, rtcp_packet_transport);
1860
deadbeef953c2ce2017-01-09 14:53:41 -08001861 media_channel()->SignalDataReceived.connect(this,
1862 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00001863 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08001864 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001865}
1866
Zhi Huang2dfc42d2017-12-04 13:38:48 -08001867void RtpDataChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
1868 BaseChannel::Init_w(rtp_transport);
1869 media_channel()->SignalDataReceived.connect(this,
1870 &RtpDataChannel::OnDataReceived);
1871 media_channel()->SignalReadyToSend.connect(
1872 this, &RtpDataChannel::OnDataChannelReadyToSend);
1873}
1874
deadbeef953c2ce2017-01-09 14:53:41 -08001875bool RtpDataChannel::SendData(const SendDataParams& params,
1876 const rtc::CopyOnWriteBuffer& payload,
1877 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07001878 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001879 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
1880 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001881}
1882
deadbeef953c2ce2017-01-09 14:53:41 -08001883bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001884 const DataContentDescription* content,
1885 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001886 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
1887 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08001888 // It's been set before, but doesn't match. That's bad.
1889 if (is_sctp) {
1890 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
1891 error_desc);
1892 return false;
1893 }
1894 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001895}
1896
deadbeef953c2ce2017-01-09 14:53:41 -08001897bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001898 SdpType type,
deadbeef953c2ce2017-01-09 14:53:41 -08001899 std::string* error_desc) {
1900 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001901 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001902 RTC_LOG(LS_INFO) << "Setting local data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001903
Steve Antonb1c1de12017-12-21 15:14:30 -08001904 RTC_DCHECK(content);
1905 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001906 SafeSetError("Can't find data content in local description.", error_desc);
1907 return false;
1908 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001909
Steve Antonb1c1de12017-12-21 15:14:30 -08001910 const DataContentDescription* data = content->as_data();
1911
deadbeef953c2ce2017-01-09 14:53:41 -08001912 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001913 return false;
1914 }
1915
jbauch5869f502017-06-29 12:31:36 -07001916 RtpHeaderExtensions rtp_header_extensions =
1917 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
1918
Steve Anton3828c062017-12-06 10:34:51 -08001919 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1920 error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001921 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001922 }
1923
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001924 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001925 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001926 if (!media_channel()->SetRecvParameters(recv_params)) {
1927 SafeSetError("Failed to set remote data description recv parameters.",
1928 error_desc);
1929 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001930 }
deadbeef953c2ce2017-01-09 14:53:41 -08001931 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001932 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001933 }
1934 last_recv_params_ = recv_params;
1935
1936 // TODO(pthatcher): Move local streams into DataSendParameters, and
1937 // only give it to the media channel once we have a remote
1938 // description too (without a remote description, we won't be able
1939 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001940 if (!UpdateLocalStreams_w(data->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001941 SafeSetError("Failed to set local data description streams.", error_desc);
1942 return false;
1943 }
1944
1945 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001946 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001947 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001948}
1949
deadbeef953c2ce2017-01-09 14:53:41 -08001950bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001951 SdpType type,
deadbeef953c2ce2017-01-09 14:53:41 -08001952 std::string* error_desc) {
1953 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001954 RTC_DCHECK_RUN_ON(worker_thread());
1955 RTC_LOG(LS_INFO) << "Setting remote data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001956
Steve Antonb1c1de12017-12-21 15:14:30 -08001957 RTC_DCHECK(content);
1958 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001959 SafeSetError("Can't find data content in remote description.", error_desc);
1960 return false;
1961 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001962
Steve Antonb1c1de12017-12-21 15:14:30 -08001963 const DataContentDescription* data = content->as_data();
1964
Zhi Huang801b8682017-11-15 11:36:43 -08001965 // If the remote data doesn't have codecs, it must be empty, so ignore it.
1966 if (!data->has_codecs()) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001967 return true;
1968 }
1969
deadbeef953c2ce2017-01-09 14:53:41 -08001970 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001971 return false;
1972 }
1973
jbauch5869f502017-06-29 12:31:36 -07001974 RtpHeaderExtensions rtp_header_extensions =
1975 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
1976
Mirko Bonadei675513b2017-11-09 11:09:25 +01001977 RTC_LOG(LS_INFO) << "Setting remote data description";
Steve Anton3828c062017-12-06 10:34:51 -08001978 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1979 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001980 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001981 }
1982
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001983 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001984 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
1985 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001986 if (!media_channel()->SetSendParameters(send_params)) {
1987 SafeSetError("Failed to set remote data description send parameters.",
1988 error_desc);
1989 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001990 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001991 last_send_params_ = send_params;
1992
1993 // TODO(pthatcher): Move remote streams into DataRecvParameters,
1994 // and only give it to the media channel once we have a local
1995 // description too (without a local description, we won't be able to
1996 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001997 if (!UpdateRemoteStreams_w(data->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001998 SafeSetError("Failed to set remote data description streams.",
1999 error_desc);
2000 return false;
2001 }
2002
2003 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002004 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002005 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002006}
2007
deadbeef953c2ce2017-01-09 14:53:41 -08002008void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002009 // Render incoming data if we're the active call, and we have the local
2010 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002011 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002012 if (!media_channel()->SetReceive(recv)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002013 RTC_LOG(LS_ERROR) << "Failed to SetReceive on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002014 }
2015
2016 // Send outgoing data if we're the active call, we have the remote content,
2017 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002018 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002019 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002020 RTC_LOG(LS_ERROR) << "Failed to SetSend on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002021 }
2022
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002023 // Trigger SignalReadyToSendData asynchronously.
2024 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002025
Mirko Bonadei675513b2017-11-09 11:09:25 +01002026 RTC_LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002027}
2028
deadbeef953c2ce2017-01-09 14:53:41 -08002029void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002030 switch (pmsg->message_id) {
2031 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002032 DataChannelReadyToSendMessageData* data =
2033 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002034 ready_to_send_data_ = data->data();
2035 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002036 delete data;
2037 break;
2038 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002039 case MSG_DATARECEIVED: {
2040 DataReceivedMessageData* data =
2041 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002042 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002043 delete data;
2044 break;
2045 }
2046 case MSG_CHANNEL_ERROR: {
2047 const DataChannelErrorMessageData* data =
2048 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002049 delete data;
2050 break;
2051 }
2052 default:
2053 BaseChannel::OnMessage(pmsg);
2054 break;
2055 }
2056}
2057
deadbeef953c2ce2017-01-09 14:53:41 -08002058void RtpDataChannel::OnConnectionMonitorUpdate(
2059 ConnectionMonitor* monitor,
2060 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002061 SignalConnectionMonitor(this, infos);
2062}
2063
deadbeef953c2ce2017-01-09 14:53:41 -08002064void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002065 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002066 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002067 media_monitor_->SignalUpdate.connect(this,
2068 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002069 media_monitor_->Start(cms);
2070}
2071
deadbeef953c2ce2017-01-09 14:53:41 -08002072void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002073 if (media_monitor_) {
2074 media_monitor_->Stop();
2075 media_monitor_->SignalUpdate.disconnect(this);
2076 media_monitor_.reset();
2077 }
2078}
2079
deadbeef953c2ce2017-01-09 14:53:41 -08002080void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2081 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002082 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002083 SignalMediaMonitor(this, info);
2084}
2085
deadbeef953c2ce2017-01-09 14:53:41 -08002086void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2087 const char* data,
2088 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002089 DataReceivedMessageData* msg = new DataReceivedMessageData(
2090 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002091 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002092}
2093
deadbeef953c2ce2017-01-09 14:53:41 -08002094void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2095 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002096 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2097 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002098 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002099}
2100
deadbeef953c2ce2017-01-09 14:53:41 -08002101void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002102 // This is usded for congestion control to indicate that the stream is ready
2103 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2104 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002105 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002106 new DataChannelReadyToSendMessageData(writable));
2107}
2108
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002109} // namespace cricket