blob: ca8736c2690862abde9b668f84b77e7983e03fca [file] [log] [blame]
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "gpu/command_buffer/client/test_shared_image_interface.h"
#include <GLES2/gl2.h>
#include <GLES2/gl2extchromium.h>
#include <utility>
#if BUILDFLAG(IS_FUCHSIA)
#include <fuchsia/sysmem2/cpp/fidl.h>
#include <lib/sys/cpp/component_context.h>
#endif
#include "base/check.h"
#include "base/notreached.h"
#include "build/build_config.h"
#include "components/viz/common/resources/shared_image_format_utils.h"
#include "gpu/command_buffer/client/client_shared_image.h"
#include "gpu/command_buffer/common/shared_image_capabilities.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/buffer_format_util.h"
#include "ui/gfx/gpu_fence.h"
#include "ui/gfx/gpu_memory_buffer.h"
#if BUILDFLAG(IS_FUCHSIA)
#include "base/fuchsia/fuchsia_logging.h"
#include "base/fuchsia/koid.h"
#include "base/fuchsia/process_context.h"
#endif
namespace gpu {
namespace {
gfx::GpuMemoryBufferType GetNativeBufferType() {
#if BUILDFLAG(IS_APPLE)
return gfx::IO_SURFACE_BUFFER;
#elif BUILDFLAG(IS_ANDROID)
return gfx::ANDROID_HARDWARE_BUFFER;
#elif BUILDFLAG(IS_WIN)
return gfx::DXGI_SHARED_HANDLE;
#else
// Ozone
return gfx::NATIVE_PIXMAP;
#endif
}
// Creates a shared memory region and returns a handle to it.
gfx::GpuMemoryBufferHandle CreateGMBHandle(
const gfx::BufferFormat& buffer_format,
const gfx::Size& size,
gfx::BufferUsage buffer_usage) {
static int last_handle_id = 0;
size_t buffer_size = 0u;
CHECK(
gfx::BufferSizeForBufferFormatChecked(size, buffer_format, &buffer_size));
auto shared_memory_region =
base::UnsafeSharedMemoryRegion::Create(buffer_size);
CHECK(shared_memory_region.IsValid());
gfx::GpuMemoryBufferHandle handle(std::move(shared_memory_region));
handle.id = gfx::GpuMemoryBufferId(last_handle_id++);
handle.offset = 0;
handle.stride = static_cast<uint32_t>(
gfx::RowSizeForBufferFormat(size.width(), buffer_format, 0));
return handle;
}
} // namespace
#if BUILDFLAG(IS_FUCHSIA)
class TestBufferCollection {
public:
TestBufferCollection(zx::eventpair handle, zx::channel collection_token)
: handle_(std::move(handle)) {
sysmem_allocator_ = base::ComponentContextForProcess()
->svc()
->Connect<fuchsia::sysmem2::Allocator>();
sysmem_allocator_.set_error_handler([](zx_status_t status) {
ZX_LOG(FATAL, status)
<< "The fuchsia.sysmem.Allocator channel was terminated.";
});
fuchsia::sysmem2::AllocatorSetDebugClientInfoRequest set_debug_request;
set_debug_request.set_name("CrTestBufferCollection");
set_debug_request.set_id(base::GetCurrentProcId());
sysmem_allocator_->SetDebugClientInfo(std::move(set_debug_request));
fuchsia::sysmem2::AllocatorBindSharedCollectionRequest bind_shared_request;
bind_shared_request.set_token(fidl::InterfaceHandle<fuchsia::sysmem2::BufferCollectionToken>(
std::move(collection_token)));
bind_shared_request.set_buffer_collection_request(buffers_collection_.NewRequest());
sysmem_allocator_->BindSharedCollection(std::move(bind_shared_request));
fuchsia::sysmem2::BufferCollectionSetConstraintsRequest set_constraints_request;
auto& buffer_constraints = *set_constraints_request.mutable_constraints();
buffer_constraints.mutable_usage()->set_cpu(fuchsia::sysmem2::CPU_USAGE_READ);
zx_status_t status = buffers_collection_->SetConstraints(std::move(set_constraints_request));
ZX_CHECK(status == ZX_OK, status) << "BufferCollection::SetConstraints()";
}
TestBufferCollection(const TestBufferCollection&) = delete;
TestBufferCollection& operator=(const TestBufferCollection&) = delete;
~TestBufferCollection() { buffers_collection_->Release(); }
size_t GetNumBuffers() {
if (!buffer_collection_info_) {
fuchsia::sysmem2::BufferCollection_WaitForAllBuffersAllocated_Result wait_result;
zx_status_t status =
buffers_collection_->WaitForAllBuffersAllocated(&wait_result);
if (status != ZX_OK) {
ZX_LOG(FATAL, status) <<
"BufferCollection::WaitForAllBuffersAllocated() (status)";
} else if (wait_result.is_framework_err()) {
LOG(FATAL) <<
"BufferCollection::WaitForAllBuffersAllocated (framework_err): " <<
fidl::ToUnderlying(wait_result.framework_err());
} else if (!wait_result.is_response()) {
LOG(FATAL) << "BufferCollection::WaitForAllBuffersAllocated (err)" <<
static_cast<uint32_t>(wait_result.err());
}
auto info = std::move(*wait_result.response().mutable_buffer_collection_info());
buffer_collection_info_ = std::move(info);
}
return buffer_collection_info_->buffers().size();
}
private:
zx::eventpair handle_;
fuchsia::sysmem2::AllocatorPtr sysmem_allocator_;
fuchsia::sysmem2::BufferCollectionSyncPtr buffers_collection_;
std::optional<fuchsia::sysmem2::BufferCollectionInfo>
buffer_collection_info_;
};
#endif
TestSharedImageInterface::TestSharedImageInterface() {
InitializeSharedImageCapabilities();
}
TestSharedImageInterface::~TestSharedImageInterface() = default;
scoped_refptr<ClientSharedImage> TestSharedImageInterface::CreateSharedImage(
const SharedImageInfo& si_info,
SurfaceHandle surface_handle,
std::optional<SharedImagePoolId> pool_id) {
SyncToken sync_token = GenUnverifiedSyncToken();
base::AutoLock locked(lock_);
auto mailbox = Mailbox::Generate();
shared_images_.insert(mailbox);
most_recent_size_ = si_info.meta.size;
auto gmb_handle_type = emulate_client_provided_native_buffer_
? GetNativeBufferType()
: gfx::EMPTY_BUFFER;
return base::MakeRefCounted<ClientSharedImage>(
mailbox, si_info.meta, sync_token, holder_, gmb_handle_type);
}
scoped_refptr<ClientSharedImage>
TestSharedImageInterface::CreateSharedImage(
const SharedImageInfo& si_info,
base::span<const uint8_t> pixel_data) {
SyncToken sync_token = GenUnverifiedSyncToken();
base::AutoLock locked(lock_);
auto mailbox = Mailbox::Generate();
shared_images_.insert(mailbox);
return base::MakeRefCounted<ClientSharedImage>(
mailbox, si_info.meta, sync_token, holder_, gfx::EMPTY_BUFFER);
}
scoped_refptr<ClientSharedImage> TestSharedImageInterface::CreateSharedImage(
const SharedImageInfo& si_info,
SurfaceHandle surface_handle,
gfx::BufferUsage buffer_usage,
std::optional<SharedImagePoolId> pool_id) {
DoCreateSharedImage(si_info.meta.size, si_info.meta.format, surface_handle,
buffer_usage);
if (fail_shared_image_creation_with_buffer_usage_) {
return nullptr;
}
SyncToken sync_token = GenUnverifiedSyncToken();
// Create a ClientSharedImage with a GMB.
base::AutoLock locked(lock_);
auto mailbox = Mailbox::Generate();
shared_images_.insert(mailbox);
most_recent_size_ = si_info.meta.size;
auto buffer_format =
viz::SharedImageFormatToBufferFormatRestrictedUtils::ToBufferFormat(
si_info.meta.format);
if (test_gmb_manager_) {
auto gpu_memory_buffer = test_gmb_manager_->CreateGpuMemoryBuffer(
si_info.meta.size, buffer_format, buffer_usage, surface_handle,
nullptr);
// Since the |gpu_memory_buffer| here is always a shared memory, clear the
// external sampler prefs if it is already set by client.
// https://issues.chromium.org/339546249.
SharedImageInfo si_info_copy = si_info;
if (si_info_copy.meta.format.PrefersExternalSampler()) {
si_info_copy.meta.format.ClearPrefersExternalSampler();
}
return ClientSharedImage::CreateForTesting(
mailbox, si_info_copy.meta, sync_token, std::move(gpu_memory_buffer),
buffer_usage, holder_);
}
auto gmb_handle =
CreateGMBHandle(buffer_format, si_info.meta.size, buffer_usage);
return base::MakeRefCounted<ClientSharedImage>(
mailbox, si_info.meta, sync_token,
GpuMemoryBufferHandleInfo(std::move(gmb_handle), si_info.meta.format,
si_info.meta.size, buffer_usage),
holder_);
}
scoped_refptr<ClientSharedImage>
TestSharedImageInterface::CreateSharedImage(
const SharedImageInfo& si_info,
SurfaceHandle surface_handle,
gfx::BufferUsage buffer_usage,
gfx::GpuMemoryBufferHandle buffer_handle) {
SyncToken sync_token = GenUnverifiedSyncToken();
base::AutoLock locked(lock_);
auto mailbox = Mailbox::Generate();
shared_images_.insert(mailbox);
most_recent_size_ = si_info.meta.size;
auto buffer_format =
viz::SharedImageFormatToBufferFormatRestrictedUtils::ToBufferFormat(
si_info.meta.format);
if (test_gmb_manager_) {
auto gpu_memory_buffer = test_gmb_manager_->CreateGpuMemoryBuffer(
si_info.meta.size, buffer_format, buffer_usage, surface_handle,
nullptr);
// Since the |gpu_memory_buffer| here is always a shared memory, clear the
// external sampler prefs if it is already set by client.
// https://issues.chromium.org/339546249.
SharedImageInfo si_info_copy = si_info;
if (si_info_copy.meta.format.PrefersExternalSampler()) {
si_info_copy.meta.format.ClearPrefersExternalSampler();
}
return ClientSharedImage::CreateForTesting(
mailbox, si_info_copy.meta, sync_token, std::move(gpu_memory_buffer),
buffer_usage, holder_);
}
return base::MakeRefCounted<ClientSharedImage>(
mailbox, si_info.meta, sync_token,
GpuMemoryBufferHandleInfo(std::move(buffer_handle),
si_info.meta.format, si_info.meta.size,
buffer_usage),
holder_);
}
scoped_refptr<ClientSharedImage>
TestSharedImageInterface::CreateSharedImage(
const SharedImageInfo& si_info,
gfx::GpuMemoryBufferHandle buffer_handle) {
SyncToken sync_token = GenUnverifiedSyncToken();
base::AutoLock locked(lock_);
#if BUILDFLAG(IS_FUCHSIA)
if (buffer_handle.type == gfx::GpuMemoryBufferType::NATIVE_PIXMAP) {
zx_koid_t id =
base::GetRelatedKoid(
buffer_handle.native_pixmap_handle().buffer_collection_handle)
.value();
auto collection_it = sysmem_buffer_collections_.find(id);
// NOTE: Not all unittests invoke RegisterSysmemBufferCollection(), but
// the below CHECK should hold for those that do.
if (collection_it != sysmem_buffer_collections_.end()) {
CHECK_LT(buffer_handle.native_pixmap_handle().buffer_index,
collection_it->second->GetNumBuffers());
}
}
#endif
auto mailbox = Mailbox::Generate();
shared_images_.insert(mailbox);
most_recent_size_ = si_info.meta.size;
return base::MakeRefCounted<ClientSharedImage>(
mailbox, si_info.meta, sync_token, holder_, buffer_handle.type);
}
scoped_refptr<ClientSharedImage>
TestSharedImageInterface::CreateSharedImageForSoftwareCompositor(
const SharedImageInfo& si_info) {
base::WritableSharedMemoryMapping mapping;
gfx::GpuMemoryBufferHandle handle;
CreateSharedMemoryRegionFromSIInfo(si_info, mapping, handle);
auto mailbox = Mailbox::Generate();
shared_images_.insert(mailbox);
most_recent_size_ = si_info.meta.size;
return base::MakeRefCounted<ClientSharedImage>(mailbox, si_info.meta,
GenUnverifiedSyncToken(),
holder_, std::move(mapping));
}
scoped_refptr<ClientSharedImage>
TestSharedImageInterface::CreateSharedImageForMLTensor(
std::string debug_label,
viz::SharedImageFormat format,
const gfx::Size& size,
gpu::SharedImageUsageSet usage) {
NOTREACHED();
}
void TestSharedImageInterface::UpdateSharedImage(
const SyncToken& sync_token,
const Mailbox& mailbox) {
base::AutoLock locked(lock_);
DCHECK(shared_images_.find(mailbox) != shared_images_.end());
num_update_shared_image_no_fence_calls_++;
}
void TestSharedImageInterface::UpdateSharedImage(
const SyncToken& sync_token,
std::unique_ptr<gfx::GpuFence> acquire_fence,
const Mailbox& mailbox) {
base::AutoLock locked(lock_);
DCHECK(shared_images_.find(mailbox) != shared_images_.end());
}
scoped_refptr<ClientSharedImage> TestSharedImageInterface::ImportSharedImage(
ExportedSharedImage exported_shared_image) {
shared_images_.insert(exported_shared_image.mailbox_);
return base::WrapRefCounted<ClientSharedImage>(
new ClientSharedImage(std::move(exported_shared_image), holder_));
}
void TestSharedImageInterface::DestroySharedImage(
const SyncToken& sync_token,
const Mailbox& mailbox) {
base::AutoLock locked(lock_);
shared_images_.erase(mailbox);
most_recent_destroy_token_ = sync_token;
if (test_client_) {
test_client_->DidDestroySharedImage();
}
}
void TestSharedImageInterface::DestroySharedImage(
const SyncToken& sync_token,
scoped_refptr<ClientSharedImage> client_shared_image) {
CHECK(client_shared_image->HasOneRef());
client_shared_image->UpdateDestructionSyncToken(sync_token);
}
SharedImageInterface::SwapChainSharedImages
TestSharedImageInterface::CreateSwapChain(viz::SharedImageFormat format,
const gfx::Size& size,
const gfx::ColorSpace& color_space,
GrSurfaceOrigin surface_origin,
SkAlphaType alpha_type,
gpu::SharedImageUsageSet usage) {
auto front_buffer = Mailbox::Generate();
auto back_buffer = Mailbox::Generate();
SyncToken sync_token = GenUnverifiedSyncToken();
shared_images_.insert(front_buffer);
shared_images_.insert(back_buffer);
return {base::MakeRefCounted<ClientSharedImage>(
front_buffer,
SharedImageMetadata(format, size, color_space,
surface_origin, alpha_type, usage),
sync_token, holder_, gfx::EMPTY_BUFFER),
base::MakeRefCounted<ClientSharedImage>(
back_buffer,
SharedImageMetadata(format, size, color_space,
surface_origin, alpha_type, usage),
sync_token, holder_, gfx::EMPTY_BUFFER)};
}
void TestSharedImageInterface::PresentSwapChain(
const SyncToken& sync_token,
const Mailbox& mailbox) {}
#if BUILDFLAG(IS_FUCHSIA)
void TestSharedImageInterface::RegisterSysmemBufferCollection(
zx::eventpair service_handle,
zx::channel sysmem_token,
const viz::SharedImageFormat& format,
gfx::BufferUsage usage,
bool register_with_image_pipe) {
EXPECT_EQ(format, viz::MultiPlaneFormat::kNV12);
EXPECT_EQ(usage, gfx::BufferUsage::GPU_READ);
zx_koid_t id = base::GetKoid(service_handle).value();
std::unique_ptr<TestBufferCollection>& collection =
sysmem_buffer_collections_[id];
EXPECT_FALSE(collection);
collection = std::make_unique<TestBufferCollection>(std::move(service_handle),
std::move(sysmem_token));
}
#endif // BUILDFLAG(IS_FUCHSIA)
SyncToken TestSharedImageInterface::GenVerifiedSyncToken() {
base::AutoLock locked(lock_);
most_recent_generated_token_ =
SyncToken(CommandBufferNamespace::GPU_IO,
CommandBufferId(), ++release_id_);
VerifySyncToken(most_recent_generated_token_);
return most_recent_generated_token_;
}
SyncToken TestSharedImageInterface::GenUnverifiedSyncToken() {
base::AutoLock locked(lock_);
most_recent_generated_token_ =
SyncToken(CommandBufferNamespace::GPU_IO,
CommandBufferId(), ++release_id_);
return most_recent_generated_token_;
}
void TestSharedImageInterface::VerifySyncToken(SyncToken& sync_token) {
sync_token.SetVerifyFlush();
}
void TestSharedImageInterface::WaitSyncToken(const SyncToken& sync_token) {
NOTREACHED();
}
scoped_refptr<ClientSharedImage>
TestSharedImageInterface::CreateSharedImageWithMapCallbackController(
const SharedImageInfo& si_info,
gfx::BufferUsage buffer_usage,
bool premapped,
FakeGpuMemoryBuffer::MapCallbackController* controller) {
CHECK(controller);
// Create a FakeGpuMemoryBuffer.
auto buffer_format =
viz::SharedImageFormatToBufferFormatRestrictedUtils::ToBufferFormat(
si_info.meta.format);
auto fake_gmb = std::make_unique<FakeGpuMemoryBuffer>(
si_info.meta.size, buffer_format, premapped, controller);
Mailbox mailbox;
// Create a ClientSharedImage with a FakeGpuMemoryBuffer.
{
base::AutoLock locked(lock_);
mailbox = Mailbox::Generate();
shared_images_.insert(mailbox);
}
auto image = ClientSharedImage::CreateForTesting(
mailbox, si_info.meta, GenUnverifiedSyncToken(), std::move(fake_gmb),
buffer_usage, holder_);
return image;
}
bool TestSharedImageInterface::CheckSharedImageExists(
const Mailbox& mailbox) const {
base::AutoLock locked(lock_);
return shared_images_.contains(mailbox);
}
const SharedImageCapabilities&
TestSharedImageInterface::GetCapabilities() {
return shared_image_capabilities_;
}
void TestSharedImageInterface::SetCapabilities(
const SharedImageCapabilities& caps) {
shared_image_capabilities_ = caps;
InitializeSharedImageCapabilities();
}
void TestSharedImageInterface::InitializeSharedImageCapabilities() {
#if BUILDFLAG(IS_MAC)
// Initialize `texture_target_for_io_surfaces` to a value that is valid for
// ClientSharedImage to use, as unittests broadly create and use
// SharedImageCapabilities instances without initializing this field. The
// specific value is chosen to match the historical default value that was
// used when this state was accessed via a global variable.
if (!shared_image_capabilities_.texture_target_for_io_surfaces) {
shared_image_capabilities_.texture_target_for_io_surfaces =
GL_TEXTURE_RECTANGLE_ARB;
}
#endif
}
} // namespace gpu