blob: ff6b46380337bc5f8de33f0eff86586866e43d73 [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
bsalomon@google.com42eff162013-04-02 12:50:49 +000012GrGLSLGeneration GrGetGLSLGeneration(GrGLBinding binding, const GrGLInterface* gl) {
tomhudson@google.com086e5352011-12-08 14:44:10 +000013 GrGLSLVersion ver = GrGLGetGLSLVersion(gl);
14 switch (binding) {
15 case kDesktop_GrGLBinding:
16 GrAssert(ver >= GR_GLSL_VER(1,10));
17 if (ver >= GR_GLSL_VER(1,50)) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000018 return k150_GrGLSLGeneration;
bsalomon@google.com281c7262012-10-23 14:31:30 +000019 } else if (ver >= GR_GLSL_VER(1,40)) {
20 return k140_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000021 } else if (ver >= GR_GLSL_VER(1,30)) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000022 return k130_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000023 } else {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000024 return k110_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000025 }
26 case kES2_GrGLBinding:
27 // version 1.00 of ES GLSL based on ver 1.20 of desktop GLSL
28 GrAssert(ver >= GR_GL_VER(1,00));
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000029 return k110_GrGLSLGeneration;
tomhudson@google.com086e5352011-12-08 14:44:10 +000030 default:
31 GrCrash("Unknown GL Binding");
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000032 return k110_GrGLSLGeneration; // suppress warning
tomhudson@google.com086e5352011-12-08 14:44:10 +000033 }
34}
35
bsalomon@google.com42eff162013-04-02 12:50:49 +000036const char* GrGetGLSLVersionDecl(GrGLBinding binding, GrGLSLGeneration gen) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000037 switch (gen) {
38 case k110_GrGLSLGeneration:
39 if (kES2_GrGLBinding == binding) {
40 // ES2s shader language is based on version 1.20 but is version
41 // 1.00 of the ES language.
42 return "#version 100\n";
43 } else {
44 GrAssert(kDesktop_GrGLBinding == binding);
45 return "#version 110\n";
46 }
47 case k130_GrGLSLGeneration:
48 GrAssert(kDesktop_GrGLBinding == binding);
49 return "#version 130\n";
bsalomon@google.com281c7262012-10-23 14:31:30 +000050 case k140_GrGLSLGeneration:
51 GrAssert(kDesktop_GrGLBinding == binding);
52 return "#version 140\n";
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000053 case k150_GrGLSLGeneration:
54 GrAssert(kDesktop_GrGLBinding == binding);
55 return "#version 150\n";
56 default:
57 GrCrash("Unknown GL version.");
58 return ""; // suppress warning
59 }
60}
61
bsalomon@google.com42eff162013-04-02 12:50:49 +000062bool GrGLSLSetupFSColorOuput(GrGLSLGeneration gen, const char* nameIfDeclared, GrGLShaderVar* var) {
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000063 bool declaredOutput = k110_GrGLSLGeneration != gen;
tomhudson@google.com168e6342012-04-18 17:49:20 +000064 var->set(kVec4f_GrSLType,
bsalomon@google.come55fd0f2012-02-10 15:56:06 +000065 GrGLShaderVar::kOut_TypeModifier,
66 declaredOutput ? nameIfDeclared : "gl_FragColor");
67 return declaredOutput;
68}
tomhudson@google.com168e6342012-04-18 17:49:20 +000069
tomhudson@google.com168e6342012-04-18 17:49:20 +000070const char* GrGLSLVectorHomogCoord(int count) {
71 static const char* HOMOGS[] = {"ERROR", "", ".y", ".z", ".w"};
72 GrAssert(count >= 1 && count < (int)GR_ARRAY_COUNT(HOMOGS));
73 return HOMOGS[count];
74}
75
bsalomon@google.com34bcb9f2012-08-28 18:20:18 +000076const char* GrGLSLVectorHomogCoord(GrSLType type) {
77 return GrGLSLVectorHomogCoord(GrSLTypeToVecLength(type));
78}
79
tomhudson@google.com168e6342012-04-18 17:49:20 +000080const char* GrGLSLVectorNonhomogCoords(int count) {
81 static const char* NONHOMOGS[] = {"ERROR", "", ".x", ".xy", ".xyz"};
82 GrAssert(count >= 1 && count < (int)GR_ARRAY_COUNT(NONHOMOGS));
83 return NONHOMOGS[count];
84}
85
robertphillips@google.com5b5bba32012-09-24 14:20:00 +000086const char* GrGLSLVectorNonhomogCoords(GrSLType type) {
bsalomon@google.com34bcb9f2012-08-28 18:20:18 +000087 return GrGLSLVectorNonhomogCoords(GrSLTypeToVecLength(type));
88}
bsalomon@google.com4af0af62012-08-29 12:59:57 +000089
bsalomon@google.com868a8e72012-08-30 19:11:34 +000090namespace {
bsalomon@google.com018f1792013-04-18 19:36:09 +000091 void append_tabs(SkString* outAppend, int tabCnt) {
92 static const char kTabs[] = "\t\t\t\t\t\t\t\t";
93 while (tabCnt) {
94 int cnt = GrMin((int)GR_ARRAY_COUNT(kTabs), tabCnt);
95 outAppend->append(kTabs, cnt);
96 tabCnt -= cnt;
97 }
bsalomon@google.com868a8e72012-08-30 19:11:34 +000098 }
99}
bsalomon@google.com868a8e72012-08-30 19:11:34 +0000100
101GrSLConstantVec GrGLSLMulVarBy4f(SkString* outAppend,
102 int tabCnt,
103 const char* vec4VarName,
104 const char* mulFactor,
105 GrSLConstantVec mulFactorDefault) {
106 bool haveFactor = NULL != mulFactor && '\0' != *mulFactor;
bsalomon@google.com018f1792013-04-18 19:36:09 +0000107
bsalomon@google.com868a8e72012-08-30 19:11:34 +0000108 GrAssert(NULL != outAppend);
109 GrAssert(NULL != vec4VarName);
110 GrAssert(kNone_GrSLConstantVec != mulFactorDefault || haveFactor);
bsalomon@google.com018f1792013-04-18 19:36:09 +0000111
bsalomon@google.com868a8e72012-08-30 19:11:34 +0000112 if (!haveFactor) {
113 if (kOnes_GrSLConstantVec == mulFactorDefault) {
114 return kNone_GrSLConstantVec;
115 } else {
116 GrAssert(kZeros_GrSLConstantVec == mulFactorDefault);
117 append_tabs(outAppend, tabCnt);
118 outAppend->appendf("%s = vec4(0, 0, 0, 0);\n", vec4VarName);
119 return kZeros_GrSLConstantVec;
120 }
121 }
122 append_tabs(outAppend, tabCnt);
123 outAppend->appendf("%s *= %s;\n", vec4VarName, mulFactor);
124 return kNone_GrSLConstantVec;
125}
126
bsalomon@google.com018f1792013-04-18 19:36:09 +0000127GrSLConstantVec GrGLSLGetComponent4f(SkString* outAppend,
128 const char* expr,
129 GrColorComponentFlags component,
130 GrSLConstantVec defaultExpr,
131 bool omitIfConst) {
132 if (NULL == expr || '\0' == *expr) {
133 GrAssert(defaultExpr != kNone_GrSLConstantVec);
134 if (!omitIfConst) {
135 if (kOnes_GrSLConstantVec == defaultExpr) {
136 outAppend->append("1.0");
137 } else {
138 GrAssert(kZeros_GrSLConstantVec == defaultExpr);
139 outAppend->append("0.0");
140 }
141 }
142 return defaultExpr;
bsalomon@google.com4af0af62012-08-29 12:59:57 +0000143 } else {
bsalomon@google.com018f1792013-04-18 19:36:09 +0000144 outAppend->appendf("(%s).%c", expr, GrColorComponentFlagToChar(component));
bsalomon@google.com4af0af62012-08-29 12:59:57 +0000145 return kNone_GrSLConstantVec;
146 }
147}