1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker * Copyright (c) 2020 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker *
4*d9f75844SAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker */
10*d9f75844SAndroid Build Coastguard Worker
11*d9f75844SAndroid Build Coastguard Worker #include "video/alignment_adjuster.h"
12*d9f75844SAndroid Build Coastguard Worker
13*d9f75844SAndroid Build Coastguard Worker #include <algorithm>
14*d9f75844SAndroid Build Coastguard Worker #include <limits>
15*d9f75844SAndroid Build Coastguard Worker
16*d9f75844SAndroid Build Coastguard Worker #include "absl/algorithm/container.h"
17*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/logging.h"
18*d9f75844SAndroid Build Coastguard Worker
19*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
20*d9f75844SAndroid Build Coastguard Worker namespace {
21*d9f75844SAndroid Build Coastguard Worker // Round each scale factor to the closest rational in form alignment/i where i
22*d9f75844SAndroid Build Coastguard Worker // is a multiple of `requested_alignment`. Each resolution divisible by
23*d9f75844SAndroid Build Coastguard Worker // `alignment` will be divisible by `requested_alignment` after the scale factor
24*d9f75844SAndroid Build Coastguard Worker // is applied.
RoundToMultiple(int alignment,int requested_alignment,VideoEncoderConfig * config,bool update_config)25*d9f75844SAndroid Build Coastguard Worker double RoundToMultiple(int alignment,
26*d9f75844SAndroid Build Coastguard Worker int requested_alignment,
27*d9f75844SAndroid Build Coastguard Worker VideoEncoderConfig* config,
28*d9f75844SAndroid Build Coastguard Worker bool update_config) {
29*d9f75844SAndroid Build Coastguard Worker double diff = 0.0;
30*d9f75844SAndroid Build Coastguard Worker for (auto& layer : config->simulcast_layers) {
31*d9f75844SAndroid Build Coastguard Worker double min_dist = std::numeric_limits<double>::max();
32*d9f75844SAndroid Build Coastguard Worker double new_scale = 1.0;
33*d9f75844SAndroid Build Coastguard Worker for (int i = requested_alignment; i <= alignment;
34*d9f75844SAndroid Build Coastguard Worker i += requested_alignment) {
35*d9f75844SAndroid Build Coastguard Worker double dist = std::abs(layer.scale_resolution_down_by -
36*d9f75844SAndroid Build Coastguard Worker alignment / static_cast<double>(i));
37*d9f75844SAndroid Build Coastguard Worker if (dist <= min_dist) {
38*d9f75844SAndroid Build Coastguard Worker min_dist = dist;
39*d9f75844SAndroid Build Coastguard Worker new_scale = alignment / static_cast<double>(i);
40*d9f75844SAndroid Build Coastguard Worker }
41*d9f75844SAndroid Build Coastguard Worker }
42*d9f75844SAndroid Build Coastguard Worker diff += std::abs(layer.scale_resolution_down_by - new_scale);
43*d9f75844SAndroid Build Coastguard Worker if (update_config) {
44*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_INFO) << "scale_resolution_down_by "
45*d9f75844SAndroid Build Coastguard Worker << layer.scale_resolution_down_by << " -> " << new_scale;
46*d9f75844SAndroid Build Coastguard Worker layer.scale_resolution_down_by = new_scale;
47*d9f75844SAndroid Build Coastguard Worker }
48*d9f75844SAndroid Build Coastguard Worker }
49*d9f75844SAndroid Build Coastguard Worker return diff;
50*d9f75844SAndroid Build Coastguard Worker }
51*d9f75844SAndroid Build Coastguard Worker } // namespace
52*d9f75844SAndroid Build Coastguard Worker
53*d9f75844SAndroid Build Coastguard Worker // Input: encoder_info.requested_resolution_alignment (K)
54*d9f75844SAndroid Build Coastguard Worker // Input: encoder_info.apply_alignment_to_all_simulcast_layers (B)
55*d9f75844SAndroid Build Coastguard Worker // Input: vector config->simulcast_layers.scale_resolution_down_by (S[i])
56*d9f75844SAndroid Build Coastguard Worker // Output:
57*d9f75844SAndroid Build Coastguard Worker // If B is false, returns K and does not adjust scaling factors.
58*d9f75844SAndroid Build Coastguard Worker // Otherwise, returns adjusted alignment (A), adjusted scaling factors (S'[i])
59*d9f75844SAndroid Build Coastguard Worker // are written in `config` such that:
60*d9f75844SAndroid Build Coastguard Worker //
61*d9f75844SAndroid Build Coastguard Worker // A / S'[i] are integers divisible by K
62*d9f75844SAndroid Build Coastguard Worker // sum abs(S'[i] - S[i]) -> min
63*d9f75844SAndroid Build Coastguard Worker // A integer <= 16
64*d9f75844SAndroid Build Coastguard Worker //
65*d9f75844SAndroid Build Coastguard Worker // Solution chooses closest S'[i] in a form A / j where j is a multiple of K.
66*d9f75844SAndroid Build Coastguard Worker
GetAlignmentAndMaybeAdjustScaleFactors(const VideoEncoder::EncoderInfo & encoder_info,VideoEncoderConfig * config,absl::optional<size_t> max_layers)67*d9f75844SAndroid Build Coastguard Worker int AlignmentAdjuster::GetAlignmentAndMaybeAdjustScaleFactors(
68*d9f75844SAndroid Build Coastguard Worker const VideoEncoder::EncoderInfo& encoder_info,
69*d9f75844SAndroid Build Coastguard Worker VideoEncoderConfig* config,
70*d9f75844SAndroid Build Coastguard Worker absl::optional<size_t> max_layers) {
71*d9f75844SAndroid Build Coastguard Worker const int requested_alignment = encoder_info.requested_resolution_alignment;
72*d9f75844SAndroid Build Coastguard Worker if (!encoder_info.apply_alignment_to_all_simulcast_layers) {
73*d9f75844SAndroid Build Coastguard Worker return requested_alignment;
74*d9f75844SAndroid Build Coastguard Worker }
75*d9f75844SAndroid Build Coastguard Worker
76*d9f75844SAndroid Build Coastguard Worker if (requested_alignment < 1 || config->number_of_streams <= 1 ||
77*d9f75844SAndroid Build Coastguard Worker config->simulcast_layers.size() <= 1) {
78*d9f75844SAndroid Build Coastguard Worker return requested_alignment;
79*d9f75844SAndroid Build Coastguard Worker }
80*d9f75844SAndroid Build Coastguard Worker
81*d9f75844SAndroid Build Coastguard Worker // Update alignment to also apply to simulcast layers.
82*d9f75844SAndroid Build Coastguard Worker const bool has_scale_resolution_down_by = absl::c_any_of(
83*d9f75844SAndroid Build Coastguard Worker config->simulcast_layers, [](const webrtc::VideoStream& layer) {
84*d9f75844SAndroid Build Coastguard Worker return layer.scale_resolution_down_by >= 1.0;
85*d9f75844SAndroid Build Coastguard Worker });
86*d9f75844SAndroid Build Coastguard Worker
87*d9f75844SAndroid Build Coastguard Worker if (!has_scale_resolution_down_by) {
88*d9f75844SAndroid Build Coastguard Worker // Default resolution downscaling used (scale factors: 1, 2, 4, ...).
89*d9f75844SAndroid Build Coastguard Worker size_t size = config->simulcast_layers.size();
90*d9f75844SAndroid Build Coastguard Worker if (max_layers && *max_layers > 0 && *max_layers < size) {
91*d9f75844SAndroid Build Coastguard Worker size = *max_layers;
92*d9f75844SAndroid Build Coastguard Worker }
93*d9f75844SAndroid Build Coastguard Worker return requested_alignment * (1 << (size - 1));
94*d9f75844SAndroid Build Coastguard Worker }
95*d9f75844SAndroid Build Coastguard Worker
96*d9f75844SAndroid Build Coastguard Worker // Get alignment for downscaled layers.
97*d9f75844SAndroid Build Coastguard Worker // Adjust `scale_resolution_down_by` to a common multiple to limit the
98*d9f75844SAndroid Build Coastguard Worker // alignment value (to avoid largely cropped frames and possibly with an
99*d9f75844SAndroid Build Coastguard Worker // aspect ratio far from the original).
100*d9f75844SAndroid Build Coastguard Worker const int kMaxAlignment = 16;
101*d9f75844SAndroid Build Coastguard Worker
102*d9f75844SAndroid Build Coastguard Worker for (auto& layer : config->simulcast_layers) {
103*d9f75844SAndroid Build Coastguard Worker layer.scale_resolution_down_by =
104*d9f75844SAndroid Build Coastguard Worker std::max(layer.scale_resolution_down_by, 1.0);
105*d9f75844SAndroid Build Coastguard Worker layer.scale_resolution_down_by =
106*d9f75844SAndroid Build Coastguard Worker std::min(layer.scale_resolution_down_by, 10000.0);
107*d9f75844SAndroid Build Coastguard Worker }
108*d9f75844SAndroid Build Coastguard Worker
109*d9f75844SAndroid Build Coastguard Worker // Decide on common multiple to use.
110*d9f75844SAndroid Build Coastguard Worker double min_diff = std::numeric_limits<double>::max();
111*d9f75844SAndroid Build Coastguard Worker int best_alignment = 1;
112*d9f75844SAndroid Build Coastguard Worker for (int alignment = requested_alignment; alignment <= kMaxAlignment;
113*d9f75844SAndroid Build Coastguard Worker ++alignment) {
114*d9f75844SAndroid Build Coastguard Worker double diff = RoundToMultiple(alignment, requested_alignment, config,
115*d9f75844SAndroid Build Coastguard Worker /*update_config=*/false);
116*d9f75844SAndroid Build Coastguard Worker if (diff < min_diff) {
117*d9f75844SAndroid Build Coastguard Worker min_diff = diff;
118*d9f75844SAndroid Build Coastguard Worker best_alignment = alignment;
119*d9f75844SAndroid Build Coastguard Worker }
120*d9f75844SAndroid Build Coastguard Worker }
121*d9f75844SAndroid Build Coastguard Worker RoundToMultiple(best_alignment, requested_alignment, config,
122*d9f75844SAndroid Build Coastguard Worker /*update_config=*/true);
123*d9f75844SAndroid Build Coastguard Worker
124*d9f75844SAndroid Build Coastguard Worker return std::max(best_alignment, requested_alignment);
125*d9f75844SAndroid Build Coastguard Worker }
126*d9f75844SAndroid Build Coastguard Worker } // namespace webrtc
127