blob: e4d9675bff246927dc48e4e48d76b047906a2229 [file] [log] [blame]
bsalomon@google.coma8e686e2011-08-16 15:45:58 +00001
2/*
3 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
bsalomon@google.comd4726202012-08-03 14:34:46 +00009// This is a GPU-backend specific test. It relies on static intializers to work
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +000010
bsalomon@google.com2a48c3a2012-08-03 14:54:45 +000011#include "SkTypes.h"
12
13#if SK_SUPPORT_GPU && SK_ALLOW_STATIC_GLOBAL_INITIALIZERS
14
bsalomon@google.com396e61f2012-10-25 19:00:29 +000015#include "GrBackendEffectFactory.h"
bsalomon@google.com67b915d2013-02-04 16:13:32 +000016#include "GrContextFactory.h"
egdanielc0648242014-09-22 13:17:02 -070017#include "GrOptDrawState.h"
bsalomon@google.coma04e8e82012-08-27 12:53:13 +000018#include "effects/GrConfigConversionEffect.h"
kkinnunenec56e452014-08-25 22:21:16 -070019#include "gl/GrGLPathRendering.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000020#include "gl/GrGpuGL.h"
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000021#include "SkChecksum.h"
tfarina@chromium.org223137f2012-11-21 22:38:36 +000022#include "SkRandom.h"
bsalomon@google.comc3841b92012-08-02 18:11:43 +000023#include "Test.h"
24
egdaniela7dc0a82014-09-17 08:25:05 -070025static void get_stage_stats(const GrEffectStage stage, bool* readsDst,
26 bool* readsFragPosition, bool* requiresVertexShader) {
27 if (stage.getEffect()->willReadDstColor()) {
28 *readsDst = true;
29 }
30 if (stage.getEffect()->willReadFragmentPosition()) {
31 *readsFragPosition = true;
32 }
33 if (stage.getEffect()->requiresVertexShader()) {
34 *requiresVertexShader = true;
35 }
36}
37
bsalomon848faf02014-07-11 10:01:02 -070038bool GrGLProgramDesc::setRandom(SkRandom* random,
egdanielae444962014-09-22 12:29:52 -070039 GrGpuGL* gpu,
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000040 const GrRenderTarget* dstRenderTarget,
41 const GrTexture* dstCopyTexture,
joshualittbd769d02014-09-04 08:56:46 -070042 const GrEffectStage* geometryProcessor,
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000043 const GrEffectStage* stages[],
44 int numColorStages,
45 int numCoverageStages,
egdanielae444962014-09-22 12:29:52 -070046 int currAttribIndex,
47 GrGpu::DrawType drawType) {
48 bool isPathRendering = GrGpu::IsPathRenderingDrawType(drawType);
49 bool useLocalCoords = !isPathRendering &&
50 random->nextBool() &&
51 currAttribIndex < GrDrawState::kMaxVertexAttribCnt;
bsalomon848faf02014-07-11 10:01:02 -070052
53 int numStages = numColorStages + numCoverageStages;
54 fKey.reset();
55
bsalomon929f29a2014-07-17 07:55:11 -070056 GR_STATIC_ASSERT(0 == kEffectKeyOffsetsAndLengthOffset % sizeof(uint32_t));
bsalomon848faf02014-07-11 10:01:02 -070057
58 // Make room for everything up to and including the array of offsets to effect keys.
joshualittbd769d02014-09-04 08:56:46 -070059 fKey.push_back_n(kEffectKeyOffsetsAndLengthOffset + 2 * sizeof(uint16_t) * (numStages +
60 (geometryProcessor ? 1 : 0)));
bsalomon848faf02014-07-11 10:01:02 -070061
62 bool dstRead = false;
63 bool fragPos = false;
bsalomon49f085d2014-09-05 13:34:00 -070064 bool vertexShader = SkToBool(geometryProcessor);
joshualittbd769d02014-09-04 08:56:46 -070065 int offset = 0;
bsalomon49f085d2014-09-05 13:34:00 -070066 if (geometryProcessor) {
joshualittbd769d02014-09-04 08:56:46 -070067 const GrEffectStage* stage = geometryProcessor;
bsalomon929f29a2014-07-17 07:55:11 -070068 uint16_t* offsetAndSize = reinterpret_cast<uint16_t*>(fKey.begin() +
69 kEffectKeyOffsetsAndLengthOffset +
joshualittbd769d02014-09-04 08:56:46 -070070 offset * 2 * sizeof(uint16_t));
bsalomon929f29a2014-07-17 07:55:11 -070071 uint32_t effectKeyOffset = fKey.count();
72 if (effectKeyOffset > SK_MaxU16) {
bsalomonc0ea3982014-07-15 19:41:17 -070073 fKey.reset();
74 return false;
bsalomon848faf02014-07-11 10:01:02 -070075 }
bsalomon929f29a2014-07-17 07:55:11 -070076 GrEffectKeyBuilder b(&fKey);
77 uint16_t effectKeySize;
egdaniela7dc0a82014-09-17 08:25:05 -070078 if (!GetEffectKey(*stage, gpu->glCaps(), useLocalCoords, &b, &effectKeySize)) {
joshualittbd769d02014-09-04 08:56:46 -070079 fKey.reset();
80 return false;
81 }
egdaniela7dc0a82014-09-17 08:25:05 -070082 get_stage_stats(*stage, &dstRead, &fragPos, &vertexShader);
joshualittbd769d02014-09-04 08:56:46 -070083 offsetAndSize[0] = effectKeyOffset;
84 offsetAndSize[1] = effectKeySize;
85 offset++;
86 }
87
88 for (int s = 0; s < numStages; ++s, ++offset) {
89 const GrEffectStage* stage = stages[s];
90 uint16_t* offsetAndSize = reinterpret_cast<uint16_t*>(fKey.begin() +
91 kEffectKeyOffsetsAndLengthOffset +
92 offset * 2 * sizeof(uint16_t));
93 uint32_t effectKeyOffset = fKey.count();
94 if (effectKeyOffset > SK_MaxU16) {
95 fKey.reset();
96 return false;
97 }
joshualittbd769d02014-09-04 08:56:46 -070098 GrEffectKeyBuilder b(&fKey);
99 uint16_t effectKeySize;
egdaniela7dc0a82014-09-17 08:25:05 -0700100 if (!GetEffectKey(*stages[s], gpu->glCaps(), useLocalCoords, &b, &effectKeySize)) {
bsalomon929f29a2014-07-17 07:55:11 -0700101 fKey.reset();
102 return false;
mtklein79401002014-07-16 06:16:43 -0700103 }
egdaniela7dc0a82014-09-17 08:25:05 -0700104 get_stage_stats(*stage, &dstRead, &fragPos, &vertexShader);
bsalomon929f29a2014-07-17 07:55:11 -0700105 offsetAndSize[0] = effectKeyOffset;
106 offsetAndSize[1] = effectKeySize;
bsalomon848faf02014-07-11 10:01:02 -0700107 }
jvanverth@google.com054ae992013-04-01 20:06:51 +0000108
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000109 KeyHeader* header = this->header();
bsalomon848faf02014-07-11 10:01:02 -0700110 memset(header, 0, kHeaderSize);
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000111 header->fEmitsPointSize = random->nextBool();
112
113 header->fPositionAttributeIndex = 0;
jvanverth@google.com054ae992013-04-01 20:06:51 +0000114
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +0000115 // if the effects have used up all off the available attributes,
jvanverth@google.com054ae992013-04-01 20:06:51 +0000116 // don't try to use color or coverage attributes as input
117 do {
egdanielae444962014-09-22 12:29:52 -0700118 header->fColorInput = static_cast<GrGLProgramDesc::ColorInput>(
119 random->nextULessThan(kColorInputCnt));
120 } while ((GrDrawState::kMaxVertexAttribCnt <= currAttribIndex || isPathRendering) &&
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000121 kAttribute_ColorInput == header->fColorInput);
122 header->fColorAttributeIndex = (header->fColorInput == kAttribute_ColorInput) ?
123 currAttribIndex++ :
124 -1;
jvanverth@google.com054ae992013-04-01 20:06:51 +0000125
126 do {
commit-bot@chromium.org949eef02013-10-01 18:43:29 +0000127 header->fCoverageInput = static_cast<GrGLProgramDesc::ColorInput>(
128 random->nextULessThan(kColorInputCnt));
egdanielae444962014-09-22 12:29:52 -0700129 } while ((GrDrawState::kMaxVertexAttribCnt <= currAttribIndex || isPathRendering) &&
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000130 kAttribute_ColorInput == header->fCoverageInput);
131 header->fCoverageAttributeIndex = (header->fCoverageInput == kAttribute_ColorInput) ?
132 currAttribIndex++ :
133 -1;
bsalomon43d361f2014-09-19 07:47:08 -0700134 bool useGS = random->nextBool();
bsalomon@google.com91207482013-02-12 21:45:24 +0000135#if GR_GL_EXPERIMENTAL_GS
bsalomon43d361f2014-09-19 07:47:08 -0700136 header->fExperimentalGS = gpu->caps()->geometryShaderSupport() && useGS;
137#else
138 (void) useGS;
bsalomon@google.com91207482013-02-12 21:45:24 +0000139#endif
140
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000141 header->fLocalCoordAttributeIndex = useLocalCoords ? currAttribIndex++ : -1;
142
143 header->fColorEffectCnt = numColorStages;
144 header->fCoverageEffectCnt = numCoverageStages;
bsalomon@google.comc7818882013-03-20 19:19:53 +0000145
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000146 if (dstRead) {
joshualitt30ba4362014-08-21 20:18:45 -0700147 header->fDstReadKey = SkToU8(GrGLFragmentShaderBuilder::KeyForDstRead(dstCopyTexture,
bsalomon848faf02014-07-11 10:01:02 -0700148 gpu->glCaps()));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000149 } else {
150 header->fDstReadKey = 0;
151 }
152 if (fragPos) {
joshualitt30ba4362014-08-21 20:18:45 -0700153 header->fFragPosKey = SkToU8(GrGLFragmentShaderBuilder::KeyForFragmentPosition(dstRenderTarget,
bsalomon848faf02014-07-11 10:01:02 -0700154 gpu->glCaps()));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000155 } else {
156 header->fFragPosKey = 0;
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000157 }
bsalomon@google.com5920ac22013-04-19 13:14:45 +0000158
egdanielae444962014-09-22 12:29:52 -0700159 header->fUseFragShaderOnly = isPathRendering && gpu->glPathRendering()->texturingMode() ==
160 GrGLPathRendering::FixedFunction_TexturingMode;
joshualittbd769d02014-09-04 08:56:46 -0700161 header->fHasGeometryProcessor = vertexShader;
commit-bot@chromium.org6b30e452013-10-04 20:02:53 +0000162
egdanielc0648242014-09-22 13:17:02 -0700163 GrOptDrawState::PrimaryOutputType primaryOutput;
164 GrOptDrawState::SecondaryOutputType secondaryOutput;
165 if (!dstRead) {
166 primaryOutput = GrOptDrawState::kModulate_PrimaryOutputType;
167 } else {
168 primaryOutput = static_cast<GrOptDrawState::PrimaryOutputType>(
169 random->nextULessThan(GrOptDrawState::kPrimaryOutputTypeCnt));
170 }
bsalomon@google.com5920ac22013-04-19 13:14:45 +0000171
egdanielc0648242014-09-22 13:17:02 -0700172 if (GrOptDrawState::kCombineWithDst_PrimaryOutputType == primaryOutput ||
173 !gpu->caps()->dualSourceBlendingSupport()) {
174 secondaryOutput = GrOptDrawState::kNone_SecondaryOutputType;
175 } else {
176 secondaryOutput = static_cast<GrOptDrawState::SecondaryOutputType>(
177 random->nextULessThan(GrOptDrawState::kSecondaryOutputTypeCnt));
178 }
179
180 header->fPrimaryOutputType = primaryOutput;
181 header->fSecondaryOutputType = secondaryOutput;
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000182
bsalomon848faf02014-07-11 10:01:02 -0700183 this->finalize();
184 return true;
bsalomon@google.com91207482013-02-12 21:45:24 +0000185}
186
joshualitt249af152014-09-15 11:41:13 -0700187// TODO clean this up, we have to do this to test geometry processors but there has got to be
188// a better way. In the mean time, we actually fill out these generic vertex attribs below with
189// the correct vertex attribs from the GP. We have to ensure, however, we don't try to add more
190// than two attributes.
191GrVertexAttrib genericVertexAttribs[] = {
192 { kVec2f_GrVertexAttribType, 0, kPosition_GrVertexAttribBinding },
193 { kVec2f_GrVertexAttribType, 0, kEffect_GrVertexAttribBinding },
194 { kVec2f_GrVertexAttribType, 0, kEffect_GrVertexAttribBinding }
195};
196
197/*
198 * convert sl type to vertexattrib type, not a complete implementation, only use for debugging
199 */
200GrVertexAttribType convert_sltype_to_attribtype(GrSLType type) {
201 switch (type) {
202 case kFloat_GrSLType:
203 return kFloat_GrVertexAttribType;
204 case kVec2f_GrSLType:
205 return kVec2f_GrVertexAttribType;
206 case kVec3f_GrSLType:
207 return kVec3f_GrVertexAttribType;
208 case kVec4f_GrSLType:
209 return kVec4f_GrVertexAttribType;
210 default:
211 SkFAIL("Type isn't convertible");
212 return kFloat_GrVertexAttribType;
213 }
214}
215// TODO end test hack
216
217
bsalomon@google.com042a2862013-02-04 18:39:24 +0000218bool GrGpuGL::programUnitTest(int maxStages) {
219
bsalomon@google.comd4726202012-08-03 14:34:46 +0000220 GrTextureDesc dummyDesc;
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000221 dummyDesc.fFlags = kRenderTarget_GrTextureFlagBit;
bsalomon@google.comfec0bc32013-02-07 14:43:04 +0000222 dummyDesc.fConfig = kSkia8888_GrPixelConfig;
bsalomon@google.comd4726202012-08-03 14:34:46 +0000223 dummyDesc.fWidth = 34;
224 dummyDesc.fHeight = 18;
225 SkAutoTUnref<GrTexture> dummyTexture1(this->createTexture(dummyDesc, NULL, 0));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000226 dummyDesc.fFlags = kNone_GrTextureFlags;
bsalomon@google.comd4726202012-08-03 14:34:46 +0000227 dummyDesc.fConfig = kAlpha_8_GrPixelConfig;
228 dummyDesc.fWidth = 16;
229 dummyDesc.fHeight = 22;
230 SkAutoTUnref<GrTexture> dummyTexture2(this->createTexture(dummyDesc, NULL, 0));
231
bsalomone904c092014-07-17 10:50:59 -0700232 if (!dummyTexture1 || ! dummyTexture2) {
233 return false;
234 }
235
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000236 static const int NUM_TESTS = 512;
237
commit-bot@chromium.orge0e7cfe2013-09-09 20:09:12 +0000238 SkRandom random;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000239 for (int t = 0; t < NUM_TESTS; ++t) {
240
241#if 0
242 GrPrintf("\nTest Program %d\n-------------\n", t);
243 static const int stop = -1;
244 if (t == stop) {
245 int breakpointhere = 9;
246 }
247#endif
248
bsalomon@google.com31ec7982013-03-27 18:14:57 +0000249 GrGLProgramDesc pdesc;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000250
jvanverth@google.com054ae992013-04-01 20:06:51 +0000251 int currAttribIndex = 1; // we need to always leave room for position
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000252 int currTextureCoordSet = 0;
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000253 GrTexture* dummyTextures[] = {dummyTexture1.get(), dummyTexture2.get()};
commit-bot@chromium.org9ae78502013-03-21 17:44:39 +0000254
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000255 int numStages = random.nextULessThan(maxStages + 1);
256 int numColorStages = random.nextULessThan(numStages + 1);
257 int numCoverageStages = numStages - numColorStages;
258
259 SkAutoSTMalloc<8, const GrEffectStage*> stages(numStages);
260
egdanielae444962014-09-22 12:29:52 -0700261 bool usePathRendering = this->glCaps().pathRenderingSupport() && random.nextBool();
egdanielc0648242014-09-22 13:17:02 -0700262
egdanielae444962014-09-22 12:29:52 -0700263 GrGpu::DrawType drawType = usePathRendering ? GrGpu::kDrawPath_DrawType :
264 GrGpu::kDrawPoints_DrawType;
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000265
joshualittbd769d02014-09-04 08:56:46 -0700266 SkAutoTDelete<GrEffectStage> geometryProcessor;
egdanielae444962014-09-22 12:29:52 -0700267 bool hasGeometryProcessor = usePathRendering ? false : random.nextBool();
joshualittbd769d02014-09-04 08:56:46 -0700268 if (hasGeometryProcessor) {
269 while (true) {
270 SkAutoTUnref<const GrEffect> effect(GrEffectTestFactory::CreateStage(
271 &random,
272 this->getContext(),
273 *this->caps(),
274 dummyTextures));
275 SkASSERT(effect);
joshualittbd769d02014-09-04 08:56:46 -0700276 // Only geometryProcessor can use vertex shader
277 if (!effect->requiresVertexShader()) {
278 continue;
279 }
280
joshualitt249af152014-09-15 11:41:13 -0700281 GrEffectStage* stage = SkNEW_ARGS(GrEffectStage, (effect.get()));
joshualittbd769d02014-09-04 08:56:46 -0700282 geometryProcessor.reset(stage);
joshualitt249af152014-09-15 11:41:13 -0700283
284 // we have to set dummy vertex attribs
285 const GrEffect::VertexAttribArray& v = effect->getVertexAttribs();
286 int numVertexAttribs = v.count();
287
288 SkASSERT(GrEffect::kMaxVertexAttribs == 2 &&
289 GrEffect::kMaxVertexAttribs >= numVertexAttribs);
290 size_t runningStride = GrVertexAttribTypeSize(genericVertexAttribs[0].fType);
291 for (int i = 0; i < numVertexAttribs; i++) {
292 genericVertexAttribs[i + 1].fOffset = runningStride;
293 genericVertexAttribs[i + 1].fType =
294 convert_sltype_to_attribtype(v[i].getType());
295 runningStride += GrVertexAttribTypeSize(genericVertexAttribs[i + 1].fType);
296 }
297
298 // update the vertex attributes with the ds
299 GrDrawState* ds = this->drawState();
300 ds->setVertexAttribs<genericVertexAttribs>(numVertexAttribs + 1, runningStride);
301 currAttribIndex = numVertexAttribs + 1;
joshualittbd769d02014-09-04 08:56:46 -0700302 break;
303 }
304 }
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000305 for (int s = 0; s < numStages;) {
bsalomon83d081a2014-07-08 09:56:10 -0700306 SkAutoTUnref<const GrEffect> effect(GrEffectTestFactory::CreateStage(
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000307 &random,
308 this->getContext(),
309 *this->caps(),
310 dummyTextures));
commit-bot@chromium.org65ee5f42014-02-04 17:49:48 +0000311 SkASSERT(effect);
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000312
joshualittbd769d02014-09-04 08:56:46 -0700313 // Only geometryProcessor can use vertex shader
314 if (effect->requiresVertexShader()) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000315 continue;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000316 }
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000317
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000318 // If adding this effect would exceed the max texture coord set count then generate a
319 // new random effect.
egdanielae444962014-09-22 12:29:52 -0700320 if (usePathRendering && this->glPathRendering()->texturingMode() ==
321 GrGLPathRendering::FixedFunction_TexturingMode) {;
bsalomon97b9ab72014-07-08 06:52:35 -0700322 int numTransforms = effect->numTransforms();
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000323 if (currTextureCoordSet + numTransforms > this->glCaps().maxFixedFunctionTextureCoords()) {
324 continue;
325 }
326 currTextureCoordSet += numTransforms;
327 }
joshualitt249af152014-09-15 11:41:13 -0700328 GrEffectStage* stage = SkNEW_ARGS(GrEffectStage, (effect.get()));
joshualittbd769d02014-09-04 08:56:46 -0700329
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000330 stages[s] = stage;
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000331 ++s;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000332 }
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000333 const GrTexture* dstTexture = random.nextBool() ? dummyTextures[0] : dummyTextures[1];
bsalomon848faf02014-07-11 10:01:02 -0700334 if (!pdesc.setRandom(&random,
335 this,
336 dummyTextures[0]->asRenderTarget(),
337 dstTexture,
joshualittbd769d02014-09-04 08:56:46 -0700338 geometryProcessor.get(),
bsalomon848faf02014-07-11 10:01:02 -0700339 stages.get(),
340 numColorStages,
341 numCoverageStages,
egdanielae444962014-09-22 12:29:52 -0700342 currAttribIndex,
343 drawType)) {
bsalomon848faf02014-07-11 10:01:02 -0700344 return false;
345 }
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +0000346
commit-bot@chromium.org9188a152013-09-05 18:28:24 +0000347 SkAutoTUnref<GrGLProgram> program(GrGLProgram::Create(this,
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000348 pdesc,
joshualittbd769d02014-09-04 08:56:46 -0700349 geometryProcessor.get(),
bsalomon@google.com2c84aa32013-06-06 20:28:57 +0000350 stages,
351 stages + numColorStages));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000352 for (int s = 0; s < numStages; ++s) {
bsalomon@google.com504976e2013-05-09 13:45:02 +0000353 SkDELETE(stages[s]);
354 }
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000355 if (NULL == program.get()) {
356 return false;
357 }
joshualitt249af152014-09-15 11:41:13 -0700358
359 // We have to reset the drawstate because we might have added a gp
360 this->drawState()->reset();
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000361 }
362 return true;
363}
bsalomon@google.coma8e686e2011-08-16 15:45:58 +0000364
tfarina@chromium.org4ee16bf2014-01-10 22:08:27 +0000365DEF_GPUTEST(GLPrograms, reporter, factory) {
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000366 for (int type = 0; type < GrContextFactory::kLastGLContextType; ++type) {
367 GrContext* context = factory->get(static_cast<GrContextFactory::GLContextType>(type));
bsalomon49f085d2014-09-05 13:34:00 -0700368 if (context) {
bsalomon@google.com042a2862013-02-04 18:39:24 +0000369 GrGpuGL* gpu = static_cast<GrGpuGL*>(context->getGpu());
bsalomon@google.comeb6879f2013-06-13 19:34:18 +0000370 int maxStages = 6;
bsalomon@google.com042a2862013-02-04 18:39:24 +0000371#if SK_ANGLE
372 // Some long shaders run out of temporary registers in the D3D compiler on ANGLE.
373 if (type == GrContextFactory::kANGLE_GLContextType) {
374 maxStages = 3;
375 }
376#endif
377 REPORTER_ASSERT(reporter, gpu->programUnitTest(maxStages));
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000378 }
379 }
bsalomon@google.coma8e686e2011-08-16 15:45:58 +0000380}
381
rmistry@google.comd6176b02012-08-23 18:14:13 +0000382// This is evil evil evil. The linker may throw away whole translation units as dead code if it
bsalomon@google.com67e78c92012-10-17 13:36:14 +0000383// thinks none of the functions are called. It will do this even if there are static initializers
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000384// in the unit that could pass pointers to functions from the unit out to other translation units!
385// We force some of the effects that would otherwise be discarded to link here.
386
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +0000387#include "SkAlphaThresholdFilter.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +0000388#include "SkColorMatrixFilter.h"
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000389#include "SkLightingImageFilter.h"
bsalomon@google.com82aa7482012-08-13 14:22:17 +0000390#include "SkMagnifierImageFilter.h"
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000391
392void forceLinking();
393
394void forceLinking() {
395 SkLightingImageFilter::CreateDistantLitDiffuse(SkPoint3(0,0,0), 0, 0, 0);
commit-bot@chromium.org9109e182014-01-07 16:04:01 +0000396 SkAlphaThresholdFilter::Create(SkRegion(), .5f, .5f);
reed9fa60da2014-08-21 07:59:51 -0700397 SkAutoTUnref<SkImageFilter> mag(SkMagnifierImageFilter::Create(
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000398 SkRect::MakeWH(SK_Scalar1, SK_Scalar1), SK_Scalar1));
bsalomon@google.comadc65362013-01-28 14:26:09 +0000399 GrConfigConversionEffect::Create(NULL,
400 false,
401 GrConfigConversionEffect::kNone_PMConversion,
402 SkMatrix::I());
bsalomon@google.com67e78c92012-10-17 13:36:14 +0000403 SkScalar matrix[20];
commit-bot@chromium.org727a3522014-02-21 18:46:30 +0000404 SkAutoTUnref<SkColorMatrixFilter> cmf(SkColorMatrixFilter::Create(matrix));
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000405}
406
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +0000407#endif