Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2019 The WebRTC Project Authors. All rights reserved. |
| 3 | * |
| 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 | |
| 11 | #ifndef VIDEO_OVERUSE_FRAME_DETECTOR_RESOURCE_ADAPTATION_MODULE_H_ |
| 12 | #define VIDEO_OVERUSE_FRAME_DETECTOR_RESOURCE_ADAPTATION_MODULE_H_ |
| 13 | |
| 14 | #include <map> |
| 15 | #include <memory> |
| 16 | #include <string> |
| 17 | #include <utility> |
| 18 | #include <vector> |
| 19 | |
| 20 | #include "absl/types/optional.h" |
| 21 | #include "api/rtp_parameters.h" |
| 22 | #include "api/video/video_frame.h" |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 23 | #include "api/video/video_source_interface.h" |
| 24 | #include "api/video/video_stream_encoder_observer.h" |
| 25 | #include "api/video_codecs/video_encoder.h" |
| 26 | #include "api/video_codecs/video_encoder_config.h" |
Henrik Boström | 382cc6d | 2020-01-07 09:15:04 | [diff] [blame] | 27 | #include "call/adaptation/resource_adaptation_module_interface.h" |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 28 | #include "rtc_base/experiments/balanced_degradation_settings.h" |
| 29 | #include "video/overuse_frame_detector.h" |
| 30 | |
| 31 | namespace webrtc { |
| 32 | |
Henrik Boström | 382cc6d | 2020-01-07 09:15:04 | [diff] [blame] | 33 | class VideoStreamEncoder; |
| 34 | |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 35 | // This class is used by the VideoStreamEncoder and is responsible for adapting |
| 36 | // resolution up or down based on encode usage percent. It keeps track of video |
| 37 | // source settings, adaptation counters and may get influenced by |
| 38 | // VideoStreamEncoder's quality scaler through AdaptUp() and AdaptDown() calls. |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 39 | // |
| 40 | // This class is single-threaded. The caller is responsible for ensuring safe |
| 41 | // usage. |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 42 | // TODO(hbos): Reduce the coupling with VideoStreamEncoder. |
| 43 | // TODO(hbos): Add unittests specific to this class, it is currently only tested |
| 44 | // indirectly in video_stream_encoder_unittest.cc and other tests exercising |
| 45 | // VideoStreamEncoder. |
| 46 | // TODO(hbos): Create and implement an abstract interface |
| 47 | // ResourceAdaptationModuleInterface and make this class inherit it. Use the |
| 48 | // generic interface in VideoStreamEncoder, unblocking other modules from being |
| 49 | // implemented and used. |
| 50 | class OveruseFrameDetectorResourceAdaptationModule |
Henrik Boström | 382cc6d | 2020-01-07 09:15:04 | [diff] [blame] | 51 | : public ResourceAdaptationModuleInterface, |
| 52 | public AdaptationObserverInterface { |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 53 | public: |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 54 | // The module can be constructed on any sequence, but must be initialized and |
| 55 | // used on a single sequence, e.g. the encoder queue. |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 56 | OveruseFrameDetectorResourceAdaptationModule( |
Henrik Boström | 382cc6d | 2020-01-07 09:15:04 | [diff] [blame] | 57 | VideoStreamEncoder* video_stream_encoder, |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 58 | std::unique_ptr<OveruseFrameDetector> overuse_detector, |
Henrik Boström | d238200 | 2020-01-10 14:44:01 | [diff] [blame] | 59 | VideoStreamEncoderObserver* encoder_stats_observer, |
| 60 | ResourceAdaptationModuleListener* adaptation_listener); |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 61 | ~OveruseFrameDetectorResourceAdaptationModule() override; |
| 62 | |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 63 | // Sets the encoder to reconfigure based on overuse. |
| 64 | // TODO(hbos): Don't reconfigure the encoder directly. Instead, define the |
| 65 | // output of a resource adaptation module as a struct and let the |
| 66 | // VideoStreamEncoder handle the interaction with the actual encoder. |
| 67 | void SetEncoder(VideoEncoder* encoder); |
| 68 | |
| 69 | DegradationPreference degradation_preference() const { |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 70 | return degradation_preference_; |
| 71 | } |
| 72 | |
Henrik Boström | 382cc6d | 2020-01-07 09:15:04 | [diff] [blame] | 73 | // ResourceAdaptationModuleInterface implementation. |
Henrik Boström | a3d4252 | 2020-01-16 12:55:29 | [diff] [blame] | 74 | void StartResourceAdaptation( |
Henrik Boström | d238200 | 2020-01-10 14:44:01 | [diff] [blame] | 75 | ResourceAdaptationModuleListener* adaptation_listener) override; |
Henrik Boström | a3d4252 | 2020-01-16 12:55:29 | [diff] [blame] | 76 | void StopResourceAdaptation() override; |
| 77 | void SetHasInputVideo(bool has_input_video) override; |
| 78 | void SetDegradationPreference( |
| 79 | DegradationPreference degradation_preference) override; |
Henrik Boström | fae6f0e | 2020-01-20 10:16:50 | [diff] [blame] | 80 | void ResetVideoSourceRestrictions() override; |
Henrik Boström | 382cc6d | 2020-01-07 09:15:04 | [diff] [blame] | 81 | |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 82 | // Input to the OveruseFrameDetector, which are required for this module to |
| 83 | // function. These map to OveruseFrameDetector methods. |
| 84 | // TODO(hbos): Define virtual methods in ResourceAdaptationModuleInterface |
| 85 | // for input that are more generic so that this class can be used without |
| 86 | // assumptions about underlying implementation. |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 87 | void FrameCaptured(const VideoFrame& frame, int64_t time_when_first_seen_us); |
| 88 | void FrameSent(uint32_t timestamp, |
| 89 | int64_t time_sent_in_us, |
| 90 | int64_t capture_time_us, |
| 91 | absl::optional<int> encode_duration_us); |
| 92 | |
| 93 | // Various other settings and feedback mechanisms. |
| 94 | // TODO(hbos): Find a common interface that would make sense for a generic |
| 95 | // resource adaptation module. Unify code paths where possible. Do we really |
| 96 | // need this many public methods? |
| 97 | void SetLastFramePixelCount(absl::optional<int> last_frame_pixel_count); |
| 98 | void SetEncoderConfig(VideoEncoderConfig encoder_config); |
Henrik Boström | fae6f0e | 2020-01-20 10:16:50 | [diff] [blame] | 99 | void SetCodecMaxFrameRate(absl::optional<double> codec_max_frame_rate); |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 100 | void SetEncoderStartBitrateBps(uint32_t encoder_start_bitrate_bps); |
| 101 | // Inform the detector whether or not the quality scaler is enabled. This |
| 102 | // helps GetActiveCounts() return absl::nullopt when appropriate. |
| 103 | // TODO(hbos): This feels really hacky, can we report the right values without |
| 104 | // this boolean? It would be really easy to report the wrong thing if this |
| 105 | // method is called incorrectly. |
| 106 | void SetIsQualityScalerEnabled(bool is_quality_scaler_enabled); |
| 107 | |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 108 | class AdaptCounter final { |
| 109 | public: |
| 110 | AdaptCounter(); |
| 111 | ~AdaptCounter(); |
| 112 | |
| 113 | // Get number of adaptation downscales for |reason|. |
| 114 | VideoStreamEncoderObserver::AdaptationSteps Counts(int reason) const; |
| 115 | |
| 116 | std::string ToString() const; |
| 117 | |
| 118 | void IncrementFramerate(int reason); |
| 119 | void IncrementResolution(int reason); |
| 120 | void DecrementFramerate(int reason); |
| 121 | void DecrementResolution(int reason); |
| 122 | void DecrementFramerate(int reason, int cur_fps); |
| 123 | |
| 124 | // Gets the total number of downgrades (for all adapt reasons). |
| 125 | int FramerateCount() const; |
| 126 | int ResolutionCount() const; |
| 127 | |
| 128 | // Gets the total number of downgrades for |reason|. |
| 129 | int FramerateCount(int reason) const; |
| 130 | int ResolutionCount(int reason) const; |
| 131 | int TotalCount(int reason) const; |
| 132 | |
| 133 | private: |
| 134 | std::string ToString(const std::vector<int>& counters) const; |
| 135 | int Count(const std::vector<int>& counters) const; |
| 136 | void MoveCount(std::vector<int>* counters, int from_reason); |
| 137 | |
| 138 | // Degradation counters holding number of framerate/resolution reductions |
| 139 | // per adapt reason. |
| 140 | std::vector<int> fps_counters_; |
| 141 | std::vector<int> resolution_counters_; |
| 142 | }; |
| 143 | |
| 144 | // AdaptationObserverInterface implementation. Used both "internally" as |
| 145 | // feedback from |overuse_detector_|, and externally from VideoStreamEncoder: |
| 146 | // - It is wired to the VideoStreamEncoder::quality_scaler_. |
| 147 | // - It is invoked by VideoStreamEncoder::MaybeEncodeVideoFrame(). |
| 148 | // TODO(hbos): Decouple quality scaling and resource adaptation, or find an |
| 149 | // interface for reconfiguring externally. |
| 150 | // TODO(hbos): VideoStreamEncoder should not be responsible for any part of |
| 151 | // the adaptation. |
| 152 | void AdaptUp(AdaptReason reason) override; |
| 153 | bool AdaptDown(AdaptReason reason) override; |
| 154 | |
| 155 | // Used by VideoStreamEncoder when ConfigureQualityScaler() occurs and the |
| 156 | // |encoder_stats_observer_| is called outside of this class. |
| 157 | // TODO(hbos): Decouple quality scaling and resource adaptation logic and make |
| 158 | // this method private. |
| 159 | VideoStreamEncoderObserver::AdaptationSteps GetActiveCounts( |
| 160 | AdaptReason reason); |
| 161 | |
| 162 | // Used by VideoStreamEncoder::MaybeEncodeVideoFrame(). |
| 163 | // TODO(hbos): VideoStreamEncoder should not be responsible for any part of |
| 164 | // the adaptation. Move this logic to this module? |
| 165 | const AdaptCounter& GetConstAdaptCounter(); |
| 166 | |
| 167 | // Used by VideoStreamEncoder::ConfigureQualityScaler(). |
| 168 | // TODO(hbos): Decouple quality scaling and resource adaptation logic and |
| 169 | // delete this method. |
| 170 | absl::optional<VideoEncoder::QpThresholds> GetQpThresholds() const; |
| 171 | |
| 172 | private: |
Henrik Boström | ce0ea49 | 2020-01-13 10:27:18 | [diff] [blame] | 173 | class VideoSourceRestrictor; |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 174 | |
| 175 | struct AdaptationRequest { |
| 176 | // The pixel count produced by the source at the time of the adaptation. |
| 177 | int input_pixel_count_; |
| 178 | // Framerate received from the source at the time of the adaptation. |
| 179 | int framerate_fps_; |
| 180 | // Indicates if request was to adapt up or down. |
| 181 | enum class Mode { kAdaptUp, kAdaptDown } mode_; |
| 182 | }; |
| 183 | |
Henrik Boström | 8234b92 | 2020-01-13 16:26:50 | [diff] [blame] | 184 | // Makes |video_source_restrictions_| up-to-date and informs the |
| 185 | // |adaptation_listener_| if restrictions are changed, allowing the listener |
| 186 | // to reconfigure the source accordingly. |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 187 | void MaybeUpdateVideoSourceRestrictions(); |
Henrik Boström | fae6f0e | 2020-01-20 10:16:50 | [diff] [blame] | 188 | // Calculates an up-to-date value of |target_frame_rate_| and informs the |
| 189 | // |overuse_detector_| of the new value if it changed and the detector is |
| 190 | // started. |
| 191 | void MaybeUpdateTargetFrameRate(); |
Henrik Boström | 8234b92 | 2020-01-13 16:26:50 | [diff] [blame] | 192 | |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 193 | void UpdateAdaptationStats(AdaptReason reason); |
| 194 | DegradationPreference EffectiveDegradataionPreference(); |
| 195 | AdaptCounter& GetAdaptCounter(); |
| 196 | bool CanAdaptUpResolution(int pixels, uint32_t bitrate_bps) const; |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 197 | |
Henrik Boström | 8234b92 | 2020-01-13 16:26:50 | [diff] [blame] | 198 | ResourceAdaptationModuleListener* const adaptation_listener_; |
| 199 | // The restrictions that |adaptation_listener_| is informed of. |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 200 | VideoSourceRestrictions video_source_restrictions_; |
Henrik Boström | 382cc6d | 2020-01-07 09:15:04 | [diff] [blame] | 201 | // Used to query CpuOveruseOptions at StartCheckForOveruse(). |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 202 | VideoStreamEncoder* video_stream_encoder_; |
Henrik Boström | a3d4252 | 2020-01-16 12:55:29 | [diff] [blame] | 203 | bool has_input_video_; |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 204 | DegradationPreference degradation_preference_; |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 205 | // Counters used for deciding if the video resolution or framerate is |
| 206 | // currently restricted, and if so, why, on a per degradation preference |
| 207 | // basis. |
| 208 | // TODO(sprang): Replace this with a state holding a relative overuse measure |
| 209 | // instead, that can be translated into suitable down-scale or fps limit. |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 210 | std::map<const DegradationPreference, AdaptCounter> adapt_counters_; |
| 211 | const BalancedDegradationSettings balanced_settings_; |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 212 | // Stores a snapshot of the last adaptation request triggered by an AdaptUp |
| 213 | // or AdaptDown signal. |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 214 | absl::optional<AdaptationRequest> last_adaptation_request_; |
| 215 | absl::optional<int> last_frame_pixel_count_; |
Henrik Boström | ce0ea49 | 2020-01-13 10:27:18 | [diff] [blame] | 216 | // Keeps track of source restrictions that this adaptation module outputs. |
| 217 | const std::unique_ptr<VideoSourceRestrictor> source_restrictor_; |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 218 | const std::unique_ptr<OveruseFrameDetector> overuse_detector_; |
Henrik Boström | fae6f0e | 2020-01-20 10:16:50 | [diff] [blame] | 219 | bool overuse_detector_is_started_; |
| 220 | absl::optional<double> codec_max_frame_rate_; |
| 221 | absl::optional<double> target_frame_rate_; |
Henrik Boström | 07b17df | 2020-01-15 10:42:12 | [diff] [blame] | 222 | uint32_t encoder_start_bitrate_bps_; |
| 223 | bool is_quality_scaler_enabled_; |
| 224 | VideoEncoderConfig encoder_config_; |
| 225 | VideoEncoder* encoder_; |
| 226 | VideoStreamEncoderObserver* const encoder_stats_observer_; |
Henrik Boström | b08882b | 2020-01-07 09:11:17 | [diff] [blame] | 227 | }; |
| 228 | |
| 229 | } // namespace webrtc |
| 230 | |
| 231 | #endif // VIDEO_OVERUSE_FRAME_DETECTOR_RESOURCE_ADAPTATION_MODULE_H_ |