blob: fe7357fc944cc4f028d82cae756b7f0a6b079a15 [file] [log] [blame]
zsteind48dbda2017-04-04 19:45:57 -07001/*
2 * Copyright 2017 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Steve Anton10542f22019-01-11 09:11:00 -080011#include "pc/rtp_transport.h"
zsteind48dbda2017-04-04 19:45:57 -070012
Yves Gerey3e707812018-11-28 16:47:49 +010013#include <errno.h>
Jonas Olssona4d87372019-07-05 19:08:33 +020014
Yves Gerey3e707812018-11-28 16:47:49 +010015#include <string>
Zhi Huang365381f2018-04-13 16:44:34 -070016#include <utility>
17
Yves Gerey3e707812018-11-28 16:47:49 +010018#include "api/rtp_headers.h"
Steve Anton10542f22019-01-11 09:11:00 -080019#include "api/rtp_parameters.h"
20#include "media/base/rtp_utils.h"
Zhi Huang365381f2018-04-13 16:44:34 -070021#include "modules/rtp_rtcp/source/rtp_packet_received.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "rtc_base/checks.h"
Steve Anton10542f22019-01-11 09:11:00 -080023#include "rtc_base/copy_on_write_buffer.h"
Yves Gerey3e707812018-11-28 16:47:49 +010024#include "rtc_base/logging.h"
25#include "rtc_base/third_party/sigslot/sigslot.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020026#include "rtc_base/trace_event.h"
zsteind48dbda2017-04-04 19:45:57 -070027
28namespace webrtc {
29
zstein56162b92017-04-24 16:54:35 -070030void RtpTransport::SetRtcpMuxEnabled(bool enable) {
31 rtcp_mux_enabled_ = enable;
32 MaybeSignalReadyToSend();
zsteind9ce7642017-04-10 16:17:57 -070033}
34
Bjorn A Mellem3a1b9272019-05-24 16:13:08 -070035const std::string& RtpTransport::transport_name() const {
36 return rtp_packet_transport_->transport_name();
37}
38
39int RtpTransport::SetRtpOption(rtc::Socket::Option opt, int value) {
40 return rtp_packet_transport_->SetOption(opt, value);
41}
42
43int RtpTransport::SetRtcpOption(rtc::Socket::Option opt, int value) {
44 if (rtcp_packet_transport_) {
45 return rtcp_packet_transport_->SetOption(opt, value);
46 }
47 return -1;
48}
49
zstein56162b92017-04-24 16:54:35 -070050void RtpTransport::SetRtpPacketTransport(
51 rtc::PacketTransportInternal* new_packet_transport) {
52 if (new_packet_transport == rtp_packet_transport_) {
53 return;
54 }
55 if (rtp_packet_transport_) {
56 rtp_packet_transport_->SignalReadyToSend.disconnect(this);
zstein3dcf0e92017-06-01 13:22:42 -070057 rtp_packet_transport_->SignalReadPacket.disconnect(this);
Zhi Huang942bc2e2017-11-13 13:26:07 -080058 rtp_packet_transport_->SignalNetworkRouteChanged.disconnect(this);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -080059 rtp_packet_transport_->SignalWritableState.disconnect(this);
60 rtp_packet_transport_->SignalSentPacket.disconnect(this);
Zhi Huang942bc2e2017-11-13 13:26:07 -080061 // Reset the network route of the old transport.
Danil Chapovalov66cadcc2018-06-19 16:47:43 +020062 SignalNetworkRouteChanged(absl::optional<rtc::NetworkRoute>());
zstein56162b92017-04-24 16:54:35 -070063 }
64 if (new_packet_transport) {
65 new_packet_transport->SignalReadyToSend.connect(
66 this, &RtpTransport::OnReadyToSend);
zstein3dcf0e92017-06-01 13:22:42 -070067 new_packet_transport->SignalReadPacket.connect(this,
68 &RtpTransport::OnReadPacket);
Zhi Huang942bc2e2017-11-13 13:26:07 -080069 new_packet_transport->SignalNetworkRouteChanged.connect(
Zhi Huang365381f2018-04-13 16:44:34 -070070 this, &RtpTransport::OnNetworkRouteChanged);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -080071 new_packet_transport->SignalWritableState.connect(
72 this, &RtpTransport::OnWritableState);
73 new_packet_transport->SignalSentPacket.connect(this,
74 &RtpTransport::OnSentPacket);
Zhi Huang942bc2e2017-11-13 13:26:07 -080075 // Set the network route for the new transport.
76 SignalNetworkRouteChanged(new_packet_transport->network_route());
zstein56162b92017-04-24 16:54:35 -070077 }
Zhi Huang8c316c12017-11-13 21:13:45 +000078
Zhi Huang942bc2e2017-11-13 13:26:07 -080079 rtp_packet_transport_ = new_packet_transport;
zstein56162b92017-04-24 16:54:35 -070080 // Assumes the transport is ready to send if it is writable. If we are wrong,
81 // ready to send will be updated the next time we try to send.
82 SetReadyToSend(false,
83 rtp_packet_transport_ && rtp_packet_transport_->writable());
84}
85
86void RtpTransport::SetRtcpPacketTransport(
87 rtc::PacketTransportInternal* new_packet_transport) {
88 if (new_packet_transport == rtcp_packet_transport_) {
89 return;
90 }
91 if (rtcp_packet_transport_) {
92 rtcp_packet_transport_->SignalReadyToSend.disconnect(this);
zstein3dcf0e92017-06-01 13:22:42 -070093 rtcp_packet_transport_->SignalReadPacket.disconnect(this);
Zhi Huang942bc2e2017-11-13 13:26:07 -080094 rtcp_packet_transport_->SignalNetworkRouteChanged.disconnect(this);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -080095 rtcp_packet_transport_->SignalWritableState.disconnect(this);
96 rtcp_packet_transport_->SignalSentPacket.disconnect(this);
Zhi Huang942bc2e2017-11-13 13:26:07 -080097 // Reset the network route of the old transport.
Danil Chapovalov66cadcc2018-06-19 16:47:43 +020098 SignalNetworkRouteChanged(absl::optional<rtc::NetworkRoute>());
zstein56162b92017-04-24 16:54:35 -070099 }
100 if (new_packet_transport) {
101 new_packet_transport->SignalReadyToSend.connect(
102 this, &RtpTransport::OnReadyToSend);
zstein3dcf0e92017-06-01 13:22:42 -0700103 new_packet_transport->SignalReadPacket.connect(this,
104 &RtpTransport::OnReadPacket);
Zhi Huang942bc2e2017-11-13 13:26:07 -0800105 new_packet_transport->SignalNetworkRouteChanged.connect(
Zhi Huang365381f2018-04-13 16:44:34 -0700106 this, &RtpTransport::OnNetworkRouteChanged);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800107 new_packet_transport->SignalWritableState.connect(
108 this, &RtpTransport::OnWritableState);
109 new_packet_transport->SignalSentPacket.connect(this,
110 &RtpTransport::OnSentPacket);
Zhi Huang942bc2e2017-11-13 13:26:07 -0800111 // Set the network route for the new transport.
112 SignalNetworkRouteChanged(new_packet_transport->network_route());
zstein56162b92017-04-24 16:54:35 -0700113 }
114 rtcp_packet_transport_ = new_packet_transport;
115
116 // Assumes the transport is ready to send if it is writable. If we are wrong,
117 // ready to send will be updated the next time we try to send.
118 SetReadyToSend(true,
119 rtcp_packet_transport_ && rtcp_packet_transport_->writable());
120}
121
122bool RtpTransport::IsWritable(bool rtcp) const {
123 rtc::PacketTransportInternal* transport = rtcp && !rtcp_mux_enabled_
124 ? rtcp_packet_transport_
125 : rtp_packet_transport_;
126 return transport && transport->writable();
127}
128
Zhi Huangcf990f52017-09-22 12:12:30 -0700129bool RtpTransport::SendRtpPacket(rtc::CopyOnWriteBuffer* packet,
130 const rtc::PacketOptions& options,
131 int flags) {
132 return SendPacket(false, packet, options, flags);
133}
134
135bool RtpTransport::SendRtcpPacket(rtc::CopyOnWriteBuffer* packet,
136 const rtc::PacketOptions& options,
137 int flags) {
138 return SendPacket(true, packet, options, flags);
139}
140
zstein56162b92017-04-24 16:54:35 -0700141bool RtpTransport::SendPacket(bool rtcp,
zstein398c3fd2017-07-19 13:38:02 -0700142 rtc::CopyOnWriteBuffer* packet,
zstein56162b92017-04-24 16:54:35 -0700143 const rtc::PacketOptions& options,
144 int flags) {
145 rtc::PacketTransportInternal* transport = rtcp && !rtcp_mux_enabled_
146 ? rtcp_packet_transport_
147 : rtp_packet_transport_;
Amit Hilbuche7a5f7b2019-03-12 11:10:27 -0700148 int ret = transport->SendPacket(packet->cdata<char>(), packet->size(),
149 options, flags);
zstein56162b92017-04-24 16:54:35 -0700150 if (ret != static_cast<int>(packet->size())) {
151 if (transport->GetError() == ENOTCONN) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100152 RTC_LOG(LS_WARNING) << "Got ENOTCONN from transport.";
zstein56162b92017-04-24 16:54:35 -0700153 SetReadyToSend(rtcp, false);
154 }
155 return false;
156 }
157 return true;
zsteind48dbda2017-04-04 19:45:57 -0700158}
159
Zhi Huang365381f2018-04-13 16:44:34 -0700160void RtpTransport::UpdateRtpHeaderExtensionMap(
161 const cricket::RtpHeaderExtensions& header_extensions) {
162 header_extension_map_ = RtpHeaderExtensionMap(header_extensions);
163}
164
165bool RtpTransport::RegisterRtpDemuxerSink(const RtpDemuxerCriteria& criteria,
166 RtpPacketSinkInterface* sink) {
167 rtp_demuxer_.RemoveSink(sink);
168 if (!rtp_demuxer_.AddSink(criteria, sink)) {
169 RTC_LOG(LS_ERROR) << "Failed to register the sink for RTP demuxer.";
170 return false;
171 }
172 return true;
173}
174
175bool RtpTransport::UnregisterRtpDemuxerSink(RtpPacketSinkInterface* sink) {
176 if (!rtp_demuxer_.RemoveSink(sink)) {
177 RTC_LOG(LS_ERROR) << "Failed to unregister the sink for RTP demuxer.";
178 return false;
179 }
180 return true;
181}
182
Amit Hilbuchedd20542019-03-18 12:33:43 -0700183void RtpTransport::DemuxPacket(rtc::CopyOnWriteBuffer packet,
Niels Möllere6933812018-11-05 13:01:41 +0100184 int64_t packet_time_us) {
Zhi Huang365381f2018-04-13 16:44:34 -0700185 webrtc::RtpPacketReceived parsed_packet(&header_extension_map_);
Amit Hilbuchedd20542019-03-18 12:33:43 -0700186 if (!parsed_packet.Parse(std::move(packet))) {
Zhi Huang365381f2018-04-13 16:44:34 -0700187 RTC_LOG(LS_ERROR)
188 << "Failed to parse the incoming RTP packet before demuxing. Drop it.";
189 return;
190 }
191
Niels Möllere6933812018-11-05 13:01:41 +0100192 if (packet_time_us != -1) {
193 parsed_packet.set_arrival_time_ms((packet_time_us + 500) / 1000);
Zhi Huang365381f2018-04-13 16:44:34 -0700194 }
Steve Anton884adca2019-04-15 16:52:27 -0700195 if (!rtp_demuxer_.OnRtpPacket(parsed_packet)) {
196 RTC_LOG(LS_WARNING) << "Failed to demux RTP packet: "
197 << RtpDemuxer::DescribePacket(parsed_packet);
198 }
Zhi Huang365381f2018-04-13 16:44:34 -0700199}
200
Zhi Huang365381f2018-04-13 16:44:34 -0700201bool RtpTransport::IsTransportWritable() {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800202 auto rtcp_packet_transport =
203 rtcp_mux_enabled_ ? nullptr : rtcp_packet_transport_;
204 return rtp_packet_transport_ && rtp_packet_transport_->writable() &&
205 (!rtcp_packet_transport || rtcp_packet_transport->writable());
206}
207
zstein56162b92017-04-24 16:54:35 -0700208void RtpTransport::OnReadyToSend(rtc::PacketTransportInternal* transport) {
209 SetReadyToSend(transport == rtcp_packet_transport_, true);
210}
211
Zhi Huang365381f2018-04-13 16:44:34 -0700212void RtpTransport::OnNetworkRouteChanged(
Danil Chapovalov66cadcc2018-06-19 16:47:43 +0200213 absl::optional<rtc::NetworkRoute> network_route) {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800214 SignalNetworkRouteChanged(network_route);
215}
216
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800217void RtpTransport::OnWritableState(
218 rtc::PacketTransportInternal* packet_transport) {
219 RTC_DCHECK(packet_transport == rtp_packet_transport_ ||
220 packet_transport == rtcp_packet_transport_);
Zhi Huang365381f2018-04-13 16:44:34 -0700221 SignalWritableState(IsTransportWritable());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800222}
223
224void RtpTransport::OnSentPacket(rtc::PacketTransportInternal* packet_transport,
225 const rtc::SentPacket& sent_packet) {
226 RTC_DCHECK(packet_transport == rtp_packet_transport_ ||
227 packet_transport == rtcp_packet_transport_);
228 SignalSentPacket(sent_packet);
229}
230
Amit Hilbuchedd20542019-03-18 12:33:43 -0700231void RtpTransport::OnRtpPacketReceived(rtc::CopyOnWriteBuffer packet,
Niels Möllere6933812018-11-05 13:01:41 +0100232 int64_t packet_time_us) {
233 DemuxPacket(packet, packet_time_us);
Zhi Huang365381f2018-04-13 16:44:34 -0700234}
235
Amit Hilbuchedd20542019-03-18 12:33:43 -0700236void RtpTransport::OnRtcpPacketReceived(rtc::CopyOnWriteBuffer packet,
Niels Möllere6933812018-11-05 13:01:41 +0100237 int64_t packet_time_us) {
Amit Hilbuchedd20542019-03-18 12:33:43 -0700238 SignalRtcpPacketReceived(&packet, packet_time_us);
Zhi Huang365381f2018-04-13 16:44:34 -0700239}
240
241void RtpTransport::OnReadPacket(rtc::PacketTransportInternal* transport,
242 const char* data,
243 size_t len,
Niels Möllere6933812018-11-05 13:01:41 +0100244 const int64_t& packet_time_us,
Zhi Huang365381f2018-04-13 16:44:34 -0700245 int flags) {
246 TRACE_EVENT0("webrtc", "RtpTransport::OnReadPacket");
247
248 // When using RTCP multiplexing we might get RTCP packets on the RTP
249 // transport. We check the RTP payload type to determine if it is RTCP.
Amit Hilbuchedd20542019-03-18 12:33:43 -0700250 auto array_view = rtc::MakeArrayView(data, len);
251 cricket::RtpPacketType packet_type = cricket::InferRtpPacketType(array_view);
Zhi Huang365381f2018-04-13 16:44:34 -0700252 // Filter out the packet that is neither RTP nor RTCP.
Amit Hilbuchedd20542019-03-18 12:33:43 -0700253 if (packet_type == cricket::RtpPacketType::kUnknown) {
254 return;
255 }
256
257 // Protect ourselves against crazy data.
258 if (!cricket::IsValidRtpPacketSize(packet_type, len)) {
259 RTC_LOG(LS_ERROR) << "Dropping incoming "
260 << cricket::RtpPacketTypeToString(packet_type)
261 << " packet: wrong size=" << len;
Zhi Huang365381f2018-04-13 16:44:34 -0700262 return;
263 }
264
265 rtc::CopyOnWriteBuffer packet(data, len);
Amit Hilbuchedd20542019-03-18 12:33:43 -0700266 if (packet_type == cricket::RtpPacketType::kRtcp) {
267 OnRtcpPacketReceived(std::move(packet), packet_time_us);
Zhi Huang365381f2018-04-13 16:44:34 -0700268 } else {
Amit Hilbuchedd20542019-03-18 12:33:43 -0700269 OnRtpPacketReceived(std::move(packet), packet_time_us);
Zhi Huang365381f2018-04-13 16:44:34 -0700270 }
271}
272
zstein56162b92017-04-24 16:54:35 -0700273void RtpTransport::SetReadyToSend(bool rtcp, bool ready) {
274 if (rtcp) {
275 rtcp_ready_to_send_ = ready;
276 } else {
277 rtp_ready_to_send_ = ready;
278 }
279
280 MaybeSignalReadyToSend();
281}
282
283void RtpTransport::MaybeSignalReadyToSend() {
284 bool ready_to_send =
285 rtp_ready_to_send_ && (rtcp_ready_to_send_ || rtcp_mux_enabled_);
286 if (ready_to_send != ready_to_send_) {
287 ready_to_send_ = ready_to_send;
288 SignalReadyToSend(ready_to_send);
289 }
290}
291
zsteind48dbda2017-04-04 19:45:57 -0700292} // namespace webrtc