| /* |
| * Copyright 2015 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "GrVkTextureRenderTarget.h" |
| |
| #include "GrTexturePriv.h" |
| #include "GrVkGpu.h" |
| #include "GrVkImageView.h" |
| #include "GrVkUtil.h" |
| |
| #include "SkMipMap.h" |
| |
| #include "vk/GrVkTypes.h" |
| |
| #define VK_CALL(GPU, X) GR_VK_CALL(GPU->vkInterface(), X) |
| |
| GrVkTextureRenderTarget::GrVkTextureRenderTarget(GrVkGpu* gpu, |
| SkBudgeted budgeted, |
| const GrSurfaceDesc& desc, |
| const GrVkImageInfo& info, |
| sk_sp<GrVkImageLayout> layout, |
| const GrVkImageView* texView, |
| const GrVkImageInfo& msaaInfo, |
| sk_sp<GrVkImageLayout> msaaLayout, |
| const GrVkImageView* colorAttachmentView, |
| const GrVkImageView* resolveAttachmentView, |
| GrMipMapsStatus mipMapsStatus, |
| GrBackendObjectOwnership ownership) |
| : GrSurface(gpu, desc) |
| , GrVkImage(info, layout, ownership) |
| , GrVkTexture(gpu, desc, info, layout, texView, mipMapsStatus, ownership) |
| , GrVkRenderTarget(gpu, desc, info, layout, msaaInfo, std::move(msaaLayout), |
| colorAttachmentView, resolveAttachmentView, |
| GrBackendObjectOwnership::kOwned) { |
| this->registerWithCache(budgeted); |
| } |
| |
| GrVkTextureRenderTarget::GrVkTextureRenderTarget(GrVkGpu* gpu, |
| SkBudgeted budgeted, |
| const GrSurfaceDesc& desc, |
| const GrVkImageInfo& info, |
| sk_sp<GrVkImageLayout> layout, |
| const GrVkImageView* texView, |
| const GrVkImageView* colorAttachmentView, |
| GrMipMapsStatus mipMapsStatus, |
| GrBackendObjectOwnership ownership) |
| : GrSurface(gpu, desc) |
| , GrVkImage(info, layout, ownership) |
| , GrVkTexture(gpu, desc, info, layout, texView, mipMapsStatus, ownership) |
| , GrVkRenderTarget(gpu, desc, info, layout, colorAttachmentView, |
| GrBackendObjectOwnership::kOwned) { |
| this->registerWithCache(budgeted); |
| } |
| |
| GrVkTextureRenderTarget::GrVkTextureRenderTarget(GrVkGpu* gpu, |
| const GrSurfaceDesc& desc, |
| const GrVkImageInfo& info, |
| sk_sp<GrVkImageLayout> layout, |
| const GrVkImageView* texView, |
| const GrVkImageInfo& msaaInfo, |
| sk_sp<GrVkImageLayout> msaaLayout, |
| const GrVkImageView* colorAttachmentView, |
| const GrVkImageView* resolveAttachmentView, |
| GrMipMapsStatus mipMapsStatus, |
| GrBackendObjectOwnership ownership) |
| : GrSurface(gpu, desc) |
| , GrVkImage(info, layout, ownership) |
| , GrVkTexture(gpu, desc, info, layout, texView, mipMapsStatus, ownership) |
| , GrVkRenderTarget(gpu, desc, info, layout, msaaInfo, std::move(msaaLayout), |
| colorAttachmentView, resolveAttachmentView, ownership) { |
| this->registerWithCacheWrapped(); |
| } |
| |
| GrVkTextureRenderTarget::GrVkTextureRenderTarget(GrVkGpu* gpu, |
| const GrSurfaceDesc& desc, |
| const GrVkImageInfo& info, |
| sk_sp<GrVkImageLayout> layout, |
| const GrVkImageView* texView, |
| const GrVkImageView* colorAttachmentView, |
| GrMipMapsStatus mipMapsStatus, |
| GrBackendObjectOwnership ownership) |
| : GrSurface(gpu, desc) |
| , GrVkImage(info, layout, ownership) |
| , GrVkTexture(gpu, desc, info, layout, texView, mipMapsStatus, ownership) |
| , GrVkRenderTarget(gpu, desc, info, layout, colorAttachmentView, ownership) { |
| this->registerWithCacheWrapped(); |
| } |
| |
| sk_sp<GrVkTextureRenderTarget> GrVkTextureRenderTarget::Make(GrVkGpu* gpu, |
| const GrSurfaceDesc& desc, |
| const GrVkImageInfo& info, |
| sk_sp<GrVkImageLayout> layout, |
| GrMipMapsStatus mipMapsStatus, |
| SkBudgeted budgeted, |
| GrBackendObjectOwnership ownership, |
| bool isWrapped) { |
| VkImage image = info.fImage; |
| // Create the texture ImageView |
| const GrVkImageView* imageView = GrVkImageView::Create(gpu, image, info.fFormat, |
| GrVkImageView::kColor_Type, |
| info.fLevelCount); |
| if (!imageView) { |
| return nullptr; |
| } |
| |
| VkFormat pixelFormat; |
| GrPixelConfigToVkFormat(desc.fConfig, &pixelFormat); |
| |
| VkImage colorImage; |
| |
| // create msaa surface if necessary |
| GrVkImageInfo msInfo; |
| sk_sp<GrVkImageLayout> msLayout; |
| const GrVkImageView* resolveAttachmentView = nullptr; |
| if (desc.fSampleCnt > 1) { |
| GrVkImage::ImageDesc msImageDesc; |
| msImageDesc.fImageType = VK_IMAGE_TYPE_2D; |
| msImageDesc.fFormat = pixelFormat; |
| msImageDesc.fWidth = desc.fWidth; |
| msImageDesc.fHeight = desc.fHeight; |
| msImageDesc.fLevels = 1; |
| msImageDesc.fSamples = desc.fSampleCnt; |
| msImageDesc.fImageTiling = VK_IMAGE_TILING_OPTIMAL; |
| msImageDesc.fUsageFlags = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | |
| VK_IMAGE_USAGE_TRANSFER_DST_BIT | |
| VK_IMAGE_USAGE_TRANSFER_SRC_BIT; |
| msImageDesc.fMemProps = VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT; |
| |
| if (!GrVkImage::InitImageInfo(gpu, msImageDesc, &msInfo)) { |
| imageView->unref(gpu); |
| return nullptr; |
| } |
| |
| // Set color attachment image |
| colorImage = msInfo.fImage; |
| |
| // Create resolve attachment view. |
| resolveAttachmentView = GrVkImageView::Create(gpu, image, pixelFormat, |
| GrVkImageView::kColor_Type, |
| info.fLevelCount); |
| if (!resolveAttachmentView) { |
| GrVkImage::DestroyImageInfo(gpu, &msInfo); |
| imageView->unref(gpu); |
| return nullptr; |
| } |
| msLayout.reset(new GrVkImageLayout(msInfo.fImageLayout)); |
| } else { |
| // Set color attachment image |
| colorImage = info.fImage; |
| } |
| |
| const GrVkImageView* colorAttachmentView = GrVkImageView::Create(gpu, colorImage, pixelFormat, |
| GrVkImageView::kColor_Type, 1); |
| if (!colorAttachmentView) { |
| if (desc.fSampleCnt > 1) { |
| resolveAttachmentView->unref(gpu); |
| GrVkImage::DestroyImageInfo(gpu, &msInfo); |
| } |
| imageView->unref(gpu); |
| return nullptr; |
| } |
| |
| sk_sp<GrVkTextureRenderTarget> texRT; |
| if (desc.fSampleCnt > 1) { |
| if (!isWrapped) { |
| texRT = sk_sp<GrVkTextureRenderTarget>(new GrVkTextureRenderTarget( |
| gpu, budgeted, desc, |
| info, std::move(layout), imageView, msInfo, |
| std::move(msLayout), colorAttachmentView, |
| resolveAttachmentView, mipMapsStatus, |
| ownership)); |
| } else { |
| texRT = sk_sp<GrVkTextureRenderTarget>(new GrVkTextureRenderTarget( |
| gpu, desc, |
| info, std::move(layout), imageView, msInfo, |
| std::move(msLayout), colorAttachmentView, |
| resolveAttachmentView, mipMapsStatus, |
| ownership)); |
| } |
| } else { |
| if (!isWrapped) { |
| texRT = sk_sp<GrVkTextureRenderTarget>(new GrVkTextureRenderTarget( |
| gpu, budgeted, desc, |
| info, std::move(layout), imageView, |
| colorAttachmentView, mipMapsStatus, |
| ownership)); |
| } else { |
| texRT = sk_sp<GrVkTextureRenderTarget>(new GrVkTextureRenderTarget( |
| gpu, desc, |
| info, std::move(layout), imageView, |
| colorAttachmentView, mipMapsStatus, |
| ownership)); |
| } |
| } |
| return texRT; |
| } |
| |
| sk_sp<GrVkTextureRenderTarget> |
| GrVkTextureRenderTarget::CreateNewTextureRenderTarget(GrVkGpu* gpu, |
| SkBudgeted budgeted, |
| const GrSurfaceDesc& desc, |
| const GrVkImage::ImageDesc& imageDesc, |
| GrMipMapsStatus mipMapsStatus) { |
| SkASSERT(imageDesc.fUsageFlags & VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT); |
| SkASSERT(imageDesc.fUsageFlags & VK_IMAGE_USAGE_SAMPLED_BIT); |
| |
| GrVkImageInfo info; |
| if (!GrVkImage::InitImageInfo(gpu, imageDesc, &info)) { |
| return nullptr; |
| } |
| sk_sp<GrVkImageLayout> layout(new GrVkImageLayout(info.fImageLayout)); |
| |
| sk_sp<GrVkTextureRenderTarget> trt = Make(gpu, desc, info, std::move(layout), mipMapsStatus, |
| budgeted, GrBackendObjectOwnership::kOwned, false); |
| if (!trt) { |
| GrVkImage::DestroyImageInfo(gpu, &info); |
| } |
| |
| return trt; |
| } |
| |
| sk_sp<GrVkTextureRenderTarget> |
| GrVkTextureRenderTarget::MakeWrappedTextureRenderTarget(GrVkGpu* gpu, |
| const GrSurfaceDesc& desc, |
| GrWrapOwnership wrapOwnership, |
| const GrVkImageInfo& info, |
| sk_sp<GrVkImageLayout> layout) { |
| // Wrapped textures require both image and allocation (because they can be mapped) |
| SkASSERT(VK_NULL_HANDLE != info.fImage && VK_NULL_HANDLE != info.fAlloc.fMemory); |
| |
| GrMipMapsStatus mipMapsStatus = info.fLevelCount > 1 ? GrMipMapsStatus::kDirty |
| : GrMipMapsStatus::kNotAllocated; |
| |
| GrBackendObjectOwnership ownership = kBorrow_GrWrapOwnership == wrapOwnership |
| ? GrBackendObjectOwnership::kBorrowed : GrBackendObjectOwnership::kOwned; |
| |
| return Make(gpu, desc, info, std::move(layout), mipMapsStatus, SkBudgeted::kNo, ownership, |
| true); |
| } |
| |
| bool GrVkTextureRenderTarget::updateForMipmap(GrVkGpu* gpu, const GrVkImageInfo& newInfo) { |
| VkFormat pixelFormat; |
| GrPixelConfigToVkFormat(this->config(), &pixelFormat); |
| if (this->numStencilSamples() > 1) { |
| const GrVkImageView* resolveAttachmentView = |
| GrVkImageView::Create(gpu, |
| newInfo.fImage, |
| pixelFormat, |
| GrVkImageView::kColor_Type, |
| newInfo.fLevelCount); |
| if (!resolveAttachmentView) { |
| return false; |
| } |
| fResolveAttachmentView->unref(gpu); |
| fResolveAttachmentView = resolveAttachmentView; |
| } else { |
| const GrVkImageView* colorAttachmentView = GrVkImageView::Create(gpu, |
| newInfo.fImage, |
| pixelFormat, |
| GrVkImageView::kColor_Type, |
| 1); |
| if (!colorAttachmentView) { |
| return false; |
| } |
| fColorAttachmentView->unref(gpu); |
| fColorAttachmentView = colorAttachmentView; |
| } |
| |
| this->createFramebuffer(gpu); |
| return true; |
| } |
| |
| size_t GrVkTextureRenderTarget::onGpuMemorySize() const { |
| int numColorSamples = this->numColorSamples(); |
| if (numColorSamples > 1) { |
| // Add one to account for the resolve VkImage. |
| ++numColorSamples; |
| } |
| return GrSurface::ComputeSize(this->config(), this->width(), this->height(), |
| numColorSamples, // TODO: this still correct? |
| this->texturePriv().mipMapped()); |
| } |