blob: dfa3ccd1dfc73ab7f2789cb96cce71a824aede60 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/base/video_util.h"
#include <stdint.h>
#include <cmath>
#include <memory>
#include "base/compiler_specific.h"
#include "base/containers/heap_array.h"
#include "base/containers/span.h"
#include "base/memory/raw_ptr_exclusion.h"
#include "media/base/limits.h"
#include "media/base/video_frame.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/geometry/rect.h"
namespace {
// Initialize a plane's visible rect with value circularly from 0 to 255.
void FillPlaneWithPattern(uint8_t* data,
int stride,
const gfx::Size& visible_size) {
DCHECK(data && visible_size.width() <= stride);
uint32_t val = 0;
uint8_t* src = data;
for (int i = 0; i < visible_size.height(); ++i, UNSAFE_TODO(src += stride)) {
for (int j = 0; j < visible_size.width(); ++j, ++val)
UNSAFE_TODO(src[j]) = val & 0xff;
}
}
// Create a VideoFrame and initialize the visible rect using
// |FillPlaneWithPattern()|. For testing purpose, the VideoFrame should be
// filled with varying values, which is different from
// |VideoFrame::CreateColorFrame()| where the entrire VideoFrame is filled
// with a given color.
scoped_refptr<media::VideoFrame> CreateFrameWithPatternFilled(
media::VideoPixelFormat format,
const gfx::Size& coded_size,
const gfx::Rect& visible_rect,
const gfx::Size& natural_size,
base::TimeDelta timestamp) {
scoped_refptr<media::VideoFrame> frame(media::VideoFrame::CreateFrame(
format, coded_size, visible_rect, natural_size, timestamp));
FillPlaneWithPattern(frame->writable_data(media::VideoFrame::Plane::kY),
frame->stride(media::VideoFrame::Plane::kY),
frame->visible_rect().size());
FillPlaneWithPattern(
frame->writable_data(media::VideoFrame::Plane::kU),
frame->stride(media::VideoFrame::Plane::kU),
media::VideoFrame::PlaneSize(format, media::VideoFrame::Plane::kU,
frame->visible_rect().size()));
FillPlaneWithPattern(
frame->writable_data(media::VideoFrame::Plane::kV),
frame->stride(media::VideoFrame::Plane::kV),
media::VideoFrame::PlaneSize(format, media::VideoFrame::Plane::kV,
frame->visible_rect().size()));
return frame;
}
// Helper function used to verify the data in the coded region after copying the
// visible region and padding the remaining area.
bool VerifyPlanCopyWithPadding(const uint8_t* src,
size_t src_stride,
// Size of visible region.
const gfx::Size& src_size,
const uint8_t* dst,
size_t dst_stride,
// Coded size of |dst|.
const gfx::Size& dst_size) {
if (!src || !dst)
return false;
const size_t src_width = src_size.width();
const size_t src_height = src_size.height();
const size_t dst_width = dst_size.width();
const size_t dst_height = dst_size.height();
if (src_width > dst_width || src_width > src_stride ||
src_height > dst_height || src_size.IsEmpty() || dst_size.IsEmpty())
return false;
const uint8_t *src_ptr = src, *dst_ptr = dst;
for (size_t i = 0; i < src_height; ++i, UNSAFE_TODO(src_ptr += src_stride),
UNSAFE_TODO(dst_ptr += dst_stride)) {
if (UNSAFE_TODO(memcmp(src_ptr, dst_ptr, src_width))) {
return false;
}
for (size_t j = src_width; j < dst_width; ++j) {
if (UNSAFE_TODO(src_ptr[src_width - 1]) != UNSAFE_TODO(dst_ptr[j])) {
return false;
}
}
}
if (src_height < dst_height) {
src_ptr = UNSAFE_TODO(dst + (src_height - 1) * dst_stride);
if (UNSAFE_TODO(memcmp(src_ptr, dst_ptr, dst_width))) {
return false;
}
}
return true;
}
bool VerifyCopyWithPadding(const media::VideoFrame& src_frame,
const media::VideoFrame& dst_frame) {
if (!src_frame.IsMappable() || !dst_frame.IsMappable() ||
src_frame.visible_rect().size() != dst_frame.visible_rect().size())
return false;
if (!VerifyPlanCopyWithPadding(
src_frame.visible_data(media::VideoFrame::Plane::kY),
src_frame.stride(media::VideoFrame::Plane::kY),
src_frame.visible_rect().size(),
dst_frame.data(media::VideoFrame::Plane::kY),
dst_frame.stride(media::VideoFrame::Plane::kY),
dst_frame.coded_size())) {
return false;
}
if (!VerifyPlanCopyWithPadding(
src_frame.visible_data(media::VideoFrame::Plane::kU),
src_frame.stride(media::VideoFrame::Plane::kU),
media::VideoFrame::PlaneSize(media::PIXEL_FORMAT_I420,
media::VideoFrame::Plane::kU,
src_frame.visible_rect().size()),
dst_frame.data(media::VideoFrame::Plane::kU),
dst_frame.stride(media::VideoFrame::Plane::kU),
media::VideoFrame::PlaneSize(media::PIXEL_FORMAT_I420,
media::VideoFrame::Plane::kU,
dst_frame.coded_size()))) {
return false;
}
if (!VerifyPlanCopyWithPadding(
src_frame.visible_data(media::VideoFrame::Plane::kV),
src_frame.stride(media::VideoFrame::Plane::kV),
media::VideoFrame::PlaneSize(media::PIXEL_FORMAT_I420,
media::VideoFrame::Plane::kV,
src_frame.visible_rect().size()),
dst_frame.data(media::VideoFrame::Plane::kV),
dst_frame.stride(media::VideoFrame::Plane::kV),
media::VideoFrame::PlaneSize(media::PIXEL_FORMAT_I420,
media::VideoFrame::Plane::kV,
dst_frame.coded_size()))) {
return false;
}
return true;
}
} // namespace
namespace media {
class VideoUtilTest : public testing::Test {
public:
VideoUtilTest() = default;
VideoUtilTest(const VideoUtilTest&) = delete;
VideoUtilTest& operator=(const VideoUtilTest&) = delete;
~VideoUtilTest() override = default;
void CreateDestinationFrame(int width, int height) {
gfx::Size size(width, height);
destination_frame_ = VideoFrame::CreateFrame(
PIXEL_FORMAT_I420, size, gfx::Rect(size), size, base::TimeDelta());
}
private:
scoped_refptr<VideoFrame> destination_frame_;
};
namespace {
uint8_t src6x4[] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11,
12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23};
// Target images, name pattern target_rotation_flipV_flipH.
uint8_t* target6x4_0_n_n = src6x4;
uint8_t target6x4_0_n_y[] = {5, 4, 3, 2, 1, 0, 11, 10, 9, 8, 7, 6,
17, 16, 15, 14, 13, 12, 23, 22, 21, 20, 19, 18};
uint8_t target6x4_0_y_n[] = {18, 19, 20, 21, 22, 23, 12, 13, 14, 15, 16, 17,
6, 7, 8, 9, 10, 11, 0, 1, 2, 3, 4, 5};
uint8_t target6x4_0_y_y[] = {23, 22, 21, 20, 19, 18, 17, 16, 15, 14, 13, 12,
11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0};
uint8_t target6x4_90_n_n[] = {255, 19, 13, 7, 1, 255, 255, 20, 14, 8, 2, 255,
255, 21, 15, 9, 3, 255, 255, 22, 16, 10, 4, 255};
uint8_t target6x4_90_n_y[] = {255, 1, 7, 13, 19, 255, 255, 2, 8, 14, 20, 255,
255, 3, 9, 15, 21, 255, 255, 4, 10, 16, 22, 255};
uint8_t target6x4_90_y_n[] = {255, 22, 16, 10, 4, 255, 255, 21, 15, 9, 3, 255,
255, 20, 14, 8, 2, 255, 255, 19, 13, 7, 1, 255};
uint8_t target6x4_90_y_y[] = {255, 4, 10, 16, 22, 255, 255, 3, 9, 15, 21, 255,
255, 2, 8, 14, 20, 255, 255, 1, 7, 13, 19, 255};
uint8_t* target6x4_180_n_n = target6x4_0_y_y;
uint8_t* target6x4_180_n_y = target6x4_0_y_n;
uint8_t* target6x4_180_y_n = target6x4_0_n_y;
uint8_t* target6x4_180_y_y = target6x4_0_n_n;
uint8_t* target6x4_270_n_n = target6x4_90_y_y;
uint8_t* target6x4_270_n_y = target6x4_90_y_n;
uint8_t* target6x4_270_y_n = target6x4_90_n_y;
uint8_t* target6x4_270_y_y = target6x4_90_n_n;
uint8_t src4x6[] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11,
12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23};
uint8_t* target4x6_0_n_n = src4x6;
uint8_t target4x6_0_n_y[] = {3, 2, 1, 0, 7, 6, 5, 4, 11, 10, 9, 8,
15, 14, 13, 12, 19, 18, 17, 16, 23, 22, 21, 20};
uint8_t target4x6_0_y_n[] = {20, 21, 22, 23, 16, 17, 18, 19, 12, 13, 14, 15,
8, 9, 10, 11, 4, 5, 6, 7, 0, 1, 2, 3};
uint8_t target4x6_0_y_y[] = {23, 22, 21, 20, 19, 18, 17, 16, 15, 14, 13, 12,
11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0};
uint8_t target4x6_90_n_n[] = {255, 255, 255, 255, 16, 12, 8, 4,
17, 13, 9, 5, 18, 14, 10, 6,
19, 15, 11, 7, 255, 255, 255, 255};
uint8_t target4x6_90_n_y[] = {255, 255, 255, 255, 4, 8, 12, 16,
5, 9, 13, 17, 6, 10, 14, 18,
7, 11, 15, 19, 255, 255, 255, 255};
uint8_t target4x6_90_y_n[] = {255, 255, 255, 255, 19, 15, 11, 7,
18, 14, 10, 6, 17, 13, 9, 5,
16, 12, 8, 4, 255, 255, 255, 255};
uint8_t target4x6_90_y_y[] = {255, 255, 255, 255, 7, 11, 15, 19,
6, 10, 14, 18, 5, 9, 13, 17,
4, 8, 12, 16, 255, 255, 255, 255};
uint8_t* target4x6_180_n_n = target4x6_0_y_y;
uint8_t* target4x6_180_n_y = target4x6_0_y_n;
uint8_t* target4x6_180_y_n = target4x6_0_n_y;
uint8_t* target4x6_180_y_y = target4x6_0_n_n;
uint8_t* target4x6_270_n_n = target4x6_90_y_y;
uint8_t* target4x6_270_n_y = target4x6_90_y_n;
uint8_t* target4x6_270_y_n = target4x6_90_n_y;
uint8_t* target4x6_270_y_y = target4x6_90_n_n;
struct VideoRotationTestData {
// These fields are not raw_ptr<>s because they only ever point to
// statically-allocated data which is never freed, and hence cannot dangle.
RAW_PTR_EXCLUSION uint8_t* src;
RAW_PTR_EXCLUSION uint8_t* target;
int width;
int height;
int rotation;
bool flip_vert;
bool flip_horiz;
};
const VideoRotationTestData kVideoRotationTestData[] = {
{ src6x4, target6x4_0_n_n, 6, 4, 0, false, false },
{ src6x4, target6x4_0_n_y, 6, 4, 0, false, true },
{ src6x4, target6x4_0_y_n, 6, 4, 0, true, false },
{ src6x4, target6x4_0_y_y, 6, 4, 0, true, true },
{ src6x4, target6x4_90_n_n, 6, 4, 90, false, false },
{ src6x4, target6x4_90_n_y, 6, 4, 90, false, true },
{ src6x4, target6x4_90_y_n, 6, 4, 90, true, false },
{ src6x4, target6x4_90_y_y, 6, 4, 90, true, true },
{ src6x4, target6x4_180_n_n, 6, 4, 180, false, false },
{ src6x4, target6x4_180_n_y, 6, 4, 180, false, true },
{ src6x4, target6x4_180_y_n, 6, 4, 180, true, false },
{ src6x4, target6x4_180_y_y, 6, 4, 180, true, true },
{ src6x4, target6x4_270_n_n, 6, 4, 270, false, false },
{ src6x4, target6x4_270_n_y, 6, 4, 270, false, true },
{ src6x4, target6x4_270_y_n, 6, 4, 270, true, false },
{ src6x4, target6x4_270_y_y, 6, 4, 270, true, true },
{ src4x6, target4x6_0_n_n, 4, 6, 0, false, false },
{ src4x6, target4x6_0_n_y, 4, 6, 0, false, true },
{ src4x6, target4x6_0_y_n, 4, 6, 0, true, false },
{ src4x6, target4x6_0_y_y, 4, 6, 0, true, true },
{ src4x6, target4x6_90_n_n, 4, 6, 90, false, false },
{ src4x6, target4x6_90_n_y, 4, 6, 90, false, true },
{ src4x6, target4x6_90_y_n, 4, 6, 90, true, false },
{ src4x6, target4x6_90_y_y, 4, 6, 90, true, true },
{ src4x6, target4x6_180_n_n, 4, 6, 180, false, false },
{ src4x6, target4x6_180_n_y, 4, 6, 180, false, true },
{ src4x6, target4x6_180_y_n, 4, 6, 180, true, false },
{ src4x6, target4x6_180_y_y, 4, 6, 180, true, true },
{ src4x6, target4x6_270_n_n, 4, 6, 270, false, false },
{ src4x6, target4x6_270_n_y, 4, 6, 270, false, true },
{ src4x6, target4x6_270_y_n, 4, 6, 270, true, false },
{ src4x6, target4x6_270_y_y, 4, 6, 270, true, true }
};
} // namespace
class VideoUtilRotationTest
: public testing::TestWithParam<VideoRotationTestData> {
public:
VideoUtilRotationTest()
: dest_(base::HeapArray<uint8_t>::Uninit(GetParam().width *
GetParam().height)) {
std::ranges::fill(dest_, 255);
}
VideoUtilRotationTest(const VideoUtilRotationTest&) = delete;
VideoUtilRotationTest& operator=(const VideoUtilRotationTest&) = delete;
~VideoUtilRotationTest() override = default;
base::span<uint8_t> dest_plane() { return dest_; }
private:
base::HeapArray<uint8_t> dest_;
};
TEST_P(VideoUtilRotationTest, Rotate) {
int rotation = GetParam().rotation;
EXPECT_TRUE((rotation >= 0) && (rotation < 360) && (rotation % 90 == 0));
base::span<uint8_t> dest = dest_plane();
RotatePlaneByPixels(GetParam().src, dest.data(), GetParam().width,
GetParam().height, rotation, GetParam().flip_vert,
GetParam().flip_horiz);
auto expected = UNSAFE_TODO(base::span(GetParam().target, dest.size()));
EXPECT_EQ(dest, expected);
}
INSTANTIATE_TEST_SUITE_P(All,
VideoUtilRotationTest,
testing::ValuesIn(kVideoRotationTestData));
// Tests the ComputeLetterboxRegion function. Also, because of shared code
// internally, this also tests ScaleSizeToFitWithinTarget().
TEST_F(VideoUtilTest, ComputeLetterboxRegion) {
EXPECT_EQ(gfx::Rect(166, 0, 667, 500),
ComputeLetterboxRegion(gfx::Rect(0, 0, 1000, 500),
gfx::Size(640, 480)));
EXPECT_EQ(gfx::Rect(0, 312, 500, 375),
ComputeLetterboxRegion(gfx::Rect(0, 0, 500, 1000),
gfx::Size(640, 480)));
EXPECT_EQ(gfx::Rect(55, 0, 889, 500),
ComputeLetterboxRegion(gfx::Rect(0, 0, 1000, 500),
gfx::Size(1920, 1080)));
EXPECT_EQ(gfx::Rect(0, 12, 100, 75),
ComputeLetterboxRegion(gfx::Rect(0, 0, 100, 100),
gfx::Size(400, 300)));
EXPECT_EQ(gfx::Rect(0, 250000000, 2000000000, 1500000000),
ComputeLetterboxRegion(gfx::Rect(0, 0, 2000000000, 2000000000),
gfx::Size(40000, 30000)));
EXPECT_TRUE(ComputeLetterboxRegion(gfx::Rect(0, 0, 2000000000, 2000000000),
gfx::Size(0, 0)).IsEmpty());
// Some operations in the internal ScaleSizeToTarget() use rounded division
// and might lose some precision, this expectation codifies that.
EXPECT_EQ(
gfx::Rect(0, 0, 1279, 720),
ComputeLetterboxRegion(gfx::Rect(0, 0, 1280, 720), gfx::Size(1057, 595)));
}
// Tests the ComputeLetterboxRegionForI420 function.
TEST_F(VideoUtilTest, ComputeLetterboxRegionForI420) {
// Note: These are the same trials as in VideoUtilTest.ComputeLetterboxRegion
// above, except that Rect coordinates are nudged into even-numbered values.
EXPECT_EQ(gfx::Rect(166, 0, 666, 500),
ComputeLetterboxRegionForI420(gfx::Rect(0, 0, 1000, 500),
gfx::Size(640, 480)));
EXPECT_EQ(gfx::Rect(0, 312, 500, 374),
ComputeLetterboxRegionForI420(gfx::Rect(0, 0, 500, 1000),
gfx::Size(640, 480)));
EXPECT_EQ(gfx::Rect(54, 0, 890, 500),
ComputeLetterboxRegionForI420(gfx::Rect(0, 0, 1000, 500),
gfx::Size(1920, 1080)));
EXPECT_EQ(gfx::Rect(0, 12, 100, 74),
ComputeLetterboxRegionForI420(gfx::Rect(0, 0, 100, 100),
gfx::Size(400, 300)));
EXPECT_EQ(
gfx::Rect(0, 250000000, 2000000000, 1500000000),
ComputeLetterboxRegionForI420(gfx::Rect(0, 0, 2000000000, 2000000000),
gfx::Size(40000, 30000)));
EXPECT_TRUE(ComputeLetterboxRegionForI420(
gfx::Rect(0, 0, 2000000000, 2000000000), gfx::Size(0, 0))
.IsEmpty());
}
// Tests the MinimallyShrinkRectForI420 function.
TEST_F(VideoUtilTest, MinimallyShrinkRectForI420) {
// A few no-ops:
EXPECT_EQ(gfx::Rect(2, 2, 100, 100),
MinimallyShrinkRectForI420(gfx::Rect(2, 2, 100, 100)));
EXPECT_EQ(gfx::Rect(2, -2, 100, 100),
MinimallyShrinkRectForI420(gfx::Rect(2, -2, 100, 100)));
EXPECT_EQ(gfx::Rect(-2, 2, 100, 100),
MinimallyShrinkRectForI420(gfx::Rect(-2, 2, 100, 100)));
// Origin has odd coordinates:
EXPECT_EQ(gfx::Rect(2, 2, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(1, 1, 100, 100)));
EXPECT_EQ(gfx::Rect(0, 2, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(-1, 1, 100, 100)));
EXPECT_EQ(gfx::Rect(2, 0, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(1, -1, 100, 100)));
// Size is odd:
EXPECT_EQ(gfx::Rect(2, 2, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(2, 2, 99, 99)));
EXPECT_EQ(gfx::Rect(-2, 2, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(-2, 2, 99, 99)));
EXPECT_EQ(gfx::Rect(2, -2, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(2, -2, 99, 99)));
// Both are odd:
EXPECT_EQ(gfx::Rect(2, 2, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(1, 1, 99, 99)));
EXPECT_EQ(gfx::Rect(0, 2, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(-1, 1, 99, 99)));
EXPECT_EQ(gfx::Rect(2, 0, 98, 98),
MinimallyShrinkRectForI420(gfx::Rect(1, -1, 99, 99)));
// Check the biggest rectangle that the function will accept:
constexpr int kMinDimension = -1 * limits::kMaxDimension;
if (limits::kMaxDimension % 2 == 0) {
EXPECT_EQ(gfx::Rect(kMinDimension, kMinDimension, 2 * limits::kMaxDimension,
2 * limits::kMaxDimension),
MinimallyShrinkRectForI420(gfx::Rect(kMinDimension, kMinDimension,
2 * limits::kMaxDimension,
2 * limits::kMaxDimension)));
} else {
EXPECT_EQ(
gfx::Rect(kMinDimension + 1, kMinDimension + 1,
2 * limits::kMaxDimension - 2, 2 * limits::kMaxDimension - 2),
MinimallyShrinkRectForI420(gfx::Rect(kMinDimension, kMinDimension,
2 * limits::kMaxDimension,
2 * limits::kMaxDimension)));
}
}
TEST_F(VideoUtilTest, ScaleSizeToEncompassTarget) {
EXPECT_EQ(gfx::Size(1000, 750),
ScaleSizeToEncompassTarget(gfx::Size(640, 480),
gfx::Size(1000, 500)));
EXPECT_EQ(gfx::Size(1333, 1000),
ScaleSizeToEncompassTarget(gfx::Size(640, 480),
gfx::Size(500, 1000)));
EXPECT_EQ(gfx::Size(1000, 563),
ScaleSizeToEncompassTarget(gfx::Size(1920, 1080),
gfx::Size(1000, 500)));
EXPECT_EQ(gfx::Size(133, 100),
ScaleSizeToEncompassTarget(gfx::Size(400, 300),
gfx::Size(100, 100)));
EXPECT_EQ(gfx::Size(266666667, 200000000),
ScaleSizeToEncompassTarget(gfx::Size(40000, 30000),
gfx::Size(200000000, 200000000)));
EXPECT_TRUE(ScaleSizeToEncompassTarget(
gfx::Size(0, 0), gfx::Size(2000000000, 2000000000)).IsEmpty());
}
TEST_F(VideoUtilTest, CropSizeForScalingToTarget) {
// Test same aspect ratios.
EXPECT_EQ(gfx::Rect(0, 0, 640, 360),
CropSizeForScalingToTarget(gfx::Size(640, 360), gfx::Size(16, 9)));
EXPECT_EQ(gfx::Rect(0, 0, 320, 240),
CropSizeForScalingToTarget(gfx::Size(320, 240), gfx::Size(4, 3)));
EXPECT_EQ(
gfx::Rect(0, 0, 320, 240),
CropSizeForScalingToTarget(gfx::Size(321, 241), gfx::Size(4, 3), 2));
// Test cropping 4:3 from 16:9.
EXPECT_EQ(gfx::Rect(80, 0, 480, 360),
CropSizeForScalingToTarget(gfx::Size(640, 360), gfx::Size(4, 3)));
EXPECT_EQ(gfx::Rect(53, 0, 320, 240),
CropSizeForScalingToTarget(gfx::Size(426, 240), gfx::Size(4, 3)));
EXPECT_EQ(
gfx::Rect(52, 0, 320, 240),
CropSizeForScalingToTarget(gfx::Size(426, 240), gfx::Size(4, 3), 2));
// Test cropping 16:9 from 4:3.
EXPECT_EQ(gfx::Rect(0, 30, 320, 180),
CropSizeForScalingToTarget(gfx::Size(320, 240), gfx::Size(16, 9)));
EXPECT_EQ(gfx::Rect(0, 9, 96, 54),
CropSizeForScalingToTarget(gfx::Size(96, 72), gfx::Size(16, 9)));
EXPECT_EQ(gfx::Rect(0, 8, 96, 54),
CropSizeForScalingToTarget(gfx::Size(96, 72), gfx::Size(16, 9), 2));
// Test abnormal inputs.
EXPECT_EQ(gfx::Rect(),
CropSizeForScalingToTarget(gfx::Size(0, 1), gfx::Size(1, 1)));
EXPECT_EQ(gfx::Rect(),
CropSizeForScalingToTarget(gfx::Size(1, 0), gfx::Size(1, 1)));
EXPECT_EQ(gfx::Rect(),
CropSizeForScalingToTarget(gfx::Size(1, 1), gfx::Size(0, 1)));
EXPECT_EQ(gfx::Rect(),
CropSizeForScalingToTarget(gfx::Size(1, 1), gfx::Size(1, 0)));
}
TEST_F(VideoUtilTest, PadToMatchAspectRatio) {
EXPECT_EQ(gfx::Size(640, 480),
PadToMatchAspectRatio(gfx::Size(640, 480), gfx::Size(640, 480)));
EXPECT_EQ(gfx::Size(640, 480),
PadToMatchAspectRatio(gfx::Size(640, 480), gfx::Size(4, 3)));
EXPECT_EQ(gfx::Size(960, 480),
PadToMatchAspectRatio(gfx::Size(640, 480), gfx::Size(1000, 500)));
EXPECT_EQ(gfx::Size(640, 1280),
PadToMatchAspectRatio(gfx::Size(640, 480), gfx::Size(500, 1000)));
EXPECT_EQ(gfx::Size(2160, 1080),
PadToMatchAspectRatio(gfx::Size(1920, 1080), gfx::Size(1000, 500)));
EXPECT_EQ(gfx::Size(400, 400),
PadToMatchAspectRatio(gfx::Size(400, 300), gfx::Size(100, 100)));
EXPECT_EQ(gfx::Size(400, 400),
PadToMatchAspectRatio(gfx::Size(300, 400), gfx::Size(100, 100)));
EXPECT_EQ(gfx::Size(40000, 40000),
PadToMatchAspectRatio(gfx::Size(40000, 30000),
gfx::Size(2000000000, 2000000000)));
EXPECT_TRUE(PadToMatchAspectRatio(
gfx::Size(40000, 30000), gfx::Size(0, 0)).IsEmpty());
}
TEST_F(VideoUtilTest, LetterboxVideoFrame) {
int width = 40;
int height = 30;
gfx::Size size(width, height);
scoped_refptr<VideoFrame> frame(VideoFrame::CreateFrame(
PIXEL_FORMAT_I420, size, gfx::Rect(size), size, base::TimeDelta()));
for (int left_margin = 0; left_margin <= 10; left_margin += 10) {
for (int right_margin = 0; right_margin <= 10; right_margin += 10) {
for (int top_margin = 0; top_margin <= 10; top_margin += 10) {
for (int bottom_margin = 0; bottom_margin <= 10; bottom_margin += 10) {
gfx::Rect view_area(left_margin, top_margin,
width - left_margin - right_margin,
height - top_margin - bottom_margin);
FillYUV(frame.get(), 0x1, 0x2, 0x3);
LetterboxVideoFrame(frame.get(), view_area);
for (int x = 0; x < width; x++) {
for (int y = 0; y < height; y++) {
bool inside = x >= view_area.x() &&
x < view_area.x() + view_area.width() &&
y >= view_area.y() &&
y < view_area.y() + view_area.height();
UNSAFE_TODO(
EXPECT_EQ(frame->data(VideoFrame::Plane::kY)
[y * frame->stride(VideoFrame::Plane::kY) + x],
inside ? 0x01 : 0x00));
UNSAFE_TODO(EXPECT_EQ(
frame->data(VideoFrame::Plane::kU)
[(y / 2) * frame->stride(VideoFrame::Plane::kU) +
(x / 2)],
inside ? 0x02 : 0x80));
UNSAFE_TODO(EXPECT_EQ(
frame->data(VideoFrame::Plane::kV)
[(y / 2) * frame->stride(VideoFrame::Plane::kV) +
(x / 2)],
inside ? 0x03 : 0x80));
}
}
}
}
}
}
}
TEST_F(VideoUtilTest, I420CopyWithPadding) {
gfx::Size visible_size(40, 30);
scoped_refptr<VideoFrame> src_frame = CreateFrameWithPatternFilled(
PIXEL_FORMAT_I420, visible_size, gfx::Rect(visible_size), visible_size,
base::TimeDelta());
// Expect to return false when copying to an empty buffer.
EXPECT_FALSE(I420CopyWithPadding(*src_frame, nullptr));
scoped_refptr<VideoFrame> dst_frame = CreateFrameWithPatternFilled(
PIXEL_FORMAT_I420, visible_size, gfx::Rect(visible_size), visible_size,
base::TimeDelta());
EXPECT_TRUE(I420CopyWithPadding(*src_frame, dst_frame.get()));
EXPECT_TRUE(VerifyCopyWithPadding(*src_frame, *dst_frame));
gfx::Size coded_size(60, 40);
dst_frame = CreateFrameWithPatternFilled(PIXEL_FORMAT_I420, coded_size,
gfx::Rect(visible_size), coded_size,
base::TimeDelta());
EXPECT_TRUE(I420CopyWithPadding(*src_frame, dst_frame.get()));
EXPECT_TRUE(VerifyCopyWithPadding(*src_frame, *dst_frame));
gfx::Size odd_size(39, 31);
src_frame = CreateFrameWithPatternFilled(PIXEL_FORMAT_I420, odd_size,
gfx::Rect(odd_size), odd_size,
base::TimeDelta());
dst_frame = CreateFrameWithPatternFilled(PIXEL_FORMAT_I420, coded_size,
gfx::Rect(odd_size), coded_size,
base::TimeDelta());
EXPECT_TRUE(I420CopyWithPadding(*src_frame, dst_frame.get()));
EXPECT_TRUE(VerifyCopyWithPadding(*src_frame, *dst_frame));
}
TEST_F(VideoUtilTest, WrapAsI420VideoFrame) {
gfx::Size size(640, 480);
scoped_refptr<VideoFrame> src_frame = VideoFrame::CreateFrame(
PIXEL_FORMAT_I420A, size, gfx::Rect(size), size, base::Days(1));
scoped_refptr<VideoFrame> dst_frame = WrapAsI420VideoFrame(src_frame);
EXPECT_EQ(dst_frame->format(), PIXEL_FORMAT_I420);
EXPECT_EQ(dst_frame->timestamp(), src_frame->timestamp());
EXPECT_EQ(dst_frame->coded_size(), src_frame->coded_size());
EXPECT_EQ(dst_frame->visible_rect(), src_frame->visible_rect());
EXPECT_EQ(dst_frame->natural_size(), src_frame->natural_size());
std::vector<size_t> planes = {VideoFrame::Plane::kY, VideoFrame::Plane::kU,
VideoFrame::Plane::kV};
for (auto plane : planes)
EXPECT_EQ(dst_frame->data(plane), src_frame->data(plane));
// Check that memory for planes is not released upon destruction of the
// original frame pointer (new frame holds a reference). This check relies on
// ASAN.
src_frame.reset();
for (auto plane : planes)
UNSAFE_TODO(
memset(dst_frame->writable_data(plane), 1, dst_frame->stride(plane)));
}
} // namespace media