blob: 5cac8e36a90f6fa7d9658de876e8cae6051ef90a [file] [log] [blame]
// Copyright 2013 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 "cc/resources/video_resource_updater.h"
#include <stddef.h>
#include <stdint.h>
#include "base/bind.h"
#include "base/memory/ptr_util.h"
#include "cc/resources/resource_provider.h"
#include "cc/test/fake_output_surface.h"
#include "cc/test/fake_output_surface_client.h"
#include "cc/test/fake_resource_provider.h"
#include "cc/test/test_shared_bitmap_manager.h"
#include "cc/test/test_web_graphics_context_3d.h"
#include "gpu/GLES2/gl2extchromium.h"
#include "media/base/video_frame.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace cc {
namespace {
class WebGraphicsContext3DUploadCounter : public TestWebGraphicsContext3D {
public:
void texSubImage2D(GLenum target,
GLint level,
GLint xoffset,
GLint yoffset,
GLsizei width,
GLsizei height,
GLenum format,
GLenum type,
const void* pixels) override {
++upload_count_;
}
void texStorage2DEXT(GLenum target,
GLint levels,
GLuint internalformat,
GLint width,
GLint height) override {
}
GLuint createTexture() override {
++created_texture_count_;
return TestWebGraphicsContext3D::createTexture();
}
void deleteTexture(GLuint texture) override {
--created_texture_count_;
TestWebGraphicsContext3D::deleteTexture(texture);
}
void deleteTextures(GLsizei count, const GLuint* ids) override {
created_texture_count_ -= count;
TestWebGraphicsContext3D::deleteTextures(count, ids);
}
int UploadCount() { return upload_count_; }
void ResetUploadCount() { upload_count_ = 0; }
int TextureCreationCount() { return created_texture_count_; }
void ResetTextureCreationCount() { created_texture_count_ = 0; }
private:
int upload_count_;
int created_texture_count_;
};
class SharedBitmapManagerAllocationCounter : public TestSharedBitmapManager {
public:
std::unique_ptr<viz::SharedBitmap> AllocateSharedBitmap(
const gfx::Size& size) override {
++allocation_count_;
return TestSharedBitmapManager::AllocateSharedBitmap(size);
}
int AllocationCount() { return allocation_count_; }
void ResetAllocationCount() { allocation_count_ = 0; }
private:
int allocation_count_;
};
class VideoResourceUpdaterTest : public testing::Test {
protected:
VideoResourceUpdaterTest() {
std::unique_ptr<WebGraphicsContext3DUploadCounter> context3d(
new WebGraphicsContext3DUploadCounter());
context3d_ = context3d.get();
context3d_->set_support_texture_storage(true);
context_provider_ = TestContextProvider::Create(std::move(context3d));
context_provider_->BindToCurrentThread();
}
void SetUp() override {
testing::Test::SetUp();
shared_bitmap_manager_.reset(new SharedBitmapManagerAllocationCounter());
resource_provider3d_ =
FakeResourceProvider::CreateLayerTreeResourceProvider(
context_provider_.get(), shared_bitmap_manager_.get(), nullptr,
high_bit_for_testing_);
resource_provider_software_ =
FakeResourceProvider::CreateLayerTreeResourceProvider(
nullptr, shared_bitmap_manager_.get(), nullptr,
high_bit_for_testing_);
}
scoped_refptr<media::VideoFrame> CreateTestYUVVideoFrame() {
const int kDimension = 10;
gfx::Size size(kDimension, kDimension);
static uint8_t y_data[kDimension * kDimension] = {0};
static uint8_t u_data[kDimension * kDimension / 2] = {0};
static uint8_t v_data[kDimension * kDimension / 2] = {0};
scoped_refptr<media::VideoFrame> video_frame =
media::VideoFrame::WrapExternalYuvData(
media::PIXEL_FORMAT_I422, // format
size, // coded_size
gfx::Rect(size), // visible_rect
size, // natural_size
size.width(), // y_stride
size.width() / 2, // u_stride
size.width() / 2, // v_stride
y_data, // y_data
u_data, // u_data
v_data, // v_data
base::TimeDelta()); // timestamp
EXPECT_TRUE(video_frame);
return video_frame;
}
scoped_refptr<media::VideoFrame> CreateWonkyTestYUVVideoFrame() {
const int kDimension = 10;
const int kYWidth = kDimension + 5;
const int kUWidth = (kYWidth + 1) / 2 + 200;
const int kVWidth = (kYWidth + 1) / 2 + 1;
static uint8_t y_data[kYWidth * kDimension] = {0};
static uint8_t u_data[kUWidth * kDimension] = {0};
static uint8_t v_data[kVWidth * kDimension] = {0};
scoped_refptr<media::VideoFrame> video_frame =
media::VideoFrame::WrapExternalYuvData(
media::PIXEL_FORMAT_I422, // format
gfx::Size(kYWidth, kDimension), // coded_size
gfx::Rect(2, 0, kDimension, kDimension), // visible_rect
gfx::Size(kDimension, kDimension), // natural_size
-kYWidth, // y_stride (negative)
kUWidth, // u_stride
kVWidth, // v_stride
y_data + kYWidth * (kDimension - 1), // y_data
u_data, // u_data
v_data, // v_data
base::TimeDelta()); // timestamp
EXPECT_TRUE(video_frame);
return video_frame;
}
scoped_refptr<media::VideoFrame> CreateTestHighBitFrame() {
const int kDimension = 10;
gfx::Size size(kDimension, kDimension);
scoped_refptr<media::VideoFrame> video_frame(media::VideoFrame::CreateFrame(
media::PIXEL_FORMAT_YUV420P10, size, gfx::Rect(size), size,
base::TimeDelta()));
EXPECT_TRUE(video_frame);
return video_frame;
}
void SetReleaseSyncToken(const gpu::SyncToken& sync_token) {
release_sync_token_ = sync_token;
}
scoped_refptr<media::VideoFrame> CreateTestHardwareVideoFrame(
media::VideoPixelFormat format,
unsigned target) {
const int kDimension = 10;
gfx::Size size(kDimension, kDimension);
gpu::Mailbox mailbox;
mailbox.name[0] = 51;
gpu::MailboxHolder mailbox_holders[media::VideoFrame::kMaxPlanes] = {
gpu::MailboxHolder(mailbox, kMailboxSyncToken, target)};
scoped_refptr<media::VideoFrame> video_frame =
media::VideoFrame::WrapNativeTextures(
format, mailbox_holders,
base::Bind(&VideoResourceUpdaterTest::SetReleaseSyncToken,
base::Unretained(this)),
size, // coded_size
gfx::Rect(size), // visible_rect
size, // natural_size
base::TimeDelta()); // timestamp
EXPECT_TRUE(video_frame);
return video_frame;
}
scoped_refptr<media::VideoFrame> CreateTestRGBAHardwareVideoFrame() {
return CreateTestHardwareVideoFrame(media::PIXEL_FORMAT_ARGB,
GL_TEXTURE_2D);
}
scoped_refptr<media::VideoFrame> CreateTestStreamTextureHardwareVideoFrame(
bool needs_copy) {
scoped_refptr<media::VideoFrame> video_frame = CreateTestHardwareVideoFrame(
media::PIXEL_FORMAT_ARGB, GL_TEXTURE_EXTERNAL_OES);
video_frame->metadata()->SetBoolean(
media::VideoFrameMetadata::COPY_REQUIRED, needs_copy);
return video_frame;
}
scoped_refptr<media::VideoFrame> CreateTestYuvHardwareVideoFrame(
media::VideoPixelFormat format,
size_t num_textures,
unsigned target) {
const int kDimension = 10;
gfx::Size size(kDimension, kDimension);
gpu::MailboxHolder mailbox_holders[media::VideoFrame::kMaxPlanes];
for (size_t i = 0; i < num_textures; ++i) {
gpu::Mailbox mailbox;
mailbox.name[0] = 50 + 1;
mailbox_holders[i] =
gpu::MailboxHolder(mailbox, kMailboxSyncToken, target);
}
scoped_refptr<media::VideoFrame> video_frame =
media::VideoFrame::WrapNativeTextures(
format, mailbox_holders,
base::Bind(&VideoResourceUpdaterTest::SetReleaseSyncToken,
base::Unretained(this)),
size, // coded_size
gfx::Rect(size), // visible_rect
size, // natural_size
base::TimeDelta()); // timestamp
EXPECT_TRUE(video_frame);
return video_frame;
}
static const gpu::SyncToken kMailboxSyncToken;
WebGraphicsContext3DUploadCounter* context3d_;
scoped_refptr<TestContextProvider> context_provider_;
std::unique_ptr<SharedBitmapManagerAllocationCounter> shared_bitmap_manager_;
std::unique_ptr<LayerTreeResourceProvider> resource_provider3d_;
std::unique_ptr<LayerTreeResourceProvider> resource_provider_software_;
gpu::SyncToken release_sync_token_;
bool high_bit_for_testing_ = false;
};
const gpu::SyncToken VideoResourceUpdaterTest::kMailboxSyncToken =
gpu::SyncToken(gpu::CommandBufferNamespace::GPU_IO,
gpu::CommandBufferId::FromUnsafeValue(0x123),
7);
TEST_F(VideoResourceUpdaterTest, SoftwareFrame) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestYUVVideoFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
}
TEST_F(VideoResourceUpdaterTest, HighBitFrameNoF16) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestHighBitFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
}
class VideoResourceUpdaterTestWithF16 : public VideoResourceUpdaterTest {
public:
VideoResourceUpdaterTestWithF16() : VideoResourceUpdaterTest() {
context3d_->set_support_texture_half_float_linear(true);
}
};
TEST_F(VideoResourceUpdaterTestWithF16, HighBitFrame) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestHighBitFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_NEAR(resources.multiplier, 2.0, 0.1);
EXPECT_NEAR(resources.offset, 0.5, 0.1);
// Create the resource again, to test the path where the
// resources are cached.
VideoFrameExternalResources resources2 =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources2.type);
EXPECT_NEAR(resources2.multiplier, 2.0, 0.1);
EXPECT_NEAR(resources2.offset, 0.5, 0.1);
}
class VideoResourceUpdaterTestWithR16 : public VideoResourceUpdaterTest {
public:
VideoResourceUpdaterTestWithR16() : VideoResourceUpdaterTest() {
high_bit_for_testing_ = true;
context3d_->set_support_texture_norm16(true);
}
};
TEST_F(VideoResourceUpdaterTestWithR16, HighBitFrame) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
updater.SetUseR16ForTesting(true);
scoped_refptr<media::VideoFrame> video_frame = CreateTestHighBitFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
// Max 10-bit values as read by a sampler.
double max_10bit_value = ((1 << 10) - 1) / 65535.0;
EXPECT_NEAR(resources.multiplier * max_10bit_value, 1.0, 0.0001);
EXPECT_NEAR(resources.offset, 0.0, 0.1);
// Create the resource again, to test the path where the
// resources are cached.
VideoFrameExternalResources resources2 =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources2.type);
EXPECT_NEAR(resources2.multiplier * max_10bit_value, 1.0, 0.0001);
EXPECT_NEAR(resources2.offset, 0.0, 0.1);
}
TEST_F(VideoResourceUpdaterTest, HighBitFrameSoftwareCompositor) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(nullptr, resource_provider_software_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestHighBitFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::SOFTWARE_RESOURCE, resources.type);
}
TEST_F(VideoResourceUpdaterTest, WonkySoftwareFrame) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateWonkyTestYUVVideoFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
}
TEST_F(VideoResourceUpdaterTest, WonkySoftwareFrameSoftwareCompositor) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(nullptr, resource_provider_software_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateWonkyTestYUVVideoFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::SOFTWARE_RESOURCE, resources.type);
}
TEST_F(VideoResourceUpdaterTest, ReuseResource) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestYUVVideoFrame();
video_frame->set_timestamp(base::TimeDelta::FromSeconds(1234));
// Allocate the resources for a YUV video frame.
context3d_->ResetUploadCount();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_EQ(3u, resources.resources.size());
EXPECT_EQ(3u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
// Expect exactly three texture uploads, one for each plane.
EXPECT_EQ(3, context3d_->UploadCount());
// Simulate the ResourceProvider releasing the resources back to the video
// updater.
for (auto& release_callback : resources.release_callbacks)
std::move(release_callback).Run(gpu::SyncToken(), false);
// Allocate resources for the same frame.
context3d_->ResetUploadCount();
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_EQ(3u, resources.resources.size());
EXPECT_EQ(3u, resources.release_callbacks.size());
// The data should be reused so expect no texture uploads.
EXPECT_EQ(0, context3d_->UploadCount());
}
TEST_F(VideoResourceUpdaterTest, ReuseResourceNoDelete) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestYUVVideoFrame();
video_frame->set_timestamp(base::TimeDelta::FromSeconds(1234));
// Allocate the resources for a YUV video frame.
context3d_->ResetUploadCount();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_EQ(3u, resources.resources.size());
EXPECT_EQ(3u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
// Expect exactly three texture uploads, one for each plane.
EXPECT_EQ(3, context3d_->UploadCount());
// Allocate resources for the same frame.
context3d_->ResetUploadCount();
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_EQ(3u, resources.resources.size());
EXPECT_EQ(3u, resources.release_callbacks.size());
// The data should be reused so expect no texture uploads.
EXPECT_EQ(0, context3d_->UploadCount());
}
TEST_F(VideoResourceUpdaterTest, SoftwareFrameSoftwareCompositor) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(nullptr, resource_provider_software_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestYUVVideoFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::SOFTWARE_RESOURCE, resources.type);
}
TEST_F(VideoResourceUpdaterTest, ReuseResourceSoftwareCompositor) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(nullptr, resource_provider_software_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestYUVVideoFrame();
video_frame->set_timestamp(base::TimeDelta::FromSeconds(1234));
// Allocate the resources for a software video frame.
shared_bitmap_manager_->ResetAllocationCount();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::SOFTWARE_RESOURCE, resources.type);
EXPECT_EQ(0u, resources.resources.size());
EXPECT_EQ(0u, resources.release_callbacks.size());
EXPECT_LT(viz::kInvalidResourceId, resources.software_resource);
// Expect exactly one allocated shared bitmap.
EXPECT_EQ(1, shared_bitmap_manager_->AllocationCount());
// Simulate the ResourceProvider releasing the resource back to the video
// updater.
std::move(resources.software_release_callback).Run(gpu::SyncToken(), false);
// Allocate resources for the same frame.
shared_bitmap_manager_->ResetAllocationCount();
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::SOFTWARE_RESOURCE, resources.type);
EXPECT_EQ(0u, resources.resources.size());
EXPECT_EQ(0u, resources.release_callbacks.size());
EXPECT_LT(viz::kInvalidResourceId, resources.software_resource);
// The data should be reused so expect no new allocations.
EXPECT_EQ(0, shared_bitmap_manager_->AllocationCount());
}
TEST_F(VideoResourceUpdaterTest, ReuseResourceNoDeleteSoftwareCompositor) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(nullptr, resource_provider_software_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame = CreateTestYUVVideoFrame();
video_frame->set_timestamp(base::TimeDelta::FromSeconds(1234));
// Allocate the resources for a software video frame.
shared_bitmap_manager_->ResetAllocationCount();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::SOFTWARE_RESOURCE, resources.type);
EXPECT_EQ(0u, resources.resources.size());
EXPECT_EQ(0u, resources.release_callbacks.size());
EXPECT_LT(viz::kInvalidResourceId, resources.software_resource);
// Expect exactly one allocated shared bitmap.
EXPECT_EQ(1, shared_bitmap_manager_->AllocationCount());
// Allocate resources for the same frame.
shared_bitmap_manager_->ResetAllocationCount();
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::SOFTWARE_RESOURCE, resources.type);
EXPECT_EQ(0u, resources.resources.size());
EXPECT_EQ(0u, resources.release_callbacks.size());
EXPECT_NE(viz::kInvalidResourceId, resources.software_resource);
// The data should be reused so expect no new allocations.
EXPECT_EQ(0, shared_bitmap_manager_->AllocationCount());
}
TEST_F(VideoResourceUpdaterTest, CreateForHardwarePlanes) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
scoped_refptr<media::VideoFrame> video_frame =
CreateTestRGBAHardwareVideoFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::RGBA_PREMULTIPLIED_RESOURCE,
resources.type);
EXPECT_EQ(1u, resources.resources.size());
EXPECT_EQ(1u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
video_frame = CreateTestYuvHardwareVideoFrame(media::PIXEL_FORMAT_I420, 3,
GL_TEXTURE_RECTANGLE_ARB);
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_EQ(3u, resources.resources.size());
EXPECT_EQ(3u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
EXPECT_FALSE(resources.resources[0].read_lock_fences_enabled);
EXPECT_FALSE(resources.resources[1].read_lock_fences_enabled);
EXPECT_FALSE(resources.resources[2].read_lock_fences_enabled);
video_frame = CreateTestYuvHardwareVideoFrame(media::PIXEL_FORMAT_I420, 3,
GL_TEXTURE_RECTANGLE_ARB);
video_frame->metadata()->SetBoolean(
media::VideoFrameMetadata::READ_LOCK_FENCES_ENABLED, true);
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_TRUE(resources.resources[0].read_lock_fences_enabled);
EXPECT_TRUE(resources.resources[1].read_lock_fences_enabled);
EXPECT_TRUE(resources.resources[2].read_lock_fences_enabled);
}
TEST_F(VideoResourceUpdaterTest, CreateForHardwarePlanes_StreamTexture) {
bool use_stream_video_draw_quad = true;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
context3d_->ResetTextureCreationCount();
scoped_refptr<media::VideoFrame> video_frame =
CreateTestStreamTextureHardwareVideoFrame(false);
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::STREAM_TEXTURE_RESOURCE,
resources.type);
EXPECT_EQ(1u, resources.resources.size());
EXPECT_EQ((GLenum)GL_TEXTURE_EXTERNAL_OES,
resources.resources[0].mailbox_holder.texture_target);
EXPECT_EQ(1u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
EXPECT_EQ(0, context3d_->TextureCreationCount());
// A copied stream texture should return an RGBA resource in a new
// GL_TEXTURE_2D texture.
context3d_->ResetTextureCreationCount();
video_frame = CreateTestStreamTextureHardwareVideoFrame(true);
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::RGBA_PREMULTIPLIED_RESOURCE,
resources.type);
EXPECT_EQ(1u, resources.resources.size());
EXPECT_EQ((GLenum)GL_TEXTURE_2D,
resources.resources[0].mailbox_holder.texture_target);
EXPECT_EQ(1u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
EXPECT_EQ(1, context3d_->TextureCreationCount());
}
TEST_F(VideoResourceUpdaterTest, CreateForHardwarePlanes_TextureQuad) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
context3d_->ResetTextureCreationCount();
scoped_refptr<media::VideoFrame> video_frame =
CreateTestStreamTextureHardwareVideoFrame(false);
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::RGBA_PREMULTIPLIED_RESOURCE,
resources.type);
EXPECT_EQ(1u, resources.resources.size());
EXPECT_EQ((GLenum)GL_TEXTURE_EXTERNAL_OES,
resources.resources[0].mailbox_holder.texture_target);
EXPECT_EQ(1u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
EXPECT_EQ(0, context3d_->TextureCreationCount());
}
// Passthrough the sync token returned by the compositor if we don't have an
// existing release sync token.
TEST_F(VideoResourceUpdaterTest, PassReleaseSyncToken) {
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
false /* use_stream_video_draw_quad */);
const gpu::SyncToken sync_token(gpu::CommandBufferNamespace::GPU_IO,
gpu::CommandBufferId::FromUnsafeValue(0x123),
123);
{
scoped_refptr<media::VideoFrame> video_frame =
CreateTestRGBAHardwareVideoFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
ASSERT_EQ(resources.release_callbacks.size(), 1u);
std::move(resources.release_callbacks[0]).Run(sync_token, false);
}
EXPECT_EQ(release_sync_token_, sync_token);
}
// Generate new sync token because video frame has an existing sync token.
TEST_F(VideoResourceUpdaterTest, GenerateReleaseSyncToken) {
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
false /* use_stream_video_draw_quad */);
const gpu::SyncToken sync_token1(gpu::CommandBufferNamespace::GPU_IO,
gpu::CommandBufferId::FromUnsafeValue(0x123),
123);
const gpu::SyncToken sync_token2(gpu::CommandBufferNamespace::GPU_IO,
gpu::CommandBufferId::FromUnsafeValue(0x234),
234);
{
scoped_refptr<media::VideoFrame> video_frame =
CreateTestRGBAHardwareVideoFrame();
VideoFrameExternalResources resources1 =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
ASSERT_EQ(resources1.release_callbacks.size(), 1u);
std::move(resources1.release_callbacks[0]).Run(sync_token1, false);
VideoFrameExternalResources resources2 =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
ASSERT_EQ(resources2.release_callbacks.size(), 1u);
std::move(resources2.release_callbacks[0]).Run(sync_token2, false);
}
EXPECT_TRUE(release_sync_token_.HasData());
EXPECT_NE(release_sync_token_, sync_token1);
EXPECT_NE(release_sync_token_, sync_token2);
}
// Pass mailbox sync token as is if no GL operations are performed before frame
// resources are handed off to the compositor.
TEST_F(VideoResourceUpdaterTest, PassMailboxSyncToken) {
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
false /* use_stream_video_draw_quad */);
scoped_refptr<media::VideoFrame> video_frame =
CreateTestRGBAHardwareVideoFrame();
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
ASSERT_EQ(resources.resources.size(), 1u);
EXPECT_TRUE(resources.resources[0].mailbox_holder.sync_token.HasData());
EXPECT_EQ(resources.resources[0].mailbox_holder.sync_token,
kMailboxSyncToken);
}
// Generate new sync token for compositor when copying the texture.
TEST_F(VideoResourceUpdaterTest, GenerateSyncTokenOnTextureCopy) {
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
false /* use_stream_video_draw_quad */);
scoped_refptr<media::VideoFrame> video_frame =
CreateTestStreamTextureHardwareVideoFrame(true /* needs_copy */);
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
ASSERT_EQ(resources.resources.size(), 1u);
EXPECT_TRUE(resources.resources[0].mailbox_holder.sync_token.HasData());
EXPECT_NE(resources.resources[0].mailbox_holder.sync_token,
kMailboxSyncToken);
}
// NV12 VideoFrames backed by a single native texture can be sampled out
// by GL as RGB. To use them as HW overlays we need to know the format
// of the underlying buffer, that is YUV_420_BIPLANAR.
TEST_F(VideoResourceUpdaterTest, CreateForHardwarePlanes_SingleNV12) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
context3d_->ResetTextureCreationCount();
scoped_refptr<media::VideoFrame> video_frame = CreateTestHardwareVideoFrame(
media::PIXEL_FORMAT_NV12, GL_TEXTURE_EXTERNAL_OES);
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::RGB_RESOURCE, resources.type);
EXPECT_EQ(1u, resources.resources.size());
EXPECT_EQ((GLenum)GL_TEXTURE_EXTERNAL_OES,
resources.resources[0].mailbox_holder.texture_target);
EXPECT_EQ(gfx::BufferFormat::YUV_420_BIPLANAR,
resources.resources[0].buffer_format);
video_frame = CreateTestYuvHardwareVideoFrame(media::PIXEL_FORMAT_NV12, 1,
GL_TEXTURE_RECTANGLE_ARB);
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::RGB_RESOURCE, resources.type);
EXPECT_EQ(1u, resources.resources.size());
EXPECT_EQ((GLenum)GL_TEXTURE_RECTANGLE_ARB,
resources.resources[0].mailbox_holder.texture_target);
EXPECT_EQ(gfx::BufferFormat::YUV_420_BIPLANAR,
resources.resources[0].buffer_format);
EXPECT_EQ(0, context3d_->TextureCreationCount());
}
TEST_F(VideoResourceUpdaterTest, CreateForHardwarePlanes_DualNV12) {
bool use_stream_video_draw_quad = false;
VideoResourceUpdater updater(context_provider_.get(),
resource_provider3d_.get(),
use_stream_video_draw_quad);
context3d_->ResetTextureCreationCount();
scoped_refptr<media::VideoFrame> video_frame =
CreateTestYuvHardwareVideoFrame(media::PIXEL_FORMAT_NV12, 2,
GL_TEXTURE_EXTERNAL_OES);
VideoFrameExternalResources resources =
updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_EQ(2u, resources.resources.size());
EXPECT_EQ(2u, resources.release_callbacks.size());
EXPECT_EQ(viz::kInvalidResourceId, resources.software_resource);
EXPECT_EQ((GLenum)GL_TEXTURE_EXTERNAL_OES,
resources.resources[0].mailbox_holder.texture_target);
// |updater| doesn't set |buffer_format| in this case.
EXPECT_EQ(gfx::BufferFormat::RGBA_8888, resources.resources[0].buffer_format);
video_frame = CreateTestYuvHardwareVideoFrame(media::PIXEL_FORMAT_NV12, 2,
GL_TEXTURE_RECTANGLE_ARB);
resources = updater.CreateExternalResourcesFromVideoFrame(video_frame);
EXPECT_EQ(VideoFrameExternalResources::YUV_RESOURCE, resources.type);
EXPECT_EQ(2u, resources.resources.size());
EXPECT_EQ((GLenum)GL_TEXTURE_RECTANGLE_ARB,
resources.resources[0].mailbox_holder.texture_target);
EXPECT_EQ(gfx::BufferFormat::RGBA_8888, resources.resources[0].buffer_format);
EXPECT_EQ(0, context3d_->TextureCreationCount());
}
} // namespace
} // namespace cc