blob: a32c8134e29dd381d558ffba9c41e8b94fa91032 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
leozwang@webrtc.org39e96592012-03-01 18:22:48 +00002 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
niklase@google.com470e71d2011-07-07 08:21:25 +00003 *
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
mflodman@webrtc.org1b1cd782012-06-28 06:34:08 +000011#include "video_engine/vie_receiver.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000012
stefan@webrtc.org976a7e62012-09-21 13:20:21 +000013#include "modules/remote_bitrate_estimator/include/remote_bitrate_estimator.h"
mflodman@webrtc.org1b1cd782012-06-28 06:34:08 +000014#include "modules/rtp_rtcp/interface/rtp_rtcp.h"
15#include "modules/utility/interface/rtp_dump.h"
16#include "modules/video_coding/main/interface/video_coding.h"
17#include "system_wrappers/interface/critical_section_wrapper.h"
stefan@webrtc.org976a7e62012-09-21 13:20:21 +000018#include "system_wrappers/interface/tick_util.h"
mflodman@webrtc.org1b1cd782012-06-28 06:34:08 +000019#include "system_wrappers/interface/trace.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000020
21namespace webrtc {
22
stefan@webrtc.org976a7e62012-09-21 13:20:21 +000023enum { kPacketOverheadBytes = 28 };
24
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +000025ViEReceiver::ViEReceiver(const int32_t channel_id,
stefan@webrtc.org976a7e62012-09-21 13:20:21 +000026 VideoCodingModule* module_vcm,
27 RemoteBitrateEstimator* remote_bitrate_estimator)
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +000028 : receive_cs_(CriticalSectionWrapper::CreateCriticalSection()),
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000029 channel_id_(channel_id),
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +000030 rtp_rtcp_(NULL),
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000031 vcm_(module_vcm),
stefan@webrtc.org976a7e62012-09-21 13:20:21 +000032 remote_bitrate_estimator_(remote_bitrate_estimator),
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000033 external_decryption_(NULL),
34 decryption_buffer_(NULL),
35 rtp_dump_(NULL),
36 receiving_(false) {
stefan@webrtc.org976a7e62012-09-21 13:20:21 +000037 assert(remote_bitrate_estimator);
niklase@google.com470e71d2011-07-07 08:21:25 +000038}
39
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000040ViEReceiver::~ViEReceiver() {
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000041 if (decryption_buffer_) {
42 delete[] decryption_buffer_;
43 decryption_buffer_ = NULL;
44 }
45 if (rtp_dump_) {
46 rtp_dump_->Stop();
47 RtpDump::DestroyRtpDump(rtp_dump_);
48 rtp_dump_ = NULL;
49 }
niklase@google.com470e71d2011-07-07 08:21:25 +000050}
51
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000052int ViEReceiver::RegisterExternalDecryption(Encryption* decryption) {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +000053 CriticalSectionScoped cs(receive_cs_.get());
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000054 if (external_decryption_) {
55 return -1;
56 }
57 decryption_buffer_ = new WebRtc_UWord8[kViEMaxMtu];
58 if (decryption_buffer_ == NULL) {
59 return -1;
60 }
61 external_decryption_ = decryption;
62 return 0;
niklase@google.com470e71d2011-07-07 08:21:25 +000063}
64
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000065int ViEReceiver::DeregisterExternalDecryption() {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +000066 CriticalSectionScoped cs(receive_cs_.get());
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000067 if (external_decryption_ == NULL) {
68 return -1;
69 }
70 external_decryption_ = NULL;
71 return 0;
niklase@google.com470e71d2011-07-07 08:21:25 +000072}
73
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +000074void ViEReceiver::SetRtpRtcpModule(RtpRtcp* module) {
75 rtp_rtcp_ = module;
76}
77
pwestin@webrtc.org1da1ce02011-10-13 15:19:55 +000078void ViEReceiver::RegisterSimulcastRtpRtcpModules(
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000079 const std::list<RtpRtcp*>& rtp_modules) {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +000080 CriticalSectionScoped cs(receive_cs_.get());
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000081 rtp_rtcp_simulcast_.clear();
82
83 if (!rtp_modules.empty()) {
84 rtp_rtcp_simulcast_.insert(rtp_rtcp_simulcast_.begin(),
85 rtp_modules.begin(),
86 rtp_modules.end());
87 }
pwestin@webrtc.org1da1ce02011-10-13 15:19:55 +000088}
89
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000090void ViEReceiver::IncomingRTPPacket(const WebRtc_Word8* rtp_packet,
91 const WebRtc_Word32 rtp_packet_length,
leozwang@webrtc.org39e96592012-03-01 18:22:48 +000092 const char* from_ip,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000093 const WebRtc_UWord16 from_port) {
94 InsertRTPPacket(rtp_packet, rtp_packet_length);
95}
niklase@google.com470e71d2011-07-07 08:21:25 +000096
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +000097void ViEReceiver::IncomingRTCPPacket(const WebRtc_Word8* rtcp_packet,
98 const WebRtc_Word32 rtcp_packet_length,
leozwang@webrtc.org39e96592012-03-01 18:22:48 +000099 const char* from_ip,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000100 const WebRtc_UWord16 from_port) {
101 InsertRTCPPacket(rtcp_packet, rtcp_packet_length);
102}
niklase@google.com470e71d2011-07-07 08:21:25 +0000103
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000104int ViEReceiver::ReceivedRTPPacket(const void* rtp_packet,
105 int rtp_packet_length) {
106 if (!receiving_) {
107 return -1;
108 }
109 return InsertRTPPacket((const WebRtc_Word8*) rtp_packet, rtp_packet_length);
110}
111
112int ViEReceiver::ReceivedRTCPPacket(const void* rtcp_packet,
113 int rtcp_packet_length) {
114 if (!receiving_) {
115 return -1;
116 }
117 return InsertRTCPPacket((const WebRtc_Word8*) rtcp_packet,
118 rtcp_packet_length);
119}
120
121WebRtc_Word32 ViEReceiver::OnReceivedPayloadData(
122 const WebRtc_UWord8* payload_data, const WebRtc_UWord16 payload_size,
123 const WebRtcRTPHeader* rtp_header) {
124 if (rtp_header == NULL) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000125 return 0;
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000126 }
127
stefan@webrtc.org976a7e62012-09-21 13:20:21 +0000128 // TODO(holmer): Make sure packets reconstructed using FEC are not passed to
129 // the bandwidth estimator.
130 // Add headers, ideally we would like to include for instance
131 // Ethernet header here as well.
132 const int packet_size = payload_size + kPacketOverheadBytes +
133 rtp_header->header.headerLength + rtp_header->header.paddingLength;
134 uint32_t compensated_timestamp = rtp_header->header.timestamp +
135 rtp_header->extension.transmissionTimeOffset;
136 remote_bitrate_estimator_->IncomingPacket(rtp_header->header.ssrc,
137 packet_size,
138 TickTime::MillisecondTimestamp(),
139 compensated_timestamp);
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000140 if (vcm_->IncomingPacket(payload_data, payload_size, *rtp_header) != 0) {
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000141 // Check this...
142 return -1;
143 }
144 return 0;
niklase@google.com470e71d2011-07-07 08:21:25 +0000145}
146
stefan@webrtc.org976a7e62012-09-21 13:20:21 +0000147void ViEReceiver::OnSendReportReceived(const WebRtc_Word32 id,
148 const WebRtc_UWord32 senderSSRC,
149 uint32_t ntp_secs,
150 uint32_t ntp_frac,
151 uint32_t timestamp) {
152 remote_bitrate_estimator_->IncomingRtcp(senderSSRC, ntp_secs, ntp_frac,
153 timestamp);
154}
155
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000156int ViEReceiver::InsertRTPPacket(const WebRtc_Word8* rtp_packet,
157 int rtp_packet_length) {
158 // TODO(mflodman) Change decrypt to get rid of this cast.
159 WebRtc_Word8* tmp_ptr = const_cast<WebRtc_Word8*>(rtp_packet);
160 unsigned char* received_packet = reinterpret_cast<unsigned char*>(tmp_ptr);
161 int received_packet_length = rtp_packet_length;
niklase@google.com470e71d2011-07-07 08:21:25 +0000162
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000163 {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +0000164 CriticalSectionScoped cs(receive_cs_.get());
niklase@google.com470e71d2011-07-07 08:21:25 +0000165
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000166 if (external_decryption_) {
mflodman@webrtc.org34e83b82012-10-17 11:05:54 +0000167 int decrypted_length = kViEMaxMtu;
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000168 external_decryption_->decrypt(channel_id_, received_packet,
169 decryption_buffer_, received_packet_length,
170 &decrypted_length);
171 if (decrypted_length <= 0) {
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000172 WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, channel_id_,
173 "RTP decryption failed");
niklase@google.com470e71d2011-07-07 08:21:25 +0000174 return -1;
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000175 } else if (decrypted_length > kViEMaxMtu) {
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000176 WEBRTC_TRACE(webrtc::kTraceCritical, webrtc::kTraceVideo, channel_id_,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000177 "InsertRTPPacket: %d bytes is allocated as RTP decrytption"
178 " output, external decryption used %d bytes. => memory is "
179 " now corrupted", kViEMaxMtu, decrypted_length);
180 return -1;
181 }
182 received_packet = decryption_buffer_;
183 received_packet_length = decrypted_length;
niklase@google.com470e71d2011-07-07 08:21:25 +0000184 }
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000185
186 if (rtp_dump_) {
187 rtp_dump_->DumpPacket(received_packet,
188 static_cast<WebRtc_UWord16>(received_packet_length));
189 }
190 }
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000191 assert(rtp_rtcp_); // Should be set by owner at construction time.
192 return rtp_rtcp_->IncomingPacket(received_packet, received_packet_length);
niklase@google.com470e71d2011-07-07 08:21:25 +0000193}
194
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000195int ViEReceiver::InsertRTCPPacket(const WebRtc_Word8* rtcp_packet,
196 int rtcp_packet_length) {
197 // TODO(mflodman) Change decrypt to get rid of this cast.
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000198 WebRtc_Word8* tmp_ptr = const_cast<WebRtc_Word8*>(rtcp_packet);
199 unsigned char* received_packet = reinterpret_cast<unsigned char*>(tmp_ptr);
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000200 int received_packet_length = rtcp_packet_length;
201 {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +0000202 CriticalSectionScoped cs(receive_cs_.get());
niklase@google.com470e71d2011-07-07 08:21:25 +0000203
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000204 if (external_decryption_) {
mflodman@webrtc.org34e83b82012-10-17 11:05:54 +0000205 int decrypted_length = kViEMaxMtu;
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000206 external_decryption_->decrypt_rtcp(channel_id_, received_packet,
207 decryption_buffer_,
208 received_packet_length,
209 &decrypted_length);
210 if (decrypted_length <= 0) {
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000211 WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, channel_id_,
212 "RTP decryption failed");
niklase@google.com470e71d2011-07-07 08:21:25 +0000213 return -1;
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000214 } else if (decrypted_length > kViEMaxMtu) {
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000215 WEBRTC_TRACE(webrtc::kTraceCritical, webrtc::kTraceVideo, channel_id_,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000216 "InsertRTCPPacket: %d bytes is allocated as RTP "
217 " decrytption output, external decryption used %d bytes. "
218 " => memory is now corrupted",
219 kViEMaxMtu, decrypted_length);
220 return -1;
221 }
222 received_packet = decryption_buffer_;
223 received_packet_length = decrypted_length;
niklase@google.com470e71d2011-07-07 08:21:25 +0000224 }
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000225
226 if (rtp_dump_) {
227 rtp_dump_->DumpPacket(
228 received_packet, static_cast<WebRtc_UWord16>(received_packet_length));
229 }
230 }
231 {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +0000232 CriticalSectionScoped cs(receive_cs_.get());
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000233 std::list<RtpRtcp*>::iterator it = rtp_rtcp_simulcast_.begin();
234 while (it != rtp_rtcp_simulcast_.end()) {
235 RtpRtcp* rtp_rtcp = *it++;
236 rtp_rtcp->IncomingPacket(received_packet, received_packet_length);
237 }
238 }
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000239 assert(rtp_rtcp_); // Should be set by owner at construction time.
240 return rtp_rtcp_->IncomingPacket(received_packet, received_packet_length);
niklase@google.com470e71d2011-07-07 08:21:25 +0000241}
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000242
243void ViEReceiver::StartReceive() {
244 receiving_ = true;
245}
246
247void ViEReceiver::StopReceive() {
248 receiving_ = false;
249}
250
251int ViEReceiver::StartRTPDump(const char file_nameUTF8[1024]) {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +0000252 CriticalSectionScoped cs(receive_cs_.get());
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000253 if (rtp_dump_) {
254 // Restart it if it already exists and is started
255 rtp_dump_->Stop();
256 } else {
257 rtp_dump_ = RtpDump::CreateRtpDump();
258 if (rtp_dump_ == NULL) {
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000259 WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, channel_id_,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000260 "StartRTPDump: Failed to create RTP dump");
261 return -1;
262 }
263 }
264 if (rtp_dump_->Start(file_nameUTF8) != 0) {
265 RtpDump::DestroyRtpDump(rtp_dump_);
266 rtp_dump_ = NULL;
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000267 WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, channel_id_,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000268 "StartRTPDump: Failed to start RTP dump");
269 return -1;
270 }
271 return 0;
272}
273
274int ViEReceiver::StopRTPDump() {
mflodman@webrtc.orgd32c4472011-12-22 14:17:53 +0000275 CriticalSectionScoped cs(receive_cs_.get());
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000276 if (rtp_dump_) {
277 if (rtp_dump_->IsActive()) {
278 rtp_dump_->Stop();
279 } else {
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000280 WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, channel_id_,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000281 "StopRTPDump: Dump not active");
282 }
283 RtpDump::DestroyRtpDump(rtp_dump_);
284 rtp_dump_ = NULL;
285 } else {
pwestin@webrtc.org2853dde2012-05-11 11:08:54 +0000286 WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideo, channel_id_,
mflodman@webrtc.orgad4ee362011-11-28 22:39:24 +0000287 "StopRTPDump: RTP dump not started");
288 return -1;
289 }
290 return 0;
291}
292
293} // namespace webrtc