blob: 5ebf150a846769269adea89d61145f8d845d4e75 [file] [log] [blame]
bsalomon95740982014-09-04 13:12:37 -07001/*
2 * Copyright 2014 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#ifndef GrProgramElement_DEFINED
9#define GrProgramElement_DEFINED
10
11#include "SkRefCnt.h"
12#include "SkTArray.h"
13
bsalomonf96ba022014-09-17 08:05:40 -070014class GrGpuResourceRef;
bsalomon95740982014-09-04 13:12:37 -070015
16/**
joshualittb0a8a372014-09-23 09:50:21 -070017 * Base class for GrProcessor. GrDrawState uses this to manage
18 * transitioning a GrProcessor from being owned by a client to being scheduled for execution. It
bsalomon95740982014-09-04 13:12:37 -070019 * converts resources owned by the effect from being ref'ed to having pending reads/writes.
20 *
21 * All GrGpuResource objects owned by a GrProgramElement or derived classes (either directly or
bsalomonf96ba022014-09-17 08:05:40 -070022 * indirectly) must be wrapped in a GrGpuResourceRef and registered with the GrProgramElement using
23 * addGpuResource(). This allows the regular refs to be converted to pending IO events
bsalomon95740982014-09-04 13:12:37 -070024 * when the program element is scheduled for deferred execution.
25 */
26class GrProgramElement : public SkNoncopyable {
27public:
mtklein6f076652015-01-13 08:22:43 -080028 SK_DECLARE_INST_COUNT(GrProgramElement)
bsalomon95740982014-09-04 13:12:37 -070029
30 virtual ~GrProgramElement() {
31 // fRefCnt can be one when an effect is created statically using GR_CREATE_STATIC_EFFECT
32 SkASSERT((0 == fRefCnt || 1 == fRefCnt) && 0 == fPendingExecutions);
33 // Set to invalid values.
34 SkDEBUGCODE(fRefCnt = fPendingExecutions = -10;)
35 }
36
37 void ref() const {
bsalomonae59b772014-11-19 08:23:49 -080038 this->validate();
bsalomon95740982014-09-04 13:12:37 -070039 // Once the ref cnt reaches zero it should never be ref'ed again.
40 SkASSERT(fRefCnt > 0);
bsalomon95740982014-09-04 13:12:37 -070041 ++fRefCnt;
bsalomonae59b772014-11-19 08:23:49 -080042 this->validate();
bsalomon95740982014-09-04 13:12:37 -070043 }
44
45 void unref() const {
46 this->validate();
47 --fRefCnt;
bsalomond0128772014-10-03 05:31:41 -070048 if (0 == fRefCnt) {
49 if (0 == fPendingExecutions) {
50 SkDELETE(this);
bsalomonae59b772014-11-19 08:23:49 -080051 return;
bsalomond0128772014-10-03 05:31:41 -070052 } else {
53 this->removeRefs();
54 }
bsalomon95740982014-09-04 13:12:37 -070055 }
bsalomonae59b772014-11-19 08:23:49 -080056 this->validate();
bsalomon95740982014-09-04 13:12:37 -070057 }
58
bsalomon52e9d632014-09-05 12:23:12 -070059 /**
60 * Gets an id that is unique for this GrProgramElement object. This will never return 0.
61 */
62 uint32_t getUniqueID() const { return fUniqueID; }
63
bsalomon95740982014-09-04 13:12:37 -070064 void validate() const {
65#ifdef SK_DEBUG
66 SkASSERT(fRefCnt >= 0);
67 SkASSERT(fPendingExecutions >= 0);
68 SkASSERT(fRefCnt + fPendingExecutions > 0);
69#endif
70 }
71
72protected:
bsalomon52e9d632014-09-05 12:23:12 -070073 GrProgramElement() : fRefCnt(1), fPendingExecutions(0), fUniqueID(CreateUniqueID()) {}
bsalomon95740982014-09-04 13:12:37 -070074
75 /** Subclasses registers their resources using this function. It is assumed the GrProgramResouce
76 is and will remain owned by the subclass and this function will retain a raw ptr. Once a
bsalomonf96ba022014-09-17 08:05:40 -070077 GrGpuResourceRef is registered its setResource must not be called.
bsalomon95740982014-09-04 13:12:37 -070078 */
bsalomonf96ba022014-09-17 08:05:40 -070079 void addGpuResource(const GrGpuResourceRef* res) {
80 fGpuResources.push_back(res);
bsalomon95740982014-09-04 13:12:37 -070081 }
82
83private:
bsalomon52e9d632014-09-05 12:23:12 -070084 static uint32_t CreateUniqueID();
85
bsalomonae59b772014-11-19 08:23:49 -080086 void addPendingExecution() const {
87 this->validate();
88 SkASSERT(fRefCnt > 0);
89 if (0 == fPendingExecutions) {
90 this->addPendingIOs();
91 }
92 ++fPendingExecutions;
93 this->validate();
94 }
bsalomon95740982014-09-04 13:12:37 -070095
bsalomonae59b772014-11-19 08:23:49 -080096 void completedExecution() const {
97 this->validate();
98 --fPendingExecutions;
99 if (0 == fPendingExecutions) {
100 if (0 == fRefCnt) {
101 SkDELETE(this);
102 return;
103 } else {
104 this->pendingIOComplete();
105 }
106 }
107 this->validate();
108 }
bsalomon95740982014-09-04 13:12:37 -0700109
bsalomond0128772014-10-03 05:31:41 -0700110 void removeRefs() const;
bsalomonae59b772014-11-19 08:23:49 -0800111 void addPendingIOs() const;
112 void pendingIOComplete() const;
bsalomond0128772014-10-03 05:31:41 -0700113
bsalomon95740982014-09-04 13:12:37 -0700114 mutable int32_t fRefCnt;
115 // Count of deferred executions not yet issued to the 3D API.
116 mutable int32_t fPendingExecutions;
bsalomon52e9d632014-09-05 12:23:12 -0700117 uint32_t fUniqueID;
bsalomon95740982014-09-04 13:12:37 -0700118
bsalomonf96ba022014-09-17 08:05:40 -0700119 SkSTArray<4, const GrGpuResourceRef*, true> fGpuResources;
bsalomon95740982014-09-04 13:12:37 -0700120
bsalomonae59b772014-11-19 08:23:49 -0800121 // Only this class can access addPendingExecution() and completedExecution().
122 template <typename T> friend class GrPendingProgramElement;
bsalomon95740982014-09-04 13:12:37 -0700123
124 typedef SkNoncopyable INHERITED;
125};
126
127#endif