Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 1 | /* |
| 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 Phillips | f95b175 | 2017-08-31 08:56:07 -0400 | [diff] [blame] | 9 | |
| 10 | #include "GrTexturePriv.h" |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 11 | #include "GrVkGpu.h" |
| 12 | #include "GrVkImageView.h" |
egdaniel | 50ead53 | 2016-07-13 14:23:26 -0700 | [diff] [blame] | 13 | #include "GrVkTextureRenderTarget.h" |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 14 | #include "GrVkUtil.h" |
| 15 | |
jvanverth | fd359ca | 2016-03-18 11:57:24 -0700 | [diff] [blame] | 16 | #include "vk/GrVkTypes.h" |
| 17 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 18 | #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. |
| 21 | GrVkTexture::GrVkTexture(GrVkGpu* gpu, |
kkinnunen | 2e6055b | 2016-04-22 01:48:29 -0700 | [diff] [blame] | 22 | SkBudgeted budgeted, |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 23 | const GrSurfaceDesc& desc, |
egdaniel | b2df0c2 | 2016-05-13 11:30:37 -0700 | [diff] [blame] | 24 | const GrVkImageInfo& info, |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 25 | sk_sp<GrVkImageLayout> layout, |
Greg Daniel | 834f120 | 2017-10-09 15:06:20 -0400 | [diff] [blame] | 26 | const GrVkImageView* view, |
Greg Daniel | 0fc4d2d | 2017-10-12 11:23:36 -0400 | [diff] [blame] | 27 | GrMipMapsStatus mipMapsStatus) |
Brian Salomon | 60dd8c7 | 2018-07-30 10:24:13 -0400 | [diff] [blame] | 28 | : GrSurface(gpu, desc) |
| 29 | , GrVkImage(info, std::move(layout), GrBackendObjectOwnership::kOwned) |
Brian Salomon | e632dfc | 2018-08-01 13:01:16 -0400 | [diff] [blame] | 30 | , INHERITED(gpu, desc, GrTextureType::k2D, mipMapsStatus) |
Brian Salomon | 60dd8c7 | 2018-07-30 10:24:13 -0400 | [diff] [blame] | 31 | , fTextureView(view) { |
Greg Daniel | 0fc4d2d | 2017-10-12 11:23:36 -0400 | [diff] [blame] | 32 | SkASSERT((GrMipMapsStatus::kNotAllocated == mipMapsStatus) == (1 == info.fLevelCount)); |
kkinnunen | 2e6055b | 2016-04-22 01:48:29 -0700 | [diff] [blame] | 33 | this->registerWithCache(budgeted); |
Jim Van Verth | 1676cb9 | 2019-01-15 13:24:45 -0500 | [diff] [blame] | 34 | if (GrPixelConfigIsCompressed(desc.fConfig)) { |
| 35 | this->setReadOnly(); |
| 36 | } |
kkinnunen | 2e6055b | 2016-04-22 01:48:29 -0700 | [diff] [blame] | 37 | } |
| 38 | |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 39 | GrVkTexture::GrVkTexture(GrVkGpu* gpu, const GrSurfaceDesc& desc, const GrVkImageInfo& info, |
| 40 | sk_sp<GrVkImageLayout> layout, const GrVkImageView* view, |
| 41 | GrMipMapsStatus mipMapsStatus, GrBackendObjectOwnership ownership, |
| 42 | GrWrapCacheable cacheable, GrIOType ioType) |
Brian Salomon | 60dd8c7 | 2018-07-30 10:24:13 -0400 | [diff] [blame] | 43 | : GrSurface(gpu, desc) |
| 44 | , GrVkImage(info, std::move(layout), ownership) |
Brian Salomon | e632dfc | 2018-08-01 13:01:16 -0400 | [diff] [blame] | 45 | , INHERITED(gpu, desc, GrTextureType::k2D, mipMapsStatus) |
Brian Salomon | 60dd8c7 | 2018-07-30 10:24:13 -0400 | [diff] [blame] | 46 | , fTextureView(view) { |
Greg Daniel | 0fc4d2d | 2017-10-12 11:23:36 -0400 | [diff] [blame] | 47 | SkASSERT((GrMipMapsStatus::kNotAllocated == mipMapsStatus) == (1 == info.fLevelCount)); |
Brian Salomon | c67c31c | 2018-12-06 10:00:03 -0500 | [diff] [blame] | 48 | if (ioType == kRead_GrIOType) { |
| 49 | this->setReadOnly(); |
| 50 | } |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 51 | this->registerWithCacheWrapped(cacheable); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 52 | } |
| 53 | |
| 54 | // Because this class is virtually derived from GrSurface we must explicitly call its constructor. |
| 55 | GrVkTexture::GrVkTexture(GrVkGpu* gpu, |
| 56 | const GrSurfaceDesc& desc, |
egdaniel | b2df0c2 | 2016-05-13 11:30:37 -0700 | [diff] [blame] | 57 | const GrVkImageInfo& info, |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 58 | sk_sp<GrVkImageLayout> layout, |
egdaniel | b2df0c2 | 2016-05-13 11:30:37 -0700 | [diff] [blame] | 59 | const GrVkImageView* view, |
Greg Daniel | 0fc4d2d | 2017-10-12 11:23:36 -0400 | [diff] [blame] | 60 | GrMipMapsStatus mipMapsStatus, |
| 61 | GrBackendObjectOwnership ownership) |
Brian Salomon | 60dd8c7 | 2018-07-30 10:24:13 -0400 | [diff] [blame] | 62 | : GrSurface(gpu, desc) |
| 63 | , GrVkImage(info, layout, ownership) |
Brian Salomon | e632dfc | 2018-08-01 13:01:16 -0400 | [diff] [blame] | 64 | , INHERITED(gpu, desc, GrTextureType::k2D, mipMapsStatus) |
Brian Salomon | 60dd8c7 | 2018-07-30 10:24:13 -0400 | [diff] [blame] | 65 | , fTextureView(view) { |
Greg Daniel | 0fc4d2d | 2017-10-12 11:23:36 -0400 | [diff] [blame] | 66 | SkASSERT((GrMipMapsStatus::kNotAllocated == mipMapsStatus) == (1 == info.fLevelCount)); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 67 | } |
| 68 | |
Greg Daniel | 475eb70 | 2018-09-28 14:16:50 -0400 | [diff] [blame] | 69 | sk_sp<GrVkTexture> GrVkTexture::MakeNewTexture(GrVkGpu* gpu, SkBudgeted budgeted, |
| 70 | const GrSurfaceDesc& desc, |
| 71 | const GrVkImage::ImageDesc& imageDesc, |
| 72 | GrMipMapsStatus mipMapsStatus) { |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 73 | SkASSERT(imageDesc.fUsageFlags & VK_IMAGE_USAGE_SAMPLED_BIT); |
| 74 | |
egdaniel | b2df0c2 | 2016-05-13 11:30:37 -0700 | [diff] [blame] | 75 | GrVkImageInfo info; |
| 76 | if (!GrVkImage::InitImageInfo(gpu, imageDesc, &info)) { |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 77 | return nullptr; |
| 78 | } |
| 79 | |
Greg Daniel | 7e00022 | 2018-12-03 10:08:21 -0500 | [diff] [blame] | 80 | const GrVkImageView* imageView = GrVkImageView::Create( |
| 81 | gpu, info.fImage, info.fFormat, GrVkImageView::kColor_Type, info.fLevelCount, |
| 82 | info.fYcbcrConversionInfo); |
egdaniel | b2df0c2 | 2016-05-13 11:30:37 -0700 | [diff] [blame] | 83 | if (!imageView) { |
| 84 | GrVkImage::DestroyImageInfo(gpu, &info); |
| 85 | return nullptr; |
| 86 | } |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 87 | sk_sp<GrVkImageLayout> layout(new GrVkImageLayout(info.fImageLayout)); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 88 | |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 89 | return sk_sp<GrVkTexture>(new GrVkTexture(gpu, budgeted, desc, info, std::move(layout), |
| 90 | imageView, mipMapsStatus)); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 91 | } |
| 92 | |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 93 | sk_sp<GrVkTexture> GrVkTexture::MakeWrappedTexture(GrVkGpu* gpu, const GrSurfaceDesc& desc, |
Greg Daniel | 1591c38 | 2017-08-17 15:37:20 -0400 | [diff] [blame] | 94 | GrWrapOwnership wrapOwnership, |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 95 | GrWrapCacheable cacheable, GrIOType ioType, |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 96 | const GrVkImageInfo& info, |
| 97 | sk_sp<GrVkImageLayout> layout) { |
jvanverth | fd359ca | 2016-03-18 11:57:24 -0700 | [diff] [blame] | 98 | // Wrapped textures require both image and allocation (because they can be mapped) |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 99 | SkASSERT(VK_NULL_HANDLE != info.fImage && VK_NULL_HANDLE != info.fAlloc.fMemory); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 100 | |
Greg Daniel | 7e00022 | 2018-12-03 10:08:21 -0500 | [diff] [blame] | 101 | const GrVkImageView* imageView = GrVkImageView::Create( |
| 102 | gpu, info.fImage, info.fFormat, GrVkImageView::kColor_Type, info.fLevelCount, |
| 103 | info.fYcbcrConversionInfo); |
egdaniel | b2df0c2 | 2016-05-13 11:30:37 -0700 | [diff] [blame] | 104 | if (!imageView) { |
jvanverth | fd359ca | 2016-03-18 11:57:24 -0700 | [diff] [blame] | 105 | return nullptr; |
| 106 | } |
| 107 | |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 108 | GrMipMapsStatus mipMapsStatus = info.fLevelCount > 1 ? GrMipMapsStatus::kValid |
| 109 | : GrMipMapsStatus::kNotAllocated; |
Greg Daniel | 0fc4d2d | 2017-10-12 11:23:36 -0400 | [diff] [blame] | 110 | |
Greg Daniel | 1591c38 | 2017-08-17 15:37:20 -0400 | [diff] [blame] | 111 | GrBackendObjectOwnership ownership = kBorrow_GrWrapOwnership == wrapOwnership |
| 112 | ? GrBackendObjectOwnership::kBorrowed : GrBackendObjectOwnership::kOwned; |
Brian Salomon | fa2ebea | 2019-01-24 15:58:58 -0500 | [diff] [blame] | 113 | return sk_sp<GrVkTexture>(new GrVkTexture(gpu, desc, info, std::move(layout), imageView, |
| 114 | mipMapsStatus, ownership, cacheable, ioType)); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 115 | } |
| 116 | |
| 117 | GrVkTexture::~GrVkTexture() { |
| 118 | // either release or abandon should have been called by the owner of this object. |
| 119 | SkASSERT(!fTextureView); |
| 120 | } |
| 121 | |
| 122 | void GrVkTexture::onRelease() { |
Brian Salomon | 9bc76d9 | 2019-01-24 12:18:33 -0500 | [diff] [blame] | 123 | // We're about to be severed from our GrVkResource. If there is an idle proc we have to decide |
| 124 | // who will handle it. If the resource is still tied to a command buffer we let it handle it. |
| 125 | // Otherwise, we handle it. |
| 126 | if (this->hasResource() && this->resource()->isOwnedByCommandBuffer()) { |
| 127 | fIdleProc = nullptr; |
| 128 | fIdleProcContext = nullptr; |
| 129 | } |
Brian Salomon | 614c1a8 | 2018-12-19 15:42:06 -0500 | [diff] [blame] | 130 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 131 | // we create this and don't hand it off, so we should always destroy it |
| 132 | if (fTextureView) { |
| 133 | fTextureView->unref(this->getVkGpu()); |
| 134 | fTextureView = nullptr; |
| 135 | } |
| 136 | |
kkinnunen | 2e6055b | 2016-04-22 01:48:29 -0700 | [diff] [blame] | 137 | this->releaseImage(this->getVkGpu()); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 138 | |
| 139 | INHERITED::onRelease(); |
| 140 | } |
| 141 | |
| 142 | void GrVkTexture::onAbandon() { |
Brian Salomon | 9bc76d9 | 2019-01-24 12:18:33 -0500 | [diff] [blame] | 143 | // We're about to be severed from our GrVkResource. If there is an idle proc we have to decide |
| 144 | // who will handle it. If the resource is still tied to a command buffer we let it handle it. |
| 145 | // Otherwise, we handle it. |
| 146 | if (this->hasResource() && this->resource()->isOwnedByCommandBuffer()) { |
| 147 | fIdleProc = nullptr; |
| 148 | fIdleProcContext = nullptr; |
| 149 | } |
| 150 | |
Brian Salomon | 614c1a8 | 2018-12-19 15:42:06 -0500 | [diff] [blame] | 151 | // we create this and don't hand it off, so we should always destroy it |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 152 | if (fTextureView) { |
| 153 | fTextureView->unrefAndAbandon(); |
| 154 | fTextureView = nullptr; |
| 155 | } |
| 156 | |
| 157 | this->abandonImage(); |
| 158 | INHERITED::onAbandon(); |
| 159 | } |
| 160 | |
Robert Phillips | b67821d | 2017-12-13 15:00:45 -0500 | [diff] [blame] | 161 | GrBackendTexture GrVkTexture::getBackendTexture() const { |
Greg Daniel | 52e16d9 | 2018-04-10 09:34:07 -0400 | [diff] [blame] | 162 | return GrBackendTexture(this->width(), this->height(), fInfo, this->grVkImageLayout()); |
Robert Phillips | b67821d | 2017-12-13 15:00:45 -0500 | [diff] [blame] | 163 | } |
| 164 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 165 | GrVkGpu* GrVkTexture::getVkGpu() const { |
| 166 | SkASSERT(!this->wasDestroyed()); |
| 167 | return static_cast<GrVkGpu*>(this->getGpu()); |
| 168 | } |
jvanverth | 6234006 | 2016-04-26 08:01:44 -0700 | [diff] [blame] | 169 | |
Brian Osman | 2b23c4b | 2018-06-01 12:25:08 -0400 | [diff] [blame] | 170 | const GrVkImageView* GrVkTexture::textureView() { |
| 171 | return fTextureView; |
brianosman | f05ab1b | 2016-05-12 11:01:10 -0700 | [diff] [blame] | 172 | } |
| 173 | |
Brian Salomon | 614c1a8 | 2018-12-19 15:42:06 -0500 | [diff] [blame] | 174 | void GrVkTexture::setIdleProc(IdleProc proc, void* context) { |
| 175 | fIdleProc = proc; |
| 176 | fIdleProcContext = context; |
| 177 | if (auto* resource = this->resource()) { |
| 178 | resource->setIdleProc(proc ? this : nullptr, proc, context); |
| 179 | } |
| 180 | } |
| 181 | |
Brian Salomon | 9bc76d9 | 2019-01-24 12:18:33 -0500 | [diff] [blame] | 182 | void GrVkTexture::removedLastRefOrPendingIO() { |
Brian Salomon | 614c1a8 | 2018-12-19 15:42:06 -0500 | [diff] [blame] | 183 | if (!fIdleProc) { |
| 184 | return; |
| 185 | } |
| 186 | // This is called when the GrTexture is purgeable. However, we need to check whether the |
| 187 | // Resource is still owned by any command buffers. If it is then it will call the proc. |
Brian Salomon | 9bc76d9 | 2019-01-24 12:18:33 -0500 | [diff] [blame] | 188 | auto* resource = this->hasResource() ? this->resource() : nullptr; |
| 189 | if (resource && resource->isOwnedByCommandBuffer()) { |
Brian Salomon | 614c1a8 | 2018-12-19 15:42:06 -0500 | [diff] [blame] | 190 | return; |
| 191 | } |
| 192 | fIdleProc(fIdleProcContext); |
| 193 | fIdleProc = nullptr; |
| 194 | fIdleProcContext = nullptr; |
Brian Salomon | 9bc76d9 | 2019-01-24 12:18:33 -0500 | [diff] [blame] | 195 | if (resource) { |
| 196 | resource->setIdleProc(nullptr, nullptr, nullptr); |
| 197 | } |
Brian Salomon | 614c1a8 | 2018-12-19 15:42:06 -0500 | [diff] [blame] | 198 | } |