blob: 534e3c38a1bc10d10e4203e94275b7a5749873ad [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 */
7
8#include "GrGLProgram.h"
9
tomhudson@google.comd8f856c2012-05-10 12:13:36 +000010#include "GrAllocator.h"
bsalomon@google.coma469c282012-10-24 18:28:34 +000011#include "GrEffect.h"
bsalomon@google.com77af6802013-10-02 13:04:56 +000012#include "GrCoordTransform.h"
bsalomon@google.comc7818882013-03-20 19:19:53 +000013#include "GrDrawEffect.h"
bsalomon@google.comd698f772012-10-25 13:22:00 +000014#include "GrGLEffect.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"
Scroggo97c88c22011-05-11 14:05:25 +000019#include "SkXfermode.h"
20
commit-bot@chromium.org9188a152013-09-05 18:28:24 +000021#define GL_CALL(X) GR_GL_CALL(fGpu->glInterface(), X)
22#define GL_CALL_RET(R, X) GR_GL_CALL_RET(fGpu->glInterface(), R, X)
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000023
commit-bot@chromium.org9188a152013-09-05 18:28:24 +000024GrGLProgram* GrGLProgram::Create(GrGpuGL* gpu,
bsalomon@google.com31ec7982013-03-27 18:14:57 +000025 const GrGLProgramDesc& desc,
bsalomon@google.com2c84aa32013-06-06 20:28:57 +000026 const GrEffectStage* colorStages[],
27 const GrEffectStage* coverageStages[]) {
joshualitt30ba4362014-08-21 20:18:45 -070028 SkAutoTDelete<GrGLProgramBuilder> builder;
kkinnunenec56e452014-08-25 22:21:16 -070029 if (!desc.getHeader().fRequiresVertexShader &&
30 gpu->glCaps().pathRenderingSupport() &&
31 gpu->glPathRendering()->texturingMode() == GrGLPathRendering::FixedFunction_TexturingMode) {
joshualitt30ba4362014-08-21 20:18:45 -070032 builder.reset(SkNEW_ARGS(GrGLFragmentOnlyProgramBuilder, (gpu, desc)));
kkinnunenec56e452014-08-25 22:21:16 -070033 } else {
34 builder.reset(SkNEW_ARGS(GrGLFullProgramBuilder, (gpu, desc)));
kkinnunendddc18a2014-08-03 23:19:46 -070035 }
kkinnunendddc18a2014-08-03 23:19:46 -070036 if (builder->genProgram(colorStages, coverageStages)) {
37 SkASSERT(0 != builder->getProgramID());
38 return SkNEW_ARGS(GrGLProgram, (gpu, desc, *builder));
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000039 }
commit-bot@chromium.org6eac42e2014-05-29 21:29:51 +000040 return NULL;
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000041}
42
commit-bot@chromium.org9188a152013-09-05 18:28:24 +000043GrGLProgram::GrGLProgram(GrGpuGL* gpu,
bsalomon@google.com31ec7982013-03-27 18:14:57 +000044 const GrGLProgramDesc& desc,
joshualitt30ba4362014-08-21 20:18:45 -070045 const GrGLProgramBuilder& builder)
commit-bot@chromium.orga05fa062014-05-30 18:55:03 +000046 : fColor(GrColor_ILLEGAL)
commit-bot@chromium.org6eac42e2014-05-29 21:29:51 +000047 , fCoverage(GrColor_ILLEGAL)
48 , fDstCopyTexUnit(-1)
kkinnunendddc18a2014-08-03 23:19:46 -070049 , fBuiltinUniformHandles(builder.getBuiltinUniformHandles())
50 , fColorEffects(SkRef(builder.getColorEffects()))
51 , fCoverageEffects(SkRef(builder.getCoverageEffects()))
52 , fProgramID(builder.getProgramID())
53 , fHasVertexShader(builder.hasVertexShader())
54 , fTexCoordSetCnt(builder.getTexCoordSetCount())
commit-bot@chromium.org6eac42e2014-05-29 21:29:51 +000055 , fDesc(desc)
56 , fGpu(gpu)
kkinnunendddc18a2014-08-03 23:19:46 -070057 , fProgramDataManager(gpu, this, builder) {
commit-bot@chromium.org6eac42e2014-05-29 21:29:51 +000058 this->initSamplerUniforms();
junov@google.comf93e7172011-03-31 21:26:24 +000059}
60
61GrGLProgram::~GrGLProgram() {
kkinnunendddc18a2014-08-03 23:19:46 -070062 if (fProgramID) {
63 GL_CALL(DeleteProgram(fProgramID));
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000064 }
junov@google.comf93e7172011-03-31 21:26:24 +000065}
66
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000067void GrGLProgram::abandon() {
kkinnunendddc18a2014-08-03 23:19:46 -070068 fProgramID = 0;
bsalomon@google.comecb60aa2012-07-18 13:20:29 +000069}
70
tomhudson@google.com0d3f1fb2011-06-01 19:27:31 +000071void GrGLProgram::overrideBlend(GrBlendCoeff* srcCoeff,
bsalomon@google.com271cffc2011-05-20 14:13:56 +000072 GrBlendCoeff* dstCoeff) const {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +000073 switch (fDesc.getHeader().fCoverageOutput) {
bsalomon@google.com5920ac22013-04-19 13:14:45 +000074 case GrGLProgramDesc::kModulate_CoverageOutput:
bsalomon@google.com271cffc2011-05-20 14:13:56 +000075 break;
bsalomon@google.com5920ac22013-04-19 13:14:45 +000076 // The prog will write a coverage value to the secondary
bsalomon@google.com271cffc2011-05-20 14:13:56 +000077 // output and the dst is blended by one minus that value.
bsalomon@google.com5920ac22013-04-19 13:14:45 +000078 case GrGLProgramDesc::kSecondaryCoverage_CoverageOutput:
79 case GrGLProgramDesc::kSecondaryCoverageISA_CoverageOutput:
80 case GrGLProgramDesc::kSecondaryCoverageISC_CoverageOutput:
81 *dstCoeff = (GrBlendCoeff)GrGpu::kIS2C_GrBlendCoeff;
82 break;
83 case GrGLProgramDesc::kCombineWithDst_CoverageOutput:
84 // We should only have set this if the blend was specified as (1, 0)
tfarina@chromium.orgf6de4752013-08-17 00:02:59 +000085 SkASSERT(kOne_GrBlendCoeff == *srcCoeff && kZero_GrBlendCoeff == *dstCoeff);
bsalomon@google.com5920ac22013-04-19 13:14:45 +000086 break;
bsalomon@google.com271cffc2011-05-20 14:13:56 +000087 default:
commit-bot@chromium.org88cb22b2014-04-30 14:17:00 +000088 SkFAIL("Unexpected coverage output");
bsalomon@google.com271cffc2011-05-20 14:13:56 +000089 break;
90 }
91}
92
bsalomon@google.comdbbc4e22012-07-25 17:48:39 +000093void GrGLProgram::initSamplerUniforms() {
kkinnunendddc18a2014-08-03 23:19:46 -070094 GL_CALL(UseProgram(fProgramID));
bsalomon@google.com34cccde2013-01-04 18:34:30 +000095 GrGLint texUnitIdx = 0;
kkinnunendddc18a2014-08-03 23:19:46 -070096 if (fBuiltinUniformHandles.fDstCopySamplerUni.isValid()) {
97 fProgramDataManager.setSampler(fBuiltinUniformHandles.fDstCopySamplerUni, texUnitIdx);
bsalomon@google.com804e9942013-06-06 18:04:38 +000098 fDstCopyTexUnit = texUnitIdx++;
bsalomon@google.com26e18b52013-03-29 19:22:36 +000099 }
kkinnunendddc18a2014-08-03 23:19:46 -0700100 fColorEffects->initSamplers(fProgramDataManager, &texUnitIdx);
101 fCoverageEffects->initSamplers(fProgramDataManager, &texUnitIdx);
bsalomon@google.com91961302011-05-09 18:39:58 +0000102}
103
bsalomon@google.comeb715c82012-07-11 15:03:31 +0000104///////////////////////////////////////////////////////////////////////////////
junov@google.comf93e7172011-03-31 21:26:24 +0000105
kkinnunenec56e452014-08-25 22:21:16 -0700106void GrGLProgram::setData(GrGpu::DrawType drawType,
107 GrDrawState::BlendOptFlags blendOpts,
bsalomon@google.com2c84aa32013-06-06 20:28:57 +0000108 const GrEffectStage* colorStages[],
109 const GrEffectStage* coverageStages[],
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000110 const GrDeviceCoordTexture* dstCopy,
bsalomon@google.com91207482013-02-12 21:45:24 +0000111 SharedGLState* sharedState) {
commit-bot@chromium.org9188a152013-09-05 18:28:24 +0000112 const GrDrawState& drawState = fGpu->getDrawState();
bsalomon@google.com9ba4fa62012-07-16 17:36:28 +0000113
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000114 GrColor color;
115 GrColor coverage;
116 if (blendOpts & GrDrawState::kEmitTransBlack_BlendOptFlag) {
117 color = 0;
118 coverage = 0;
119 } else if (blendOpts & GrDrawState::kEmitCoverage_BlendOptFlag) {
120 color = 0xffffffff;
commit-bot@chromium.orge0a868c2013-11-22 07:02:11 +0000121 coverage = drawState.getCoverageColor();
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000122 } else {
123 color = drawState.getColor();
commit-bot@chromium.orge0a868c2013-11-22 07:02:11 +0000124 coverage = drawState.getCoverageColor();
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000125 }
126
bsalomon@google.com91207482013-02-12 21:45:24 +0000127 this->setColor(drawState, color, sharedState);
128 this->setCoverage(drawState, coverage, sharedState);
kkinnunenec56e452014-08-25 22:21:16 -0700129 this->setMatrixAndRenderTargetHeight(drawType, drawState);
bsalomon@google.com91207482013-02-12 21:45:24 +0000130
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000131 if (NULL != dstCopy) {
kkinnunendddc18a2014-08-03 23:19:46 -0700132 if (fBuiltinUniformHandles.fDstCopyTopLeftUni.isValid()) {
133 fProgramDataManager.set2f(fBuiltinUniformHandles.fDstCopyTopLeftUni,
kkinnunen7510b222014-07-30 00:04:16 -0700134 static_cast<GrGLfloat>(dstCopy->offset().fX),
135 static_cast<GrGLfloat>(dstCopy->offset().fY));
kkinnunendddc18a2014-08-03 23:19:46 -0700136 fProgramDataManager.set2f(fBuiltinUniformHandles.fDstCopyScaleUni,
kkinnunen7510b222014-07-30 00:04:16 -0700137 1.f / dstCopy->texture()->width(),
138 1.f / dstCopy->texture()->height());
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000139 GrGLTexture* texture = static_cast<GrGLTexture*>(dstCopy->texture());
140 static GrTextureParams kParams; // the default is clamp, nearest filtering.
commit-bot@chromium.org9188a152013-09-05 18:28:24 +0000141 fGpu->bindTexture(fDstCopyTexUnit, kParams, texture);
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000142 } else {
kkinnunendddc18a2014-08-03 23:19:46 -0700143 SkASSERT(!fBuiltinUniformHandles.fDstCopyScaleUni.isValid());
144 SkASSERT(!fBuiltinUniformHandles.fDstCopySamplerUni.isValid());
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000145 }
146 } else {
kkinnunendddc18a2014-08-03 23:19:46 -0700147 SkASSERT(!fBuiltinUniformHandles.fDstCopyTopLeftUni.isValid());
148 SkASSERT(!fBuiltinUniformHandles.fDstCopyScaleUni.isValid());
149 SkASSERT(!fBuiltinUniformHandles.fDstCopySamplerUni.isValid());
bsalomon@google.com26e18b52013-03-29 19:22:36 +0000150 }
bsalomon@google.comc7818882013-03-20 19:19:53 +0000151
kkinnunenec56e452014-08-25 22:21:16 -0700152 fColorEffects->setData(fGpu, drawType,fProgramDataManager, colorStages);
153 fCoverageEffects->setData(fGpu, drawType,fProgramDataManager, coverageStages);
commit-bot@chromium.org20807222013-11-01 11:54:54 +0000154
commit-bot@chromium.org0a6fe712014-04-23 19:26:26 +0000155 // PathTexGen state applies to the the fixed function vertex shader. For
156 // custom shaders, it's ignored, so we don't need to change the texgen
157 // settings in that case.
kkinnunendddc18a2014-08-03 23:19:46 -0700158 if (!fHasVertexShader) {
kkinnunenccdaa042014-08-20 01:36:23 -0700159 fGpu->glPathRendering()->flushPathTexGenSettings(fTexCoordSetCnt);
commit-bot@chromium.org6b30e452013-10-04 20:02:53 +0000160 }
skia.committer@gmail.com8ae714b2013-01-05 02:02:05 +0000161}
bsalomon@google.com91207482013-02-12 21:45:24 +0000162
163void GrGLProgram::setColor(const GrDrawState& drawState,
164 GrColor color,
165 SharedGLState* sharedState) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000166 const GrGLProgramDesc::KeyHeader& header = fDesc.getHeader();
egdaniel02cafcc2014-07-21 11:37:28 -0700167 if (!drawState.hasColorVertexAttribute() || drawState.canIgnoreColorAttribute()) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000168 switch (header.fColorInput) {
bsalomon@google.com31ec7982013-03-27 18:14:57 +0000169 case GrGLProgramDesc::kAttribute_ColorInput:
tfarina@chromium.orgf6de4752013-08-17 00:02:59 +0000170 SkASSERT(-1 != header.fColorAttributeIndex);
jvanverth@google.com054ae992013-04-01 20:06:51 +0000171 if (sharedState->fConstAttribColor != color ||
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000172 sharedState->fConstAttribColorIndex != header.fColorAttributeIndex) {
bsalomon@google.com91207482013-02-12 21:45:24 +0000173 // OpenGL ES only supports the float varieties of glVertexAttrib
174 GrGLfloat c[4];
175 GrColorToRGBAFloat(color, c);
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000176 GL_CALL(VertexAttrib4fv(header.fColorAttributeIndex, c));
bsalomon@google.com91207482013-02-12 21:45:24 +0000177 sharedState->fConstAttribColor = color;
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000178 sharedState->fConstAttribColorIndex = header.fColorAttributeIndex;
bsalomon@google.com91207482013-02-12 21:45:24 +0000179 }
180 break;
bsalomon@google.com31ec7982013-03-27 18:14:57 +0000181 case GrGLProgramDesc::kUniform_ColorInput:
kkinnunendddc18a2014-08-03 23:19:46 -0700182 if (fColor != color && fBuiltinUniformHandles.fColorUni.isValid()) {
bsalomon@google.com91207482013-02-12 21:45:24 +0000183 // OpenGL ES doesn't support unsigned byte varieties of glUniform
184 GrGLfloat c[4];
185 GrColorToRGBAFloat(color, c);
kkinnunendddc18a2014-08-03 23:19:46 -0700186 fProgramDataManager.set4fv(fBuiltinUniformHandles.fColorUni, 1, c);
bsalomon@google.com91207482013-02-12 21:45:24 +0000187 fColor = color;
188 }
jvanverth@google.com054ae992013-04-01 20:06:51 +0000189 sharedState->fConstAttribColorIndex = -1;
bsalomon@google.com91207482013-02-12 21:45:24 +0000190 break;
egdaniel842b0862014-09-02 10:01:30 -0700191 case GrGLProgramDesc::kAllOnes_ColorInput:
192 sharedState->fConstAttribColorIndex = -1;
193 break;
bsalomon@google.com91207482013-02-12 21:45:24 +0000194 default:
egdaniel02cafcc2014-07-21 11:37:28 -0700195 SkFAIL("Unexpected color type.");
bsalomon@google.com91207482013-02-12 21:45:24 +0000196 }
jvanverth@google.com054ae992013-04-01 20:06:51 +0000197 } else {
198 sharedState->fConstAttribColorIndex = -1;
bsalomon@google.com91207482013-02-12 21:45:24 +0000199 }
200}
201
202void GrGLProgram::setCoverage(const GrDrawState& drawState,
203 GrColor coverage,
204 SharedGLState* sharedState) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000205 const GrGLProgramDesc::KeyHeader& header = fDesc.getHeader();
jvanverth@google.com054ae992013-04-01 20:06:51 +0000206 if (!drawState.hasCoverageVertexAttribute()) {
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000207 switch (header.fCoverageInput) {
bsalomon@google.com31ec7982013-03-27 18:14:57 +0000208 case GrGLProgramDesc::kAttribute_ColorInput:
jvanverth@google.com054ae992013-04-01 20:06:51 +0000209 if (sharedState->fConstAttribCoverage != coverage ||
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000210 sharedState->fConstAttribCoverageIndex != header.fCoverageAttributeIndex) {
bsalomon@google.com91207482013-02-12 21:45:24 +0000211 // OpenGL ES only supports the float varieties of glVertexAttrib
212 GrGLfloat c[4];
213 GrColorToRGBAFloat(coverage, c);
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000214 GL_CALL(VertexAttrib4fv(header.fCoverageAttributeIndex, c));
bsalomon@google.com91207482013-02-12 21:45:24 +0000215 sharedState->fConstAttribCoverage = coverage;
bsalomon@google.com2db3ded2013-05-22 14:34:04 +0000216 sharedState->fConstAttribCoverageIndex = header.fCoverageAttributeIndex;
bsalomon@google.com91207482013-02-12 21:45:24 +0000217 }
218 break;
bsalomon@google.com31ec7982013-03-27 18:14:57 +0000219 case GrGLProgramDesc::kUniform_ColorInput:
bsalomon@google.com91207482013-02-12 21:45:24 +0000220 if (fCoverage != coverage) {
221 // OpenGL ES doesn't support unsigned byte varieties of glUniform
222 GrGLfloat c[4];
223 GrColorToRGBAFloat(coverage, c);
kkinnunendddc18a2014-08-03 23:19:46 -0700224 fProgramDataManager.set4fv(fBuiltinUniformHandles.fCoverageUni, 1, c);
bsalomon@google.com91207482013-02-12 21:45:24 +0000225 fCoverage = coverage;
226 }
jvanverth@google.com054ae992013-04-01 20:06:51 +0000227 sharedState->fConstAttribCoverageIndex = -1;
bsalomon@google.com91207482013-02-12 21:45:24 +0000228 break;
egdaniel842b0862014-09-02 10:01:30 -0700229 case GrGLProgramDesc::kAllOnes_ColorInput:
jvanverth@google.com054ae992013-04-01 20:06:51 +0000230 sharedState->fConstAttribCoverageIndex = -1;
bsalomon@google.com91207482013-02-12 21:45:24 +0000231 break;
232 default:
egdaniel02cafcc2014-07-21 11:37:28 -0700233 SkFAIL("Unexpected coverage type.");
bsalomon@google.com91207482013-02-12 21:45:24 +0000234 }
jvanverth@google.com054ae992013-04-01 20:06:51 +0000235 } else {
236 sharedState->fConstAttribCoverageIndex = -1;
bsalomon@google.com91207482013-02-12 21:45:24 +0000237 }
238}
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000239
kkinnunenec56e452014-08-25 22:21:16 -0700240void GrGLProgram::setMatrixAndRenderTargetHeight(GrGpu::DrawType drawType,
241 const GrDrawState& drawState) {
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000242 const GrRenderTarget* rt = drawState.getRenderTarget();
243 SkISize size;
244 size.set(rt->width(), rt->height());
245
246 // Load the RT height uniform if it is needed to y-flip gl_FragCoord.
kkinnunendddc18a2014-08-03 23:19:46 -0700247 if (fBuiltinUniformHandles.fRTHeightUni.isValid() &&
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000248 fMatrixState.fRenderTargetSize.fHeight != size.fHeight) {
kkinnunendddc18a2014-08-03 23:19:46 -0700249 fProgramDataManager.set1f(fBuiltinUniformHandles.fRTHeightUni,
kkinnunen7510b222014-07-30 00:04:16 -0700250 SkIntToScalar(size.fHeight));
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000251 }
252
kkinnunenec56e452014-08-25 22:21:16 -0700253 if (GrGpu::IsPathRenderingDrawType(drawType)) {
kkinnunenccdaa042014-08-20 01:36:23 -0700254 fGpu->glPathRendering()->setProjectionMatrix(drawState.getViewMatrix(), size, rt->origin());
commit-bot@chromium.org6b30e452013-10-04 20:02:53 +0000255 } else if (fMatrixState.fRenderTargetOrigin != rt->origin() ||
256 fMatrixState.fRenderTargetSize != size ||
257 !fMatrixState.fViewMatrix.cheapEqualTo(drawState.getViewMatrix())) {
kkinnunendddc18a2014-08-03 23:19:46 -0700258 SkASSERT(fBuiltinUniformHandles.fViewMatrixUni.isValid());
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000259
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000260 fMatrixState.fViewMatrix = drawState.getViewMatrix();
261 fMatrixState.fRenderTargetSize = size;
262 fMatrixState.fRenderTargetOrigin = rt->origin();
commit-bot@chromium.org215a6822013-09-05 18:28:42 +0000263
264 GrGLfloat viewMatrix[3 * 3];
265 fMatrixState.getGLMatrix<3>(viewMatrix);
kkinnunendddc18a2014-08-03 23:19:46 -0700266 fProgramDataManager.setMatrix3f(fBuiltinUniformHandles.fViewMatrixUni, viewMatrix);
commit-bot@chromium.org47c66dd2014-05-29 01:12:10 +0000267
268 GrGLfloat rtAdjustmentVec[4];
269 fMatrixState.getRTAdjustmentVec(rtAdjustmentVec);
kkinnunendddc18a2014-08-03 23:19:46 -0700270 fProgramDataManager.set4fv(fBuiltinUniformHandles.fRTAdjustmentUni, 1, rtAdjustmentVec);
bsalomon@google.com6a51dcb2013-02-13 16:03:51 +0000271 }
272}