blob: 2f96ade45081f3dd199c4f87b4b1dfb20fc1dc28 [file] [log] [blame]
joshualitt4973d9d2014-11-08 09:24:25 -08001/*
2 * Copyright 2014 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
8#include "GrDefaultGeoProcFactory.h"
9
bungeman06ca8ec2016-06-09 08:01:03 -070010#include "SkRefCnt.h"
Brian Osmanfa6d8652017-05-31 09:37:27 -040011#include "glsl/GrGLSLColorSpaceXformHelper.h"
egdaniel2d721d32015-11-11 13:06:05 -080012#include "glsl/GrGLSLFragmentShaderBuilder.h"
egdaniele659a582015-11-13 09:55:43 -080013#include "glsl/GrGLSLGeometryProcessor.h"
Chris Daltonc17bf322017-10-24 10:59:03 -060014#include "glsl/GrGLSLVertexGeoBuilder.h"
egdaniel0eafe792015-11-20 14:01:22 -080015#include "glsl/GrGLSLVarying.h"
egdaniel7ea439b2015-12-03 09:20:44 -080016#include "glsl/GrGLSLUniformHandler.h"
egdaniel64c47282015-11-13 06:54:19 -080017#include "glsl/GrGLSLUtil.h"
joshualitt4973d9d2014-11-08 09:24:25 -080018
19/*
20 * The default Geometry Processor simply takes position and multiplies it by the uniform view
21 * matrix. It also leaves coverage untouched. Behind the scenes, we may add per vertex color or
22 * local coords.
23 */
joshualittb2aa7cb2015-08-05 11:05:22 -070024
25enum GPFlag {
Brian Salomon3de0aee2017-01-29 09:34:17 -050026 kColorAttribute_GPFlag = 0x1,
27 kColorAttributeIsSkColor_GPFlag = 0x2,
28 kLocalCoordAttribute_GPFlag = 0x4,
29 kCoverageAttribute_GPFlag = 0x8,
Brian Osmanfa6d8652017-05-31 09:37:27 -040030
31 kLinearizeColorAttribute_GPFlag = 0x10,
joshualittb2aa7cb2015-08-05 11:05:22 -070032};
33
joshualitt4973d9d2014-11-08 09:24:25 -080034class DefaultGeoProc : public GrGeometryProcessor {
35public:
bungeman06ca8ec2016-06-09 08:01:03 -070036 static sk_sp<GrGeometryProcessor> Make(uint32_t gpTypeFlags,
Brian Salomon3de0aee2017-01-29 09:34:17 -050037 GrColor color,
Brian Osmanfa6d8652017-05-31 09:37:27 -040038 sk_sp<GrColorSpaceXform> colorSpaceXform,
Brian Salomon3de0aee2017-01-29 09:34:17 -050039 const SkMatrix& viewMatrix,
40 const SkMatrix& localMatrix,
41 bool localCoordsWillBeRead,
42 uint8_t coverage) {
bungeman06ca8ec2016-06-09 08:01:03 -070043 return sk_sp<GrGeometryProcessor>(new DefaultGeoProc(
Brian Osmanfa6d8652017-05-31 09:37:27 -040044 gpTypeFlags, color, std::move(colorSpaceXform), viewMatrix, localMatrix, coverage,
45 localCoordsWillBeRead));
joshualitt4973d9d2014-11-08 09:24:25 -080046 }
47
mtklein36352bf2015-03-25 18:17:31 -070048 const char* name() const override { return "DefaultGeometryProcessor"; }
joshualitt4973d9d2014-11-08 09:24:25 -080049
joshualitt71c92602015-01-14 08:12:47 -080050 const Attribute* inPosition() const { return fInPosition; }
51 const Attribute* inColor() const { return fInColor; }
52 const Attribute* inLocalCoords() const { return fInLocalCoords; }
53 const Attribute* inCoverage() const { return fInCoverage; }
joshualitt88c23fc2015-05-13 14:18:07 -070054 GrColor color() const { return fColor; }
joshualittb8c241a2015-05-19 08:23:30 -070055 bool hasVertexColor() const { return SkToBool(fInColor); }
joshualitte578a952015-05-14 10:09:13 -070056 const SkMatrix& viewMatrix() const { return fViewMatrix; }
joshualitte3ababe2015-05-15 07:56:07 -070057 const SkMatrix& localMatrix() const { return fLocalMatrix; }
bsalomon7765a472015-07-08 11:26:37 -070058 bool localCoordsWillBeRead() const { return fLocalCoordsWillBeRead; }
joshualitt9b989322014-12-15 14:16:27 -080059 uint8_t coverage() const { return fCoverage; }
joshualittb8c241a2015-05-19 08:23:30 -070060 bool hasVertexCoverage() const { return SkToBool(fInCoverage); }
Brian Osmanfa6d8652017-05-31 09:37:27 -040061 bool linearizeColor() const {
62 // Linearization should only happen with SkColor
63 bool linearize = SkToBool(fFlags & kLinearizeColorAttribute_GPFlag);
64 SkASSERT(!linearize || (fFlags & kColorAttributeIsSkColor_GPFlag));
65 return linearize;
66 }
joshualitt9b989322014-12-15 14:16:27 -080067
egdaniel57d3b032015-11-13 11:57:27 -080068 class GLSLProcessor : public GrGLSLGeometryProcessor {
joshualitt4973d9d2014-11-08 09:24:25 -080069 public:
egdaniel57d3b032015-11-13 11:57:27 -080070 GLSLProcessor()
joshualitt5559ca22015-05-21 15:50:36 -070071 : fViewMatrix(SkMatrix::InvalidMatrix()), fColor(GrColor_ILLEGAL), fCoverage(0xff) {}
joshualitt4973d9d2014-11-08 09:24:25 -080072
mtklein36352bf2015-03-25 18:17:31 -070073 void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
joshualitt2dd1ae02014-12-03 06:24:10 -080074 const DefaultGeoProc& gp = args.fGP.cast<DefaultGeoProc>();
egdaniel4ca2e602015-11-18 08:01:26 -080075 GrGLSLVertexBuilder* vertBuilder = args.fVertBuilder;
Chris Dalton60283612018-02-14 13:38:14 -070076 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
egdaniel0eafe792015-11-20 14:01:22 -080077 GrGLSLVaryingHandler* varyingHandler = args.fVaryingHandler;
egdaniel7ea439b2015-12-03 09:20:44 -080078 GrGLSLUniformHandler* uniformHandler = args.fUniformHandler;
joshualitt4973d9d2014-11-08 09:24:25 -080079
joshualittabb52a12015-01-13 15:02:10 -080080 // emit attributes
egdaniel0eafe792015-11-20 14:01:22 -080081 varyingHandler->emitAttributes(gp);
joshualitt2dd1ae02014-12-03 06:24:10 -080082
83 // Setup pass through color
Brian Salomonbfd51832017-01-04 13:22:08 -050084 if (gp.hasVertexColor()) {
Chris Dalton27372882017-12-08 13:34:21 -070085 GrGLSLVarying varying(kHalf4_GrSLType);
Brian Salomon3de0aee2017-01-29 09:34:17 -050086 varyingHandler->addVarying("color", &varying);
Brian Osmanfa6d8652017-05-31 09:37:27 -040087
88 // There are several optional steps to process the color. Start with the attribute:
Brian Salomon70132d02018-05-29 15:33:06 -040089 vertBuilder->codeAppendf("half4 color = %s;", gp.inColor()->name());
Brian Osmanfa6d8652017-05-31 09:37:27 -040090
91 // Linearize
92 if (gp.linearizeColor()) {
93 SkString srgbFuncName;
94 static const GrShaderVar gSrgbArgs[] = {
Ethan Nicholasf7b88202017-09-18 14:10:39 -040095 GrShaderVar("x", kHalf_GrSLType),
Brian Osmanfa6d8652017-05-31 09:37:27 -040096 };
Ethan Nicholasf7b88202017-09-18 14:10:39 -040097 vertBuilder->emitFunction(kHalf_GrSLType,
Brian Osmanfa6d8652017-05-31 09:37:27 -040098 "srgb_to_linear",
99 SK_ARRAY_COUNT(gSrgbArgs),
100 gSrgbArgs,
101 "return (x <= 0.04045) ? (x / 12.92) "
102 ": pow((x + 0.055) / 1.055, 2.4);",
103 &srgbFuncName);
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400104 vertBuilder->codeAppendf("color = half4(%s(%s.r), %s(%s.g), %s(%s.b), %s.a);",
Brian Salomon70132d02018-05-29 15:33:06 -0400105 srgbFuncName.c_str(), gp.inColor()->name(),
106 srgbFuncName.c_str(), gp.inColor()->name(),
107 srgbFuncName.c_str(), gp.inColor()->name(),
108 gp.inColor()->name());
Brian Salomon3de0aee2017-01-29 09:34:17 -0500109 }
Brian Osmanfa6d8652017-05-31 09:37:27 -0400110
111 // For SkColor, do a red/blue swap and premul
112 if (gp.fFlags & kColorAttributeIsSkColor_GPFlag) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400113 vertBuilder->codeAppend("color = half4(color.a * color.bgr, color.a);");
Brian Osmanfa6d8652017-05-31 09:37:27 -0400114 }
115
116 // Do color-correction to destination gamut
117 if (gp.linearizeColor()) {
118 fColorSpaceHelper.emitCode(uniformHandler, gp.fColorSpaceXform.get(),
119 kVertex_GrShaderFlag);
Brian Osmanc891b102018-06-14 14:50:17 -0400120 SkString xformedColor;
121 vertBuilder->appendColorGamutXform(&xformedColor, "color", &fColorSpaceHelper);
122 vertBuilder->codeAppendf("color = %s;", xformedColor.c_str());
Brian Osmanfa6d8652017-05-31 09:37:27 -0400123 }
124 vertBuilder->codeAppendf("%s = color;\n", varying.vsOut());
Brian Salomon3de0aee2017-01-29 09:34:17 -0500125 fragBuilder->codeAppendf("%s = %s;", args.fOutputColor, varying.fsIn());
Brian Salomonbfd51832017-01-04 13:22:08 -0500126 } else {
127 this->setupUniformColor(fragBuilder, uniformHandler, args.fOutputColor,
128 &fColorUniform);
joshualittb8c241a2015-05-19 08:23:30 -0700129 }
130
joshualittabb52a12015-01-13 15:02:10 -0800131 // Setup position
Brian Salomon7f235432017-08-16 09:41:48 -0400132 this->writeOutputPosition(vertBuilder,
133 uniformHandler,
134 gpArgs,
Brian Salomon70132d02018-05-29 15:33:06 -0400135 gp.inPosition()->name(),
Brian Salomon7f235432017-08-16 09:41:48 -0400136 gp.viewMatrix(),
137 &fViewMatrixUniform);
joshualitt4973d9d2014-11-08 09:24:25 -0800138
Brian Salomon969bdef2018-06-06 17:16:05 -0400139 if (gp.inLocalCoords()) {
joshualittabb52a12015-01-13 15:02:10 -0800140 // emit transforms with explicit local coords
egdaniel7ea439b2015-12-03 09:20:44 -0800141 this->emitTransforms(vertBuilder,
egdaniel0eafe792015-11-20 14:01:22 -0800142 varyingHandler,
egdaniel7ea439b2015-12-03 09:20:44 -0800143 uniformHandler,
Brian Salomon04460cc2017-12-06 14:47:42 -0500144 gp.inLocalCoords()->asShaderVar(),
egdaniel4ca2e602015-11-18 08:01:26 -0800145 gp.localMatrix(),
bsalomona624bf32016-09-20 09:12:47 -0700146 args.fFPCoordTransformHandler);
joshualittabb52a12015-01-13 15:02:10 -0800147 } else {
148 // emit transforms with position
egdaniel7ea439b2015-12-03 09:20:44 -0800149 this->emitTransforms(vertBuilder,
egdaniel0eafe792015-11-20 14:01:22 -0800150 varyingHandler,
egdaniel7ea439b2015-12-03 09:20:44 -0800151 uniformHandler,
Brian Salomon04460cc2017-12-06 14:47:42 -0500152 gp.inPosition()->asShaderVar(),
egdaniel4ca2e602015-11-18 08:01:26 -0800153 gp.localMatrix(),
bsalomona624bf32016-09-20 09:12:47 -0700154 args.fFPCoordTransformHandler);
joshualittabb52a12015-01-13 15:02:10 -0800155 }
156
joshualitt2dd1ae02014-12-03 06:24:10 -0800157 // Setup coverage as pass through
Brian Salomon8c852be2017-01-04 10:44:42 -0500158 if (gp.hasVertexCoverage()) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400159 fragBuilder->codeAppendf("half alpha = 1.0;");
Brian Salomon8c852be2017-01-04 10:44:42 -0500160 varyingHandler->addPassThroughAttribute(gp.inCoverage(), "alpha");
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400161 fragBuilder->codeAppendf("%s = half4(alpha);", args.fOutputCoverage);
Brian Salomon8c852be2017-01-04 10:44:42 -0500162 } else if (gp.coverage() == 0xff) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400163 fragBuilder->codeAppendf("%s = half4(1);", args.fOutputCoverage);
Brian Salomon8c852be2017-01-04 10:44:42 -0500164 } else {
165 const char* fragCoverage;
166 fCoverageUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400167 kHalf_GrSLType,
Brian Salomon8c852be2017-01-04 10:44:42 -0500168 "Coverage",
169 &fragCoverage);
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400170 fragBuilder->codeAppendf("%s = half4(%s);", args.fOutputCoverage, fragCoverage);
joshualitt2dd1ae02014-12-03 06:24:10 -0800171 }
joshualitt4973d9d2014-11-08 09:24:25 -0800172 }
173
joshualitt87f48d92014-12-04 10:41:40 -0800174 static inline void GenKey(const GrGeometryProcessor& gp,
Brian Salomon94efbf52016-11-29 13:43:05 -0500175 const GrShaderCaps&,
joshualitt87f48d92014-12-04 10:41:40 -0800176 GrProcessorKeyBuilder* b) {
joshualitt2dd1ae02014-12-03 06:24:10 -0800177 const DefaultGeoProc& def = gp.cast<DefaultGeoProc>();
joshualitt8fc6c2d2014-12-22 15:27:05 -0800178 uint32_t key = def.fFlags;
Brian Salomon3de0aee2017-01-29 09:34:17 -0500179 key |= (def.coverage() == 0xff) ? 0x10 : 0;
180 key |= (def.localCoordsWillBeRead() && def.localMatrix().hasPerspective()) ? 0x20 : 0x0;
181 key |= ComputePosKey(def.viewMatrix()) << 20;
joshualitt8fc6c2d2014-12-22 15:27:05 -0800182 b->add32(key);
Brian Osmanfa6d8652017-05-31 09:37:27 -0400183 if (def.linearizeColor()) {
184 b->add32(GrColorSpaceXform::XformKey(def.fColorSpaceXform.get()));
185 }
joshualitt2dd1ae02014-12-03 06:24:10 -0800186 }
joshualitt4973d9d2014-11-08 09:24:25 -0800187
egdaniel018fb622015-10-28 07:26:40 -0700188 void setData(const GrGLSLProgramDataManager& pdman,
bsalomona624bf32016-09-20 09:12:47 -0700189 const GrPrimitiveProcessor& gp,
190 FPCoordTransformIter&& transformIter) override {
joshualitte578a952015-05-14 10:09:13 -0700191 const DefaultGeoProc& dgp = gp.cast<DefaultGeoProc>();
joshualitt5559ca22015-05-21 15:50:36 -0700192
193 if (!dgp.viewMatrix().isIdentity() && !fViewMatrix.cheapEqualTo(dgp.viewMatrix())) {
194 fViewMatrix = dgp.viewMatrix();
egdaniel018fb622015-10-28 07:26:40 -0700195 float viewMatrix[3 * 3];
egdaniel64c47282015-11-13 06:54:19 -0800196 GrGLSLGetMatrix<3>(viewMatrix, fViewMatrix);
joshualitt5559ca22015-05-21 15:50:36 -0700197 pdman.setMatrix3f(fViewMatrixUniform, viewMatrix);
198 }
joshualittee2af952014-12-30 09:04:15 -0800199
joshualittb8c241a2015-05-19 08:23:30 -0700200 if (dgp.color() != fColor && !dgp.hasVertexColor()) {
egdaniel018fb622015-10-28 07:26:40 -0700201 float c[4];
joshualittb8c241a2015-05-19 08:23:30 -0700202 GrColorToRGBAFloat(dgp.color(), c);
joshualitt9b989322014-12-15 14:16:27 -0800203 pdman.set4fv(fColorUniform, 1, c);
joshualittb8c241a2015-05-19 08:23:30 -0700204 fColor = dgp.color();
joshualitt9b989322014-12-15 14:16:27 -0800205 }
joshualittb8c241a2015-05-19 08:23:30 -0700206
Brian Salomon8c852be2017-01-04 10:44:42 -0500207 if (dgp.coverage() != fCoverage && !dgp.hasVertexCoverage()) {
joshualittb8c241a2015-05-19 08:23:30 -0700208 pdman.set1f(fCoverageUniform, GrNormalizeByteToFloat(dgp.coverage()));
209 fCoverage = dgp.coverage();
joshualitt9b989322014-12-15 14:16:27 -0800210 }
bsalomona624bf32016-09-20 09:12:47 -0700211 this->setTransformDataHelper(dgp.fLocalMatrix, pdman, &transformIter);
Brian Osmanfa6d8652017-05-31 09:37:27 -0400212
213 if (dgp.linearizeColor() && dgp.fColorSpaceXform) {
214 fColorSpaceHelper.setData(pdman, dgp.fColorSpaceXform.get());
215 }
joshualitte3ababe2015-05-15 07:56:07 -0700216 }
217
joshualitt4973d9d2014-11-08 09:24:25 -0800218 private:
joshualitt5559ca22015-05-21 15:50:36 -0700219 SkMatrix fViewMatrix;
joshualitt9b989322014-12-15 14:16:27 -0800220 GrColor fColor;
221 uint8_t fCoverage;
joshualitt5559ca22015-05-21 15:50:36 -0700222 UniformHandle fViewMatrixUniform;
joshualitt9b989322014-12-15 14:16:27 -0800223 UniformHandle fColorUniform;
224 UniformHandle fCoverageUniform;
Brian Osmanfa6d8652017-05-31 09:37:27 -0400225 GrGLSLColorSpaceXformHelper fColorSpaceHelper;
joshualitt9b989322014-12-15 14:16:27 -0800226
egdaniele659a582015-11-13 09:55:43 -0800227 typedef GrGLSLGeometryProcessor INHERITED;
joshualitt4973d9d2014-11-08 09:24:25 -0800228 };
229
Brian Salomon94efbf52016-11-29 13:43:05 -0500230 void getGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override {
egdaniel57d3b032015-11-13 11:57:27 -0800231 GLSLProcessor::GenKey(*this, caps, b);
joshualitteb2a6762014-12-04 11:35:33 -0800232 }
233
Brian Salomon94efbf52016-11-29 13:43:05 -0500234 GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override {
egdaniel57d3b032015-11-13 11:57:27 -0800235 return new GLSLProcessor();
joshualitteb2a6762014-12-04 11:35:33 -0800236 }
237
joshualitt4973d9d2014-11-08 09:24:25 -0800238private:
joshualitt8059eb92014-12-29 15:10:07 -0800239 DefaultGeoProc(uint32_t gpTypeFlags,
240 GrColor color,
Brian Osmanfa6d8652017-05-31 09:37:27 -0400241 sk_sp<GrColorSpaceXform> colorSpaceXform,
joshualitt8059eb92014-12-29 15:10:07 -0800242 const SkMatrix& viewMatrix,
243 const SkMatrix& localMatrix,
joshualittb8c241a2015-05-19 08:23:30 -0700244 uint8_t coverage,
Brian Salomon8c852be2017-01-04 10:44:42 -0500245 bool localCoordsWillBeRead)
Ethan Nicholasabff9562017-10-09 10:54:08 -0400246 : INHERITED(kDefaultGeoProc_ClassID)
247 , fColor(color)
Brian Salomon8c852be2017-01-04 10:44:42 -0500248 , fViewMatrix(viewMatrix)
249 , fLocalMatrix(localMatrix)
250 , fCoverage(coverage)
251 , fFlags(gpTypeFlags)
Brian Osmanfa6d8652017-05-31 09:37:27 -0400252 , fLocalCoordsWillBeRead(localCoordsWillBeRead)
253 , fColorSpaceXform(std::move(colorSpaceXform)) {
Ethan Nicholasfa7ee242017-09-25 09:52:04 -0400254 fInPosition = &this->addVertexAttrib("inPosition", kFloat2_GrVertexAttribType);
Brian Salomon3de0aee2017-01-29 09:34:17 -0500255 if (fFlags & kColorAttribute_GPFlag) {
Ethan Nicholasfa7ee242017-09-25 09:52:04 -0400256 fInColor = &this->addVertexAttrib("inColor", kUByte4_norm_GrVertexAttribType);
joshualitt2dd1ae02014-12-03 06:24:10 -0800257 }
Brian Salomon3de0aee2017-01-29 09:34:17 -0500258 if (fFlags & kLocalCoordAttribute_GPFlag) {
Ethan Nicholasfa7ee242017-09-25 09:52:04 -0400259 fInLocalCoords = &this->addVertexAttrib("inLocalCoord", kFloat2_GrVertexAttribType);
joshualitt2dd1ae02014-12-03 06:24:10 -0800260 }
Brian Salomon3de0aee2017-01-29 09:34:17 -0500261 if (fFlags & kCoverageAttribute_GPFlag) {
Ethan Nicholasfa7ee242017-09-25 09:52:04 -0400262 fInCoverage = &this->addVertexAttrib("inCoverage", kHalf_GrVertexAttribType);
joshualitt2dd1ae02014-12-03 06:24:10 -0800263 }
264 }
joshualitt4973d9d2014-11-08 09:24:25 -0800265
Brian Salomon3de0aee2017-01-29 09:34:17 -0500266 const Attribute* fInPosition = nullptr;
267 const Attribute* fInColor = nullptr;
268 const Attribute* fInLocalCoords = nullptr;
269 const Attribute* fInCoverage = nullptr;
joshualitt88c23fc2015-05-13 14:18:07 -0700270 GrColor fColor;
joshualitte578a952015-05-14 10:09:13 -0700271 SkMatrix fViewMatrix;
joshualitte3ababe2015-05-15 07:56:07 -0700272 SkMatrix fLocalMatrix;
joshualitt9b989322014-12-15 14:16:27 -0800273 uint8_t fCoverage;
joshualitt2dd1ae02014-12-03 06:24:10 -0800274 uint32_t fFlags;
bsalomon7765a472015-07-08 11:26:37 -0700275 bool fLocalCoordsWillBeRead;
Brian Osmanfa6d8652017-05-31 09:37:27 -0400276 sk_sp<GrColorSpaceXform> fColorSpaceXform;
joshualitt4973d9d2014-11-08 09:24:25 -0800277
Brian Salomon0c26a9d2017-07-06 10:09:38 -0400278 GR_DECLARE_GEOMETRY_PROCESSOR_TEST
joshualitt5478d422014-11-14 16:00:38 -0800279
joshualitt2e3b3e32014-12-09 13:31:14 -0800280 typedef GrGeometryProcessor INHERITED;
joshualitt4973d9d2014-11-08 09:24:25 -0800281};
282
283GR_DEFINE_GEOMETRY_PROCESSOR_TEST(DefaultGeoProc);
284
Hal Canary6f6961e2017-01-31 13:50:44 -0500285#if GR_TEST_UTILS
bungeman06ca8ec2016-06-09 08:01:03 -0700286sk_sp<GrGeometryProcessor> DefaultGeoProc::TestCreate(GrProcessorTestData* d) {
joshualitt2dd1ae02014-12-03 06:24:10 -0800287 uint32_t flags = 0;
joshualitt0067ff52015-07-08 14:26:19 -0700288 if (d->fRandom->nextBool()) {
Brian Salomon3de0aee2017-01-29 09:34:17 -0500289 flags |= kColorAttribute_GPFlag;
joshualitt4973d9d2014-11-08 09:24:25 -0800290 }
joshualitt0067ff52015-07-08 14:26:19 -0700291 if (d->fRandom->nextBool()) {
Brian Salomon3de0aee2017-01-29 09:34:17 -0500292 flags |= kColorAttributeIsSkColor_GPFlag;
joshualitt4973d9d2014-11-08 09:24:25 -0800293 }
joshualitt0067ff52015-07-08 14:26:19 -0700294 if (d->fRandom->nextBool()) {
Brian Salomon3de0aee2017-01-29 09:34:17 -0500295 flags |= kCoverageAttribute_GPFlag;
296 }
297 if (d->fRandom->nextBool()) {
298 flags |= kLocalCoordAttribute_GPFlag;
joshualittb2aa7cb2015-08-05 11:05:22 -0700299 }
joshualitt2dd1ae02014-12-03 06:24:10 -0800300
bungeman06ca8ec2016-06-09 08:01:03 -0700301 return DefaultGeoProc::Make(flags,
302 GrRandomColor(d->fRandom),
Brian Osmanfa6d8652017-05-31 09:37:27 -0400303 GrTest::TestColorXform(d->fRandom),
bungeman06ca8ec2016-06-09 08:01:03 -0700304 GrTest::TestMatrix(d->fRandom),
305 GrTest::TestMatrix(d->fRandom),
306 d->fRandom->nextBool(),
bungeman06ca8ec2016-06-09 08:01:03 -0700307 GrRandomCoverage(d->fRandom));
joshualitt4973d9d2014-11-08 09:24:25 -0800308}
Hal Canary6f6961e2017-01-31 13:50:44 -0500309#endif
joshualitt4973d9d2014-11-08 09:24:25 -0800310
bungeman06ca8ec2016-06-09 08:01:03 -0700311sk_sp<GrGeometryProcessor> GrDefaultGeoProcFactory::Make(const Color& color,
312 const Coverage& coverage,
313 const LocalCoords& localCoords,
314 const SkMatrix& viewMatrix) {
joshualitte9d60952015-07-27 12:13:14 -0700315 uint32_t flags = 0;
Brian Salomon3de0aee2017-01-29 09:34:17 -0500316 if (Color::kPremulGrColorAttribute_Type == color.fType) {
317 flags |= kColorAttribute_GPFlag;
318 } else if (Color::kUnpremulSkColorAttribute_Type == color.fType) {
319 flags |= kColorAttribute_GPFlag | kColorAttributeIsSkColor_GPFlag;
320 }
Brian Osmanfa6d8652017-05-31 09:37:27 -0400321 if (color.fLinearize) {
322 // It only makes sense to linearize SkColors (which are always sRGB). GrColor values should
323 // have been linearized and gamut-converted during paint conversion
324 SkASSERT(Color::kUnpremulSkColorAttribute_Type == color.fType);
325 flags |= kLinearizeColorAttribute_GPFlag;
326 }
Brian Salomon3de0aee2017-01-29 09:34:17 -0500327 flags |= coverage.fType == Coverage::kAttribute_Type ? kCoverageAttribute_GPFlag : 0;
328 flags |= localCoords.fType == LocalCoords::kHasExplicit_Type ? kLocalCoordAttribute_GPFlag : 0;
joshualitte9d60952015-07-27 12:13:14 -0700329
330 uint8_t inCoverage = coverage.fCoverage;
joshualitt0d986d82015-07-28 10:01:18 -0700331 bool localCoordsWillBeRead = localCoords.fType != LocalCoords::kUnused_Type;
joshualitte9d60952015-07-27 12:13:14 -0700332
333 GrColor inColor = color.fColor;
bungeman06ca8ec2016-06-09 08:01:03 -0700334 return DefaultGeoProc::Make(flags,
335 inColor,
Brian Osmanfa6d8652017-05-31 09:37:27 -0400336 color.fColorSpaceXform,
bungeman06ca8ec2016-06-09 08:01:03 -0700337 viewMatrix,
338 localCoords.fMatrix ? *localCoords.fMatrix : SkMatrix::I(),
339 localCoordsWillBeRead,
bungeman06ca8ec2016-06-09 08:01:03 -0700340 inCoverage);
joshualitte9d60952015-07-27 12:13:14 -0700341}
joshualitt0d986d82015-07-28 10:01:18 -0700342
bungeman06ca8ec2016-06-09 08:01:03 -0700343sk_sp<GrGeometryProcessor> GrDefaultGeoProcFactory::MakeForDeviceSpace(
joshualitt0d986d82015-07-28 10:01:18 -0700344 const Color& color,
345 const Coverage& coverage,
346 const LocalCoords& localCoords,
347 const SkMatrix& viewMatrix) {
joshualitt0d986d82015-07-28 10:01:18 -0700348 SkMatrix invert = SkMatrix::I();
joshualittdf0c5572015-08-03 11:35:28 -0700349 if (LocalCoords::kUnused_Type != localCoords.fType) {
350 SkASSERT(LocalCoords::kUsePosition_Type == localCoords.fType);
351 if (!viewMatrix.isIdentity() && !viewMatrix.invert(&invert)) {
halcanary96fcdcc2015-08-27 07:41:13 -0700352 return nullptr;
joshualittdf0c5572015-08-03 11:35:28 -0700353 }
joshualitt0d986d82015-07-28 10:01:18 -0700354
joshualittdf0c5572015-08-03 11:35:28 -0700355 if (localCoords.hasLocalMatrix()) {
356 invert.preConcat(*localCoords.fMatrix);
357 }
joshualitt0d986d82015-07-28 10:01:18 -0700358 }
359
360 LocalCoords inverted(LocalCoords::kUsePosition_Type, &invert);
bungeman06ca8ec2016-06-09 08:01:03 -0700361 return Make(color, coverage, inverted, SkMatrix::I());
joshualitt0d986d82015-07-28 10:01:18 -0700362}