/* * Copyright 2015 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "GrMeshDrawOp.h" #include "GrOpFlushState.h" #include "GrResourceProvider.h" GrMeshDrawOp::GrMeshDrawOp(uint32_t classID) : INHERITED(classID), fBaseDrawToken(GrDrawOpUploadToken::AlreadyFlushedToken()) {} void GrMeshDrawOp::onPrepare(GrOpFlushState* state) { Target target(state, this); this->onPrepareDraws(&target); } void* GrMeshDrawOp::InstancedHelper::init(Target* target, GrPrimitiveType primType, size_t vertexStride, const GrBuffer* indexBuffer, int verticesPerInstance, int indicesPerInstance, int instancesToDraw) { SkASSERT(target); if (!indexBuffer) { return nullptr; } const GrBuffer* vertexBuffer; int firstVertex; int vertexCount = verticesPerInstance * instancesToDraw; void* vertices = target->makeVertexSpace(vertexStride, vertexCount, &vertexBuffer, &firstVertex); if (!vertices) { SkDebugf("Vertices could not be allocated for instanced rendering."); return nullptr; } SkASSERT(vertexBuffer); size_t ibSize = indexBuffer->gpuMemorySize(); int maxInstancesPerDraw = static_cast(ibSize / (sizeof(uint16_t) * indicesPerInstance)); fMesh.initInstanced(primType, vertexBuffer, indexBuffer, firstVertex, verticesPerInstance, indicesPerInstance, instancesToDraw, maxInstancesPerDraw); return vertices; } void GrMeshDrawOp::InstancedHelper::recordDraw(Target* target, const GrGeometryProcessor* gp) { SkASSERT(fMesh.instanceCount()); target->draw(gp, fMesh); } void* GrMeshDrawOp::QuadHelper::init(Target* target, size_t vertexStride, int quadsToDraw) { sk_sp quadIndexBuffer(target->resourceProvider()->refQuadIndexBuffer()); if (!quadIndexBuffer) { SkDebugf("Could not get quad index buffer."); return nullptr; } return this->INHERITED::init(target, kTriangles_GrPrimitiveType, vertexStride, quadIndexBuffer.get(), kVerticesPerQuad, kIndicesPerQuad, quadsToDraw); } void GrMeshDrawOp::onExecute(GrOpFlushState* state) { int currUploadIdx = 0; int currMeshIdx = 0; SkASSERT(fQueuedDraws.empty() || fBaseDrawToken == state->nextTokenToFlush()); for (int currDrawIdx = 0; currDrawIdx < fQueuedDraws.count(); ++currDrawIdx) { GrDrawOpUploadToken drawToken = state->nextTokenToFlush(); while (currUploadIdx < fInlineUploads.count() && fInlineUploads[currUploadIdx].fUploadBeforeToken == drawToken) { state->commandBuffer()->inlineUpload(state, fInlineUploads[currUploadIdx++].fUpload); } const QueuedDraw& draw = fQueuedDraws[currDrawIdx]; state->commandBuffer()->draw(*this->pipeline(), *draw.fGeometryProcessor.get(), fMeshes.begin() + currMeshIdx, draw.fMeshCnt, this->bounds()); currMeshIdx += draw.fMeshCnt; state->flushToken(); } SkASSERT(currUploadIdx == fInlineUploads.count()); SkASSERT(currMeshIdx == fMeshes.count()); fQueuedDraws.reset(); fInlineUploads.reset(); } ////////////////////////////////////////////////////////////////////////////// void GrMeshDrawOp::Target::draw(const GrGeometryProcessor* gp, const GrMesh& mesh) { GrMeshDrawOp* op = this->meshDrawOp(); op->fMeshes.push_back(mesh); if (!op->fQueuedDraws.empty()) { // If the last draw shares a geometry processor and there are no intervening uploads, // add this mesh to it. GrMeshDrawOp::QueuedDraw& lastDraw = op->fQueuedDraws.back(); if (lastDraw.fGeometryProcessor == gp && (op->fInlineUploads.empty() || op->fInlineUploads.back().fUploadBeforeToken != this->nextDrawToken())) { ++lastDraw.fMeshCnt; return; } } GrMeshDrawOp::QueuedDraw& draw = op->fQueuedDraws.push_back(); GrDrawOpUploadToken token = this->state()->issueDrawToken(); draw.fGeometryProcessor.reset(gp); draw.fMeshCnt = 1; if (op->fQueuedDraws.count() == 1) { op->fBaseDrawToken = token; } }