blob: ae6fe7b3c4fecbcde464a88584e3a261569a66f7 [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.com5739d2c2012-05-31 15:07:19 +000015#include "gl/GrGpuGL.h"
bsalomon@google.com396e61f2012-10-25 19:00:29 +000016#include "GrBackendEffectFactory.h"
bsalomon@google.com67b915d2013-02-04 16:13:32 +000017#include "GrContextFactory.h"
bsalomon@google.comc7818882013-03-20 19:19:53 +000018#include "GrDrawEffect.h"
bsalomon@google.coma04e8e82012-08-27 12:53:13 +000019#include "effects/GrConfigConversionEffect.h"
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +000020
tfarina@chromium.org223137f2012-11-21 22:38:36 +000021#include "SkRandom.h"
bsalomon@google.comc3841b92012-08-02 18:11:43 +000022#include "Test.h"
23
bsalomon@google.com31ec7982013-03-27 18:14:57 +000024void GrGLProgramDesc::setRandom(SkMWCRandom* random,
25 const GrGpuGL* gpu,
bsalomon@google.comb79d8652013-03-29 20:30:50 +000026 const GrTexture* dstTexture,
jvanverth@google.com054ae992013-04-01 20:06:51 +000027 const GrEffectStage stages[GrDrawState::kNumStages],
28 int currAttribIndex) {
bsalomon@google.com91207482013-02-12 21:45:24 +000029 fEmitsPointSize = random->nextBool();
jvanverth@google.com054ae992013-04-01 20:06:51 +000030
31 fPositionAttributeIndex = 0;
32
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +000033 // if the effects have used up all off the available attributes,
jvanverth@google.com054ae992013-04-01 20:06:51 +000034 // don't try to use color or coverage attributes as input
35 do {
36 fColorInput = random->nextULessThan(kColorInputCnt);
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +000037 } while (GrDrawState::kMaxVertexAttribCnt <= currAttribIndex &&
jvanverth@google.com054ae992013-04-01 20:06:51 +000038 kAttribute_ColorInput == fColorInput);
39 fColorAttributeIndex = (fColorInput == kAttribute_ColorInput) ? currAttribIndex++ : -1;
40
41 do {
42 fCoverageInput = random->nextULessThan(kColorInputCnt);
43 } while (GrDrawState::kMaxVertexAttribCnt <= currAttribIndex &&
44 kAttribute_ColorInput == fCoverageInput);
45 fCoverageAttributeIndex = (fCoverageInput == kAttribute_ColorInput) ? currAttribIndex++ : -1;
bsalomon@google.com91207482013-02-12 21:45:24 +000046
bsalomon@google.com8da9bc72013-04-19 15:03:21 +000047 fColorFilterXfermode = random->nextULessThan(SkXfermode::kLastCoeffMode + 1);
bsalomon@google.com91207482013-02-12 21:45:24 +000048
49 fFirstCoverageStage = random->nextULessThan(GrDrawState::kNumStages);
50
bsalomon@google.com91207482013-02-12 21:45:24 +000051#if GR_GL_EXPERIMENTAL_GS
bsalomon@google.combcce8922013-03-25 15:38:39 +000052 fExperimentalGS = gpu->caps()->geometryShaderSupport() && random->nextBool();
bsalomon@google.com91207482013-02-12 21:45:24 +000053#endif
54
bsalomon@google.com26ec00e2013-03-26 16:49:37 +000055 fDiscardIfZeroCoverage = random->nextBool();
skia.committer@gmail.com37cbc7f2013-03-27 07:01:04 +000056
jvanverth@google.com054ae992013-04-01 20:06:51 +000057 bool useLocalCoords = random->nextBool() && currAttribIndex < GrDrawState::kMaxVertexAttribCnt;
58 fLocalCoordAttributeIndex = useLocalCoords ? currAttribIndex++ : -1;
bsalomon@google.comc7818882013-03-20 19:19:53 +000059
bsalomon@google.comb79d8652013-03-29 20:30:50 +000060 bool dstRead = false;
bsalomon@google.com91207482013-02-12 21:45:24 +000061 for (int s = 0; s < GrDrawState::kNumStages; ++s) {
62 if (NULL != stages[s].getEffect()) {
63 const GrBackendEffectFactory& factory = (*stages[s].getEffect())->getFactory();
jvanverth@google.com054ae992013-04-01 20:06:51 +000064 GrDrawEffect drawEffect(stages[s], useLocalCoords);
bsalomon@google.comc7818882013-03-20 19:19:53 +000065 fEffectKeys[s] = factory.glEffectKey(drawEffect, gpu->glCaps());
bsalomon@google.comb79d8652013-03-29 20:30:50 +000066 if ((*stages[s].getEffect())->willReadDst()) {
67 dstRead = true;
68 }
bsalomon@google.com91207482013-02-12 21:45:24 +000069 }
70 }
jvanverth@google.com9b855c72013-03-01 18:21:22 +000071
bsalomon@google.comb79d8652013-03-29 20:30:50 +000072 if (dstRead) {
73 this->fDstRead = GrGLShaderBuilder::KeyForDstRead(dstTexture, gpu->glCaps());
74 }
bsalomon@google.com5920ac22013-04-19 13:14:45 +000075
76 CoverageOutput coverageOutput;
77 bool illegalCoverageOutput;
78 do {
79 coverageOutput = static_cast<CoverageOutput>(random->nextULessThan(kCoverageOutputCnt));
80 illegalCoverageOutput = (!gpu->caps()->dualSourceBlendingSupport() &&
81 CoverageOutputUsesSecondaryOutput(coverageOutput)) ||
bsalomon@google.com72993ab2013-04-19 13:25:28 +000082 (!dstRead && kCombineWithDst_CoverageOutput == coverageOutput);
bsalomon@google.com5920ac22013-04-19 13:14:45 +000083 } while (illegalCoverageOutput);
84
85 fCoverageOutput = coverageOutput;
bsalomon@google.com91207482013-02-12 21:45:24 +000086}
87
bsalomon@google.com042a2862013-02-04 18:39:24 +000088bool GrGpuGL::programUnitTest(int maxStages) {
89
90 maxStages = GrMin(maxStages, (int)GrDrawState::kNumStages);
bsalomon@google.comc3841b92012-08-02 18:11:43 +000091
bsalomon@google.comd4726202012-08-03 14:34:46 +000092 GrTextureDesc dummyDesc;
bsalomon@google.comfec0bc32013-02-07 14:43:04 +000093 dummyDesc.fConfig = kSkia8888_GrPixelConfig;
bsalomon@google.comd4726202012-08-03 14:34:46 +000094 dummyDesc.fWidth = 34;
95 dummyDesc.fHeight = 18;
96 SkAutoTUnref<GrTexture> dummyTexture1(this->createTexture(dummyDesc, NULL, 0));
97 dummyDesc.fConfig = kAlpha_8_GrPixelConfig;
98 dummyDesc.fWidth = 16;
99 dummyDesc.fHeight = 22;
100 SkAutoTUnref<GrTexture> dummyTexture2(this->createTexture(dummyDesc, NULL, 0));
101
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000102 static const int NUM_TESTS = 512;
103
bsalomon@google.com91207482013-02-12 21:45:24 +0000104 SkMWCRandom random;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000105 for (int t = 0; t < NUM_TESTS; ++t) {
106
107#if 0
108 GrPrintf("\nTest Program %d\n-------------\n", t);
109 static const int stop = -1;
110 if (t == stop) {
111 int breakpointhere = 9;
112 }
113#endif
114
bsalomon@google.com31ec7982013-03-27 18:14:57 +0000115 GrGLProgramDesc pdesc;
bsalomon@google.com2eaaefd2012-10-29 19:51:22 +0000116 GrEffectStage stages[GrDrawState::kNumStages];
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000117
jvanverth@google.com054ae992013-04-01 20:06:51 +0000118 int currAttribIndex = 1; // we need to always leave room for position
commit-bot@chromium.orgff6ea262013-03-12 12:26:08 +0000119 int attribIndices[2];
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000120 GrTexture* dummyTextures[] = {dummyTexture1.get(), dummyTexture2.get()};
bsalomon@google.com1d1d4242013-02-05 15:44:21 +0000121 for (int s = 0; s < maxStages; ++s) {
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000122 // enable the stage?
bsalomon@google.com91207482013-02-12 21:45:24 +0000123 if (random.nextBool()) {
bsalomon@google.com73a96942013-02-13 16:31:19 +0000124 SkAutoTUnref<const GrEffectRef> effect(GrEffectTestFactory::CreateStage(
125 &random,
126 this->getContext(),
bsalomon@google.comc26d94f2013-03-25 18:19:00 +0000127 *this->caps(),
bsalomon@google.com73a96942013-02-13 16:31:19 +0000128 dummyTextures));
commit-bot@chromium.org9ae78502013-03-21 17:44:39 +0000129 int numAttribs = (*effect)->numVertexAttribs();
130
jvanverth@google.com054ae992013-04-01 20:06:51 +0000131 // If adding this effect would exceed the max attrib count then generate a
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +0000132 // new random effect.
jvanverth@google.com054ae992013-04-01 20:06:51 +0000133 if (currAttribIndex + numAttribs > GrDrawState::kMaxVertexAttribCnt) {
commit-bot@chromium.org9ae78502013-03-21 17:44:39 +0000134 --s;
135 continue;
136 }
137 for (int i = 0; i < numAttribs; ++i) {
commit-bot@chromium.orgff6ea262013-03-12 12:26:08 +0000138 attribIndices[i] = currAttribIndex++;
139 }
jvanverth@google.com65eb4d52013-03-19 18:51:02 +0000140 stages[s].setEffect(effect.get(), attribIndices[0], attribIndices[1]);
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000141 }
142 }
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000143 const GrTexture* dstTexture = random.nextBool() ? dummyTextures[0] : dummyTextures[1];
jvanverth@google.com054ae992013-04-01 20:06:51 +0000144 pdesc.setRandom(&random, this, dstTexture, stages, currAttribIndex);
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +0000145
bsalomon@google.com2eaaefd2012-10-29 19:51:22 +0000146 const GrEffectStage* stagePtrs[GrDrawState::kNumStages];
147 for (int s = 0; s < GrDrawState::kNumStages; ++s) {
148 stagePtrs[s] = &stages[s];
149 }
robertphillips@google.com6177e692013-02-28 20:16:25 +0000150 SkAutoTUnref<GrGLProgram> program(GrGLProgram::Create(this->glContext(),
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000151 pdesc,
bsalomon@google.com2eaaefd2012-10-29 19:51:22 +0000152 stagePtrs));
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000153 if (NULL == program.get()) {
154 return false;
155 }
156 }
157 return true;
158}
bsalomon@google.coma8e686e2011-08-16 15:45:58 +0000159
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000160static void GLProgramsTest(skiatest::Reporter* reporter, GrContextFactory* factory) {
161 for (int type = 0; type < GrContextFactory::kLastGLContextType; ++type) {
162 GrContext* context = factory->get(static_cast<GrContextFactory::GLContextType>(type));
163 if (NULL != context) {
bsalomon@google.com042a2862013-02-04 18:39:24 +0000164 GrGpuGL* gpu = static_cast<GrGpuGL*>(context->getGpu());
165 int maxStages = GrDrawState::kNumStages;
166#if SK_ANGLE
167 // Some long shaders run out of temporary registers in the D3D compiler on ANGLE.
168 if (type == GrContextFactory::kANGLE_GLContextType) {
169 maxStages = 3;
170 }
171#endif
172 REPORTER_ASSERT(reporter, gpu->programUnitTest(maxStages));
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000173 }
174 }
bsalomon@google.coma8e686e2011-08-16 15:45:58 +0000175}
176
bsalomon@google.coma8e686e2011-08-16 15:45:58 +0000177#include "TestClassDef.h"
178DEFINE_GPUTESTCLASS("GLPrograms", GLProgramsTestClass, GLProgramsTest)
179
rmistry@google.comd6176b02012-08-23 18:14:13 +0000180// 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 +0000181// 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 +0000182// in the unit that could pass pointers to functions from the unit out to other translation units!
183// We force some of the effects that would otherwise be discarded to link here.
184
185#include "SkLightingImageFilter.h"
bsalomon@google.com82aa7482012-08-13 14:22:17 +0000186#include "SkMagnifierImageFilter.h"
bsalomon@google.com67e78c92012-10-17 13:36:14 +0000187#include "SkColorMatrixFilter.h"
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000188
189void forceLinking();
190
191void forceLinking() {
192 SkLightingImageFilter::CreateDistantLitDiffuse(SkPoint3(0,0,0), 0, 0, 0);
bsalomon@google.com82aa7482012-08-13 14:22:17 +0000193 SkMagnifierImageFilter mag(SkRect::MakeWH(SK_Scalar1, SK_Scalar1), SK_Scalar1);
bsalomon@google.comadc65362013-01-28 14:26:09 +0000194 GrConfigConversionEffect::Create(NULL,
195 false,
196 GrConfigConversionEffect::kNone_PMConversion,
197 SkMatrix::I());
bsalomon@google.com67e78c92012-10-17 13:36:14 +0000198 SkScalar matrix[20];
199 SkColorMatrixFilter cmf(matrix);
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000200}
201
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +0000202#endif