blob: 038c5230943df7714f1e5ee046ecaad89baaebae [file] [log] [blame]
Brian Salomoncfe910d2017-07-06 16:40:18 -04001/*
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/mock/GrMockBuffer.h"
9#include "src/gpu/mock/GrMockCaps.h"
10#include "src/gpu/mock/GrMockGpu.h"
11#include "src/gpu/mock/GrMockGpuCommandBuffer.h"
12#include "src/gpu/mock/GrMockStencilAttachment.h"
13#include "src/gpu/mock/GrMockTexture.h"
Mike Kleinde2244c2018-12-04 11:16:08 -050014#include <atomic>
Brian Salomoncfe910d2017-07-06 16:40:18 -040015
Brian Salomon8fe24272017-07-07 12:56:11 -040016int GrMockGpu::NextInternalTextureID() {
Mike Kleinde2244c2018-12-04 11:16:08 -050017 static std::atomic<int> nextID{1};
Chris Dalton351e80c2019-01-06 22:51:00 -070018 int id;
19 do {
20 id = nextID.fetch_add(1);
21 } while (0 == id); // Reserve 0 for an invalid ID.
22 return id;
Brian Salomon8fe24272017-07-07 12:56:11 -040023}
24
25int GrMockGpu::NextExternalTextureID() {
26 // We use negative ints for the "testing only external textures" so they can easily be
27 // identified when debugging.
Mike Kleinde2244c2018-12-04 11:16:08 -050028 static std::atomic<int> nextID{-1};
29 return nextID--;
Brian Salomon8fe24272017-07-07 12:56:11 -040030}
31
Brian Salomon0c51eea2018-03-09 17:02:09 -050032int GrMockGpu::NextInternalRenderTargetID() {
Mike Kleinde2244c2018-12-04 11:16:08 -050033 // We start off with large numbers to differentiate from texture IDs, even though they're
Brian Salomon0c51eea2018-03-09 17:02:09 -050034 // technically in a different space.
Mike Kleinde2244c2018-12-04 11:16:08 -050035 static std::atomic<int> nextID{SK_MaxS32};
36 return nextID--;
Brian Salomon0c51eea2018-03-09 17:02:09 -050037}
38
39int GrMockGpu::NextExternalRenderTargetID() {
40 // We use large negative ints for the "testing only external render targets" so they can easily
41 // be identified when debugging.
Mike Kleinde2244c2018-12-04 11:16:08 -050042 static std::atomic<int> nextID{SK_MinS32};
43 return nextID++;
Brian Salomon0c51eea2018-03-09 17:02:09 -050044}
45
Brian Salomon384fab42017-12-07 12:33:05 -050046sk_sp<GrGpu> GrMockGpu::Make(const GrMockOptions* mockOptions,
47 const GrContextOptions& contextOptions, GrContext* context) {
Greg Daniel02611d92017-07-25 10:05:01 -040048 static const GrMockOptions kDefaultOptions = GrMockOptions();
49 if (!mockOptions) {
50 mockOptions = &kDefaultOptions;
51 }
Brian Salomon384fab42017-12-07 12:33:05 -050052 return sk_sp<GrGpu>(new GrMockGpu(context, *mockOptions, contextOptions));
Greg Daniel02611d92017-07-25 10:05:01 -040053}
54
Robert Phillips5b5d84c2018-08-09 15:12:18 -040055GrGpuRTCommandBuffer* GrMockGpu::getCommandBuffer(
Ethan Nicholas56d19a52018-10-15 11:26:20 -040056 GrRenderTarget* rt, GrSurfaceOrigin origin, const SkRect& bounds,
Robert Phillips5b5d84c2018-08-09 15:12:18 -040057 const GrGpuRTCommandBuffer::LoadAndStoreInfo&,
58 const GrGpuRTCommandBuffer::StencilLoadAndStoreInfo&) {
Greg Daniel500d58b2017-08-24 15:59:33 -040059 return new GrMockGpuRTCommandBuffer(this, rt, origin);
Brian Salomoncfe910d2017-07-06 16:40:18 -040060}
61
Robert Phillips5b5d84c2018-08-09 15:12:18 -040062GrGpuTextureCommandBuffer* GrMockGpu::getCommandBuffer(GrTexture* texture, GrSurfaceOrigin origin) {
Greg Daniel500d58b2017-08-24 15:59:33 -040063 return new GrMockGpuTextureCommandBuffer(texture, origin);
64}
65
Robert Phillips5b5d84c2018-08-09 15:12:18 -040066void GrMockGpu::submit(GrGpuCommandBuffer* buffer) {
67 if (buffer->asRTCommandBuffer()) {
68 this->submitCommandBuffer(
69 static_cast<GrMockGpuRTCommandBuffer*>(buffer->asRTCommandBuffer()));
70 }
71
72 delete buffer;
73}
Greg Daniel500d58b2017-08-24 15:59:33 -040074
75void GrMockGpu::submitCommandBuffer(const GrMockGpuRTCommandBuffer* cmdBuffer) {
Brian Salomoncfe910d2017-07-06 16:40:18 -040076 for (int i = 0; i < cmdBuffer->numDraws(); ++i) {
77 fStats.incNumDraws();
78 }
79}
80
81GrMockGpu::GrMockGpu(GrContext* context, const GrMockOptions& options,
82 const GrContextOptions& contextOptions)
Chris Dalton91ab1552018-04-18 13:24:25 -060083 : INHERITED(context)
84 , fMockOptions(options) {
Brian Salomoncfe910d2017-07-06 16:40:18 -040085 fCaps.reset(new GrMockCaps(contextOptions, options));
86}
87
88sk_sp<GrTexture> GrMockGpu::onCreateTexture(const GrSurfaceDesc& desc, SkBudgeted budgeted,
Brian Salomon58389b92018-03-07 13:01:25 -050089 const GrMipLevel texels[], int mipLevelCount) {
Chris Dalton91ab1552018-04-18 13:24:25 -060090 if (fMockOptions.fFailTextureAllocations) {
91 return nullptr;
92 }
93
Greg Daniel0fc4d2d2017-10-12 11:23:36 -040094 GrMipMapsStatus mipMapsStatus = mipLevelCount > 1 ? GrMipMapsStatus::kValid
95 : GrMipMapsStatus::kNotAllocated;
Brian Salomon0c51eea2018-03-09 17:02:09 -050096 GrMockTextureInfo texInfo;
97 texInfo.fConfig = desc.fConfig;
98 texInfo.fID = NextInternalTextureID();
Brian Salomoncfe910d2017-07-06 16:40:18 -040099 if (desc.fFlags & kRenderTarget_GrSurfaceFlag) {
Brian Salomon0c51eea2018-03-09 17:02:09 -0500100 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));
Brian Salomoncfe910d2017-07-06 16:40:18 -0400105 }
Brian Salomon0c51eea2018-03-09 17:02:09 -0500106 return sk_sp<GrTexture>(new GrMockTexture(this, budgeted, desc, mipMapsStatus, texInfo));
Brian Salomoncfe910d2017-07-06 16:40:18 -0400107}
108
Greg Daniel4684f822018-03-08 15:27:36 -0500109sk_sp<GrTexture> GrMockGpu::onWrapBackendTexture(const GrBackendTexture& tex,
Brian Salomonfa2ebea2019-01-24 15:58:58 -0500110 GrWrapOwnership ownership,
111 GrWrapCacheable wrapType, GrIOType ioType) {
Greg Daniel4684f822018-03-08 15:27:36 -0500112 GrSurfaceDesc desc;
113 desc.fWidth = tex.width();
114 desc.fHeight = tex.height();
Greg Daniel52e16d92018-04-10 09:34:07 -0400115
116 GrMockTextureInfo info;
117 SkAssertResult(tex.getMockTextureInfo(&info));
Greg Daniel4684f822018-03-08 15:27:36 -0500118 desc.fConfig = info.fConfig;
119
120 GrMipMapsStatus mipMapsStatus = tex.hasMipMaps() ? GrMipMapsStatus::kValid
121 : GrMipMapsStatus::kNotAllocated;
122
Brian Salomonfa2ebea2019-01-24 15:58:58 -0500123 return sk_sp<GrTexture>(new GrMockTexture(this, desc, mipMapsStatus, info, wrapType, ioType));
Greg Daniel4684f822018-03-08 15:27:36 -0500124}
125
Brian Salomon0c51eea2018-03-09 17:02:09 -0500126sk_sp<GrTexture> GrMockGpu::onWrapRenderableBackendTexture(const GrBackendTexture& tex,
127 int sampleCnt,
Brian Salomonaa6ca0a2019-01-24 16:03:07 -0500128 GrWrapOwnership ownership,
129 GrWrapCacheable cacheable) {
Brian Salomon0c51eea2018-03-09 17:02:09 -0500130 GrSurfaceDesc desc;
131 desc.fFlags = kRenderTarget_GrSurfaceFlag;
132 desc.fWidth = tex.width();
133 desc.fHeight = tex.height();
Greg Daniel52e16d92018-04-10 09:34:07 -0400134
135 GrMockTextureInfo texInfo;
136 SkAssertResult(tex.getMockTextureInfo(&texInfo));
Brian Salomon0c51eea2018-03-09 17:02:09 -0500137 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>(
Brian Salomonaa6ca0a2019-01-24 16:03:07 -0500148 new GrMockTextureRenderTarget(this, desc, mipMapsStatus, texInfo, rtInfo, cacheable));
Brian Salomon0c51eea2018-03-09 17:02:09 -0500149}
150
151sk_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();
Greg Daniel323fbcf2018-04-10 13:46:30 -0400156
157 GrMockRenderTargetInfo info;
158 SkAssertResult(rt.getMockRenderTargetInfo(&info));
Brian Salomon0c51eea2018-03-09 17:02:09 -0500159 desc.fConfig = info.fConfig;
160
161 return sk_sp<GrRenderTarget>(
162 new GrMockRenderTarget(this, GrMockRenderTarget::kWrapped, desc, info));
163}
164
165sk_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();
Greg Daniel52e16d92018-04-10 09:34:07 -0400171
172 GrMockTextureInfo texInfo;
173 SkAssertResult(tex.getMockTextureInfo(&texInfo));
Brian Salomon0c51eea2018-03-09 17:02:09 -0500174 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
Brian Salomondbf70722019-02-07 11:31:24 -0500186sk_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));
Brian Salomoncfe910d2017-07-06 16:40:18 -0400189}
190
191GrStencilAttachment* 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}
Brian Salomon8fe24272017-07-07 12:56:11 -0400198
Robert Phillips9dbcdcc2019-05-13 10:40:06 -0400199GrBackendTexture GrMockGpu::createTestingOnlyBackendTexture(int w, int h,
200 const GrBackendFormat& format,
Robert Phillips646f6372018-09-25 09:31:10 -0400201 GrMipMapped mipMapped,
Robert Phillips9dbcdcc2019-05-13 10:40:06 -0400202 GrRenderable renderable,
203 const void* pixels,
Robert Phillips646f6372018-09-25 09:31:10 -0400204 size_t rowBytes) {
205
Robert Phillips9dbcdcc2019-05-13 10:40:06 -0400206 const GrPixelConfig* pixelConfig = format.getMockFormat();
207 if (!pixelConfig) {
208 return GrBackendTexture(); // invalid
209 }
210
211 if (!this->caps()->isConfigTexturable(*pixelConfig)) {
Robert Phillips646f6372018-09-25 09:31:10 -0400212 return GrBackendTexture(); // invalid
213 }
214
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500215 GrMockTextureInfo info;
Robert Phillips9dbcdcc2019-05-13 10:40:06 -0400216 info.fConfig = *pixelConfig;
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500217 info.fID = NextExternalTextureID();
218 fOutstandingTestingOnlyTextureIDs.add(info.fID);
Brian Salomon0c51eea2018-03-09 17:02:09 -0500219 return GrBackendTexture(w, h, mipMapped, info);
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500220}
221
Robert Phillipsf0ced622019-05-16 09:06:25 -0400222void GrMockGpu::deleteTestingOnlyBackendTexture(const GrBackendTexture& tex) {
223 SkASSERT(GrBackendApi::kMock == tex.backend());
224
225 GrMockTextureInfo info;
226 if (tex.getMockTextureInfo(&info)) {
227 fOutstandingTestingOnlyTextureIDs.remove(info.fID);
228 }
229}
230
231#if GR_TEST_UTILS
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500232bool GrMockGpu::isTestingOnlyBackendTexture(const GrBackendTexture& tex) const {
Greg Danielbdf12ad2018-10-12 09:31:11 -0400233 SkASSERT(GrBackendApi::kMock == tex.backend());
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500234
Greg Daniel52e16d92018-04-10 09:34:07 -0400235 GrMockTextureInfo info;
236 if (!tex.getMockTextureInfo(&info)) {
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500237 return false;
238 }
239
Greg Daniel52e16d92018-04-10 09:34:07 -0400240 return fOutstandingTestingOnlyTextureIDs.contains(info.fID);
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500241}
242
Brian Salomon0c51eea2018-03-09 17:02:09 -0500243GrBackendRenderTarget GrMockGpu::createTestingOnlyBackendRenderTarget(int w, int h,
Brian Osman2d010b62018-08-09 10:55:09 -0400244 GrColorType colorType) {
245 auto config = GrColorTypeToPixelConfig(colorType, GrSRGBEncoded::kNo);
Brian Salomon0c51eea2018-03-09 17:02:09 -0500246 if (kUnknown_GrPixelConfig == config) {
247 return {};
248 }
249 GrMockRenderTargetInfo info = {config, NextExternalRenderTargetID()};
250 static constexpr int kSampleCnt = 1;
251 static constexpr int kStencilBits = 8;
252 return {w, h, kSampleCnt, kStencilBits, info};
Brian Salomonf865b052018-03-09 09:01:53 -0500253}
254
255void GrMockGpu::deleteTestingOnlyBackendRenderTarget(const GrBackendRenderTarget&) {}
256#endif