blob: 496ae311fa9ab04f49a0e674002a39201f2ad246 [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/capture/content/capture_resolution_chooser.h"
#include <algorithm>
#include <limits>
#include "base/logging.h"
#include "base/strings/string_util.h"
#include "media/base/video_util.h"
namespace media {
namespace {
// Each snapped frame size is an integer multiple of this many lines apart.
// This is ideal for 16:9 content, but seems to also work well for many
// arbitrary aspect ratios.
const int kSnappedHeightStep = 90;
// The minimum amount of decrease in area between consecutive snapped frame
// sizes. This matters externally, where the end-to-end system is hunting for a
// capture size that works within all resource bottlenecks. If the snapped
// frame sizes are too-close together, the end-to-end system cannot stablize.
// If they are too-far apart, quality is being sacrificed.
const int kMinAreaDecreasePercent = 15;
// Returns |size|, unless it exceeds |max_size| or is under |min_size|. When
// the bounds are exceeded, computes and returns an alternate size of similar
// aspect ratio that is within the bounds.
gfx::Size ComputeBoundedCaptureSize(const gfx::Size& size,
const gfx::Size& min_size,
const gfx::Size& max_size) {
if (size.width() > max_size.width() || size.height() > max_size.height()) {
gfx::Size result = ScaleSizeToFitWithinTarget(size, max_size);
result.SetToMax(min_size);
return result;
} else if (size.width() < min_size.width() ||
size.height() < min_size.height()) {
gfx::Size result = ScaleSizeToEncompassTarget(size, min_size);
result.SetToMin(max_size);
return result;
} else {
return size;
}
}
// Returns true if the area of |a| is less than that of |b|.
bool CompareByArea(const gfx::Size& a, const gfx::Size& b) {
return a.GetArea() < b.GetArea();
}
} // namespace
// static
constexpr gfx::Size CaptureResolutionChooser::kDefaultCaptureSize;
CaptureResolutionChooser::CaptureResolutionChooser()
: min_frame_size_(kDefaultCaptureSize),
max_frame_size_(kDefaultCaptureSize),
apply_aspect_ratio_adjustment_(false),
target_area_(std::numeric_limits<decltype(target_area_)>::max()),
capture_size_(kDefaultCaptureSize),
snapped_sizes_({kDefaultCaptureSize}) {}
CaptureResolutionChooser::~CaptureResolutionChooser() = default;
void CaptureResolutionChooser::SetConstraints(const gfx::Size& min_frame_size,
const gfx::Size& max_frame_size,
bool use_fixed_aspect_ratio) {
DCHECK_LT(0, min_frame_size.width());
DCHECK_LT(0, min_frame_size.height());
DCHECK_LE(min_frame_size.width(), max_frame_size.width());
DCHECK_LE(min_frame_size.height(), max_frame_size.height());
min_frame_size_ = min_frame_size;
max_frame_size_ = max_frame_size;
apply_aspect_ratio_adjustment_ =
(min_frame_size != max_frame_size && use_fixed_aspect_ratio);
UpdateSnappedFrameSizes();
RecomputeCaptureSize();
}
void CaptureResolutionChooser::SetSourceSize(const gfx::Size& source_size) {
source_size_ = source_size;
UpdateSnappedFrameSizes();
RecomputeCaptureSize();
}
void CaptureResolutionChooser::SetTargetFrameArea(int area) {
DCHECK_GE(area, 0);
target_area_ = area;
RecomputeCaptureSize();
}
gfx::Size CaptureResolutionChooser::FindNearestFrameSize(int area) const {
const auto begin = snapped_sizes_.begin();
const auto end = snapped_sizes_.end();
DCHECK(begin != end);
const gfx::Size area_as_size(area, 1); // A facade for CompareByArea().
const auto p = std::lower_bound(begin, end, area_as_size, &CompareByArea);
if (p == end) {
// Boundary case: The target |area| is greater than or equal to the
// largest, so the largest size is closest.
return *(end - 1);
} else if (p == begin) {
// Boundary case: The target |area| is smaller than the smallest, so the
// smallest size is closest.
return *begin;
} else {
// |p| points to the smallest size whose area is greater than or equal to
// the target |area|. The next smaller size could be closer to the target
// |area|, so it must also be considered.
const auto q = p - 1;
return ((p->GetArea() - area) < (area - q->GetArea())) ? *p : *q;
}
}
gfx::Size CaptureResolutionChooser::FindLargerFrameSize(
int area,
int num_steps_up) const {
DCHECK_GT(num_steps_up, 0);
const auto begin = snapped_sizes_.begin();
const auto end = snapped_sizes_.end();
DCHECK(begin != end);
const gfx::Size area_as_size(area, 1); // A facade for CompareByArea().
auto p = std::upper_bound(begin, end, area_as_size, &CompareByArea);
// |p| is already pointing one step up.
const int additional_steps_up = num_steps_up - 1;
if ((end - p) > additional_steps_up)
return *(p + additional_steps_up);
else
return *(end - 1);
}
gfx::Size CaptureResolutionChooser::FindSmallerFrameSize(
int area,
int num_steps_down) const {
DCHECK_GT(num_steps_down, 0);
const auto begin = snapped_sizes_.begin();
const auto end = snapped_sizes_.end();
DCHECK(begin != end);
const gfx::Size area_as_size(area, 1); // A facade for CompareByArea().
const auto p = std::lower_bound(begin, end, area_as_size, &CompareByArea);
if ((p - begin) >= num_steps_down)
return *(p - num_steps_down);
else
return *begin;
}
void CaptureResolutionChooser::RecomputeCaptureSize() {
const gfx::Size old_capture_size = capture_size_;
capture_size_ = FindNearestFrameSize(target_area_);
VLOG_IF(1, capture_size_ != old_capture_size)
<< "Recomputed capture size from " << old_capture_size.ToString()
<< " to " << capture_size_.ToString() << " ("
<< (100.0 * capture_size_.height() / snapped_sizes_.back().height())
<< "% of ideal size)";
}
void CaptureResolutionChooser::UpdateSnappedFrameSizes() {
// Compute the largest snapped frame size, and the one that will determine the
// aspect ratio of all the snapped frame sizes. If the |source_size_| has not
// yet been set, use the |capture_size_| as a substitute.
gfx::Size constrained_size =
source_size_.IsEmpty() ? capture_size_ : source_size_;
constrained_size = ComputeBoundedCaptureSize(
apply_aspect_ratio_adjustment_
? PadToMatchAspectRatio(constrained_size, max_frame_size_)
: constrained_size,
min_frame_size_, max_frame_size_);
// Start with the constrained size as the largest in the set.
snapped_sizes_.clear();
snapped_sizes_.push_back(constrained_size);
// Repeatedly decrease the size in steps, adding each to |snapped_sizes_|.
// However, skip the sizes that do not decrease in area by enough, relative to
// the prior size.
int last_area = constrained_size.GetArea();
for (int height = constrained_size.height() - kSnappedHeightStep;
height >= min_frame_size_.height(); height -= kSnappedHeightStep) {
const int width =
height * constrained_size.width() / constrained_size.height();
if (width < min_frame_size_.width())
break;
const int smaller_area = width * height;
const int percent_decrease = 100 * (last_area - smaller_area) / last_area;
if (percent_decrease >= kMinAreaDecreasePercent) {
snapped_sizes_.push_back(gfx::Size(width, height));
last_area = smaller_area;
}
}
// Reverse ordering, so that sizes are from smallest to largest.
std::reverse(snapped_sizes_.begin(), snapped_sizes_.end());
if (VLOG_IS_ON(1)) {
std::vector<std::string> stringified_sizes;
for (const gfx::Size& size : snapped_sizes_)
stringified_sizes.push_back(size.ToString());
VLOG_STREAM(1) << "Recomputed snapped frame sizes: "
<< base::JoinString(stringified_sizes, " <--> ");
}
}
} // namespace media