blob: a699b0410ea0031197513f67c168f9fc53040e40 [file] [log] [blame]
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +00001/*
2 * Copyright 2013 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
egdaniel309e3462014-12-09 10:35:58 -08008#include "GrBitmapTextGeoProc.h"
egdaniel605dd0f2014-11-12 08:35:25 -08009#include "GrInvariantOutput.h"
joshualitteb2a6762014-12-04 11:35:33 -080010#include "GrTexture.h"
joshualittb0a8a372014-09-23 09:50:21 -070011#include "gl/GrGLProcessor.h"
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000012#include "gl/GrGLSL.h"
13#include "gl/GrGLTexture.h"
joshualitt249af152014-09-15 11:41:13 -070014#include "gl/GrGLGeometryProcessor.h"
joshualitteb2a6762014-12-04 11:35:33 -080015#include "gl/builders/GrGLProgramBuilder.h"
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000016
joshualitt9b989322014-12-15 14:16:27 -080017struct BitmapTextBatchTracker {
18 GrGPInput fInputColorType;
19 GrColor fColor;
joshualitt290c09b2014-12-19 13:45:20 -080020 bool fUsesLocalCoords;
joshualitt9b989322014-12-15 14:16:27 -080021};
22
egdaniel309e3462014-12-09 10:35:58 -080023class GrGLBitmapTextGeoProc : public GrGLGeometryProcessor {
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000024public:
joshualitt9b989322014-12-15 14:16:27 -080025 GrGLBitmapTextGeoProc(const GrGeometryProcessor&, const GrBatchTracker&)
26 : fColor(GrColor_ILLEGAL) {}
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000027
robertphillips46d36f02015-01-18 08:14:14 -080028 void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) SK_OVERRIDE{
egdaniel309e3462014-12-09 10:35:58 -080029 const GrBitmapTextGeoProc& cte = args.fGP.cast<GrBitmapTextGeoProc>();
joshualitt9b989322014-12-15 14:16:27 -080030 const BitmapTextBatchTracker& local = args.fBT.cast<BitmapTextBatchTracker>();
joshualitt2dd1ae02014-12-03 06:24:10 -080031
joshualitt9b989322014-12-15 14:16:27 -080032 GrGLGPBuilder* pb = args.fPB;
33 GrGLVertexBuilder* vsBuilder = pb->getVertexShaderBuilder();
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000034
joshualittabb52a12015-01-13 15:02:10 -080035 // emit attributes
36 vsBuilder->emitAttributes(cte);
37
joshualitt74077b92014-10-24 11:26:03 -070038 GrGLVertToFrag v(kVec2f_GrSLType);
joshualitt9b989322014-12-15 14:16:27 -080039 pb->addVarying("TextureCoords", &v);
joshualitt2dd1ae02014-12-03 06:24:10 -080040 vsBuilder->codeAppendf("%s = %s;", v.vsOut(), cte.inTextureCoords()->fName);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000041
joshualitt9b989322014-12-15 14:16:27 -080042 // Setup pass through color
43 this->setupColorPassThrough(pb, local.fInputColorType, args.fOutputColor, cte.inColor(),
44 &fColorUniform);
joshualitt2dd1ae02014-12-03 06:24:10 -080045
joshualittee2af952014-12-30 09:04:15 -080046 // setup uniform viewMatrix
47 this->addUniformViewMatrix(pb);
48
joshualittabb52a12015-01-13 15:02:10 -080049 // Setup position
joshualitt02b05012015-02-11 06:56:30 -080050 SetupPosition(vsBuilder, gpArgs, cte.inPosition()->fName, cte.viewMatrix(), this->uViewM());
joshualitt4973d9d2014-11-08 09:24:25 -080051
joshualittabb52a12015-01-13 15:02:10 -080052 // emit transforms
robertphillips46d36f02015-01-18 08:14:14 -080053 this->emitTransforms(args.fPB, gpArgs->fPositionVar, cte.inPosition()->fName,
joshualittabb52a12015-01-13 15:02:10 -080054 cte.localMatrix(), args.fTransformsIn, args.fTransformsOut);
55
joshualitt9b989322014-12-15 14:16:27 -080056 GrGLGPFragmentBuilder* fsBuilder = pb->getFragmentShaderBuilder();
joshualitt02b05012015-02-11 06:56:30 -080057 if (cte.maskFormat() == kARGB_GrMaskFormat) {
58 fsBuilder->codeAppendf("%s = ", args.fOutputColor);
59 fsBuilder->appendTextureLookupAndModulate(args.fOutputColor,
60 args.fSamplers[0],
61 v.fsIn(),
62 kVec2f_GrSLType);
63 fsBuilder->codeAppend(";");
64 fsBuilder->codeAppendf("%s = vec4(1);", args.fOutputCoverage);
65 } else {
66 fsBuilder->codeAppendf("%s = ", args.fOutputCoverage);
67 fsBuilder->appendTextureLookup(args.fSamplers[0], v.fsIn(), kVec2f_GrSLType);
68 fsBuilder->codeAppend(";");
69 }
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000070 }
71
joshualitt9b989322014-12-15 14:16:27 -080072 virtual void setData(const GrGLProgramDataManager& pdman,
73 const GrPrimitiveProcessor& gp,
74 const GrBatchTracker& bt) SK_OVERRIDE {
joshualittee2af952014-12-30 09:04:15 -080075 this->setUniformViewMatrix(pdman, gp.viewMatrix());
76
joshualitt9b989322014-12-15 14:16:27 -080077 const BitmapTextBatchTracker& local = bt.cast<BitmapTextBatchTracker>();
78 if (kUniform_GrGPInput == local.fInputColorType && local.fColor != fColor) {
79 GrGLfloat c[4];
80 GrColorToRGBAFloat(local.fColor, c);
81 pdman.set4fv(fColorUniform, 1, c);
82 fColor = local.fColor;
83 }
joshualitt2dd1ae02014-12-03 06:24:10 -080084 }
85
joshualitt9b989322014-12-15 14:16:27 -080086 static inline void GenKey(const GrGeometryProcessor& proc,
87 const GrBatchTracker& bt,
88 const GrGLCaps&,
89 GrProcessorKeyBuilder* b) {
90 const BitmapTextBatchTracker& local = bt.cast<BitmapTextBatchTracker>();
91 // We have to put the optional vertex attribute as part of the key. See the comment
92 // on addVertexAttrib.
93 // TODO When we have deferred geometry we can fix this
94 const GrBitmapTextGeoProc& gp = proc.cast<GrBitmapTextGeoProc>();
joshualitt8fc6c2d2014-12-22 15:27:05 -080095 uint32_t key = 0;
96 key |= SkToBool(gp.inColor()) ? 0x1 : 0x0;
97 key |= local.fUsesLocalCoords && proc.localMatrix().hasPerspective() ? 0x2 : 0x0;
joshualitt02b05012015-02-11 06:56:30 -080098 key |= gp.maskFormat() == kARGB_GrMaskFormat ? 0x4 : 0x0;
99 key |= ComputePosKey(gp.viewMatrix()) << 3;
joshualitt8fc6c2d2014-12-22 15:27:05 -0800100 b->add32(local.fInputColorType << 16 | key);
joshualitt9b989322014-12-15 14:16:27 -0800101 }
joshualitt2dd1ae02014-12-03 06:24:10 -0800102
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000103private:
joshualitt9b989322014-12-15 14:16:27 -0800104 GrColor fColor;
105 UniformHandle fColorUniform;
106
joshualitt249af152014-09-15 11:41:13 -0700107 typedef GrGLGeometryProcessor INHERITED;
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000108};
109
110///////////////////////////////////////////////////////////////////////////////
111
joshualitt2e3b3e32014-12-09 13:31:14 -0800112GrBitmapTextGeoProc::GrBitmapTextGeoProc(GrColor color, GrTexture* texture,
joshualitt02b05012015-02-11 06:56:30 -0800113 const GrTextureParams& params, GrMaskFormat format,
joshualitt8fc6c2d2014-12-22 15:27:05 -0800114 bool opaqueVertexColors, const SkMatrix& localMatrix)
joshualitt8059eb92014-12-29 15:10:07 -0800115 : INHERITED(color, SkMatrix::I(), localMatrix, opaqueVertexColors)
joshualitt8fc6c2d2014-12-22 15:27:05 -0800116 , fTextureAccess(texture, params)
joshualitt02b05012015-02-11 06:56:30 -0800117 , fInColor(NULL)
118 , fMaskFormat(format) {
egdaniel309e3462014-12-09 10:35:58 -0800119 this->initClassID<GrBitmapTextGeoProc>();
joshualitt71c92602015-01-14 08:12:47 -0800120 fInPosition = &this->addVertexAttrib(Attribute("inPosition", kVec2f_GrVertexAttribType));
joshualitt02b05012015-02-11 06:56:30 -0800121
122 bool hasVertexColor = kA8_GrMaskFormat == fMaskFormat;
123 if (hasVertexColor) {
joshualitt71c92602015-01-14 08:12:47 -0800124 fInColor = &this->addVertexAttrib(Attribute("inColor", kVec4ub_GrVertexAttribType));
joshualitt2dd1ae02014-12-03 06:24:10 -0800125 this->setHasVertexColor();
126 }
joshualitt71c92602015-01-14 08:12:47 -0800127 fInTextureCoords = &this->addVertexAttrib(Attribute("inTextureCoords",
joshualitt02b05012015-02-11 06:56:30 -0800128 kVec2f_GrVertexAttribType));
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000129 this->addTextureAccess(&fTextureAccess);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000130}
131
egdaniel309e3462014-12-09 10:35:58 -0800132bool GrBitmapTextGeoProc::onIsEqual(const GrGeometryProcessor& other) const {
133 const GrBitmapTextGeoProc& gp = other.cast<GrBitmapTextGeoProc>();
joshualitt2dd1ae02014-12-03 06:24:10 -0800134 return SkToBool(this->inColor()) == SkToBool(gp.inColor());
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000135}
136
joshualitt02b05012015-02-11 06:56:30 -0800137void GrBitmapTextGeoProc::onGetInvariantOutputColor(GrInitInvariantOutput* out) const {
138 if (kARGB_GrMaskFormat == fMaskFormat) {
joshualitt56995b52014-12-11 15:44:02 -0800139 out->setUnknownFourComponents();
joshualitt02b05012015-02-11 06:56:30 -0800140 }
141}
142
143void GrBitmapTextGeoProc::onGetInvariantOutputCoverage(GrInitInvariantOutput* out) const {
144 if (kARGB_GrMaskFormat != fMaskFormat) {
145 if (GrPixelConfigIsAlphaOnly(this->texture(0)->config())) {
146 out->setUnknownSingleComponent();
147 } else if (GrPixelConfigIsOpaque(this->texture(0)->config())) {
148 out->setUnknownOpaqueFourComponents();
149 out->setUsingLCDCoverage();
150 } else {
151 out->setUnknownFourComponents();
152 out->setUsingLCDCoverage();
153 }
154 } else {
155 out->setKnownSingleComponent(0xff);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000156 }
157}
158
egdaniel309e3462014-12-09 10:35:58 -0800159void GrBitmapTextGeoProc::getGLProcessorKey(const GrBatchTracker& bt,
160 const GrGLCaps& caps,
161 GrProcessorKeyBuilder* b) const {
162 GrGLBitmapTextGeoProc::GenKey(*this, bt, caps, b);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000163}
164
joshualittabb52a12015-01-13 15:02:10 -0800165GrGLPrimitiveProcessor*
166GrBitmapTextGeoProc::createGLInstance(const GrBatchTracker& bt,
167 const GrGLCaps& caps) const {
egdaniel309e3462014-12-09 10:35:58 -0800168 return SkNEW_ARGS(GrGLBitmapTextGeoProc, (*this, bt));
joshualitteb2a6762014-12-04 11:35:33 -0800169}
joshualitt9b989322014-12-15 14:16:27 -0800170
joshualitt4d8da812015-01-28 12:53:54 -0800171void GrBitmapTextGeoProc::initBatchTracker(GrBatchTracker* bt, const GrPipelineInfo& init) const {
joshualitt9b989322014-12-15 14:16:27 -0800172 BitmapTextBatchTracker* local = bt->cast<BitmapTextBatchTracker>();
173 local->fInputColorType = GetColorInputType(&local->fColor, this->color(), init,
174 SkToBool(fInColor));
joshualitt290c09b2014-12-19 13:45:20 -0800175 local->fUsesLocalCoords = init.fUsesLocalCoords;
joshualitt9b989322014-12-15 14:16:27 -0800176}
177
joshualitt290c09b2014-12-19 13:45:20 -0800178bool GrBitmapTextGeoProc::onCanMakeEqual(const GrBatchTracker& m,
179 const GrGeometryProcessor& that,
180 const GrBatchTracker& t) const {
joshualitt9b989322014-12-15 14:16:27 -0800181 const BitmapTextBatchTracker& mine = m.cast<BitmapTextBatchTracker>();
182 const BitmapTextBatchTracker& theirs = t.cast<BitmapTextBatchTracker>();
joshualitt290c09b2014-12-19 13:45:20 -0800183 return CanCombineLocalMatrices(*this, mine.fUsesLocalCoords,
184 that, theirs.fUsesLocalCoords) &&
185 CanCombineOutput(mine.fInputColorType, mine.fColor,
joshualitt9b989322014-12-15 14:16:27 -0800186 theirs.fInputColorType, theirs.fColor);
187}
188
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000189///////////////////////////////////////////////////////////////////////////////
190
egdaniel309e3462014-12-09 10:35:58 -0800191GR_DEFINE_GEOMETRY_PROCESSOR_TEST(GrBitmapTextGeoProc);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000192
egdaniel309e3462014-12-09 10:35:58 -0800193GrGeometryProcessor* GrBitmapTextGeoProc::TestCreate(SkRandom* random,
194 GrContext*,
195 const GrDrawTargetCaps&,
196 GrTexture* textures[]) {
joshualittb0a8a372014-09-23 09:50:21 -0700197 int texIdx = random->nextBool() ? GrProcessorUnitTest::kSkiaPMTextureIdx :
198 GrProcessorUnitTest::kAlphaTextureIdx;
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000199 static const SkShader::TileMode kTileModes[] = {
200 SkShader::kClamp_TileMode,
201 SkShader::kRepeat_TileMode,
202 SkShader::kMirror_TileMode,
203 };
204 SkShader::TileMode tileModes[] = {
205 kTileModes[random->nextULessThan(SK_ARRAY_COUNT(kTileModes))],
206 kTileModes[random->nextULessThan(SK_ARRAY_COUNT(kTileModes))],
207 };
208 GrTextureParams params(tileModes, random->nextBool() ? GrTextureParams::kBilerp_FilterMode :
209 GrTextureParams::kNone_FilterMode);
210
joshualitt02b05012015-02-11 06:56:30 -0800211 GrMaskFormat format;
212 switch (random->nextULessThan(3)) {
213 default:
214 SkFAIL("Incomplete enum\n");
215 case 0:
216 format = kA8_GrMaskFormat;
217 break;
218 case 1:
219 format = kA565_GrMaskFormat;
220 break;
221 case 2:
222 format = kARGB_GrMaskFormat;
223 break;
224 }
225
joshualitt2e3b3e32014-12-09 13:31:14 -0800226 return GrBitmapTextGeoProc::Create(GrRandomColor(random), textures[texIdx], params,
joshualitt02b05012015-02-11 06:56:30 -0800227 format, random->nextBool(),
joshualitt8fc6c2d2014-12-22 15:27:05 -0800228 GrProcessorUnitTest::TestMatrix(random));
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000229}