blob: 28cd7d459f5f63a61f9e929f74ae565701a24520 [file] [log] [blame]
/*
* Copyright 2023 Google LLC
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/gpu/graphite/Image_Base_Graphite.h"
#include "include/core/SkColorSpace.h"
#include "include/gpu/graphite/Image.h"
#include "include/gpu/graphite/Recorder.h"
#include "include/gpu/graphite/Surface.h"
#include "src/gpu/graphite/Device.h"
#include "src/gpu/graphite/DrawContext.h"
#include "src/gpu/graphite/Image_Graphite.h"
#include "src/gpu/graphite/Image_YUVA_Graphite.h"
#include "src/gpu/graphite/Log.h"
#include "src/gpu/graphite/RecorderPriv.h"
#include "src/gpu/graphite/Surface_Graphite.h"
#include "src/gpu/graphite/TextureUtils.h"
#include "src/gpu/graphite/task/Task.h"
namespace skgpu::graphite {
Image_Base::Image_Base(const SkImageInfo& info, uint32_t uniqueID)
: SkImage_Base(info, uniqueID) {}
Image_Base::~Image_Base() = default;
void Image_Base::linkDevices(const Image_Base* other) {
SkASSERT(other);
SkAutoSpinlock lock{other->fDeviceLinkLock};
for (const auto& device : other->fLinkedDevices) {
this->linkDevice(device);
}
}
void Image_Base::linkDevice(sk_sp<Device> device) {
// Technically this lock isn't needed since this is only called before the Image is returned to
// user code that could expose it to multiple threads. But this quiets threading warnings and
// should be uncontested.
SkAutoSpinlock lock{fDeviceLinkLock};
fLinkedDevices.push_back(std::move(device));
}
void Image_Base::notifyInUse(Recorder* recorder, DrawContext* drawContext) const {
SkASSERT(recorder);
// The ref counts stored on each linked device are thread safe, but the Image's sk_sp's that
// track the refs its responsible for are *not* thread safe. Use a spin lock since the majority
// of device-linked images will be used only on the Recorder's thread. Since it should be
// uncontended, the empty check is also done inside the lock vs. a double-checked locking
// pattern that is non-trivial to ensure correctness in C++.
SkAutoSpinlock lock{fDeviceLinkLock};
if (!fLinkedDevices.empty()) {
int emptyCount = 0;
for (sk_sp<Device>& device : fLinkedDevices) {
if (!device || device->notifyInUse(recorder, drawContext)) {
// Already unlinked or notifyInUse() signals the device doesn't need to be linked
// anymore. reset() is a no-op if device is already holding null.
device.reset();
emptyCount++;
}
}
if (emptyCount == fLinkedDevices.size()) {
fLinkedDevices.clear();
}
}
}
bool Image_Base::isDynamic() const {
SkAutoSpinlock lock{fDeviceLinkLock};
int emptyCount = 0;
if (!fLinkedDevices.empty()) {
for (sk_sp<Device>& device : fLinkedDevices) {
if (!device || !device->recorder() || device->unique()) {
device.reset();
emptyCount++;
}
}
if (emptyCount == fLinkedDevices.size()) {
fLinkedDevices.clear();
emptyCount = 0;
}
}
return emptyCount > 0;
}
// For now, CopyAsDraw is called with a nullptr for the drawContext, which causes the draw task
// to be pushed onto the root task list. However, this could be given a drawContext in the future.
sk_sp<Image> Image_Base::copyImage(Recorder* recorder,
const SkIRect& subset,
Budgeted budgeted,
Mipmapped mipmapped,
SkBackingFit backingFit,
std::string_view label) const {
return CopyAsDraw(recorder,
/*drawContext=*/nullptr, this, subset, this->imageInfo().colorInfo(),
budgeted, mipmapped, backingFit, std::move(label));
}
namespace {
TextureProxy* get_base_proxy_for_label(const Image_Base* baseImage) {
if (baseImage->type() == SkImage_Base::Type::kGraphite) {
const Image* img = static_cast<const Image*>(baseImage);
return img->textureProxyView().proxy();
}
SkASSERT(baseImage->type() == SkImage_Base::Type::kGraphiteYUVA);
// We will end up flattening to RGBA for a YUVA image when we get a subset. We just grab
// the label off of the first channel's proxy and use that to be the stand in label.
const Image_YUVA* img = static_cast<const Image_YUVA*>(baseImage);
return img->proxyView(0).proxy();
}
} // anonymous namespace
sk_sp<SkImage> Image_Base::onMakeSubset(SkRecorder* recorder,
const SkIRect& subset,
RequiredProperties requiredProps) const {
auto gRecorder = AsGraphiteRecorder(recorder);
if (!gRecorder) {
return nullptr;
}
// optimization : return self if the subset == our bounds and requirements met and the image's
// texture is immutable
if (this->bounds() == subset &&
(!requiredProps.fMipmapped || this->hasMipmaps()) &&
!this->isDynamic()) {
return sk_ref_sp(this);
}
TextureProxy* proxy = get_base_proxy_for_label(this);
SkASSERT(proxy);
std::string label = proxy->label();
if (label.empty()) {
label = "ImageSubsetTexture";
} else {
label += "_Subset";
}
// The copied image is not considered budgeted because this is a client-invoked API and they
// will own the image.
return this->copyImage(gRecorder,
subset,
Budgeted::kNo,
requiredProps.fMipmapped ? Mipmapped::kYes : Mipmapped::kNo,
SkBackingFit::kExact,
label);
}
sk_sp<SkSurface> Image_Base::onMakeSurface(SkRecorder* recorder, const SkImageInfo& info) const {
auto gRecorder = AsGraphiteRecorder(recorder);
if (!gRecorder) {
return nullptr;
}
return SkSurfaces::RenderTarget(gRecorder, info);
}
sk_sp<SkImage> Image_Base::makeColorTypeAndColorSpace(SkRecorder* recorder,
SkColorType targetCT,
sk_sp<SkColorSpace> targetCS,
RequiredProperties requiredProps) const {
auto gRecorder = AsGraphiteRecorder(recorder);
if (!gRecorder) {
return nullptr;
}
SkColorInfo dstColorInfo{targetCT, this->alphaType(), std::move(targetCS)};
// optimization : return self if there's no color type/space change and the image's texture
// is immutable
if (this->imageInfo().colorInfo() == dstColorInfo && !this->isDynamic()) {
return sk_ref_sp(this);
}
TextureProxy* proxy = get_base_proxy_for_label(this);
SkASSERT(proxy);
std::string label = proxy->label();
if (label.empty()) {
label = "ImageMakeCTandCSTexture";
} else {
label += "_CTandCSConversion";
}
// Use CopyAsDraw directly to perform the color space changes. The copied image is not
// considered budgeted because this is a client-invoked API and they will own the image.
return CopyAsDraw(gRecorder,
/*drawContext=*/nullptr,
this,
this->bounds(),
dstColorInfo,
Budgeted::kNo,
requiredProps.fMipmapped ? Mipmapped::kYes : Mipmapped::kNo,
SkBackingFit::kExact,
label);
}
// Ganesh APIs are no-ops
void Image_Base::onAsyncRescaleAndReadPixels(const SkImageInfo& info,
SkIRect srcRect,
RescaleGamma rescaleGamma,
RescaleMode rescaleMode,
ReadPixelsCallback callback,
ReadPixelsContext context) const {
SKGPU_LOG_W("Cannot use Ganesh async API with Graphite-backed image, use API on Context");
callback(context, nullptr);
}
void Image_Base::onAsyncRescaleAndReadPixelsYUV420(SkYUVColorSpace yuvColorSpace,
bool readAlpha,
sk_sp<SkColorSpace> dstColorSpace,
const SkIRect srcRect,
const SkISize dstSize,
RescaleGamma rescaleGamma,
RescaleMode rescaleMode,
ReadPixelsCallback callback,
ReadPixelsContext context) const {
SKGPU_LOG_W("Cannot use Ganesh async API with Graphite-backed image, use API on Context");
callback(context, nullptr);
}
} // namespace skgpu::graphite