| // Copyright 2016 The SwiftShader Authors. All Rights Reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #ifndef _TYPES_INCLUDED |
| #define _TYPES_INCLUDED |
| |
| #include "BaseTypes.h" |
| #include "Common.h" |
| #include "debug.h" |
| |
| #include <algorithm> |
| |
| class TType; |
| struct TPublicType; |
| |
| class TField |
| { |
| public: |
| POOL_ALLOCATOR_NEW_DELETE() |
| TField(TType *type, TString *name, const TSourceLoc &line) |
| : mType(type), |
| mName(name), |
| mLine(line) |
| { |
| } |
| |
| // TODO(alokp): We should only return const type. |
| // Fix it by tweaking grammar. |
| TType *type() |
| { |
| return mType; |
| } |
| const TType *type() const |
| { |
| return mType; |
| } |
| |
| const TString &name() const |
| { |
| return *mName; |
| } |
| const TSourceLoc &line() const |
| { |
| return mLine; |
| } |
| |
| private: |
| TType *mType; |
| TString *mName; |
| TSourceLoc mLine; |
| }; |
| |
| typedef TVector<TField *> TFieldList; |
| inline TFieldList *NewPoolTFieldList() |
| { |
| void *memory = GetGlobalPoolAllocator()->allocate(sizeof(TFieldList)); |
| return new(memory)TFieldList; |
| } |
| |
| class TFieldListCollection |
| { |
| public: |
| virtual ~TFieldListCollection() { } |
| const TString &name() const |
| { |
| return *mName; |
| } |
| const TFieldList &fields() const |
| { |
| return *mFields; |
| } |
| |
| const TString &mangledName() const |
| { |
| if(mMangledName.empty()) |
| mMangledName = buildMangledName(); |
| return mMangledName; |
| } |
| size_t objectSize() const |
| { |
| if(mObjectSize == 0) |
| mObjectSize = calculateObjectSize(); |
| return mObjectSize; |
| } |
| |
| protected: |
| TFieldListCollection(const TString *name, TFieldList *fields) |
| : mName(name), |
| mFields(fields), |
| mObjectSize(0) |
| { |
| } |
| TString buildMangledName() const; |
| size_t calculateObjectSize() const; |
| virtual TString mangledNamePrefix() const = 0; |
| |
| const TString *mName; |
| TFieldList *mFields; |
| |
| mutable TString mMangledName; |
| mutable size_t mObjectSize; |
| }; |
| |
| // May also represent interface blocks |
| class TStructure : public TFieldListCollection |
| { |
| public: |
| POOL_ALLOCATOR_NEW_DELETE() |
| TStructure(const TString *name, TFieldList *fields) |
| : TFieldListCollection(name, fields), |
| mDeepestNesting(0), |
| mUniqueId(0), |
| mAtGlobalScope(false) |
| { |
| } |
| |
| int deepestNesting() const |
| { |
| if(mDeepestNesting == 0) |
| mDeepestNesting = calculateDeepestNesting(); |
| return mDeepestNesting; |
| } |
| bool containsArrays() const; |
| bool containsType(TBasicType type) const; |
| bool containsSamplers() const; |
| |
| bool equals(const TStructure &other) const; |
| |
| void setMatrixPackingIfUnspecified(TLayoutMatrixPacking matrixPacking); |
| |
| void setUniqueId(int uniqueId) |
| { |
| mUniqueId = uniqueId; |
| } |
| |
| int uniqueId() const |
| { |
| ASSERT(mUniqueId != 0); |
| return mUniqueId; |
| } |
| |
| void setAtGlobalScope(bool atGlobalScope) |
| { |
| mAtGlobalScope = atGlobalScope; |
| } |
| |
| bool atGlobalScope() const |
| { |
| return mAtGlobalScope; |
| } |
| |
| private: |
| // TODO(zmo): Find a way to get rid of the const_cast in function |
| // setName(). At the moment keep this function private so only |
| // friend class RegenerateStructNames may call it. |
| friend class RegenerateStructNames; |
| void setName(const TString &name) |
| { |
| TString *mutableName = const_cast<TString *>(mName); |
| *mutableName = name; |
| } |
| |
| virtual TString mangledNamePrefix() const |
| { |
| return "struct-"; |
| } |
| int calculateDeepestNesting() const; |
| |
| mutable int mDeepestNesting; |
| int mUniqueId; |
| bool mAtGlobalScope; |
| }; |
| |
| class TInterfaceBlock : public TFieldListCollection |
| { |
| public: |
| POOL_ALLOCATOR_NEW_DELETE() |
| TInterfaceBlock(const TString *name, TFieldList *fields, const TString *instanceName, |
| int arraySize, const TLayoutQualifier &layoutQualifier) |
| : TFieldListCollection(name, fields), |
| mInstanceName(instanceName), |
| mArraySize(arraySize), |
| mBlockStorage(layoutQualifier.blockStorage), |
| mMatrixPacking(layoutQualifier.matrixPacking) |
| { |
| } |
| |
| const TString &instanceName() const |
| { |
| return *mInstanceName; |
| } |
| bool hasInstanceName() const |
| { |
| return mInstanceName != nullptr; |
| } |
| bool isArray() const |
| { |
| return mArraySize > 0; |
| } |
| int arraySize() const |
| { |
| return mArraySize; |
| } |
| TLayoutBlockStorage blockStorage() const |
| { |
| return mBlockStorage; |
| } |
| TLayoutMatrixPacking matrixPacking() const |
| { |
| return mMatrixPacking; |
| } |
| |
| private: |
| virtual TString mangledNamePrefix() const |
| { |
| return "iblock-"; |
| } |
| |
| const TString *mInstanceName; // for interface block instance names |
| int mArraySize; // 0 if not an array |
| TLayoutBlockStorage mBlockStorage; |
| TLayoutMatrixPacking mMatrixPacking; |
| }; |
| |
| // |
| // Base class for things that have a type. |
| // |
| class TType |
| { |
| public: |
| POOL_ALLOCATOR_NEW_DELETE() |
| |
| TType(TBasicType t, int s0 = 1, int s1 = 1) : |
| type(t), precision(EbpUndefined), qualifier(EvqGlobal), |
| primarySize(s0), secondarySize(s1), array(false), arraySize(0), maxArraySize(0), arrayInformationType(0), interfaceBlock(0), layoutQualifier(TLayoutQualifier::create()), |
| structure(0), mangled(0) |
| { |
| } |
| |
| TType(TBasicType t, TPrecision p, TQualifier q = EvqTemporary, int s0 = 1, int s1 = 1, bool a = false) : |
| type(t), precision(p), qualifier(q), |
| primarySize(s0), secondarySize(s1), array(a), arraySize(0), maxArraySize(0), arrayInformationType(0), interfaceBlock(0), layoutQualifier(TLayoutQualifier::create()), |
| structure(0), mangled(0) |
| { |
| } |
| |
| TType(TStructure* userDef, TPrecision p = EbpUndefined) : |
| type(EbtStruct), precision(p), qualifier(EvqTemporary), |
| primarySize(1), secondarySize(1), array(false), arraySize(0), maxArraySize(0), arrayInformationType(0), interfaceBlock(0), layoutQualifier(TLayoutQualifier::create()), |
| structure(userDef), mangled(0) |
| { |
| } |
| |
| TType(TInterfaceBlock *interfaceBlockIn, TQualifier qualifierIn, |
| TLayoutQualifier layoutQualifierIn, int arraySizeIn) |
| : type(EbtInterfaceBlock), precision(EbpUndefined), qualifier(qualifierIn), |
| primarySize(1), secondarySize(1), array(arraySizeIn > 0), arraySize(arraySizeIn), maxArraySize(0), arrayInformationType(0), |
| interfaceBlock(interfaceBlockIn), layoutQualifier(layoutQualifierIn), structure(0), mangled(0) |
| { |
| } |
| |
| explicit TType(const TPublicType &p); |
| |
| TBasicType getBasicType() const { return type; } |
| void setBasicType(TBasicType t) { type = t; } |
| |
| TPrecision getPrecision() const { return precision; } |
| void setPrecision(TPrecision p) { precision = p; } |
| |
| TQualifier getQualifier() const { return qualifier; } |
| void setQualifier(TQualifier q) { qualifier = q; } |
| |
| TLayoutQualifier getLayoutQualifier() const { return layoutQualifier; } |
| void setLayoutQualifier(TLayoutQualifier lq) { layoutQualifier = lq; } |
| |
| void setMatrixPackingIfUnspecified(TLayoutMatrixPacking matrixPacking) |
| { |
| if(isStruct()) |
| { |
| // If the structure's matrix packing is specified, it overrules the block's matrix packing |
| structure->setMatrixPackingIfUnspecified((layoutQualifier.matrixPacking == EmpUnspecified) ? |
| matrixPacking : layoutQualifier.matrixPacking); |
| } |
| // If the member's matrix packing is specified, it overrules any higher level matrix packing |
| if(layoutQualifier.matrixPacking == EmpUnspecified) |
| { |
| layoutQualifier.matrixPacking = matrixPacking; |
| } |
| } |
| |
| // One-dimensional size of single instance type |
| int getNominalSize() const { return primarySize; } |
| void setNominalSize(int s) { primarySize = s; } |
| // Full size of single instance of type |
| size_t getObjectSize() const |
| { |
| if(isArray()) |
| { |
| return getElementSize() * std::max(getArraySize(), getMaxArraySize()); |
| } |
| else |
| { |
| return getElementSize(); |
| } |
| } |
| |
| size_t getElementSize() const |
| { |
| if(getBasicType() == EbtStruct) |
| { |
| return getStructSize(); |
| } |
| else if(isInterfaceBlock()) |
| { |
| return interfaceBlock->objectSize(); |
| } |
| else if(isMatrix()) |
| { |
| return primarySize * secondarySize; |
| } |
| else // Vector or scalar |
| { |
| return primarySize; |
| } |
| } |
| |
| int samplerRegisterCount() const |
| { |
| if(structure) |
| { |
| int registerCount = 0; |
| |
| const TFieldList& fields = isInterfaceBlock() ? interfaceBlock->fields() : structure->fields(); |
| for(size_t i = 0; i < fields.size(); i++) |
| { |
| registerCount += fields[i]->type()->totalSamplerRegisterCount(); |
| } |
| |
| return registerCount; |
| } |
| |
| return IsSampler(getBasicType()) ? 1 : 0; |
| } |
| |
| int elementRegisterCount() const |
| { |
| if(structure || isInterfaceBlock()) |
| { |
| int registerCount = 0; |
| |
| const TFieldList& fields = isInterfaceBlock() ? interfaceBlock->fields() : structure->fields(); |
| for(size_t i = 0; i < fields.size(); i++) |
| { |
| registerCount += fields[i]->type()->totalRegisterCount(); |
| } |
| |
| return registerCount; |
| } |
| else if(isMatrix()) |
| { |
| return getNominalSize(); |
| } |
| else |
| { |
| return 1; |
| } |
| } |
| |
| int blockRegisterCount() const |
| { |
| // If this TType object is a block member, return the register count of the parent block |
| // Otherwise, return the register count of the current TType object |
| if(interfaceBlock && !isInterfaceBlock()) |
| { |
| int registerCount = 0; |
| const TFieldList& fieldList = interfaceBlock->fields(); |
| for(size_t i = 0; i < fieldList.size(); i++) |
| { |
| const TType &fieldType = *(fieldList[i]->type()); |
| registerCount += fieldType.totalRegisterCount(); |
| } |
| return registerCount; |
| } |
| return totalRegisterCount(); |
| } |
| |
| int totalSamplerRegisterCount() const |
| { |
| if(array) |
| { |
| return arraySize * samplerRegisterCount(); |
| } |
| else |
| { |
| return samplerRegisterCount(); |
| } |
| } |
| |
| int totalRegisterCount() const |
| { |
| if(array) |
| { |
| return arraySize * elementRegisterCount(); |
| } |
| else |
| { |
| return elementRegisterCount(); |
| } |
| } |
| |
| int registerSize() const |
| { |
| return isMatrix() ? secondarySize : primarySize; |
| } |
| |
| bool isMatrix() const { return secondarySize > 1; } |
| void setSecondarySize(int s1) { secondarySize = s1; } |
| int getSecondarySize() const { return secondarySize; } |
| |
| bool isArray() const { return array ? true : false; } |
| bool isUnsizedArray() const { return array && arraySize == 0; } |
| int getArraySize() const { return arraySize; } |
| void setArraySize(int s) { array = true; arraySize = s; } |
| int getMaxArraySize () const { return maxArraySize; } |
| void setMaxArraySize (int s) { maxArraySize = s; } |
| void clearArrayness() { array = false; arraySize = 0; maxArraySize = 0; } |
| void setArrayInformationType(TType* t) { arrayInformationType = t; } |
| TType* getArrayInformationType() const { return arrayInformationType; } |
| |
| TInterfaceBlock *getInterfaceBlock() const { return interfaceBlock; } |
| void setInterfaceBlock(TInterfaceBlock *interfaceBlockIn) { interfaceBlock = interfaceBlockIn; } |
| bool isInterfaceBlock() const { return type == EbtInterfaceBlock; } |
| TInterfaceBlock *getAsInterfaceBlock() const { return isInterfaceBlock() ? getInterfaceBlock() : nullptr; } |
| |
| bool isVector() const { return primarySize > 1 && !isMatrix(); } |
| bool isScalar() const { return primarySize == 1 && !isMatrix() && !structure && !isInterfaceBlock() && !IsSampler(getBasicType()); } |
| bool isRegister() const { return !isMatrix() && !structure && !array && !isInterfaceBlock(); } // Fits in a 4-element register |
| bool isStruct() const { return structure != 0; } |
| bool isScalarInt() const { return isScalar() && IsInteger(type); } |
| |
| TStructure* getStruct() const { return structure; } |
| void setStruct(TStructure* s) { structure = s; } |
| |
| TString& getMangledName() { |
| if (!mangled) { |
| mangled = NewPoolTString(""); |
| buildMangledName(*mangled); |
| *mangled += ';' ; |
| } |
| |
| return *mangled; |
| } |
| |
| bool sameElementType(const TType& right) const { |
| return type == right.type && |
| primarySize == right.primarySize && |
| secondarySize == right.secondarySize && |
| structure == right.structure; |
| } |
| bool operator==(const TType& right) const { |
| return type == right.type && |
| primarySize == right.primarySize && |
| secondarySize == right.secondarySize && |
| array == right.array && (!array || arraySize == right.arraySize) && |
| structure == right.structure; |
| // don't check the qualifier, it's not ever what's being sought after |
| } |
| bool operator!=(const TType& right) const { |
| return !operator==(right); |
| } |
| bool operator<(const TType& right) const { |
| if (type != right.type) return type < right.type; |
| if(primarySize != right.primarySize) return (primarySize * secondarySize) < (right.primarySize * right.secondarySize); |
| if(secondarySize != right.secondarySize) return secondarySize < right.secondarySize; |
| if (array != right.array) return array < right.array; |
| if (arraySize != right.arraySize) return arraySize < right.arraySize; |
| if (structure != right.structure) return structure < right.structure; |
| |
| return false; |
| } |
| |
| const char* getBasicString() const { return ::getBasicString(type); } |
| const char* getPrecisionString() const { return ::getPrecisionString(precision); } |
| const char* getQualifierString() const { return ::getQualifierString(qualifier); } |
| TString getCompleteString() const; |
| |
| // If this type is a struct, returns the deepest struct nesting of |
| // any field in the struct. For example: |
| // struct nesting1 { |
| // vec4 position; |
| // }; |
| // struct nesting2 { |
| // nesting1 field1; |
| // vec4 field2; |
| // }; |
| // For type "nesting2", this method would return 2 -- the number |
| // of structures through which indirection must occur to reach the |
| // deepest field (nesting2.field1.position). |
| int getDeepestStructNesting() const |
| { |
| return structure ? structure->deepestNesting() : 0; |
| } |
| |
| bool isStructureContainingArrays() const |
| { |
| return structure ? structure->containsArrays() : false; |
| } |
| |
| bool isStructureContainingType(TBasicType t) const |
| { |
| return structure ? structure->containsType(t) : false; |
| } |
| |
| bool isStructureContainingSamplers() const |
| { |
| return structure ? structure->containsSamplers() : false; |
| } |
| |
| protected: |
| void buildMangledName(TString&); |
| size_t getStructSize() const; |
| |
| TBasicType type = EbtVoid; |
| TPrecision precision = EbpUndefined; |
| TQualifier qualifier = EvqTemporary; |
| unsigned char primarySize = 0; // size of vector or matrix, not size of array |
| unsigned char secondarySize = 0; // 1 for vectors, > 1 for matrices |
| bool array = false; |
| int arraySize = 0; |
| int maxArraySize = 0; |
| TType *arrayInformationType = nullptr; |
| |
| // null unless this is an interface block, or interface block member variable |
| TInterfaceBlock *interfaceBlock = nullptr; |
| TLayoutQualifier layoutQualifier; |
| |
| TStructure *structure = nullptr; // null unless this is a struct |
| |
| TString *mangled = nullptr; |
| }; |
| |
| // |
| // This is a workaround for a problem with the yacc stack, It can't have |
| // types that it thinks have non-trivial constructors. It should |
| // just be used while recognizing the grammar, not anything else. Pointers |
| // could be used, but also trying to avoid lots of memory management overhead. |
| // |
| // Not as bad as it looks, there is no actual assumption that the fields |
| // match up or are name the same or anything like that. |
| // |
| struct TPublicType |
| { |
| TBasicType type; |
| TLayoutQualifier layoutQualifier; |
| TQualifier qualifier; |
| bool invariant; |
| TPrecision precision; |
| int primarySize; // size of vector or matrix, not size of array |
| int secondarySize; // 1 for scalars/vectors, >1 for matrices |
| bool array; |
| int arraySize; |
| TType* userDef; |
| TSourceLoc line; |
| |
| void setBasic(TBasicType bt, TQualifier q, const TSourceLoc &ln) |
| { |
| type = bt; |
| layoutQualifier = TLayoutQualifier::create(); |
| qualifier = q; |
| invariant = false; |
| precision = EbpUndefined; |
| primarySize = 1; |
| secondarySize = 1; |
| array = false; |
| arraySize = 0; |
| userDef = 0; |
| line = ln; |
| } |
| |
| void setAggregate(int s) |
| { |
| primarySize = s; |
| secondarySize = 1; |
| } |
| |
| void setMatrix(int s0, int s1) |
| { |
| primarySize = s0; |
| secondarySize = s1; |
| } |
| |
| bool isUnsizedArray() const |
| { |
| return array && arraySize == 0; |
| } |
| |
| void setArray(bool a, int s = 0) |
| { |
| array = a; |
| arraySize = s; |
| } |
| |
| void clearArrayness() |
| { |
| array = false; |
| arraySize = 0; |
| } |
| |
| bool isStructureContainingArrays() const |
| { |
| if (!userDef) |
| { |
| return false; |
| } |
| |
| return userDef->isStructureContainingArrays(); |
| } |
| |
| bool isStructureContainingType(TBasicType t) const |
| { |
| if(!userDef) |
| { |
| return false; |
| } |
| |
| return userDef->isStructureContainingType(t); |
| } |
| |
| bool isMatrix() const |
| { |
| return primarySize > 1 && secondarySize > 1; |
| } |
| |
| bool isVector() const |
| { |
| return primarySize > 1 && secondarySize == 1; |
| } |
| |
| int getCols() const |
| { |
| ASSERT(isMatrix()); |
| return primarySize; |
| } |
| |
| int getRows() const |
| { |
| ASSERT(isMatrix()); |
| return secondarySize; |
| } |
| |
| int getNominalSize() const |
| { |
| return primarySize; |
| } |
| |
| bool isAggregate() const |
| { |
| return array || isMatrix() || isVector(); |
| } |
| }; |
| |
| #endif // _TYPES_INCLUDED_ |