Index: src/sksl/ir/SkSLType.h |
diff --git a/src/sksl/ir/SkSLType.h b/src/sksl/ir/SkSLType.h |
new file mode 100644 |
index 0000000000000000000000000000000000000000..e17bae68db93f368b0e6bea5139795ac6beb0b29 |
--- /dev/null |
+++ b/src/sksl/ir/SkSLType.h |
@@ -0,0 +1,438 @@ |
+/* |
+ * Copyright 2016 Google Inc. |
+ * |
+ * Use of this source code is governed by a BSD-style license that can be |
+ * found in the LICENSE file. |
+ */ |
+ |
+#ifndef SKIASL_TYPE |
+#define SKIASL_TYPE |
+ |
+#include "SkSLModifiers.h" |
+#include "SkSLSymbol.h" |
+#include "../SkSLPosition.h" |
+#include "../SkSLUtil.h" |
+#include "../spirv.h" |
+#include <vector> |
+#include <memory> |
+ |
+namespace SkSL { |
+ |
+/** |
+ * Represents a type, such as int or vec4. |
+ */ |
+class Type : public Symbol { |
+public: |
+ struct Field { |
+ Field(Modifiers modifiers, std::string name, std::shared_ptr<Type> type) |
+ : fModifiers(modifiers) |
+ , fName(std::move(name)) |
+ , fType(std::move(type)) {} |
+ |
+ const std::string description() { |
+ return fType->description() + " " + fName + ";"; |
+ } |
+ |
+ const Modifiers fModifiers; |
+ const std::string fName; |
+ const std::shared_ptr<Type> fType; |
+ }; |
+ |
+ enum Kind { |
+ kScalar_Kind, |
+ kVector_Kind, |
+ kMatrix_Kind, |
+ kArray_Kind, |
+ kStruct_Kind, |
+ kGeneric_Kind, |
+ kSampler_Kind, |
+ kOther_Kind |
+ }; |
+ |
+ // Create an "other" (special) type with the given name. These types cannot be directly |
+ // referenced from user code. |
+ Type(std::string name) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kOther_Kind) {} |
+ |
+ // Create a generic type which maps to the listed types. |
+ Type(std::string name, std::vector<std::shared_ptr<Type>> types) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kGeneric_Kind) |
+ , fCoercibleTypes(std::move(types)) { |
+ ASSERT(fCoercibleTypes.size() == 4); |
+ } |
+ |
+ // Create a struct type with the given fields. |
+ Type(std::string name, std::vector<Field> fields) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kStruct_Kind) |
+ , fFields(std::move(fields)) {} |
+ |
+ // Create a scalar type. |
+ Type(std::string name, bool isNumber) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kScalar_Kind) |
+ , fIsNumber(isNumber) |
+ , fColumns(1) |
+ , fRows(1) {} |
+ |
+ // Create a scalar type which can be coerced to the listed types. |
+ Type(std::string name, bool isNumber, std::vector<std::shared_ptr<Type>> coercibleTypes) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kScalar_Kind) |
+ , fIsNumber(isNumber) |
+ , fCoercibleTypes(std::move(coercibleTypes)) |
+ , fColumns(1) |
+ , fRows(1) {} |
+ |
+ // Create a vector type. |
+ Type(std::string name, std::shared_ptr<Type> componentType, int columns) |
+ : Type(name, kVector_Kind, componentType, columns) {} |
+ |
+ // Create a vector or array type. |
+ Type(std::string name, Kind kind, std::shared_ptr<Type> componentType, int columns) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kind) |
+ , fComponentType(std::move(componentType)) |
+ , fColumns(columns) |
+ , fRows(1) |
+ , fDimensions(SpvDim1D) {} |
+ |
+ // Create a matrix type. |
+ Type(std::string name, std::shared_ptr<Type> componentType, int columns, int rows) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kMatrix_Kind) |
+ , fComponentType(std::move(componentType)) |
+ , fColumns(columns) |
+ , fRows(rows) |
+ , fDimensions(SpvDim1D) {} |
+ |
+ // Create a sampler type. |
+ Type(std::string name, SpvDim_ dimensions, bool isDepth, bool isArrayed, bool isMultisampled, |
+ bool isSampled) |
+ : INHERITED(Position(), kType_Kind, std::move(name)) |
+ , fTypeKind(kSampler_Kind) |
+ , fDimensions(dimensions) |
+ , fIsDepth(isDepth) |
+ , fIsArrayed(isArrayed) |
+ , fIsMultisampled(isMultisampled) |
+ , fIsSampled(isSampled) {} |
+ |
+ std::string name() const { |
+ return fName; |
+ } |
+ |
+ std::string description() const override { |
+ return fName; |
+ } |
+ |
+ bool operator==(const Type& other) const { |
+ return fName == other.fName; |
+ } |
+ |
+ bool operator!=(const Type& other) const { |
+ return fName != other.fName; |
+ } |
+ |
+ /** |
+ * Returns the category (scalar, vector, matrix, etc.) of this type. |
+ */ |
+ Kind kind() const { |
+ return fTypeKind; |
+ } |
+ |
+ /** |
+ * Returns true if this is a numeric scalar type. |
+ */ |
+ bool isNumber() const { |
+ return fIsNumber; |
+ } |
+ |
+ /** |
+ * Returns true if an instance of this type can be freely coerced (implicitly converted) to |
+ * another type. |
+ */ |
+ bool canCoerceTo(std::shared_ptr<Type> other) const { |
+ int cost; |
+ return determineCoercionCost(other, &cost); |
+ } |
+ |
+ /** |
+ * Determines the "cost" of coercing (implicitly converting) this type to another type. The cost |
+ * is a number with no particular meaning other than that lower costs are preferable to higher |
+ * costs. Returns true if a conversion is possible, false otherwise. The value of the out |
+ * parameter is undefined if false is returned. |
+ */ |
+ bool determineCoercionCost(std::shared_ptr<Type> other, int* outCost) const; |
+ |
+ /** |
+ * For matrices and vectors, returns the type of individual cells (e.g. mat2 has a component |
+ * type of kFloat_Type). For all other types, causes an assertion failure. |
+ */ |
+ std::shared_ptr<Type> componentType() const { |
+ ASSERT(fComponentType); |
+ return fComponentType; |
+ } |
+ |
+ /** |
+ * For matrices and vectors, returns the number of columns (e.g. both mat3 and vec3 return 3). |
+ * For scalars, returns 1. For arrays, returns either the size of the array (if known) or -1. |
+ * For all other types, causes an assertion failure. |
+ */ |
+ int columns() const { |
+ ASSERT(fTypeKind == kScalar_Kind || fTypeKind == kVector_Kind || |
+ fTypeKind == kMatrix_Kind || fTypeKind == kArray_Kind); |
+ return fColumns; |
+ } |
+ |
+ /** |
+ * For matrices, returns the number of rows (e.g. mat2x4 returns 4). For vectors and scalars, |
+ * returns 1. For all other types, causes an assertion failure. |
+ */ |
+ int rows() const { |
+ ASSERT(fRows > 0); |
+ return fRows; |
+ } |
+ |
+ std::vector<Field> fields() const { |
+ ASSERT(fTypeKind == kStruct_Kind); |
+ return fFields; |
+ } |
+ |
+ /** |
+ * For generic types, returns the types that this generic type can substitute for. For other |
+ * types, returns a list of other types that this type can be coerced into. |
+ */ |
+ std::vector<std::shared_ptr<Type>> coercibleTypes() const { |
+ ASSERT(fCoercibleTypes.size() > 0); |
+ return fCoercibleTypes; |
+ } |
+ |
+ int dimensions() const { |
+ ASSERT(fTypeKind == kSampler_Kind); |
+ return fDimensions; |
+ } |
+ |
+ bool isDepth() const { |
+ ASSERT(fTypeKind == kSampler_Kind); |
+ return fIsDepth; |
+ } |
+ |
+ bool isArrayed() const { |
+ ASSERT(fTypeKind == kSampler_Kind); |
+ return fIsArrayed; |
+ } |
+ |
+ bool isMultisampled() const { |
+ ASSERT(fTypeKind == kSampler_Kind); |
+ return fIsMultisampled; |
+ } |
+ |
+ bool isSampled() const { |
+ ASSERT(fTypeKind == kSampler_Kind); |
+ return fIsSampled; |
+ } |
+ |
+ static size_t vector_alignment(size_t componentSize, int columns) { |
+ return componentSize * (columns + columns % 2); |
+ } |
+ |
+ /** |
+ * Returns the type's required alignment (when putting this type into a struct, the offset must |
+ * be a multiple of the alignment). |
+ */ |
+ size_t alignment() const { |
+ // See OpenGL Spec 7.6.2.2 Standard Uniform Block Layout |
+ switch (fTypeKind) { |
+ case kScalar_Kind: |
+ return this->size(); |
+ case kVector_Kind: |
+ return vector_alignment(fComponentType->size(), fColumns); |
+ case kMatrix_Kind: |
+ return (vector_alignment(fComponentType->size(), fRows) + 15) & ~15; |
+ case kArray_Kind: |
+ // round up to next multiple of 16 |
+ return (fComponentType->alignment() + 15) & ~15; |
+ case kStruct_Kind: { |
+ size_t result = 16; |
+ for (size_t i = 0; i < fFields.size(); i++) { |
+ size_t alignment = fFields[i].fType->alignment(); |
+ if (alignment > result) { |
+ result = alignment; |
+ } |
+ } |
+ } |
+ default: |
+ ABORT(("cannot determine size of type " + fName).c_str()); |
+ } |
+ } |
+ |
+ /** |
+ * For matrices and arrays, returns the number of bytes from the start of one entry (row, in |
+ * the case of matrices) to the start of the next. |
+ */ |
+ size_t stride() const { |
+ switch (fTypeKind) { |
+ case kMatrix_Kind: // fall through |
+ case kArray_Kind: |
+ return this->alignment(); |
+ default: |
+ ABORT("type does not have a stride"); |
+ } |
+ } |
+ |
+ /** |
+ * Returns the size of this type in bytes. |
+ */ |
+ size_t size() const { |
+ switch (fTypeKind) { |
+ case kScalar_Kind: |
+ // FIXME need to take precision into account, once we figure out how we want to |
+ // handle it... |
+ return 4; |
+ case kVector_Kind: |
+ return fColumns * fComponentType->size(); |
+ case kMatrix_Kind: |
+ return vector_alignment(fComponentType->size(), fRows) * fColumns; |
+ case kArray_Kind: |
+ return fColumns * this->stride(); |
+ case kStruct_Kind: { |
+ size_t total = 0; |
+ for (size_t i = 0; i < fFields.size(); i++) { |
+ size_t alignment = fFields[i].fType->alignment(); |
+ if (total % alignment != 0) { |
+ total += alignment - total % alignment; |
+ } |
+ ASSERT(false); |
+ ASSERT(total % alignment == 0); |
+ total += fFields[i].fType->size(); |
+ } |
+ return total; |
+ } |
+ default: |
+ ABORT(("cannot determine size of type " + fName).c_str()); |
+ } |
+ } |
+ |
+ /** |
+ * Returns the corresponding vector or matrix type with the specified number of columns and |
+ * rows. |
+ */ |
+ std::shared_ptr<Type> toCompound(int columns, int rows); |
+ |
+private: |
+ typedef Symbol INHERITED; |
+ |
+ const Kind fTypeKind; |
+ const bool fIsNumber = false; |
+ const std::shared_ptr<Type> fComponentType = nullptr; |
+ const std::vector<std::shared_ptr<Type>> fCoercibleTypes = { }; |
+ const int fColumns = -1; |
+ const int fRows = -1; |
+ const std::vector<Field> fFields = { }; |
+ const SpvDim_ fDimensions = SpvDim1D; |
+ const bool fIsDepth = false; |
+ const bool fIsArrayed = false; |
+ const bool fIsMultisampled = false; |
+ const bool fIsSampled = false; |
+}; |
+ |
+extern const std::shared_ptr<Type> kVoid_Type; |
+ |
+extern const std::shared_ptr<Type> kFloat_Type; |
+extern const std::shared_ptr<Type> kVec2_Type; |
+extern const std::shared_ptr<Type> kVec3_Type; |
+extern const std::shared_ptr<Type> kVec4_Type; |
+extern const std::shared_ptr<Type> kDouble_Type; |
+extern const std::shared_ptr<Type> kDVec2_Type; |
+extern const std::shared_ptr<Type> kDVec3_Type; |
+extern const std::shared_ptr<Type> kDVec4_Type; |
+extern const std::shared_ptr<Type> kInt_Type; |
+extern const std::shared_ptr<Type> kIVec2_Type; |
+extern const std::shared_ptr<Type> kIVec3_Type; |
+extern const std::shared_ptr<Type> kIVec4_Type; |
+extern const std::shared_ptr<Type> kUInt_Type; |
+extern const std::shared_ptr<Type> kUVec2_Type; |
+extern const std::shared_ptr<Type> kUVec3_Type; |
+extern const std::shared_ptr<Type> kUVec4_Type; |
+extern const std::shared_ptr<Type> kBool_Type; |
+extern const std::shared_ptr<Type> kBVec2_Type; |
+extern const std::shared_ptr<Type> kBVec3_Type; |
+extern const std::shared_ptr<Type> kBVec4_Type; |
+ |
+extern const std::shared_ptr<Type> kMat2x2_Type; |
+extern const std::shared_ptr<Type> kMat2x3_Type; |
+extern const std::shared_ptr<Type> kMat2x4_Type; |
+extern const std::shared_ptr<Type> kMat3x2_Type; |
+extern const std::shared_ptr<Type> kMat3x3_Type; |
+extern const std::shared_ptr<Type> kMat3x4_Type; |
+extern const std::shared_ptr<Type> kMat4x2_Type; |
+extern const std::shared_ptr<Type> kMat4x3_Type; |
+extern const std::shared_ptr<Type> kMat4x4_Type; |
+ |
+extern const std::shared_ptr<Type> kDMat2x2_Type; |
+extern const std::shared_ptr<Type> kDMat2x3_Type; |
+extern const std::shared_ptr<Type> kDMat2x4_Type; |
+extern const std::shared_ptr<Type> kDMat3x2_Type; |
+extern const std::shared_ptr<Type> kDMat3x3_Type; |
+extern const std::shared_ptr<Type> kDMat3x4_Type; |
+extern const std::shared_ptr<Type> kDMat4x2_Type; |
+extern const std::shared_ptr<Type> kDMat4x3_Type; |
+extern const std::shared_ptr<Type> kDMat4x4_Type; |
+ |
+extern const std::shared_ptr<Type> kSampler1D_Type; |
+extern const std::shared_ptr<Type> kSampler2D_Type; |
+extern const std::shared_ptr<Type> kSampler3D_Type; |
+extern const std::shared_ptr<Type> kSamplerCube_Type; |
+extern const std::shared_ptr<Type> kSampler2DRect_Type; |
+extern const std::shared_ptr<Type> kSampler1DArray_Type; |
+extern const std::shared_ptr<Type> kSampler2DArray_Type; |
+extern const std::shared_ptr<Type> kSamplerCubeArray_Type; |
+extern const std::shared_ptr<Type> kSamplerBuffer_Type; |
+extern const std::shared_ptr<Type> kSampler2DMS_Type; |
+extern const std::shared_ptr<Type> kSampler2DMSArray_Type; |
+ |
+extern const std::shared_ptr<Type> kGSampler1D_Type; |
+extern const std::shared_ptr<Type> kGSampler2D_Type; |
+extern const std::shared_ptr<Type> kGSampler3D_Type; |
+extern const std::shared_ptr<Type> kGSamplerCube_Type; |
+extern const std::shared_ptr<Type> kGSampler2DRect_Type; |
+extern const std::shared_ptr<Type> kGSampler1DArray_Type; |
+extern const std::shared_ptr<Type> kGSampler2DArray_Type; |
+extern const std::shared_ptr<Type> kGSamplerCubeArray_Type; |
+extern const std::shared_ptr<Type> kGSamplerBuffer_Type; |
+extern const std::shared_ptr<Type> kGSampler2DMS_Type; |
+extern const std::shared_ptr<Type> kGSampler2DMSArray_Type; |
+ |
+extern const std::shared_ptr<Type> kSampler1DShadow_Type; |
+extern const std::shared_ptr<Type> kSampler2DShadow_Type; |
+extern const std::shared_ptr<Type> kSamplerCubeShadow_Type; |
+extern const std::shared_ptr<Type> kSampler2DRectShadow_Type; |
+extern const std::shared_ptr<Type> kSampler1DArrayShadow_Type; |
+extern const std::shared_ptr<Type> kSampler2DArrayShadow_Type; |
+extern const std::shared_ptr<Type> kSamplerCubeArrayShadow_Type; |
+extern const std::shared_ptr<Type> kGSampler2DArrayShadow_Type; |
+extern const std::shared_ptr<Type> kGSamplerCubeArrayShadow_Type; |
+ |
+extern const std::shared_ptr<Type> kGenType_Type; |
+extern const std::shared_ptr<Type> kGenDType_Type; |
+extern const std::shared_ptr<Type> kGenIType_Type; |
+extern const std::shared_ptr<Type> kGenUType_Type; |
+extern const std::shared_ptr<Type> kGenBType_Type; |
+extern const std::shared_ptr<Type> kMat_Type; |
+extern const std::shared_ptr<Type> kVec_Type; |
+extern const std::shared_ptr<Type> kGVec_Type; |
+extern const std::shared_ptr<Type> kGVec2_Type; |
+extern const std::shared_ptr<Type> kGVec3_Type; |
+extern const std::shared_ptr<Type> kGVec4_Type; |
+extern const std::shared_ptr<Type> kDVec_Type; |
+extern const std::shared_ptr<Type> kIVec_Type; |
+extern const std::shared_ptr<Type> kUVec_Type; |
+extern const std::shared_ptr<Type> kBVec_Type; |
+ |
+extern const std::shared_ptr<Type> kInvalid_Type; |
+ |
+} // namespace |
+ |
+#endif |