1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
|
// Copyright 2020 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/service/shared_image/android_image_backing.h"
#include "base/logging.h"
#include "base/posix/eintr_wrapper.h"
#include "ui/gl/gl_utils.h"
namespace gpu {
AndroidImageBacking::AndroidImageBacking(const Mailbox& mailbox,
viz::SharedImageFormat format,
const gfx::Size& size,
const gfx::ColorSpace& color_space,
GrSurfaceOrigin surface_origin,
SkAlphaType alpha_type,
gpu::SharedImageUsageSet usage,
std::string debug_label,
size_t estimated_size,
bool is_thread_safe,
base::ScopedFD initial_upload_fd)
: ClearTrackingSharedImageBacking(mailbox,
format,
size,
color_space,
surface_origin,
alpha_type,
usage,
std::move(debug_label),
estimated_size,
is_thread_safe),
write_sync_fd_(std::move(initial_upload_fd)) {}
AndroidImageBacking::~AndroidImageBacking() = default;
bool AndroidImageBacking::BeginWrite(base::ScopedFD* fd_to_wait_on) {
AutoLock auto_lock(this);
if (is_writing_) {
LOG(ERROR)
<< "BeginWrite should only be called when there are no other writers";
return false;
}
if (!allow_concurrent_read_write() &&
(!active_readers_.empty() || is_overlay_accessing_)) {
LOG(ERROR)
<< "BeginWrite should only be called when there are no other readers";
return false;
}
is_writing_ = true;
if (allow_concurrent_read_write()) {
if (write_sync_fd_.is_valid()) {
(*fd_to_wait_on) =
base::ScopedFD(HANDLE_EINTR(dup(write_sync_fd_.get())));
} else {
fd_to_wait_on->reset();
}
} else {
(*fd_to_wait_on) =
gl::MergeFDs(std::move(read_sync_fd_), std::move(write_sync_fd_));
}
return true;
}
void AndroidImageBacking::EndWrite(base::ScopedFD end_write_fd) {
AutoLock auto_lock(this);
if (!is_writing_) {
LOG(ERROR) << "Attempt to end write to a SharedImageBacking without a "
"successful begin write";
return;
}
is_writing_ = false;
write_sync_fd_ = std::move(end_write_fd);
}
bool AndroidImageBacking::BeginRead(const SharedImageRepresentation* reader,
base::ScopedFD* fd_to_wait_on) {
AutoLock auto_lock(this);
if (!allow_concurrent_read_write() && is_writing_) {
LOG(ERROR) << "BeginRead should only be called when there are no writers";
return false;
}
if (active_readers_.contains(reader)) {
LOG(ERROR) << "BeginRead was called twice on the same representation";
return false;
}
active_readers_.insert(reader);
if (write_sync_fd_.is_valid()) {
(*fd_to_wait_on) = base::ScopedFD(HANDLE_EINTR(dup(write_sync_fd_.get())));
} else {
(*fd_to_wait_on) = base::ScopedFD{};
}
return true;
}
void AndroidImageBacking::EndRead(const SharedImageRepresentation* reader,
base::ScopedFD end_read_fd) {
AutoLock auto_lock(this);
if (!active_readers_.contains(reader)) {
LOG(ERROR) << "Attempt to end read to a SharedImageBacking without a "
"successful begin read";
return;
}
active_readers_.erase(reader);
if (!allow_concurrent_read_write()) {
read_sync_fd_ =
gl::MergeFDs(std::move(read_sync_fd_), std::move(end_read_fd));
}
}
base::ScopedFD AndroidImageBacking::TakeReadFence() {
AutoLock auto_lock(this);
return std::move(read_sync_fd_);
}
} // namespace gpu
|