blob: b7946fc07a95435fb59e469fa91a7dc85ce0893f [file] [log] [blame]
tomhudson@google.com086e5352011-12-08 14:44:10 +00001/*
2 * 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.
6 */
7
8#include "GrGLSL.h"
bsalomon@google.come55fd0f2012-02-10 15:56:06 +00009#include "GrGLShaderVar.h"
bsalomon@google.com4af0af62012-08-29 12:59:57 +000010#include "SkString.h"
tomhudson@google.com086e5352011-12-08 14:44:10 +000011
robertphillips@google.com97055962013-04-01 22:20:59 +000012GrGLSLGeneration GrGetGLSLGeneration(GrGLBinding binding,
13 const GrGLInterface* gl) {
tomhudson@google.com086e5352011-12-08 14:44:10 +000014 GrGLSLVersion ver = GrGLGetGLSLVersion(gl);
15 switch (binding) {
16 case kDesktop_GrGLBinding:
17 GrAssert(ver >= GR_GLSL_VER(1,10));
18 if (ver >= GR_GLSL_VER(1,50)) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000019 return k150_GrGLSLGeneration;
bsalomon@google.com281c7262012-10-23 14:31:30 +000020 } else if (ver >= GR_GLSL_VER(1,40)) {
21 return k140_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000022 } else if (ver >= GR_GLSL_VER(1,30)) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000023 return k130_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000024 } else {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000025 return k110_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000026 }
27 case kES2_GrGLBinding:
28 // version 1.00 of ES GLSL based on ver 1.20 of desktop GLSL
29 GrAssert(ver >= GR_GL_VER(1,00));
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000030 return k110_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000031 default:
32 GrCrash("Unknown GL Binding");
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000033 return k110_GrGLSLGeneration; // suppress warning
tomhudson@google.com086e5352011-12-08 14:44:10 +000034 }
35}
36
robertphillips@google.com97055962013-04-01 22:20:59 +000037const char* GrGetGLSLVersionDecl(GrGLBinding binding,
38 GrGLSLGeneration gen) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000039 switch (gen) {
40 case k110_GrGLSLGeneration:
41 if (kES2_GrGLBinding == binding) {
42 // ES2s shader language is based on version 1.20 but is version
43 // 1.00 of the ES language.
44 return "#version 100\n";
45 } else {
46 GrAssert(kDesktop_GrGLBinding == binding);
47 return "#version 110\n";
48 }
49 case k130_GrGLSLGeneration:
50 GrAssert(kDesktop_GrGLBinding == binding);
51 return "#version 130\n";
bsalomon@google.com281c7262012-10-23 14:31:30 +000052 case k140_GrGLSLGeneration:
53 GrAssert(kDesktop_GrGLBinding == binding);
54 return "#version 140\n";
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000055 case k150_GrGLSLGeneration:
56 GrAssert(kDesktop_GrGLBinding == binding);
57 return "#version 150\n";
58 default:
59 GrCrash("Unknown GL version.");
60 return ""; // suppress warning
61 }
62}
63
robertphillips@google.com97055962013-04-01 22:20:59 +000064bool GrGLSLSetupFSColorOuput(GrGLSLGeneration gen,
65 const char* nameIfDeclared,
66 GrGLShaderVar* var) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000067 bool declaredOutput = k110_GrGLSLGeneration != gen;
tomhudson@google.com168e6342012-04-18 17:49:20 +000068 var->set(kVec4f_GrSLType,
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000069 GrGLShaderVar::kOut_TypeModifier,
70 declaredOutput ? nameIfDeclared : "gl_FragColor");
71 return declaredOutput;
72}
tomhudson@google.com168e6342012-04-18 17:49:20 +000073
tomhudson@google.com168e6342012-04-18 17:49:20 +000074const char* GrGLSLVectorHomogCoord(int count) {
75 static const char* HOMOGS[] = {"ERROR", "", ".y", ".z", ".w"};
76 GrAssert(count >= 1 && count < (int)GR_ARRAY_COUNT(HOMOGS));
77 return HOMOGS[count];
78}
79
bsalomon@google.com34bcb9f2012-08-28 18:20:18 +000080const char* GrGLSLVectorHomogCoord(GrSLType type) {
81 return GrGLSLVectorHomogCoord(GrSLTypeToVecLength(type));
82}
83
tomhudson@google.com168e6342012-04-18 17:49:20 +000084const char* GrGLSLVectorNonhomogCoords(int count) {
85 static const char* NONHOMOGS[] = {"ERROR", "", ".x", ".xy", ".xyz"};
86 GrAssert(count >= 1 && count < (int)GR_ARRAY_COUNT(NONHOMOGS));
87 return NONHOMOGS[count];
88}
89
robertphillips@google.com5b5bba32012-09-24 14:20:00 +000090const char* GrGLSLVectorNonhomogCoords(GrSLType type) {
bsalomon@google.com34bcb9f2012-08-28 18:20:18 +000091 return GrGLSLVectorNonhomogCoords(GrSLTypeToVecLength(type));
92}
bsalomon@google.com4af0af62012-08-29 12:59:57 +000093
94GrSLConstantVec GrGLSLModulate4f(SkString* outAppend,
95 const char* in0,
96 const char* in1,
97 GrSLConstantVec default0,
98 GrSLConstantVec default1) {
99 GrAssert(NULL != outAppend);
100
101 bool has0 = NULL != in0 && '\0' != *in0;
102 bool has1 = NULL != in1 && '\0' != *in1;
103
104 GrAssert(has0 || kNone_GrSLConstantVec != default0);
105 GrAssert(has1 || kNone_GrSLConstantVec != default1);
106
107 if (!has0 && !has1) {
108 GrAssert(kZeros_GrSLConstantVec == default0 || kOnes_GrSLConstantVec == default0);
109 GrAssert(kZeros_GrSLConstantVec == default1 || kOnes_GrSLConstantVec == default1);
110 if (kZeros_GrSLConstantVec == default0 || kZeros_GrSLConstantVec == default1) {
111 outAppend->append(GrGLSLZerosVecf(4));
112 return kZeros_GrSLConstantVec;
113 } else {
bsalomon@google.com868a8e72012-08-30 19:11:34 +0000114 // both inputs are ones vectors
bsalomon@google.comd9e01812012-08-29 19:35:44 +0000115 outAppend->append(GrGLSLOnesVecf(4));
bsalomon@google.com4af0af62012-08-29 12:59:57 +0000116 return kOnes_GrSLConstantVec;
117 }
118 } else if (!has0) {
119 GrAssert(kZeros_GrSLConstantVec == default0 || kOnes_GrSLConstantVec == default0);
120 if (kZeros_GrSLConstantVec == default0) {
121 outAppend->append(GrGLSLZerosVecf(4));
122 return kZeros_GrSLConstantVec;
123 } else {
124 outAppend->appendf("vec4(%s)", in1);
125 return kNone_GrSLConstantVec;
126 }
127 } else if (!has1) {
128 GrAssert(kZeros_GrSLConstantVec == default1 || kOnes_GrSLConstantVec == default1);
129 if (kZeros_GrSLConstantVec == default1) {
130 outAppend->append(GrGLSLZerosVecf(4));
131 return kZeros_GrSLConstantVec;
132 } else {
133 outAppend->appendf("vec4(%s)", in0);
134 return kNone_GrSLConstantVec;
135 }
136 } else {
137 outAppend->appendf("vec4(%s * %s)", in0, in1);
138 return kNone_GrSLConstantVec;
139 }
140}
141
bsalomon@google.com868a8e72012-08-30 19:11:34 +0000142namespace {
143void append_tabs(SkString* outAppend, int tabCnt) {
144 static const char kTabs[] = "\t\t\t\t\t\t\t\t";
145 while (tabCnt) {
146 int cnt = GrMin((int)GR_ARRAY_COUNT(kTabs), tabCnt);
147 outAppend->append(kTabs, cnt);
148 tabCnt -= cnt;
149 }
150}
151}
152
153GrSLConstantVec GrGLSLMulVarBy4f(SkString* outAppend,
154 int tabCnt,
155 const char* vec4VarName,
156 const char* mulFactor,
157 GrSLConstantVec mulFactorDefault) {
158 bool haveFactor = NULL != mulFactor && '\0' != *mulFactor;
159
160 GrAssert(NULL != outAppend);
161 GrAssert(NULL != vec4VarName);
162 GrAssert(kNone_GrSLConstantVec != mulFactorDefault || haveFactor);
163
164 if (!haveFactor) {
165 if (kOnes_GrSLConstantVec == mulFactorDefault) {
166 return kNone_GrSLConstantVec;
167 } else {
168 GrAssert(kZeros_GrSLConstantVec == mulFactorDefault);
169 append_tabs(outAppend, tabCnt);
170 outAppend->appendf("%s = vec4(0, 0, 0, 0);\n", vec4VarName);
171 return kZeros_GrSLConstantVec;
172 }
173 }
174 append_tabs(outAppend, tabCnt);
175 outAppend->appendf("%s *= %s;\n", vec4VarName, mulFactor);
176 return kNone_GrSLConstantVec;
177}
178
bsalomon@google.com4af0af62012-08-29 12:59:57 +0000179GrSLConstantVec GrGLSLAdd4f(SkString* outAppend,
180 const char* in0,
181 const char* in1,
182 GrSLConstantVec default0,
183 GrSLConstantVec default1) {
184 GrAssert(NULL != outAppend);
185
186 bool has0 = NULL != in0 && '\0' != *in0;
187 bool has1 = NULL != in1 && '\0' != *in1;
188
189 if (!has0 && !has1) {
190 GrAssert(kZeros_GrSLConstantVec == default0);
191 GrAssert(kZeros_GrSLConstantVec == default1);
192 outAppend->append(GrGLSLZerosVecf(4));
193 return kZeros_GrSLConstantVec;
194 } else if (!has0) {
195 GrAssert(kZeros_GrSLConstantVec == default0);
196 outAppend->appendf("vec4(%s)", in1);
197 return kNone_GrSLConstantVec;
198 } else if (!has1) {
199 GrAssert(kZeros_GrSLConstantVec == default1);
200 outAppend->appendf("vec4(%s)", in0);
201 return kNone_GrSLConstantVec;
202 } else {
203 outAppend->appendf("(vec4(%s) + vec4(%s))", in0, in1);
204 return kNone_GrSLConstantVec;
205 }
206}