1 /*
2  * Copyright 2017 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 "GrMockGpu.h"
9 #include "GrMockBuffer.h"
10 #include "GrMockCaps.h"
11 #include "GrMockGpuCommandBuffer.h"
12 #include "GrMockStencilAttachment.h"
13 #include "GrMockTexture.h"
14 #include <atomic>
15 
NextInternalTextureID()16 int GrMockGpu::NextInternalTextureID() {
17     static std::atomic<int> nextID{1};
18     int id;
19     do {
20         id = nextID.fetch_add(1);
21     } while (0 == id);  // Reserve 0 for an invalid ID.
22     return id;
23 }
24 
NextExternalTextureID()25 int GrMockGpu::NextExternalTextureID() {
26     // We use negative ints for the "testing only external textures" so they can easily be
27     // identified when debugging.
28     static std::atomic<int> nextID{-1};
29     return nextID--;
30 }
31 
NextInternalRenderTargetID()32 int GrMockGpu::NextInternalRenderTargetID() {
33     // We start off with large numbers to differentiate from texture IDs, even though they're
34     // technically in a different space.
35     static std::atomic<int> nextID{SK_MaxS32};
36     return nextID--;
37 }
38 
NextExternalRenderTargetID()39 int GrMockGpu::NextExternalRenderTargetID() {
40     // We use large negative ints for the "testing only external render targets" so they can easily
41     // be identified when debugging.
42     static std::atomic<int> nextID{SK_MinS32};
43     return nextID++;
44 }
45 
Make(const GrMockOptions * mockOptions,const GrContextOptions & contextOptions,GrContext * context)46 sk_sp<GrGpu> GrMockGpu::Make(const GrMockOptions* mockOptions,
47                              const GrContextOptions& contextOptions, GrContext* context) {
48     static const GrMockOptions kDefaultOptions = GrMockOptions();
49     if (!mockOptions) {
50         mockOptions = &kDefaultOptions;
51     }
52     return sk_sp<GrGpu>(new GrMockGpu(context, *mockOptions, contextOptions));
53 }
54 
getCommandBuffer(GrRenderTarget * rt,GrSurfaceOrigin origin,const SkRect & bounds,const GrGpuRTCommandBuffer::LoadAndStoreInfo &,const GrGpuRTCommandBuffer::StencilLoadAndStoreInfo &)55 GrGpuRTCommandBuffer* GrMockGpu::getCommandBuffer(
56                                 GrRenderTarget* rt, GrSurfaceOrigin origin, const SkRect& bounds,
57                                 const GrGpuRTCommandBuffer::LoadAndStoreInfo&,
58                                 const GrGpuRTCommandBuffer::StencilLoadAndStoreInfo&) {
59     return new GrMockGpuRTCommandBuffer(this, rt, origin);
60 }
61 
getCommandBuffer(GrTexture * texture,GrSurfaceOrigin origin)62 GrGpuTextureCommandBuffer* GrMockGpu::getCommandBuffer(GrTexture* texture, GrSurfaceOrigin origin) {
63     return new GrMockGpuTextureCommandBuffer(texture, origin);
64 }
65 
submit(GrGpuCommandBuffer * buffer)66 void GrMockGpu::submit(GrGpuCommandBuffer* buffer) {
67     if (buffer->asRTCommandBuffer()) {
68         this->submitCommandBuffer(
69                         static_cast<GrMockGpuRTCommandBuffer*>(buffer->asRTCommandBuffer()));
70     }
71 
72     delete buffer;
73 }
74 
submitCommandBuffer(const GrMockGpuRTCommandBuffer * cmdBuffer)75 void GrMockGpu::submitCommandBuffer(const GrMockGpuRTCommandBuffer* cmdBuffer) {
76     for (int i = 0; i < cmdBuffer->numDraws(); ++i) {
77         fStats.incNumDraws();
78     }
79 }
80 
GrMockGpu(GrContext * context,const GrMockOptions & options,const GrContextOptions & contextOptions)81 GrMockGpu::GrMockGpu(GrContext* context, const GrMockOptions& options,
82                      const GrContextOptions& contextOptions)
83         : INHERITED(context)
84         , fMockOptions(options) {
85     fCaps.reset(new GrMockCaps(contextOptions, options));
86 }
87 
onCreateTexture(const GrSurfaceDesc & desc,SkBudgeted budgeted,const GrMipLevel texels[],int mipLevelCount)88 sk_sp<GrTexture> GrMockGpu::onCreateTexture(const GrSurfaceDesc& desc, SkBudgeted budgeted,
89                                             const GrMipLevel texels[], int mipLevelCount) {
90     if (fMockOptions.fFailTextureAllocations) {
91         return nullptr;
92     }
93 
94     GrMipMapsStatus mipMapsStatus = mipLevelCount > 1 ? GrMipMapsStatus::kValid
95                                                       : GrMipMapsStatus::kNotAllocated;
96     GrMockTextureInfo texInfo;
97     texInfo.fConfig = desc.fConfig;
98     texInfo.fID = NextInternalTextureID();
99     if (desc.fFlags & kRenderTarget_GrSurfaceFlag) {
100         GrMockRenderTargetInfo rtInfo;
101         rtInfo.fConfig = desc.fConfig;
102         rtInfo.fID = NextInternalRenderTargetID();
103         return sk_sp<GrTexture>(new GrMockTextureRenderTarget(this, budgeted, desc, mipMapsStatus,
104                                                               texInfo, rtInfo));
105     }
106     return sk_sp<GrTexture>(new GrMockTexture(this, budgeted, desc, mipMapsStatus, texInfo));
107 }
108 
onWrapBackendTexture(const GrBackendTexture & tex,GrWrapOwnership ownership,GrWrapCacheable wrapType,GrIOType ioType)109 sk_sp<GrTexture> GrMockGpu::onWrapBackendTexture(const GrBackendTexture& tex,
110                                                  GrWrapOwnership ownership,
111                                                  GrWrapCacheable wrapType, GrIOType ioType) {
112     GrSurfaceDesc desc;
113     desc.fWidth = tex.width();
114     desc.fHeight = tex.height();
115 
116     GrMockTextureInfo info;
117     SkAssertResult(tex.getMockTextureInfo(&info));
118     desc.fConfig = info.fConfig;
119 
120     GrMipMapsStatus mipMapsStatus = tex.hasMipMaps() ? GrMipMapsStatus::kValid
121                                                      : GrMipMapsStatus::kNotAllocated;
122 
123     return sk_sp<GrTexture>(new GrMockTexture(this, desc, mipMapsStatus, info, wrapType, ioType));
124 }
125 
onWrapRenderableBackendTexture(const GrBackendTexture & tex,int sampleCnt,GrWrapOwnership ownership,GrWrapCacheable cacheable)126 sk_sp<GrTexture> GrMockGpu::onWrapRenderableBackendTexture(const GrBackendTexture& tex,
127                                                            int sampleCnt,
128                                                            GrWrapOwnership ownership,
129                                                            GrWrapCacheable cacheable) {
130     GrSurfaceDesc desc;
131     desc.fFlags = kRenderTarget_GrSurfaceFlag;
132     desc.fWidth = tex.width();
133     desc.fHeight = tex.height();
134 
135     GrMockTextureInfo texInfo;
136     SkAssertResult(tex.getMockTextureInfo(&texInfo));
137     desc.fConfig = texInfo.fConfig;
138 
139     GrMipMapsStatus mipMapsStatus =
140             tex.hasMipMaps() ? GrMipMapsStatus::kValid : GrMipMapsStatus::kNotAllocated;
141 
142     GrMockRenderTargetInfo rtInfo;
143     rtInfo.fConfig = texInfo.fConfig;
144     // The client gave us the texture ID but we supply the render target ID.
145     rtInfo.fID = NextInternalRenderTargetID();
146 
147     return sk_sp<GrTexture>(
148             new GrMockTextureRenderTarget(this, desc, mipMapsStatus, texInfo, rtInfo, cacheable));
149 }
150 
onWrapBackendRenderTarget(const GrBackendRenderTarget & rt)151 sk_sp<GrRenderTarget> GrMockGpu::onWrapBackendRenderTarget(const GrBackendRenderTarget& rt) {
152     GrSurfaceDesc desc;
153     desc.fFlags = kRenderTarget_GrSurfaceFlag;
154     desc.fWidth = rt.width();
155     desc.fHeight = rt.height();
156 
157     GrMockRenderTargetInfo info;
158     SkAssertResult(rt.getMockRenderTargetInfo(&info));
159     desc.fConfig = info.fConfig;
160 
161     return sk_sp<GrRenderTarget>(
162             new GrMockRenderTarget(this, GrMockRenderTarget::kWrapped, desc, info));
163 }
164 
onWrapBackendTextureAsRenderTarget(const GrBackendTexture & tex,int sampleCnt)165 sk_sp<GrRenderTarget> GrMockGpu::onWrapBackendTextureAsRenderTarget(const GrBackendTexture& tex,
166                                                                     int sampleCnt) {
167     GrSurfaceDesc desc;
168     desc.fFlags = kRenderTarget_GrSurfaceFlag;
169     desc.fWidth = tex.width();
170     desc.fHeight = tex.height();
171 
172     GrMockTextureInfo texInfo;
173     SkAssertResult(tex.getMockTextureInfo(&texInfo));
174     desc.fConfig = texInfo.fConfig;
175     desc.fSampleCnt = sampleCnt;
176 
177     GrMockRenderTargetInfo rtInfo;
178     rtInfo.fConfig = texInfo.fConfig;
179     // The client gave us the texture ID but we supply the render target ID.
180     rtInfo.fID = NextInternalRenderTargetID();
181 
182     return sk_sp<GrRenderTarget>(
183             new GrMockRenderTarget(this, GrMockRenderTarget::kWrapped, desc, rtInfo));
184 }
185 
onCreateBuffer(size_t sizeInBytes,GrGpuBufferType type,GrAccessPattern accessPattern,const void *)186 sk_sp<GrGpuBuffer> GrMockGpu::onCreateBuffer(size_t sizeInBytes, GrGpuBufferType type,
187                                              GrAccessPattern accessPattern, const void*) {
188     return sk_sp<GrGpuBuffer>(new GrMockBuffer(this, sizeInBytes, type, accessPattern));
189 }
190 
createStencilAttachmentForRenderTarget(const GrRenderTarget * rt,int width,int height)191 GrStencilAttachment* GrMockGpu::createStencilAttachmentForRenderTarget(const GrRenderTarget* rt,
192                                                                        int width,
193                                                                        int height) {
194     static constexpr int kBits = 8;
195     fStats.incStencilAttachmentCreates();
196     return new GrMockStencilAttachment(this, width, height, kBits, rt->numColorSamples());
197 }
198 
199 #if GR_TEST_UTILS
createTestingOnlyBackendTexture(const void * pixels,int w,int h,GrColorType colorType,bool isRT,GrMipMapped mipMapped,size_t rowBytes)200 GrBackendTexture GrMockGpu::createTestingOnlyBackendTexture(const void* pixels, int w, int h,
201                                                             GrColorType colorType, bool isRT,
202                                                             GrMipMapped mipMapped,
203                                                             size_t rowBytes) {
204 
205     GrPixelConfig config = GrColorTypeToPixelConfig(colorType, GrSRGBEncoded::kNo);
206     if (!this->caps()->isConfigTexturable(config)) {
207         return GrBackendTexture();  // invalid
208     }
209 
210     GrMockTextureInfo info;
211     info.fConfig = config;
212     info.fID = NextExternalTextureID();
213     fOutstandingTestingOnlyTextureIDs.add(info.fID);
214     return GrBackendTexture(w, h, mipMapped, info);
215 }
216 
isTestingOnlyBackendTexture(const GrBackendTexture & tex) const217 bool GrMockGpu::isTestingOnlyBackendTexture(const GrBackendTexture& tex) const {
218     SkASSERT(GrBackendApi::kMock == tex.backend());
219 
220     GrMockTextureInfo info;
221     if (!tex.getMockTextureInfo(&info)) {
222         return false;
223     }
224 
225     return fOutstandingTestingOnlyTextureIDs.contains(info.fID);
226 }
227 
deleteTestingOnlyBackendTexture(const GrBackendTexture & tex)228 void GrMockGpu::deleteTestingOnlyBackendTexture(const GrBackendTexture& tex) {
229     SkASSERT(GrBackendApi::kMock == tex.backend());
230 
231     GrMockTextureInfo info;
232     if (tex.getMockTextureInfo(&info)) {
233         fOutstandingTestingOnlyTextureIDs.remove(info.fID);
234     }
235 }
236 
createTestingOnlyBackendRenderTarget(int w,int h,GrColorType colorType)237 GrBackendRenderTarget GrMockGpu::createTestingOnlyBackendRenderTarget(int w, int h,
238                                                                       GrColorType colorType) {
239     auto config = GrColorTypeToPixelConfig(colorType, GrSRGBEncoded::kNo);
240     if (kUnknown_GrPixelConfig == config) {
241         return {};
242     }
243     GrMockRenderTargetInfo info = {config, NextExternalRenderTargetID()};
244     static constexpr int kSampleCnt = 1;
245     static constexpr int kStencilBits = 8;
246     return {w, h, kSampleCnt, kStencilBits, info};
247 }
248 
deleteTestingOnlyBackendRenderTarget(const GrBackendRenderTarget &)249 void GrMockGpu::deleteTestingOnlyBackendRenderTarget(const GrBackendRenderTarget&) {}
250 #endif
251