blob: 9bc5923c3be8d5391e3ac31796f2680d7e9ad0c1 [file] [log] [blame]
/*
* Copyright 2022 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/SpecialImage_Graphite.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkColorSpace.h"
#include "src/core/SkSpecialImage.h"
#include "src/gpu/graphite/Device.h"
#include "src/gpu/graphite/Image_Graphite.h"
#include "src/gpu/graphite/Surface_Graphite.h"
#include "src/gpu/graphite/TextureUtils.h"
namespace skgpu::graphite {
class SpecialImage final : public SkSpecialImage {
public:
SpecialImage(const SkIRect& subset, sk_sp<SkImage> image, const SkSurfaceProps& props)
: SkSpecialImage(subset, image->uniqueID(), image->imageInfo().colorInfo(), props)
, fImage(std::move(image)) {
SkASSERT(as_IB(fImage)->isGraphiteBacked());
}
size_t getSize() const override {
return fImage->textureSize();
}
bool isGraphiteBacked() const override { return true; }
SkISize backingStoreDimensions() const override {
return fImage->dimensions();
}
sk_sp<SkSpecialImage> onMakeBackingStoreSubset(const SkIRect& subset) const override {
SkASSERT(fImage->bounds().contains(subset));
return sk_make_sp<skgpu::graphite::SpecialImage>(subset, fImage, this->props());
}
sk_sp<SkImage> asImage() const override { return fImage; }
private:
// TODO(b/299474380): SkSpecialImage is intended to go away in favor of just using SkImages
// and tracking the intended srcRect explicitly in skif::FilterResult. Since Graphite tracks
// device-linked textures via Images, the graphite special image just wraps an image.
sk_sp<SkImage> fImage;
};
} // namespace skgpu::graphite
namespace SkSpecialImages {
sk_sp<SkSpecialImage> MakeGraphite(skgpu::graphite::Recorder* recorder,
const SkIRect& subset,
sk_sp<SkImage> image,
const SkSurfaceProps& props) {
// 'recorder' can be null if we're wrapping a graphite-backed image since there's no work that
// needs to be added. This can happen when snapping a special image from a Device that's been
// marked as immutable and abandoned its recorder.
if (!image || subset.isEmpty()) {
return nullptr;
}
SkASSERT(image->bounds().contains(subset));
// Use the Recorder's client ImageProvider to convert to a graphite-backed image when
// possible, but this does not necessarily mean the provider will produce a valid image.
if (!as_IB(image)->isGraphiteBacked()) {
if (!recorder) {
return nullptr;
}
auto [graphiteImage, _] =
skgpu::graphite::GetGraphiteBacked(recorder, image.get(), {});
if (!graphiteImage) {
return nullptr;
}
image = graphiteImage;
}
return sk_make_sp<skgpu::graphite::SpecialImage>(subset, std::move(image), props);
}
} // namespace SkSpecialImages