blob: 58ae2a90ec199ed3a3f77ed97bf38c00e5134eb1 [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"
egdaniel2d721d32015-11-11 13:06:05 -080014#include "glsl/GrGLSLVertexShaderBuilder.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;
cdalton85285412016-02-18 12:37:07 -080076 GrGLSLPPFragmentBuilder* 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()) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -040085 GrGLSLVertToFrag 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:
Ethan Nicholasf7b88202017-09-18 14:10:39 -040089 vertBuilder->codeAppendf("half4 color = %s;", gp.inColor()->fName);
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 Osmanfa6d8652017-05-31 09:37:27 -0400105 srgbFuncName.c_str(), gp.inColor()->fName,
106 srgbFuncName.c_str(), gp.inColor()->fName,
107 srgbFuncName.c_str(), gp.inColor()->fName,
Brian Salomon3de0aee2017-01-29 09:34:17 -0500108 gp.inColor()->fName);
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);
120 if (fColorSpaceHelper.isValid()) {
121 SkString xformedColor;
122 vertBuilder->appendColorGamutXform(&xformedColor, "color",
123 &fColorSpaceHelper);
124 vertBuilder->codeAppendf("color = %s;", xformedColor.c_str());
125 }
126 }
127 vertBuilder->codeAppendf("%s = color;\n", varying.vsOut());
Brian Salomon3de0aee2017-01-29 09:34:17 -0500128 fragBuilder->codeAppendf("%s = %s;", args.fOutputColor, varying.fsIn());
Brian Salomonbfd51832017-01-04 13:22:08 -0500129 } else {
130 this->setupUniformColor(fragBuilder, uniformHandler, args.fOutputColor,
131 &fColorUniform);
joshualittb8c241a2015-05-19 08:23:30 -0700132 }
133
joshualittabb52a12015-01-13 15:02:10 -0800134 // Setup position
Brian Salomon7f235432017-08-16 09:41:48 -0400135 this->writeOutputPosition(vertBuilder,
136 uniformHandler,
137 gpArgs,
138 gp.inPosition()->fName,
139 gp.viewMatrix(),
140 &fViewMatrixUniform);
joshualitt4973d9d2014-11-08 09:24:25 -0800141
joshualittb2aa7cb2015-08-05 11:05:22 -0700142 if (gp.hasExplicitLocalCoords()) {
joshualittabb52a12015-01-13 15:02:10 -0800143 // emit transforms with explicit local coords
egdaniel7ea439b2015-12-03 09:20:44 -0800144 this->emitTransforms(vertBuilder,
egdaniel0eafe792015-11-20 14:01:22 -0800145 varyingHandler,
egdaniel7ea439b2015-12-03 09:20:44 -0800146 uniformHandler,
egdaniel4ca2e602015-11-18 08:01:26 -0800147 gpArgs->fPositionVar,
148 gp.inLocalCoords()->fName,
149 gp.localMatrix(),
bsalomona624bf32016-09-20 09:12:47 -0700150 args.fFPCoordTransformHandler);
joshualittabb52a12015-01-13 15:02:10 -0800151 } else {
152 // emit transforms with position
egdaniel7ea439b2015-12-03 09:20:44 -0800153 this->emitTransforms(vertBuilder,
egdaniel0eafe792015-11-20 14:01:22 -0800154 varyingHandler,
egdaniel7ea439b2015-12-03 09:20:44 -0800155 uniformHandler,
egdaniel4ca2e602015-11-18 08:01:26 -0800156 gpArgs->fPositionVar,
157 gp.inPosition()->fName,
158 gp.localMatrix(),
bsalomona624bf32016-09-20 09:12:47 -0700159 args.fFPCoordTransformHandler);
joshualittabb52a12015-01-13 15:02:10 -0800160 }
161
joshualitt2dd1ae02014-12-03 06:24:10 -0800162 // Setup coverage as pass through
Brian Salomon8c852be2017-01-04 10:44:42 -0500163 if (gp.hasVertexCoverage()) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400164 fragBuilder->codeAppendf("half alpha = 1.0;");
Brian Salomon8c852be2017-01-04 10:44:42 -0500165 varyingHandler->addPassThroughAttribute(gp.inCoverage(), "alpha");
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400166 fragBuilder->codeAppendf("%s = half4(alpha);", args.fOutputCoverage);
Brian Salomon8c852be2017-01-04 10:44:42 -0500167 } else if (gp.coverage() == 0xff) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400168 fragBuilder->codeAppendf("%s = half4(1);", args.fOutputCoverage);
Brian Salomon8c852be2017-01-04 10:44:42 -0500169 } else {
170 const char* fragCoverage;
171 fCoverageUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400172 kHalf_GrSLType,
Brian Salomon8c852be2017-01-04 10:44:42 -0500173 "Coverage",
174 &fragCoverage);
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400175 fragBuilder->codeAppendf("%s = half4(%s);", args.fOutputCoverage, fragCoverage);
joshualitt2dd1ae02014-12-03 06:24:10 -0800176 }
joshualitt4973d9d2014-11-08 09:24:25 -0800177 }
178
joshualitt87f48d92014-12-04 10:41:40 -0800179 static inline void GenKey(const GrGeometryProcessor& gp,
Brian Salomon94efbf52016-11-29 13:43:05 -0500180 const GrShaderCaps&,
joshualitt87f48d92014-12-04 10:41:40 -0800181 GrProcessorKeyBuilder* b) {
joshualitt2dd1ae02014-12-03 06:24:10 -0800182 const DefaultGeoProc& def = gp.cast<DefaultGeoProc>();
joshualitt8fc6c2d2014-12-22 15:27:05 -0800183 uint32_t key = def.fFlags;
Brian Salomon3de0aee2017-01-29 09:34:17 -0500184 key |= (def.coverage() == 0xff) ? 0x10 : 0;
185 key |= (def.localCoordsWillBeRead() && def.localMatrix().hasPerspective()) ? 0x20 : 0x0;
186 key |= ComputePosKey(def.viewMatrix()) << 20;
joshualitt8fc6c2d2014-12-22 15:27:05 -0800187 b->add32(key);
Brian Osmanfa6d8652017-05-31 09:37:27 -0400188 if (def.linearizeColor()) {
189 b->add32(GrColorSpaceXform::XformKey(def.fColorSpaceXform.get()));
190 }
joshualitt2dd1ae02014-12-03 06:24:10 -0800191 }
joshualitt4973d9d2014-11-08 09:24:25 -0800192
egdaniel018fb622015-10-28 07:26:40 -0700193 void setData(const GrGLSLProgramDataManager& pdman,
bsalomona624bf32016-09-20 09:12:47 -0700194 const GrPrimitiveProcessor& gp,
195 FPCoordTransformIter&& transformIter) override {
joshualitte578a952015-05-14 10:09:13 -0700196 const DefaultGeoProc& dgp = gp.cast<DefaultGeoProc>();
joshualitt5559ca22015-05-21 15:50:36 -0700197
198 if (!dgp.viewMatrix().isIdentity() && !fViewMatrix.cheapEqualTo(dgp.viewMatrix())) {
199 fViewMatrix = dgp.viewMatrix();
egdaniel018fb622015-10-28 07:26:40 -0700200 float viewMatrix[3 * 3];
egdaniel64c47282015-11-13 06:54:19 -0800201 GrGLSLGetMatrix<3>(viewMatrix, fViewMatrix);
joshualitt5559ca22015-05-21 15:50:36 -0700202 pdman.setMatrix3f(fViewMatrixUniform, viewMatrix);
203 }
joshualittee2af952014-12-30 09:04:15 -0800204
joshualittb8c241a2015-05-19 08:23:30 -0700205 if (dgp.color() != fColor && !dgp.hasVertexColor()) {
egdaniel018fb622015-10-28 07:26:40 -0700206 float c[4];
joshualittb8c241a2015-05-19 08:23:30 -0700207 GrColorToRGBAFloat(dgp.color(), c);
joshualitt9b989322014-12-15 14:16:27 -0800208 pdman.set4fv(fColorUniform, 1, c);
joshualittb8c241a2015-05-19 08:23:30 -0700209 fColor = dgp.color();
joshualitt9b989322014-12-15 14:16:27 -0800210 }
joshualittb8c241a2015-05-19 08:23:30 -0700211
Brian Salomon8c852be2017-01-04 10:44:42 -0500212 if (dgp.coverage() != fCoverage && !dgp.hasVertexCoverage()) {
joshualittb8c241a2015-05-19 08:23:30 -0700213 pdman.set1f(fCoverageUniform, GrNormalizeByteToFloat(dgp.coverage()));
214 fCoverage = dgp.coverage();
joshualitt9b989322014-12-15 14:16:27 -0800215 }
bsalomona624bf32016-09-20 09:12:47 -0700216 this->setTransformDataHelper(dgp.fLocalMatrix, pdman, &transformIter);
Brian Osmanfa6d8652017-05-31 09:37:27 -0400217
218 if (dgp.linearizeColor() && dgp.fColorSpaceXform) {
219 fColorSpaceHelper.setData(pdman, dgp.fColorSpaceXform.get());
220 }
joshualitte3ababe2015-05-15 07:56:07 -0700221 }
222
joshualitt4973d9d2014-11-08 09:24:25 -0800223 private:
joshualitt5559ca22015-05-21 15:50:36 -0700224 SkMatrix fViewMatrix;
joshualitt9b989322014-12-15 14:16:27 -0800225 GrColor fColor;
226 uint8_t fCoverage;
joshualitt5559ca22015-05-21 15:50:36 -0700227 UniformHandle fViewMatrixUniform;
joshualitt9b989322014-12-15 14:16:27 -0800228 UniformHandle fColorUniform;
229 UniformHandle fCoverageUniform;
Brian Osmanfa6d8652017-05-31 09:37:27 -0400230 GrGLSLColorSpaceXformHelper fColorSpaceHelper;
joshualitt9b989322014-12-15 14:16:27 -0800231
egdaniele659a582015-11-13 09:55:43 -0800232 typedef GrGLSLGeometryProcessor INHERITED;
joshualitt4973d9d2014-11-08 09:24:25 -0800233 };
234
Brian Salomon94efbf52016-11-29 13:43:05 -0500235 void getGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override {
egdaniel57d3b032015-11-13 11:57:27 -0800236 GLSLProcessor::GenKey(*this, caps, b);
joshualitteb2a6762014-12-04 11:35:33 -0800237 }
238
Brian Salomon94efbf52016-11-29 13:43:05 -0500239 GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override {
egdaniel57d3b032015-11-13 11:57:27 -0800240 return new GLSLProcessor();
joshualitteb2a6762014-12-04 11:35:33 -0800241 }
242
joshualitt4973d9d2014-11-08 09:24:25 -0800243private:
joshualitt8059eb92014-12-29 15:10:07 -0800244 DefaultGeoProc(uint32_t gpTypeFlags,
245 GrColor color,
Brian Osmanfa6d8652017-05-31 09:37:27 -0400246 sk_sp<GrColorSpaceXform> colorSpaceXform,
joshualitt8059eb92014-12-29 15:10:07 -0800247 const SkMatrix& viewMatrix,
248 const SkMatrix& localMatrix,
joshualittb8c241a2015-05-19 08:23:30 -0700249 uint8_t coverage,
Brian Salomon8c852be2017-01-04 10:44:42 -0500250 bool localCoordsWillBeRead)
Brian Salomon3de0aee2017-01-29 09:34:17 -0500251 : fColor(color)
Brian Salomon8c852be2017-01-04 10:44:42 -0500252 , fViewMatrix(viewMatrix)
253 , fLocalMatrix(localMatrix)
254 , fCoverage(coverage)
255 , fFlags(gpTypeFlags)
Brian Osmanfa6d8652017-05-31 09:37:27 -0400256 , fLocalCoordsWillBeRead(localCoordsWillBeRead)
257 , fColorSpaceXform(std::move(colorSpaceXform)) {
joshualitteb2a6762014-12-04 11:35:33 -0800258 this->initClassID<DefaultGeoProc>();
bsalomon6cb807b2016-08-17 11:33:39 -0700259 fInPosition = &this->addVertexAttrib("inPosition", kVec2f_GrVertexAttribType,
260 kHigh_GrSLPrecision);
Brian Salomon3de0aee2017-01-29 09:34:17 -0500261 if (fFlags & kColorAttribute_GPFlag) {
bsalomon6cb807b2016-08-17 11:33:39 -0700262 fInColor = &this->addVertexAttrib("inColor", kVec4ub_GrVertexAttribType);
joshualitt2dd1ae02014-12-03 06:24:10 -0800263 }
Brian Salomon3de0aee2017-01-29 09:34:17 -0500264 if (fFlags & kLocalCoordAttribute_GPFlag) {
Brian Salomon9b345e32016-10-05 12:09:26 -0400265 fInLocalCoords = &this->addVertexAttrib("inLocalCoord", kVec2f_GrVertexAttribType,
266 kHigh_GrSLPrecision);
bsalomoncc6aeee2016-09-12 12:02:11 -0700267 this->setHasExplicitLocalCoords();
joshualitt2dd1ae02014-12-03 06:24:10 -0800268 }
Brian Salomon3de0aee2017-01-29 09:34:17 -0500269 if (fFlags & kCoverageAttribute_GPFlag) {
bsalomon6cb807b2016-08-17 11:33:39 -0700270 fInCoverage = &this->addVertexAttrib("inCoverage", kFloat_GrVertexAttribType);
joshualitt2dd1ae02014-12-03 06:24:10 -0800271 }
272 }
joshualitt4973d9d2014-11-08 09:24:25 -0800273
Brian Salomon3de0aee2017-01-29 09:34:17 -0500274 const Attribute* fInPosition = nullptr;
275 const Attribute* fInColor = nullptr;
276 const Attribute* fInLocalCoords = nullptr;
277 const Attribute* fInCoverage = nullptr;
joshualitt88c23fc2015-05-13 14:18:07 -0700278 GrColor fColor;
joshualitte578a952015-05-14 10:09:13 -0700279 SkMatrix fViewMatrix;
joshualitte3ababe2015-05-15 07:56:07 -0700280 SkMatrix fLocalMatrix;
joshualitt9b989322014-12-15 14:16:27 -0800281 uint8_t fCoverage;
joshualitt2dd1ae02014-12-03 06:24:10 -0800282 uint32_t fFlags;
bsalomon7765a472015-07-08 11:26:37 -0700283 bool fLocalCoordsWillBeRead;
Brian Osmanfa6d8652017-05-31 09:37:27 -0400284 sk_sp<GrColorSpaceXform> fColorSpaceXform;
joshualitt4973d9d2014-11-08 09:24:25 -0800285
Brian Salomon0c26a9d2017-07-06 10:09:38 -0400286 GR_DECLARE_GEOMETRY_PROCESSOR_TEST
joshualitt5478d422014-11-14 16:00:38 -0800287
joshualitt2e3b3e32014-12-09 13:31:14 -0800288 typedef GrGeometryProcessor INHERITED;
joshualitt4973d9d2014-11-08 09:24:25 -0800289};
290
291GR_DEFINE_GEOMETRY_PROCESSOR_TEST(DefaultGeoProc);
292
Hal Canary6f6961e2017-01-31 13:50:44 -0500293#if GR_TEST_UTILS
bungeman06ca8ec2016-06-09 08:01:03 -0700294sk_sp<GrGeometryProcessor> DefaultGeoProc::TestCreate(GrProcessorTestData* d) {
joshualitt2dd1ae02014-12-03 06:24:10 -0800295 uint32_t flags = 0;
joshualitt0067ff52015-07-08 14:26:19 -0700296 if (d->fRandom->nextBool()) {
Brian Salomon3de0aee2017-01-29 09:34:17 -0500297 flags |= kColorAttribute_GPFlag;
joshualitt4973d9d2014-11-08 09:24:25 -0800298 }
joshualitt0067ff52015-07-08 14:26:19 -0700299 if (d->fRandom->nextBool()) {
Brian Salomon3de0aee2017-01-29 09:34:17 -0500300 flags |= kColorAttributeIsSkColor_GPFlag;
joshualitt4973d9d2014-11-08 09:24:25 -0800301 }
joshualitt0067ff52015-07-08 14:26:19 -0700302 if (d->fRandom->nextBool()) {
Brian Salomon3de0aee2017-01-29 09:34:17 -0500303 flags |= kCoverageAttribute_GPFlag;
304 }
305 if (d->fRandom->nextBool()) {
306 flags |= kLocalCoordAttribute_GPFlag;
joshualittb2aa7cb2015-08-05 11:05:22 -0700307 }
joshualitt2dd1ae02014-12-03 06:24:10 -0800308
bungeman06ca8ec2016-06-09 08:01:03 -0700309 return DefaultGeoProc::Make(flags,
310 GrRandomColor(d->fRandom),
Brian Osmanfa6d8652017-05-31 09:37:27 -0400311 GrTest::TestColorXform(d->fRandom),
bungeman06ca8ec2016-06-09 08:01:03 -0700312 GrTest::TestMatrix(d->fRandom),
313 GrTest::TestMatrix(d->fRandom),
314 d->fRandom->nextBool(),
bungeman06ca8ec2016-06-09 08:01:03 -0700315 GrRandomCoverage(d->fRandom));
joshualitt4973d9d2014-11-08 09:24:25 -0800316}
Hal Canary6f6961e2017-01-31 13:50:44 -0500317#endif
joshualitt4973d9d2014-11-08 09:24:25 -0800318
bungeman06ca8ec2016-06-09 08:01:03 -0700319sk_sp<GrGeometryProcessor> GrDefaultGeoProcFactory::Make(const Color& color,
320 const Coverage& coverage,
321 const LocalCoords& localCoords,
322 const SkMatrix& viewMatrix) {
joshualitte9d60952015-07-27 12:13:14 -0700323 uint32_t flags = 0;
Brian Salomon3de0aee2017-01-29 09:34:17 -0500324 if (Color::kPremulGrColorAttribute_Type == color.fType) {
325 flags |= kColorAttribute_GPFlag;
326 } else if (Color::kUnpremulSkColorAttribute_Type == color.fType) {
327 flags |= kColorAttribute_GPFlag | kColorAttributeIsSkColor_GPFlag;
328 }
Brian Osmanfa6d8652017-05-31 09:37:27 -0400329 if (color.fLinearize) {
330 // It only makes sense to linearize SkColors (which are always sRGB). GrColor values should
331 // have been linearized and gamut-converted during paint conversion
332 SkASSERT(Color::kUnpremulSkColorAttribute_Type == color.fType);
333 flags |= kLinearizeColorAttribute_GPFlag;
334 }
Brian Salomon3de0aee2017-01-29 09:34:17 -0500335 flags |= coverage.fType == Coverage::kAttribute_Type ? kCoverageAttribute_GPFlag : 0;
336 flags |= localCoords.fType == LocalCoords::kHasExplicit_Type ? kLocalCoordAttribute_GPFlag : 0;
joshualitte9d60952015-07-27 12:13:14 -0700337
338 uint8_t inCoverage = coverage.fCoverage;
joshualitt0d986d82015-07-28 10:01:18 -0700339 bool localCoordsWillBeRead = localCoords.fType != LocalCoords::kUnused_Type;
joshualitte9d60952015-07-27 12:13:14 -0700340
341 GrColor inColor = color.fColor;
bungeman06ca8ec2016-06-09 08:01:03 -0700342 return DefaultGeoProc::Make(flags,
343 inColor,
Brian Osmanfa6d8652017-05-31 09:37:27 -0400344 color.fColorSpaceXform,
bungeman06ca8ec2016-06-09 08:01:03 -0700345 viewMatrix,
346 localCoords.fMatrix ? *localCoords.fMatrix : SkMatrix::I(),
347 localCoordsWillBeRead,
bungeman06ca8ec2016-06-09 08:01:03 -0700348 inCoverage);
joshualitte9d60952015-07-27 12:13:14 -0700349}
joshualitt0d986d82015-07-28 10:01:18 -0700350
bungeman06ca8ec2016-06-09 08:01:03 -0700351sk_sp<GrGeometryProcessor> GrDefaultGeoProcFactory::MakeForDeviceSpace(
joshualitt0d986d82015-07-28 10:01:18 -0700352 const Color& color,
353 const Coverage& coverage,
354 const LocalCoords& localCoords,
355 const SkMatrix& viewMatrix) {
joshualitt0d986d82015-07-28 10:01:18 -0700356 SkMatrix invert = SkMatrix::I();
joshualittdf0c5572015-08-03 11:35:28 -0700357 if (LocalCoords::kUnused_Type != localCoords.fType) {
358 SkASSERT(LocalCoords::kUsePosition_Type == localCoords.fType);
359 if (!viewMatrix.isIdentity() && !viewMatrix.invert(&invert)) {
halcanary96fcdcc2015-08-27 07:41:13 -0700360 return nullptr;
joshualittdf0c5572015-08-03 11:35:28 -0700361 }
joshualitt0d986d82015-07-28 10:01:18 -0700362
joshualittdf0c5572015-08-03 11:35:28 -0700363 if (localCoords.hasLocalMatrix()) {
364 invert.preConcat(*localCoords.fMatrix);
365 }
joshualitt0d986d82015-07-28 10:01:18 -0700366 }
367
368 LocalCoords inverted(LocalCoords::kUsePosition_Type, &invert);
bungeman06ca8ec2016-06-09 08:01:03 -0700369 return Make(color, coverage, inverted, SkMatrix::I());
joshualitt0d986d82015-07-28 10:01:18 -0700370}