blob: ac8816639420442b421e2746100feda90c347504 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
pwestin@webrtc.org52fd98d2012-02-13 09:03:53 +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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#ifndef MODULES_VIDEO_CODING_VIDEO_CODING_IMPL_H_
12#define MODULES_VIDEO_CODING_VIDEO_CODING_IMPL_H_
niklase@google.com470e71d2011-07-07 08:21:25 +000013
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020014#include "modules/video_coding/include/video_coding.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000015
kwiberg3f55dea2016-02-29 05:51:59 -080016#include <memory>
perkj376b1922016-05-02 11:35:24 -070017#include <string>
stefan@webrtc.orgc5300432012-10-08 07:06:53 +000018#include <vector>
19
Niels Möllerf9063782018-02-20 16:09:48 +010020#include "modules/video_coding/decoder_database.h"
21#include "modules/video_coding/encoder_database.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "modules/video_coding/frame_buffer.h"
23#include "modules/video_coding/generic_decoder.h"
24#include "modules/video_coding/generic_encoder.h"
25#include "modules/video_coding/jitter_buffer.h"
26#include "modules/video_coding/media_optimization.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020027#include "modules/video_coding/receiver.h"
28#include "modules/video_coding/timing.h"
29#include "rtc_base/onetimeevent.h"
30#include "rtc_base/sequenced_task_checker.h"
31#include "rtc_base/thread_annotations.h"
32#include "rtc_base/thread_checker.h"
33#include "system_wrappers/include/clock.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000034
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000035namespace webrtc {
sprang@webrtc.org40709352013-11-26 11:41:59 +000036
Erik Språng08127a92016-11-16 16:41:30 +010037class VideoBitrateAllocator;
sprang1a646ee2016-12-01 06:34:11 -080038class VideoBitrateAllocationObserver;
Erik Språng08127a92016-11-16 16:41:30 +010039
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000040namespace vcm {
niklase@google.com470e71d2011-07-07 08:21:25 +000041
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000042class VCMProcessTimer {
43 public:
sprang40217c32016-11-21 05:41:52 -080044 static const int64_t kDefaultProcessIntervalMs = 1000;
45
pkasting@chromium.org0b1534c2014-12-15 22:09:40 +000046 VCMProcessTimer(int64_t periodMs, Clock* clock)
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000047 : _clock(clock),
48 _periodMs(periodMs),
49 _latestMs(_clock->TimeInMilliseconds()) {}
pkasting@chromium.org0b1534c2014-12-15 22:09:40 +000050 int64_t Period() const;
51 int64_t TimeUntilProcess() const;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000052 void Processed();
niklase@google.com470e71d2011-07-07 08:21:25 +000053
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000054 private:
55 Clock* _clock;
pkasting@chromium.org0b1534c2014-12-15 22:09:40 +000056 int64_t _periodMs;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000057 int64_t _latestMs;
niklase@google.com470e71d2011-07-07 08:21:25 +000058};
59
Niels Möllera0565992017-10-24 11:37:08 +020060class VideoSender {
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000061 public:
62 typedef VideoCodingModule::SenderNackMode SenderNackMode;
63
Yves Gerey665174f2018-06-19 15:03:05 +020064 VideoSender(Clock* clock, EncodedImageCallback* post_encode_callback);
andresp@webrtc.org1df9dc32014-01-09 08:01:57 +000065
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000066 ~VideoSender();
67
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000068 // Register the send codec to be used.
tommi@webrtc.orge07710c2015-02-19 17:43:25 +000069 // This method must be called on the construction thread.
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000070 int32_t RegisterSendCodec(const VideoCodec* sendCodec,
71 uint32_t numberOfCores,
72 uint32_t maxPayloadSize);
tommi@webrtc.orge07710c2015-02-19 17:43:25 +000073
Peter Boström795dbe42015-11-27 14:09:07 +010074 void RegisterExternalEncoder(VideoEncoder* externalEncoder,
Peter Boström795dbe42015-11-27 14:09:07 +010075 bool internalSource);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000076
sprang1a646ee2016-12-01 06:34:11 -080077 // Update the channel parameters based on new rates and rtt. This will also
78 // cause an immediate call to VideoEncoder::SetRateAllocation().
79 int32_t SetChannelParameters(
80 uint32_t target_bitrate_bps,
81 uint8_t loss_rate,
82 int64_t rtt,
83 VideoBitrateAllocator* bitrate_allocator,
84 VideoBitrateAllocationObserver* bitrate_updated_callback);
85
86 // Updates the channel parameters with a new bitrate allocation, but using the
87 // current targit_bitrate, loss rate and rtt. That is, the distribution or
88 // caps may be updated to a change to a new VideoCodec or allocation mode.
89 // The new parameters will be stored as pending EncoderParameters, and the
90 // encoder will only be updated on the next frame.
Niels Möller96d7f762018-01-30 11:27:16 +010091 void UpdateChannelParameters(
sprang1a646ee2016-12-01 06:34:11 -080092 VideoBitrateAllocator* bitrate_allocator,
93 VideoBitrateAllocationObserver* bitrate_updated_callback);
Erik Språng08127a92016-11-16 16:41:30 +010094
Miguel Casas-Sanchez47650702015-05-29 17:21:40 -070095 int32_t AddVideoFrame(const VideoFrame& videoFrame,
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000096 const CodecSpecificInfo* codecSpecificInfo);
97
perkj600246e2016-05-04 11:26:51 -070098 int32_t IntraFrameRequest(size_t stream_index);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +000099 int32_t EnableFrameDropper(bool enable);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000100
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000101 private:
Erik Språng08127a92016-11-16 16:41:30 +0100102 EncoderParameters UpdateEncoderParameters(
103 const EncoderParameters& params,
104 VideoBitrateAllocator* bitrate_allocator,
105 uint32_t target_bitrate_bps);
perkj57c21f92016-06-17 07:27:16 -0700106 void SetEncoderParameters(EncoderParameters params, bool has_internal_source)
danilchap56359be2017-09-07 07:53:45 -0700107 RTC_EXCLUSIVE_LOCKS_REQUIRED(encoder_crit_);
Peter Boströmdcb89982015-09-15 14:43:47 +0200108
pbos5ad935c2016-01-25 03:52:44 -0800109 rtc::CriticalSection encoder_crit_;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000110 VCMGenericEncoder* _encoder;
henrik.lundin@webrtc.orgbec11ef2013-09-23 19:54:25 +0000111 media_optimization::MediaOptimization _mediaOpt;
danilchap56359be2017-09-07 07:53:45 -0700112 VCMEncodedFrameCallback _encodedFrameCallback RTC_GUARDED_BY(encoder_crit_);
kthelgason876222f2016-11-29 01:44:11 -0800113 EncodedImageCallback* const post_encode_callback_;
Niels Möllerf9063782018-02-20 16:09:48 +0100114 VCMEncoderDataBase _codecDataBase RTC_GUARDED_BY(encoder_crit_);
Erik Språngd3438aa2018-11-08 16:56:43 +0100115
116 // |frame_dropper_requested_| specifies if the user of this class has
117 // requested frame dropping to be enabled, via EnableFrameDropper().
118 // Depending on video encoder configuration, this setting may be overridden
119 // and the frame dropper be force disabled. If so,
120 // |force_disable_frame_dropper_| will be set to true.
121 // If frame dropper is requested, and is not force disabled, frame dropping
122 // might still be disabled if VideoEncoder::GetEncoderInfo() indicates that
123 // the encoder has a trusted rate controller. This is determined on a
124 // per-frame basis, as the encoder behavior might dynamically change.
125 bool frame_dropper_requested_ RTC_GUARDED_BY(encoder_crit_);
126 bool force_disable_frame_dropper_ RTC_GUARDED_BY(encoder_crit_);
andresp@webrtc.orge682aa52013-12-19 10:59:48 +0000127
tommi@webrtc.orge07710c2015-02-19 17:43:25 +0000128 // Must be accessed on the construction thread of VideoSender.
129 VideoCodec current_codec_;
perkj4e417b22016-07-14 23:35:55 -0700130 rtc::SequencedTaskChecker sequenced_checker_;
Erik Språng66a641a2015-06-11 14:20:07 +0200131
Peter Boström233bfd22016-01-18 20:23:40 +0100132 rtc::CriticalSection params_crit_;
danilchap56359be2017-09-07 07:53:45 -0700133 EncoderParameters encoder_params_ RTC_GUARDED_BY(params_crit_);
134 bool encoder_has_internal_source_ RTC_GUARDED_BY(params_crit_);
135 std::vector<FrameType> next_frame_types_ RTC_GUARDED_BY(params_crit_);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000136};
137
Peter Boström0b250722016-04-22 18:23:15 +0200138class VideoReceiver : public Module {
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000139 public:
philipel83f831a2016-03-12 03:30:23 -0800140 VideoReceiver(Clock* clock,
philipel721d4022016-12-15 07:10:57 -0800141 VCMTiming* timing,
philipel83f831a2016-03-12 03:30:23 -0800142 NackSender* nack_sender = nullptr,
143 KeyFrameRequestSender* keyframe_request_sender = nullptr);
Tommifbf3bce2018-02-21 15:56:05 +0100144 ~VideoReceiver() override;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000145
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000146 int32_t RegisterReceiveCodec(const VideoCodec* receiveCodec,
147 int32_t numberOfCores,
148 bool requireKeyFrame);
149
Peter Boström795dbe42015-11-27 14:09:07 +0100150 void RegisterExternalDecoder(VideoDecoder* externalDecoder,
perkj796cfaf2015-12-10 09:27:38 -0800151 uint8_t payloadType);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000152 int32_t RegisterReceiveCallback(VCMReceiveCallback* receiveCallback);
153 int32_t RegisterReceiveStatisticsCallback(
154 VCMReceiveStatisticsCallback* receiveStats);
155 int32_t RegisterFrameTypeCallback(VCMFrameTypeCallback* frameTypeCallback);
156 int32_t RegisterPacketRequestCallback(VCMPacketRequestCallback* callback);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000157
158 int32_t Decode(uint16_t maxWaitTimeMs);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000159
philipelfd5a20f2016-11-15 00:57:57 -0800160 int32_t Decode(const webrtc::VCMEncodedFrame* frame);
161
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000162 int32_t IncomingPacket(const uint8_t* incomingPayload,
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000163 size_t payloadLength,
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000164 const WebRtcRTPHeader& rtpInfo);
165 int32_t SetMinimumPlayoutDelay(uint32_t minPlayoutDelayMs);
166 int32_t SetRenderDelay(uint32_t timeMS);
167 int32_t Delay() const;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000168
tommia5c18d72017-03-20 10:43:23 -0700169 // DEPRECATED.
170 int SetReceiverRobustnessMode(
171 VideoCodingModule::ReceiverRobustness robustnessMode,
172 VCMDecodeErrorMode errorMode);
173
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000174 void SetNackSettings(size_t max_nack_list_size,
175 int max_packet_age_to_nack,
176 int max_incomplete_time_ms);
177
178 void SetDecodeErrorMode(VCMDecodeErrorMode decode_error_mode);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000179
pkasting@chromium.org16825b12015-01-12 21:51:21 +0000180 int32_t SetReceiveChannelParameters(int64_t rtt);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000181 int32_t SetVideoProtection(VCMVideoProtection videoProtection, bool enable);
182
Peter Boström0b250722016-04-22 18:23:15 +0200183 int64_t TimeUntilNextProcess() override;
184 void Process() override;
Tommifbf3bce2018-02-21 15:56:05 +0100185 void ProcessThreadAttached(ProcessThread* process_thread) override;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000186
pbos@webrtc.org4dd40d62015-02-17 13:22:43 +0000187 void TriggerDecoderShutdown();
sprang@webrtc.org40709352013-11-26 11:41:59 +0000188
Tommifbf3bce2018-02-21 15:56:05 +0100189 // Notification methods that are used to check our internal state and validate
190 // threading assumptions. These are called by VideoReceiveStream.
191 // See |IsDecoderThreadRunning()| for more details.
192 void DecoderThreadStarting();
193 void DecoderThreadStopped();
194
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000195 protected:
Tommifbf3bce2018-02-21 15:56:05 +0100196 int32_t Decode(const webrtc::VCMEncodedFrame& frame);
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000197 int32_t RequestKeyFrame();
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000198
199 private:
Tommifbf3bce2018-02-21 15:56:05 +0100200 // Used for DCHECKing thread correctness.
201 // In build where DCHECKs are enabled, will return false before
202 // DecoderThreadStarting is called, then true until DecoderThreadStopped
203 // is called.
204 // In builds where DCHECKs aren't enabled, it will return true.
205 bool IsDecoderThreadRunning();
206
207 rtc::ThreadChecker construction_thread_checker_;
208 rtc::ThreadChecker decoder_thread_checker_;
209 rtc::ThreadChecker module_thread_checker_;
pbos@webrtc.org20c1f562014-07-04 10:58:12 +0000210 Clock* const clock_;
sprang3911c262016-04-15 01:24:14 -0700211 rtc::CriticalSection process_crit_;
philipel721d4022016-12-15 07:10:57 -0800212 VCMTiming* _timing;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000213 VCMReceiver _receiver;
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000214 VCMDecodedFrameCallback _decodedFrameCallback;
niklase@google.com470e71d2011-07-07 08:21:25 +0000215
Tommifbf3bce2018-02-21 15:56:05 +0100216 // These callbacks are set on the construction thread before being attached
217 // to the module thread or decoding started, so a lock is not required.
218 VCMFrameTypeCallback* _frameTypeCallback;
219 VCMReceiveStatisticsCallback* _receiveStatsCallback;
220 VCMPacketRequestCallback* _packetRequestCallback;
221
222 // Used on both the module and decoder thread.
danilchap56359be2017-09-07 07:53:45 -0700223 bool _scheduleKeyRequest RTC_GUARDED_BY(process_crit_);
224 bool drop_frames_until_keyframe_ RTC_GUARDED_BY(process_crit_);
sprang3911c262016-04-15 01:24:14 -0700225
Tommifbf3bce2018-02-21 15:56:05 +0100226 // Modified on the construction thread while not attached to the process
227 // thread. Once attached to the process thread, its value is only read
228 // so a lock is not required.
229 size_t max_nack_list_size_;
Peter Boströmed3277b2016-02-02 15:40:04 +0100230
Tommifbf3bce2018-02-21 15:56:05 +0100231 // Callbacks are set before the decoder thread starts.
232 // Once the decoder thread has been started, usage of |_codecDataBase| moves
233 // over to the decoder thread.
234 VCMDecoderDataBase _codecDataBase;
Tommifbf3bce2018-02-21 15:56:05 +0100235
236 VCMProcessTimer _receiveStatsTimer RTC_GUARDED_BY(module_thread_checker_);
237 VCMProcessTimer _retransmissionTimer RTC_GUARDED_BY(module_thread_checker_);
238 VCMProcessTimer _keyRequestTimer RTC_GUARDED_BY(module_thread_checker_);
Tommifbf3bce2018-02-21 15:56:05 +0100239 ThreadUnsafeOneTimeEvent first_frame_received_
240 RTC_GUARDED_BY(decoder_thread_checker_);
241 // Modified on the construction thread. Can be read without a lock and assumed
242 // to be non-null on the module and decoder threads.
243 ProcessThread* process_thread_ = nullptr;
244 bool is_attached_to_process_thread_
245 RTC_GUARDED_BY(construction_thread_checker_) = false;
246#if RTC_DCHECK_IS_ON
247 bool decoder_thread_is_running_ = false;
248#endif
niklase@google.com470e71d2011-07-07 08:21:25 +0000249};
andresp@webrtc.orgf7eb75b2013-09-14 00:25:28 +0000250
251} // namespace vcm
pbos@webrtc.orgd900e8b2013-07-03 15:12:26 +0000252} // namespace webrtc
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200253#endif // MODULES_VIDEO_CODING_VIDEO_CODING_IMPL_H_