/* * Copyright 2015 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef GrMesh_DEFINED #define GrMesh_DEFINED #include "src/gpu/GrBuffer.h" #include "src/gpu/GrGpuBuffer.h" #include "src/gpu/GrPendingIOResource.h" class GrPrimitiveProcessor; /** * Used to communicate index and vertex buffers, counts, and offsets for a draw from GrOp to * GrGpu. It also holds the primitive type for the draw. TODO: Consider moving ownership of this * and draw-issuing responsibility to GrPrimitiveProcessor. The rest of the vertex info lives there * already (stride, attribute mappings). */ class GrMesh { public: GrMesh(GrPrimitiveType primitiveType = GrPrimitiveType::kTriangles) : fPrimitiveType(primitiveType), fBaseVertex(0) { SkDEBUGCODE(fNonIndexNonInstanceData.fVertexCount = -1;) } void setPrimitiveType(GrPrimitiveType type) { fPrimitiveType = type; } GrPrimitiveType primitiveType() const { return fPrimitiveType; } bool isIndexed() const { return SkToBool(fIndexBuffer.get()); } GrPrimitiveRestart primitiveRestart() const { return GrPrimitiveRestart(fFlags & Flags::kUsePrimitiveRestart); } bool isInstanced() const { return fFlags & Flags::kIsInstanced; } bool hasInstanceData() const { return SkToBool(fInstanceBuffer.get()); } bool hasVertexData() const { return SkToBool(fVertexBuffer.get()); } void setNonIndexedNonInstanced(int vertexCount); void setIndexed(sk_sp indexBuffer, int indexCount, int baseIndex, uint16_t minIndexValue, uint16_t maxIndexValue, GrPrimitiveRestart); void setIndexedPatterned(sk_sp indexBuffer, int indexCount, int vertexCount, int patternRepeatCount, int maxPatternRepetitionsInIndexBuffer); void setInstanced(sk_sp instanceBuffer, int instanceCount, int baseInstance, int vertexCount); void setIndexedInstanced(sk_sp indexBuffer, int indexCount, sk_sp instanceBuffer, int instanceCount, int baseInstance, GrPrimitiveRestart); void setVertexData(sk_sp vertexBuffer, int baseVertex = 0); class SendToGpuImpl { public: virtual void sendMeshToGpu(GrPrimitiveType, const GrBuffer* vertexBuffer, int vertexCount, int baseVertex) = 0; virtual void sendIndexedMeshToGpu(GrPrimitiveType, const GrBuffer* indexBuffer, int indexCount, int baseIndex, uint16_t minIndexValue, uint16_t maxIndexValue, const GrBuffer* vertexBuffer, int baseVertex, GrPrimitiveRestart) = 0; virtual void sendInstancedMeshToGpu(GrPrimitiveType, const GrBuffer* vertexBuffer, int vertexCount, int baseVertex, const GrBuffer* instanceBuffer, int instanceCount, int baseInstance) = 0; virtual void sendIndexedInstancedMeshToGpu(GrPrimitiveType, const GrBuffer* indexBuffer, int indexCount, int baseIndex, const GrBuffer* vertexBuffer, int baseVertex, const GrBuffer* instanceBuffer, int instanceCount, int baseInstance, GrPrimitiveRestart) = 0; virtual ~SendToGpuImpl() {} }; void sendToGpu(SendToGpuImpl*) const; private: enum class Flags { kNone = 0, kUsePrimitiveRestart = 1 << 0, kIsInstanced = 1 << 1, }; GR_DECL_BITFIELD_CLASS_OPS_FRIENDS(Flags); GR_STATIC_ASSERT(Flags(GrPrimitiveRestart::kNo) == Flags::kNone); GR_STATIC_ASSERT(Flags(GrPrimitiveRestart::kYes) == Flags::kUsePrimitiveRestart); GrPrimitiveType fPrimitiveType; sk_sp fIndexBuffer; sk_sp fInstanceBuffer; sk_sp fVertexBuffer; int fBaseVertex; Flags fFlags; union { struct { // When fIndexBuffer == nullptr and isInstanced() == false. int fVertexCount; } fNonIndexNonInstanceData; struct { // When fIndexBuffer != nullptr and isInstanced() == false. struct { int fIndexCount; int fPatternRepeatCount; } fIndexData; union { struct { // When fPatternRepeatCount == 0. int fBaseIndex; uint16_t fMinIndexValue; uint16_t fMaxIndexValue; } fNonPatternIndexData; struct { // When fPatternRepeatCount != 0. int fVertexCount; int fMaxPatternRepetitionsInIndexBuffer; } fPatternData; }; }; struct { // When isInstanced() != false. struct { int fInstanceCount; int fBaseInstance; } fInstanceData; union { // When fIndexBuffer == nullptr. struct { int fVertexCount; } fInstanceNonIndexData; struct { // When fIndexBuffer != nullptr. int fIndexCount; } fInstanceIndexData; }; }; }; }; GR_MAKE_BITFIELD_CLASS_OPS(GrMesh::Flags); inline void GrMesh::setNonIndexedNonInstanced(int vertexCount) { fIndexBuffer.reset(); fInstanceBuffer.reset(); fNonIndexNonInstanceData.fVertexCount = vertexCount; fFlags = Flags::kNone; } inline void GrMesh::setIndexed(sk_sp indexBuffer, int indexCount, int baseIndex, uint16_t minIndexValue, uint16_t maxIndexValue, GrPrimitiveRestart primitiveRestart) { SkASSERT(indexBuffer); SkASSERT(indexCount >= 1); SkASSERT(baseIndex >= 0); SkASSERT(maxIndexValue >= minIndexValue); fIndexBuffer = std::move(indexBuffer); fInstanceBuffer.reset(); fIndexData.fIndexCount = indexCount; fIndexData.fPatternRepeatCount = 0; fNonPatternIndexData.fBaseIndex = baseIndex; fNonPatternIndexData.fMinIndexValue = minIndexValue; fNonPatternIndexData.fMaxIndexValue = maxIndexValue; fFlags = Flags(primitiveRestart); } inline void GrMesh::setIndexedPatterned(sk_sp indexBuffer, int indexCount, int vertexCount, int patternRepeatCount, int maxPatternRepetitionsInIndexBuffer) { SkASSERT(indexBuffer); SkASSERT(indexCount >= 1); SkASSERT(vertexCount >= 1); SkASSERT(patternRepeatCount >= 1); SkASSERT(maxPatternRepetitionsInIndexBuffer >= 1); fIndexBuffer = std::move(indexBuffer); fInstanceBuffer.reset(); fIndexData.fIndexCount = indexCount; fIndexData.fPatternRepeatCount = patternRepeatCount; fPatternData.fVertexCount = vertexCount; fPatternData.fMaxPatternRepetitionsInIndexBuffer = maxPatternRepetitionsInIndexBuffer; fFlags = Flags::kNone; } inline void GrMesh::setInstanced(sk_sp instanceBuffer, int instanceCount, int baseInstance, int vertexCount) { SkASSERT(instanceCount >= 1); SkASSERT(baseInstance >= 0); fIndexBuffer.reset(); fInstanceBuffer = std::move(instanceBuffer); fInstanceData.fInstanceCount = instanceCount; fInstanceData.fBaseInstance = baseInstance; fInstanceNonIndexData.fVertexCount = vertexCount; fFlags = Flags::kIsInstanced; } inline void GrMesh::setIndexedInstanced(sk_sp indexBuffer, int indexCount, sk_sp instanceBuffer, int instanceCount, int baseInstance, GrPrimitiveRestart primitiveRestart) { SkASSERT(indexBuffer); SkASSERT(indexCount >= 1); SkASSERT(instanceCount >= 1); SkASSERT(baseInstance >= 0); fIndexBuffer = std::move(indexBuffer); fInstanceBuffer = std::move(instanceBuffer); fInstanceData.fInstanceCount = instanceCount; fInstanceData.fBaseInstance = baseInstance; fInstanceIndexData.fIndexCount = indexCount; fFlags = Flags::kIsInstanced | Flags(primitiveRestart); } inline void GrMesh::setVertexData(sk_sp vertexBuffer, int baseVertex) { SkASSERT(baseVertex >= 0); fVertexBuffer = std::move(vertexBuffer); fBaseVertex = baseVertex; } inline void GrMesh::sendToGpu(SendToGpuImpl* impl) const { if (this->isInstanced()) { if (!this->isIndexed()) { impl->sendInstancedMeshToGpu(fPrimitiveType, fVertexBuffer.get(), fInstanceNonIndexData.fVertexCount, fBaseVertex, fInstanceBuffer.get(), fInstanceData.fInstanceCount, fInstanceData.fBaseInstance); } else { impl->sendIndexedInstancedMeshToGpu( fPrimitiveType, fIndexBuffer.get(), fInstanceIndexData.fIndexCount, 0, fVertexBuffer.get(), fBaseVertex, fInstanceBuffer.get(), fInstanceData.fInstanceCount, fInstanceData.fBaseInstance, this->primitiveRestart()); } return; } if (!this->isIndexed()) { SkASSERT(fNonIndexNonInstanceData.fVertexCount > 0); impl->sendMeshToGpu(fPrimitiveType, fVertexBuffer.get(), fNonIndexNonInstanceData.fVertexCount, fBaseVertex); return; } if (0 == fIndexData.fPatternRepeatCount) { impl->sendIndexedMeshToGpu( fPrimitiveType, fIndexBuffer.get(), fIndexData.fIndexCount, fNonPatternIndexData.fBaseIndex, fNonPatternIndexData.fMinIndexValue, fNonPatternIndexData.fMaxIndexValue, fVertexBuffer.get(), fBaseVertex, this->primitiveRestart()); return; } SkASSERT(fIndexData.fPatternRepeatCount > 0); int baseRepetition = 0; do { int repeatCount = SkTMin(fPatternData.fMaxPatternRepetitionsInIndexBuffer, fIndexData.fPatternRepeatCount - baseRepetition); // A patterned index buffer must contain indices in the range [0..vertexCount]. int minIndexValue = 0; int maxIndexValue = fPatternData.fVertexCount * repeatCount - 1; SkASSERT(!(fFlags & Flags::kUsePrimitiveRestart)); impl->sendIndexedMeshToGpu( fPrimitiveType, fIndexBuffer.get(), fIndexData.fIndexCount * repeatCount, 0, minIndexValue, maxIndexValue, fVertexBuffer.get(), fBaseVertex + fPatternData.fVertexCount * baseRepetition, GrPrimitiveRestart::kNo); baseRepetition += repeatCount; } while (baseRepetition < fIndexData.fPatternRepeatCount); } #endif