blob: 21dd5e72652f6d2522e871a89ec393886077844f [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"
wangyix6af0c932015-07-22 10:21:17 -070011#include "gl/GrGLFragmentProcessor.h"
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000012#include "gl/GrGLTexture.h"
joshualitt249af152014-09-15 11:41:13 -070013#include "gl/GrGLGeometryProcessor.h"
egdaniel2d721d32015-11-11 13:06:05 -080014#include "glsl/GrGLSLFragmentShaderBuilder.h"
15#include "glsl/GrGLSLProgramBuilder.h"
egdaniel018fb622015-10-28 07:26:40 -070016#include "glsl/GrGLSLProgramDataManager.h"
egdaniel2d721d32015-11-11 13:06:05 -080017#include "glsl/GrGLSLVertexShaderBuilder.h"
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000018
egdaniel309e3462014-12-09 10:35:58 -080019class GrGLBitmapTextGeoProc : public GrGLGeometryProcessor {
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000020public:
joshualitt465283c2015-09-11 08:19:35 -070021 GrGLBitmapTextGeoProc() : fColor(GrColor_ILLEGAL) {}
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000022
joshualitt465283c2015-09-11 08:19:35 -070023 void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
egdaniel309e3462014-12-09 10:35:58 -080024 const GrBitmapTextGeoProc& cte = args.fGP.cast<GrBitmapTextGeoProc>();
joshualitt2dd1ae02014-12-03 06:24:10 -080025
egdaniel8dcdedc2015-11-11 06:27:20 -080026 GrGLSLGPBuilder* pb = args.fPB;
egdaniel2d721d32015-11-11 13:06:05 -080027 GrGLSLVertexBuilder* vsBuilder = pb->getVertexShaderBuilder();
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000028
joshualittabb52a12015-01-13 15:02:10 -080029 // emit attributes
30 vsBuilder->emitAttributes(cte);
31
joshualitt922c8b12015-08-07 09:55:23 -070032 // compute numbers to be hardcoded to convert texture coordinates from int to float
33 SkASSERT(cte.numTextures() == 1);
34 GrTexture* atlas = cte.textureAccess(0).getTexture();
joshualitt7375d6b2015-08-07 13:36:44 -070035 SkASSERT(atlas && SkIsPow2(atlas->width()) && SkIsPow2(atlas->height()));
joshualitt922c8b12015-08-07 09:55:23 -070036 SkScalar recipWidth = 1.0f / atlas->width();
37 SkScalar recipHeight = 1.0f / atlas->height();
38
egdaniel8dcdedc2015-11-11 06:27:20 -080039 GrGLSLVertToFrag v(kVec2f_GrSLType);
joshualitt9b989322014-12-15 14:16:27 -080040 pb->addVarying("TextureCoords", &v);
joshualitt922c8b12015-08-07 09:55:23 -070041 vsBuilder->codeAppendf("%s = vec2(%.*f, %.*f) * %s;", v.vsOut(),
joshualitt7375d6b2015-08-07 13:36:44 -070042 GR_SIGNIFICANT_POW2_DECIMAL_DIG, recipWidth,
43 GR_SIGNIFICANT_POW2_DECIMAL_DIG, recipHeight,
joshualitt922c8b12015-08-07 09:55:23 -070044 cte.inTextureCoords()->fName);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000045
joshualitt9b989322014-12-15 14:16:27 -080046 // Setup pass through color
joshualittb8c241a2015-05-19 08:23:30 -070047 if (!cte.colorIgnored()) {
48 if (cte.hasVertexColor()) {
49 pb->addPassThroughAttribute(cte.inColor(), args.fOutputColor);
50 } else {
51 this->setupUniformColor(pb, args.fOutputColor, &fColorUniform);
52 }
53 }
joshualitt2dd1ae02014-12-03 06:24:10 -080054
joshualittabb52a12015-01-13 15:02:10 -080055 // Setup position
joshualitte578a952015-05-14 10:09:13 -070056 this->setupPosition(pb, gpArgs, cte.inPosition()->fName);
joshualitt4973d9d2014-11-08 09:24:25 -080057
joshualittabb52a12015-01-13 15:02:10 -080058 // emit transforms
robertphillips46d36f02015-01-18 08:14:14 -080059 this->emitTransforms(args.fPB, gpArgs->fPositionVar, cte.inPosition()->fName,
joshualittabb52a12015-01-13 15:02:10 -080060 cte.localMatrix(), args.fTransformsIn, args.fTransformsOut);
61
egdaniel2d721d32015-11-11 13:06:05 -080062 GrGLSLFragmentBuilder* fsBuilder = pb->getFragmentShaderBuilder();
joshualitt02b05012015-02-11 06:56:30 -080063 if (cte.maskFormat() == kARGB_GrMaskFormat) {
64 fsBuilder->codeAppendf("%s = ", args.fOutputColor);
65 fsBuilder->appendTextureLookupAndModulate(args.fOutputColor,
66 args.fSamplers[0],
67 v.fsIn(),
68 kVec2f_GrSLType);
69 fsBuilder->codeAppend(";");
70 fsBuilder->codeAppendf("%s = vec4(1);", args.fOutputCoverage);
71 } else {
72 fsBuilder->codeAppendf("%s = ", args.fOutputCoverage);
73 fsBuilder->appendTextureLookup(args.fSamplers[0], v.fsIn(), kVec2f_GrSLType);
74 fsBuilder->codeAppend(";");
egdaniel27b63352015-09-15 13:13:50 -070075 if (cte.maskFormat() == kA565_GrMaskFormat) {
76 // set alpha to be max of rgb coverage
77 fsBuilder->codeAppendf("%s.a = max(max(%s.r, %s.g), %s.b);",
78 args.fOutputCoverage, args.fOutputCoverage,
79 args.fOutputCoverage, args.fOutputCoverage);
80 }
joshualitt02b05012015-02-11 06:56:30 -080081 }
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +000082 }
83
egdaniel018fb622015-10-28 07:26:40 -070084 void setData(const GrGLSLProgramDataManager& pdman, const GrPrimitiveProcessor& gp) override {
joshualittb8c241a2015-05-19 08:23:30 -070085 const GrBitmapTextGeoProc& btgp = gp.cast<GrBitmapTextGeoProc>();
86 if (btgp.color() != fColor && !btgp.hasVertexColor()) {
egdaniel018fb622015-10-28 07:26:40 -070087 float c[4];
joshualittb8c241a2015-05-19 08:23:30 -070088 GrColorToRGBAFloat(btgp.color(), c);
joshualitt9b989322014-12-15 14:16:27 -080089 pdman.set4fv(fColorUniform, 1, c);
joshualittb8c241a2015-05-19 08:23:30 -070090 fColor = btgp.color();
joshualitt9b989322014-12-15 14:16:27 -080091 }
joshualitt2dd1ae02014-12-03 06:24:10 -080092 }
93
joshualitte3ababe2015-05-15 07:56:07 -070094 void setTransformData(const GrPrimitiveProcessor& primProc,
egdaniel018fb622015-10-28 07:26:40 -070095 const GrGLSLProgramDataManager& pdman,
joshualitte3ababe2015-05-15 07:56:07 -070096 int index,
97 const SkTArray<const GrCoordTransform*, true>& transforms) override {
98 this->setTransformDataHelper<GrBitmapTextGeoProc>(primProc, pdman, index, transforms);
99 }
100
joshualitt9b989322014-12-15 14:16:27 -0800101 static inline void GenKey(const GrGeometryProcessor& proc,
jvanverthcfc18862015-04-28 08:48:20 -0700102 const GrGLSLCaps&,
joshualitt9b989322014-12-15 14:16:27 -0800103 GrProcessorKeyBuilder* b) {
joshualitt9b989322014-12-15 14:16:27 -0800104 const GrBitmapTextGeoProc& gp = proc.cast<GrBitmapTextGeoProc>();
joshualitt8fc6c2d2014-12-22 15:27:05 -0800105 uint32_t key = 0;
joshualittb8c241a2015-05-19 08:23:30 -0700106 key |= gp.usesLocalCoords() && gp.localMatrix().hasPerspective() ? 0x1 : 0x0;
107 key |= gp.colorIgnored() ? 0x2 : 0x0;
108 key |= gp.maskFormat() << 3;
109 b->add32(key);
joshualitt922c8b12015-08-07 09:55:23 -0700110
111 // Currently we hardcode numbers to convert atlas coordinates to normalized floating point
112 SkASSERT(gp.numTextures() == 1);
113 GrTexture* atlas = gp.textureAccess(0).getTexture();
114 SkASSERT(atlas);
115 b->add32(atlas->width());
116 b->add32(atlas->height());
joshualitt9b989322014-12-15 14:16:27 -0800117 }
joshualitt2dd1ae02014-12-03 06:24:10 -0800118
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000119private:
joshualitt9b989322014-12-15 14:16:27 -0800120 GrColor fColor;
121 UniformHandle fColorUniform;
122
joshualitt249af152014-09-15 11:41:13 -0700123 typedef GrGLGeometryProcessor INHERITED;
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000124};
125
126///////////////////////////////////////////////////////////////////////////////
127
joshualitt2e3b3e32014-12-09 13:31:14 -0800128GrBitmapTextGeoProc::GrBitmapTextGeoProc(GrColor color, GrTexture* texture,
joshualitt02b05012015-02-11 06:56:30 -0800129 const GrTextureParams& params, GrMaskFormat format,
joshualittb8c241a2015-05-19 08:23:30 -0700130 const SkMatrix& localMatrix, bool usesLocalCoords)
joshualitte3ababe2015-05-15 07:56:07 -0700131 : fColor(color)
132 , fLocalMatrix(localMatrix)
joshualittb8c241a2015-05-19 08:23:30 -0700133 , fUsesLocalCoords(usesLocalCoords)
joshualitt8fc6c2d2014-12-22 15:27:05 -0800134 , fTextureAccess(texture, params)
halcanary96fcdcc2015-08-27 07:41:13 -0700135 , fInColor(nullptr)
joshualitt02b05012015-02-11 06:56:30 -0800136 , fMaskFormat(format) {
egdaniel309e3462014-12-09 10:35:58 -0800137 this->initClassID<GrBitmapTextGeoProc>();
joshualitt71c92602015-01-14 08:12:47 -0800138 fInPosition = &this->addVertexAttrib(Attribute("inPosition", kVec2f_GrVertexAttribType));
joshualitt02b05012015-02-11 06:56:30 -0800139
joshualittb8c241a2015-05-19 08:23:30 -0700140 // TODO we could think about removing this attribute if color is ignored, but unfortunately
141 // we don't do text positioning in batch, so we can't quite do that yet.
joshualitt02b05012015-02-11 06:56:30 -0800142 bool hasVertexColor = kA8_GrMaskFormat == fMaskFormat;
143 if (hasVertexColor) {
joshualitt71c92602015-01-14 08:12:47 -0800144 fInColor = &this->addVertexAttrib(Attribute("inColor", kVec4ub_GrVertexAttribType));
joshualitt2dd1ae02014-12-03 06:24:10 -0800145 }
joshualitt71c92602015-01-14 08:12:47 -0800146 fInTextureCoords = &this->addVertexAttrib(Attribute("inTextureCoords",
jvanverth5a105ff2015-02-18 11:36:35 -0800147 kVec2s_GrVertexAttribType));
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000148 this->addTextureAccess(&fTextureAccess);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000149}
150
joshualitt465283c2015-09-11 08:19:35 -0700151void GrBitmapTextGeoProc::getGLProcessorKey(const GrGLSLCaps& caps,GrProcessorKeyBuilder* b) const {
152 GrGLBitmapTextGeoProc::GenKey(*this, caps, b);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000153}
154
joshualitt465283c2015-09-11 08:19:35 -0700155GrGLPrimitiveProcessor* GrBitmapTextGeoProc::createGLInstance(const GrGLSLCaps& caps) const {
156 return new GrGLBitmapTextGeoProc();
joshualitteb2a6762014-12-04 11:35:33 -0800157}
joshualitt9b989322014-12-15 14:16:27 -0800158
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000159///////////////////////////////////////////////////////////////////////////////
160
egdaniel309e3462014-12-09 10:35:58 -0800161GR_DEFINE_GEOMETRY_PROCESSOR_TEST(GrBitmapTextGeoProc);
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000162
bsalomonc21b09e2015-08-28 18:46:56 -0700163const GrGeometryProcessor* GrBitmapTextGeoProc::TestCreate(GrProcessorTestData* d) {
joshualitt0067ff52015-07-08 14:26:19 -0700164 int texIdx = d->fRandom->nextBool() ? GrProcessorUnitTest::kSkiaPMTextureIdx :
165 GrProcessorUnitTest::kAlphaTextureIdx;
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000166 static const SkShader::TileMode kTileModes[] = {
167 SkShader::kClamp_TileMode,
168 SkShader::kRepeat_TileMode,
169 SkShader::kMirror_TileMode,
170 };
171 SkShader::TileMode tileModes[] = {
joshualitt0067ff52015-07-08 14:26:19 -0700172 kTileModes[d->fRandom->nextULessThan(SK_ARRAY_COUNT(kTileModes))],
173 kTileModes[d->fRandom->nextULessThan(SK_ARRAY_COUNT(kTileModes))],
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000174 };
joshualitt0067ff52015-07-08 14:26:19 -0700175 GrTextureParams params(tileModes, d->fRandom->nextBool() ? GrTextureParams::kBilerp_FilterMode :
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000176 GrTextureParams::kNone_FilterMode);
177
joshualitt02b05012015-02-11 06:56:30 -0800178 GrMaskFormat format;
joshualitt0067ff52015-07-08 14:26:19 -0700179 switch (d->fRandom->nextULessThan(3)) {
joshualitt02b05012015-02-11 06:56:30 -0800180 case 0:
181 format = kA8_GrMaskFormat;
182 break;
183 case 1:
184 format = kA565_GrMaskFormat;
185 break;
186 case 2:
187 format = kARGB_GrMaskFormat;
188 break;
189 }
190
joshualitt0067ff52015-07-08 14:26:19 -0700191 return GrBitmapTextGeoProc::Create(GrRandomColor(d->fRandom), d->fTextures[texIdx], params,
192 format, GrTest::TestMatrix(d->fRandom),
193 d->fRandom->nextBool());
commit-bot@chromium.org76eaf742013-09-30 18:41:38 +0000194}