blob: 9abf1b195629f32f862ec05a4cd6de18cb425b06 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
junov@google.comf93e7172011-03-31 21:26:24 +00002/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
junov@google.comf93e7172011-03-31 21:26:24 +00007 */
8
epoger@google.comec3ed6a2011-07-28 14:26:00 +00009
junov@google.comf93e7172011-03-31 21:26:24 +000010#ifndef GrGLProgram_DEFINED
11#define GrGLProgram_DEFINED
12
tomhudson@google.com93813632011-10-27 20:21:16 +000013#include "GrDrawState.h"
junov@google.comf93e7172011-03-31 21:26:24 +000014#include "GrGLInterface.h"
bsalomon@google.com91961302011-05-09 18:39:58 +000015#include "GrStringBuilder.h"
bsalomon@google.com271cffc2011-05-20 14:13:56 +000016#include "GrGpu.h"
junov@google.comf93e7172011-03-31 21:26:24 +000017
Scroggo97c88c22011-05-11 14:05:25 +000018#include "SkXfermode.h"
19
junov@google.comf93e7172011-03-31 21:26:24 +000020class GrBinHashKeyBuilder;
junov@google.comd31cbc42011-05-17 17:01:17 +000021
bsalomon@google.com4fa66942011-09-20 19:06:12 +000022struct ShaderCodeSegments;
junov@google.comf93e7172011-03-31 21:26:24 +000023
bsalomon@google.comedfe1aa2011-09-29 14:40:26 +000024// optionally compile the experimental GS code. Set to GR_DEBUG
25// so that debug build bots will execute the code.
26#define GR_GL_EXPERIMENTAL_GS GR_DEBUG
27
junov@google.comf93e7172011-03-31 21:26:24 +000028/**
29 * This class manages a GPU program and records per-program information.
30 * We can specify the attribute locations so that they are constant
31 * across our shaders. But the driver determines the uniform locations
32 * at link time. We don't need to remember the sampler uniform location
33 * because we will bind a texture slot to it and never change it
34 * Uniforms are program-local so we can't rely on fHWState to hold the
35 * previous uniform state after a program change.
36 */
37class GrGLProgram {
38public:
bsalomon@google.com373a6632011-10-19 20:43:20 +000039 // Limited set of GLSL versions we build shaders for. Caller should round
40 // down the GLSL version to one of these enums.
bsalomon@google.com4fa66942011-09-20 19:06:12 +000041 enum GLSLVersion {
bsalomon@google.com373a6632011-10-19 20:43:20 +000042 /**
43 * Desktop GLSL 1.10 and ES2 shading lang (based on desktop GLSL 1.20)
44 */
45 k110_GLSLVersion,
46 /**
47 * Desktop GLSL 1.30
48 */
49 k130_GLSLVersion,
50 /**
51 * Dekstop GLSL 1.50
52 */
53 k150_GLSLVersion,
bsalomon@google.com4fa66942011-09-20 19:06:12 +000054 };
55
junov@google.comf93e7172011-03-31 21:26:24 +000056 class CachedData;
57
58 GrGLProgram();
59 ~GrGLProgram();
60
61 /**
junov@google.comf93e7172011-03-31 21:26:24 +000062 * This is the heavy initilization routine for building a GLProgram.
63 * The result of heavy init is not stored in datamembers of GrGLProgam,
64 * but in a separate cacheable container.
65 */
bsalomon@google.com0b77d682011-08-19 13:28:54 +000066 bool genProgram(const GrGLInterface* gl,
bsalomon@google.com4fa66942011-09-20 19:06:12 +000067 GLSLVersion glslVersion,
bsalomon@google.com0b77d682011-08-19 13:28:54 +000068 CachedData* programData) const;
junov@google.comf93e7172011-03-31 21:26:24 +000069
bsalomon@google.com271cffc2011-05-20 14:13:56 +000070 /**
71 * The shader may modify the blend coeffecients. Params are in/out
72 */
73 void overrideBlend(GrBlendCoeff* srcCoeff, GrBlendCoeff* dstCoeff) const;
74
75 /**
bsalomon@google.comb5b5eaf2011-10-19 13:25:46 +000076 * Attribute indices. These should not overlap. Matrices consume 3 slots.
bsalomon@google.com271cffc2011-05-20 14:13:56 +000077 */
bsalomon@google.com91961302011-05-09 18:39:58 +000078 static int PositionAttributeIdx() { return 0; }
79 static int TexCoordAttributeIdx(int tcIdx) { return 1 + tcIdx; }
tomhudson@google.com93813632011-10-27 20:21:16 +000080 static int ColorAttributeIdx() { return 1 + GrDrawState::kMaxTexCoords; }
bsalomon@google.coma3108262011-10-10 14:08:47 +000081 static int CoverageAttributeIdx() {
tomhudson@google.com93813632011-10-27 20:21:16 +000082 return 2 + GrDrawState::kMaxTexCoords;
bsalomon@google.coma3108262011-10-10 14:08:47 +000083 }
tomhudson@google.com93813632011-10-27 20:21:16 +000084 static int EdgeAttributeIdx() { return 3 + GrDrawState::kMaxTexCoords; }
bsalomon@google.comaeb21602011-08-30 18:13:44 +000085
tomhudson@google.com0d831722011-06-02 15:37:14 +000086 static int ViewMatrixAttributeIdx() {
tomhudson@google.com93813632011-10-27 20:21:16 +000087 return 4 + GrDrawState::kMaxTexCoords;
bsalomon@google.com91961302011-05-09 18:39:58 +000088 }
tomhudson@google.com0d831722011-06-02 15:37:14 +000089 static int TextureMatrixAttributeIdx(int stage) {
tomhudson@google.com93813632011-10-27 20:21:16 +000090 return 7 + GrDrawState::kMaxTexCoords + 3 * stage;
bsalomon@google.com91961302011-05-09 18:39:58 +000091 }
92
tomhudson@google.com2a2e3ef2011-10-25 19:51:09 +000093public:
junov@google.comf93e7172011-03-31 21:26:24 +000094
tomhudson@google.com0d831722011-06-02 15:37:14 +000095 // Parameters that affect code generation
96 // These structs should be kept compact; they are the input to an
97 // expensive hash key generator.
junov@google.comf93e7172011-03-31 21:26:24 +000098 struct ProgramDesc {
bsalomon@google.com4be283f2011-04-19 21:15:09 +000099 ProgramDesc() {
100 // since we use this as part of a key we can't have any unitialized
101 // padding
102 memset(this, 0, sizeof(ProgramDesc));
103 }
104
bsalomon@google.comc4364992011-11-07 15:54:49 +0000105 enum OutputPM {
106 // PM-color OR color with no alpha channel
107 kYes_OutputPM,
108 // nonPM-color with alpha channel
109 kNo_OutputPM,
110
111 kOutputPMCnt
112 };
113
tomhudson@google.com0d831722011-06-02 15:37:14 +0000114 struct StageDesc {
115 enum OptFlagBits {
116 kNoPerspective_OptFlagBit = 1 << 0,
117 kIdentityMatrix_OptFlagBit = 1 << 1,
118 kCustomTextureDomain_OptFlagBit = 1 << 2,
119 kIsEnabled_OptFlagBit = 1 << 7
120 };
tomhudson@google.com0d831722011-06-02 15:37:14 +0000121 enum FetchMode {
122 kSingle_FetchMode,
bsalomon@google.com1e257a52011-07-06 19:52:16 +0000123 k2x2_FetchMode,
senorblanco@chromium.org027de5f2011-07-08 18:03:33 +0000124 kConvolution_FetchMode,
bsalomon@google.com1e257a52011-07-06 19:52:16 +0000125
126 kFetchModeCnt,
tomhudson@google.com0d831722011-06-02 15:37:14 +0000127 };
bsalomon@google.com85b505b2011-11-07 14:56:51 +0000128 enum InputConfig {
129 /**
130 The texture has r,g,b, and optionally a.
131 */
132 kColor_InputConfig,
133 /**
134 The texture is alpha only. This should only be used if the
135 caller is unable to map the r,g,b sample values to the
136 texture's alpha channel (GL_ARB_texture_swizzle).
137 */
138 kAlphaOnly_InputConfig,
139
140 kInputConfigCnt
141 };
tomhudson@google.com0d831722011-06-02 15:37:14 +0000142 enum CoordMapping {
143 kIdentity_CoordMapping,
144 kRadialGradient_CoordMapping,
145 kSweepGradient_CoordMapping,
bsalomon@google.com1e257a52011-07-06 19:52:16 +0000146 kRadial2Gradient_CoordMapping,
bsalomon@google.com22c5dea2011-07-07 14:38:03 +0000147 // need different shader computation when quadratic
148 // eq describing the gradient degenerates to a linear eq.
149 kRadial2GradientDegenerate_CoordMapping,
bsalomon@google.com1e257a52011-07-06 19:52:16 +0000150 kCoordMappingCnt
tomhudson@google.com0d831722011-06-02 15:37:14 +0000151 };
junov@google.comf93e7172011-03-31 21:26:24 +0000152
tomhudson@google.com0d831722011-06-02 15:37:14 +0000153 uint8_t fOptFlags;
bsalomon@google.com85b505b2011-11-07 14:56:51 +0000154 uint8_t fInputConfig; // casts to enum InputConfig
155 uint8_t fFetchMode; // casts to enum FetchMode
tomhudson@google.com0d831722011-06-02 15:37:14 +0000156 uint8_t fCoordMapping; // casts to enum CoordMapping
senorblanco@chromium.org027de5f2011-07-08 18:03:33 +0000157 uint8_t fKernelWidth;
tomhudson@google.com0d831722011-06-02 15:37:14 +0000158
159 inline bool isEnabled() const {
bsalomon@google.comc2c9b972011-10-03 13:17:22 +0000160 return SkToBool(fOptFlags & kIsEnabled_OptFlagBit);
tomhudson@google.com0d831722011-06-02 15:37:14 +0000161 }
162 inline void setEnabled(bool newValue) {
163 if (newValue) {
164 fOptFlags |= kIsEnabled_OptFlagBit;
165 } else {
166 fOptFlags &= ~kIsEnabled_OptFlagBit;
167 }
168 }
169 };
170
bsalomon@google.com86c1f712011-10-12 14:54:26 +0000171 // Specifies where the intitial color comes from before the stages are
172 // applied.
bsalomon@google.com85b505b2011-11-07 14:56:51 +0000173 enum ColorInput {
174 kSolidWhite_ColorInput,
175 kTransBlack_ColorInput,
176 kAttribute_ColorInput,
177 kUniform_ColorInput,
bsalomon@google.com86c1f712011-10-12 14:54:26 +0000178
bsalomon@google.com85b505b2011-11-07 14:56:51 +0000179 kColorInputCnt
tomhudson@google.com0d831722011-06-02 15:37:14 +0000180 };
181 // Dual-src blending makes use of a secondary output color that can be
bsalomon@google.com271cffc2011-05-20 14:13:56 +0000182 // used as a per-pixel blend coeffecient. This controls whether a
183 // secondary source is output and what value it holds.
184 enum DualSrcOutput {
185 kNone_DualSrcOutput,
186 kCoverage_DualSrcOutput,
187 kCoverageISA_DualSrcOutput,
188 kCoverageISC_DualSrcOutput,
bsalomon@google.com1e257a52011-07-06 19:52:16 +0000189
bsalomon@google.com271cffc2011-05-20 14:13:56 +0000190 kDualSrcOutputCnt
tomhudson@google.com0d831722011-06-02 15:37:14 +0000191 };
bsalomon@google.com271cffc2011-05-20 14:13:56 +0000192
tomhudson@google.com93813632011-10-27 20:21:16 +0000193 GrDrawState::VertexEdgeType fVertexEdgeType;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000194
tomhudson@google.com0d831722011-06-02 15:37:14 +0000195 // stripped of bits that don't affect prog generation
196 GrVertexLayout fVertexLayout;
junov@google.comf93e7172011-03-31 21:26:24 +0000197
tomhudson@google.com93813632011-10-27 20:21:16 +0000198 StageDesc fStages[GrDrawState::kNumStages];
Scroggo97c88c22011-05-11 14:05:25 +0000199
bsalomon@google.comedfe1aa2011-09-29 14:40:26 +0000200 // To enable experimental geometry shader code (not for use in
201 // production)
202#if GR_GL_EXPERIMENTAL_GS
203 bool fExperimentalGS;
204#endif
205
bsalomon@google.com85b505b2011-11-07 14:56:51 +0000206 uint8_t fColorInput; // casts to enum ColorInput
bsalomon@google.comc4364992011-11-07 15:54:49 +0000207 uint8_t fOutputPM; // cases to enum OutputPM
bsalomon@google.com85b505b2011-11-07 14:56:51 +0000208 uint8_t fDualSrcOutput; // casts to enum DualSrcOutput
tomhudson@google.com0d831722011-06-02 15:37:14 +0000209 int8_t fFirstCoverageStage;
210 SkBool8 fEmitsPointSize;
senorblanco@chromium.org129b8e32011-06-15 17:52:09 +0000211 SkBool8 fEdgeAAConcave;
junov@google.comf93e7172011-03-31 21:26:24 +0000212
tomhudson@google.com0d831722011-06-02 15:37:14 +0000213 int8_t fEdgeAANumEdges;
214 uint8_t fColorFilterXfermode; // casts to enum SkXfermode::Mode
junov@google.comf93e7172011-03-31 21:26:24 +0000215
junov@google.comf93e7172011-03-31 21:26:24 +0000216 } fProgramDesc;
bsalomon@google.comedfe1aa2011-09-29 14:40:26 +0000217 GR_STATIC_ASSERT(!(sizeof(ProgramDesc) % 4));
junov@google.comf93e7172011-03-31 21:26:24 +0000218
bsalomon@google.com22c5dea2011-07-07 14:38:03 +0000219 // for code readability
220 typedef ProgramDesc::StageDesc StageDesc;
221
tomhudson@google.com2a2e3ef2011-10-25 19:51:09 +0000222private:
223
224 const ProgramDesc& getDesc() { return fProgramDesc; }
225
junov@google.comf93e7172011-03-31 21:26:24 +0000226public:
bsalomon@google.com91961302011-05-09 18:39:58 +0000227 enum {
228 kUnusedUniform = -1,
229 kSetAsAttribute = 1000,
230 };
231
junov@google.comf93e7172011-03-31 21:26:24 +0000232 struct StageUniLocations {
233 GrGLint fTextureMatrixUni;
bsalomon@google.com6aef1fb2011-05-05 12:33:22 +0000234 GrGLint fNormalizedTexelSizeUni;
junov@google.comf93e7172011-03-31 21:26:24 +0000235 GrGLint fSamplerUni;
236 GrGLint fRadial2Uni;
junov@google.com6acc9b32011-05-16 18:32:07 +0000237 GrGLint fTexDomUni;
senorblanco@chromium.org027de5f2011-07-08 18:03:33 +0000238 GrGLint fKernelUni;
239 GrGLint fImageIncrementUni;
bsalomon@google.com91961302011-05-09 18:39:58 +0000240 void reset() {
241 fTextureMatrixUni = kUnusedUniform;
242 fNormalizedTexelSizeUni = kUnusedUniform;
243 fSamplerUni = kUnusedUniform;
244 fRadial2Uni = kUnusedUniform;
junov@google.com6acc9b32011-05-16 18:32:07 +0000245 fTexDomUni = kUnusedUniform;
senorblanco@chromium.org027de5f2011-07-08 18:03:33 +0000246 fKernelUni = kUnusedUniform;
247 fImageIncrementUni = kUnusedUniform;
bsalomon@google.com91961302011-05-09 18:39:58 +0000248 }
junov@google.comf93e7172011-03-31 21:26:24 +0000249 };
250
251 struct UniLocations {
252 GrGLint fViewMatrixUni;
bsalomon@google.com4be283f2011-04-19 21:15:09 +0000253 GrGLint fColorUni;
senorblanco@chromium.org92e0f222011-05-12 15:49:15 +0000254 GrGLint fEdgesUni;
Scroggo97c88c22011-05-11 14:05:25 +0000255 GrGLint fColorFilterUni;
tomhudson@google.com93813632011-10-27 20:21:16 +0000256 StageUniLocations fStages[GrDrawState::kNumStages];
bsalomon@google.com91961302011-05-09 18:39:58 +0000257 void reset() {
258 fViewMatrixUni = kUnusedUniform;
259 fColorUni = kUnusedUniform;
senorblanco@chromium.org92e0f222011-05-12 15:49:15 +0000260 fEdgesUni = kUnusedUniform;
Scroggo97c88c22011-05-11 14:05:25 +0000261 fColorFilterUni = kUnusedUniform;
tomhudson@google.com93813632011-10-27 20:21:16 +0000262 for (int s = 0; s < GrDrawState::kNumStages; ++s) {
bsalomon@google.com91961302011-05-09 18:39:58 +0000263 fStages[s].reset();
264 }
265 }
junov@google.comf93e7172011-03-31 21:26:24 +0000266 };
267
268 class CachedData : public ::GrNoncopyable {
269 public:
270 CachedData() {
junov@google.comf93e7172011-03-31 21:26:24 +0000271 }
272
273 ~CachedData() {
junov@google.comf93e7172011-03-31 21:26:24 +0000274 }
275
276 void copyAndTakeOwnership(CachedData& other) {
bsalomon@google.com2d9ddf92011-05-11 16:52:59 +0000277 memcpy(this, &other, sizeof(*this));
junov@google.comf93e7172011-03-31 21:26:24 +0000278 }
279
junov@google.comf93e7172011-03-31 21:26:24 +0000280 public:
281
282 // IDs
283 GrGLuint fVShaderID;
bsalomon@google.comedfe1aa2011-09-29 14:40:26 +0000284 GrGLuint fGShaderID;
junov@google.comf93e7172011-03-31 21:26:24 +0000285 GrGLuint fFShaderID;
286 GrGLuint fProgramID;
287 // shader uniform locations (-1 if shader doesn't use them)
288 UniLocations fUniLocations;
289
290 GrMatrix fViewMatrix;
291
292 // these reflect the current values of uniforms
293 // (GL uniform values travel with program)
bsalomon@google.com4be283f2011-04-19 21:15:09 +0000294 GrColor fColor;
Scroggo97c88c22011-05-11 14:05:25 +0000295 GrColor fColorFilterColor;
tomhudson@google.com93813632011-10-27 20:21:16 +0000296 GrMatrix fTextureMatrices[GrDrawState::kNumStages];
bsalomon@google.com6aef1fb2011-05-05 12:33:22 +0000297 // width and height used for normalized texel size
tomhudson@google.com93813632011-10-27 20:21:16 +0000298 int fTextureWidth[GrDrawState::kNumStages];
299 int fTextureHeight[GrDrawState::kNumStages];
300 GrScalar fRadial2CenterX1[GrDrawState::kNumStages];
301 GrScalar fRadial2Radius0[GrDrawState::kNumStages];
302 bool fRadial2PosRoot[GrDrawState::kNumStages];
303 GrRect fTextureDomain[GrDrawState::kNumStages];
junov@google.comf93e7172011-03-31 21:26:24 +0000304
305 private:
306 enum Constants {
307 kUniLocationPreAllocSize = 8
308 };
309
junov@google.comf93e7172011-03-31 21:26:24 +0000310 }; // CachedData
311
junov@google.comf7c00f62011-08-18 18:15:16 +0000312 enum Constants {
313 kProgramKeySize = sizeof(ProgramDesc)
314 };
315
316 // Provide an opaque ProgramDesc
317 const uint32_t* keyData() const{
318 return reinterpret_cast<const uint32_t*>(&fProgramDesc);
319 }
320
junov@google.comf93e7172011-03-31 21:26:24 +0000321private:
bsalomon@google.com91961302011-05-09 18:39:58 +0000322
tomhudson@google.com2a2e3ef2011-10-25 19:51:09 +0000323 // Determines which uniforms will need to be bound.
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000324 void genStageCode(const GrGLInterface* gl,
325 int stageNum,
junov@google.comf93e7172011-03-31 21:26:24 +0000326 const ProgramDesc::StageDesc& desc,
327 const char* fsInColor, // NULL means no incoming color
328 const char* fsOutColor,
329 const char* vsInCoord,
330 ShaderCodeSegments* segments,
331 StageUniLocations* locations) const;
bsalomon@google.com4fa66942011-09-20 19:06:12 +0000332
bsalomon@google.comedfe1aa2011-09-29 14:40:26 +0000333 void genGeometryShader(const GrGLInterface* gl,
334 GLSLVersion glslVersion,
335 ShaderCodeSegments* segments) const;
336
bsalomon@google.com66105672011-09-15 15:12:00 +0000337 // generates code to compute coverage based on edge AA.
338 void genEdgeCoverage(const GrGLInterface* gl,
339 GrVertexLayout layout,
340 CachedData* programData,
341 GrStringBuilder* coverageVar,
342 ShaderCodeSegments* segments) const;
junov@google.comf93e7172011-03-31 21:26:24 +0000343
bsalomon@google.comedfe1aa2011-09-29 14:40:26 +0000344 static bool CompileShaders(const GrGLInterface* gl,
bsalomon@google.com4fa66942011-09-20 19:06:12 +0000345 GLSLVersion glslVersion,
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000346 const ShaderCodeSegments& segments,
bsalomon@google.com91961302011-05-09 18:39:58 +0000347 CachedData* programData);
348
junov@google.comf93e7172011-03-31 21:26:24 +0000349 // Compiles a GL shader, returns shader ID or 0 if failed
350 // params have same meaning as glShaderSource
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000351 static GrGLuint CompileShader(const GrGLInterface* gl,
352 GrGLenum type, int stringCnt,
junov@google.comf93e7172011-03-31 21:26:24 +0000353 const char** strings,
354 int* stringLengths);
355
bsalomon@google.com91961302011-05-09 18:39:58 +0000356 // Creates a GL program ID, binds shader attributes to GL vertex attrs, and
357 // links the program
bsalomon@google.com271cffc2011-05-20 14:13:56 +0000358 bool bindOutputsAttribsAndLinkProgram(
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000359 const GrGLInterface* gl,
tomhudson@google.com93813632011-10-27 20:21:16 +0000360 GrStringBuilder texCoordAttrNames[GrDrawState::kMaxTexCoords],
bsalomon@google.com271cffc2011-05-20 14:13:56 +0000361 bool bindColorOut,
362 bool bindDualSrcOut,
363 CachedData* programData) const;
bsalomon@google.com91961302011-05-09 18:39:58 +0000364
tomhudson@google.com2a2e3ef2011-10-25 19:51:09 +0000365 // Binds uniforms; initializes cache to invalid values.
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000366 void getUniformLocationsAndInitCache(const GrGLInterface* gl,
367 CachedData* programData) const;
bsalomon@google.com91961302011-05-09 18:39:58 +0000368
junov@google.comf93e7172011-03-31 21:26:24 +0000369 friend class GrGpuGLShaders;
370};
371
372#endif