1 // Copyright (C) 2016 and later: Unicode, Inc. and others. 2 // License & terms of use: http://www.unicode.org/copyright.html 3 /* 4 ****************************************************************************** 5 * Copyright (C) 2015-2016, International Business Machines 6 * Corporation and others. All Rights Reserved. 7 ****************************************************************************** 8 * sharedobject.h 9 */ 10 11 #ifndef __SHAREDOBJECT_H__ 12 #define __SHAREDOBJECT_H__ 13 14 15 #include "unicode/uobject.h" 16 #include "umutex.h" 17 18 U_NAMESPACE_BEGIN 19 20 /** 21 * Base class for unified cache exposing enough methods to SharedObject 22 * instances to allow their addRef() and removeRef() methods to 23 * update cache metrics. No other part of ICU, except for SharedObject, 24 * should directly call the methods of this base class. 25 */ 26 class U_COMMON_API UnifiedCacheBase : public UObject { 27 public: UnifiedCacheBase()28 UnifiedCacheBase() { } 29 30 /** 31 * Called by addRefWhileHoldingCacheLock() when the hard reference count 32 * of its instance goes from 0 to 1. 33 */ 34 virtual void incrementItemsInUse() const = 0; 35 36 /** 37 * Called by removeRef() when the hard reference count of its instance 38 * drops from 1 to 0. 39 */ 40 virtual void decrementItemsInUseWithLockingAndEviction() const = 0; 41 42 /** 43 * Called by removeRefWhileHoldingCacheLock() when the hard reference 44 * count of its instance drops from 1 to 0. 45 */ 46 virtual void decrementItemsInUse() const = 0; 47 virtual ~UnifiedCacheBase(); 48 private: 49 UnifiedCacheBase(const UnifiedCacheBase &); 50 UnifiedCacheBase &operator=(const UnifiedCacheBase &); 51 }; 52 53 /** 54 * Base class for shared, reference-counted, auto-deleted objects. 55 * Subclasses can be immutable. 56 * If they are mutable, then they must implement their copy constructor 57 * so that copyOnWrite() works. 58 * 59 * Either stack-allocate, use LocalPointer, or use addRef()/removeRef(). 60 * Sharing requires reference-counting. 61 */ 62 class U_COMMON_API SharedObject : public UObject { 63 public: 64 /** Initializes totalRefCount, softRefCount to 0. */ SharedObject()65 SharedObject() : 66 totalRefCount(0), 67 softRefCount(0), 68 hardRefCount(0), 69 cachePtr(NULL) {} 70 71 /** Initializes totalRefCount, softRefCount to 0. */ SharedObject(const SharedObject & other)72 SharedObject(const SharedObject &other) : 73 UObject(other), 74 totalRefCount(0), 75 softRefCount(0), 76 hardRefCount(0), 77 cachePtr(NULL) {} 78 79 virtual ~SharedObject(); 80 81 /** 82 * Increments the number of references to this object. Thread-safe. 83 */ addRef()84 void addRef() const { addRef(FALSE); } 85 86 /** 87 * Increments the number of references to this object. 88 * Must be called only from within the internals of UnifiedCache and 89 * only while the cache global mutex is held. 90 */ addRefWhileHoldingCacheLock()91 void addRefWhileHoldingCacheLock() const { addRef(TRUE); } 92 93 /** 94 * Increments the number of soft references to this object. 95 * Must be called only from within the internals of UnifiedCache and 96 * only while the cache global mutex is held. 97 */ 98 void addSoftRef() const; 99 100 /** 101 * Decrements the number of references to this object. Thread-safe. 102 */ removeRef()103 void removeRef() const { removeRef(FALSE); } 104 105 /** 106 * Decrements the number of references to this object. 107 * Must be called only from within the internals of UnifiedCache and 108 * only while the cache global mutex is held. 109 */ removeRefWhileHoldingCacheLock()110 void removeRefWhileHoldingCacheLock() const { removeRef(TRUE); } 111 112 /** 113 * Decrements the number of soft references to this object. 114 * Must be called only from within the internals of UnifiedCache and 115 * only while the cache global mutex is held. 116 */ 117 void removeSoftRef() const; 118 119 /** 120 * Returns the reference counter including soft references. 121 * Uses a memory barrier. 122 */ 123 int32_t getRefCount() const; 124 125 /** 126 * Returns the count of soft references only. 127 * Must be called only from within the internals of UnifiedCache and 128 * only while the cache global mutex is held. 129 */ getSoftRefCount()130 int32_t getSoftRefCount() const { return softRefCount; } 131 132 /** 133 * Returns the count of hard references only. Uses a memory barrier. 134 * Used for testing the cache. Regular clients won't need this. 135 */ 136 int32_t getHardRefCount() const; 137 138 /** 139 * If noHardReferences() == TRUE then this object has no hard references. 140 * Must be called only from within the internals of UnifiedCache. 141 */ noHardReferences()142 inline UBool noHardReferences() const { return getHardRefCount() == 0; } 143 144 /** 145 * If hasHardReferences() == TRUE then this object has hard references. 146 * Must be called only from within the internals of UnifiedCache. 147 */ hasHardReferences()148 inline UBool hasHardReferences() const { return getHardRefCount() != 0; } 149 150 /** 151 * If noSoftReferences() == TRUE then this object has no soft references. 152 * Must be called only from within the internals of UnifiedCache and 153 * only while the cache global mutex is held. 154 */ noSoftReferences()155 UBool noSoftReferences() const { return (softRefCount == 0); } 156 157 /** 158 * Deletes this object if it has no references or soft references. 159 */ 160 void deleteIfZeroRefCount() const; 161 162 /** 163 * @internal For UnifedCache use only to register this object with itself. 164 * Must be called before this object is exposed to multiple threads. 165 */ registerWithCache(const UnifiedCacheBase * ptr)166 void registerWithCache(const UnifiedCacheBase *ptr) const { 167 cachePtr = ptr; 168 } 169 170 /** 171 * Returns a writable version of ptr. 172 * If there is exactly one owner, then ptr itself is returned as a 173 * non-const pointer. 174 * If there are multiple owners, then ptr is replaced with a 175 * copy-constructed clone, 176 * and that is returned. 177 * Returns NULL if cloning failed. 178 * 179 * T must be a subclass of SharedObject. 180 */ 181 template<typename T> copyOnWrite(const T * & ptr)182 static T *copyOnWrite(const T *&ptr) { 183 const T *p = ptr; 184 if(p->getRefCount() <= 1) { return const_cast<T *>(p); } 185 T *p2 = new T(*p); 186 if(p2 == NULL) { return NULL; } 187 p->removeRef(); 188 ptr = p2; 189 p2->addRef(); 190 return p2; 191 } 192 193 /** 194 * Makes dest an owner of the object pointed to by src while adjusting 195 * reference counts and deleting the previous object dest pointed to 196 * if necessary. Before this call is made, dest must either be NULL or 197 * be included in the reference count of the object it points to. 198 * 199 * T must be a subclass of SharedObject. 200 */ 201 template<typename T> copyPtr(const T * src,const T * & dest)202 static void copyPtr(const T *src, const T *&dest) { 203 if(src != dest) { 204 if(dest != NULL) { dest->removeRef(); } 205 dest = src; 206 if(src != NULL) { src->addRef(); } 207 } 208 } 209 210 /** 211 * Equivalent to copyPtr(NULL, dest). 212 */ 213 template<typename T> clearPtr(const T * & ptr)214 static void clearPtr(const T *&ptr) { 215 if (ptr != NULL) { 216 ptr->removeRef(); 217 ptr = NULL; 218 } 219 } 220 221 private: 222 mutable u_atomic_int32_t totalRefCount; 223 224 // Any thread modifying softRefCount must hold the global cache mutex 225 mutable int32_t softRefCount; 226 227 mutable u_atomic_int32_t hardRefCount; 228 mutable const UnifiedCacheBase *cachePtr; 229 void addRef(UBool withCacheLock) const; 230 void removeRef(UBool withCacheLock) const; 231 232 }; 233 234 U_NAMESPACE_END 235 236 #endif 237