blob: 219f66805f34d8a5d98338bd53a88f8dfa35e210 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
mflodman@webrtc.orgc80d9d92012-02-06 10:11:25 +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
philipel5908c712015-12-21 08:23:20 -080011#include "webrtc/modules/video_coding/video_coding_impl.h"
12
13#include <algorithm>
14
pbos@webrtc.orga4407322013-07-16 12:32:05 +000015#include "webrtc/common_types.h"
16#include "webrtc/common_video/libyuv/include/webrtc_libyuv.h"
sprang3911c262016-04-15 01:24:14 -070017#include "webrtc/base/criticalsection.h"
Henrik Kjellander2557b862015-11-18 22:00:21 +010018#include "webrtc/modules/video_coding/include/video_codec_interface.h"
19#include "webrtc/modules/video_coding/encoded_frame.h"
20#include "webrtc/modules/video_coding/jitter_buffer.h"
21#include "webrtc/modules/video_coding/packet.h"
Henrik Kjellander98f53512015-10-28 18:17:40 +010022#include "webrtc/system_wrappers/include/clock.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000023
agalusza@google.coma7e360e2013-08-01 03:15:08 +000024namespace webrtc {
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000025namespace vcm {
niklase@google.com470e71d2011-07-07 08:21:25 +000026
philipel5908c712015-12-21 08:23:20 -080027int64_t VCMProcessTimer::Period() const {
28 return _periodMs;
niklase@google.com470e71d2011-07-07 08:21:25 +000029}
30
philipel5908c712015-12-21 08:23:20 -080031int64_t VCMProcessTimer::TimeUntilProcess() const {
32 const int64_t time_since_process = _clock->TimeInMilliseconds() - _latestMs;
33 const int64_t time_until_process = _periodMs - time_since_process;
34 return std::max<int64_t>(time_until_process, 0);
niklase@google.com470e71d2011-07-07 08:21:25 +000035}
36
philipel5908c712015-12-21 08:23:20 -080037void VCMProcessTimer::Processed() {
38 _latestMs = _clock->TimeInMilliseconds();
niklase@google.com470e71d2011-07-07 08:21:25 +000039}
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000040} // namespace vcm
niklase@google.com470e71d2011-07-07 08:21:25 +000041
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000042namespace {
andresp@webrtc.org1df9dc32014-01-09 08:01:57 +000043// This wrapper provides a way to modify the callback without the need to expose
44// a register method all the way down to the function calling it.
45class EncodedImageCallbackWrapper : public EncodedImageCallback {
46 public:
47 EncodedImageCallbackWrapper()
48 : cs_(CriticalSectionWrapper::CreateCriticalSection()), callback_(NULL) {}
49
50 virtual ~EncodedImageCallbackWrapper() {}
51
52 void Register(EncodedImageCallback* callback) {
53 CriticalSectionScoped cs(cs_.get());
54 callback_ = callback;
55 }
56
pbos@webrtc.org273a4142014-12-01 15:23:21 +000057 virtual int32_t Encoded(const EncodedImage& encoded_image,
andresp@webrtc.org1df9dc32014-01-09 08:01:57 +000058 const CodecSpecificInfo* codec_specific_info,
59 const RTPFragmentationHeader* fragmentation) {
60 CriticalSectionScoped cs(cs_.get());
61 if (callback_)
philipel5908c712015-12-21 08:23:20 -080062 return callback_->Encoded(encoded_image, codec_specific_info,
63 fragmentation);
andresp@webrtc.org1df9dc32014-01-09 08:01:57 +000064 return 0;
65 }
66
67 private:
kwiberg3f55dea2016-02-29 05:51:59 -080068 std::unique_ptr<CriticalSectionWrapper> cs_;
andresp@webrtc.org1df9dc32014-01-09 08:01:57 +000069 EncodedImageCallback* callback_ GUARDED_BY(cs_);
70};
71
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000072class VideoCodingModuleImpl : public VideoCodingModule {
73 public:
stefan@webrtc.org34c5da62014-04-11 14:08:35 +000074 VideoCodingModuleImpl(Clock* clock,
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000075 EventFactory* event_factory,
mflodmanfcf54bd2015-04-14 21:28:08 +020076 VideoEncoderRateObserver* encoder_rate_observer,
philipel83f831a2016-03-12 03:30:23 -080077 VCMQMSettingsCallback* qm_settings_callback,
78 NackSender* nack_sender,
sprang3911c262016-04-15 01:24:14 -070079 KeyFrameRequestSender* keyframe_request_sender,
80 EncodedImageCallback* pre_decode_image_callback)
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000081 : VideoCodingModule(),
pbos0fa9b222015-11-13 05:59:57 -080082 sender_(clock,
83 &post_encode_callback_,
84 encoder_rate_observer,
perkj376b1922016-05-02 11:35:24 -070085 qm_settings_callback,
86 nullptr),
philipel83f831a2016-03-12 03:30:23 -080087 receiver_(clock,
88 event_factory,
sprang3911c262016-04-15 01:24:14 -070089 pre_decode_image_callback,
philipel83f831a2016-03-12 03:30:23 -080090 nack_sender,
Peter Boström0b250722016-04-22 18:23:15 +020091 keyframe_request_sender) {}
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000092
Peter Boström0b250722016-04-22 18:23:15 +020093 virtual ~VideoCodingModuleImpl() {}
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000094
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000095 int64_t TimeUntilNextProcess() override {
pbos0fa9b222015-11-13 05:59:57 -080096 int64_t sender_time = sender_.TimeUntilNextProcess();
97 int64_t receiver_time = receiver_.TimeUntilNextProcess();
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000098 assert(sender_time >= 0);
99 assert(receiver_time >= 0);
100 return VCM_MIN(sender_time, receiver_time);
101 }
102
pbosa26ac922016-02-25 04:50:01 -0800103 void Process() override {
104 sender_.Process();
105 receiver_.Process();
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000106 }
107
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000108 int32_t RegisterSendCodec(const VideoCodec* sendCodec,
109 uint32_t numberOfCores,
110 uint32_t maxPayloadSize) override {
pbos0fa9b222015-11-13 05:59:57 -0800111 return sender_.RegisterSendCodec(sendCodec, numberOfCores, maxPayloadSize);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000112 }
113
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000114 int32_t RegisterExternalEncoder(VideoEncoder* externalEncoder,
115 uint8_t payloadType,
116 bool internalSource) override {
Peter Boström795dbe42015-11-27 14:09:07 +0100117 sender_.RegisterExternalEncoder(externalEncoder, payloadType,
118 internalSource);
119 return 0;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000120 }
121
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000122 int Bitrate(unsigned int* bitrate) const override {
pbos0fa9b222015-11-13 05:59:57 -0800123 return sender_.Bitrate(bitrate);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000124 }
125
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000126 int FrameRate(unsigned int* framerate) const override {
pbos0fa9b222015-11-13 05:59:57 -0800127 return sender_.FrameRate(framerate);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000128 }
129
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000130 int32_t SetChannelParameters(uint32_t target_bitrate, // bits/s.
131 uint8_t lossRate,
132 int64_t rtt) override {
pbos0fa9b222015-11-13 05:59:57 -0800133 return sender_.SetChannelParameters(target_bitrate, lossRate, rtt);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000134 }
135
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000136 int32_t RegisterProtectionCallback(
137 VCMProtectionCallback* protection) override {
pbos0fa9b222015-11-13 05:59:57 -0800138 return sender_.RegisterProtectionCallback(protection);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000139 }
140
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000141 int32_t SetVideoProtection(VCMVideoProtection videoProtection,
142 bool enable) override {
pbosba8c15b2015-07-14 09:36:34 -0700143 // TODO(pbos): Remove enable from receive-side protection modes as well.
144 if (enable)
pbos0fa9b222015-11-13 05:59:57 -0800145 sender_.SetVideoProtection(videoProtection);
146 return receiver_.SetVideoProtection(videoProtection, enable);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000147 }
148
Miguel Casas-Sanchez47650702015-05-29 17:21:40 -0700149 int32_t AddVideoFrame(const VideoFrame& videoFrame,
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000150 const VideoContentMetrics* contentMetrics,
151 const CodecSpecificInfo* codecSpecificInfo) override {
pbos0fa9b222015-11-13 05:59:57 -0800152 return sender_.AddVideoFrame(videoFrame, contentMetrics, codecSpecificInfo);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000153 }
154
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000155 int32_t IntraFrameRequest(int stream_index) override {
pbos0fa9b222015-11-13 05:59:57 -0800156 return sender_.IntraFrameRequest(stream_index);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000157 }
158
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000159 int32_t EnableFrameDropper(bool enable) override {
pbos0fa9b222015-11-13 05:59:57 -0800160 return sender_.EnableFrameDropper(enable);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000161 }
162
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000163 void SuspendBelowMinBitrate() override {
pbos0fa9b222015-11-13 05:59:57 -0800164 return sender_.SuspendBelowMinBitrate();
henrik.lundin@webrtc.org572699d2013-09-30 12:16:08 +0000165 }
166
pbos0fa9b222015-11-13 05:59:57 -0800167 bool VideoSuspended() const override { return sender_.VideoSuspended(); }
henrik.lundin@webrtc.org572699d2013-09-30 12:16:08 +0000168
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000169 int32_t RegisterReceiveCodec(const VideoCodec* receiveCodec,
170 int32_t numberOfCores,
171 bool requireKeyFrame) override {
pbos0fa9b222015-11-13 05:59:57 -0800172 return receiver_.RegisterReceiveCodec(receiveCodec, numberOfCores,
173 requireKeyFrame);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000174 }
175
Peter Boström795dbe42015-11-27 14:09:07 +0100176 void RegisterExternalDecoder(VideoDecoder* externalDecoder,
perkj796cfaf2015-12-10 09:27:38 -0800177 uint8_t payloadType) override {
178 receiver_.RegisterExternalDecoder(externalDecoder, payloadType);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000179 }
180
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000181 int32_t RegisterReceiveCallback(
182 VCMReceiveCallback* receiveCallback) override {
pbos0fa9b222015-11-13 05:59:57 -0800183 return receiver_.RegisterReceiveCallback(receiveCallback);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000184 }
185
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000186 int32_t RegisterReceiveStatisticsCallback(
187 VCMReceiveStatisticsCallback* receiveStats) override {
pbos0fa9b222015-11-13 05:59:57 -0800188 return receiver_.RegisterReceiveStatisticsCallback(receiveStats);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000189 }
190
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000191 int32_t RegisterDecoderTimingCallback(
192 VCMDecoderTimingCallback* decoderTiming) override {
pbos0fa9b222015-11-13 05:59:57 -0800193 return receiver_.RegisterDecoderTimingCallback(decoderTiming);
fischman@webrtc.org37bb4972013-10-23 23:59:45 +0000194 }
195
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000196 int32_t RegisterFrameTypeCallback(
197 VCMFrameTypeCallback* frameTypeCallback) override {
pbos0fa9b222015-11-13 05:59:57 -0800198 return receiver_.RegisterFrameTypeCallback(frameTypeCallback);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000199 }
200
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000201 int32_t RegisterPacketRequestCallback(
202 VCMPacketRequestCallback* callback) override {
pbos0fa9b222015-11-13 05:59:57 -0800203 return receiver_.RegisterPacketRequestCallback(callback);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000204 }
205
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000206 int32_t Decode(uint16_t maxWaitTimeMs) override {
pbos0fa9b222015-11-13 05:59:57 -0800207 return receiver_.Decode(maxWaitTimeMs);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000208 }
209
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000210 int32_t ReceiveCodec(VideoCodec* currentReceiveCodec) const override {
pbos0fa9b222015-11-13 05:59:57 -0800211 return receiver_.ReceiveCodec(currentReceiveCodec);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000212 }
213
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000214 VideoCodecType ReceiveCodec() const override {
pbos0fa9b222015-11-13 05:59:57 -0800215 return receiver_.ReceiveCodec();
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000216 }
217
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000218 int32_t IncomingPacket(const uint8_t* incomingPayload,
219 size_t payloadLength,
220 const WebRtcRTPHeader& rtpInfo) override {
pbos0fa9b222015-11-13 05:59:57 -0800221 return receiver_.IncomingPacket(incomingPayload, payloadLength, rtpInfo);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000222 }
223
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000224 int32_t SetMinimumPlayoutDelay(uint32_t minPlayoutDelayMs) override {
pbos0fa9b222015-11-13 05:59:57 -0800225 return receiver_.SetMinimumPlayoutDelay(minPlayoutDelayMs);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000226 }
227
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000228 int32_t SetRenderDelay(uint32_t timeMS) override {
pbos0fa9b222015-11-13 05:59:57 -0800229 return receiver_.SetRenderDelay(timeMS);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000230 }
231
pbos0fa9b222015-11-13 05:59:57 -0800232 int32_t Delay() const override { return receiver_.Delay(); }
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000233
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000234 uint32_t DiscardedPackets() const override {
pbos0fa9b222015-11-13 05:59:57 -0800235 return receiver_.DiscardedPackets();
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000236 }
237
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000238 int SetReceiverRobustnessMode(ReceiverRobustness robustnessMode,
239 VCMDecodeErrorMode errorMode) override {
pbos0fa9b222015-11-13 05:59:57 -0800240 return receiver_.SetReceiverRobustnessMode(robustnessMode, errorMode);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000241 }
242
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000243 void SetNackSettings(size_t max_nack_list_size,
244 int max_packet_age_to_nack,
245 int max_incomplete_time_ms) override {
pbos0fa9b222015-11-13 05:59:57 -0800246 return receiver_.SetNackSettings(max_nack_list_size, max_packet_age_to_nack,
247 max_incomplete_time_ms);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000248 }
249
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000250 void SetDecodeErrorMode(VCMDecodeErrorMode decode_error_mode) override {
pbos0fa9b222015-11-13 05:59:57 -0800251 return receiver_.SetDecodeErrorMode(decode_error_mode);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000252 }
253
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000254 int SetMinReceiverDelay(int desired_delay_ms) override {
pbos0fa9b222015-11-13 05:59:57 -0800255 return receiver_.SetMinReceiverDelay(desired_delay_ms);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000256 }
257
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000258 int32_t SetReceiveChannelParameters(int64_t rtt) override {
pbos0fa9b222015-11-13 05:59:57 -0800259 return receiver_.SetReceiveChannelParameters(rtt);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000260 }
261
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000262 void RegisterPostEncodeImageCallback(
263 EncodedImageCallback* observer) override {
andresp@webrtc.org1df9dc32014-01-09 08:01:57 +0000264 post_encode_callback_.Register(observer);
sprang@webrtc.org40709352013-11-26 11:41:59 +0000265 }
266
pbos0fa9b222015-11-13 05:59:57 -0800267 void TriggerDecoderShutdown() override { receiver_.TriggerDecoderShutdown(); }
pbos@webrtc.org4dd40d62015-02-17 13:22:43 +0000268
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000269 private:
andresp@webrtc.org1df9dc32014-01-09 08:01:57 +0000270 EncodedImageCallbackWrapper post_encode_callback_;
pbos0fa9b222015-11-13 05:59:57 -0800271 vcm::VideoSender sender_;
272 vcm::VideoReceiver receiver_;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000273};
274} // namespace
275
Peter Boström7776e782016-01-07 15:42:47 +0100276void VideoCodingModule::Codec(VideoCodecType codecType, VideoCodec* codec) {
Peter Boström92f8dbd2015-11-24 13:55:55 +0100277 VCMCodecDataBase::Codec(codecType, codec);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000278}
279
philipel83f831a2016-03-12 03:30:23 -0800280// Create method for current interface, will be removed when the
281// new jitter buffer is in place.
pbos@webrtc.org891d4832015-02-26 13:15:22 +0000282VideoCodingModule* VideoCodingModule::Create(
mflodmanfcf54bd2015-04-14 21:28:08 +0200283 Clock* clock,
284 VideoEncoderRateObserver* encoder_rate_observer,
285 VCMQMSettingsCallback* qm_settings_callback) {
philipel83f831a2016-03-12 03:30:23 -0800286 return VideoCodingModule::Create(clock, encoder_rate_observer,
287 qm_settings_callback,
288 nullptr, // NackSender
sprang3911c262016-04-15 01:24:14 -0700289 nullptr, // KeyframeRequestSender
290 nullptr); // Pre-decode image callback
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000291}
292
philipel83f831a2016-03-12 03:30:23 -0800293// Create method for the new jitter buffer.
294VideoCodingModule* VideoCodingModule::Create(
295 Clock* clock,
296 VideoEncoderRateObserver* encoder_rate_observer,
297 VCMQMSettingsCallback* qm_settings_callback,
298 NackSender* nack_sender,
sprang3911c262016-04-15 01:24:14 -0700299 KeyFrameRequestSender* keyframe_request_sender,
300 EncodedImageCallback* pre_decode_image_callback) {
Peter Boström0b250722016-04-22 18:23:15 +0200301 return new VideoCodingModuleImpl(
302 clock, nullptr, encoder_rate_observer, qm_settings_callback, nack_sender,
303 keyframe_request_sender, pre_decode_image_callback);
philipel83f831a2016-03-12 03:30:23 -0800304}
305
306// Create method for current interface, will be removed when the
307// new jitter buffer is in place.
philipel5908c712015-12-21 08:23:20 -0800308VideoCodingModule* VideoCodingModule::Create(Clock* clock,
309 EventFactory* event_factory) {
philipel83f831a2016-03-12 03:30:23 -0800310 return VideoCodingModule::Create(clock, event_factory,
311 nullptr, // NackSender
312 nullptr); // KeyframeRequestSender
313}
314
315// Create method for the new jitter buffer.
316VideoCodingModule* VideoCodingModule::Create(
317 Clock* clock,
318 EventFactory* event_factory,
319 NackSender* nack_sender,
320 KeyFrameRequestSender* keyframe_request_sender) {
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000321 assert(clock);
322 assert(event_factory);
Peter Boström0b250722016-04-22 18:23:15 +0200323 return new VideoCodingModuleImpl(clock, event_factory, nullptr, nullptr,
324 nack_sender, keyframe_request_sender,
325 nullptr);
niklase@google.com470e71d2011-07-07 08:21:25 +0000326}
327
stefan@webrtc.org791eec72011-10-11 07:53:43 +0000328} // namespace webrtc