1 /* 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 #include "GrGpuResourceRef.h" 9 10 GrGpuResourceRef::GrGpuResourceRef() { 11 fResource = nullptr; 12 fOwnRef = false; 13 fPendingIO = false; 14 } 15 16 GrGpuResourceRef::GrGpuResourceRef(GrGpuResource* resource, GrIOType ioType) { 17 fResource = nullptr; 18 fOwnRef = false; 19 fPendingIO = false; 20 this->setResource(resource, ioType); 21 } 22 23 GrGpuResourceRef::~GrGpuResourceRef() { 24 if (fOwnRef) { 25 SkASSERT(fResource); 26 fResource->unref(); 27 } 28 if (fPendingIO) { 29 switch (fIOType) { 30 case kRead_GrIOType: 31 fResource->completedRead(); 32 break; 33 case kWrite_GrIOType: 34 fResource->completedWrite(); 35 break; 36 case kRW_GrIOType: 37 fResource->completedRead(); 38 fResource->completedWrite(); 39 break; 40 } 41 } 42 } 43 44 void GrGpuResourceRef::reset() { 45 SkASSERT(!fPendingIO); 46 SkASSERT(SkToBool(fResource) == fOwnRef); 47 if (fOwnRef) { 48 fResource->unref(); 49 fOwnRef = false; 50 fResource = nullptr; 51 } 52 } 53 54 void GrGpuResourceRef::setResource(GrGpuResource* resource, GrIOType ioType) { 55 SkASSERT(!fPendingIO); 56 SkASSERT(SkToBool(fResource) == fOwnRef); 57 SkSafeUnref(fResource); 58 if (nullptr == resource) { 59 fResource = nullptr; 60 fOwnRef = false; 61 } else { 62 fResource = resource; 63 fOwnRef = true; 64 fIOType = ioType; 65 } 66 } 67 68 void GrGpuResourceRef::markPendingIO() const { 69 if (!fResource) { 70 return; 71 } 72 73 // This should only be called when the owning GrProgramElement gets its first 74 // pendingExecution ref. 75 SkASSERT(!fPendingIO); 76 fPendingIO = true; 77 switch (fIOType) { 78 case kRead_GrIOType: 79 fResource->addPendingRead(); 80 break; 81 case kWrite_GrIOType: 82 fResource->addPendingWrite(); 83 break; 84 case kRW_GrIOType: 85 fResource->addPendingRead(); 86 fResource->addPendingWrite(); 87 break; 88 } 89 } 90 91 void GrGpuResourceRef::pendingIOComplete() const { 92 if (!fResource) { 93 return; 94 } 95 96 // This should only be called when the owner's pending executions have ocurred but it is still 97 // reffed. 98 SkASSERT(fOwnRef); 99 SkASSERT(fPendingIO); 100 switch (fIOType) { 101 case kRead_GrIOType: 102 fResource->completedRead(); 103 break; 104 case kWrite_GrIOType: 105 fResource->completedWrite(); 106 break; 107 case kRW_GrIOType: 108 fResource->completedRead(); 109 fResource->completedWrite(); 110 break; 111 112 } 113 fPendingIO = false; 114 } 115 116 void GrGpuResourceRef::removeRef() const { 117 if (!fResource) { 118 return; 119 } 120 121 // This should only be called once, when the owners last ref goes away and 122 // there is a pending execution. 123 SkASSERT(fOwnRef); 124 SkASSERT(fPendingIO); 125 fResource->unref(); 126 fOwnRef = false; 127 } 128