blob: f7c40c4d42d216b5fe85b9b0b89a30948478d876 [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
Chris Daltonc3318f02019-07-19 14:20:53 -060088void GrMockGpu::querySampleLocations(GrRenderTarget* rt, SkTArray<SkPoint>* sampleLocations) {
89 sampleLocations->reset();
90 int numRemainingSamples = rt->numSamples();
91 while (numRemainingSamples > 0) {
92 // Use standard D3D sample locations.
93 switch (numRemainingSamples) {
94 case 0:
95 case 1:
96 sampleLocations->push_back().set(.5, .5);
97 break;
98 case 2:
99 sampleLocations->push_back().set(.75, .75);
100 sampleLocations->push_back().set(.25, .25);
101 break;
102 case 3:
103 case 4:
104 sampleLocations->push_back().set(.375, .125);
105 sampleLocations->push_back().set(.875, .375);
106 sampleLocations->push_back().set(.125, .625);
107 sampleLocations->push_back().set(.625, .875);
108 break;
109 case 5:
110 case 6:
111 case 7:
112 case 8:
113 sampleLocations->push_back().set(.5625, .3125);
114 sampleLocations->push_back().set(.4375, .6875);
115 sampleLocations->push_back().set(.8125, .5625);
116 sampleLocations->push_back().set(.3125, .1875);
117 sampleLocations->push_back().set(.1875, .8125);
118 sampleLocations->push_back().set(.0625, .4375);
119 sampleLocations->push_back().set(.6875, .4375);
120 sampleLocations->push_back().set(.4375, .0625);
121 break;
122 default:
123 sampleLocations->push_back().set(.5625, .5625);
124 sampleLocations->push_back().set(.4375, .3125);
125 sampleLocations->push_back().set(.3125, .6250);
126 sampleLocations->push_back().set(.2500, .4375);
127 sampleLocations->push_back().set(.1875, .3750);
128 sampleLocations->push_back().set(.6250, .8125);
129 sampleLocations->push_back().set(.8125, .6875);
130 sampleLocations->push_back().set(.6875, .1875);
131 sampleLocations->push_back().set(.3750, .8750);
132 sampleLocations->push_back().set(.5000, .0625);
133 sampleLocations->push_back().set(.2500, .1250);
134 sampleLocations->push_back().set(.1250, .2500);
135 sampleLocations->push_back().set(.0000, .5000);
136 sampleLocations->push_back().set(.4375, .2500);
137 sampleLocations->push_back().set(.8750, .4375);
138 sampleLocations->push_back().set(.0625, .0000);
139 break;
140 }
141 numRemainingSamples = rt->numSamples() - sampleLocations->count();
142 }
143}
144
Brian Salomonf2c2ba92019-07-17 09:59:59 -0400145sk_sp<GrTexture> GrMockGpu::onCreateTexture(const GrSurfaceDesc& desc, GrRenderable renderable,
Brian Salomone8a766b2019-07-19 14:24:36 -0400146 SkBudgeted budgeted, GrProtected isProtected,
147 const GrMipLevel texels[], int mipLevelCount) {
Chris Dalton91ab1552018-04-18 13:24:25 -0600148 if (fMockOptions.fFailTextureAllocations) {
149 return nullptr;
150 }
151
Greg Daniele877dce2019-07-11 10:52:43 -0400152 GrColorType ct = GrPixelConfigToColorType(desc.fConfig);
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400153 if (GrColorType::kUnknown == ct) {
154 return nullptr;
155 }
156
Greg Daniel0fc4d2d2017-10-12 11:23:36 -0400157 GrMipMapsStatus mipMapsStatus = mipLevelCount > 1 ? GrMipMapsStatus::kValid
158 : GrMipMapsStatus::kNotAllocated;
Greg Daniele877dce2019-07-11 10:52:43 -0400159 GrMockTextureInfo texInfo(ct, NextInternalTextureID());
Brian Salomonf2c2ba92019-07-17 09:59:59 -0400160 if (renderable == GrRenderable::kYes) {
Greg Daniele877dce2019-07-11 10:52:43 -0400161 GrMockRenderTargetInfo rtInfo(ct, NextInternalRenderTargetID());
Brian Salomone8a766b2019-07-19 14:24:36 -0400162 return sk_sp<GrTexture>(new GrMockTextureRenderTarget(this, budgeted, desc, isProtected,
163 mipMapsStatus, texInfo, rtInfo));
Brian Salomoncfe910d2017-07-06 16:40:18 -0400164 }
Brian Salomone8a766b2019-07-19 14:24:36 -0400165 return sk_sp<GrTexture>(
166 new GrMockTexture(this, budgeted, desc, isProtected, mipMapsStatus, texInfo));
Brian Salomoncfe910d2017-07-06 16:40:18 -0400167}
168
Brian Salomonbb8dde82019-06-27 10:52:13 -0400169sk_sp<GrTexture> GrMockGpu::onCreateCompressedTexture(int width, int height,
170 SkImage::CompressionType compressionType,
171 SkBudgeted budgeted, const void* data) {
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400172 return nullptr;
Brian Salomonbb8dde82019-06-27 10:52:13 -0400173}
174
Robert Phillipsdd399802019-07-18 12:28:00 +0000175sk_sp<GrTexture> GrMockGpu::onWrapBackendTexture(const GrBackendTexture& tex,
Brian Salomonfa2ebea2019-01-24 15:58:58 -0500176 GrWrapOwnership ownership,
177 GrWrapCacheable wrapType, GrIOType ioType) {
Robert Phillipsdd399802019-07-18 12:28:00 +0000178 GrMockTextureInfo info;
179 SkAssertResult(tex.getMockTextureInfo(&info));
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400180
Greg Daniel4684f822018-03-08 15:27:36 -0500181 GrSurfaceDesc desc;
182 desc.fWidth = tex.width();
183 desc.fHeight = tex.height();
Robert Phillipsdd399802019-07-18 12:28:00 +0000184 desc.fConfig = info.pixelConfig();
Greg Daniel4684f822018-03-08 15:27:36 -0500185
186 GrMipMapsStatus mipMapsStatus = tex.hasMipMaps() ? GrMipMapsStatus::kValid
187 : GrMipMapsStatus::kNotAllocated;
Brian Salomone8a766b2019-07-19 14:24:36 -0400188 auto isProtected = GrProtected(tex.isProtected());
189 return sk_sp<GrTexture>(
190 new GrMockTexture(this, desc, isProtected, mipMapsStatus, info, wrapType, ioType));
Greg Daniel4684f822018-03-08 15:27:36 -0500191}
192
Brian Salomon0c51eea2018-03-09 17:02:09 -0500193sk_sp<GrTexture> GrMockGpu::onWrapRenderableBackendTexture(const GrBackendTexture& tex,
194 int sampleCnt,
Robert Phillips0902c982019-07-16 07:47:56 -0400195 GrColorType colorType,
Brian Salomonaa6ca0a2019-01-24 16:03:07 -0500196 GrWrapOwnership ownership,
197 GrWrapCacheable cacheable) {
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400198 GrMockTextureInfo texInfo;
199 SkAssertResult(tex.getMockTextureInfo(&texInfo));
200
Robert Phillips0902c982019-07-16 07:47:56 -0400201 SkASSERT(colorType == texInfo.fColorType);
Brian Salomon0c51eea2018-03-09 17:02:09 -0500202 GrSurfaceDesc desc;
Brian Salomon0c51eea2018-03-09 17:02:09 -0500203 desc.fWidth = tex.width();
204 desc.fHeight = tex.height();
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400205 desc.fConfig = texInfo.pixelConfig();
Brian Salomon0c51eea2018-03-09 17:02:09 -0500206
207 GrMipMapsStatus mipMapsStatus =
208 tex.hasMipMaps() ? GrMipMapsStatus::kValid : GrMipMapsStatus::kNotAllocated;
209
Brian Salomon0c51eea2018-03-09 17:02:09 -0500210 // The client gave us the texture ID but we supply the render target ID.
Greg Daniele877dce2019-07-11 10:52:43 -0400211 GrMockRenderTargetInfo rtInfo(texInfo.fColorType, NextInternalRenderTargetID());
Brian Salomon0c51eea2018-03-09 17:02:09 -0500212
Brian Salomone8a766b2019-07-19 14:24:36 -0400213 auto isProtected = GrProtected(tex.isProtected());
214 return sk_sp<GrTexture>(new GrMockTextureRenderTarget(this, desc, isProtected, mipMapsStatus,
215 texInfo, rtInfo, cacheable));
Brian Salomon0c51eea2018-03-09 17:02:09 -0500216}
217
Robert Phillipsdd399802019-07-18 12:28:00 +0000218sk_sp<GrRenderTarget> GrMockGpu::onWrapBackendRenderTarget(const GrBackendRenderTarget& rt) {
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400219 GrMockRenderTargetInfo info;
220 SkAssertResult(rt.getMockRenderTargetInfo(&info));
221
Brian Salomon0c51eea2018-03-09 17:02:09 -0500222 GrSurfaceDesc desc;
Brian Salomon0c51eea2018-03-09 17:02:09 -0500223 desc.fWidth = rt.width();
224 desc.fHeight = rt.height();
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400225 desc.fConfig = info.pixelConfig();
Brian Salomon0c51eea2018-03-09 17:02:09 -0500226
Brian Salomone8a766b2019-07-19 14:24:36 -0400227 auto isProtected = GrProtected(rt.isProtected());
Brian Salomon0c51eea2018-03-09 17:02:09 -0500228 return sk_sp<GrRenderTarget>(
Brian Salomone8a766b2019-07-19 14:24:36 -0400229 new GrMockRenderTarget(this, GrMockRenderTarget::kWrapped, desc, isProtected, info));
Brian Salomon0c51eea2018-03-09 17:02:09 -0500230}
231
232sk_sp<GrRenderTarget> GrMockGpu::onWrapBackendTextureAsRenderTarget(const GrBackendTexture& tex,
Robert Phillipsdd399802019-07-18 12:28:00 +0000233 int sampleCnt) {
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400234 GrMockTextureInfo texInfo;
235 SkAssertResult(tex.getMockTextureInfo(&texInfo));
236
Brian Salomon0c51eea2018-03-09 17:02:09 -0500237 GrSurfaceDesc desc;
Brian Salomon0c51eea2018-03-09 17:02:09 -0500238 desc.fWidth = tex.width();
239 desc.fHeight = tex.height();
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400240 desc.fConfig = texInfo.pixelConfig();
Brian Salomon0c51eea2018-03-09 17:02:09 -0500241 desc.fSampleCnt = sampleCnt;
242
Brian Salomon0c51eea2018-03-09 17:02:09 -0500243 // The client gave us the texture ID but we supply the render target ID.
Greg Daniele877dce2019-07-11 10:52:43 -0400244 GrMockRenderTargetInfo rtInfo(texInfo.fColorType, NextInternalRenderTargetID());
Brian Salomon0c51eea2018-03-09 17:02:09 -0500245
Brian Salomone8a766b2019-07-19 14:24:36 -0400246 auto isProtected = GrProtected(tex.isProtected());
Brian Salomon0c51eea2018-03-09 17:02:09 -0500247 return sk_sp<GrRenderTarget>(
Brian Salomone8a766b2019-07-19 14:24:36 -0400248 new GrMockRenderTarget(this, GrMockRenderTarget::kWrapped, desc, isProtected, rtInfo));
Brian Salomon0c51eea2018-03-09 17:02:09 -0500249}
250
Brian Salomondbf70722019-02-07 11:31:24 -0500251sk_sp<GrGpuBuffer> GrMockGpu::onCreateBuffer(size_t sizeInBytes, GrGpuBufferType type,
252 GrAccessPattern accessPattern, const void*) {
253 return sk_sp<GrGpuBuffer>(new GrMockBuffer(this, sizeInBytes, type, accessPattern));
Brian Salomoncfe910d2017-07-06 16:40:18 -0400254}
255
Chris Daltoneffee202019-07-01 22:28:03 -0600256GrStencilAttachment* GrMockGpu::createStencilAttachmentForRenderTarget(
257 const GrRenderTarget* rt, int width, int height, int numStencilSamples) {
258 SkASSERT(numStencilSamples == rt->numSamples());
Brian Salomoncfe910d2017-07-06 16:40:18 -0400259 static constexpr int kBits = 8;
260 fStats.incStencilAttachmentCreates();
Chris Dalton6ce447a2019-06-23 18:07:38 -0600261 return new GrMockStencilAttachment(this, width, height, kBits, rt->numSamples());
Brian Salomoncfe910d2017-07-06 16:40:18 -0400262}
Brian Salomon8fe24272017-07-07 12:56:11 -0400263
Robert Phillipsf0313ee2019-05-21 13:51:11 -0400264GrBackendTexture GrMockGpu::createBackendTexture(int w, int h,
265 const GrBackendFormat& format,
266 GrMipMapped mipMapped,
267 GrRenderable /* renderable */,
268 const void* /* pixels */,
Robert Phillips459b2952019-05-23 09:38:27 -0400269 size_t /* rowBytes */,
Emircan Uysaler23ca4e72019-06-24 10:53:09 -0400270 const SkColor4f* /* color */,
271 GrProtected /* isProtected */) {
Robert Phillips646f6372018-09-25 09:31:10 -0400272
Greg Daniele877dce2019-07-11 10:52:43 -0400273 if (!format.getMockColorType()) {
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400274 return GrBackendTexture(); // invalid;
275 }
276
Greg Daniele877dce2019-07-11 10:52:43 -0400277 if (!this->caps()->isFormatTexturable(*format.getMockColorType(), format)) {
Robert Phillips9dbcdcc2019-05-13 10:40:06 -0400278 return GrBackendTexture(); // invalid
279 }
280
Greg Daniele877dce2019-07-11 10:52:43 -0400281 GrMockTextureInfo info(*format.getMockColorType(), NextExternalTextureID());
Robert Phillips646f6372018-09-25 09:31:10 -0400282
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500283 fOutstandingTestingOnlyTextureIDs.add(info.fID);
Brian Salomon0c51eea2018-03-09 17:02:09 -0500284 return GrBackendTexture(w, h, mipMapped, info);
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500285}
286
Robert Phillipsf0313ee2019-05-21 13:51:11 -0400287void GrMockGpu::deleteBackendTexture(const GrBackendTexture& tex) {
Robert Phillipsf0ced622019-05-16 09:06:25 -0400288 SkASSERT(GrBackendApi::kMock == tex.backend());
289
290 GrMockTextureInfo info;
291 if (tex.getMockTextureInfo(&info)) {
292 fOutstandingTestingOnlyTextureIDs.remove(info.fID);
293 }
294}
295
296#if GR_TEST_UTILS
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500297bool GrMockGpu::isTestingOnlyBackendTexture(const GrBackendTexture& tex) const {
Greg Danielbdf12ad2018-10-12 09:31:11 -0400298 SkASSERT(GrBackendApi::kMock == tex.backend());
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500299
Greg Daniel52e16d92018-04-10 09:34:07 -0400300 GrMockTextureInfo info;
301 if (!tex.getMockTextureInfo(&info)) {
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500302 return false;
303 }
304
Greg Daniel52e16d92018-04-10 09:34:07 -0400305 return fOutstandingTestingOnlyTextureIDs.contains(info.fID);
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500306}
307
Brian Salomon0c51eea2018-03-09 17:02:09 -0500308GrBackendRenderTarget GrMockGpu::createTestingOnlyBackendRenderTarget(int w, int h,
Brian Osman2d010b62018-08-09 10:55:09 -0400309 GrColorType colorType) {
Greg Daniele877dce2019-07-11 10:52:43 -0400310 GrMockRenderTargetInfo info(colorType, NextExternalRenderTargetID());
Brian Salomon0c51eea2018-03-09 17:02:09 -0500311 static constexpr int kSampleCnt = 1;
312 static constexpr int kStencilBits = 8;
Robert Phillipsa5e78be2019-07-09 12:34:38 -0400313 return GrBackendRenderTarget(w, h, kSampleCnt, kStencilBits, info);
Brian Salomonf865b052018-03-09 09:01:53 -0500314}
315
316void GrMockGpu::deleteTestingOnlyBackendRenderTarget(const GrBackendRenderTarget&) {}
317#endif