blob: d3b88e5aeb6f2355585ccf20305ff735e8e5af78 [file] [log] [blame]
Henrik Boströmb08882b2020-01-07 09:11:171/*
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ömb08882b2020-01-07 09:11:1723#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öm382cc6d2020-01-07 09:15:0427#include "call/adaptation/resource_adaptation_module_interface.h"
Henrik Boströmb08882b2020-01-07 09:11:1728#include "rtc_base/experiments/balanced_degradation_settings.h"
29#include "video/overuse_frame_detector.h"
30
31namespace webrtc {
32
Henrik Boström382cc6d2020-01-07 09:15:0433class VideoStreamEncoder;
34
Henrik Boströmb08882b2020-01-07 09:11:1735// 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öm07b17df2020-01-15 10:42:1239//
40// This class is single-threaded. The caller is responsible for ensuring safe
41// usage.
Henrik Boströmb08882b2020-01-07 09:11:1742// 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.
50class OveruseFrameDetectorResourceAdaptationModule
Henrik Boström382cc6d2020-01-07 09:15:0451 : public ResourceAdaptationModuleInterface,
52 public AdaptationObserverInterface {
Henrik Boströmb08882b2020-01-07 09:11:1753 public:
Henrik Boström07b17df2020-01-15 10:42:1254 // 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ömb08882b2020-01-07 09:11:1756 OveruseFrameDetectorResourceAdaptationModule(
Henrik Boström382cc6d2020-01-07 09:15:0457 VideoStreamEncoder* video_stream_encoder,
Henrik Boströmb08882b2020-01-07 09:11:1758 std::unique_ptr<OveruseFrameDetector> overuse_detector,
Henrik Boströmd2382002020-01-10 14:44:0159 VideoStreamEncoderObserver* encoder_stats_observer,
60 ResourceAdaptationModuleListener* adaptation_listener);
Henrik Boströmb08882b2020-01-07 09:11:1761 ~OveruseFrameDetectorResourceAdaptationModule() override;
62
Henrik Boströmb08882b2020-01-07 09:11:1763 // 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ömb08882b2020-01-07 09:11:1770 return degradation_preference_;
71 }
72
Henrik Boström382cc6d2020-01-07 09:15:0473 // ResourceAdaptationModuleInterface implementation.
Henrik Boströma3d42522020-01-16 12:55:2974 void StartResourceAdaptation(
Henrik Boströmd2382002020-01-10 14:44:0175 ResourceAdaptationModuleListener* adaptation_listener) override;
Henrik Boströma3d42522020-01-16 12:55:2976 void StopResourceAdaptation() override;
77 void SetHasInputVideo(bool has_input_video) override;
78 void SetDegradationPreference(
79 DegradationPreference degradation_preference) override;
Henrik Boströmfae6f0e2020-01-20 10:16:5080 void ResetVideoSourceRestrictions() override;
Henrik Boström382cc6d2020-01-07 09:15:0481
Henrik Boströmb08882b2020-01-07 09:11:1782 // 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ömb08882b2020-01-07 09:11:1787 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ömfae6f0e2020-01-20 10:16:5099 void SetCodecMaxFrameRate(absl::optional<double> codec_max_frame_rate);
Henrik Boströmb08882b2020-01-07 09:11:17100 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ömb08882b2020-01-07 09:11:17108 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ömce0ea492020-01-13 10:27:18173 class VideoSourceRestrictor;
Henrik Boströmb08882b2020-01-07 09:11:17174
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öm8234b922020-01-13 16:26:50184 // 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öm07b17df2020-01-15 10:42:12187 void MaybeUpdateVideoSourceRestrictions();
Henrik Boströmfae6f0e2020-01-20 10:16:50188 // 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öm8234b922020-01-13 16:26:50192
Henrik Boström07b17df2020-01-15 10:42:12193 void UpdateAdaptationStats(AdaptReason reason);
194 DegradationPreference EffectiveDegradataionPreference();
195 AdaptCounter& GetAdaptCounter();
196 bool CanAdaptUpResolution(int pixels, uint32_t bitrate_bps) const;
Henrik Boströmb08882b2020-01-07 09:11:17197
Henrik Boström8234b922020-01-13 16:26:50198 ResourceAdaptationModuleListener* const adaptation_listener_;
199 // The restrictions that |adaptation_listener_| is informed of.
Henrik Boström07b17df2020-01-15 10:42:12200 VideoSourceRestrictions video_source_restrictions_;
Henrik Boström382cc6d2020-01-07 09:15:04201 // Used to query CpuOveruseOptions at StartCheckForOveruse().
Henrik Boström07b17df2020-01-15 10:42:12202 VideoStreamEncoder* video_stream_encoder_;
Henrik Boströma3d42522020-01-16 12:55:29203 bool has_input_video_;
Henrik Boström07b17df2020-01-15 10:42:12204 DegradationPreference degradation_preference_;
Henrik Boströmb08882b2020-01-07 09:11:17205 // 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öm07b17df2020-01-15 10:42:12210 std::map<const DegradationPreference, AdaptCounter> adapt_counters_;
211 const BalancedDegradationSettings balanced_settings_;
Henrik Boströmb08882b2020-01-07 09:11:17212 // Stores a snapshot of the last adaptation request triggered by an AdaptUp
213 // or AdaptDown signal.
Henrik Boström07b17df2020-01-15 10:42:12214 absl::optional<AdaptationRequest> last_adaptation_request_;
215 absl::optional<int> last_frame_pixel_count_;
Henrik Boströmce0ea492020-01-13 10:27:18216 // Keeps track of source restrictions that this adaptation module outputs.
217 const std::unique_ptr<VideoSourceRestrictor> source_restrictor_;
Henrik Boström07b17df2020-01-15 10:42:12218 const std::unique_ptr<OveruseFrameDetector> overuse_detector_;
Henrik Boströmfae6f0e2020-01-20 10:16:50219 bool overuse_detector_is_started_;
220 absl::optional<double> codec_max_frame_rate_;
221 absl::optional<double> target_frame_rate_;
Henrik Boström07b17df2020-01-15 10:42:12222 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ömb08882b2020-01-07 09:11:17227};
228
229} // namespace webrtc
230
231#endif // VIDEO_OVERUSE_FRAME_DETECTOR_RESOURCE_ADAPTATION_MODULE_H_