blob: f593cf2a81928a3fe4e47c201d18221fee8b3e0e [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
stefan@webrtc.org07b45a52012-02-02 08:37: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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#ifndef VIDEO_VIDEO_STREAM_ENCODER_H_
12#define VIDEO_VIDEO_STREAM_ENCODER_H_
niklase@google.com470e71d2011-07-07 08:21:25 +000013
Yuwei Huangd9f99c12017-10-24 15:40:52 -070014#include <atomic>
sprangc5d62e22017-04-02 23:53:04 -070015#include <map>
kwiberg27f982b2016-03-01 11:52:33 -080016#include <memory>
perkj376b1922016-05-02 11:35:24 -070017#include <string>
mflodman@webrtc.org02270cd2015-02-06 13:10:19 +000018#include <vector>
mflodman@webrtc.orgd6ec3862012-10-25 11:30:29 +000019
philipeld9cc8c02019-09-16 14:53:40 +020020#include "api/units/data_rate.h"
Jiawei Ou4206a0a2018-07-20 15:49:43 -070021#include "api/video/video_bitrate_allocator.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "api/video/video_rotation.h"
Niels Möllerc6ce9c52018-05-11 11:15:30 +020023#include "api/video/video_sink_interface.h"
Niels Möller0327c2d2018-05-21 14:09:31 +020024#include "api/video/video_stream_encoder_interface.h"
Niels Möller213618e2018-07-24 09:29:58 +020025#include "api/video/video_stream_encoder_observer.h"
26#include "api/video/video_stream_encoder_settings.h"
Erik Språng6a7baa72019-02-26 18:31:00 +010027#include "api/video_codecs/video_codec.h"
Niels Möller0327c2d2018-05-21 14:09:31 +020028#include "api/video_codecs/video_encoder.h"
Henrik Boström62057622020-03-10 19:08:05 +010029#include "call/adaptation/resource_adaptation_processor_interface.h"
Henrik Boströmce0ea492020-01-13 11:27:18 +010030#include "call/adaptation/video_source_restrictions.h"
Henrik Boströmd516b252020-04-17 12:10:59 +020031#include "call/adaptation/video_stream_input_state_provider.h"
Niels Möller6bb5ab92019-01-11 11:11:10 +010032#include "modules/video_coding/utility/frame_dropper.h"
philipelda5aa4d2019-04-26 13:37:37 +020033#include "rtc_base/critical_section.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020034#include "rtc_base/event.h"
Erik Språng7ca375c2019-02-06 16:20:17 +010035#include "rtc_base/experiments/rate_control_settings.h"
philipeld9cc8c02019-09-16 14:53:40 +020036#include "rtc_base/numerics/exp_filter.h"
Erik Språng6a7baa72019-02-26 18:31:00 +010037#include "rtc_base/race_checker.h"
Niels Möller6bb5ab92019-01-11 11:11:10 +010038#include "rtc_base/rate_statistics.h"
Sebastian Janssonb55015e2019-04-09 13:44:04 +020039#include "rtc_base/synchronization/sequence_checker.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020040#include "rtc_base/task_queue.h"
Niels Möllerfe407b72019-09-10 10:48:48 +020041#include "system_wrappers/include/clock.h"
Henrik Boströmde8d5512020-04-17 17:15:26 +020042#include "video/adaptation/video_stream_encoder_resource_manager.h"
Erik Språng7ca375c2019-02-06 16:20:17 +010043#include "video/encoder_bitrate_adjuster.h"
Ilya Nikolaevskiy2ebf5232019-05-13 16:13:36 +020044#include "video/frame_encode_metadata_writer.h"
Henrik Boströmce0ea492020-01-13 11:27:18 +010045#include "video/video_source_sink_controller.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000046
47namespace webrtc {
mflodman@webrtc.org84d17832011-12-01 17:02:23 +000048
Sebastian Jansson652dc912018-04-19 17:09:15 +020049// VideoStreamEncoder represent a video encoder that accepts raw video frames as
50// input and produces an encoded bit stream.
51// Usage:
52// Instantiate.
53// Call SetSink.
54// Call SetSource.
55// Call ConfigureEncoder with the codec settings.
56// Call Stop() when done.
57class VideoStreamEncoder : public VideoStreamEncoderInterface,
Henrik Boströmd2382002020-01-10 15:44:01 +010058 private EncodedImageCallback,
Henrik Boström62057622020-03-10 19:08:05 +010059 public ResourceAdaptationProcessorListener {
Sebastian Jansson652dc912018-04-19 17:09:15 +020060 public:
Sebastian Jansson572c60f2019-03-04 18:30:41 +010061 VideoStreamEncoder(Clock* clock,
62 uint32_t number_of_cores,
Niels Möller213618e2018-07-24 09:29:58 +020063 VideoStreamEncoderObserver* encoder_stats_observer,
64 const VideoStreamEncoderSettings& settings,
Sebastian Jansson74682c12019-03-01 11:50:20 +010065 std::unique_ptr<OveruseFrameDetector> overuse_detector,
66 TaskQueueFactory* task_queue_factory);
Stefan Holmerdbdb3a02018-07-17 16:03:46 +020067 ~VideoStreamEncoder() override;
niklase@google.com470e71d2011-07-07 08:21:25 +000068
Sebastian Jansson652dc912018-04-19 17:09:15 +020069 void SetSource(rtc::VideoSourceInterface<VideoFrame>* source,
Taylor Brandstetter49fcc102018-05-16 14:20:41 -070070 const DegradationPreference& degradation_preference) override;
perkj803d97f2016-11-01 11:45:46 -070071
Sebastian Jansson652dc912018-04-19 17:09:15 +020072 void SetSink(EncoderSink* sink, bool rotation_applied) override;
mflodman@webrtc.org02270cd2015-02-06 13:10:19 +000073
perkj26091b12016-09-01 01:17:40 -070074 // TODO(perkj): Can we remove VideoCodec.startBitrate ?
Sebastian Jansson652dc912018-04-19 17:09:15 +020075 void SetStartBitrate(int start_bitrate_bps) override;
mflodman@webrtc.org9ec883e2012-03-05 17:12:41 +000076
Niels Möller0327c2d2018-05-21 14:09:31 +020077 void SetBitrateAllocationObserver(
Sebastian Jansson652dc912018-04-19 17:09:15 +020078 VideoBitrateAllocationObserver* bitrate_observer) override;
sprang1a646ee2016-12-01 06:34:11 -080079
Elad Alon8f01c4e2019-06-28 15:19:43 +020080 void SetFecControllerOverride(
81 FecControllerOverride* fec_controller_override) override;
82
Per512ecb32016-09-23 15:52:06 +020083 void ConfigureEncoder(VideoEncoderConfig config,
Niels Möllerf1338562018-04-26 09:51:47 +020084 size_t max_data_payload_length) override;
niklase@google.com470e71d2011-07-07 08:21:25 +000085
perkj26091b12016-09-01 01:17:40 -070086 // Permanently stop encoding. After this method has returned, it is
87 // guaranteed that no encoded frames will be delivered to the sink.
Sebastian Jansson652dc912018-04-19 17:09:15 +020088 void Stop() override;
perkj26091b12016-09-01 01:17:40 -070089
Sebastian Jansson652dc912018-04-19 17:09:15 +020090 void SendKeyFrame() override;
mflodman@webrtc.orgd6ec3862012-10-25 11:30:29 +000091
Elad Alonb6ef99b2019-04-10 16:37:07 +020092 void OnLossNotification(
93 const VideoEncoder::LossNotification& loss_notification) override;
94
Erik Språng610c7632019-03-06 15:37:33 +010095 void OnBitrateUpdated(DataRate target_bitrate,
Florent Castellia8336d32019-09-09 13:36:55 +020096 DataRate stable_target_bitrate,
Erik Språng610c7632019-03-06 15:37:33 +010097 DataRate target_headroom,
stefan@webrtc.orgedeea912014-12-08 19:46:23 +000098 uint8_t fraction_lost,
Ying Wang9b881ab2020-02-07 14:29:32 +010099 int64_t round_trip_time_ms,
100 double cwnd_reduce_ratio) override;
101
102 DataRate UpdateTargetBitrate(DataRate target_bitrate,
103 double cwnd_reduce_ratio);
pwestin@webrtc.org49888ce2012-04-27 05:25:53 +0000104
perkj803d97f2016-11-01 11:45:46 -0700105 protected:
kthelgason876222f2016-11-29 01:44:11 -0800106 // Used for testing. For example the |ScalingObserverInterface| methods must
107 // be called on |encoder_queue_|.
perkj803d97f2016-11-01 11:45:46 -0700108 rtc::TaskQueue* encoder_queue() { return &encoder_queue_; }
109
Henrik Boströmd2382002020-01-10 15:44:01 +0100110 void OnVideoSourceRestrictionsUpdated(
Henrik Boströmd2930c62020-04-17 15:31:48 +0200111 VideoSourceRestrictions restrictions,
112 const VideoAdaptationCounters& adaptation_counters,
113 const Resource* reason) override;
Henrik Boströmd2382002020-01-10 15:44:01 +0100114
Evan Shrubsoleaa6fbc12020-02-25 16:26:01 +0100115 // Used for injected test resources.
116 // TODO(eshr): Move all adaptation tests out of VideoStreamEncoder tests.
117 void InjectAdaptationResource(Resource* resource,
Evan Shrubsolece0a11d2020-04-16 11:36:55 +0200118 VideoAdaptationReason reason)
Evan Shrubsoleaa6fbc12020-02-25 16:26:01 +0100119 RTC_RUN_ON(&encoder_queue_);
120
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000121 private:
kthelgason93f16d72017-01-16 06:15:23 -0800122 class VideoFrameInfo {
123 public:
Yves Gerey665174f2018-06-19 15:03:05 +0200124 VideoFrameInfo(int width, int height, bool is_texture)
125 : width(width), height(height), is_texture(is_texture) {}
perkjfa10b552016-10-02 23:45:26 -0700126 int width;
127 int height;
perkjfa10b552016-10-02 23:45:26 -0700128 bool is_texture;
kthelgason93f16d72017-01-16 06:15:23 -0800129 int pixel_count() const { return width * height; }
perkjfa10b552016-10-02 23:45:26 -0700130 };
131
Evan Shrubsole7c079f62019-09-26 09:55:03 +0200132 struct EncoderRateSettings {
Erik Språng4c6ca302019-04-08 15:14:01 +0200133 EncoderRateSettings();
134 EncoderRateSettings(const VideoBitrateAllocation& bitrate,
135 double framerate_fps,
136 DataRate bandwidth_allocation,
Florent Castellia8336d32019-09-09 13:36:55 +0200137 DataRate encoder_target,
138 DataRate stable_encoder_target);
Erik Språng4c6ca302019-04-08 15:14:01 +0200139 bool operator==(const EncoderRateSettings& rhs) const;
140 bool operator!=(const EncoderRateSettings& rhs) const;
141
Evan Shrubsole7c079f62019-09-26 09:55:03 +0200142 VideoEncoder::RateControlParameters rate_control;
Erik Språng4c6ca302019-04-08 15:14:01 +0200143 // This is the scalar target bitrate before the VideoBitrateAllocator, i.e.
144 // the |target_bitrate| argument of the OnBitrateUpdated() method. This is
145 // needed because the bitrate allocator may truncate the total bitrate and a
146 // later call to the same allocator instance, e.g.
147 // |using last_encoder_rate_setings_->bitrate.get_sum_bps()|, may trick it
148 // into thinking the available bitrate has decreased since the last call.
149 DataRate encoder_target;
Florent Castellia8336d32019-09-09 13:36:55 +0200150 DataRate stable_encoder_target;
Erik Språng4c6ca302019-04-08 15:14:01 +0200151 };
152
Niels Möllera8b15082018-02-07 13:42:09 +0100153 void ReconfigureEncoder() RTC_RUN_ON(&encoder_queue_);
Henrik Boströmd516b252020-04-17 12:10:59 +0200154 void OnEncoderSettingsChanged() RTC_RUN_ON(&encoder_queue_);
perkj26091b12016-09-01 01:17:40 -0700155
perkja49cbd32016-09-16 07:53:41 -0700156 // Implements VideoSinkInterface.
157 void OnFrame(const VideoFrame& video_frame) override;
Ilya Nikolaevskiyd79314f2017-10-23 10:45:37 +0200158 void OnDiscardedFrame() override;
perkja49cbd32016-09-16 07:53:41 -0700159
Sebastian Janssona3177052018-04-10 13:05:49 +0200160 void MaybeEncodeVideoFrame(const VideoFrame& frame,
161 int64_t time_when_posted_in_ms);
162
perkjd52063f2016-09-07 06:32:18 -0700163 void EncodeVideoFrame(const VideoFrame& frame,
164 int64_t time_when_posted_in_ms);
Sebastian Janssona3177052018-04-10 13:05:49 +0200165 // Indicates wether frame should be dropped because the pixel count is too
166 // large for the current bitrate configuration.
167 bool DropDueToSize(uint32_t pixel_count) const RTC_RUN_ON(&encoder_queue_);
perkj26091b12016-09-01 01:17:40 -0700168
169 // Implements EncodedImageCallback.
170 EncodedImageCallback::Result OnEncodedImage(
171 const EncodedImage& encoded_image,
172 const CodecSpecificInfo* codec_specific_info,
173 const RTPFragmentationHeader* fragmentation) override;
174
Ilya Nikolaevskiyd79314f2017-10-23 10:45:37 +0200175 void OnDroppedFrame(EncodedImageCallback::DropReason reason) override;
kthelgason876222f2016-11-29 01:44:11 -0800176
perkj26091b12016-09-01 01:17:40 -0700177 bool EncoderPaused() const;
178 void TraceFrameDropStart();
179 void TraceFrameDropEnd();
180
Erik Språng4c6ca302019-04-08 15:14:01 +0200181 // Returns a copy of |rate_settings| with the |bitrate| field updated using
182 // the current VideoBitrateAllocator, and notifies any listeners of the new
183 // allocation.
184 EncoderRateSettings UpdateBitrateAllocationAndNotifyObserver(
185 const EncoderRateSettings& rate_settings) RTC_RUN_ON(&encoder_queue_);
186
Niels Möller6bb5ab92019-01-11 11:11:10 +0100187 uint32_t GetInputFramerateFps() RTC_RUN_ON(&encoder_queue_);
Erik Språng4c6ca302019-04-08 15:14:01 +0200188 void SetEncoderRates(const EncoderRateSettings& rate_settings)
189 RTC_RUN_ON(&encoder_queue_);
Niels Möller6bb5ab92019-01-11 11:11:10 +0100190
Evan Shrubsole2bc91e82020-01-17 11:44:24 +0100191 void RunPostEncode(const EncodedImage& encoded_image,
Niels Möller6bb5ab92019-01-11 11:11:10 +0100192 int64_t time_sent_us,
philipelc7a46c42019-11-25 15:01:09 +0100193 int temporal_index,
194 DataSize frame_size);
Erik Språngd7329ca2019-02-21 21:19:53 +0100195 bool HasInternalSource() const RTC_RUN_ON(&encoder_queue_);
Erik Språng6a7baa72019-02-26 18:31:00 +0100196 void ReleaseEncoder() RTC_RUN_ON(&encoder_queue_);
sprangc5d62e22017-04-02 23:53:04 -0700197
Ilya Nikolaevskiy648b9d72019-12-03 16:54:17 +0100198 void CheckForAnimatedContent(const VideoFrame& frame,
199 int64_t time_when_posted_in_ms)
200 RTC_RUN_ON(&encoder_queue_);
201
perkj26091b12016-09-01 01:17:40 -0700202 rtc::Event shutdown_event_;
stefan@webrtc.orgbfacda62013-03-27 16:36:01 +0000203
pbos@webrtc.orgb238d122013-04-09 13:41:51 +0000204 const uint32_t number_of_cores_;
perkja49cbd32016-09-16 07:53:41 -0700205
Åsa Perssona945aee2018-04-24 16:53:25 +0200206 const bool quality_scaling_experiment_enabled_;
207
Per512ecb32016-09-23 15:52:06 +0200208 EncoderSink* sink_;
Niels Möller213618e2018-07-24 09:29:58 +0200209 const VideoStreamEncoderSettings settings_;
Erik Språng7ca375c2019-02-06 16:20:17 +0100210 const RateControlSettings rate_control_settings_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000211
philipel9b058032020-02-10 11:30:00 +0100212 std::unique_ptr<VideoEncoderFactory::EncoderSelectorInterface> const
213 encoder_selector_;
Niels Möller213618e2018-07-24 09:29:58 +0200214 VideoStreamEncoderObserver* const encoder_stats_observer_;
perkja49cbd32016-09-16 07:53:41 -0700215 // |thread_checker_| checks that public methods that are related to lifetime
mflodmancc3d4422017-08-03 08:27:51 -0700216 // of VideoStreamEncoder are called on the same thread.
perkja49cbd32016-09-16 07:53:41 -0700217 rtc::ThreadChecker thread_checker_;
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000218
Niels Möller1e062892018-02-07 10:18:32 +0100219 VideoEncoderConfig encoder_config_ RTC_GUARDED_BY(&encoder_queue_);
Niels Möller4db138e2018-04-19 09:04:13 +0200220 std::unique_ptr<VideoEncoder> encoder_ RTC_GUARDED_BY(&encoder_queue_)
221 RTC_PT_GUARDED_BY(&encoder_queue_);
Erik Språng6a7baa72019-02-26 18:31:00 +0100222 bool encoder_initialized_;
Erik Språng08127a92016-11-16 16:41:30 +0100223 std::unique_ptr<VideoBitrateAllocator> rate_allocator_
Yves Gerey665174f2018-06-19 15:03:05 +0200224 RTC_GUARDED_BY(&encoder_queue_) RTC_PT_GUARDED_BY(&encoder_queue_);
Niels Möller1e062892018-02-07 10:18:32 +0100225 int max_framerate_ RTC_GUARDED_BY(&encoder_queue_);
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000226
perkjfa10b552016-10-02 23:45:26 -0700227 // Set when ConfigureEncoder has been called in order to lazy reconfigure the
228 // encoder on the next frame.
Niels Möller1e062892018-02-07 10:18:32 +0100229 bool pending_encoder_reconfiguration_ RTC_GUARDED_BY(&encoder_queue_);
Mirta Dvornicic1ec2a162018-12-10 09:47:34 +0000230 // Set when configuration must create a new encoder object, e.g.,
231 // because of a codec change.
232 bool pending_encoder_creation_ RTC_GUARDED_BY(&encoder_queue_);
Sergey Silkin5ee69672019-07-02 14:18:34 +0200233
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200234 absl::optional<VideoFrameInfo> last_frame_info_
Niels Möller1e062892018-02-07 10:18:32 +0100235 RTC_GUARDED_BY(&encoder_queue_);
236 int crop_width_ RTC_GUARDED_BY(&encoder_queue_);
237 int crop_height_ RTC_GUARDED_BY(&encoder_queue_);
Henrik Boströmede69c02020-01-21 17:45:35 +0100238 absl::optional<uint32_t> encoder_target_bitrate_bps_
239 RTC_GUARDED_BY(&encoder_queue_);
Niels Möller1e062892018-02-07 10:18:32 +0100240 size_t max_data_payload_length_ RTC_GUARDED_BY(&encoder_queue_);
Erik Språng4c6ca302019-04-08 15:14:01 +0200241 absl::optional<EncoderRateSettings> last_encoder_rate_settings_
242 RTC_GUARDED_BY(&encoder_queue_);
Niels Möller1e062892018-02-07 10:18:32 +0100243 bool encoder_paused_and_dropped_frame_ RTC_GUARDED_BY(&encoder_queue_);
Sergey Silkin5ee69672019-07-02 14:18:34 +0200244
245 // Set to true if at least one frame was sent to encoder since last encoder
246 // initialization.
247 bool was_encode_called_since_last_initialization_
248 RTC_GUARDED_BY(&encoder_queue_);
249
philipele8ed8302019-07-03 11:53:48 +0200250 bool encoder_failed_ RTC_GUARDED_BY(&encoder_queue_);
perkj26091b12016-09-01 01:17:40 -0700251 Clock* const clock_;
perkj803d97f2016-11-01 11:45:46 -0700252
253 rtc::RaceChecker incoming_frame_race_checker_
danilchapa37de392017-09-09 04:17:22 -0700254 RTC_GUARDED_BY(incoming_frame_race_checker_);
Yuwei Huangd9f99c12017-10-24 15:40:52 -0700255 std::atomic<int> posted_frames_waiting_for_encode_;
perkj26091b12016-09-01 01:17:40 -0700256 // Used to make sure incoming time stamp is increasing for every frame.
danilchapa37de392017-09-09 04:17:22 -0700257 int64_t last_captured_timestamp_ RTC_GUARDED_BY(incoming_frame_race_checker_);
perkj26091b12016-09-01 01:17:40 -0700258 // Delta used for translating between NTP and internal timestamps.
danilchapa37de392017-09-09 04:17:22 -0700259 const int64_t delta_ntp_internal_ms_
260 RTC_GUARDED_BY(incoming_frame_race_checker_);
perkj26091b12016-09-01 01:17:40 -0700261
danilchapa37de392017-09-09 04:17:22 -0700262 int64_t last_frame_log_ms_ RTC_GUARDED_BY(incoming_frame_race_checker_);
Niels Möller1e062892018-02-07 10:18:32 +0100263 int captured_frame_count_ RTC_GUARDED_BY(&encoder_queue_);
Ying Wang9b881ab2020-02-07 14:29:32 +0100264 int dropped_frame_cwnd_pushback_count_ RTC_GUARDED_BY(&encoder_queue_);
265 int dropped_frame_encoder_block_count_ RTC_GUARDED_BY(&encoder_queue_);
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200266 absl::optional<VideoFrame> pending_frame_ RTC_GUARDED_BY(&encoder_queue_);
Sebastian Janssona3177052018-04-10 13:05:49 +0200267 int64_t pending_frame_post_time_us_ RTC_GUARDED_BY(&encoder_queue_);
asapersson6ffb67d2016-09-12 00:10:45 -0700268
Ilya Nikolaevskiy71aee3a2019-02-18 13:01:26 +0100269 VideoFrame::UpdateRect accumulated_update_rect_
270 RTC_GUARDED_BY(&encoder_queue_);
Ilya Nikolaevskiy9560d7d2019-10-30 11:19:47 +0100271 bool accumulated_update_rect_is_valid_ RTC_GUARDED_BY(&encoder_queue_);
Ilya Nikolaevskiy71aee3a2019-02-18 13:01:26 +0100272
Ilya Nikolaevskiy648b9d72019-12-03 16:54:17 +0100273 // Used for automatic content type detection.
274 absl::optional<VideoFrame::UpdateRect> last_update_rect_
275 RTC_GUARDED_BY(&encoder_queue_);
276 Timestamp animation_start_time_ RTC_GUARDED_BY(&encoder_queue_);
277 bool cap_resolution_due_to_video_content_ RTC_GUARDED_BY(&encoder_queue_);
278 // Used to correctly ignore changes in update_rect introduced by
279 // resize triggered by animation detection.
280 enum class ExpectResizeState {
281 kNoResize, // Normal operation.
282 kResize, // Resize was triggered by the animation detection.
283 kFirstFrameAfterResize // Resize observed.
284 } expect_resize_state_ RTC_GUARDED_BY(&encoder_queue_);
285
danilchapa37de392017-09-09 04:17:22 -0700286 VideoBitrateAllocationObserver* bitrate_observer_
Niels Möller1e062892018-02-07 10:18:32 +0100287 RTC_GUARDED_BY(&encoder_queue_);
Elad Alon8f01c4e2019-06-28 15:19:43 +0200288 FecControllerOverride* fec_controller_override_
289 RTC_GUARDED_BY(&encoder_queue_);
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200290 absl::optional<int64_t> last_parameters_update_ms_
Niels Möller1e062892018-02-07 10:18:32 +0100291 RTC_GUARDED_BY(&encoder_queue_);
Åsa Perssonc29cb2c2019-03-25 12:06:59 +0100292 absl::optional<int64_t> last_encode_info_ms_ RTC_GUARDED_BY(&encoder_queue_);
sprang1a646ee2016-12-01 06:34:11 -0800293
Erik Språnge2fd86a2018-10-24 11:32:39 +0200294 VideoEncoder::EncoderInfo encoder_info_ RTC_GUARDED_BY(&encoder_queue_);
Sergey Silkin6456e352019-07-08 17:56:40 +0200295 absl::optional<VideoEncoder::ResolutionBitrateLimits> encoder_bitrate_limits_
296 RTC_GUARDED_BY(&encoder_queue_);
Erik Språngd7329ca2019-02-21 21:19:53 +0100297 VideoEncoderFactory::CodecInfo codec_info_ RTC_GUARDED_BY(&encoder_queue_);
Erik Språng6a7baa72019-02-26 18:31:00 +0100298 VideoCodec send_codec_ RTC_GUARDED_BY(&encoder_queue_);
Niels Möller6bb5ab92019-01-11 11:11:10 +0100299
Erik Språng6a7baa72019-02-26 18:31:00 +0100300 FrameDropper frame_dropper_ RTC_GUARDED_BY(&encoder_queue_);
Niels Möller6bb5ab92019-01-11 11:11:10 +0100301 // If frame dropper is not force disabled, frame dropping might still be
302 // disabled if VideoEncoder::GetEncoderInfo() indicates that the encoder has a
303 // trusted rate controller. This is determined on a per-frame basis, as the
304 // encoder behavior might dynamically change.
305 bool force_disable_frame_dropper_ RTC_GUARDED_BY(&encoder_queue_);
306 RateStatistics input_framerate_ RTC_GUARDED_BY(&encoder_queue_);
307 // Incremented on worker thread whenever |frame_dropper_| determines that a
308 // frame should be dropped. Decremented on whichever thread runs
309 // OnEncodedImage(), which is only called by one thread but not necessarily
310 // the worker thread.
311 std::atomic<int> pending_frame_drops_;
Erik Språnge2fd86a2018-10-24 11:32:39 +0200312
Ying Wang9b881ab2020-02-07 14:29:32 +0100313 // Congestion window frame drop ratio (drop 1 in every
314 // cwnd_frame_drop_interval_ frames).
315 absl::optional<int> cwnd_frame_drop_interval_ RTC_GUARDED_BY(&encoder_queue_);
316 // Frame counter for congestion window frame drop.
317 int cwnd_frame_counter_ RTC_GUARDED_BY(&encoder_queue_);
318
Erik Språng7ca375c2019-02-06 16:20:17 +0100319 std::unique_ptr<EncoderBitrateAdjuster> bitrate_adjuster_
320 RTC_GUARDED_BY(&encoder_queue_);
321
Erik Språngd7329ca2019-02-21 21:19:53 +0100322 // TODO(sprang): Change actually support keyframe per simulcast stream, or
323 // turn this into a simple bool |pending_keyframe_request_|.
Niels Möller87e2d782019-03-07 10:18:23 +0100324 std::vector<VideoFrameType> next_frame_types_ RTC_GUARDED_BY(&encoder_queue_);
Erik Språngd7329ca2019-02-21 21:19:53 +0100325
Ilya Nikolaevskiy2ebf5232019-05-13 16:13:36 +0200326 FrameEncodeMetadataWriter frame_encode_metadata_writer_;
Erik Språng6a7baa72019-02-26 18:31:00 +0100327
328 // Experiment groups parsed from field trials for realtime video ([0]) and
329 // screenshare ([1]). 0 means no group specified. Positive values are
330 // experiment group numbers incremented by 1.
331 const std::array<uint8_t, 2> experiment_groups_;
332
philipelda5aa4d2019-04-26 13:37:37 +0200333 // TODO(philipel): Remove this lock and run on |encoder_queue_| instead.
334 rtc::CriticalSection encoded_image_lock_;
335
336 int64_t next_frame_id_ RTC_GUARDED_BY(encoded_image_lock_);
337
338 // This array is used as a map from simulcast id to an encoder's buffer
339 // state. For every buffer of the encoder we keep track of the last frame id
340 // that updated that buffer.
341 std::array<std::array<int64_t, kMaxEncoderBuffers>, kMaxSimulcastStreams>
342 encoder_buffer_state_ RTC_GUARDED_BY(encoded_image_lock_);
343
philipeld9cc8c02019-09-16 14:53:40 +0200344 struct EncoderSwitchExperiment {
345 struct Thresholds {
346 absl::optional<DataRate> bitrate;
347 absl::optional<int> pixel_count;
348 };
349
350 // Codec --> switching thresholds
351 std::map<VideoCodecType, Thresholds> codec_thresholds;
352
353 // To smooth out the target bitrate so that we don't trigger a switch
354 // too easily.
355 rtc::ExpFilter bitrate_filter{1.0};
356
357 // Codec/implementation to switch to
358 std::string to_codec;
359 absl::optional<std::string> to_param;
360 absl::optional<std::string> to_value;
361
362 // Thresholds for the currently used codecs.
363 Thresholds current_thresholds;
364
365 // Updates the |bitrate_filter|, so not const.
366 bool IsBitrateBelowThreshold(const DataRate& target_bitrate);
367 bool IsPixelCountBelowThreshold(int pixel_count) const;
368 void SetCodec(VideoCodecType codec);
369 };
370
371 EncoderSwitchExperiment ParseEncoderSwitchFieldTrial() const;
372
373 EncoderSwitchExperiment encoder_switch_experiment_
374 RTC_GUARDED_BY(&encoder_queue_);
375
Ilya Nikolaevskiy648b9d72019-12-03 16:54:17 +0100376 struct AutomaticAnimationDetectionExperiment {
377 bool enabled = false;
378 int min_duration_ms = 2000;
379 double min_area_ratio = 0.8;
380 int min_fps = 10;
381 std::unique_ptr<StructParametersParser> Parser() {
382 return StructParametersParser::Create(
383 "enabled", &enabled, //
384 "min_duration_ms", &min_duration_ms, //
385 "min_area_ratio", &min_area_ratio, //
386 "min_fps", &min_fps);
387 }
388 };
389
390 AutomaticAnimationDetectionExperiment
391 ParseAutomatincAnimationDetectionFieldTrial() const;
392
393 AutomaticAnimationDetectionExperiment
394 automatic_animation_detection_experiment_ RTC_GUARDED_BY(&encoder_queue_);
395
philipeld9cc8c02019-09-16 14:53:40 +0200396 // An encoder switch is only requested once, this variable is used to keep
397 // track of whether a request has been made or not.
398 bool encoder_switch_requested_ RTC_GUARDED_BY(&encoder_queue_);
399
Henrik Boström87eece92020-04-17 18:36:19 +0200400 // Provies video stream input states: current resolution and frame rate.
401 VideoStreamInputStateProvider input_state_provider_
402 RTC_GUARDED_BY(&encoder_queue_);
403 // Responsible for adapting input resolution or frame rate to ensure resources
404 // (e.g. CPU or bandwidth) are not overused.
405 ResourceAdaptationProcessor resource_adaptation_processor_
406 RTC_GUARDED_BY(&encoder_queue_);
407 // Handles input, output and stats reporting related to VideoStreamEncoder
408 // specific resources, such as "encode usage percent" measurements and "QP
409 // scaling". Also involved with various mitigations such as inital frame
410 // dropping.
411 VideoStreamEncoderResourceManager stream_resource_manager_
412 RTC_GUARDED_BY(&encoder_queue_);
413 // Carries out the VideoSourceRestrictions provided by the
414 // ResourceAdaptationProcessor, i.e. reconfigures the source of video frames
415 // to provide us with different resolution or frame rate.
Henrik Boström07b17df2020-01-15 11:42:12 +0100416 //
Henrik Boström87eece92020-04-17 18:36:19 +0200417 // Used on the |encoder_queue_| with a few exceptions:
Henrik Boström07b17df2020-01-15 11:42:12 +0100418 // - VideoStreamEncoder::SetSource() invokes SetSource().
419 // - VideoStreamEncoder::SetSink() invokes SetRotationApplied() and
420 // PushSourceSinkSettings().
421 // - VideoStreamEncoder::Stop() invokes SetSource().
Henrik Boström87eece92020-04-17 18:36:19 +0200422 VideoSourceSinkController video_source_sink_controller_;
Henrik Boströmb08882b2020-01-07 10:11:17 +0100423
philipele5d0fe02019-10-15 11:02:53 +0200424 // All public methods are proxied to |encoder_queue_|. It must must be
425 // destroyed first to make sure no tasks are run that use other members.
426 rtc::TaskQueue encoder_queue_;
427
mflodmancc3d4422017-08-03 08:27:51 -0700428 RTC_DISALLOW_COPY_AND_ASSIGN(VideoStreamEncoder);
niklase@google.com470e71d2011-07-07 08:21:25 +0000429};
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000430
431} // namespace webrtc
432
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200433#endif // VIDEO_VIDEO_STREAM_ENCODER_H_