alokp@chromium.org | 07620a5 | 2010-09-23 17:53:56 +0000 | [diff] [blame] | 1 | // |
shannonwoods@chromium.org | e429ab7 | 2013-05-30 00:12:52 +0000 | [diff] [blame] | 2 | // Copyright (c) 2002-2013 The ANGLE Project Authors. All rights reserved. |
alokp@chromium.org | 07620a5 | 2010-09-23 17:53:56 +0000 | [diff] [blame] | 3 | // Use of this source code is governed by a BSD-style license that can be |
| 4 | // found in the LICENSE file. |
| 5 | // |
| 6 | |
Geoff Lang | 1773282 | 2013-08-29 13:46:49 -0400 | [diff] [blame] | 7 | #include "compiler/translator/VariableInfo.h" |
Jamie Madill | aa72d78 | 2014-07-02 15:31:19 -0400 | [diff] [blame] | 8 | #include "compiler/translator/util.h" |
Jamie Madill | 183bde5 | 2014-07-02 15:31:19 -0400 | [diff] [blame] | 9 | #include "angle_gl.h" |
alokp@chromium.org | 07620a5 | 2010-09-23 17:53:56 +0000 | [diff] [blame] | 10 | |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 11 | namespace { |
| 12 | |
| 13 | TString arrayBrackets(int index) |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 14 | { |
| 15 | TStringStream stream; |
| 16 | stream << "[" << index << "]"; |
| 17 | return stream.str(); |
| 18 | } |
| 19 | |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 20 | template <typename VarT> |
| 21 | void getBuiltInVariableInfo(const TType &type, |
| 22 | const TString &name, |
| 23 | const TString &mappedName, |
| 24 | std::vector<VarT> &infoList); |
| 25 | |
| 26 | template <typename VarT> |
| 27 | void getUserDefinedVariableInfo(const TType &type, |
| 28 | const TString &name, |
| 29 | const TString &mappedName, |
| 30 | std::vector<VarT> &infoList, |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 31 | ShHashFunction64 hashFunction); |
| 32 | |
| 33 | // Returns info for an attribute, uniform, or varying. |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 34 | template <typename VarT> |
| 35 | void getVariableInfo(const TType &type, |
| 36 | const TString &name, |
| 37 | const TString &mappedName, |
| 38 | std::vector<VarT> &infoList, |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 39 | ShHashFunction64 hashFunction) |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 40 | { |
Dan Glastonbury | b66feb3 | 2014-05-14 15:38:55 +1000 | [diff] [blame] | 41 | if (type.getBasicType() == EbtStruct || type.isInterfaceBlock()) { |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 42 | if (type.isArray()) { |
| 43 | for (int i = 0; i < type.getArraySize(); ++i) { |
| 44 | TString lname = name + arrayBrackets(i); |
zmo@google.com | fd747b8 | 2011-04-23 01:30:07 +0000 | [diff] [blame] | 45 | TString lmappedName = mappedName + arrayBrackets(i); |
daniel@transgaming.com | 0aa3b5a | 2012-11-28 19:43:24 +0000 | [diff] [blame] | 46 | getUserDefinedVariableInfo(type, lname, lmappedName, infoList, hashFunction); |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 47 | } |
| 48 | } else { |
daniel@transgaming.com | 0aa3b5a | 2012-11-28 19:43:24 +0000 | [diff] [blame] | 49 | getUserDefinedVariableInfo(type, name, mappedName, infoList, hashFunction); |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 50 | } |
| 51 | } else { |
zmo@google.com | fd747b8 | 2011-04-23 01:30:07 +0000 | [diff] [blame] | 52 | getBuiltInVariableInfo(type, name, mappedName, infoList); |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 53 | } |
| 54 | } |
| 55 | |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 56 | template <class VarT> |
| 57 | void getBuiltInVariableInfo(const TType &type, |
| 58 | const TString &name, |
| 59 | const TString &mappedName, |
| 60 | std::vector<VarT> &infoList) |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 61 | { |
| 62 | ASSERT(type.getBasicType() != EbtStruct); |
| 63 | |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 64 | VarT varInfo; |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 65 | if (type.isArray()) { |
| 66 | varInfo.name = (name + "[0]").c_str(); |
zmo@google.com | fd747b8 | 2011-04-23 01:30:07 +0000 | [diff] [blame] | 67 | varInfo.mappedName = (mappedName + "[0]").c_str(); |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 68 | varInfo.arraySize = type.getArraySize(); |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 69 | } else { |
| 70 | varInfo.name = name.c_str(); |
zmo@google.com | fd747b8 | 2011-04-23 01:30:07 +0000 | [diff] [blame] | 71 | varInfo.mappedName = mappedName.c_str(); |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 72 | varInfo.arraySize = 0; |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 73 | } |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 74 | varInfo.precision = sh::GLVariablePrecision(type); |
Jamie Madill | aa72d78 | 2014-07-02 15:31:19 -0400 | [diff] [blame] | 75 | varInfo.type = sh::GLVariableType(type); |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 76 | infoList.push_back(varInfo); |
| 77 | } |
| 78 | |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 79 | template <class VarT> |
| 80 | void getUserDefinedVariableInfo(const TType &type, |
| 81 | const TString &name, |
| 82 | const TString &mappedName, |
| 83 | std::vector<VarT> &infoList, |
daniel@transgaming.com | 0aa3b5a | 2012-11-28 19:43:24 +0000 | [diff] [blame] | 84 | ShHashFunction64 hashFunction) |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 85 | { |
Jamie Madill | 98493dd | 2013-07-08 14:39:03 -0400 | [diff] [blame] | 86 | ASSERT(type.getBasicType() == EbtStruct || type.isInterfaceBlock()); |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 87 | |
Dan Glastonbury | b66feb3 | 2014-05-14 15:38:55 +1000 | [diff] [blame] | 88 | const TFieldList& fields = type.isInterfaceBlock() ? |
| 89 | type.getInterfaceBlock()->fields() : |
| 90 | type.getStruct()->fields(); |
Jamie Madill | 98493dd | 2013-07-08 14:39:03 -0400 | [diff] [blame] | 91 | for (size_t i = 0; i < fields.size(); ++i) { |
| 92 | const TType& fieldType = *(fields[i]->type()); |
| 93 | const TString& fieldName = fields[i]->name(); |
| 94 | getVariableInfo(fieldType, |
| 95 | name + "." + fieldName, |
| 96 | mappedName + "." + TIntermTraverser::hash(fieldName, hashFunction), |
daniel@transgaming.com | 0aa3b5a | 2012-11-28 19:43:24 +0000 | [diff] [blame] | 97 | infoList, |
| 98 | hashFunction); |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 99 | } |
| 100 | } |
| 101 | |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 102 | template <class VarT> |
| 103 | VarT* findVariable(const TType &type, |
| 104 | const TString &name, |
| 105 | std::vector<VarT> *infoList) |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 106 | { |
| 107 | // TODO(zmo): optimize this function. |
| 108 | TString myName = name; |
| 109 | if (type.isArray()) |
| 110 | myName += "[0]"; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 111 | for (size_t ii = 0; ii < infoList->size(); ++ii) |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 112 | { |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 113 | if ((*infoList)[ii].name.c_str() == myName) |
| 114 | return &((*infoList)[ii]); |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 115 | } |
| 116 | return NULL; |
| 117 | } |
| 118 | |
| 119 | } // namespace anonymous |
| 120 | |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 121 | CollectVariables::CollectVariables(std::vector<sh::Attribute> *attribs, |
| 122 | std::vector<sh::Uniform> *uniforms, |
| 123 | std::vector<sh::Varying> *varyings, |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 124 | ShHashFunction64 hashFunction) |
alokp@chromium.org | 07620a5 | 2010-09-23 17:53:56 +0000 | [diff] [blame] | 125 | : mAttribs(attribs), |
daniel@transgaming.com | 0aa3b5a | 2012-11-28 19:43:24 +0000 | [diff] [blame] | 126 | mUniforms(uniforms), |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 127 | mVaryings(varyings), |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 128 | mPointCoordAdded(false), |
| 129 | mFrontFacingAdded(false), |
| 130 | mFragCoordAdded(false), |
daniel@transgaming.com | 0aa3b5a | 2012-11-28 19:43:24 +0000 | [diff] [blame] | 131 | mHashFunction(hashFunction) |
alokp@chromium.org | 07620a5 | 2010-09-23 17:53:56 +0000 | [diff] [blame] | 132 | { |
| 133 | } |
| 134 | |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 135 | // We want to check whether a uniform/varying is statically used |
| 136 | // because we only count the used ones in packing computing. |
| 137 | // Also, gl_FragCoord, gl_PointCoord, and gl_FrontFacing count |
| 138 | // toward varying counting if they are statically used in a fragment |
| 139 | // shader. |
| 140 | void CollectVariables::visitSymbol(TIntermSymbol* symbol) |
alokp@chromium.org | 07620a5 | 2010-09-23 17:53:56 +0000 | [diff] [blame] | 141 | { |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 142 | ASSERT(symbol != NULL); |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 143 | sh::ShaderVariable *var = NULL; |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 144 | switch (symbol->getQualifier()) |
| 145 | { |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 146 | case EvqVaryingOut: |
| 147 | case EvqInvariantVaryingOut: |
| 148 | case EvqVaryingIn: |
| 149 | case EvqInvariantVaryingIn: |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 150 | var = findVariable(symbol->getType(), symbol->getSymbol(), mVaryings); |
| 151 | break; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 152 | case EvqUniform: |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 153 | var = findVariable(symbol->getType(), symbol->getSymbol(), mUniforms); |
| 154 | break; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 155 | case EvqFragCoord: |
| 156 | if (!mFragCoordAdded) |
| 157 | { |
| 158 | sh::Varying info; |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 159 | info.name = "gl_FragCoord"; |
| 160 | info.mappedName = "gl_FragCoord"; |
Jamie Madill | 183bde5 | 2014-07-02 15:31:19 -0400 | [diff] [blame] | 161 | info.type = GL_FLOAT_VEC4; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 162 | info.arraySize = 0; |
| 163 | info.precision = GL_MEDIUM_FLOAT; // Use mediump as it doesn't really matter. |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 164 | info.staticUse = true; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 165 | mVaryings->push_back(info); |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 166 | mFragCoordAdded = true; |
| 167 | } |
| 168 | return; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 169 | case EvqFrontFacing: |
| 170 | if (!mFrontFacingAdded) |
| 171 | { |
| 172 | sh::Varying info; |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 173 | info.name = "gl_FrontFacing"; |
| 174 | info.mappedName = "gl_FrontFacing"; |
Jamie Madill | 183bde5 | 2014-07-02 15:31:19 -0400 | [diff] [blame] | 175 | info.type = GL_BOOL; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 176 | info.arraySize = 0; |
| 177 | info.precision = GL_NONE; |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 178 | info.staticUse = true; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 179 | mVaryings->push_back(info); |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 180 | mFrontFacingAdded = true; |
| 181 | } |
| 182 | return; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 183 | case EvqPointCoord: |
| 184 | if (!mPointCoordAdded) |
| 185 | { |
| 186 | sh::Varying info; |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 187 | info.name = "gl_PointCoord"; |
| 188 | info.mappedName = "gl_PointCoord"; |
Jamie Madill | 183bde5 | 2014-07-02 15:31:19 -0400 | [diff] [blame] | 189 | info.type = GL_FLOAT_VEC2; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 190 | info.arraySize = 0; |
| 191 | info.precision = GL_MEDIUM_FLOAT; // Use mediump as it doesn't really matter. |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 192 | info.staticUse = true; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 193 | mVaryings->push_back(info); |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 194 | mPointCoordAdded = true; |
| 195 | } |
| 196 | return; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 197 | default: |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 198 | break; |
| 199 | } |
| 200 | if (var) |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 201 | { |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 202 | var->staticUse = true; |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 203 | } |
| 204 | } |
| 205 | |
| 206 | template <typename VarT> |
| 207 | void CollectVariables::visitInfoList(const TIntermSequence& sequence, std::vector<VarT> *infoList) const |
| 208 | { |
| 209 | for (size_t seqIndex = 0; seqIndex < sequence.size(); seqIndex++) |
| 210 | { |
| 211 | const TIntermSymbol* variable = sequence[seqIndex]->getAsSymbolNode(); |
| 212 | // The only case in which the sequence will not contain a |
| 213 | // TIntermSymbol node is initialization. It will contain a |
| 214 | // TInterBinary node in that case. Since attributes, uniforms, |
| 215 | // and varyings cannot be initialized in a shader, we must have |
| 216 | // only TIntermSymbol nodes in the sequence. |
| 217 | ASSERT(variable != NULL); |
| 218 | TString processedSymbol; |
| 219 | if (mHashFunction == NULL) |
| 220 | processedSymbol = variable->getSymbol(); |
| 221 | else |
| 222 | processedSymbol = TIntermTraverser::hash(variable->getSymbol(), mHashFunction); |
| 223 | getVariableInfo(variable->getType(), |
| 224 | variable->getSymbol(), |
| 225 | processedSymbol, |
| 226 | *infoList, |
| 227 | mHashFunction); |
| 228 | } |
alokp@chromium.org | 07620a5 | 2010-09-23 17:53:56 +0000 | [diff] [blame] | 229 | } |
| 230 | |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 231 | bool CollectVariables::visitAggregate(Visit, TIntermAggregate* node) |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 232 | { |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 233 | bool visitChildren = true; |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 234 | |
| 235 | switch (node->getOp()) |
| 236 | { |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 237 | case EOpDeclaration: { |
| 238 | const TIntermSequence& sequence = node->getSequence(); |
alokp@chromium.org | 10e6e9e | 2010-09-27 21:03:45 +0000 | [diff] [blame] | 239 | TQualifier qualifier = sequence.front()->getAsTyped()->getQualifier(); |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 240 | if (qualifier == EvqAttribute || qualifier == EvqVertexIn || qualifier == EvqUniform || |
| 241 | qualifier == EvqVaryingIn || qualifier == EvqVaryingOut || |
| 242 | qualifier == EvqInvariantVaryingIn || qualifier == EvqInvariantVaryingOut) |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 243 | { |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 244 | switch (qualifier) |
| 245 | { |
| 246 | case EvqAttribute: |
| 247 | case EvqVertexIn: |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 248 | visitInfoList(sequence, mAttribs); |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 249 | break; |
| 250 | case EvqUniform: |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 251 | visitInfoList(sequence, mUniforms); |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 252 | break; |
| 253 | default: |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 254 | visitInfoList(sequence, mVaryings); |
Zhenyao Mo | 74da9f2 | 2013-09-23 14:57:01 -0400 | [diff] [blame] | 255 | break; |
| 256 | } |
| 257 | |
Jamie Madill | a718c1e | 2014-07-02 15:31:22 -0400 | [diff] [blame^] | 258 | if (!sequence.empty()) |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 259 | { |
Zhenyao Mo | d2d340b | 2013-09-23 14:57:05 -0400 | [diff] [blame] | 260 | visitChildren = false; |
alokp@chromium.org | ee76f6a | 2010-09-27 19:28:55 +0000 | [diff] [blame] | 261 | } |
| 262 | } |
| 263 | break; |
| 264 | } |
| 265 | default: break; |
| 266 | } |
| 267 | |
| 268 | return visitChildren; |
| 269 | } |