blob: 797a876f2b4049c38049bf1477b5ecb80819e022 [file] [log] [blame]
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:131/*
kjellander1afca732016-02-08 04:46:452 * Copyright (c) 2010 The WebRTC project authors. All Rights Reserved.
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:133 *
kjellander1afca732016-02-08 04:46:454 * 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.
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:139 */
henrike@webrtc.org28e20752013-07-10 00:45:3610
kjellandera96e2d72016-02-05 07:52:2811#include "webrtc/media/base/videoadapter.h"
henrike@webrtc.org28e20752013-07-10 00:45:3612
andresp@webrtc.orgff689be2015-02-12 11:54:2613#include <algorithm>
Per766ad3b92016-04-05 13:23:4914#include <limits>
henrike@webrtc.org28e20752013-07-10 00:45:3615
buildbot@webrtc.orga09a9992014-08-13 17:26:0816#include "webrtc/base/logging.h"
kjellanderf4752772016-03-02 13:42:3017#include "webrtc/media/base/mediaconstants.h"
kjellandera96e2d72016-02-05 07:52:2818#include "webrtc/media/base/videocommon.h"
henrike@webrtc.org28e20752013-07-10 00:45:3619
Per766ad3b92016-04-05 13:23:4920namespace {
henrike@webrtc.org28e20752013-07-10 00:45:3621
Per766ad3b92016-04-05 13:23:4922// Scale factors optimized for in libYUV that we accept.
23// Must be sorted in decreasing scale factors for FindScaleLargerThan to work.
24const float kScaleFactors[] = {
25 1.f / 1.f, // Full size.
26 3.f / 4.f, // 3/4 scale.
27 1.f / 2.f, // 1/2 scale.
28 3.f / 8.f, // 3/8 scale.
29 1.f / 4.f, // 1/4 scale.
30 3.f / 16.f, // 3/16 scale.
henrike@webrtc.org28e20752013-07-10 00:45:3631};
henrike@webrtc.org28e20752013-07-10 00:45:3632
Per766ad3b92016-04-05 13:23:4933float FindScaleLessThanOrEqual(int width,
34 int height,
35 int target_num_pixels,
36 int* resulting_number_of_pixels) {
37 float best_distance = std::numeric_limits<float>::max();
38 float best_scale = 0.0f; // Default to 0 if nothing matches.
39 float pixels = width * height;
40 float best_number_of_pixels = 0.0f;
41 for (const auto& scale : kScaleFactors) {
wu@webrtc.orgcadf9042013-08-30 21:24:1642 float test_num_pixels = pixels * scale * scale;
wu@webrtc.orgcadf9042013-08-30 21:24:1643 float diff = target_num_pixels - test_num_pixels;
henrike@webrtc.org28e20752013-07-10 00:45:3644 if (diff < 0) {
Per766ad3b92016-04-05 13:23:4945 continue;
henrike@webrtc.org28e20752013-07-10 00:45:3646 }
47 if (diff < best_distance) {
48 best_distance = diff;
wu@webrtc.orgcadf9042013-08-30 21:24:1649 best_scale = scale;
Per766ad3b92016-04-05 13:23:4950 best_number_of_pixels = test_num_pixels;
henrike@webrtc.org28e20752013-07-10 00:45:3651 if (best_distance == 0) { // Found exact match.
52 break;
53 }
54 }
55 }
Per766ad3b92016-04-05 13:23:4956 if (resulting_number_of_pixels) {
57 *resulting_number_of_pixels = static_cast<int>(best_number_of_pixels + .5f);
58 }
wu@webrtc.orgcadf9042013-08-30 21:24:1659 return best_scale;
60}
61
Per766ad3b92016-04-05 13:23:4962float FindScaleLargerThan(int width,
63 int height,
64 int target_num_pixels,
65 int* resulting_number_of_pixels) {
66 float best_distance = std::numeric_limits<float>::max();
67 float best_scale = 1.f; // Default to unscaled if nothing matches.
68 float pixels = width * height;
69 float best_number_of_pixels = pixels; // Default to input number of pixels.
70 for (const auto& scale : kScaleFactors) {
71 float test_num_pixels = pixels * scale * scale;
72 float diff = test_num_pixels - target_num_pixels;
73 if (diff <= 0) {
74 break;
75 }
76 if (diff < best_distance) {
77 best_distance = diff;
78 best_scale = scale;
79 best_number_of_pixels = test_num_pixels;
80 }
81 }
82
83 *resulting_number_of_pixels = static_cast<int>(best_number_of_pixels + .5f);
84 return best_scale;
wu@webrtc.orgcadf9042013-08-30 21:24:1685}
86
Per766ad3b92016-04-05 13:23:4987} // namespace
henrike@webrtc.org28e20752013-07-10 00:45:3688
Per766ad3b92016-04-05 13:23:4989namespace cricket {
henrike@webrtc.org28e20752013-07-10 00:45:3690
henrike@webrtc.org28e20752013-07-10 00:45:3691VideoAdapter::VideoAdapter()
Per766ad3b92016-04-05 13:23:4992 : output_num_pixels_(std::numeric_limits<int>::max()),
sergeyu@chromium.org9cf037b2014-02-07 19:03:2693 frames_in_(0),
94 frames_out_(0),
95 frames_scaled_(0),
wu@webrtc.orgcadf9042013-08-30 21:24:1696 adaption_changes_(0),
magjed@webrtc.orga73d7462014-11-14 13:25:2597 previous_width_(0),
98 previous_height_(0),
Per766ad3b92016-04-05 13:23:4999 interval_next_frame_(0),
100 format_request_max_pixel_count_(std::numeric_limits<int>::max()),
101 resolution_request_max_pixel_count_(std::numeric_limits<int>::max()) {}
henrike@webrtc.org28e20752013-07-10 00:45:36102
Per766ad3b92016-04-05 13:23:49103VideoAdapter::~VideoAdapter() {}
104
105void VideoAdapter::SetExpectedInputFrameInterval(int64_t interval) {
106 // TODO(perkj): Consider measuring input frame rate instead.
107 // Frame rate typically varies depending on lighting.
108 rtc::CritScope cs(&critical_section_);
109 input_format_.interval = interval;
henrike@webrtc.org28e20752013-07-10 00:45:36110}
111
henrike@webrtc.org28e20752013-07-10 00:45:36112void VideoAdapter::SetInputFormat(const VideoFormat& format) {
Per766ad3b92016-04-05 13:23:49113 bool is_resolution_change = (input_format().width != format.width ||
114 input_format().height != format.height);
Peter Boström0c4e06b2015-10-07 10:23:21115 int64_t old_input_interval = input_format_.interval;
henrike@webrtc.org28e20752013-07-10 00:45:36116 input_format_ = format;
andresp@webrtc.orgff689be2015-02-12 11:54:26117 output_format_.interval =
118 std::max(output_format_.interval, input_format_.interval);
sergeyu@chromium.org9cf037b2014-02-07 19:03:26119 if (old_input_interval != input_format_.interval) {
wu@webrtc.orgb9a088b2014-02-13 23:18:49120 LOG(LS_INFO) << "VAdapt input interval changed from "
121 << old_input_interval << " to " << input_format_.interval;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26122 }
mallinath@webrtc.org1b15f422013-09-06 22:56:28123 if (is_resolution_change) {
mallinath@webrtc.org1b15f422013-09-06 22:56:28124 // Trigger the adaptation logic again, to potentially reset the adaptation
125 // state for things like view requests that may not longer be capping
126 // output (or may now cap output).
Per766ad3b92016-04-05 13:23:49127 Adapt(std::min(format_request_max_pixel_count_,
128 resolution_request_max_pixel_count_),
129 0);
mallinath@webrtc.org1b15f422013-09-06 22:56:28130 }
131}
132
Per766ad3b92016-04-05 13:23:49133const VideoFormat& VideoAdapter::input_format() const {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52134 rtc::CritScope cs(&critical_section_);
henrike@webrtc.org28e20752013-07-10 00:45:36135 return input_format_;
136}
137
magjed@webrtc.orgf58b4552014-11-19 18:09:14138VideoFormat VideoAdapter::AdaptFrameResolution(int in_width, int in_height) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52139 rtc::CritScope cs(&critical_section_);
sergeyu@chromium.org9cf037b2014-02-07 19:03:26140 ++frames_in_;
henrike@webrtc.org28e20752013-07-10 00:45:36141
magjed@webrtc.orgf58b4552014-11-19 18:09:14142 SetInputFormat(VideoFormat(
143 in_width, in_height, input_format_.interval, input_format_.fourcc));
henrike@webrtc.org28e20752013-07-10 00:45:36144
145 // Drop the input frame if necessary.
146 bool should_drop = false;
147 if (!output_num_pixels_) {
148 // Drop all frames as the output format is 0x0.
149 should_drop = true;
150 } else {
151 // Drop some frames based on input fps and output fps.
152 // Normally output fps is less than input fps.
henrike@webrtc.org28e20752013-07-10 00:45:36153 interval_next_frame_ += input_format_.interval;
154 if (output_format_.interval > 0) {
155 if (interval_next_frame_ >= output_format_.interval) {
156 interval_next_frame_ %= output_format_.interval;
157 } else {
158 should_drop = true;
159 }
160 }
161 }
162 if (should_drop) {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26163 // Show VAdapt log every 90 frames dropped. (3 seconds)
wu@webrtc.orgb9a088b2014-02-13 23:18:49164 if ((frames_in_ - frames_out_) % 90 == 0) {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26165 // TODO(fbarchard): Reduce to LS_VERBOSE when adapter info is not needed
166 // in default calls.
wu@webrtc.orgb9a088b2014-02-13 23:18:49167 LOG(LS_INFO) << "VAdapt Drop Frame: scaled " << frames_scaled_
168 << " / out " << frames_out_
169 << " / in " << frames_in_
sergeyu@chromium.org9cf037b2014-02-07 19:03:26170 << " Changes: " << adaption_changes_
magjed@webrtc.orgf58b4552014-11-19 18:09:14171 << " Input: " << in_width
172 << "x" << in_height
sergeyu@chromium.org9cf037b2014-02-07 19:03:26173 << " i" << input_format_.interval
174 << " Output: i" << output_format_.interval;
175 }
magjed@webrtc.orgf58b4552014-11-19 18:09:14176
177 return VideoFormat(); // Drop frame.
henrike@webrtc.org28e20752013-07-10 00:45:36178 }
179
Per766ad3b92016-04-05 13:23:49180 const float scale = FindScaleLessThanOrEqual(in_width, in_height,
181 output_num_pixels_, nullptr);
182 const int output_width = static_cast<int>(in_width * scale + .5f);
183 const int output_height = static_cast<int>(in_height * scale + .5f);
wu@webrtc.orgcadf9042013-08-30 21:24:16184
sergeyu@chromium.org9cf037b2014-02-07 19:03:26185 ++frames_out_;
magjed@webrtc.orgf58b4552014-11-19 18:09:14186 if (scale != 1)
sergeyu@chromium.org9cf037b2014-02-07 19:03:26187 ++frames_scaled_;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26188
magjed@webrtc.orgf58b4552014-11-19 18:09:14189 if (previous_width_ && (previous_width_ != output_width ||
190 previous_height_ != output_height)) {
wu@webrtc.orgcadf9042013-08-30 21:24:16191 ++adaption_changes_;
Per766ad3b92016-04-05 13:23:49192 LOG(LS_INFO) << "Frame size changed: scaled " << frames_scaled_ << " / out "
193 << frames_out_ << " / in " << frames_in_
194 << " Changes: " << adaption_changes_ << " Input: " << in_width
195 << "x" << in_height << " i" << input_format_.interval
196 << " Scale: " << scale << " Output: " << output_width << "x"
197 << output_height << " i" << output_format_.interval;
wu@webrtc.orgcadf9042013-08-30 21:24:16198 }
magjed@webrtc.orgf58b4552014-11-19 18:09:14199
200 output_format_.width = output_width;
201 output_format_.height = output_height;
202 previous_width_ = output_width;
203 previous_height_ = output_height;
204
205 return output_format_;
206}
207
Per766ad3b92016-04-05 13:23:49208void VideoAdapter::OnOutputFormatRequest(const VideoFormat& format) {
209 rtc::CritScope cs(&critical_section_);
210 format_request_max_pixel_count_ = format.width * format.height;
211 output_format_.interval = format.interval;
212 Adapt(std::min(format_request_max_pixel_count_,
213 resolution_request_max_pixel_count_),
214 0);
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24215}
216
Per766ad3b92016-04-05 13:23:49217void VideoAdapter::OnResolutionRequest(
perkj2d5f0912016-02-29 08:04:41218 rtc::Optional<int> max_pixel_count,
219 rtc::Optional<int> max_pixel_count_step_up) {
Per766ad3b92016-04-05 13:23:49220 rtc::CritScope cs(&critical_section_);
221 resolution_request_max_pixel_count_ =
222 max_pixel_count.value_or(std::numeric_limits<int>::max());
223 Adapt(std::min(format_request_max_pixel_count_,
224 resolution_request_max_pixel_count_),
225 max_pixel_count_step_up.value_or(0));
perkj2d5f0912016-02-29 08:04:41226}
227
Per766ad3b92016-04-05 13:23:49228bool VideoAdapter::Adapt(int max_num_pixels, int max_pixel_count_step_up) {
229 float scale_lower =
230 FindScaleLessThanOrEqual(input_format_.width, input_format_.height,
231 max_num_pixels, &max_num_pixels);
232 float scale_upper =
233 max_pixel_count_step_up > 0
234 ? FindScaleLargerThan(input_format_.width, input_format_.height,
235 max_pixel_count_step_up,
236 &max_pixel_count_step_up)
237 : 1.f;
perkj2d5f0912016-02-29 08:04:41238
Per766ad3b92016-04-05 13:23:49239 bool use_max_pixel_count_step_up =
240 max_pixel_count_step_up > 0 && max_num_pixels > max_pixel_count_step_up;
henrike@webrtc.org28e20752013-07-10 00:45:36241
Per766ad3b92016-04-05 13:23:49242 int old_num_pixels = output_num_pixels_;
243 output_num_pixels_ =
244 use_max_pixel_count_step_up ? max_pixel_count_step_up : max_num_pixels;
245 // Log the new size.
246 float scale = use_max_pixel_count_step_up ? scale_upper : scale_lower;
247 int new_width = static_cast<int>(input_format_.width * scale + .5f);
248 int new_height = static_cast<int>(input_format_.height * scale + .5f);
wu@webrtc.orgcadf9042013-08-30 21:24:16249
Per766ad3b92016-04-05 13:23:49250 bool changed = output_num_pixels_ != old_num_pixels;
251 LOG(LS_INFO) << "OnResolutionRequest: "
252 << " Max pixels: " << max_num_pixels
253 << " Max pixels step up: " << max_pixel_count_step_up
254 << " Output Pixels: " << output_num_pixels_
255 << " Input: " << input_format_.width << "x"
256 << input_format_.height << " Scale: " << scale
257 << " Resolution: " << new_width << "x" << new_height
258 << " Changed: " << (changed ? "true" : "false");
henrike@webrtc.org28654cb2013-07-22 21:07:49259
henrike@webrtc.org28e20752013-07-10 00:45:36260 return changed;
261}
262
263} // namespace cricket