blob: bfa5f3cbc155e7f18581e361933653d40bdaf357 [file] [log] [blame]
junov@google.comf93e7172011-03-31 21:26:24 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2011 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.
junov@google.comf93e7172011-03-31 21:26:24 +00006 */
joshualitt23e280d2014-09-18 12:26:38 -07007
junov@google.comf93e7172011-03-31 21:26:24 +00008#include "GrGLProgram.h"
9
tomhudson@google.comd8f856c2012-05-10 12:13:36 +000010#include "GrAllocator.h"
joshualittb0a8a372014-09-23 09:50:21 -070011#include "GrProcessor.h"
bsalomon@google.com77af6802013-10-02 13:04:56 +000012#include "GrCoordTransform.h"
joshualitta5305a12014-10-10 17:47:00 -070013#include "GrGLGeometryProcessor.h"
joshualittb0a8a372014-09-23 09:50:21 -070014#include "GrGLProcessor.h"
bsalomon@google.com34cccde2013-01-04 18:34:30 +000015#include "GrGpuGL.h"
kkinnunenec56e452014-08-25 22:21:16 -070016#include "GrGLPathRendering.h"
bsalomon@google.com4fa66942011-09-20 19:06:12 +000017#include "GrGLShaderVar.h"
bsalomon@google.com018f1792013-04-18 19:36:09 +000018#include "GrGLSL.h"
egdaniel170f90b2014-09-16 12:54:40 -070019#include "GrOptDrawState.h"
Scroggo97c88c22011-05-11 14:05:25 +000020#include "SkXfermode.h"
21
commit-bot@chromium.org9188a152013-09-05 18:28:24 +000022#define GL_CALL(X) GR_GL_CALL(fGpu->glInterface(), X)
23#define GL_CALL_RET(R, X) GR_GL_CALL_RET(fGpu->glInterface(), R, X)
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000024
joshualitt47bb3822014-10-07 16:43:25 -070025/**
26 * Retrieves the final matrix that a transform needs to apply to its source coords.
27 */
joshualitta5305a12014-10-10 17:47:00 -070028static SkMatrix get_transform_matrix(const GrFragmentStage& processorStage,
joshualitt47bb3822014-10-07 16:43:25 -070029 bool useExplicitLocalCoords,
30 int transformIdx) {
31 const GrCoordTransform& coordTransform =
32 processorStage.getProcessor()->coordTransform(transformIdx);
33 SkMatrix combined;
34
35 if (kLocal_GrCoordSet == coordTransform.sourceCoords()) {
36 // If we have explicit local coords then we shouldn't need a coord change.
37 const SkMatrix& ccm =
38 useExplicitLocalCoords ? SkMatrix::I() : processorStage.getCoordChangeMatrix();
39 combined.setConcat(coordTransform.getMatrix(), ccm);
kkinnunenec56e452014-08-25 22:21:16 -070040 } else {
joshualitt47bb3822014-10-07 16:43:25 -070041 combined = coordTransform.getMatrix();
kkinnunendddc18a2014-08-03 23:19:46 -070042 }
joshualitt47bb3822014-10-07 16:43:25 -070043 if (coordTransform.reverseY()) {
44 // combined.postScale(1,-1);
45 // combined.postTranslate(0,1);
46 combined.set(SkMatrix::kMSkewY,
47 combined[SkMatrix::kMPersp0] - combined[SkMatrix::kMSkewY]);
48 combined.set(SkMatrix::kMScaleY,
49 combined[SkMatrix::kMPersp1] - combined[SkMatrix::kMScaleY]);
50 combined.set(SkMatrix::kMTransY,
51 combined[SkMatrix::kMPersp2] - combined[SkMatrix::kMTransY]);
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000052 }
joshualitt47bb3822014-10-07 16:43:25 -070053 return combined;
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000054}
55
joshualitt47bb3822014-10-07 16:43:25 -070056///////////////////////////////////////////////////////////////////////////////////////////////////
57
commit-bot@chromium.org9188a152013-09-05 18:28:24 +000058GrGLProgram::GrGLProgram(GrGpuGL* gpu,
bsalomon@google.com31ec7982013-03-27 18:14:57 +000059 const GrGLProgramDesc& desc,
joshualitt47bb3822014-10-07 16:43:25 -070060 const BuiltinUniformHandles& builtinUniforms,
61 GrGLuint programID,
62 const UniformInfoArray& uniforms,
joshualitta5305a12014-10-10 17:47:00 -070063 GrGLInstalledGeoProc* geometryProcessor,
64 GrGLInstalledFragProcs* fragmentProcessors)
commit-bot@chromium.orga05fa062014-05-30 18:55:03 +000065 : fColor(GrColor_ILLEGAL)
commit-bot@chromium.org6eac42e2014-05-29 21:29:51 +000066 , fCoverage(GrColor_ILLEGAL)
67 , fDstCopyTexUnit(-1)
joshualitt47bb3822014-10-07 16:43:25 -070068 , fBuiltinUniformHandles(builtinUniforms)
69 , fProgramID(programID)
joshualitta5305a12014-10-10 17:47:00 -070070 , fGeometryProcessor(geometryProcessor)
71 , fFragmentProcessors(SkRef(fragmentProcessors))
commit-bot@chromium.org6eac42e2014-05-29 21:29:51 +000072 , fDesc(desc)
73 , fGpu(gpu)
joshualitt47bb3822014-10-07 16:43:25 -070074 , fProgramDataManager(gpu, uniforms) {
commit-bot@chromium.org6eac42e2014-05-29 21:29:51 +000075 this->initSamplerUniforms();
junov@google.comf93e7172011-03-31 21:26:24 +000076}
77
78GrGLProgram::~GrGLProgram() {
kkinnunendddc18a2014-08-03 23:19:46 -070079 if (fProgramID) {
80 GL_CALL(DeleteProgram(fProgramID));
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000081 }
junov@google.comf93e7172011-03-31 21:26:24 +000082}
83
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000084void GrGLProgram::abandon() {
kkinnunendddc18a2014-08-03 23:19:46 -070085 fProgramID = 0;
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000086}
87
bsalomon@google.comdbbc4e22012-07-25 17:48:39 +000088void GrGLProgram::initSamplerUniforms() {
kkinnunendddc18a2014-08-03 23:19:46 -070089 GL_CALL(UseProgram(fProgramID));
bsalomon@google.com34cccde2013-01-04 18:34:30 +000090 GrGLint texUnitIdx = 0;
kkinnunendddc18a2014-08-03 23:19:46 -070091 if (fBuiltinUniformHandles.fDstCopySamplerUni.isValid()) {
92 fProgramDataManager.setSampler(fBuiltinUniformHandles.fDstCopySamplerUni, texUnitIdx);
bsalomon@google.com804e9942013-06-06 18:04:38 +000093 fDstCopyTexUnit = texUnitIdx++;
bsalomon@google.com26e18b52013-03-29 19:22:36 +000094 }
bsalomon49f085d2014-09-05 13:34:00 -070095 if (fGeometryProcessor.get()) {
joshualitt47bb3822014-10-07 16:43:25 -070096 this->initSamplers(fGeometryProcessor.get(), &texUnitIdx);
joshualittbd769d02014-09-04 08:56:46 -070097 }
joshualitta5305a12014-10-10 17:47:00 -070098 int numProcs = fFragmentProcessors->fProcs.count();
99 for (int i = 0; i < numProcs; i++) {
100 this->initSamplers(fFragmentProcessors->fProcs[i], &texUnitIdx);
joshualitt47bb3822014-10-07 16:43:25 -0700101 }
102}
103
joshualitta5305a12014-10-10 17:47:00 -0700104void GrGLProgram::initSamplers(GrGLInstalledProc* ip, int* texUnitIdx) {
105 SkTArray<GrGLInstalledProc::Sampler, true>& samplers = ip->fSamplers;
106 int numSamplers = samplers.count();
107 for (int s = 0; s < numSamplers; ++s) {
108 SkASSERT(samplers[s].fUniform.isValid());
109 fProgramDataManager.setSampler(samplers[s].fUniform, *texUnitIdx);
110 samplers[s].fTextureUnit = (*texUnitIdx)++;
111 }
112}
113
114void GrGLProgram::bindTextures(const GrGLInstalledProc* ip, const GrProcessor& processor) {
115 const SkTArray<GrGLInstalledProc::Sampler, true>& samplers = ip->fSamplers;
joshualitt47bb3822014-10-07 16:43:25 -0700116 int numSamplers = samplers.count();
117 SkASSERT(numSamplers == processor.numTextures());
118 for (int s = 0; s < numSamplers; ++s) {
119 SkASSERT(samplers[s].fTextureUnit >= 0);
120 const GrTextureAccess& textureAccess = processor.textureAccess(s);
121 fGpu->bindTexture(samplers[s].fTextureUnit,
122 textureAccess.getParams(),
123 static_cast<GrGLTexture*>(textureAccess.getTexture()));
124 }
125}
126
127
bsalomon@google.comeb715c82012-07-11 15:03:31 +0000128///////////////////////////////////////////////////////////////////////////////
junov@google.comf93e7172011-03-31 21:26:24 +0000129
egdaniel170f90b2014-09-16 12:54:40 -0700130void GrGLProgram::setData(const GrOptDrawState& optState,
131 GrGpu::DrawType drawType,
joshualitt0e602822014-10-28 10:27:44 -0700132 const GrDeviceCoordTexture* dstCopy) {
egdaniel170f90b2014-09-16 12:54:40 -0700133 GrColor color = optState.getColor();
134 GrColor coverage = optState.getCoverageColor();
bsalomon@google.com9ba4fa62012-07-16 17:36:28 +0000135
joshualitt0e602822014-10-28 10:27:44 -0700136 this->setColor(optState, color);
137 this->setCoverage(optState, coverage);
egdaniel170f90b2014-09-16 12:54:40 -0700138 this->setMatrixAndRenderTargetHeight(drawType, optState);
bsalomon@google.com91207482013-02-12 21:45:24 +0000139
bsalomon49f085d2014-09-05 13:34:00 -0700140 if (dstCopy) {
kkinnunendddc18a2014-08-03 23:19:46 -0700141 if (fBuiltinUniformHandles.fDstCopyTopLeftUni.isValid()) {
142 fProgramDataManager.set2f(fBuiltinUniformHandles.fDstCopyTopLeftUni,
kkinnunen7510b222014-07-30 00:04:16 -0700143 static_cast<GrGLfloat>(dstCopy->offset().fX),
144 static_cast<GrGLfloat>(dstCopy->offset().fY));
kkinnunendddc18a2014-08-03 23:19:46 -0700145 fProgramDataManager.set2f(fBuiltinUniformHandles.fDstCopyScaleUni,
kkinnunen7510b222014-07-30 00:04:16 -0700146 1.f / dstCopy->texture()->width(),
147 1.f / dstCopy->texture()->height());
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000148 GrGLTexture* texture = static_cast<GrGLTexture*>(dstCopy->texture());
149 static GrTextureParams kParams; // the default is clamp, nearest filtering.
commit-bot@chromium.org9188a152013-09-05 18:28:24 +0000150 fGpu->bindTexture(fDstCopyTexUnit, kParams, texture);
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000151 } else {
kkinnunendddc18a2014-08-03 23:19:46 -0700152 SkASSERT(!fBuiltinUniformHandles.fDstCopyScaleUni.isValid());
153 SkASSERT(!fBuiltinUniformHandles.fDstCopySamplerUni.isValid());
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000154 }
155 } else {
kkinnunendddc18a2014-08-03 23:19:46 -0700156 SkASSERT(!fBuiltinUniformHandles.fDstCopyTopLeftUni.isValid());
157 SkASSERT(!fBuiltinUniformHandles.fDstCopyScaleUni.isValid());
158 SkASSERT(!fBuiltinUniformHandles.fDstCopySamplerUni.isValid());
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000159 }
bsalomon@google.comc7818882013-03-20 19:19:53 +0000160
joshualitt47bb3822014-10-07 16:43:25 -0700161 // we set the textures, and uniforms for installed processors in a generic way, but subclasses
162 // of GLProgram determine how to set coord transforms
bsalomon49f085d2014-09-05 13:34:00 -0700163 if (fGeometryProcessor.get()) {
joshualitta5305a12014-10-10 17:47:00 -0700164 SkASSERT(optState.hasGeometryProcessor());
165 const GrGeometryProcessor& gp = *optState.getGeometryProcessor();
166 fGeometryProcessor->fGLProc->setData(fProgramDataManager, gp);
167 this->bindTextures(fGeometryProcessor, gp);
joshualittbd769d02014-09-04 08:56:46 -0700168 }
joshualitta5305a12014-10-10 17:47:00 -0700169 this->setFragmentData(optState);
commit-bot@chromium.org20807222013-11-01 11:54:54 +0000170
joshualitt47bb3822014-10-07 16:43:25 -0700171 // Some of GrGLProgram subclasses need to update state here
172 this->didSetData(drawType);
173}
174
joshualitta5305a12014-10-10 17:47:00 -0700175void GrGLProgram::setFragmentData(const GrOptDrawState& optState) {
176 int numProcessors = fFragmentProcessors->fProcs.count();
177 for (int e = 0; e < numProcessors; ++e) {
178 const GrFragmentStage& stage = optState.getFragmentStage(e);
179 const GrProcessor& processor = *stage.getProcessor();
180 fFragmentProcessors->fProcs[e]->fGLProc->setData(fProgramDataManager, processor);
181 this->setTransformData(stage, fFragmentProcessors->fProcs[e]);
182 this->bindTextures(fFragmentProcessors->fProcs[e], processor);
183 }
184}
185void GrGLProgram::setTransformData(const GrFragmentStage& processor, GrGLInstalledFragProc* ip) {
186 SkTArray<GrGLInstalledFragProc::Transform, true>& transforms = ip->fTransforms;
joshualitt47bb3822014-10-07 16:43:25 -0700187 int numTransforms = transforms.count();
188 SkASSERT(numTransforms == processor.getProcessor()->numTransforms());
189 for (int t = 0; t < numTransforms; ++t) {
190 SkASSERT(transforms[t].fHandle.isValid());
joshualitta5305a12014-10-10 17:47:00 -0700191 const SkMatrix& matrix = get_transform_matrix(processor, ip->fLocalCoordAttrib, t);
joshualitt47bb3822014-10-07 16:43:25 -0700192 if (!transforms[t].fCurrentValue.cheapEqualTo(matrix)) {
193 fProgramDataManager.setSkMatrix(transforms[t].fHandle.convertToUniformHandle(), matrix);
194 transforms[t].fCurrentValue = matrix;
195 }
commit-bot@chromium.org6b30e452013-10-04 20:02:53 +0000196 }
skia.committer@gmail.com8ae714b2013-01-05 02:02:05 +0000197}
bsalomon@google.com91207482013-02-12 21:45:24 +0000198
joshualitt47bb3822014-10-07 16:43:25 -0700199void GrGLProgram::didSetData(GrGpu::DrawType drawType) {
200 SkASSERT(!GrGpu::IsPathRenderingDrawType(drawType));
201}
202
joshualitt0e602822014-10-28 10:27:44 -0700203void GrGLProgram::setColor(const GrOptDrawState& optState, GrColor color) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000204 const GrGLProgramDesc::KeyHeader& header = fDesc.getHeader();
joshualitt0e602822014-10-28 10:27:44 -0700205 switch (header.fColorInput) {
206 case GrGLProgramDesc::kAttribute_ColorInput:
207 // Attribute case is handled in GrGpuGL::setupGeometry
208 break;
209 case GrGLProgramDesc::kUniform_ColorInput:
210 if (fColor != color && fBuiltinUniformHandles.fColorUni.isValid()) {
211 // OpenGL ES doesn't support unsigned byte varieties of glUniform
212 GrGLfloat c[4];
213 GrColorToRGBAFloat(color, c);
214 fProgramDataManager.set4fv(fBuiltinUniformHandles.fColorUni, 1, c);
215 fColor = color;
216 }
217 break;
218 case GrGLProgramDesc::kAllOnes_ColorInput:
219 // Handled by shader creation
220 break;
221 default:
222 SkFAIL("Unexpected color type.");
bsalomon@google.com91207482013-02-12 21:45:24 +0000223 }
224}
225
joshualitt0e602822014-10-28 10:27:44 -0700226void GrGLProgram::setCoverage(const GrOptDrawState& optState, GrColor coverage) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000227 const GrGLProgramDesc::KeyHeader& header = fDesc.getHeader();
joshualitt0e602822014-10-28 10:27:44 -0700228 switch (header.fCoverageInput) {
229 case GrGLProgramDesc::kAttribute_ColorInput:
230 // Attribute case is handled in GrGpuGL::setupGeometry
231 break;
232 case GrGLProgramDesc::kUniform_ColorInput:
233 if (fCoverage != coverage) {
234 // OpenGL ES doesn't support unsigned byte varieties of glUniform
235 GrGLfloat c[4];
236 GrColorToRGBAFloat(coverage, c);
237 fProgramDataManager.set4fv(fBuiltinUniformHandles.fCoverageUni, 1, c);
238 fCoverage = coverage;
239 }
240 break;
241 case GrGLProgramDesc::kAllOnes_ColorInput:
242 // Handled by shader creation
243 break;
244 default:
245 SkFAIL("Unexpected coverage type.");
bsalomon@google.com91207482013-02-12 21:45:24 +0000246 }
247}
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000248
kkinnunenec56e452014-08-25 22:21:16 -0700249void GrGLProgram::setMatrixAndRenderTargetHeight(GrGpu::DrawType drawType,
egdaniel170f90b2014-09-16 12:54:40 -0700250 const GrOptDrawState& optState) {
joshualitt47bb3822014-10-07 16:43:25 -0700251 // Load the RT height uniform if it is needed to y-flip gl_FragCoord.
252 if (fBuiltinUniformHandles.fRTHeightUni.isValid() &&
253 fMatrixState.fRenderTargetSize.fHeight != optState.getRenderTarget()->height()) {
254 fProgramDataManager.set1f(fBuiltinUniformHandles.fRTHeightUni,
255 SkIntToScalar(optState.getRenderTarget()->height()));
256 }
257
258 // call subclasses to set the actual view matrix
259 this->onSetMatrixAndRenderTargetHeight(drawType, optState);
260}
261
262void GrGLProgram::onSetMatrixAndRenderTargetHeight(GrGpu::DrawType drawType,
263 const GrOptDrawState& optState) {
egdaniel170f90b2014-09-16 12:54:40 -0700264 const GrRenderTarget* rt = optState.getRenderTarget();
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000265 SkISize size;
266 size.set(rt->width(), rt->height());
joshualitt47bb3822014-10-07 16:43:25 -0700267 if (fMatrixState.fRenderTargetOrigin != rt->origin() ||
268 fMatrixState.fRenderTargetSize != size ||
269 !fMatrixState.fViewMatrix.cheapEqualTo(optState.getViewMatrix())) {
kkinnunendddc18a2014-08-03 23:19:46 -0700270 SkASSERT(fBuiltinUniformHandles.fViewMatrixUni.isValid());
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000271
egdaniel170f90b2014-09-16 12:54:40 -0700272 fMatrixState.fViewMatrix = optState.getViewMatrix();
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000273 fMatrixState.fRenderTargetSize = size;
274 fMatrixState.fRenderTargetOrigin = rt->origin();
commit-bot@chromium.org215a6822013-09-05 18:28:42 +0000275
276 GrGLfloat viewMatrix[3 * 3];
277 fMatrixState.getGLMatrix<3>(viewMatrix);
kkinnunendddc18a2014-08-03 23:19:46 -0700278 fProgramDataManager.setMatrix3f(fBuiltinUniformHandles.fViewMatrixUni, viewMatrix);
commit-bot@chromium.org47c66dd2014-05-29 01:12:10 +0000279
280 GrGLfloat rtAdjustmentVec[4];
281 fMatrixState.getRTAdjustmentVec(rtAdjustmentVec);
kkinnunendddc18a2014-08-03 23:19:46 -0700282 fProgramDataManager.set4fv(fBuiltinUniformHandles.fRTAdjustmentUni, 1, rtAdjustmentVec);
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000283 }
284}
joshualitt47bb3822014-10-07 16:43:25 -0700285
286/////////////////////////////////////////////////////////////////////////////////////////
287
288GrGLNvprProgramBase::GrGLNvprProgramBase(GrGpuGL* gpu,
289 const GrGLProgramDesc& desc,
290 const BuiltinUniformHandles& builtinUniforms,
291 GrGLuint programID,
292 const UniformInfoArray& uniforms,
joshualitta5305a12014-10-10 17:47:00 -0700293 GrGLInstalledFragProcs* fragmentProcessors)
294 : INHERITED(gpu, desc, builtinUniforms, programID, uniforms, NULL, fragmentProcessors) {
joshualitt47bb3822014-10-07 16:43:25 -0700295}
296
297void GrGLNvprProgramBase::onSetMatrixAndRenderTargetHeight(GrGpu::DrawType drawType,
298 const GrOptDrawState& optState) {
299 SkASSERT(GrGpu::IsPathRenderingDrawType(drawType));
300 const GrRenderTarget* rt = optState.getRenderTarget();
301 SkISize size;
302 size.set(rt->width(), rt->height());
303 fGpu->glPathRendering()->setProjectionMatrix(optState.getViewMatrix(), size, rt->origin());
304}
305
306/////////////////////////////////////////////////////////////////////////////////////////
307
308GrGLNvprProgram::GrGLNvprProgram(GrGpuGL* gpu,
309 const GrGLProgramDesc& desc,
310 const BuiltinUniformHandles& builtinUniforms,
311 GrGLuint programID,
312 const UniformInfoArray& uniforms,
joshualitta5305a12014-10-10 17:47:00 -0700313 GrGLInstalledFragProcs* fragmentProcessors,
joshualitt47bb3822014-10-07 16:43:25 -0700314 const SeparableVaryingInfoArray& separableVaryings)
joshualitta5305a12014-10-10 17:47:00 -0700315 : INHERITED(gpu, desc, builtinUniforms, programID, uniforms, fragmentProcessors) {
joshualitt47bb3822014-10-07 16:43:25 -0700316 int count = separableVaryings.count();
317 fVaryings.push_back_n(count);
318 for (int i = 0; i < count; i++) {
319 Varying& varying = fVaryings[i];
320 const SeparableVaryingInfo& builderVarying = separableVaryings[i];
321 SkASSERT(GrGLShaderVar::kNonArray == builderVarying.fVariable.getArrayCount());
322 SkDEBUGCODE(
323 varying.fType = builderVarying.fVariable.getType();
324 );
325 varying.fLocation = builderVarying.fLocation;
326 }
327}
328
329void GrGLNvprProgram::didSetData(GrGpu::DrawType drawType) {
330 SkASSERT(GrGpu::IsPathRenderingDrawType(drawType));
331}
332
joshualitta5305a12014-10-10 17:47:00 -0700333void GrGLNvprProgram::setTransformData(const GrFragmentStage& proc, GrGLInstalledFragProc* ip) {
334 SkTArray<GrGLInstalledFragProc::Transform, true>& transforms = ip->fTransforms;
joshualitt47bb3822014-10-07 16:43:25 -0700335 int numTransforms = transforms.count();
joshualitta5305a12014-10-10 17:47:00 -0700336 SkASSERT(numTransforms == proc.getProcessor()->numTransforms());
joshualitt47bb3822014-10-07 16:43:25 -0700337 for (int t = 0; t < numTransforms; ++t) {
338 SkASSERT(transforms[t].fHandle.isValid());
joshualitta5305a12014-10-10 17:47:00 -0700339 const SkMatrix& transform = get_transform_matrix(proc, false, t);
joshualitt47bb3822014-10-07 16:43:25 -0700340 if (transforms[t].fCurrentValue.cheapEqualTo(transform)) {
341 continue;
342 }
343 transforms[t].fCurrentValue = transform;
344 const Varying& fragmentInput = fVaryings[transforms[t].fHandle.handle()];
345 SkASSERT(transforms[t].fType == kVec2f_GrSLType || transforms[t].fType == kVec3f_GrSLType);
346 unsigned components = transforms[t].fType == kVec2f_GrSLType ? 2 : 3;
347 fGpu->glPathRendering()->setProgramPathFragmentInputTransform(fProgramID,
348 fragmentInput.fLocation,
349 GR_GL_OBJECT_LINEAR,
350 components,
351 transform);
352 }
353}
354
355//////////////////////////////////////////////////////////////////////////////////////
356
357GrGLLegacyNvprProgram::GrGLLegacyNvprProgram(GrGpuGL* gpu,
joshualitta5305a12014-10-10 17:47:00 -0700358 const GrGLProgramDesc& desc,
359 const BuiltinUniformHandles& builtinUniforms,
360 GrGLuint programID,
361 const UniformInfoArray& uniforms,
362 GrGLInstalledFragProcs* fps,
363 int texCoordSetCnt)
364 : INHERITED(gpu, desc, builtinUniforms, programID, uniforms, fps)
joshualitt47bb3822014-10-07 16:43:25 -0700365 , fTexCoordSetCnt(texCoordSetCnt) {
366}
367
368void GrGLLegacyNvprProgram::didSetData(GrGpu::DrawType drawType) {
369 SkASSERT(GrGpu::IsPathRenderingDrawType(drawType));
370 fGpu->glPathRendering()->flushPathTexGenSettings(fTexCoordSetCnt);
371}
372
joshualitta5305a12014-10-10 17:47:00 -0700373void
374GrGLLegacyNvprProgram::setTransformData(const GrFragmentStage& proc, GrGLInstalledFragProc* ip) {
joshualitt47bb3822014-10-07 16:43:25 -0700375 // We've hidden the texcoord index in the first entry of the transforms array for each effect
joshualitta5305a12014-10-10 17:47:00 -0700376 int texCoordIndex = ip->fTransforms[0].fHandle.handle();
377 int numTransforms = proc.getProcessor()->numTransforms();
joshualitt47bb3822014-10-07 16:43:25 -0700378 for (int t = 0; t < numTransforms; ++t) {
joshualitta5305a12014-10-10 17:47:00 -0700379 const SkMatrix& transform = get_transform_matrix(proc, false, t);
joshualitt47bb3822014-10-07 16:43:25 -0700380 GrGLPathRendering::PathTexGenComponents components =
381 GrGLPathRendering::kST_PathTexGenComponents;
joshualitta5305a12014-10-10 17:47:00 -0700382 if (proc.isPerspectiveCoordTransform(t, false)) {
joshualitt47bb3822014-10-07 16:43:25 -0700383 components = GrGLPathRendering::kSTR_PathTexGenComponents;
384 }
385 fGpu->glPathRendering()->enablePathTexGen(texCoordIndex++, components, transform);
386 }
387}