/* * Copyright 2019 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef GrRecordingContext_DEFINED #define GrRecordingContext_DEFINED #include "include/core/SkRefCnt.h" #include "include/private/GrImageContext.h" #include "include/private/SkTArray.h" class GrAuditTrail; class GrBackendFormat; class GrDrawingManager; class GrOnFlushCallbackObject; class GrOpMemoryPool; class GrProgramDesc; class GrProgramInfo; class GrRecordingContextPriv; class GrSurfaceContext; class GrSurfaceProxy; class GrTextBlobCache; class SkArenaAlloc; class SkJSONWriter; class GrRecordingContext : public GrImageContext { public: ~GrRecordingContext() override; SK_API GrBackendFormat defaultBackendFormat(SkColorType ct, GrRenderable renderable) const { return INHERITED::defaultBackendFormat(ct, renderable); } /** * Reports whether the GrDirectContext associated with this GrRecordingContext is abandoned. * When called on a GrDirectContext it may actively check whether the underlying 3D API * device/context has been disconnected before reporting the status. If so, calling this * method will transition the GrDirectContext to the abandoned state. */ bool abandoned() override { return INHERITED::abandoned(); } // Provides access to functions that aren't part of the public API. GrRecordingContextPriv priv(); const GrRecordingContextPriv priv() const; // The collection of specialized memory arenas for different types of data recorded by a // GrRecordingContext. Arenas does not maintain ownership of the pools it groups together. class Arenas { public: Arenas(GrOpMemoryPool*, SkArenaAlloc*); // For storing GrOp-derived classes recorded by a GrRecordingContext GrOpMemoryPool* opMemoryPool() { return fOpMemoryPool; } // For storing pipelines and other complex data as-needed by ops SkArenaAlloc* recordTimeAllocator() { return fRecordTimeAllocator; } private: GrOpMemoryPool* fOpMemoryPool; SkArenaAlloc* fRecordTimeAllocator; }; protected: friend class GrRecordingContextPriv; // for hidden functions friend class SkDeferredDisplayList; // for OwnedArenas friend class SkDeferredDisplayListPriv; // for ProgramData // Like Arenas, but preserves ownership of the underlying pools. class OwnedArenas { public: OwnedArenas(); ~OwnedArenas(); Arenas get(); OwnedArenas& operator=(OwnedArenas&&); private: std::unique_ptr fOpMemoryPool; std::unique_ptr fRecordTimeAllocator; }; GrRecordingContext(sk_sp); bool init() override; void setupDrawingManager(bool sortOpsTasks, bool reduceOpsTaskSplitting); void abandonContext() override; GrDrawingManager* drawingManager(); // There is no going back from this method. It should only be called to control the timing // during abandon or destruction of the context. void destroyDrawingManager(); Arenas arenas() { return fArenas.get(); } // This entry point should only be used for DDL creation where we want the ops' lifetime to // match that of the DDL. OwnedArenas&& detachArenas(); struct ProgramData { ProgramData(std::unique_ptr, const GrProgramInfo*); ProgramData(ProgramData&&); // for SkTArray ProgramData(const ProgramData&) = delete; ~ProgramData(); const GrProgramDesc& desc() const { return *fDesc; } const GrProgramInfo& info() const { return *fInfo; } private: // TODO: store the GrProgramDescs in the 'fRecordTimeData' arena std::unique_ptr fDesc; // The program infos should be stored in 'fRecordTimeData' so do not need to be ref // counted or deleted in the destructor. const GrProgramInfo* fInfo = nullptr; }; // This entry point gives the recording context a chance to cache the provided // programInfo. The DDL context takes this opportunity to store programInfos as a sidecar // to the DDL. virtual void recordProgramInfo(const GrProgramInfo*) {} // This asks the recording context to return any programInfos it may have collected // via the 'recordProgramInfo' call. It is up to the caller to ensure that the lifetime // of the programInfos matches the intended use. For example, in DDL-record mode it // is known that all the programInfos will have been allocated in an arena with the // same lifetime at the DDL itself. virtual void detachProgramData(SkTArray*) {} GrTextBlobCache* getTextBlobCache(); const GrTextBlobCache* getTextBlobCache() const; /** * Registers an object for flush-related callbacks. (See GrOnFlushCallbackObject.) * * NOTE: the drawing manager tracks this object as a raw pointer; it is up to the caller to * ensure its lifetime is tied to that of the context. */ void addOnFlushCallbackObject(GrOnFlushCallbackObject*); GrAuditTrail* auditTrail() { return fAuditTrail.get(); } GrRecordingContext* asRecordingContext() override { return this; } class Stats { public: Stats() = default; #if GR_GPU_STATS void reset() { *this = {}; } int numPathMasksGenerated() const { return fNumPathMasksGenerated; } void incNumPathMasksGenerated() { fNumPathMasksGenerated++; } int numPathMaskCacheHits() const { return fNumPathMaskCacheHits; } void incNumPathMasksCacheHits() { fNumPathMaskCacheHits++; } #if GR_TEST_UTILS void dump(SkString* out); void dumpKeyValuePairs(SkTArray* keys, SkTArray* values); #endif private: int fNumPathMasksGenerated{0}; int fNumPathMaskCacheHits{0}; #else // GR_GPU_STATS void incNumPathMasksGenerated() {} void incNumPathMasksCacheHits() {} #if GR_TEST_UTILS void dump(SkString*) {} void dumpKeyValuePairs(SkTArray* keys, SkTArray* values) {} #endif #endif // GR_GPU_STATS } fStats; Stats* stats() { return &fStats; } const Stats* stats() const { return &fStats; } void dumpJSON(SkJSONWriter*) const; private: OwnedArenas fArenas; std::unique_ptr fDrawingManager; std::unique_ptr fTextBlobCache; std::unique_ptr fAuditTrail; #ifdef GR_TEST_UTILS int fSuppressWarningMessages = 0; #endif typedef GrImageContext INHERITED; }; #endif