/* * Copyright 2017 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "GrMockGpu.h" #include "GrMockBuffer.h" #include "GrMockCaps.h" #include "GrMockGpuCommandBuffer.h" #include "GrMockStencilAttachment.h" #include "GrMockTexture.h" #include int GrMockGpu::NextInternalTextureID() { static std::atomic nextID{1}; int id; do { id = nextID.fetch_add(1); } while (0 == id); // Reserve 0 for an invalid ID. return id; } int GrMockGpu::NextExternalTextureID() { // We use negative ints for the "testing only external textures" so they can easily be // identified when debugging. static std::atomic nextID{-1}; return nextID--; } int GrMockGpu::NextInternalRenderTargetID() { // We start off with large numbers to differentiate from texture IDs, even though they're // technically in a different space. static std::atomic nextID{SK_MaxS32}; return nextID--; } int GrMockGpu::NextExternalRenderTargetID() { // We use large negative ints for the "testing only external render targets" so they can easily // be identified when debugging. static std::atomic nextID{SK_MinS32}; return nextID++; } sk_sp GrMockGpu::Make(const GrMockOptions* mockOptions, const GrContextOptions& contextOptions, GrContext* context) { static const GrMockOptions kDefaultOptions = GrMockOptions(); if (!mockOptions) { mockOptions = &kDefaultOptions; } return sk_sp(new GrMockGpu(context, *mockOptions, contextOptions)); } GrGpuRTCommandBuffer* GrMockGpu::getCommandBuffer( GrRenderTarget* rt, GrSurfaceOrigin origin, const SkRect& bounds, const GrGpuRTCommandBuffer::LoadAndStoreInfo&, const GrGpuRTCommandBuffer::StencilLoadAndStoreInfo&) { return new GrMockGpuRTCommandBuffer(this, rt, origin); } GrGpuTextureCommandBuffer* GrMockGpu::getCommandBuffer(GrTexture* texture, GrSurfaceOrigin origin) { return new GrMockGpuTextureCommandBuffer(texture, origin); } void GrMockGpu::submit(GrGpuCommandBuffer* buffer) { if (buffer->asRTCommandBuffer()) { this->submitCommandBuffer( static_cast(buffer->asRTCommandBuffer())); } delete buffer; } void GrMockGpu::submitCommandBuffer(const GrMockGpuRTCommandBuffer* cmdBuffer) { for (int i = 0; i < cmdBuffer->numDraws(); ++i) { fStats.incNumDraws(); } } GrMockGpu::GrMockGpu(GrContext* context, const GrMockOptions& options, const GrContextOptions& contextOptions) : INHERITED(context) , fMockOptions(options) { fCaps.reset(new GrMockCaps(contextOptions, options)); } sk_sp GrMockGpu::onCreateTexture(const GrSurfaceDesc& desc, SkBudgeted budgeted, const GrMipLevel texels[], int mipLevelCount) { if (fMockOptions.fFailTextureAllocations) { return nullptr; } GrMipMapsStatus mipMapsStatus = mipLevelCount > 1 ? GrMipMapsStatus::kValid : GrMipMapsStatus::kNotAllocated; GrMockTextureInfo texInfo; texInfo.fConfig = desc.fConfig; texInfo.fID = NextInternalTextureID(); if (desc.fFlags & kRenderTarget_GrSurfaceFlag) { GrMockRenderTargetInfo rtInfo; rtInfo.fConfig = desc.fConfig; rtInfo.fID = NextInternalRenderTargetID(); return sk_sp(new GrMockTextureRenderTarget(this, budgeted, desc, mipMapsStatus, texInfo, rtInfo)); } return sk_sp(new GrMockTexture(this, budgeted, desc, mipMapsStatus, texInfo)); } sk_sp GrMockGpu::onWrapBackendTexture(const GrBackendTexture& tex, GrWrapOwnership ownership, GrWrapCacheable wrapType, GrIOType ioType) { GrSurfaceDesc desc; desc.fWidth = tex.width(); desc.fHeight = tex.height(); GrMockTextureInfo info; SkAssertResult(tex.getMockTextureInfo(&info)); desc.fConfig = info.fConfig; GrMipMapsStatus mipMapsStatus = tex.hasMipMaps() ? GrMipMapsStatus::kValid : GrMipMapsStatus::kNotAllocated; return sk_sp(new GrMockTexture(this, desc, mipMapsStatus, info, wrapType, ioType)); } sk_sp GrMockGpu::onWrapRenderableBackendTexture(const GrBackendTexture& tex, int sampleCnt, GrWrapOwnership ownership, GrWrapCacheable cacheable) { GrSurfaceDesc desc; desc.fFlags = kRenderTarget_GrSurfaceFlag; desc.fWidth = tex.width(); desc.fHeight = tex.height(); GrMockTextureInfo texInfo; SkAssertResult(tex.getMockTextureInfo(&texInfo)); desc.fConfig = texInfo.fConfig; GrMipMapsStatus mipMapsStatus = tex.hasMipMaps() ? GrMipMapsStatus::kValid : GrMipMapsStatus::kNotAllocated; GrMockRenderTargetInfo rtInfo; rtInfo.fConfig = texInfo.fConfig; // The client gave us the texture ID but we supply the render target ID. rtInfo.fID = NextInternalRenderTargetID(); return sk_sp( new GrMockTextureRenderTarget(this, desc, mipMapsStatus, texInfo, rtInfo, cacheable)); } sk_sp GrMockGpu::onWrapBackendRenderTarget(const GrBackendRenderTarget& rt) { GrSurfaceDesc desc; desc.fFlags = kRenderTarget_GrSurfaceFlag; desc.fWidth = rt.width(); desc.fHeight = rt.height(); GrMockRenderTargetInfo info; SkAssertResult(rt.getMockRenderTargetInfo(&info)); desc.fConfig = info.fConfig; return sk_sp( new GrMockRenderTarget(this, GrMockRenderTarget::kWrapped, desc, info)); } sk_sp GrMockGpu::onWrapBackendTextureAsRenderTarget(const GrBackendTexture& tex, int sampleCnt) { GrSurfaceDesc desc; desc.fFlags = kRenderTarget_GrSurfaceFlag; desc.fWidth = tex.width(); desc.fHeight = tex.height(); GrMockTextureInfo texInfo; SkAssertResult(tex.getMockTextureInfo(&texInfo)); desc.fConfig = texInfo.fConfig; desc.fSampleCnt = sampleCnt; GrMockRenderTargetInfo rtInfo; rtInfo.fConfig = texInfo.fConfig; // The client gave us the texture ID but we supply the render target ID. rtInfo.fID = NextInternalRenderTargetID(); return sk_sp( new GrMockRenderTarget(this, GrMockRenderTarget::kWrapped, desc, rtInfo)); } sk_sp GrMockGpu::onCreateBuffer(size_t sizeInBytes, GrBufferType type, GrAccessPattern accessPattern, const void*) { return sk_sp(new GrMockBuffer(this, sizeInBytes, type, accessPattern)); } GrStencilAttachment* GrMockGpu::createStencilAttachmentForRenderTarget(const GrRenderTarget* rt, int width, int height) { static constexpr int kBits = 8; fStats.incStencilAttachmentCreates(); return new GrMockStencilAttachment(this, width, height, kBits, rt->numColorSamples()); } #if GR_TEST_UTILS GrBackendTexture GrMockGpu::createTestingOnlyBackendTexture(const void* pixels, int w, int h, GrColorType colorType, bool isRT, GrMipMapped mipMapped, size_t rowBytes) { GrPixelConfig config = GrColorTypeToPixelConfig(colorType, GrSRGBEncoded::kNo); if (!this->caps()->isConfigTexturable(config)) { return GrBackendTexture(); // invalid } GrMockTextureInfo info; info.fConfig = config; info.fID = NextExternalTextureID(); fOutstandingTestingOnlyTextureIDs.add(info.fID); return GrBackendTexture(w, h, mipMapped, info); } bool GrMockGpu::isTestingOnlyBackendTexture(const GrBackendTexture& tex) const { SkASSERT(GrBackendApi::kMock == tex.backend()); GrMockTextureInfo info; if (!tex.getMockTextureInfo(&info)) { return false; } return fOutstandingTestingOnlyTextureIDs.contains(info.fID); } void GrMockGpu::deleteTestingOnlyBackendTexture(const GrBackendTexture& tex) { SkASSERT(GrBackendApi::kMock == tex.backend()); GrMockTextureInfo info; if (tex.getMockTextureInfo(&info)) { fOutstandingTestingOnlyTextureIDs.remove(info.fID); } } GrBackendRenderTarget GrMockGpu::createTestingOnlyBackendRenderTarget(int w, int h, GrColorType colorType) { auto config = GrColorTypeToPixelConfig(colorType, GrSRGBEncoded::kNo); if (kUnknown_GrPixelConfig == config) { return {}; } GrMockRenderTargetInfo info = {config, NextExternalRenderTargetID()}; static constexpr int kSampleCnt = 1; static constexpr int kStencilBits = 8; return {w, h, kSampleCnt, kStencilBits, info}; } void GrMockGpu::deleteTestingOnlyBackendRenderTarget(const GrBackendRenderTarget&) {} #endif