Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(247)

Side by Side Diff: src/effects/SkAvoidXfermode.cpp

Issue 1658623002: Add gpu implementation of SkAvoidXfermode (Closed) Base URL: https://skia.googlesource.com/skia.git@master
Patch Set: Address code review comments & fix shader test Created 4 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « include/client/android/SkAvoidXfermode.h ('k') | src/effects/SkPixelXorXfermode.cpp » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright 2006 The Android Open Source Project 2 * Copyright 2006 The Android Open Source Project
3 * 3 *
4 * Use of this source code is governed by a BSD-style license that can be 4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file. 5 * found in the LICENSE file.
6 */ 6 */
7 7
8 #include "SkAvoidXfermode.h" 8 #include "SkAvoidXfermode.h"
9 #include "SkColorPriv.h" 9 #include "SkColorPriv.h"
10 #include "SkReadBuffer.h" 10 #include "SkReadBuffer.h"
(...skipping 146 matching lines...) Expand 10 before | Expand all | Expand 10 after
157 } 157 }
158 dst[i] = SkBlend3216(src[i], dst[i], d); 158 dst[i] = SkBlend3216(src[i], dst[i], d);
159 } 159 }
160 } 160 }
161 } 161 }
162 162
163 void SkAvoidXfermode::xferA8(SkAlpha dst[], const SkPMColor src[], int count, 163 void SkAvoidXfermode::xferA8(SkAlpha dst[], const SkPMColor src[], int count,
164 const SkAlpha aa[]) const { 164 const SkAlpha aa[]) const {
165 } 165 }
166 166
167
168 #if SK_SUPPORT_GPU
169
170 #include "GrFragmentProcessor.h"
171 #include "GrInvariantOutput.h"
172 #include "GrXferProcessor.h"
173 #include "glsl/GrGLSLFragmentProcessor.h"
174 #include "glsl/GrGLSLFragmentShaderBuilder.h"
175 #include "glsl/GrGLSLUniformHandler.h"
176 #include "glsl/GrGLSLXferProcessor.h"
177
178 ///////////////////////////////////////////////////////////////////////////////
179 // Fragment Processor
180 ///////////////////////////////////////////////////////////////////////////////
181
182 class GLAvoidFP;
183
184 class AvoidFP : public GrFragmentProcessor {
185 public:
186 static const GrFragmentProcessor* Create(SkColor opColor, uint8_t tolerance,
187 SkAvoidXfermode::Mode mode,
188 const GrFragmentProcessor* dst) {
189 return new AvoidFP(opColor, tolerance, mode, dst);
190 }
191
192 ~AvoidFP() override { }
193
194 const char* name() const override { return "Avoid"; }
195
196 SkString dumpInfo() const override {
197 SkString str;
198 str.appendf("Color: 0x%08x Tol: %d Mode: %s",
199 fOpColor, fTolerance,
200 fMode == SkAvoidXfermode::kAvoidColor_Mode ? "Avoid" : "Targ et");
201 return str;
202 }
203
204 SkColor opColor() const { return fOpColor; }
205 uint8_t tol() const { return fTolerance; }
206 SkAvoidXfermode::Mode mode() const { return fMode; }
207
208 private:
209 GrGLSLFragmentProcessor* onCreateGLSLInstance() const override;
210
211 void onGetGLSLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
212
213 bool onIsEqual(const GrFragmentProcessor& fpBase) const override {
214 const AvoidFP& fp = fpBase.cast<AvoidFP>();
215
216 return fOpColor == fp.fOpColor &&
217 fTolerance == fp.fTolerance &&
218 fMode == fp.fMode;
219 }
220
221 void onComputeInvariantOutput(GrInvariantOutput* inout) const override {
222 inout->setToUnknown(GrInvariantOutput::kWill_ReadInput);
223 }
224
225 AvoidFP(SkColor opColor, uint8_t tolerance,
226 SkAvoidXfermode::Mode mode, const GrFragmentProcessor* dst)
227 : fOpColor(opColor), fTolerance(tolerance), fMode(mode) {
228 this->initClassID<AvoidFP>();
229
230 SkASSERT(dst);
231 SkDEBUGCODE(int dstIndex = )this->registerChildProcessor(dst);
232 SkASSERT(0 == dstIndex);
233 }
234
235 SkColor fOpColor;
236 uint8_t fTolerance;
237 SkAvoidXfermode::Mode fMode;
238
239 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
240 typedef GrFragmentProcessor INHERITED;
241 };
242
243 ///////////////////////////////////////////////////////////////////////////////
244
245 // Add common code for calculating avoid's distance value
246 static void add_avoid_code(GrGLSLFragmentBuilder* fragBuilder,
247 const char* srcColor,
egdaniel 2016/02/03 16:41:24 why are src and output color neededed?
robertphillips 2016/02/03 16:56:31 Done.
248 const char* dstColor,
249 const char* outputColor,
250 const char* srcCoverage,
251 const char* kColorAndTolUni,
252 const char* kCoverageName,
253 SkAvoidXfermode::Mode mode) {
254
255 fragBuilder->codeAppendf("vec3 temp = %s.rgb - %s.rgb;", dstColor, kColorAnd TolUni);
256 fragBuilder->codeAppendf("float dist = max(max(abs(temp.r), abs(temp.g)), ab s(temp.b));");
257
258 if (SkAvoidXfermode::kTargetColor_Mode == mode) {
259 fragBuilder->codeAppendf("dist = 1.0 - dist;");
260 }
261
262 // the 'a' portion of the uniform is the scaled and inverted tolerance
263 fragBuilder->codeAppendf("dist = dist * %s.a - (%s.a - 1.0);",
264 kColorAndTolUni, kColorAndTolUni);
265
266 fragBuilder->codeAppendf("vec4 %s = vec4(dist);", kCoverageName);
267 if (srcCoverage) {
268 fragBuilder->codeAppendf("%s *= %s;", kCoverageName, srcCoverage);
269 }
270 }
271
272 class GLAvoidFP : public GrGLSLFragmentProcessor {
273 public:
274 void emitCode(EmitArgs& args) override {
275 const AvoidFP& avoid = args.fFp.cast<AvoidFP>();
276
277 GrGLSLFragmentBuilder* fragBuilder = args.fFragBuilder;
278 SkString dstColor("dstColor");
279 this->emitChild(0, nullptr, &dstColor, args);
280
281 fColorAndTolUni = args.fUniformHandler->addUniform(
282 GrGLSLUniformHandler::kFragment _Visibility,
283 kVec4f_GrSLType, kDefault_GrSLP recision,
284 "colorAndTol");
285 const char* kColorAndTolUni = args.fUniformHandler->getUniformCStr(fColo rAndTolUni);
286
287 const char* kCoverageName = "newCoverage1";
288
289 // add_avoid_code emits the code needed to compute the new coverage
290 add_avoid_code(fragBuilder,
291 args.fInputColor, dstColor.c_str(), args.fOutputColor, nu llptr,
292 kColorAndTolUni, kCoverageName, avoid.mode());
293
294 // The raster implementation's quantization and behavior yield a very no ticeable
295 // effect near zero (0.0039 = 1/256).
296 fragBuilder->codeAppendf("if (%s.r < 0.0039) { %s = %s; } else {",
297 kCoverageName, args.fOutputColor, dstColor);
298 fragBuilder->codeAppendf("%s = %s * %s + (vec4(1.0)-%s) * %s;",
299 args.fOutputColor,
300 kCoverageName, args.fInputColor ? args.fInputCo lor : "vec4(1.0)",
301 kCoverageName, dstColor.c_str());
302 fragBuilder->codeAppend("}");
303 }
304
305 static void GenKey(const GrProcessor& proc, const GrGLSLCaps&, GrProcessorKe yBuilder* b) {
306 const AvoidFP& avoid = proc.cast<AvoidFP>();
307 uint32_t key = avoid.mode() == SkAvoidXfermode::kTargetColor_Mode ? 1 : 0;
308 b->add32(key);
309 }
310
311 protected:
312 void onSetData(const GrGLSLProgramDataManager& pdman, const GrProcessor& pro c) override {
313 const AvoidFP& avoid = proc.cast<AvoidFP>();
314 pdman.set4f(fColorAndTolUni,
315 SkColorGetR(avoid.opColor())/255.0f,
316 SkColorGetG(avoid.opColor())/255.0f,
317 SkColorGetB(avoid.opColor())/255.0f,
318 256.0f/(avoid.tol()+1.0f));
319 }
320
321 private:
322 GrGLSLProgramDataManager::UniformHandle fColorAndTolUni;
323
324 typedef GrGLSLFragmentProcessor INHERITED;
325 };
326
327 ///////////////////////////////////////////////////////////////////////////////
328
329 GrGLSLFragmentProcessor* AvoidFP::onCreateGLSLInstance() const {
330 return new GLAvoidFP;
331 }
332
333 void AvoidFP::onGetGLSLProcessorKey(const GrGLSLCaps& caps, GrProcessorKeyBuilde r* b) const {
334 GLAvoidFP::GenKey(*this, caps, b);
335 }
336
337 const GrFragmentProcessor* AvoidFP::TestCreate(GrProcessorTestData* d) {
338 SkColor opColor = d->fRandom->nextU();
339 uint8_t tolerance = d->fRandom->nextBits(8);
340 SkAvoidXfermode::Mode mode = d->fRandom->nextBool() ? SkAvoidXfermode::kAvoi dColor_Mode
341 : SkAvoidXfermode::kTarg etColor_Mode;
342
343 SkAutoTUnref<const GrFragmentProcessor> dst(GrProcessorUnitTest::CreateChild FP(d));
344 return new AvoidFP(opColor, tolerance, mode, dst);
345 }
346
347 GR_DEFINE_FRAGMENT_PROCESSOR_TEST(AvoidFP);
348
349 ///////////////////////////////////////////////////////////////////////////////
350 // Xfer Processor
351 ///////////////////////////////////////////////////////////////////////////////
352
353 class AvoidXP : public GrXferProcessor {
354 public:
355 AvoidXP(const DstTexture* dstTexture, bool hasMixedSamples,
356 SkColor opColor, uint8_t tolerance, SkAvoidXfermode::Mode mode)
357 : INHERITED(dstTexture, true, hasMixedSamples)
358 , fOpColor(opColor)
359 , fTolerance(tolerance)
360 , fMode(mode) {
361 this->initClassID<AvoidXP>();
362 }
363
364 const char* name() const override { return "Avoid"; }
365
366 GrGLSLXferProcessor* createGLSLInstance() const override;
367
368 SkColor opColor() const { return fOpColor; }
369 uint8_t tol() const { return fTolerance; }
370 SkAvoidXfermode::Mode mode() const { return fMode; }
371
372 private:
373 GrXferProcessor::OptFlags onGetOptimizations(const GrPipelineOptimizations& optimizations,
374 bool doesStencilWrite,
375 GrColor* overrideColor,
376 const GrCaps& caps) const overr ide {
377 return GrXferProcessor::kNone_OptFlags;
378 }
379
380 void onGetGLSLProcessorKey(const GrGLSLCaps& caps, GrProcessorKeyBuilder* b) const override;
381
382 bool onIsEqual(const GrXferProcessor& xpBase) const override {
383 const AvoidXP& xp = xpBase.cast<AvoidXP>();
384
385 return fOpColor == xp.fOpColor &&
386 fTolerance == xp.fTolerance &&
387 fMode == xp.fMode;
388 }
389
390 SkColor fOpColor;
391 uint8_t fTolerance;
392 SkAvoidXfermode::Mode fMode;
393
394 typedef GrXferProcessor INHERITED;
395 };
396
397 ///////////////////////////////////////////////////////////////////////////////
398
399 class GLAvoidXP : public GrGLSLXferProcessor {
400 public:
401 static void GenKey(const GrProcessor& processor, const GrGLSLCaps&, GrProces sorKeyBuilder* b) {
402 const AvoidXP& avoid = processor.cast<AvoidXP>();
403 uint32_t key = SkAvoidXfermode::kTargetColor_Mode == avoid.mode() ? 1 : 0;
404 b->add32(key);
405 }
406
407 private:
408 void emitBlendCodeForDstRead(GrGLSLXPFragmentBuilder* fragBuilder,
409 GrGLSLUniformHandler* uniformHandler,
410 const char* srcColor,
411 const char* srcCoverage,
412 const char* dstColor,
413 const char* outColor,
414 const char* outColorSecondary,
415 const GrXferProcessor& proc) override {
416 const AvoidXP& avoid = proc.cast<AvoidXP>();
417
418 fColorAndTolUni = uniformHandler->addUniform(GrGLSLUniformHandler::kFrag ment_Visibility,
419 kVec4f_GrSLType, kDefault_G rSLPrecision,
420 "colorAndTol");
421 const char* kColorandTolUni = uniformHandler->getUniformCStr(fColorAndTo lUni);
422
423 const char* kCoverageName = "newCoverage1";
424
425 // add_avoid_code emits the code needed to compute the new coverage
426 add_avoid_code(fragBuilder,
427 srcColor, dstColor, outColor, srcCoverage,
428 kColorandTolUni, kCoverageName, avoid.mode());
429
430 // The raster implementation's quantization and behavior yield a very no ticeable
431 // effect near zero (0.0039 = 1/256).
432 fragBuilder->codeAppendf("if (%s.r < 0.0039) { %s = %s; } else {",
433 kCoverageName, outColor, dstColor);
434 fragBuilder->codeAppendf("%s = %s;", outColor, srcColor ? srcColor : "ve c4(1.0)");
435 INHERITED::DefaultCoverageModulation(fragBuilder, kCoverageName, dstColo r, outColor,
436 outColorSecondary, proc);
437 fragBuilder->codeAppend("}");
438 }
439
440 void onSetData(const GrGLSLProgramDataManager& pdman,
441 const GrXferProcessor& processor) override {
442 const AvoidXP& avoid = processor.cast<AvoidXP>();
443 pdman.set4f(fColorAndTolUni,
444 SkColorGetR(avoid.opColor())/255.0f,
445 SkColorGetG(avoid.opColor())/255.0f,
446 SkColorGetB(avoid.opColor())/255.0f,
447 256.0f/(avoid.tol()+1.0f));
448 };
449
450 GrGLSLProgramDataManager::UniformHandle fColorAndTolUni;
451
452 typedef GrGLSLXferProcessor INHERITED;
453 };
454
455 ///////////////////////////////////////////////////////////////////////////////
456
457 void AvoidXP::onGetGLSLProcessorKey(const GrGLSLCaps& caps, GrProcessorKeyBuilde r* b) const {
458 GLAvoidXP::GenKey(*this, caps, b);
459 }
460
461 GrGLSLXferProcessor* AvoidXP::createGLSLInstance() const { return new GLAvoidXP; }
462
463 ///////////////////////////////////////////////////////////////////////////////
464 class GrAvoidXPFactory : public GrXPFactory {
465 public:
466 static GrXPFactory* Create(SkColor opColor, uint8_t tolerance,
467 SkAvoidXfermode::Mode mode) {
468 return new GrAvoidXPFactory(opColor, tolerance, mode);
469 }
470
471 void getInvariantBlendedColor(const GrProcOptInfo& colorPOI,
472 GrXPFactory::InvariantBlendedColor* blendedCol or) const override {
473 blendedColor->fWillBlendWithDst = true;
474 blendedColor->fKnownColorFlags = kNone_GrColorComponentFlags;
475 }
476
477 private:
478 GrAvoidXPFactory(SkColor opColor, uint8_t tolerance, SkAvoidXfermode::Mode m ode)
479 : fOpColor(opColor)
480 , fTolerance(tolerance)
481 , fMode(mode) {
482 this->initClassID<GrAvoidXPFactory>();
483 }
484
485 GrXferProcessor* onCreateXferProcessor(const GrCaps& caps,
486 const GrPipelineOptimizations& optimi zations,
487 bool hasMixedSamples,
488 const DstTexture* dstTexture) const o verride {
489 return new AvoidXP(dstTexture, hasMixedSamples, fOpColor, fTolerance, fM ode);
490 }
491
492 bool onWillReadDstColor(const GrCaps& caps,
493 const GrPipelineOptimizations& optimizations,
494 bool hasMixedSamples) const override {
495 return true;
496 }
497
498 bool onIsEqual(const GrXPFactory& xpfBase) const override {
499 const GrAvoidXPFactory& xpf = xpfBase.cast<GrAvoidXPFactory>();
500 return fOpColor == xpf.fOpColor &&
501 fTolerance == xpf.fTolerance &&
502 fMode == xpf.fMode;
503 }
504
505 GR_DECLARE_XP_FACTORY_TEST;
506
507 SkColor fOpColor;
508 uint8_t fTolerance;
509 SkAvoidXfermode::Mode fMode;
510
511 typedef GrXPFactory INHERITED;
512 };
513
514 GR_DEFINE_XP_FACTORY_TEST(GrAvoidXPFactory);
515
516 const GrXPFactory* GrAvoidXPFactory::TestCreate(GrProcessorTestData* d) {
517 SkColor opColor = d->fRandom->nextU();
518 uint8_t tolerance = d->fRandom->nextBits(8);
519 SkAvoidXfermode::Mode mode = d->fRandom->nextBool() ? SkAvoidXfermode::kAvoi dColor_Mode
520 : SkAvoidXfermode::kTarg etColor_Mode;
521 return GrAvoidXPFactory::Create(opColor, tolerance, mode);
522 }
523
524 ///////////////////////////////////////////////////////////////////////////////
525
526 bool SkAvoidXfermode::asFragmentProcessor(const GrFragmentProcessor** output,
527 const GrFragmentProcessor* dst) const {
528 if (output) {
529 *output = AvoidFP::Create(fOpColor, fTolerance, fMode, dst);
530 }
531 return true;
532 }
533
534 bool SkAvoidXfermode::asXPFactory(GrXPFactory** xpf) const {
535 if (xpf) {
536 *xpf = GrAvoidXPFactory::Create(fOpColor, fTolerance, fMode);
537 }
538 return true;
539 }
540 #endif
541
167 #ifndef SK_IGNORE_TO_STRING 542 #ifndef SK_IGNORE_TO_STRING
168 void SkAvoidXfermode::toString(SkString* str) const { 543 void SkAvoidXfermode::toString(SkString* str) const {
169 str->append("AvoidXfermode: opColor: "); 544 str->append("AvoidXfermode: opColor: ");
170 str->appendHex(fOpColor); 545 str->appendHex(fOpColor);
171 str->appendf("distMul: %d ", fDistMul); 546 str->appendf("tolerance: %d ", fTolerance);
172 547
173 static const char* gModeStrings[] = { "Avoid", "Target" }; 548 static const char* gModeStrings[] = { "Avoid", "Target" };
174 549
175 str->appendf("mode: %s", gModeStrings[fMode]); 550 str->appendf("mode: %s", gModeStrings[fMode]);
176 } 551 }
177 #endif 552 #endif
OLDNEW
« no previous file with comments | « include/client/android/SkAvoidXfermode.h ('k') | src/effects/SkPixelXorXfermode.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698