blob: 53a44b62cea3d339050e6d4d83d2770f22ab3fba [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"
Patrik Höglund42805f32018-01-18 19:15:38 +000029// Adding 'nogncheck' to disable the gn include headers check to support modular
30// WebRTC build targets.
31#include "media/engine/webrtcvoiceengine.h" // nogncheck
Zhi Huangea8b62a2018-03-26 14:31:17 -070032#include "modules/rtp_rtcp/source/rtp_packet_received.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020033#include "p2p/base/packettransportinternal.h"
34#include "pc/channelmanager.h"
Steve Anton4e70a722017-11-28 14:57:10 -080035#include "pc/rtpmediautils.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000036
37namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000038using rtc::Bind;
Steve Anton3828c062017-12-06 10:34:51 -080039using webrtc::SdpType;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000040
deadbeef2d110be2016-01-13 12:00:26 -080041namespace {
Danil Chapovalov33b01f22016-05-11 19:55:27 +020042
43struct SendPacketMessageData : public rtc::MessageData {
44 rtc::CopyOnWriteBuffer packet;
45 rtc::PacketOptions options;
46};
47
deadbeef2d110be2016-01-13 12:00:26 -080048} // namespace
49
henrike@webrtc.org28e20752013-07-10 00:45:36 +000050enum {
Steve Anton0807d152018-03-05 11:23:09 -080051 MSG_SEND_RTP_PACKET = 1,
Danil Chapovalov33b01f22016-05-11 19:55:27 +020052 MSG_SEND_RTCP_PACKET,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000053 MSG_READYTOSENDDATA,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000054 MSG_DATARECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000055 MSG_FIRSTPACKETRECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000056};
57
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000058static void SafeSetError(const std::string& message, std::string* error_desc) {
59 if (error_desc) {
60 *error_desc = message;
61 }
62}
63
jbaucheec21bd2016-03-20 06:15:43 -070064static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000065 // Check the packet size. We could check the header too if needed.
zstein3dcf0e92017-06-01 13:22:42 -070066 return packet && IsValidRtpRtcpPacketSize(rtcp, packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000067}
68
Peter Thatcherc2ee2c82015-08-07 16:05:34 -070069template <class Codec>
70void RtpParametersFromMediaDescription(
71 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -070072 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -070073 RtpParameters<Codec>* params) {
74 // TODO(pthatcher): Remove this once we're sure no one will give us
Zhi Huang801b8682017-11-15 11:36:43 -080075 // a description without codecs. Currently the ORTC implementation is relying
76 // on this.
Peter Thatcherc2ee2c82015-08-07 16:05:34 -070077 if (desc->has_codecs()) {
78 params->codecs = desc->codecs();
79 }
80 // TODO(pthatcher): See if we really need
81 // rtp_header_extensions_set() and remove it if we don't.
82 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -070083 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -070084 }
deadbeef13871492015-12-09 12:37:51 -080085 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -070086}
87
nisse05103312016-03-16 02:22:50 -070088template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -070089void RtpSendParametersFromMediaDescription(
90 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -070091 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -070092 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -070093 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -070094 send_params->max_bandwidth_bps = desc->bandwidth();
95}
96
Danil Chapovalov33b01f22016-05-11 19:55:27 +020097BaseChannel::BaseChannel(rtc::Thread* worker_thread,
98 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -080099 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -0800100 std::unique_ptr<MediaChannel> media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700101 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800102 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800103 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200104 : worker_thread_(worker_thread),
105 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800106 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000107 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700108 rtcp_mux_required_(rtcp_mux_required),
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800109 unencrypted_rtp_transport_(
110 rtc::MakeUnique<webrtc::RtpTransport>(rtcp_mux_required)),
deadbeef7af91dd2016-12-13 11:29:11 -0800111 srtp_required_(srtp_required),
Zhi Huang1d88d742017-11-15 15:58:49 -0800112 media_channel_(std::move(media_channel)) {
Steve Anton8699a322017-11-06 15:53:33 -0800113 RTC_DCHECK_RUN_ON(worker_thread_);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700114 demuxer_criteria_.mid = content_name;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800115 rtp_transport_ = unencrypted_rtp_transport_.get();
116 ConnectToRtpTransport();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100117 RTC_LOG(LS_INFO) << "Created channel for " << content_name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000118}
119
120BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800121 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Steve Anton8699a322017-11-06 15:53:33 -0800122 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.org78187522013-10-07 23:32:02 +0000123 Deinit();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200124 // Eats any outstanding messages or packets.
125 worker_thread_->Clear(&invoker_);
126 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000127 // We must destroy the media channel before the transport channel, otherwise
128 // the media channel may try to send on the dead transport channel. NULLing
129 // is not an effective strategy since the sends will come on another thread.
Steve Anton8699a322017-11-06 15:53:33 -0800130 media_channel_.reset();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100131 RTC_LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200132}
133
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800134void BaseChannel::ConnectToRtpTransport() {
135 RTC_DCHECK(rtp_transport_);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700136 RTC_DCHECK(RegisterRtpDemuxerSink());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800137 rtp_transport_->SignalReadyToSend.connect(
138 this, &BaseChannel::OnTransportReadyToSend);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700139 rtp_transport_->SignalRtcpPacketReceived.connect(
140 this, &BaseChannel::OnRtcpPacketReceived);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800141 rtp_transport_->SignalNetworkRouteChanged.connect(
142 this, &BaseChannel::OnNetworkRouteChanged);
143 rtp_transport_->SignalWritableState.connect(this,
144 &BaseChannel::OnWritableState);
145 rtp_transport_->SignalSentPacket.connect(this,
146 &BaseChannel::SignalSentPacket_n);
Steve Antondb67ba12018-03-19 17:41:42 -0700147 // TODO(bugs.webrtc.org/8587): Set the metrics observer through
148 // JsepTransportController once it takes responsibility for creating
149 // RtpTransports.
150 if (metrics_observer_) {
151 rtp_transport_->SetMetricsObserver(metrics_observer_);
152 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800153}
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200154
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800155void BaseChannel::DisconnectFromRtpTransport() {
156 RTC_DCHECK(rtp_transport_);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700157 rtp_transport_->UnregisterRtpDemuxerSink(this);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800158 rtp_transport_->SignalReadyToSend.disconnect(this);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700159 rtp_transport_->SignalRtcpPacketReceived.disconnect(this);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800160 rtp_transport_->SignalNetworkRouteChanged.disconnect(this);
161 rtp_transport_->SignalWritableState.disconnect(this);
162 rtp_transport_->SignalSentPacket.disconnect(this);
Steve Antondb67ba12018-03-19 17:41:42 -0700163 rtp_transport_->SetMetricsObserver(nullptr);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200164}
165
Steve Anton8699a322017-11-06 15:53:33 -0800166void BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800167 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800168 rtc::PacketTransportInternal* rtp_packet_transport,
169 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -0800170 RTC_DCHECK_RUN_ON(worker_thread_);
171 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800172 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport,
173 rtp_packet_transport, rtcp_packet_transport);
174
175 if (rtcp_mux_required_) {
176 rtcp_mux_filter_.SetActive();
177 }
Steve Anton8699a322017-11-06 15:53:33 -0800178 });
179
deadbeeff5346592017-01-24 21:51:21 -0800180 // Both RTP and RTCP channels should be set, we can call SetInterface on
181 // the media channel and it can set network options.
wu@webrtc.orgde305012013-10-31 15:40:38 +0000182 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000183}
184
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800185void BaseChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
186 RTC_DCHECK_RUN_ON(worker_thread_);
187 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
188 SetRtpTransport(rtp_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200189
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800190 if (rtcp_mux_required_) {
191 rtcp_mux_filter_.SetActive();
192 }
193 });
194
195 // Both RTP and RTCP channels should be set, we can call SetInterface on
196 // the media channel and it can set network options.
197 media_channel_->SetInterface(this);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200198}
199
wu@webrtc.org78187522013-10-07 23:32:02 +0000200void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200201 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000202 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200203 // Packets arrive on the network thread, processing packets calls virtual
204 // functions, so need to stop this process in Deinit that is called in
205 // derived classes destructor.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800206 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
Zhi Huangea8b62a2018-03-26 14:31:17 -0700207 if (rtp_transport_) {
208 FlushRtcpMessages_n();
209 if (dtls_srtp_transport_) {
210 dtls_srtp_transport_->SetDtlsTransports(nullptr, nullptr);
211 } else {
212 rtp_transport_->SetRtpPacketTransport(nullptr);
213 rtp_transport_->SetRtcpPacketTransport(nullptr);
214 }
215 DisconnectFromRtpTransport();
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800216 }
Zhi Huangea8b62a2018-03-26 14:31:17 -0700217
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800218 // Clear pending read packets/messages.
219 network_thread_->Clear(&invoker_);
220 network_thread_->Clear(this);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700221 // Because RTP level transports are accessed from the |network_thread_|,
222 // it's safer to release them from the |network_thread_| as well.
223 unencrypted_rtp_transport_.reset();
224 sdes_transport_.reset();
225 dtls_srtp_transport_.reset();
226 rtp_transport_ = nullptr;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800227 });
wu@webrtc.org78187522013-10-07 23:32:02 +0000228}
229
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800230void BaseChannel::SetRtpTransport(webrtc::RtpTransportInternal* rtp_transport) {
231 if (!network_thread_->IsCurrent()) {
232 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
233 SetRtpTransport(rtp_transport);
234 return;
235 });
236 }
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800237 RTC_DCHECK(rtp_transport);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700238 if (rtp_transport == rtp_transport_) {
239 return;
240 }
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800241
242 if (rtp_transport_) {
243 DisconnectFromRtpTransport();
244 }
245 rtp_transport_ = rtp_transport;
246 RTC_LOG(LS_INFO) << "Setting the RtpTransport for " << content_name();
247 ConnectToRtpTransport();
248
249 UpdateWritableState_n();
250}
251
zhihuangb2cdd932017-01-19 16:54:25 -0800252void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
253 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800254 network_thread_->Invoke<void>(
255 RTC_FROM_HERE,
256 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
257 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000258}
259
deadbeeff5346592017-01-24 21:51:21 -0800260void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800261 rtc::PacketTransportInternal* rtp_packet_transport,
262 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800263 network_thread_->Invoke<void>(
264 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
265 rtp_packet_transport, rtcp_packet_transport));
266}
zhihuangf5b251b2017-01-12 19:37:48 -0800267
deadbeeff5346592017-01-24 21:51:21 -0800268void BaseChannel::SetTransports_n(
269 DtlsTransportInternal* rtp_dtls_transport,
270 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800271 rtc::PacketTransportInternal* rtp_packet_transport,
272 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800273 RTC_DCHECK(network_thread_->IsCurrent());
274 // Validate some assertions about the input.
275 RTC_DCHECK(rtp_packet_transport);
276 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
277 if (rtp_dtls_transport || rtcp_dtls_transport) {
278 // DTLS/non-DTLS pointers should be to the same object.
279 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
280 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
281 // Can't go from non-DTLS to DTLS.
zsteine8ab5432017-07-12 11:48:11 -0700282 RTC_DCHECK(!rtp_transport_->rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800283 } else {
284 // Can't go from DTLS to non-DTLS.
285 RTC_DCHECK(!rtp_dtls_transport_);
286 }
287 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800288 if (rtp_dtls_transport && rtcp_dtls_transport) {
289 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
290 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800291 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800292
293 if (rtp_packet_transport == rtp_transport_->rtp_packet_transport()) {
294 // Nothing to do if transport isn't changing.
295 return;
296 }
297
deadbeeff5346592017-01-24 21:51:21 -0800298 std::string debug_name;
299 if (rtp_dtls_transport) {
300 transport_name_ = rtp_dtls_transport->transport_name();
301 debug_name = transport_name_;
302 } else {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800303 debug_name = rtp_packet_transport->transport_name();
deadbeeff5346592017-01-24 21:51:21 -0800304 }
deadbeefac22f702017-01-12 21:59:29 -0800305 // If this BaseChannel doesn't require RTCP mux and we haven't fully
306 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800307 if (rtcp_packet_transport) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100308 RTC_LOG(LS_INFO) << "Setting RTCP Transport for " << content_name()
309 << " on " << debug_name << " transport "
310 << rtcp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800311 SetTransport_n(/*rtcp=*/true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000312 }
313
Mirko Bonadei675513b2017-11-09 11:09:25 +0100314 RTC_LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
315 << debug_name << " transport " << rtp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800316 SetTransport_n(/*rtcp=*/false, rtp_dtls_transport, rtp_packet_transport);
317
318 // Set DtlsTransport/PacketTransport for RTP-level transport.
319 if ((rtp_dtls_transport_ || rtcp_dtls_transport_) && dtls_srtp_transport_) {
320 // When setting the transport with non-null |dtls_srtp_transport_|, we are
321 // using DTLS-SRTP. This could happen for bundling. If the
322 // |dtls_srtp_transport| is null, we cannot tell if it doing DTLS-SRTP or
323 // SDES until the description is set. So don't call |EnableDtlsSrtp_n| here.
324 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport,
325 rtcp_dtls_transport);
326 } else {
327 rtp_transport_->SetRtpPacketTransport(rtp_packet_transport);
328 rtp_transport_->SetRtcpPacketTransport(rtcp_packet_transport);
329 }
guoweis46383312015-12-17 16:45:59 -0800330
deadbeefcbecd352015-09-23 11:50:27 -0700331 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700332 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200333 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000334}
335
deadbeeff5346592017-01-24 21:51:21 -0800336void BaseChannel::SetTransport_n(
337 bool rtcp,
338 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800339 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200340 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800341 if (new_dtls_transport) {
342 RTC_DCHECK(new_dtls_transport == new_packet_transport);
343 }
deadbeeff5346592017-01-24 21:51:21 -0800344 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800345 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700346 rtc::PacketTransportInternal* old_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700347 rtcp ? rtp_transport_->rtcp_packet_transport()
348 : rtp_transport_->rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800349
deadbeeff5346592017-01-24 21:51:21 -0800350 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700351 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000352 return;
353 }
zhihuangb2cdd932017-01-19 16:54:25 -0800354
deadbeeff5346592017-01-24 21:51:21 -0800355 RTC_DCHECK(old_packet_transport != new_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000356
deadbeeff5346592017-01-24 21:51:21 -0800357 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000358
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800359 // If there's no new transport, we're done.
deadbeeff5346592017-01-24 21:51:21 -0800360 if (!new_packet_transport) {
361 return;
362 }
363
364 if (rtcp && new_dtls_transport) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700365 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_active()))
366 << "Setting RTCP for DTLS/SRTP after the DTLS is active "
deadbeeff5346592017-01-24 21:51:21 -0800367 << "should never happen.";
368 }
zstein56162b92017-04-24 16:54:35 -0700369
deadbeeff5346592017-01-24 21:51:21 -0800370 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
371 for (const auto& pair : socket_options) {
372 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800373 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000374}
375
Steve Antondb67ba12018-03-19 17:41:42 -0700376void BaseChannel::SetMetricsObserver(
377 rtc::scoped_refptr<webrtc::MetricsObserverInterface> metrics_observer) {
378 metrics_observer_ = metrics_observer;
379 if (rtp_transport_) {
380 rtp_transport_->SetMetricsObserver(metrics_observer);
381 }
382}
383
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000384bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700385 worker_thread_->Invoke<void>(
386 RTC_FROM_HERE,
387 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
388 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000389 return true;
390}
391
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000392bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700393 return InvokeOnWorker<bool>(RTC_FROM_HERE,
394 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000395}
396
Peter Boström0c4e06b2015-10-07 12:23:21 +0200397bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700398 return InvokeOnWorker<bool>(
399 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000400}
401
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000402bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700403 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700404 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000405}
406
Peter Boström0c4e06b2015-10-07 12:23:21 +0200407bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700408 return InvokeOnWorker<bool>(
409 RTC_FROM_HERE,
410 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000411}
412
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000413bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800414 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000415 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100416 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700417 return InvokeOnWorker<bool>(
418 RTC_FROM_HERE,
Steve Anton3828c062017-12-06 10:34:51 -0800419 Bind(&BaseChannel::SetLocalContent_w, this, content, type, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000420}
421
422bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800423 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000424 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100425 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700426 return InvokeOnWorker<bool>(
Steve Anton3828c062017-12-06 10:34:51 -0800427 RTC_FROM_HERE,
428 Bind(&BaseChannel::SetRemoteContent_w, this, content, type, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000429}
430
zhihuangf5b251b2017-01-12 19:37:48 -0800431bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800432 // If this BaseChannel doesn't require RTCP mux and we haven't fully
433 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700434 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000435}
436
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700437bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000438 // Receive data if we are enabled and have local content,
Steve Anton4e70a722017-11-28 14:57:10 -0800439 return enabled() &&
440 webrtc::RtpTransceiverDirectionHasRecv(local_content_direction_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000441}
442
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700443bool BaseChannel::IsReadyToSendMedia_w() const {
444 // Need to access some state updated on the network thread.
445 return network_thread_->Invoke<bool>(
446 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
447}
448
449bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000450 // Send outgoing data if we are enabled, have local and remote content,
451 // and we have had some form of connectivity.
Steve Anton4e70a722017-11-28 14:57:10 -0800452 return enabled() &&
453 webrtc::RtpTransceiverDirectionHasRecv(remote_content_direction_) &&
454 webrtc::RtpTransceiverDirectionHasSend(local_content_direction_) &&
Zhi Huangcf990f52017-09-22 12:12:30 -0700455 was_ever_writable() && (srtp_active() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000456}
457
jbaucheec21bd2016-03-20 06:15:43 -0700458bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700459 const rtc::PacketOptions& options) {
460 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000461}
462
jbaucheec21bd2016-03-20 06:15:43 -0700463bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700464 const rtc::PacketOptions& options) {
465 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000466}
467
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000468int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000469 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200470 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700471 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200472}
473
474int BaseChannel::SetOption_n(SocketType type,
475 rtc::Socket::Option opt,
476 int value) {
477 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800478 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000479 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000480 case ST_RTP:
zsteine8ab5432017-07-12 11:48:11 -0700481 transport = rtp_transport_->rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700482 socket_options_.push_back(
483 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000484 break;
485 case ST_RTCP:
zsteine8ab5432017-07-12 11:48:11 -0700486 transport = rtp_transport_->rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700487 rtcp_socket_options_.push_back(
488 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000489 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000490 }
deadbeeff5346592017-01-24 21:51:21 -0800491 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000492}
493
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800494void BaseChannel::OnWritableState(bool writable) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200495 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800496 if (writable) {
497 // This is used to cover the scenario when the DTLS handshake is completed
498 // and DtlsTransport becomes writable before the remote description is set.
499 if (ShouldSetupDtlsSrtp_n()) {
500 EnableDtlsSrtp_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700501 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800502 ChannelWritable_n();
503 } else {
504 ChannelNotWritable_n();
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800505 }
506}
507
Zhi Huang942bc2e2017-11-13 13:26:07 -0800508void BaseChannel::OnNetworkRouteChanged(
509 rtc::Optional<rtc::NetworkRoute> network_route) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200510 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800511 rtc::NetworkRoute new_route;
512 if (network_route) {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800513 new_route = *(network_route);
Zhi Huang8c316c12017-11-13 21:13:45 +0000514 }
Zhi Huang942bc2e2017-11-13 13:26:07 -0800515 // Note: When the RTCP-muxing is not enabled, RTCP transport and RTP transport
516 // use the same transport name and MediaChannel::OnNetworkRouteChanged cannot
517 // work correctly. Intentionally leave it broken to simplify the code and
518 // encourage the users to stop using non-muxing RTCP.
Steve Anton8699a322017-11-06 15:53:33 -0800519 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_, [=] {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800520 media_channel_->OnNetworkRouteChanged(transport_name_, new_route);
Steve Anton8699a322017-11-06 15:53:33 -0800521 });
Honghai Zhangcc411c02016-03-29 17:27:21 -0700522}
523
zstein56162b92017-04-24 16:54:35 -0700524void BaseChannel::OnTransportReadyToSend(bool ready) {
Steve Anton8699a322017-11-06 15:53:33 -0800525 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_,
526 [=] { media_channel_->OnReadyToSend(ready); });
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000527}
528
stefanc1aeaf02015-10-15 07:26:07 -0700529bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700530 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700531 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200532 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
533 // If the thread is not our network thread, we will post to our network
534 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000535 // synchronize access to all the pieces of the send path, including
536 // SRTP and the inner workings of the transport channels.
537 // The only downside is that we can't return a proper failure code if
538 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200539 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000540 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200541 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
542 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800543 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700544 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700545 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000546 return true;
547 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200548 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000549
550 // Now that we are on the correct thread, ensure we have a place to send this
551 // packet before doing anything. (We might get RTCP packets that we don't
552 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
553 // transport.
zsteine8ab5432017-07-12 11:48:11 -0700554 if (!rtp_transport_->IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000555 return false;
556 }
557
558 // Protect ourselves against crazy data.
559 if (!ValidPacket(rtcp, packet)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100560 RTC_LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
561 << RtpRtcpStringLiteral(rtcp)
562 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000563 return false;
564 }
565
Zhi Huangcf990f52017-09-22 12:12:30 -0700566 if (!srtp_active()) {
567 if (srtp_required_) {
568 // The audio/video engines may attempt to send RTCP packets as soon as the
569 // streams are created, so don't treat this as an error for RTCP.
570 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
571 if (rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000572 return false;
573 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700574 // However, there shouldn't be any RTP packets sent before SRTP is set up
575 // (and SetSend(true) is called).
Mirko Bonadei675513b2017-11-09 11:09:25 +0100576 RTC_LOG(LS_ERROR)
577 << "Can't send outgoing RTP packet when SRTP is inactive"
578 << " and crypto is required";
Zhi Huangcf990f52017-09-22 12:12:30 -0700579 RTC_NOTREACHED();
deadbeef8f425f92016-12-01 12:26:27 -0800580 return false;
581 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800582
583 std::string packet_type = rtcp ? "RTCP" : "RTP";
584 RTC_LOG(LS_WARNING) << "Sending an " << packet_type
585 << " packet without encryption.";
586 } else {
587 // Make sure we didn't accidentally send any packets without encryption.
588 RTC_DCHECK(rtp_transport_ == sdes_transport_.get() ||
589 rtp_transport_ == dtls_srtp_transport_.get());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000590 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000591 // Bon voyage.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800592 return rtcp ? rtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
593 : rtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000594}
595
Zhi Huangea8b62a2018-03-26 14:31:17 -0700596void BaseChannel::OnRtpPacket(const webrtc::RtpPacketReceived& parsed_packet) {
597 // Reconstruct the PacketTime from the |parsed_packet|.
598 // RtpPacketReceived.arrival_time_ms = (PacketTime + 500) / 1000;
599 // Note: The |not_before| field is always 0 here. This field is not currently
600 // used, so it should be fine.
601 int64_t timestamp = -1;
602 if (parsed_packet.arrival_time_ms() > 0) {
603 timestamp = parsed_packet.arrival_time_ms() * 1000;
604 }
605 rtc::PacketTime packet_time(timestamp, /*not_before=*/0);
606 OnPacketReceived(/*rtcp=*/false, parsed_packet.Buffer(), packet_time);
607}
608
609void BaseChannel::UpdateRtpHeaderExtensionMap(
610 const RtpHeaderExtensions& header_extensions) {
611 RTC_DCHECK(rtp_transport_);
612 rtp_transport_->UpdateRtpHeaderExtensionMap(header_extensions);
613}
614
615bool BaseChannel::RegisterRtpDemuxerSink() {
616 RTC_DCHECK(rtp_transport_);
617 return rtp_transport_->RegisterRtpDemuxerSink(demuxer_criteria_, this);
618}
619
620void BaseChannel::OnRtcpPacketReceived(rtc::CopyOnWriteBuffer* packet,
621 const rtc::PacketTime& packet_time) {
622 OnPacketReceived(/*rtcp=*/true, *packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000623}
624
zstein3dcf0e92017-06-01 13:22:42 -0700625void BaseChannel::OnPacketReceived(bool rtcp,
Zhi Huangea8b62a2018-03-26 14:31:17 -0700626 const rtc::CopyOnWriteBuffer& packet,
zstein3dcf0e92017-06-01 13:22:42 -0700627 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000628 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000629 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700630 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000631 }
632
Zhi Huangcf990f52017-09-22 12:12:30 -0700633 if (!srtp_active() && srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000634 // Our session description indicates that SRTP is required, but we got a
635 // packet before our SRTP filter is active. This means either that
636 // a) we got SRTP packets before we received the SDES keys, in which case
637 // we can't decrypt it anyway, or
638 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800639 // transports, so we haven't yet extracted keys, even if DTLS did
640 // complete on the transport that the packets are being sent on. It's
641 // really good practice to wait for both RTP and RTCP to be good to go
642 // before sending media, to prevent weird failure modes, so it's fine
643 // for us to just eat packets here. This is all sidestepped if RTCP mux
644 // is used anyway.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100645 RTC_LOG(LS_WARNING)
646 << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
647 << " packet when SRTP is inactive and crypto is required";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000648 return;
649 }
650
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200651 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700652 RTC_FROM_HERE, worker_thread_,
Zhi Huangea8b62a2018-03-26 14:31:17 -0700653 Bind(&BaseChannel::ProcessPacket, this, rtcp, packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200654}
655
zstein3dcf0e92017-06-01 13:22:42 -0700656void BaseChannel::ProcessPacket(bool rtcp,
657 const rtc::CopyOnWriteBuffer& packet,
658 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200659 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700660
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200661 // Need to copy variable because OnRtcpReceived/OnPacketReceived
662 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
663 rtc::CopyOnWriteBuffer data(packet);
664 if (rtcp) {
665 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000666 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200667 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000668 }
669}
670
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000671void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700672 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000673 if (enabled_)
674 return;
675
Mirko Bonadei675513b2017-11-09 11:09:25 +0100676 RTC_LOG(LS_INFO) << "Channel enabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000677 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700678 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000679}
680
681void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700682 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000683 if (!enabled_)
684 return;
685
Mirko Bonadei675513b2017-11-09 11:09:25 +0100686 RTC_LOG(LS_INFO) << "Channel disabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000687 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700688 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000689}
690
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200691void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700692 rtc::PacketTransportInternal* rtp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700693 rtp_transport_->rtp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700694 rtc::PacketTransportInternal* rtcp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700695 rtp_transport_->rtcp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700696 if (rtp_packet_transport && rtp_packet_transport->writable() &&
697 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200698 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700699 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200700 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700701 }
702}
703
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200704void BaseChannel::ChannelWritable_n() {
705 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800706 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000707 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800708 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000709
Mirko Bonadei675513b2017-11-09 11:09:25 +0100710 RTC_LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
711 << (was_ever_writable_ ? "" : " for the first time");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000712
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000713 was_ever_writable_ = true;
714 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700715 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000716}
717
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200718bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800719 // Since DTLS is applied to all transports, checking RTP should be enough.
720 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000721}
722
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200723void BaseChannel::ChannelNotWritable_n() {
724 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000725 if (!writable_)
726 return;
727
Mirko Bonadei675513b2017-11-09 11:09:25 +0100728 RTC_LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000729 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700730 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000731}
732
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200733bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700734 const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800735 SdpType type,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700736 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700737 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700738 std::string* error_desc) {
jbauch5869f502017-06-29 12:31:36 -0700739 std::vector<int> encrypted_extension_ids;
740 for (const webrtc::RtpExtension& extension : extensions) {
741 if (extension.encrypt) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100742 RTC_LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
743 << " encrypted extension: " << extension.ToString();
jbauch5869f502017-06-29 12:31:36 -0700744 encrypted_extension_ids.push_back(extension.id);
745 }
746 }
747
deadbeef7af91dd2016-12-13 11:29:11 -0800748 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200749 return network_thread_->Invoke<bool>(
Steve Anton3828c062017-12-06 10:34:51 -0800750 RTC_FROM_HERE,
751 Bind(&BaseChannel::SetRtpTransportParameters_n, this, content, type, src,
752 encrypted_extension_ids, error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200753}
754
755bool BaseChannel::SetRtpTransportParameters_n(
756 const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800757 SdpType type,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200758 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700759 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200760 std::string* error_desc) {
761 RTC_DCHECK(network_thread_->IsCurrent());
762
Steve Anton3828c062017-12-06 10:34:51 -0800763 if (!SetSrtp_n(content->cryptos(), type, src, encrypted_extension_ids,
764 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700765 return false;
766 }
767
Steve Anton3828c062017-12-06 10:34:51 -0800768 if (!SetRtcpMux_n(content->rtcp_mux(), type, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700769 return false;
770 }
771
772 return true;
773}
774
zhihuangb2cdd932017-01-19 16:54:25 -0800775// |dtls| will be set to true if DTLS is active for transport and crypto is
776// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200777bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
778 bool* dtls,
779 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -0800780 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000781 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200782 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000783 return false;
784 }
785 return true;
786}
787
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800788void BaseChannel::EnableSdes_n() {
789 if (sdes_transport_) {
790 return;
Zhi Huangcf990f52017-09-22 12:12:30 -0700791 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800792 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
793 // time.
794 RTC_DCHECK(!dtls_srtp_transport_);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700795
796 sdes_transport_ = rtc::MakeUnique<webrtc::SrtpTransport>(rtcp_mux_required_);
Zhi Huangd7455782017-11-30 14:50:52 -0800797#if defined(ENABLE_EXTERNAL_AUTH)
798 sdes_transport_->EnableExternalAuth();
799#endif
Zhi Huangea8b62a2018-03-26 14:31:17 -0700800 sdes_transport_->SetRtpPacketTransport(
801 rtp_transport_->rtp_packet_transport());
802 sdes_transport_->SetRtcpPacketTransport(
803 rtp_transport_->rtcp_packet_transport());
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800804 SetRtpTransport(sdes_transport_.get());
Zhi Huangea8b62a2018-03-26 14:31:17 -0700805 RTC_LOG(LS_INFO) << "SrtpTransport is created for SDES.";
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800806}
807
808void BaseChannel::EnableDtlsSrtp_n() {
809 if (dtls_srtp_transport_) {
810 return;
811 }
812 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
813 // time.
814 RTC_DCHECK(!sdes_transport_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800815
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800816 dtls_srtp_transport_ =
Zhi Huangea8b62a2018-03-26 14:31:17 -0700817 rtc::MakeUnique<webrtc::DtlsSrtpTransport>(rtcp_mux_required_);
818#if defined(ENABLE_EXTERNAL_AUTH)
819 dtls_srtp_transport_->EnableExternalAuth();
820#endif
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800821
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800822 SetRtpTransport(dtls_srtp_transport_.get());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800823 if (cached_send_extension_ids_) {
824 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
825 *cached_send_extension_ids_);
826 }
827 if (cached_recv_extension_ids_) {
828 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
829 *cached_recv_extension_ids_);
830 }
831 // Set the DtlsTransport and the |dtls_srtp_transport_| will handle the DTLS
832 // relate signal internally.
833 RTC_DCHECK(rtp_dtls_transport_);
834 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
835 rtcp_dtls_transport_);
Zhi Huangea8b62a2018-03-26 14:31:17 -0700836 RTC_LOG(LS_INFO) << "DtlsSrtpTransport is created for DTLS-SRTP.";
Zhi Huangcf990f52017-09-22 12:12:30 -0700837}
838
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200839bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
Steve Anton3828c062017-12-06 10:34:51 -0800840 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000841 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700842 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000843 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -0800844 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000845 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000846 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200847 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000848 if (!ret) {
849 return false;
850 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700851
852 // If SRTP was not required, but we're setting a description that uses SDES,
853 // we need to upgrade to an SrtpTransport.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800854 if (!sdes_transport_ && !dtls && !cryptos.empty()) {
855 EnableSdes_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700856 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800857
Steve Anton3828c062017-12-06 10:34:51 -0800858 if ((type == SdpType::kAnswer || type == SdpType::kPrAnswer) && dtls) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800859 EnableDtlsSrtp_n();
860 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800861
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800862 UpdateEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
863
864 if (!dtls) {
Steve Anton3828c062017-12-06 10:34:51 -0800865 switch (type) {
866 case SdpType::kOffer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700867 ret = sdes_negotiator_.SetOffer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800868 break;
Steve Anton3828c062017-12-06 10:34:51 -0800869 case SdpType::kPrAnswer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700870 ret = sdes_negotiator_.SetProvisionalAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800871 break;
Steve Anton3828c062017-12-06 10:34:51 -0800872 case SdpType::kAnswer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700873 ret = sdes_negotiator_.SetAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800874 break;
875 default:
876 break;
877 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700878
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800879 // If setting an SDES answer succeeded, apply the negotiated parameters
880 // to the SRTP transport.
Steve Anton3828c062017-12-06 10:34:51 -0800881 if ((type == SdpType::kPrAnswer || type == SdpType::kAnswer) && ret) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800882 if (sdes_negotiator_.send_cipher_suite() &&
883 sdes_negotiator_.recv_cipher_suite()) {
884 RTC_DCHECK(cached_send_extension_ids_);
885 RTC_DCHECK(cached_recv_extension_ids_);
886 ret = sdes_transport_->SetRtpParams(
887 *(sdes_negotiator_.send_cipher_suite()),
888 sdes_negotiator_.send_key().data(),
889 static_cast<int>(sdes_negotiator_.send_key().size()),
890 *(cached_send_extension_ids_),
891 *(sdes_negotiator_.recv_cipher_suite()),
892 sdes_negotiator_.recv_key().data(),
893 static_cast<int>(sdes_negotiator_.recv_key().size()),
894 *(cached_recv_extension_ids_));
895 } else {
896 RTC_LOG(LS_INFO) << "No crypto keys are provided for SDES.";
Steve Anton3828c062017-12-06 10:34:51 -0800897 if (type == SdpType::kAnswer && sdes_transport_) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800898 // Explicitly reset the |sdes_transport_| if no crypto param is
899 // provided in the answer. No need to call |ResetParams()| for
900 // |sdes_negotiator_| because it resets the params inside |SetAnswer|.
901 sdes_transport_->ResetParams();
902 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700903 }
904 }
905 }
906
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000907 if (!ret) {
Zhi Huangc99b6c72017-11-10 16:44:46 -0800908 SafeSetError("Failed to setup SRTP.", error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000909 return false;
910 }
911 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000912}
913
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200914bool BaseChannel::SetRtcpMux_n(bool enable,
Steve Anton3828c062017-12-06 10:34:51 -0800915 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000916 ContentSource src,
917 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -0800918 // Provide a more specific error message for the RTCP mux "require" policy
919 // case.
zstein56162b92017-04-24 16:54:35 -0700920 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -0800921 SafeSetError(
922 "rtcpMuxPolicy is 'require', but media description does not "
923 "contain 'a=rtcp-mux'.",
924 error_desc);
925 return false;
926 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000927 bool ret = false;
Steve Anton3828c062017-12-06 10:34:51 -0800928 switch (type) {
929 case SdpType::kOffer:
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000930 ret = rtcp_mux_filter_.SetOffer(enable, src);
931 break;
Steve Anton3828c062017-12-06 10:34:51 -0800932 case SdpType::kPrAnswer:
zhihuangb2cdd932017-01-19 16:54:25 -0800933 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700934 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000935 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
936 break;
Steve Anton3828c062017-12-06 10:34:51 -0800937 case SdpType::kAnswer:
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000938 ret = rtcp_mux_filter_.SetAnswer(enable, src);
939 if (ret && rtcp_mux_filter_.IsActive()) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800940 ActivateRtcpMux();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000941 }
942 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000943 default:
944 break;
945 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000946 if (!ret) {
947 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
948 return false;
949 }
zsteine8ab5432017-07-12 11:48:11 -0700950 rtp_transport_->SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
Steve Anton3828c062017-12-06 10:34:51 -0800951 // |rtcp_mux_filter_| can be active if |action| is SdpType::kPrAnswer or
952 // SdpType::kAnswer, but we only want to tear down the RTCP transport if we
953 // received a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000954 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000955 // If the RTP transport is already writable, then so are we.
zsteine8ab5432017-07-12 11:48:11 -0700956 if (rtp_transport_->rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200957 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000958 }
959 }
960
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000961 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000962}
963
964bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700965 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -0800966 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000967}
968
Peter Boström0c4e06b2015-10-07 12:23:21 +0200969bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700970 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000971 return media_channel()->RemoveRecvStream(ssrc);
972}
973
974bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
Steve Anton3828c062017-12-06 10:34:51 -0800975 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000976 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000977 // Check for streams that have been removed.
978 bool ret = true;
979 for (StreamParamsVec::const_iterator it = local_streams_.begin();
980 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +0000981 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000982 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000983 std::ostringstream desc;
984 desc << "Failed to remove send stream with ssrc "
985 << it->first_ssrc() << ".";
986 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000987 ret = false;
988 }
989 }
990 }
991 // Check for new streams.
992 for (StreamParamsVec::const_iterator it = streams.begin();
993 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +0000994 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000995 if (media_channel()->AddSendStream(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100996 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000997 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000998 std::ostringstream desc;
999 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1000 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001001 ret = false;
1002 }
1003 }
1004 }
1005 local_streams_ = streams;
1006 return ret;
1007}
1008
1009bool BaseChannel::UpdateRemoteStreams_w(
1010 const std::vector<StreamParams>& streams,
Steve Anton3828c062017-12-06 10:34:51 -08001011 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001012 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001013 // Check for streams that have been removed.
1014 bool ret = true;
1015 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1016 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001017 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001018 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001019 std::ostringstream desc;
1020 desc << "Failed to remove remote stream with ssrc "
1021 << it->first_ssrc() << ".";
1022 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001023 ret = false;
1024 }
1025 }
1026 }
1027 // Check for new streams.
1028 for (StreamParamsVec::const_iterator it = streams.begin();
1029 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001030 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001031 if (AddRecvStream_w(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001032 RTC_LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001033 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001034 std::ostringstream desc;
1035 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1036 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001037 ret = false;
1038 }
1039 }
1040 }
1041 remote_streams_ = streams;
1042 return ret;
1043}
1044
jbauch5869f502017-06-29 12:31:36 -07001045RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1046 const RtpHeaderExtensions& extensions) {
1047 if (!rtp_dtls_transport_ ||
1048 !rtp_dtls_transport_->crypto_options()
1049 .enable_encrypted_rtp_header_extensions) {
1050 RtpHeaderExtensions filtered;
1051 auto pred = [](const webrtc::RtpExtension& extension) {
1052 return !extension.encrypt;
1053 };
1054 std::copy_if(extensions.begin(), extensions.end(),
1055 std::back_inserter(filtered), pred);
1056 return filtered;
1057 }
1058
1059 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1060}
1061
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001062void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001063 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001064// Absolute Send Time extension id is used only with external auth,
1065// so do not bother searching for it and making asyncronious call to set
1066// something that is not used.
1067#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001068 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001069 webrtc::RtpExtension::FindHeaderExtensionByUri(
1070 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001071 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001072 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001073 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001074 RTC_FROM_HERE, network_thread_,
1075 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1076 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001077#endif
1078}
1079
1080void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1081 int rtp_abs_sendtime_extn_id) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001082 if (sdes_transport_) {
1083 sdes_transport_->CacheRtpAbsSendTimeHeaderExtension(
Zhi Huangcf990f52017-09-22 12:12:30 -07001084 rtp_abs_sendtime_extn_id);
Zhi Huang2a4d70c2017-11-29 15:41:59 -08001085 } else if (dtls_srtp_transport_) {
1086 dtls_srtp_transport_->CacheRtpAbsSendTimeHeaderExtension(
1087 rtp_abs_sendtime_extn_id);
Zhi Huangcf990f52017-09-22 12:12:30 -07001088 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001089 RTC_LOG(LS_WARNING)
1090 << "Trying to cache the Absolute Send Time extension id "
1091 "but the SRTP is not active.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001092 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001093}
1094
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001095void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001096 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001097 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001098 case MSG_SEND_RTP_PACKET:
1099 case MSG_SEND_RTCP_PACKET: {
1100 RTC_DCHECK(network_thread_->IsCurrent());
1101 SendPacketMessageData* data =
1102 static_cast<SendPacketMessageData*>(pmsg->pdata);
1103 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1104 SendPacket(rtcp, &data->packet, data->options);
1105 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001106 break;
1107 }
1108 case MSG_FIRSTPACKETRECEIVED: {
1109 SignalFirstPacketReceived(this);
1110 break;
1111 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001112 }
1113}
1114
zstein3dcf0e92017-06-01 13:22:42 -07001115void BaseChannel::AddHandledPayloadType(int payload_type) {
Zhi Huangea8b62a2018-03-26 14:31:17 -07001116 demuxer_criteria_.payload_types.insert(static_cast<uint8_t>(payload_type));
zstein3dcf0e92017-06-01 13:22:42 -07001117}
1118
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001119void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001120 // Flush all remaining RTCP messages. This should only be called in
1121 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001122 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001123 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001124 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1125 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001126 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1127 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001128 }
1129}
1130
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001131void BaseChannel::SignalSentPacket_n(const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001132 RTC_DCHECK(network_thread_->IsCurrent());
1133 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001134 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001135 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1136}
1137
1138void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1139 RTC_DCHECK(worker_thread_->IsCurrent());
1140 SignalSentPacket(sent_packet);
1141}
1142
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001143void BaseChannel::UpdateEncryptedHeaderExtensionIds(
Zhi Huangc99b6c72017-11-10 16:44:46 -08001144 cricket::ContentSource source,
1145 const std::vector<int>& extension_ids) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001146 if (source == ContentSource::CS_LOCAL) {
1147 cached_recv_extension_ids_ = std::move(extension_ids);
1148 if (dtls_srtp_transport_) {
1149 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
1150 extension_ids);
1151 }
1152 } else {
1153 cached_send_extension_ids_ = std::move(extension_ids);
1154 if (dtls_srtp_transport_) {
1155 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
1156 extension_ids);
1157 }
1158 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001159}
1160
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001161void BaseChannel::ActivateRtcpMux() {
1162 // We permanently activated RTCP muxing; signal that we no longer need
1163 // the RTCP transport.
1164 std::string debug_name =
1165 transport_name_.empty()
1166 ? rtp_transport_->rtp_packet_transport()->transport_name()
1167 : transport_name_;
1168 RTC_LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1169 << "; no longer need RTCP transport for " << debug_name;
1170 if (rtp_transport_->rtcp_packet_transport()) {
1171 SetTransport_n(/*rtcp=*/true, nullptr, nullptr);
1172 if (dtls_srtp_transport_) {
1173 RTC_DCHECK(rtp_dtls_transport_);
1174 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
1175 /*rtcp_dtls_transport_=*/nullptr);
1176 } else {
1177 rtp_transport_->SetRtcpPacketTransport(nullptr);
1178 }
1179 SignalRtcpMuxFullyActive(transport_name_);
Zhi Huangc99b6c72017-11-10 16:44:46 -08001180 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001181 UpdateWritableState_n();
Zhi Huangc99b6c72017-11-10 16:44:46 -08001182}
1183
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001184VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1185 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001186 rtc::Thread* signaling_thread,
Niels Möllerf120cba2018-01-30 09:33:03 +01001187 // TODO(nisse): Delete unused argument.
1188 MediaEngineInterface* /* media_engine */,
Steve Anton8699a322017-11-06 15:53:33 -08001189 std::unique_ptr<VoiceMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001190 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001191 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001192 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001193 : BaseChannel(worker_thread,
1194 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001195 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001196 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001197 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001198 rtcp_mux_required,
Niels Möllerf120cba2018-01-30 09:33:03 +01001199 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001200
1201VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001202 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001203 // this can't be done in the base class, since it calls a virtual
1204 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001205 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001206}
1207
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001208void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001209 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001210 invoker_.AsyncInvoke<void>(
1211 RTC_FROM_HERE, worker_thread_,
1212 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001213}
1214
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001215void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001216 // Render incoming data if we're the active call, and we have the local
1217 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001218 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001219 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001220
1221 // Send outgoing data if we're the active call, we have the remote content,
1222 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001223 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001224 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001225
Mirko Bonadei675513b2017-11-09 11:09:25 +01001226 RTC_LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001227}
1228
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001229bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001230 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001231 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001232 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001233 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001234 RTC_LOG(LS_INFO) << "Setting local voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001235
Steve Antonb1c1de12017-12-21 15:14:30 -08001236 RTC_DCHECK(content);
1237 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001238 SafeSetError("Can't find audio content in local description.", error_desc);
1239 return false;
1240 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001241
Steve Antonb1c1de12017-12-21 15:14:30 -08001242 const AudioContentDescription* audio = content->as_audio();
1243
jbauch5869f502017-06-29 12:31:36 -07001244 RtpHeaderExtensions rtp_header_extensions =
1245 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
Zhi Huangea8b62a2018-03-26 14:31:17 -07001246 UpdateRtpHeaderExtensionMap(rtp_header_extensions);
jbauch5869f502017-06-29 12:31:36 -07001247
Steve Anton3828c062017-12-06 10:34:51 -08001248 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1249 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001250 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001251 }
1252
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001253 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001254 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001255 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001256 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001257 error_desc);
1258 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001259 }
Zhi Huangea8b62a2018-03-26 14:31:17 -07001260
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001261 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001262 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001263 }
Zhi Huangea8b62a2018-03-26 14:31:17 -07001264 // Need to re-register the sink to update the handled payload.
1265 if (!RegisterRtpDemuxerSink()) {
1266 RTC_LOG(LS_ERROR) << "Failed to set up audio demuxing.";
1267 return false;
1268 }
1269
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001270 last_recv_params_ = recv_params;
1271
1272 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1273 // only give it to the media channel once we have a remote
1274 // description too (without a remote description, we won't be able
1275 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001276 if (!UpdateLocalStreams_w(audio->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001277 SafeSetError("Failed to set local audio description streams.", error_desc);
1278 return false;
1279 }
1280
1281 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001282 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001283 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001284}
1285
1286bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001287 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001288 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001289 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001290 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001291 RTC_LOG(LS_INFO) << "Setting remote voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001292
Steve Antonb1c1de12017-12-21 15:14:30 -08001293 RTC_DCHECK(content);
1294 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001295 SafeSetError("Can't find audio content in remote description.", error_desc);
1296 return false;
1297 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001298
Steve Antonb1c1de12017-12-21 15:14:30 -08001299 const AudioContentDescription* audio = content->as_audio();
1300
jbauch5869f502017-06-29 12:31:36 -07001301 RtpHeaderExtensions rtp_header_extensions =
1302 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1303
Zhi Huangea8b62a2018-03-26 14:31:17 -07001304 UpdateRtpHeaderExtensionMap(rtp_header_extensions);
Steve Anton3828c062017-12-06 10:34:51 -08001305 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1306 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001307 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001308 }
1309
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001310 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001311 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1312 &send_params);
Steve Antonbb50ce52018-03-26 10:24:32 -07001313 send_params.mid = content_name();
skvladdc1c62c2016-03-16 19:07:43 -07001314
1315 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1316 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001317 SafeSetError("Failed to set remote audio description send parameters.",
1318 error_desc);
1319 return false;
1320 }
1321 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001322
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001323 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1324 // and only give it to the media channel once we have a local
1325 // description too (without a local description, we won't be able to
1326 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001327 if (!UpdateRemoteStreams_w(audio->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001328 SafeSetError("Failed to set remote audio description streams.", error_desc);
1329 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001330 }
1331
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001332 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001333 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001334 }
1335
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001336 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001337 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001338 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001339}
1340
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001341VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1342 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001343 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001344 std::unique_ptr<VideoMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001345 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001346 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001347 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001348 : BaseChannel(worker_thread,
1349 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001350 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001351 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001352 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001353 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001354 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001355
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001356VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001357 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001358 // this can't be done in the base class, since it calls a virtual
1359 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001360 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001361}
1362
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001363void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001364 // Send outgoing data if we're the active call, we have the remote content,
1365 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001366 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001367 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001368 RTC_LOG(LS_ERROR) << "Failed to SetSend on video channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001369 // TODO(gangji): Report error back to server.
1370 }
1371
Mirko Bonadei675513b2017-11-09 11:09:25 +01001372 RTC_LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001373}
1374
stefanf79ade12017-06-02 06:44:03 -07001375void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
1376 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
1377 media_channel(), bwe_info));
1378}
1379
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001380bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001381 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001382 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001383 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001384 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001385 RTC_LOG(LS_INFO) << "Setting local video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001386
Steve Antonb1c1de12017-12-21 15:14:30 -08001387 RTC_DCHECK(content);
1388 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001389 SafeSetError("Can't find video content in local description.", error_desc);
1390 return false;
1391 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001392
Steve Antonb1c1de12017-12-21 15:14:30 -08001393 const VideoContentDescription* video = content->as_video();
1394
jbauch5869f502017-06-29 12:31:36 -07001395 RtpHeaderExtensions rtp_header_extensions =
1396 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
Zhi Huangea8b62a2018-03-26 14:31:17 -07001397 UpdateRtpHeaderExtensionMap(rtp_header_extensions);
jbauch5869f502017-06-29 12:31:36 -07001398
Steve Anton3828c062017-12-06 10:34:51 -08001399 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1400 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001401 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001402 }
1403
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001404 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001405 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001406 if (!media_channel()->SetRecvParameters(recv_params)) {
1407 SafeSetError("Failed to set local video description recv parameters.",
1408 error_desc);
1409 return false;
1410 }
Zhi Huangea8b62a2018-03-26 14:31:17 -07001411
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001412 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001413 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001414 }
Zhi Huangea8b62a2018-03-26 14:31:17 -07001415 // Need to re-register the sink to update the handled payload.
1416 if (!RegisterRtpDemuxerSink()) {
1417 RTC_LOG(LS_ERROR) << "Failed to set up video demuxing.";
1418 return false;
1419 }
1420
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001421 last_recv_params_ = recv_params;
1422
1423 // TODO(pthatcher): Move local streams into VideoSendParameters, and
1424 // only give it to the media channel once we have a remote
1425 // description too (without a remote description, we won't be able
1426 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001427 if (!UpdateLocalStreams_w(video->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001428 SafeSetError("Failed to set local video description streams.", error_desc);
1429 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001430 }
1431
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001432 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001433 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001434 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001435}
1436
1437bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001438 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001439 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001440 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001441 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001442 RTC_LOG(LS_INFO) << "Setting remote video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001443
Steve Antonb1c1de12017-12-21 15:14:30 -08001444 RTC_DCHECK(content);
1445 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001446 SafeSetError("Can't find video content in remote description.", error_desc);
1447 return false;
1448 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001449
Steve Antonb1c1de12017-12-21 15:14:30 -08001450 const VideoContentDescription* video = content->as_video();
1451
jbauch5869f502017-06-29 12:31:36 -07001452 RtpHeaderExtensions rtp_header_extensions =
1453 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1454
Zhi Huangea8b62a2018-03-26 14:31:17 -07001455 UpdateRtpHeaderExtensionMap(rtp_header_extensions);
Steve Anton3828c062017-12-06 10:34:51 -08001456 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1457 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001458 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001459 }
1460
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001461 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001462 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
1463 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001464 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08001465 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001466 }
Steve Antonbb50ce52018-03-26 10:24:32 -07001467 send_params.mid = content_name();
skvladdc1c62c2016-03-16 19:07:43 -07001468
1469 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1470
1471 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001472 SafeSetError("Failed to set remote video description send parameters.",
1473 error_desc);
1474 return false;
1475 }
1476 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001477
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001478 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
1479 // and only give it to the media channel once we have a local
1480 // description too (without a local description, we won't be able to
1481 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001482 if (!UpdateRemoteStreams_w(video->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001483 SafeSetError("Failed to set remote video description streams.", error_desc);
1484 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001485 }
1486
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001487 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001488 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001489 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001490
1491 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001492 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001493 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001494}
1495
deadbeef953c2ce2017-01-09 14:53:41 -08001496RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
1497 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001498 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001499 std::unique_ptr<DataMediaChannel> media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08001500 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001501 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001502 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001503 : BaseChannel(worker_thread,
1504 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001505 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001506 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001507 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001508 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001509 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001510
deadbeef953c2ce2017-01-09 14:53:41 -08001511RtpDataChannel::~RtpDataChannel() {
1512 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001513 // this can't be done in the base class, since it calls a virtual
1514 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001515 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001516}
1517
Steve Anton8699a322017-11-06 15:53:33 -08001518void RtpDataChannel::Init_w(
deadbeeff5346592017-01-24 21:51:21 -08001519 DtlsTransportInternal* rtp_dtls_transport,
1520 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08001521 rtc::PacketTransportInternal* rtp_packet_transport,
1522 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -08001523 BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
1524 rtp_packet_transport, rtcp_packet_transport);
1525
deadbeef953c2ce2017-01-09 14:53:41 -08001526 media_channel()->SignalDataReceived.connect(this,
1527 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00001528 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08001529 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001530}
1531
Zhi Huang2dfc42d2017-12-04 13:38:48 -08001532void RtpDataChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
1533 BaseChannel::Init_w(rtp_transport);
1534 media_channel()->SignalDataReceived.connect(this,
1535 &RtpDataChannel::OnDataReceived);
1536 media_channel()->SignalReadyToSend.connect(
1537 this, &RtpDataChannel::OnDataChannelReadyToSend);
1538}
1539
deadbeef953c2ce2017-01-09 14:53:41 -08001540bool RtpDataChannel::SendData(const SendDataParams& params,
1541 const rtc::CopyOnWriteBuffer& payload,
1542 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07001543 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001544 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
1545 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001546}
1547
deadbeef953c2ce2017-01-09 14:53:41 -08001548bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001549 const DataContentDescription* content,
1550 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001551 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
1552 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08001553 // It's been set before, but doesn't match. That's bad.
1554 if (is_sctp) {
1555 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
1556 error_desc);
1557 return false;
1558 }
1559 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001560}
1561
deadbeef953c2ce2017-01-09 14:53:41 -08001562bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001563 SdpType type,
deadbeef953c2ce2017-01-09 14:53:41 -08001564 std::string* error_desc) {
1565 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001566 RTC_DCHECK_RUN_ON(worker_thread());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001567 RTC_LOG(LS_INFO) << "Setting local data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001568
Steve Antonb1c1de12017-12-21 15:14:30 -08001569 RTC_DCHECK(content);
1570 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001571 SafeSetError("Can't find data content in local description.", error_desc);
1572 return false;
1573 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001574
Steve Antonb1c1de12017-12-21 15:14:30 -08001575 const DataContentDescription* data = content->as_data();
1576
deadbeef953c2ce2017-01-09 14:53:41 -08001577 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001578 return false;
1579 }
1580
jbauch5869f502017-06-29 12:31:36 -07001581 RtpHeaderExtensions rtp_header_extensions =
1582 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
Zhi Huangea8b62a2018-03-26 14:31:17 -07001583 UpdateRtpHeaderExtensionMap(rtp_header_extensions);
jbauch5869f502017-06-29 12:31:36 -07001584
Steve Anton3828c062017-12-06 10:34:51 -08001585 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1586 error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001587 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001588 }
1589
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001590 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001591 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001592 if (!media_channel()->SetRecvParameters(recv_params)) {
1593 SafeSetError("Failed to set remote data description recv parameters.",
1594 error_desc);
1595 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001596 }
Zhi Huangea8b62a2018-03-26 14:31:17 -07001597
deadbeef953c2ce2017-01-09 14:53:41 -08001598 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001599 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001600 }
Zhi Huangea8b62a2018-03-26 14:31:17 -07001601 // Need to re-register the sink to update the handled payload.
1602 if (!RegisterRtpDemuxerSink()) {
1603 RTC_LOG(LS_ERROR) << "Failed to set up data demuxing.";
1604 return false;
1605 }
1606
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001607 last_recv_params_ = recv_params;
1608
1609 // TODO(pthatcher): Move local streams into DataSendParameters, and
1610 // only give it to the media channel once we have a remote
1611 // description too (without a remote description, we won't be able
1612 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001613 if (!UpdateLocalStreams_w(data->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001614 SafeSetError("Failed to set local data description streams.", error_desc);
1615 return false;
1616 }
1617
1618 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001619 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001620 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001621}
1622
deadbeef953c2ce2017-01-09 14:53:41 -08001623bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001624 SdpType type,
deadbeef953c2ce2017-01-09 14:53:41 -08001625 std::string* error_desc) {
1626 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Steve Antonb1c1de12017-12-21 15:14:30 -08001627 RTC_DCHECK_RUN_ON(worker_thread());
1628 RTC_LOG(LS_INFO) << "Setting remote data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001629
Steve Antonb1c1de12017-12-21 15:14:30 -08001630 RTC_DCHECK(content);
1631 if (!content) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001632 SafeSetError("Can't find data content in remote description.", error_desc);
1633 return false;
1634 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001635
Steve Antonb1c1de12017-12-21 15:14:30 -08001636 const DataContentDescription* data = content->as_data();
1637
Zhi Huang801b8682017-11-15 11:36:43 -08001638 // If the remote data doesn't have codecs, it must be empty, so ignore it.
1639 if (!data->has_codecs()) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001640 return true;
1641 }
1642
deadbeef953c2ce2017-01-09 14:53:41 -08001643 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001644 return false;
1645 }
1646
jbauch5869f502017-06-29 12:31:36 -07001647 RtpHeaderExtensions rtp_header_extensions =
1648 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
1649
Zhi Huangea8b62a2018-03-26 14:31:17 -07001650 UpdateRtpHeaderExtensionMap(rtp_header_extensions);
Mirko Bonadei675513b2017-11-09 11:09:25 +01001651 RTC_LOG(LS_INFO) << "Setting remote data description";
Steve Anton3828c062017-12-06 10:34:51 -08001652 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1653 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001654 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001655 }
1656
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001657 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001658 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
1659 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001660 if (!media_channel()->SetSendParameters(send_params)) {
1661 SafeSetError("Failed to set remote data description send parameters.",
1662 error_desc);
1663 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001664 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001665 last_send_params_ = send_params;
1666
1667 // TODO(pthatcher): Move remote streams into DataRecvParameters,
1668 // and only give it to the media channel once we have a local
1669 // description too (without a local description, we won't be able to
1670 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001671 if (!UpdateRemoteStreams_w(data->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001672 SafeSetError("Failed to set remote data description streams.",
1673 error_desc);
1674 return false;
1675 }
1676
1677 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001678 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001679 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001680}
1681
deadbeef953c2ce2017-01-09 14:53:41 -08001682void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001683 // Render incoming data if we're the active call, and we have the local
1684 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001685 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001686 if (!media_channel()->SetReceive(recv)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001687 RTC_LOG(LS_ERROR) << "Failed to SetReceive on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001688 }
1689
1690 // Send outgoing data if we're the active call, we have the remote content,
1691 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001692 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001693 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001694 RTC_LOG(LS_ERROR) << "Failed to SetSend on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001695 }
1696
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001697 // Trigger SignalReadyToSendData asynchronously.
1698 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001699
Mirko Bonadei675513b2017-11-09 11:09:25 +01001700 RTC_LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001701}
1702
deadbeef953c2ce2017-01-09 14:53:41 -08001703void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001704 switch (pmsg->message_id) {
1705 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00001706 DataChannelReadyToSendMessageData* data =
1707 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00001708 ready_to_send_data_ = data->data();
1709 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001710 delete data;
1711 break;
1712 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001713 case MSG_DATARECEIVED: {
1714 DataReceivedMessageData* data =
1715 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08001716 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001717 delete data;
1718 break;
1719 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001720 default:
1721 BaseChannel::OnMessage(pmsg);
1722 break;
1723 }
1724}
1725
deadbeef953c2ce2017-01-09 14:53:41 -08001726void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
1727 const char* data,
1728 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001729 DataReceivedMessageData* msg = new DataReceivedMessageData(
1730 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001731 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001732}
1733
deadbeef953c2ce2017-01-09 14:53:41 -08001734void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00001735 // This is usded for congestion control to indicate that the stream is ready
1736 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
1737 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001738 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00001739 new DataChannelReadyToSendMessageData(writable));
1740}
1741
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001742} // namespace cricket