blob: e8b6da5193f6cc4fc4067867641d3338f8b68db8 [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"
bsalomon@google.coma04e8e82012-08-27 12:53:13 +000017#include "effects/GrConfigConversionEffect.h"
kkinnunenec56e452014-08-25 22:21:16 -070018#include "gl/GrGLPathRendering.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000019#include "gl/GrGpuGL.h"
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000020#include "SkChecksum.h"
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
bsalomon848faf02014-07-11 10:01:02 -070024bool GrGLProgramDesc::setRandom(SkRandom* random,
bsalomon@google.com31ec7982013-03-27 18:14:57 +000025 const GrGpuGL* gpu,
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000026 const GrRenderTarget* dstRenderTarget,
27 const GrTexture* dstCopyTexture,
joshualittbd769d02014-09-04 08:56:46 -070028 const GrEffectStage* geometryProcessor,
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000029 const GrEffectStage* stages[],
30 int numColorStages,
31 int numCoverageStages,
jvanverth@google.com054ae992013-04-01 20:06:51 +000032 int currAttribIndex) {
bsalomon848faf02014-07-11 10:01:02 -070033 bool useLocalCoords = random->nextBool() && currAttribIndex < GrDrawState::kMaxVertexAttribCnt;
34
35 int numStages = numColorStages + numCoverageStages;
36 fKey.reset();
37
bsalomon929f29a2014-07-17 07:55:11 -070038 GR_STATIC_ASSERT(0 == kEffectKeyOffsetsAndLengthOffset % sizeof(uint32_t));
bsalomon848faf02014-07-11 10:01:02 -070039
40 // Make room for everything up to and including the array of offsets to effect keys.
joshualittbd769d02014-09-04 08:56:46 -070041 fKey.push_back_n(kEffectKeyOffsetsAndLengthOffset + 2 * sizeof(uint16_t) * (numStages +
42 (geometryProcessor ? 1 : 0)));
bsalomon848faf02014-07-11 10:01:02 -070043
44 bool dstRead = false;
45 bool fragPos = false;
bsalomon49f085d2014-09-05 13:34:00 -070046 bool vertexShader = SkToBool(geometryProcessor);
joshualittbd769d02014-09-04 08:56:46 -070047 int offset = 0;
bsalomon49f085d2014-09-05 13:34:00 -070048 if (geometryProcessor) {
joshualittbd769d02014-09-04 08:56:46 -070049 const GrEffectStage* stage = geometryProcessor;
bsalomon929f29a2014-07-17 07:55:11 -070050 uint16_t* offsetAndSize = reinterpret_cast<uint16_t*>(fKey.begin() +
51 kEffectKeyOffsetsAndLengthOffset +
joshualittbd769d02014-09-04 08:56:46 -070052 offset * 2 * sizeof(uint16_t));
bsalomon929f29a2014-07-17 07:55:11 -070053 uint32_t effectKeyOffset = fKey.count();
54 if (effectKeyOffset > SK_MaxU16) {
bsalomonc0ea3982014-07-15 19:41:17 -070055 fKey.reset();
56 return false;
bsalomon848faf02014-07-11 10:01:02 -070057 }
bsalomon929f29a2014-07-17 07:55:11 -070058 GrEffectKeyBuilder b(&fKey);
59 uint16_t effectKeySize;
joshualittbd769d02014-09-04 08:56:46 -070060 if (!GetEffectKeyAndUpdateStats(*stage, gpu->glCaps(), useLocalCoords, &b,
61 &effectKeySize, &dstRead, &fragPos, &vertexShader)) {
62 fKey.reset();
63 return false;
64 }
65 offsetAndSize[0] = effectKeyOffset;
66 offsetAndSize[1] = effectKeySize;
67 offset++;
68 }
69
70 for (int s = 0; s < numStages; ++s, ++offset) {
71 const GrEffectStage* stage = stages[s];
72 uint16_t* offsetAndSize = reinterpret_cast<uint16_t*>(fKey.begin() +
73 kEffectKeyOffsetsAndLengthOffset +
74 offset * 2 * sizeof(uint16_t));
75 uint32_t effectKeyOffset = fKey.count();
76 if (effectKeyOffset > SK_MaxU16) {
77 fKey.reset();
78 return false;
79 }
joshualittbd769d02014-09-04 08:56:46 -070080 GrEffectKeyBuilder b(&fKey);
81 uint16_t effectKeySize;
82 if (!GetEffectKeyAndUpdateStats(*stage, gpu->glCaps(), useLocalCoords, &b,
kkinnunenec56e452014-08-25 22:21:16 -070083 &effectKeySize, &dstRead, &fragPos, &vertexShader)) {
bsalomon929f29a2014-07-17 07:55:11 -070084 fKey.reset();
85 return false;
mtklein79401002014-07-16 06:16:43 -070086 }
bsalomon929f29a2014-07-17 07:55:11 -070087 offsetAndSize[0] = effectKeyOffset;
88 offsetAndSize[1] = effectKeySize;
bsalomon848faf02014-07-11 10:01:02 -070089 }
jvanverth@google.com054ae992013-04-01 20:06:51 +000090
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000091 KeyHeader* header = this->header();
bsalomon848faf02014-07-11 10:01:02 -070092 memset(header, 0, kHeaderSize);
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000093 header->fEmitsPointSize = random->nextBool();
94
95 header->fPositionAttributeIndex = 0;
jvanverth@google.com054ae992013-04-01 20:06:51 +000096
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +000097 // if the effects have used up all off the available attributes,
jvanverth@google.com054ae992013-04-01 20:06:51 +000098 // don't try to use color or coverage attributes as input
99 do {
egdaniel02cafcc2014-07-21 11:37:28 -0700100 uint32_t colorRand = random->nextULessThan(2);
101 header->fColorInput = (0 == colorRand) ? GrGLProgramDesc::kAttribute_ColorInput :
102 GrGLProgramDesc::kUniform_ColorInput;
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +0000103 } while (GrDrawState::kMaxVertexAttribCnt <= currAttribIndex &&
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000104 kAttribute_ColorInput == header->fColorInput);
bsalomon848faf02014-07-11 10:01:02 -0700105
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000106 header->fColorAttributeIndex = (header->fColorInput == kAttribute_ColorInput) ?
107 currAttribIndex++ :
108 -1;
jvanverth@google.com054ae992013-04-01 20:06:51 +0000109
110 do {
commit-bot@chromium.org949eef02013-10-01 18:43:29 +0000111 header->fCoverageInput = static_cast<GrGLProgramDesc::ColorInput>(
112 random->nextULessThan(kColorInputCnt));
jvanverth@google.com054ae992013-04-01 20:06:51 +0000113 } while (GrDrawState::kMaxVertexAttribCnt <= currAttribIndex &&
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000114 kAttribute_ColorInput == header->fCoverageInput);
115 header->fCoverageAttributeIndex = (header->fCoverageInput == kAttribute_ColorInput) ?
116 currAttribIndex++ :
117 -1;
bsalomon@google.com91207482013-02-12 21:45:24 +0000118
bsalomon@google.com91207482013-02-12 21:45:24 +0000119#if GR_GL_EXPERIMENTAL_GS
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000120 header->fExperimentalGS = gpu->caps()->geometryShaderSupport() && random->nextBool();
bsalomon@google.com91207482013-02-12 21:45:24 +0000121#endif
122
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000123 header->fLocalCoordAttributeIndex = useLocalCoords ? currAttribIndex++ : -1;
124
125 header->fColorEffectCnt = numColorStages;
126 header->fCoverageEffectCnt = numCoverageStages;
bsalomon@google.comc7818882013-03-20 19:19:53 +0000127
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000128 if (dstRead) {
joshualitt30ba4362014-08-21 20:18:45 -0700129 header->fDstReadKey = SkToU8(GrGLFragmentShaderBuilder::KeyForDstRead(dstCopyTexture,
bsalomon848faf02014-07-11 10:01:02 -0700130 gpu->glCaps()));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000131 } else {
132 header->fDstReadKey = 0;
133 }
134 if (fragPos) {
joshualitt30ba4362014-08-21 20:18:45 -0700135 header->fFragPosKey = SkToU8(GrGLFragmentShaderBuilder::KeyForFragmentPosition(dstRenderTarget,
bsalomon848faf02014-07-11 10:01:02 -0700136 gpu->glCaps()));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000137 } else {
138 header->fFragPosKey = 0;
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000139 }
bsalomon@google.com5920ac22013-04-19 13:14:45 +0000140
kkinnunenec56e452014-08-25 22:21:16 -0700141 header->fRequiresVertexShader = vertexShader ||
142 useLocalCoords ||
143 kAttribute_ColorInput == header->fColorInput ||
144 kAttribute_ColorInput == header->fCoverageInput;
joshualittbd769d02014-09-04 08:56:46 -0700145 header->fHasGeometryProcessor = vertexShader;
commit-bot@chromium.org6b30e452013-10-04 20:02:53 +0000146
bsalomon@google.com5920ac22013-04-19 13:14:45 +0000147 CoverageOutput coverageOutput;
148 bool illegalCoverageOutput;
149 do {
150 coverageOutput = static_cast<CoverageOutput>(random->nextULessThan(kCoverageOutputCnt));
151 illegalCoverageOutput = (!gpu->caps()->dualSourceBlendingSupport() &&
152 CoverageOutputUsesSecondaryOutput(coverageOutput)) ||
bsalomon@google.com72993ab2013-04-19 13:25:28 +0000153 (!dstRead && kCombineWithDst_CoverageOutput == coverageOutput);
bsalomon@google.com5920ac22013-04-19 13:14:45 +0000154 } while (illegalCoverageOutput);
155
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000156 header->fCoverageOutput = coverageOutput;
157
bsalomon848faf02014-07-11 10:01:02 -0700158 this->finalize();
159 return true;
bsalomon@google.com91207482013-02-12 21:45:24 +0000160}
161
joshualitt249af152014-09-15 11:41:13 -0700162// TODO clean this up, we have to do this to test geometry processors but there has got to be
163// a better way. In the mean time, we actually fill out these generic vertex attribs below with
164// the correct vertex attribs from the GP. We have to ensure, however, we don't try to add more
165// than two attributes.
166GrVertexAttrib genericVertexAttribs[] = {
167 { kVec2f_GrVertexAttribType, 0, kPosition_GrVertexAttribBinding },
168 { kVec2f_GrVertexAttribType, 0, kEffect_GrVertexAttribBinding },
169 { kVec2f_GrVertexAttribType, 0, kEffect_GrVertexAttribBinding }
170};
171
172/*
173 * convert sl type to vertexattrib type, not a complete implementation, only use for debugging
174 */
175GrVertexAttribType convert_sltype_to_attribtype(GrSLType type) {
176 switch (type) {
177 case kFloat_GrSLType:
178 return kFloat_GrVertexAttribType;
179 case kVec2f_GrSLType:
180 return kVec2f_GrVertexAttribType;
181 case kVec3f_GrSLType:
182 return kVec3f_GrVertexAttribType;
183 case kVec4f_GrSLType:
184 return kVec4f_GrVertexAttribType;
185 default:
186 SkFAIL("Type isn't convertible");
187 return kFloat_GrVertexAttribType;
188 }
189}
190// TODO end test hack
191
192
bsalomon@google.com042a2862013-02-04 18:39:24 +0000193bool GrGpuGL::programUnitTest(int maxStages) {
194
bsalomon@google.comd4726202012-08-03 14:34:46 +0000195 GrTextureDesc dummyDesc;
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000196 dummyDesc.fFlags = kRenderTarget_GrTextureFlagBit;
bsalomon@google.comfec0bc32013-02-07 14:43:04 +0000197 dummyDesc.fConfig = kSkia8888_GrPixelConfig;
bsalomon@google.comd4726202012-08-03 14:34:46 +0000198 dummyDesc.fWidth = 34;
199 dummyDesc.fHeight = 18;
200 SkAutoTUnref<GrTexture> dummyTexture1(this->createTexture(dummyDesc, NULL, 0));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000201 dummyDesc.fFlags = kNone_GrTextureFlags;
bsalomon@google.comd4726202012-08-03 14:34:46 +0000202 dummyDesc.fConfig = kAlpha_8_GrPixelConfig;
203 dummyDesc.fWidth = 16;
204 dummyDesc.fHeight = 22;
205 SkAutoTUnref<GrTexture> dummyTexture2(this->createTexture(dummyDesc, NULL, 0));
206
bsalomone904c092014-07-17 10:50:59 -0700207 if (!dummyTexture1 || ! dummyTexture2) {
208 return false;
209 }
210
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000211 static const int NUM_TESTS = 512;
212
commit-bot@chromium.orge0e7cfe2013-09-09 20:09:12 +0000213 SkRandom random;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000214 for (int t = 0; t < NUM_TESTS; ++t) {
215
216#if 0
217 GrPrintf("\nTest Program %d\n-------------\n", t);
218 static const int stop = -1;
219 if (t == stop) {
220 int breakpointhere = 9;
221 }
222#endif
223
bsalomon@google.com31ec7982013-03-27 18:14:57 +0000224 GrGLProgramDesc pdesc;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000225
jvanverth@google.com054ae992013-04-01 20:06:51 +0000226 int currAttribIndex = 1; // we need to always leave room for position
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000227 int currTextureCoordSet = 0;
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000228 GrTexture* dummyTextures[] = {dummyTexture1.get(), dummyTexture2.get()};
commit-bot@chromium.org9ae78502013-03-21 17:44:39 +0000229
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000230 int numStages = random.nextULessThan(maxStages + 1);
231 int numColorStages = random.nextULessThan(numStages + 1);
232 int numCoverageStages = numStages - numColorStages;
233
234 SkAutoSTMalloc<8, const GrEffectStage*> stages(numStages);
235
kkinnunenec56e452014-08-25 22:21:16 -0700236 bool useFixedFunctionPathRendering = this->glCaps().pathRenderingSupport() &&
joshualittbd769d02014-09-04 08:56:46 -0700237 this->glPathRendering()->texturingMode() == GrGLPathRendering::FixedFunction_TexturingMode &&
238 random.nextBool();
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000239
joshualittbd769d02014-09-04 08:56:46 -0700240 SkAutoTDelete<GrEffectStage> geometryProcessor;
241 bool hasGeometryProcessor = useFixedFunctionPathRendering ? false : random.nextBool();
242 if (hasGeometryProcessor) {
243 while (true) {
244 SkAutoTUnref<const GrEffect> effect(GrEffectTestFactory::CreateStage(
245 &random,
246 this->getContext(),
247 *this->caps(),
248 dummyTextures));
249 SkASSERT(effect);
joshualittbd769d02014-09-04 08:56:46 -0700250 // Only geometryProcessor can use vertex shader
251 if (!effect->requiresVertexShader()) {
252 continue;
253 }
254
joshualitt249af152014-09-15 11:41:13 -0700255 GrEffectStage* stage = SkNEW_ARGS(GrEffectStage, (effect.get()));
joshualittbd769d02014-09-04 08:56:46 -0700256 geometryProcessor.reset(stage);
joshualitt249af152014-09-15 11:41:13 -0700257
258 // we have to set dummy vertex attribs
259 const GrEffect::VertexAttribArray& v = effect->getVertexAttribs();
260 int numVertexAttribs = v.count();
261
262 SkASSERT(GrEffect::kMaxVertexAttribs == 2 &&
263 GrEffect::kMaxVertexAttribs >= numVertexAttribs);
264 size_t runningStride = GrVertexAttribTypeSize(genericVertexAttribs[0].fType);
265 for (int i = 0; i < numVertexAttribs; i++) {
266 genericVertexAttribs[i + 1].fOffset = runningStride;
267 genericVertexAttribs[i + 1].fType =
268 convert_sltype_to_attribtype(v[i].getType());
269 runningStride += GrVertexAttribTypeSize(genericVertexAttribs[i + 1].fType);
270 }
271
272 // update the vertex attributes with the ds
273 GrDrawState* ds = this->drawState();
274 ds->setVertexAttribs<genericVertexAttribs>(numVertexAttribs + 1, runningStride);
275 currAttribIndex = numVertexAttribs + 1;
joshualittbd769d02014-09-04 08:56:46 -0700276 break;
277 }
278 }
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000279 for (int s = 0; s < numStages;) {
bsalomon83d081a2014-07-08 09:56:10 -0700280 SkAutoTUnref<const GrEffect> effect(GrEffectTestFactory::CreateStage(
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000281 &random,
282 this->getContext(),
283 *this->caps(),
284 dummyTextures));
commit-bot@chromium.org65ee5f42014-02-04 17:49:48 +0000285 SkASSERT(effect);
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000286
joshualittbd769d02014-09-04 08:56:46 -0700287 // Only geometryProcessor can use vertex shader
288 if (effect->requiresVertexShader()) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000289 continue;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000290 }
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000291
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000292 // If adding this effect would exceed the max texture coord set count then generate a
293 // new random effect.
joshualittbd769d02014-09-04 08:56:46 -0700294 if (useFixedFunctionPathRendering) {
bsalomon97b9ab72014-07-08 06:52:35 -0700295 int numTransforms = effect->numTransforms();
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000296 if (currTextureCoordSet + numTransforms > this->glCaps().maxFixedFunctionTextureCoords()) {
297 continue;
298 }
299 currTextureCoordSet += numTransforms;
300 }
joshualitt249af152014-09-15 11:41:13 -0700301 GrEffectStage* stage = SkNEW_ARGS(GrEffectStage, (effect.get()));
joshualittbd769d02014-09-04 08:56:46 -0700302
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000303 stages[s] = stage;
commit-bot@chromium.org8e919ad2013-10-21 14:48:23 +0000304 ++s;
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000305 }
bsalomon@google.comb79d8652013-03-29 20:30:50 +0000306 const GrTexture* dstTexture = random.nextBool() ? dummyTextures[0] : dummyTextures[1];
bsalomon848faf02014-07-11 10:01:02 -0700307 if (!pdesc.setRandom(&random,
308 this,
309 dummyTextures[0]->asRenderTarget(),
310 dstTexture,
joshualittbd769d02014-09-04 08:56:46 -0700311 geometryProcessor.get(),
bsalomon848faf02014-07-11 10:01:02 -0700312 stages.get(),
313 numColorStages,
314 numCoverageStages,
315 currAttribIndex)) {
316 return false;
317 }
skia.committer@gmail.com05a2ee02013-04-02 07:01:34 +0000318
commit-bot@chromium.org9188a152013-09-05 18:28:24 +0000319 SkAutoTUnref<GrGLProgram> program(GrGLProgram::Create(this,
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000320 pdesc,
joshualittbd769d02014-09-04 08:56:46 -0700321 geometryProcessor.get(),
bsalomon@google.com2c84aa32013-06-06 20:28:57 +0000322 stages,
323 stages + numColorStages));
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000324 for (int s = 0; s < numStages; ++s) {
bsalomon@google.com504976e2013-05-09 13:45:02 +0000325 SkDELETE(stages[s]);
326 }
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000327 if (NULL == program.get()) {
328 return false;
329 }
joshualitt249af152014-09-15 11:41:13 -0700330
331 // We have to reset the drawstate because we might have added a gp
332 this->drawState()->reset();
bsalomon@google.comc3841b92012-08-02 18:11:43 +0000333 }
334 return true;
335}
bsalomon@google.coma8e686e2011-08-16 15:45:58 +0000336
tfarina@chromium.org4ee16bf2014-01-10 22:08:27 +0000337DEF_GPUTEST(GLPrograms, reporter, factory) {
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000338 for (int type = 0; type < GrContextFactory::kLastGLContextType; ++type) {
339 GrContext* context = factory->get(static_cast<GrContextFactory::GLContextType>(type));
bsalomon49f085d2014-09-05 13:34:00 -0700340 if (context) {
bsalomon@google.com042a2862013-02-04 18:39:24 +0000341 GrGpuGL* gpu = static_cast<GrGpuGL*>(context->getGpu());
bsalomon@google.comeb6879f2013-06-13 19:34:18 +0000342 int maxStages = 6;
bsalomon@google.com042a2862013-02-04 18:39:24 +0000343#if SK_ANGLE
344 // Some long shaders run out of temporary registers in the D3D compiler on ANGLE.
345 if (type == GrContextFactory::kANGLE_GLContextType) {
346 maxStages = 3;
347 }
348#endif
349 REPORTER_ASSERT(reporter, gpu->programUnitTest(maxStages));
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000350 }
351 }
bsalomon@google.coma8e686e2011-08-16 15:45:58 +0000352}
353
rmistry@google.comd6176b02012-08-23 18:14:13 +0000354// 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 +0000355// 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 +0000356// in the unit that could pass pointers to functions from the unit out to other translation units!
357// We force some of the effects that would otherwise be discarded to link here.
358
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +0000359#include "SkAlphaThresholdFilter.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +0000360#include "SkColorMatrixFilter.h"
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000361#include "SkLightingImageFilter.h"
bsalomon@google.com82aa7482012-08-13 14:22:17 +0000362#include "SkMagnifierImageFilter.h"
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000363
364void forceLinking();
365
366void forceLinking() {
367 SkLightingImageFilter::CreateDistantLitDiffuse(SkPoint3(0,0,0), 0, 0, 0);
commit-bot@chromium.org9109e182014-01-07 16:04:01 +0000368 SkAlphaThresholdFilter::Create(SkRegion(), .5f, .5f);
reed9fa60da2014-08-21 07:59:51 -0700369 SkAutoTUnref<SkImageFilter> mag(SkMagnifierImageFilter::Create(
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000370 SkRect::MakeWH(SK_Scalar1, SK_Scalar1), SK_Scalar1));
bsalomon@google.comadc65362013-01-28 14:26:09 +0000371 GrConfigConversionEffect::Create(NULL,
372 false,
373 GrConfigConversionEffect::kNone_PMConversion,
374 SkMatrix::I());
bsalomon@google.com67e78c92012-10-17 13:36:14 +0000375 SkScalar matrix[20];
commit-bot@chromium.org727a3522014-02-21 18:46:30 +0000376 SkAutoTUnref<SkColorMatrixFilter> cmf(SkColorMatrixFilter::Create(matrix));
bsalomon@google.coma1bf0ff2012-08-07 17:36:29 +0000377}
378
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +0000379#endif