blob: 8e36f517d434b525a5d49afbe3c27dcede106003 [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öm62057622020-03-10 19:08:05 +010042#include "video/adaptation/resource_adaptation_processor.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(
111 VideoSourceRestrictions restrictions) override;
112
Evan Shrubsoleaa6fbc12020-02-25 16:26:01 +0100113 // Used for injected test resources.
114 // TODO(eshr): Move all adaptation tests out of VideoStreamEncoder tests.
115 void InjectAdaptationResource(Resource* resource,
Evan Shrubsolece0a11d2020-04-16 11:36:55 +0200116 VideoAdaptationReason reason)
Evan Shrubsoleaa6fbc12020-02-25 16:26:01 +0100117 RTC_RUN_ON(&encoder_queue_);
118
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000119 private:
kthelgason93f16d72017-01-16 06:15:23 -0800120 class VideoFrameInfo {
121 public:
Yves Gerey665174f2018-06-19 15:03:05 +0200122 VideoFrameInfo(int width, int height, bool is_texture)
123 : width(width), height(height), is_texture(is_texture) {}
perkjfa10b552016-10-02 23:45:26 -0700124 int width;
125 int height;
perkjfa10b552016-10-02 23:45:26 -0700126 bool is_texture;
kthelgason93f16d72017-01-16 06:15:23 -0800127 int pixel_count() const { return width * height; }
perkjfa10b552016-10-02 23:45:26 -0700128 };
129
Evan Shrubsole7c079f62019-09-26 09:55:03 +0200130 struct EncoderRateSettings {
Erik Språng4c6ca302019-04-08 15:14:01 +0200131 EncoderRateSettings();
132 EncoderRateSettings(const VideoBitrateAllocation& bitrate,
133 double framerate_fps,
134 DataRate bandwidth_allocation,
Florent Castellia8336d32019-09-09 13:36:55 +0200135 DataRate encoder_target,
136 DataRate stable_encoder_target);
Erik Språng4c6ca302019-04-08 15:14:01 +0200137 bool operator==(const EncoderRateSettings& rhs) const;
138 bool operator!=(const EncoderRateSettings& rhs) const;
139
Evan Shrubsole7c079f62019-09-26 09:55:03 +0200140 VideoEncoder::RateControlParameters rate_control;
Erik Språng4c6ca302019-04-08 15:14:01 +0200141 // This is the scalar target bitrate before the VideoBitrateAllocator, i.e.
142 // the |target_bitrate| argument of the OnBitrateUpdated() method. This is
143 // needed because the bitrate allocator may truncate the total bitrate and a
144 // later call to the same allocator instance, e.g.
145 // |using last_encoder_rate_setings_->bitrate.get_sum_bps()|, may trick it
146 // into thinking the available bitrate has decreased since the last call.
147 DataRate encoder_target;
Florent Castellia8336d32019-09-09 13:36:55 +0200148 DataRate stable_encoder_target;
Erik Språng4c6ca302019-04-08 15:14:01 +0200149 };
150
Niels Möllera8b15082018-02-07 13:42:09 +0100151 void ReconfigureEncoder() RTC_RUN_ON(&encoder_queue_);
Henrik Boströmd516b252020-04-17 12:10:59 +0200152 void OnEncoderSettingsChanged() RTC_RUN_ON(&encoder_queue_);
perkj26091b12016-09-01 01:17:40 -0700153
perkja49cbd32016-09-16 07:53:41 -0700154 // Implements VideoSinkInterface.
155 void OnFrame(const VideoFrame& video_frame) override;
Ilya Nikolaevskiyd79314f2017-10-23 10:45:37 +0200156 void OnDiscardedFrame() override;
perkja49cbd32016-09-16 07:53:41 -0700157
Sebastian Janssona3177052018-04-10 13:05:49 +0200158 void MaybeEncodeVideoFrame(const VideoFrame& frame,
159 int64_t time_when_posted_in_ms);
160
perkjd52063f2016-09-07 06:32:18 -0700161 void EncodeVideoFrame(const VideoFrame& frame,
162 int64_t time_when_posted_in_ms);
Sebastian Janssona3177052018-04-10 13:05:49 +0200163 // Indicates wether frame should be dropped because the pixel count is too
164 // large for the current bitrate configuration.
165 bool DropDueToSize(uint32_t pixel_count) const RTC_RUN_ON(&encoder_queue_);
perkj26091b12016-09-01 01:17:40 -0700166
167 // Implements EncodedImageCallback.
168 EncodedImageCallback::Result OnEncodedImage(
169 const EncodedImage& encoded_image,
170 const CodecSpecificInfo* codec_specific_info,
171 const RTPFragmentationHeader* fragmentation) override;
172
Ilya Nikolaevskiyd79314f2017-10-23 10:45:37 +0200173 void OnDroppedFrame(EncodedImageCallback::DropReason reason) override;
kthelgason876222f2016-11-29 01:44:11 -0800174
perkj26091b12016-09-01 01:17:40 -0700175 bool EncoderPaused() const;
176 void TraceFrameDropStart();
177 void TraceFrameDropEnd();
178
Erik Språng4c6ca302019-04-08 15:14:01 +0200179 // Returns a copy of |rate_settings| with the |bitrate| field updated using
180 // the current VideoBitrateAllocator, and notifies any listeners of the new
181 // allocation.
182 EncoderRateSettings UpdateBitrateAllocationAndNotifyObserver(
183 const EncoderRateSettings& rate_settings) RTC_RUN_ON(&encoder_queue_);
184
Niels Möller6bb5ab92019-01-11 11:11:10 +0100185 uint32_t GetInputFramerateFps() RTC_RUN_ON(&encoder_queue_);
Erik Språng4c6ca302019-04-08 15:14:01 +0200186 void SetEncoderRates(const EncoderRateSettings& rate_settings)
187 RTC_RUN_ON(&encoder_queue_);
Niels Möller6bb5ab92019-01-11 11:11:10 +0100188
Evan Shrubsole2bc91e82020-01-17 11:44:24 +0100189 void RunPostEncode(const EncodedImage& encoded_image,
Niels Möller6bb5ab92019-01-11 11:11:10 +0100190 int64_t time_sent_us,
philipelc7a46c42019-11-25 15:01:09 +0100191 int temporal_index,
192 DataSize frame_size);
Erik Språngd7329ca2019-02-21 21:19:53 +0100193 bool HasInternalSource() const RTC_RUN_ON(&encoder_queue_);
Erik Språng6a7baa72019-02-26 18:31:00 +0100194 void ReleaseEncoder() RTC_RUN_ON(&encoder_queue_);
sprangc5d62e22017-04-02 23:53:04 -0700195
Ilya Nikolaevskiy648b9d72019-12-03 16:54:17 +0100196 void CheckForAnimatedContent(const VideoFrame& frame,
197 int64_t time_when_posted_in_ms)
198 RTC_RUN_ON(&encoder_queue_);
199
perkj26091b12016-09-01 01:17:40 -0700200 rtc::Event shutdown_event_;
stefan@webrtc.orgbfacda62013-03-27 16:36:01 +0000201
pbos@webrtc.orgb238d122013-04-09 13:41:51 +0000202 const uint32_t number_of_cores_;
perkja49cbd32016-09-16 07:53:41 -0700203
Åsa Perssona945aee2018-04-24 16:53:25 +0200204 const bool quality_scaling_experiment_enabled_;
205
Per512ecb32016-09-23 15:52:06 +0200206 EncoderSink* sink_;
Niels Möller213618e2018-07-24 09:29:58 +0200207 const VideoStreamEncoderSettings settings_;
Erik Språng7ca375c2019-02-06 16:20:17 +0100208 const RateControlSettings rate_control_settings_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000209
philipel9b058032020-02-10 11:30:00 +0100210 std::unique_ptr<VideoEncoderFactory::EncoderSelectorInterface> const
211 encoder_selector_;
Niels Möller213618e2018-07-24 09:29:58 +0200212 VideoStreamEncoderObserver* const encoder_stats_observer_;
perkja49cbd32016-09-16 07:53:41 -0700213 // |thread_checker_| checks that public methods that are related to lifetime
mflodmancc3d4422017-08-03 08:27:51 -0700214 // of VideoStreamEncoder are called on the same thread.
perkja49cbd32016-09-16 07:53:41 -0700215 rtc::ThreadChecker thread_checker_;
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000216
Niels Möller1e062892018-02-07 10:18:32 +0100217 VideoEncoderConfig encoder_config_ RTC_GUARDED_BY(&encoder_queue_);
Niels Möller4db138e2018-04-19 09:04:13 +0200218 std::unique_ptr<VideoEncoder> encoder_ RTC_GUARDED_BY(&encoder_queue_)
219 RTC_PT_GUARDED_BY(&encoder_queue_);
Erik Språng6a7baa72019-02-26 18:31:00 +0100220 bool encoder_initialized_;
Erik Språng08127a92016-11-16 16:41:30 +0100221 std::unique_ptr<VideoBitrateAllocator> rate_allocator_
Yves Gerey665174f2018-06-19 15:03:05 +0200222 RTC_GUARDED_BY(&encoder_queue_) RTC_PT_GUARDED_BY(&encoder_queue_);
Niels Möller1e062892018-02-07 10:18:32 +0100223 int max_framerate_ RTC_GUARDED_BY(&encoder_queue_);
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000224
perkjfa10b552016-10-02 23:45:26 -0700225 // Set when ConfigureEncoder has been called in order to lazy reconfigure the
226 // encoder on the next frame.
Niels Möller1e062892018-02-07 10:18:32 +0100227 bool pending_encoder_reconfiguration_ RTC_GUARDED_BY(&encoder_queue_);
Mirta Dvornicic1ec2a162018-12-10 09:47:34 +0000228 // Set when configuration must create a new encoder object, e.g.,
229 // because of a codec change.
230 bool pending_encoder_creation_ RTC_GUARDED_BY(&encoder_queue_);
Sergey Silkin5ee69672019-07-02 14:18:34 +0200231
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200232 absl::optional<VideoFrameInfo> last_frame_info_
Niels Möller1e062892018-02-07 10:18:32 +0100233 RTC_GUARDED_BY(&encoder_queue_);
234 int crop_width_ RTC_GUARDED_BY(&encoder_queue_);
235 int crop_height_ RTC_GUARDED_BY(&encoder_queue_);
Henrik Boströmede69c02020-01-21 17:45:35 +0100236 absl::optional<uint32_t> encoder_target_bitrate_bps_
237 RTC_GUARDED_BY(&encoder_queue_);
Niels Möller1e062892018-02-07 10:18:32 +0100238 size_t max_data_payload_length_ RTC_GUARDED_BY(&encoder_queue_);
Erik Språng4c6ca302019-04-08 15:14:01 +0200239 absl::optional<EncoderRateSettings> last_encoder_rate_settings_
240 RTC_GUARDED_BY(&encoder_queue_);
Niels Möller1e062892018-02-07 10:18:32 +0100241 bool encoder_paused_and_dropped_frame_ RTC_GUARDED_BY(&encoder_queue_);
Sergey Silkin5ee69672019-07-02 14:18:34 +0200242
243 // Set to true if at least one frame was sent to encoder since last encoder
244 // initialization.
245 bool was_encode_called_since_last_initialization_
246 RTC_GUARDED_BY(&encoder_queue_);
247
philipele8ed8302019-07-03 11:53:48 +0200248 bool encoder_failed_ RTC_GUARDED_BY(&encoder_queue_);
perkj26091b12016-09-01 01:17:40 -0700249 Clock* const clock_;
perkj803d97f2016-11-01 11:45:46 -0700250
251 rtc::RaceChecker incoming_frame_race_checker_
danilchapa37de392017-09-09 04:17:22 -0700252 RTC_GUARDED_BY(incoming_frame_race_checker_);
Yuwei Huangd9f99c12017-10-24 15:40:52 -0700253 std::atomic<int> posted_frames_waiting_for_encode_;
perkj26091b12016-09-01 01:17:40 -0700254 // Used to make sure incoming time stamp is increasing for every frame.
danilchapa37de392017-09-09 04:17:22 -0700255 int64_t last_captured_timestamp_ RTC_GUARDED_BY(incoming_frame_race_checker_);
perkj26091b12016-09-01 01:17:40 -0700256 // Delta used for translating between NTP and internal timestamps.
danilchapa37de392017-09-09 04:17:22 -0700257 const int64_t delta_ntp_internal_ms_
258 RTC_GUARDED_BY(incoming_frame_race_checker_);
perkj26091b12016-09-01 01:17:40 -0700259
danilchapa37de392017-09-09 04:17:22 -0700260 int64_t last_frame_log_ms_ RTC_GUARDED_BY(incoming_frame_race_checker_);
Niels Möller1e062892018-02-07 10:18:32 +0100261 int captured_frame_count_ RTC_GUARDED_BY(&encoder_queue_);
Ying Wang9b881ab2020-02-07 14:29:32 +0100262 int dropped_frame_cwnd_pushback_count_ RTC_GUARDED_BY(&encoder_queue_);
263 int dropped_frame_encoder_block_count_ RTC_GUARDED_BY(&encoder_queue_);
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200264 absl::optional<VideoFrame> pending_frame_ RTC_GUARDED_BY(&encoder_queue_);
Sebastian Janssona3177052018-04-10 13:05:49 +0200265 int64_t pending_frame_post_time_us_ RTC_GUARDED_BY(&encoder_queue_);
asapersson6ffb67d2016-09-12 00:10:45 -0700266
Ilya Nikolaevskiy71aee3a2019-02-18 13:01:26 +0100267 VideoFrame::UpdateRect accumulated_update_rect_
268 RTC_GUARDED_BY(&encoder_queue_);
Ilya Nikolaevskiy9560d7d2019-10-30 11:19:47 +0100269 bool accumulated_update_rect_is_valid_ RTC_GUARDED_BY(&encoder_queue_);
Ilya Nikolaevskiy71aee3a2019-02-18 13:01:26 +0100270
Ilya Nikolaevskiy648b9d72019-12-03 16:54:17 +0100271 // Used for automatic content type detection.
272 absl::optional<VideoFrame::UpdateRect> last_update_rect_
273 RTC_GUARDED_BY(&encoder_queue_);
274 Timestamp animation_start_time_ RTC_GUARDED_BY(&encoder_queue_);
275 bool cap_resolution_due_to_video_content_ RTC_GUARDED_BY(&encoder_queue_);
276 // Used to correctly ignore changes in update_rect introduced by
277 // resize triggered by animation detection.
278 enum class ExpectResizeState {
279 kNoResize, // Normal operation.
280 kResize, // Resize was triggered by the animation detection.
281 kFirstFrameAfterResize // Resize observed.
282 } expect_resize_state_ RTC_GUARDED_BY(&encoder_queue_);
283
danilchapa37de392017-09-09 04:17:22 -0700284 VideoBitrateAllocationObserver* bitrate_observer_
Niels Möller1e062892018-02-07 10:18:32 +0100285 RTC_GUARDED_BY(&encoder_queue_);
Elad Alon8f01c4e2019-06-28 15:19:43 +0200286 FecControllerOverride* fec_controller_override_
287 RTC_GUARDED_BY(&encoder_queue_);
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200288 absl::optional<int64_t> last_parameters_update_ms_
Niels Möller1e062892018-02-07 10:18:32 +0100289 RTC_GUARDED_BY(&encoder_queue_);
Åsa Perssonc29cb2c2019-03-25 12:06:59 +0100290 absl::optional<int64_t> last_encode_info_ms_ RTC_GUARDED_BY(&encoder_queue_);
sprang1a646ee2016-12-01 06:34:11 -0800291
Erik Språnge2fd86a2018-10-24 11:32:39 +0200292 VideoEncoder::EncoderInfo encoder_info_ RTC_GUARDED_BY(&encoder_queue_);
Sergey Silkin6456e352019-07-08 17:56:40 +0200293 absl::optional<VideoEncoder::ResolutionBitrateLimits> encoder_bitrate_limits_
294 RTC_GUARDED_BY(&encoder_queue_);
Erik Språngd7329ca2019-02-21 21:19:53 +0100295 VideoEncoderFactory::CodecInfo codec_info_ RTC_GUARDED_BY(&encoder_queue_);
Erik Språng6a7baa72019-02-26 18:31:00 +0100296 VideoCodec send_codec_ RTC_GUARDED_BY(&encoder_queue_);
Niels Möller6bb5ab92019-01-11 11:11:10 +0100297
Erik Språng6a7baa72019-02-26 18:31:00 +0100298 FrameDropper frame_dropper_ RTC_GUARDED_BY(&encoder_queue_);
Niels Möller6bb5ab92019-01-11 11:11:10 +0100299 // If frame dropper is not force disabled, frame dropping might still be
300 // disabled if VideoEncoder::GetEncoderInfo() indicates that the encoder has a
301 // trusted rate controller. This is determined on a per-frame basis, as the
302 // encoder behavior might dynamically change.
303 bool force_disable_frame_dropper_ RTC_GUARDED_BY(&encoder_queue_);
304 RateStatistics input_framerate_ RTC_GUARDED_BY(&encoder_queue_);
305 // Incremented on worker thread whenever |frame_dropper_| determines that a
306 // frame should be dropped. Decremented on whichever thread runs
307 // OnEncodedImage(), which is only called by one thread but not necessarily
308 // the worker thread.
309 std::atomic<int> pending_frame_drops_;
Erik Språnge2fd86a2018-10-24 11:32:39 +0200310
Ying Wang9b881ab2020-02-07 14:29:32 +0100311 // Congestion window frame drop ratio (drop 1 in every
312 // cwnd_frame_drop_interval_ frames).
313 absl::optional<int> cwnd_frame_drop_interval_ RTC_GUARDED_BY(&encoder_queue_);
314 // Frame counter for congestion window frame drop.
315 int cwnd_frame_counter_ RTC_GUARDED_BY(&encoder_queue_);
316
Erik Språng7ca375c2019-02-06 16:20:17 +0100317 std::unique_ptr<EncoderBitrateAdjuster> bitrate_adjuster_
318 RTC_GUARDED_BY(&encoder_queue_);
319
Erik Språngd7329ca2019-02-21 21:19:53 +0100320 // TODO(sprang): Change actually support keyframe per simulcast stream, or
321 // turn this into a simple bool |pending_keyframe_request_|.
Niels Möller87e2d782019-03-07 10:18:23 +0100322 std::vector<VideoFrameType> next_frame_types_ RTC_GUARDED_BY(&encoder_queue_);
Erik Språngd7329ca2019-02-21 21:19:53 +0100323
Ilya Nikolaevskiy2ebf5232019-05-13 16:13:36 +0200324 FrameEncodeMetadataWriter frame_encode_metadata_writer_;
Erik Språng6a7baa72019-02-26 18:31:00 +0100325
326 // Experiment groups parsed from field trials for realtime video ([0]) and
327 // screenshare ([1]). 0 means no group specified. Positive values are
328 // experiment group numbers incremented by 1.
329 const std::array<uint8_t, 2> experiment_groups_;
330
philipelda5aa4d2019-04-26 13:37:37 +0200331 // TODO(philipel): Remove this lock and run on |encoder_queue_| instead.
332 rtc::CriticalSection encoded_image_lock_;
333
334 int64_t next_frame_id_ RTC_GUARDED_BY(encoded_image_lock_);
335
336 // This array is used as a map from simulcast id to an encoder's buffer
337 // state. For every buffer of the encoder we keep track of the last frame id
338 // that updated that buffer.
339 std::array<std::array<int64_t, kMaxEncoderBuffers>, kMaxSimulcastStreams>
340 encoder_buffer_state_ RTC_GUARDED_BY(encoded_image_lock_);
341
philipeld9cc8c02019-09-16 14:53:40 +0200342 struct EncoderSwitchExperiment {
343 struct Thresholds {
344 absl::optional<DataRate> bitrate;
345 absl::optional<int> pixel_count;
346 };
347
348 // Codec --> switching thresholds
349 std::map<VideoCodecType, Thresholds> codec_thresholds;
350
351 // To smooth out the target bitrate so that we don't trigger a switch
352 // too easily.
353 rtc::ExpFilter bitrate_filter{1.0};
354
355 // Codec/implementation to switch to
356 std::string to_codec;
357 absl::optional<std::string> to_param;
358 absl::optional<std::string> to_value;
359
360 // Thresholds for the currently used codecs.
361 Thresholds current_thresholds;
362
363 // Updates the |bitrate_filter|, so not const.
364 bool IsBitrateBelowThreshold(const DataRate& target_bitrate);
365 bool IsPixelCountBelowThreshold(int pixel_count) const;
366 void SetCodec(VideoCodecType codec);
367 };
368
369 EncoderSwitchExperiment ParseEncoderSwitchFieldTrial() const;
370
371 EncoderSwitchExperiment encoder_switch_experiment_
372 RTC_GUARDED_BY(&encoder_queue_);
373
Ilya Nikolaevskiy648b9d72019-12-03 16:54:17 +0100374 struct AutomaticAnimationDetectionExperiment {
375 bool enabled = false;
376 int min_duration_ms = 2000;
377 double min_area_ratio = 0.8;
378 int min_fps = 10;
379 std::unique_ptr<StructParametersParser> Parser() {
380 return StructParametersParser::Create(
381 "enabled", &enabled, //
382 "min_duration_ms", &min_duration_ms, //
383 "min_area_ratio", &min_area_ratio, //
384 "min_fps", &min_fps);
385 }
386 };
387
388 AutomaticAnimationDetectionExperiment
389 ParseAutomatincAnimationDetectionFieldTrial() const;
390
391 AutomaticAnimationDetectionExperiment
392 automatic_animation_detection_experiment_ RTC_GUARDED_BY(&encoder_queue_);
393
philipeld9cc8c02019-09-16 14:53:40 +0200394 // An encoder switch is only requested once, this variable is used to keep
395 // track of whether a request has been made or not.
396 bool encoder_switch_requested_ RTC_GUARDED_BY(&encoder_queue_);
397
Henrik Boström07b17df2020-01-15 11:42:12 +0100398 // The controller updates the sink wants based on restrictions that come from
Henrik Boström62057622020-03-10 19:08:05 +0100399 // the resource adaptation processor or adaptation due to bandwidth
400 // adaptation.
Henrik Boström07b17df2020-01-15 11:42:12 +0100401 //
402 // This is used on the encoder queue, with a few exceptions:
403 // - VideoStreamEncoder::SetSource() invokes SetSource().
404 // - VideoStreamEncoder::SetSink() invokes SetRotationApplied() and
405 // PushSourceSinkSettings().
406 // - VideoStreamEncoder::Stop() invokes SetSource().
407 // TODO(hbos): If these can be moved to the encoder queue,
408 // VideoSourceSinkController can be made single-threaded, and its lock can be
409 // replaced with a sequence checker.
Henrik Boströmce0ea492020-01-13 11:27:18 +0100410 std::unique_ptr<VideoSourceSinkController> video_source_sink_controller_;
Henrik Boströmd516b252020-04-17 12:10:59 +0200411 std::unique_ptr<VideoStreamInputStateProvider> input_state_provider_
412 RTC_GUARDED_BY(&encoder_queue_);
Henrik Boström62057622020-03-10 19:08:05 +0100413 std::unique_ptr<ResourceAdaptationProcessor> resource_adaptation_processor_
414 RTC_GUARDED_BY(&encoder_queue_);
Henrik Boströmb08882b2020-01-07 10:11:17 +0100415
philipele5d0fe02019-10-15 11:02:53 +0200416 // All public methods are proxied to |encoder_queue_|. It must must be
417 // destroyed first to make sure no tasks are run that use other members.
418 rtc::TaskQueue encoder_queue_;
419
mflodmancc3d4422017-08-03 08:27:51 -0700420 RTC_DISALLOW_COPY_AND_ASSIGN(VideoStreamEncoder);
niklase@google.com470e71d2011-07-07 08:21:25 +0000421};
mflodman@webrtc.org84d17832011-12-01 17:02:23 +0000422
423} // namespace webrtc
424
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200425#endif // VIDEO_VIDEO_STREAM_ENCODER_H_