/* * Copyright 2011 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "GrGLSL.h" #include "GrGLShaderVar.h" #include "SkString.h" GrGLSLGeneration GrGetGLSLGeneration(GrGLBinding binding, const GrGLInterface* gl) { GrGLSLVersion ver = GrGLGetGLSLVersion(gl); switch (binding) { case kDesktop_GrGLBinding: GrAssert(ver >= GR_GLSL_VER(1,10)); if (ver >= GR_GLSL_VER(1,50)) { return k150_GrGLSLGeneration; } else if (ver >= GR_GLSL_VER(1,40)) { return k140_GrGLSLGeneration; } else if (ver >= GR_GLSL_VER(1,30)) { return k130_GrGLSLGeneration; } else { return k110_GrGLSLGeneration; } case kES2_GrGLBinding: // version 1.00 of ES GLSL based on ver 1.20 of desktop GLSL GrAssert(ver >= GR_GL_VER(1,00)); return k110_GrGLSLGeneration; default: GrCrash("Unknown GL Binding"); return k110_GrGLSLGeneration; // suppress warning } } const char* GrGetGLSLVersionDecl(GrGLBinding binding, GrGLSLGeneration gen) { switch (gen) { case k110_GrGLSLGeneration: if (kES2_GrGLBinding == binding) { // ES2s shader language is based on version 1.20 but is version // 1.00 of the ES language. return "#version 100\n"; } else { GrAssert(kDesktop_GrGLBinding == binding); return "#version 110\n"; } case k130_GrGLSLGeneration: GrAssert(kDesktop_GrGLBinding == binding); return "#version 130\n"; case k140_GrGLSLGeneration: GrAssert(kDesktop_GrGLBinding == binding); return "#version 140\n"; case k150_GrGLSLGeneration: GrAssert(kDesktop_GrGLBinding == binding); return "#version 150\n"; default: GrCrash("Unknown GL version."); return ""; // suppress warning } } bool GrGLSLSetupFSColorOuput(GrGLSLGeneration gen, const char* nameIfDeclared, GrGLShaderVar* var) { bool declaredOutput = k110_GrGLSLGeneration != gen; var->set(kVec4f_GrSLType, GrGLShaderVar::kOut_TypeModifier, declaredOutput ? nameIfDeclared : "gl_FragColor"); return declaredOutput; } GrSLType GrSLFloatVectorType (int count) { GR_STATIC_ASSERT(kFloat_GrSLType == 1); GR_STATIC_ASSERT(kVec2f_GrSLType == 2); GR_STATIC_ASSERT(kVec3f_GrSLType == 3); GR_STATIC_ASSERT(kVec4f_GrSLType == 4); GrAssert(count > 0 && count <= 4); return (GrSLType)(count); } const char* GrGLSLVectorHomogCoord(int count) { static const char* HOMOGS[] = {"ERROR", "", ".y", ".z", ".w"}; GrAssert(count >= 1 && count < (int)GR_ARRAY_COUNT(HOMOGS)); return HOMOGS[count]; } const char* GrGLSLVectorHomogCoord(GrSLType type) { return GrGLSLVectorHomogCoord(GrSLTypeToVecLength(type)); } const char* GrGLSLVectorNonhomogCoords(int count) { static const char* NONHOMOGS[] = {"ERROR", "", ".x", ".xy", ".xyz"}; GrAssert(count >= 1 && count < (int)GR_ARRAY_COUNT(NONHOMOGS)); return NONHOMOGS[count]; } const char* GrGLSLVectorNonhomogCoords(GrSLType type) { return GrGLSLVectorNonhomogCoords(GrSLTypeToVecLength(type)); } GrSLConstantVec GrGLSLModulate4f(SkString* outAppend, const char* in0, const char* in1, GrSLConstantVec default0, GrSLConstantVec default1) { GrAssert(NULL != outAppend); bool has0 = NULL != in0 && '\0' != *in0; bool has1 = NULL != in1 && '\0' != *in1; GrAssert(has0 || kNone_GrSLConstantVec != default0); GrAssert(has1 || kNone_GrSLConstantVec != default1); if (!has0 && !has1) { GrAssert(kZeros_GrSLConstantVec == default0 || kOnes_GrSLConstantVec == default0); GrAssert(kZeros_GrSLConstantVec == default1 || kOnes_GrSLConstantVec == default1); if (kZeros_GrSLConstantVec == default0 || kZeros_GrSLConstantVec == default1) { outAppend->append(GrGLSLZerosVecf(4)); return kZeros_GrSLConstantVec; } else { // both inputs are ones vectors outAppend->append(GrGLSLOnesVecf(4)); return kOnes_GrSLConstantVec; } } else if (!has0) { GrAssert(kZeros_GrSLConstantVec == default0 || kOnes_GrSLConstantVec == default0); if (kZeros_GrSLConstantVec == default0) { outAppend->append(GrGLSLZerosVecf(4)); return kZeros_GrSLConstantVec; } else { outAppend->appendf("vec4(%s)", in1); return kNone_GrSLConstantVec; } } else if (!has1) { GrAssert(kZeros_GrSLConstantVec == default1 || kOnes_GrSLConstantVec == default1); if (kZeros_GrSLConstantVec == default1) { outAppend->append(GrGLSLZerosVecf(4)); return kZeros_GrSLConstantVec; } else { outAppend->appendf("vec4(%s)", in0); return kNone_GrSLConstantVec; } } else { outAppend->appendf("vec4(%s * %s)", in0, in1); return kNone_GrSLConstantVec; } } namespace { void append_tabs(SkString* outAppend, int tabCnt) { static const char kTabs[] = "\t\t\t\t\t\t\t\t"; while (tabCnt) { int cnt = GrMin((int)GR_ARRAY_COUNT(kTabs), tabCnt); outAppend->append(kTabs, cnt); tabCnt -= cnt; } } } GrSLConstantVec GrGLSLMulVarBy4f(SkString* outAppend, int tabCnt, const char* vec4VarName, const char* mulFactor, GrSLConstantVec mulFactorDefault) { bool haveFactor = NULL != mulFactor && '\0' != *mulFactor; GrAssert(NULL != outAppend); GrAssert(NULL != vec4VarName); GrAssert(kNone_GrSLConstantVec != mulFactorDefault || haveFactor); if (!haveFactor) { if (kOnes_GrSLConstantVec == mulFactorDefault) { return kNone_GrSLConstantVec; } else { GrAssert(kZeros_GrSLConstantVec == mulFactorDefault); append_tabs(outAppend, tabCnt); outAppend->appendf("%s = vec4(0, 0, 0, 0);\n", vec4VarName); return kZeros_GrSLConstantVec; } } append_tabs(outAppend, tabCnt); outAppend->appendf("%s *= %s;\n", vec4VarName, mulFactor); return kNone_GrSLConstantVec; } GrSLConstantVec GrGLSLAdd4f(SkString* outAppend, const char* in0, const char* in1, GrSLConstantVec default0, GrSLConstantVec default1) { GrAssert(NULL != outAppend); bool has0 = NULL != in0 && '\0' != *in0; bool has1 = NULL != in1 && '\0' != *in1; if (!has0 && !has1) { GrAssert(kZeros_GrSLConstantVec == default0); GrAssert(kZeros_GrSLConstantVec == default1); outAppend->append(GrGLSLZerosVecf(4)); return kZeros_GrSLConstantVec; } else if (!has0) { GrAssert(kZeros_GrSLConstantVec == default0); outAppend->appendf("vec4(%s)", in1); return kNone_GrSLConstantVec; } else if (!has1) { GrAssert(kZeros_GrSLConstantVec == default1); outAppend->appendf("vec4(%s)", in0); return kNone_GrSLConstantVec; } else { outAppend->appendf("(vec4(%s) + vec4(%s))", in0, in1); return kNone_GrSLConstantVec; } }