| 1 | /* |
| 2 | * Copyright 2015 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
| 8 | #include "src/gpu/ops/GrMeshDrawOp.h" |
| 9 | |
| 10 | #include "src/gpu/GrOpFlushState.h" |
| 11 | #include "src/gpu/GrOpsRenderPass.h" |
| 12 | #include "src/gpu/GrRecordingContextPriv.h" |
| 13 | #include "src/gpu/GrResourceProvider.h" |
| 14 | |
| 15 | GrMeshDrawOp::GrMeshDrawOp(uint32_t classID) : INHERITED(classID) {} |
| 16 | |
| 17 | void GrMeshDrawOp::onPrepare(GrOpFlushState* state) { this->onPrepareDraws(state); } |
| 18 | |
| 19 | void GrMeshDrawOp::createProgramInfo(Target* target) { |
| 20 | this->createProgramInfo(&target->caps(), |
| 21 | target->allocator(), |
| 22 | target->writeView(), |
| 23 | target->detachAppliedClip(), |
| 24 | target->dstProxyView()); |
| 25 | } |
| 26 | |
| 27 | // This onPrepareDraws implementation assumes the derived Op only has a single programInfo - |
| 28 | // which is the majority of the cases. |
| 29 | void GrMeshDrawOp::onPrePrepareDraws(GrRecordingContext* context, |
| 30 | const GrSurfaceProxyView* writeView, |
| 31 | GrAppliedClip* clip, |
| 32 | const GrXferProcessor::DstProxyView& dstProxyView) { |
| 33 | SkArenaAlloc* arena = context->priv().recordTimeAllocator(); |
| 34 | |
| 35 | // This is equivalent to a GrOpFlushState::detachAppliedClip |
| 36 | GrAppliedClip appliedClip = clip ? std::move(*clip) : GrAppliedClip(); |
| 37 | |
| 38 | this->createProgramInfo(context->priv().caps(), arena, writeView, |
| 39 | std::move(appliedClip), dstProxyView); |
| 40 | |
| 41 | // TODO: at this point we've created both the program info and desc in the recording context's |
| 42 | // arena. In the DDL case, it would be cool if 'recordProgramInfo' could return the |
| 43 | // pre-existing versions if the program has already been seen. We could then return the |
| 44 | // memory for the current copy to the arena. |
| 45 | context->priv().recordProgramInfo(this->programInfo()); |
| 46 | } |
| 47 | |
| 48 | ////////////////////////////////////////////////////////////////////////////// |
| 49 | |
| 50 | GrMeshDrawOp::PatternHelper::PatternHelper(Target* target, GrPrimitiveType primitiveType, |
| 51 | size_t vertexStride, sk_sp<const GrBuffer> indexBuffer, |
| 52 | int verticesPerRepetition, int indicesPerRepetition, |
| 53 | int repeatCount, int maxRepetitions) { |
| 54 | this->init(target, primitiveType, vertexStride, std::move(indexBuffer), verticesPerRepetition, |
| 55 | indicesPerRepetition, repeatCount, maxRepetitions); |
| 56 | } |
| 57 | |
| 58 | void GrMeshDrawOp::PatternHelper::init(Target* target, GrPrimitiveType primitiveType, |
| 59 | size_t vertexStride, sk_sp<const GrBuffer> indexBuffer, |
| 60 | int verticesPerRepetition, int indicesPerRepetition, |
| 61 | int repeatCount, int maxRepetitions) { |
| 62 | SkASSERT(target); |
| 63 | if (!indexBuffer) { |
| 64 | return; |
| 65 | } |
| 66 | sk_sp<const GrBuffer> vertexBuffer; |
| 67 | int firstVertex; |
| 68 | int vertexCount = verticesPerRepetition * repeatCount; |
| 69 | fVertices = target->makeVertexSpace(vertexStride, vertexCount, &vertexBuffer, &firstVertex); |
| 70 | if (!fVertices) { |
| 71 | SkDebugf("Vertices could not be allocated for patterned rendering." ); |
| 72 | return; |
| 73 | } |
| 74 | SkASSERT(vertexBuffer); |
| 75 | fMesh = target->allocMesh(); |
| 76 | fPrimitiveType = primitiveType; |
| 77 | |
| 78 | SkASSERT(maxRepetitions == |
| 79 | static_cast<int>(indexBuffer->size() / (sizeof(uint16_t) * indicesPerRepetition))); |
| 80 | fMesh->setIndexedPatterned(std::move(indexBuffer), indicesPerRepetition, repeatCount, |
| 81 | maxRepetitions, std::move(vertexBuffer), verticesPerRepetition, |
| 82 | firstVertex); |
| 83 | } |
| 84 | |
| 85 | void GrMeshDrawOp::PatternHelper::recordDraw(Target* target, const GrGeometryProcessor* gp) const { |
| 86 | target->recordDraw(gp, fMesh, 1, fPrimitiveType); |
| 87 | } |
| 88 | |
| 89 | void GrMeshDrawOp::PatternHelper::recordDraw( |
| 90 | Target* target, |
| 91 | const GrGeometryProcessor* gp, |
| 92 | const GrSurfaceProxy* const primProcProxies[]) const { |
| 93 | target->recordDraw(gp, fMesh, 1, primProcProxies, fPrimitiveType); |
| 94 | } |
| 95 | |
| 96 | ////////////////////////////////////////////////////////////////////////////// |
| 97 | |
| 98 | GrMeshDrawOp::QuadHelper::QuadHelper(Target* target, size_t vertexStride, int quadsToDraw) { |
| 99 | sk_sp<const GrGpuBuffer> indexBuffer = target->resourceProvider()->refNonAAQuadIndexBuffer(); |
| 100 | if (!indexBuffer) { |
| 101 | SkDebugf("Could not get quad index buffer." ); |
| 102 | return; |
| 103 | } |
| 104 | this->init(target, GrPrimitiveType::kTriangles, vertexStride, std::move(indexBuffer), |
| 105 | GrResourceProvider::NumVertsPerNonAAQuad(), |
| 106 | GrResourceProvider::NumIndicesPerNonAAQuad(), quadsToDraw, |
| 107 | GrResourceProvider::MaxNumNonAAQuads()); |
| 108 | } |
| 109 | |