|  | /* | 
|  | * Copyright 2011 Google Inc. | 
|  | * | 
|  | * Use of this source code is governed by a BSD-style license that can be | 
|  | * found in the LICENSE file. | 
|  | */ | 
|  |  | 
|  | #include "include/core/SkMath.h" | 
|  | #include "include/core/SkTypes.h" | 
|  | #include "include/gpu/GrContext.h" | 
|  | #include "include/gpu/GrTexture.h" | 
|  | #include "include/gpu/GrTypes.h" | 
|  | #include "include/private/GrResourceKey.h" | 
|  | #include "src/core/SkMipMap.h" | 
|  | #include "src/gpu/GrCaps.h" | 
|  | #include "src/gpu/GrContextPriv.h" | 
|  | #include "src/gpu/GrGpu.h" | 
|  | #include "src/gpu/GrRenderTarget.h" | 
|  | #include "src/gpu/GrSurfacePriv.h" | 
|  | #include "src/gpu/GrTexturePriv.h" | 
|  |  | 
|  | void GrTexture::markMipMapsDirty() { | 
|  | if (GrMipMapsStatus::kValid == fMipMapsStatus) { | 
|  | fMipMapsStatus = GrMipMapsStatus::kDirty; | 
|  | } | 
|  | } | 
|  |  | 
|  | void GrTexture::markMipMapsClean() { | 
|  | SkASSERT(GrMipMapsStatus::kNotAllocated != fMipMapsStatus); | 
|  | fMipMapsStatus = GrMipMapsStatus::kValid; | 
|  | } | 
|  |  | 
|  | size_t GrTexture::onGpuMemorySize() const { | 
|  | const GrCaps& caps = *this->getGpu()->caps(); | 
|  | return GrSurface::ComputeSize(caps, this->backendFormat(), this->width(), this->height(), | 
|  | 1, this->texturePriv().mipMapped()); | 
|  | } | 
|  |  | 
|  | ///////////////////////////////////////////////////////////////////////////// | 
|  | GrTexture::GrTexture(GrGpu* gpu, const SkISize& size, GrPixelConfig config, GrProtected isProtected, | 
|  | GrTextureType textureType, GrMipMapsStatus mipMapsStatus) | 
|  | : INHERITED(gpu, size, config, isProtected) | 
|  | , fTextureType(textureType) | 
|  | , fMipMapsStatus(mipMapsStatus) { | 
|  | if (GrMipMapsStatus::kNotAllocated == fMipMapsStatus) { | 
|  | fMaxMipMapLevel = 0; | 
|  | } else { | 
|  | fMaxMipMapLevel = SkMipMap::ComputeLevelCount(this->width(), this->height()); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool GrTexture::StealBackendTexture(sk_sp<GrTexture> texture, | 
|  | GrBackendTexture* backendTexture, | 
|  | SkImage::BackendTextureReleaseProc* releaseProc) { | 
|  | if (!texture->unique()) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | if (!texture->onStealBackendTexture(backendTexture, releaseProc)) { | 
|  | return false; | 
|  | } | 
|  | #ifdef SK_DEBUG | 
|  | GrResourceCache* cache = texture->getContext()->priv().getResourceCache(); | 
|  | int preCount = cache->getResourceCount(); | 
|  | #endif | 
|  | // Ensure that the texture will be released by the cache when we drop the last ref. | 
|  | // A texture that has no refs and no keys should be immediately removed. | 
|  | if (texture->getUniqueKey().isValid()) { | 
|  | texture->resourcePriv().removeUniqueKey(); | 
|  | } | 
|  | if (texture->resourcePriv().getScratchKey().isValid()) { | 
|  | texture->resourcePriv().removeScratchKey(); | 
|  | } | 
|  | #ifdef SK_DEBUG | 
|  | texture.reset(); | 
|  | int postCount = cache->getResourceCount(); | 
|  | SkASSERT(postCount < preCount); | 
|  | #endif | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void GrTexture::computeScratchKey(GrScratchKey* key) const { | 
|  | if (!this->getGpu()->caps()->isFormatCompressed(this->backendFormat())) { | 
|  | int sampleCount = 1; | 
|  | GrRenderable renderable = GrRenderable::kNo; | 
|  | if (const auto* rt = this->asRenderTarget()) { | 
|  | sampleCount = rt->numSamples(); | 
|  | renderable = GrRenderable::kYes; | 
|  | } | 
|  | auto isProtected = this->isProtected() ? GrProtected::kYes : GrProtected::kNo; | 
|  | GrTexturePriv::ComputeScratchKey(this->config(), this->width(), this->height(), renderable, | 
|  | sampleCount, this->texturePriv().mipMapped(), isProtected, | 
|  | key); | 
|  | } | 
|  | } | 
|  |  | 
|  | void GrTexturePriv::ComputeScratchKey(GrPixelConfig config, | 
|  | int width, | 
|  | int height, | 
|  | GrRenderable renderable, | 
|  | int sampleCnt, | 
|  | GrMipMapped mipMapped, | 
|  | GrProtected isProtected, | 
|  | GrScratchKey* key) { | 
|  | static const GrScratchKey::ResourceType kType = GrScratchKey::GenerateResourceType(); | 
|  | SkASSERT(width > 0); | 
|  | SkASSERT(height > 0); | 
|  | SkASSERT(sampleCnt > 0); | 
|  | SkASSERT(1 == sampleCnt || renderable == GrRenderable::kYes); | 
|  |  | 
|  | // make sure desc.fConfig fits in 5 bits | 
|  | SkASSERT(sk_float_log2(kLast_GrPixelConfig) <= 5); | 
|  | SkASSERT(static_cast<uint32_t>(config) < (1 << 5)); | 
|  | SkASSERT(static_cast<uint32_t>(mipMapped) <= 1); | 
|  | SkASSERT(static_cast<uint32_t>(isProtected) <= 1); | 
|  | SkASSERT(static_cast<uint32_t>(renderable) <= 1); | 
|  | SkASSERT(static_cast<uint32_t>(sampleCnt) < (1 << (32 - 8))); | 
|  |  | 
|  | GrScratchKey::Builder builder(key, kType, 3); | 
|  | builder[0] = width; | 
|  | builder[1] = height; | 
|  | builder[2] = (static_cast<uint32_t>(config)      << 0) | 
|  | | (static_cast<uint32_t>(mipMapped)   << 5) | 
|  | | (static_cast<uint32_t>(isProtected) << 6) | 
|  | | (static_cast<uint32_t>(renderable)  << 7) | 
|  | | (static_cast<uint32_t>(sampleCnt)   << 8); | 
|  | } |