blob: 802bbcd6fc4d49768e1fb56822d6d38463256647 [file] [log] [blame]
Greg Daniel164a9f02016-02-22 09:56:40 -05001/*
2 * Copyright 2015 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "GrVkTexture.h"
Robert Phillipsf95b1752017-08-31 08:56:07 -04009
10#include "GrTexturePriv.h"
Greg Daniel164a9f02016-02-22 09:56:40 -050011#include "GrVkGpu.h"
12#include "GrVkImageView.h"
egdaniel50ead532016-07-13 14:23:26 -070013#include "GrVkTextureRenderTarget.h"
Greg Daniel164a9f02016-02-22 09:56:40 -050014#include "GrVkUtil.h"
15
jvanverthfd359ca2016-03-18 11:57:24 -070016#include "vk/GrVkTypes.h"
17
Greg Daniel164a9f02016-02-22 09:56:40 -050018#define VK_CALL(GPU, X) GR_VK_CALL(GPU->vkInterface(), X)
19
20// Because this class is virtually derived from GrSurface we must explicitly call its constructor.
21GrVkTexture::GrVkTexture(GrVkGpu* gpu,
kkinnunen2e6055b2016-04-22 01:48:29 -070022 SkBudgeted budgeted,
Greg Daniel164a9f02016-02-22 09:56:40 -050023 const GrSurfaceDesc& desc,
egdanielb2df0c22016-05-13 11:30:37 -070024 const GrVkImageInfo& info,
Greg Daniel52e16d92018-04-10 09:34:07 -040025 sk_sp<GrVkImageLayout> layout,
Greg Daniel834f1202017-10-09 15:06:20 -040026 const GrVkImageView* view,
Greg Daniel0fc4d2d2017-10-12 11:23:36 -040027 GrMipMapsStatus mipMapsStatus)
Brian Salomon60dd8c72018-07-30 10:24:13 -040028 : GrSurface(gpu, desc)
29 , GrVkImage(info, std::move(layout), GrBackendObjectOwnership::kOwned)
Brian Salomone632dfc2018-08-01 13:01:16 -040030 , INHERITED(gpu, desc, GrTextureType::k2D, mipMapsStatus)
Brian Salomon60dd8c72018-07-30 10:24:13 -040031 , fTextureView(view) {
Greg Daniel0fc4d2d2017-10-12 11:23:36 -040032 SkASSERT((GrMipMapsStatus::kNotAllocated == mipMapsStatus) == (1 == info.fLevelCount));
kkinnunen2e6055b2016-04-22 01:48:29 -070033 this->registerWithCache(budgeted);
Jim Van Verth1676cb92019-01-15 13:24:45 -050034 if (GrPixelConfigIsCompressed(desc.fConfig)) {
35 this->setReadOnly();
36 }
kkinnunen2e6055b2016-04-22 01:48:29 -070037}
38
39GrVkTexture::GrVkTexture(GrVkGpu* gpu,
40 Wrapped,
41 const GrSurfaceDesc& desc,
egdanielb2df0c22016-05-13 11:30:37 -070042 const GrVkImageInfo& info,
Greg Daniel52e16d92018-04-10 09:34:07 -040043 sk_sp<GrVkImageLayout> layout,
egdanielb2df0c22016-05-13 11:30:37 -070044 const GrVkImageView* view,
Greg Daniel0fc4d2d2017-10-12 11:23:36 -040045 GrMipMapsStatus mipMapsStatus,
Greg Daniel2268ad22018-11-15 09:27:38 -050046 GrBackendObjectOwnership ownership,
Brian Salomonc67c31c2018-12-06 10:00:03 -050047 GrIOType ioType,
Greg Daniel2268ad22018-11-15 09:27:38 -050048 bool purgeImmediately)
Brian Salomon60dd8c72018-07-30 10:24:13 -040049 : GrSurface(gpu, desc)
50 , GrVkImage(info, std::move(layout), ownership)
Brian Salomone632dfc2018-08-01 13:01:16 -040051 , INHERITED(gpu, desc, GrTextureType::k2D, mipMapsStatus)
Brian Salomon60dd8c72018-07-30 10:24:13 -040052 , fTextureView(view) {
Greg Daniel0fc4d2d2017-10-12 11:23:36 -040053 SkASSERT((GrMipMapsStatus::kNotAllocated == mipMapsStatus) == (1 == info.fLevelCount));
Brian Salomonc67c31c2018-12-06 10:00:03 -050054 if (ioType == kRead_GrIOType) {
55 this->setReadOnly();
56 }
Greg Daniel2268ad22018-11-15 09:27:38 -050057 this->registerWithCacheWrapped(purgeImmediately);
Greg Daniel164a9f02016-02-22 09:56:40 -050058}
59
60// Because this class is virtually derived from GrSurface we must explicitly call its constructor.
61GrVkTexture::GrVkTexture(GrVkGpu* gpu,
62 const GrSurfaceDesc& desc,
egdanielb2df0c22016-05-13 11:30:37 -070063 const GrVkImageInfo& info,
Greg Daniel52e16d92018-04-10 09:34:07 -040064 sk_sp<GrVkImageLayout> layout,
egdanielb2df0c22016-05-13 11:30:37 -070065 const GrVkImageView* view,
Greg Daniel0fc4d2d2017-10-12 11:23:36 -040066 GrMipMapsStatus mipMapsStatus,
67 GrBackendObjectOwnership ownership)
Brian Salomon60dd8c72018-07-30 10:24:13 -040068 : GrSurface(gpu, desc)
69 , GrVkImage(info, layout, ownership)
Brian Salomone632dfc2018-08-01 13:01:16 -040070 , INHERITED(gpu, desc, GrTextureType::k2D, mipMapsStatus)
Brian Salomon60dd8c72018-07-30 10:24:13 -040071 , fTextureView(view) {
Greg Daniel0fc4d2d2017-10-12 11:23:36 -040072 SkASSERT((GrMipMapsStatus::kNotAllocated == mipMapsStatus) == (1 == info.fLevelCount));
Greg Daniel164a9f02016-02-22 09:56:40 -050073}
74
Greg Daniel475eb702018-09-28 14:16:50 -040075sk_sp<GrVkTexture> GrVkTexture::MakeNewTexture(GrVkGpu* gpu, SkBudgeted budgeted,
76 const GrSurfaceDesc& desc,
77 const GrVkImage::ImageDesc& imageDesc,
78 GrMipMapsStatus mipMapsStatus) {
Greg Daniel164a9f02016-02-22 09:56:40 -050079 SkASSERT(imageDesc.fUsageFlags & VK_IMAGE_USAGE_SAMPLED_BIT);
80
egdanielb2df0c22016-05-13 11:30:37 -070081 GrVkImageInfo info;
82 if (!GrVkImage::InitImageInfo(gpu, imageDesc, &info)) {
Greg Daniel164a9f02016-02-22 09:56:40 -050083 return nullptr;
84 }
85
Greg Daniel7e000222018-12-03 10:08:21 -050086 const GrVkImageView* imageView = GrVkImageView::Create(
87 gpu, info.fImage, info.fFormat, GrVkImageView::kColor_Type, info.fLevelCount,
88 info.fYcbcrConversionInfo);
egdanielb2df0c22016-05-13 11:30:37 -070089 if (!imageView) {
90 GrVkImage::DestroyImageInfo(gpu, &info);
91 return nullptr;
92 }
Greg Daniel52e16d92018-04-10 09:34:07 -040093 sk_sp<GrVkImageLayout> layout(new GrVkImageLayout(info.fImageLayout));
Greg Daniel164a9f02016-02-22 09:56:40 -050094
Greg Daniel52e16d92018-04-10 09:34:07 -040095 return sk_sp<GrVkTexture>(new GrVkTexture(gpu, budgeted, desc, info, std::move(layout),
96 imageView, mipMapsStatus));
Greg Daniel164a9f02016-02-22 09:56:40 -050097}
98
bungeman6bd52842016-10-27 09:30:08 -070099sk_sp<GrVkTexture> GrVkTexture::MakeWrappedTexture(GrVkGpu* gpu,
100 const GrSurfaceDesc& desc,
Greg Daniel1591c382017-08-17 15:37:20 -0400101 GrWrapOwnership wrapOwnership,
Brian Salomonc67c31c2018-12-06 10:00:03 -0500102 GrIOType ioType,
Greg Daniel2268ad22018-11-15 09:27:38 -0500103 bool purgeImmediately,
Greg Daniel52e16d92018-04-10 09:34:07 -0400104 const GrVkImageInfo& info,
105 sk_sp<GrVkImageLayout> layout) {
jvanverthfd359ca2016-03-18 11:57:24 -0700106 // Wrapped textures require both image and allocation (because they can be mapped)
Greg Daniel52e16d92018-04-10 09:34:07 -0400107 SkASSERT(VK_NULL_HANDLE != info.fImage && VK_NULL_HANDLE != info.fAlloc.fMemory);
Greg Daniel164a9f02016-02-22 09:56:40 -0500108
Greg Daniel7e000222018-12-03 10:08:21 -0500109 const GrVkImageView* imageView = GrVkImageView::Create(
110 gpu, info.fImage, info.fFormat, GrVkImageView::kColor_Type, info.fLevelCount,
111 info.fYcbcrConversionInfo);
egdanielb2df0c22016-05-13 11:30:37 -0700112 if (!imageView) {
jvanverthfd359ca2016-03-18 11:57:24 -0700113 return nullptr;
114 }
115
Greg Daniel52e16d92018-04-10 09:34:07 -0400116 GrMipMapsStatus mipMapsStatus = info.fLevelCount > 1 ? GrMipMapsStatus::kValid
117 : GrMipMapsStatus::kNotAllocated;
Greg Daniel0fc4d2d2017-10-12 11:23:36 -0400118
Greg Daniel1591c382017-08-17 15:37:20 -0400119 GrBackendObjectOwnership ownership = kBorrow_GrWrapOwnership == wrapOwnership
120 ? GrBackendObjectOwnership::kBorrowed : GrBackendObjectOwnership::kOwned;
Greg Daniel52e16d92018-04-10 09:34:07 -0400121 return sk_sp<GrVkTexture>(new GrVkTexture(gpu, kWrapped, desc, info, std::move(layout),
Brian Salomonc67c31c2018-12-06 10:00:03 -0500122 imageView, mipMapsStatus, ownership, ioType,
Greg Daniel2268ad22018-11-15 09:27:38 -0500123 purgeImmediately));
Greg Daniel164a9f02016-02-22 09:56:40 -0500124}
125
126GrVkTexture::~GrVkTexture() {
127 // either release or abandon should have been called by the owner of this object.
128 SkASSERT(!fTextureView);
129}
130
131void GrVkTexture::onRelease() {
Brian Salomon614c1a82018-12-19 15:42:06 -0500132 // When there is an idle proc, the Resource will call the proc in releaseImage() so
133 // we clear it here.
134 fIdleProc = nullptr;
135 fIdleProcContext = nullptr;
136
Greg Daniel164a9f02016-02-22 09:56:40 -0500137 // we create this and don't hand it off, so we should always destroy it
138 if (fTextureView) {
139 fTextureView->unref(this->getVkGpu());
140 fTextureView = nullptr;
141 }
142
kkinnunen2e6055b2016-04-22 01:48:29 -0700143 this->releaseImage(this->getVkGpu());
Greg Daniel164a9f02016-02-22 09:56:40 -0500144
145 INHERITED::onRelease();
146}
147
148void GrVkTexture::onAbandon() {
Brian Salomon614c1a82018-12-19 15:42:06 -0500149 // When there is an idle proc, the Resource will call the proc in abandonImage() so
150 // we clear it here.
151 fIdleProc = nullptr;
152 fIdleProcContext = nullptr;
153 // we create this and don't hand it off, so we should always destroy it
Greg Daniel164a9f02016-02-22 09:56:40 -0500154 if (fTextureView) {
155 fTextureView->unrefAndAbandon();
156 fTextureView = nullptr;
157 }
158
159 this->abandonImage();
160 INHERITED::onAbandon();
161}
162
Robert Phillipsb67821d2017-12-13 15:00:45 -0500163GrBackendTexture GrVkTexture::getBackendTexture() const {
Greg Daniel52e16d92018-04-10 09:34:07 -0400164 return GrBackendTexture(this->width(), this->height(), fInfo, this->grVkImageLayout());
Robert Phillipsb67821d2017-12-13 15:00:45 -0500165}
166
Greg Daniel164a9f02016-02-22 09:56:40 -0500167GrVkGpu* GrVkTexture::getVkGpu() const {
168 SkASSERT(!this->wasDestroyed());
169 return static_cast<GrVkGpu*>(this->getGpu());
170}
jvanverth62340062016-04-26 08:01:44 -0700171
Brian Osman2b23c4b2018-06-01 12:25:08 -0400172const GrVkImageView* GrVkTexture::textureView() {
173 return fTextureView;
brianosmanf05ab1b2016-05-12 11:01:10 -0700174}
175
Brian Salomon614c1a82018-12-19 15:42:06 -0500176void GrVkTexture::setIdleProc(IdleProc proc, void* context) {
177 fIdleProc = proc;
178 fIdleProcContext = context;
179 if (auto* resource = this->resource()) {
180 resource->setIdleProc(proc ? this : nullptr, proc, context);
181 }
182}
183
184void GrVkTexture::becamePurgeable() {
185 if (!fIdleProc) {
186 return;
187 }
188 // This is called when the GrTexture is purgeable. However, we need to check whether the
189 // Resource is still owned by any command buffers. If it is then it will call the proc.
190 auto* resource = this->resource();
191 SkASSERT(resource);
192 if (resource->isOwnedByCommandBuffer()) {
193 return;
194 }
195 fIdleProc(fIdleProcContext);
196 fIdleProc = nullptr;
197 fIdleProcContext = nullptr;
198 resource->setIdleProc(nullptr, nullptr, nullptr);
199}