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