/* * 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 GrDrawOp_DEFINED #define GrDrawOp_DEFINED #include #include "GrOp.h" #include "GrPipeline.h" class GrAppliedClip; /** * GrDrawOps are flushed in two phases (preDraw, and draw). In preDraw uploads to GrGpuResources * and draws are determined and scheduled. They are issued in the draw phase. GrDrawOpUploadToken is * used to sequence the uploads relative to each other and to draws. **/ class GrDrawOpUploadToken { public: static GrDrawOpUploadToken AlreadyFlushedToken() { return GrDrawOpUploadToken(0); } GrDrawOpUploadToken(const GrDrawOpUploadToken& that) : fSequenceNumber(that.fSequenceNumber) {} GrDrawOpUploadToken& operator =(const GrDrawOpUploadToken& that) { fSequenceNumber = that.fSequenceNumber; return *this; } bool operator==(const GrDrawOpUploadToken& that) const { return fSequenceNumber == that.fSequenceNumber; } bool operator!=(const GrDrawOpUploadToken& that) const { return !(*this == that); } private: GrDrawOpUploadToken(); explicit GrDrawOpUploadToken(uint64_t sequenceNumber) : fSequenceNumber(sequenceNumber) {} friend class GrOpFlushState; uint64_t fSequenceNumber; }; /** * Base class for GrOps that draw. These ops have a GrPipeline installed by GrOpList. */ class GrDrawOp : public GrOp { public: /** Method that performs an upload on behalf of a DeferredUploadFn. */ using WritePixelsFn = std::function; /** See comments before GrDrawOp::Target definition on how deferred uploaders work. */ using DeferredUploadFn = std::function; class Target; GrDrawOp(uint32_t classID) : INHERITED(classID) {} /** * This information is required to determine how to compute a GrAppliedClip from a GrClip for * this op. */ enum class FixedFunctionFlags : uint32_t { kNone = 0x0, /** Indices that the op will enable MSAA or mixed samples rendering. */ kUsesHWAA = 0x1, /** Indices that the op reads and/or writes the stencil buffer */ kUsesStencil = 0x2, }; GR_DECL_BITFIELD_CLASS_OPS_FRIENDS(FixedFunctionFlags); virtual FixedFunctionFlags fixedFunctionFlags() const = 0; /** * This is called after the GrAppliedClip has been computed and just prior to recording the op * or combining it with a previously recorded op. It is used to determine whether a copy of the * destination (or destination texture itself) needs to be provided to the xp when this op * executes. This is guaranteed to be called before an op is recorded. However, this is also * called on ops that are not recorded because they combine with a previously recorded op. */ virtual bool xpRequiresDstTexture(const GrCaps&, const GrAppliedClip*) = 0; protected: static SkString DumpPipelineInfo(const GrPipeline& pipeline) { SkString string; string.appendf("RT: %d\n", pipeline.getRenderTarget()->uniqueID().asUInt()); string.append("ColorStages:\n"); for (int i = 0; i < pipeline.numColorFragmentProcessors(); i++) { string.appendf("\t\t%s\n\t\t%s\n", pipeline.getColorFragmentProcessor(i).name(), pipeline.getColorFragmentProcessor(i).dumpInfo().c_str()); } string.append("CoverageStages:\n"); for (int i = 0; i < pipeline.numCoverageFragmentProcessors(); i++) { string.appendf("\t\t%s\n\t\t%s\n", pipeline.getCoverageFragmentProcessor(i).name(), pipeline.getCoverageFragmentProcessor(i).dumpInfo().c_str()); } string.appendf("XP: %s\n", pipeline.getXferProcessor().name()); bool scissorEnabled = pipeline.getScissorState().enabled(); string.appendf("Scissor: "); if (scissorEnabled) { string.appendf("[L: %d, T: %d, R: %d, B: %d]\n", pipeline.getScissorState().rect().fLeft, pipeline.getScissorState().rect().fTop, pipeline.getScissorState().rect().fRight, pipeline.getScissorState().rect().fBottom); } else { string.appendf("\n"); } return string; } struct QueuedUpload { QueuedUpload(DeferredUploadFn&& upload, GrDrawOpUploadToken token) : fUpload(std::move(upload)) , fUploadBeforeToken(token) {} DeferredUploadFn fUpload; GrDrawOpUploadToken fUploadBeforeToken; }; SkTArray fInlineUploads; private: typedef GrOp INHERITED; }; GR_MAKE_BITFIELD_CLASS_OPS(GrDrawOp::FixedFunctionFlags); #endif