blob: 0435cc69a587add3d375dca4b0878c6f4656ee1d [file] [log] [blame]
// Copyright 2021 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/fuchsia/common/sysmem_client.h"
#include <lib/sys/cpp/component_context.h>
#include <zircon/rights.h>
#include <algorithm>
#include "base/fuchsia/fuchsia_logging.h"
#include "base/fuchsia/process_context.h"
#include "base/functional/bind.h"
#include "base/process/process_handle.h"
#include "media/fuchsia/common/vmo_buffer.h"
namespace media {
SysmemCollectionClient::SysmemCollectionClient(
fuchsia::sysmem::Allocator* allocator,
fuchsia::sysmem::BufferCollectionTokenPtr collection_token)
: allocator_(allocator), collection_token_(std::move(collection_token)) {
DCHECK(allocator_);
DCHECK(collection_token_);
}
SysmemCollectionClient::~SysmemCollectionClient() {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
if (collection_)
collection_->Close();
}
void SysmemCollectionClient::Initialize(
fuchsia::sysmem::BufferCollectionConstraints constraints,
base::StringPiece name,
uint32_t name_priority) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
writable_ = (constraints.usage.cpu & fuchsia::sysmem::cpuUsageWrite) ==
fuchsia::sysmem::cpuUsageWrite;
allocator_->BindSharedCollection(std::move(collection_token_),
collection_.NewRequest());
collection_.set_error_handler(
fit::bind_member(this, &SysmemCollectionClient::OnError));
collection_->SetName(name_priority, std::string(name));
// We may need to send a Sync to ensure previously-started CreateSharedToken()
// calls can complete. The Sync completion is how we know that sysmem knows
// about the existence of the tokens created by the CreateSharedToken() calls,
// which is needed before we can send the token to a different participant.
//
// CreateSharedToken can complete as soon as this Sync is done.
if (!shared_token_ready_closures_.empty()) {
collection_->Sync(
fit::bind_member(this, &SysmemCollectionClient::OnSyncComplete));
}
collection_->SetConstraints(/*have_constraints=*/true,
std::move(constraints));
}
void SysmemCollectionClient::CreateSharedToken(
GetSharedTokenCB cb,
base::StringPiece debug_client_name,
uint64_t debug_client_id) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
DCHECK(collection_token_);
fuchsia::sysmem::BufferCollectionTokenPtr token;
collection_token_->Duplicate(ZX_RIGHT_SAME_RIGHTS, token.NewRequest());
if (!debug_client_name.empty()) {
token->SetDebugClientInfo(std::string(debug_client_name), debug_client_id);
}
shared_token_ready_closures_.push_back(
base::BindOnce(std::move(cb), std::move(token)));
}
void SysmemCollectionClient::AcquireBuffers(AcquireBuffersCB cb) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
DCHECK(!collection_token_);
if (!collection_) {
std::move(cb).Run({}, {});
return;
}
acquire_buffers_cb_ = std::move(cb);
collection_->WaitForBuffersAllocated(
fit::bind_member(this, &SysmemCollectionClient::OnBuffersAllocated));
}
void SysmemCollectionClient::OnSyncComplete() {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
std::vector<base::OnceClosure> sync_closures =
std::move(shared_token_ready_closures_);
for (auto& cb : sync_closures) {
std::move(cb).Run();
}
}
void SysmemCollectionClient::OnBuffersAllocated(
zx_status_t status,
fuchsia::sysmem::BufferCollectionInfo_2 buffer_collection_info) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
if (status != ZX_OK) {
ZX_LOG(ERROR, status) << "Failed to allocate sysmem buffers.";
OnError(status);
return;
}
if (acquire_buffers_cb_) {
auto buffers = VmoBuffer::CreateBuffersFromSysmemCollection(
&buffer_collection_info, writable_);
std::move(acquire_buffers_cb_)
.Run(std::move(buffers), buffer_collection_info.settings);
}
}
void SysmemCollectionClient::OnError(zx_status_t status) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
ZX_DLOG(ERROR, status) << "Connection to BufferCollection was disconnected.";
collection_.Unbind();
if (acquire_buffers_cb_)
std::move(acquire_buffers_cb_).Run({}, {});
}
SysmemAllocatorClient::SysmemAllocatorClient(base::StringPiece client_name) {
allocator_ = base::ComponentContextForProcess()
->svc()
->Connect<fuchsia::sysmem::Allocator>();
allocator_->SetDebugClientInfo(std::string(client_name),
base::GetCurrentProcId());
allocator_.set_error_handler([](zx_status_t status) {
// Just log a warning. We will handle BufferCollection the failure when
// trying to create a new BufferCollection.
ZX_DLOG(WARNING, status)
<< "The fuchsia.sysmem.Allocator channel was disconnected.";
});
}
SysmemAllocatorClient::~SysmemAllocatorClient() = default;
fuchsia::sysmem::BufferCollectionTokenPtr
SysmemAllocatorClient::CreateNewToken() {
fuchsia::sysmem::BufferCollectionTokenPtr collection_token;
allocator_->AllocateSharedCollection(collection_token.NewRequest());
return collection_token;
}
std::unique_ptr<SysmemCollectionClient>
SysmemAllocatorClient::AllocateNewCollection() {
return std::make_unique<SysmemCollectionClient>(allocator_.get(),
CreateNewToken());
}
std::unique_ptr<SysmemCollectionClient>
SysmemAllocatorClient::BindSharedCollection(
fuchsia::sysmem::BufferCollectionTokenPtr token) {
return std::make_unique<SysmemCollectionClient>(allocator_.get(),
std::move(token));
}
} // namespace media