1 /*
2  * Copyright 2011 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 "SkBitmapCache.h"
9 #include "SkMutex.h"
10 #include "SkPixelRef.h"
11 #include "SkTraceEvent.h"
12 
13 //#define SK_SUPPORT_LEGACY_UNBALANCED_PIXELREF_LOCKCOUNT
14 //#define SK_TRACE_PIXELREF_LIFETIME
15 
16 #include "SkNextID.h"
17 
ImageID()18 uint32_t SkNextID::ImageID() {
19     static uint32_t gID = 0;
20     uint32_t id;
21     // Loop in case our global wraps around, as we never want to return a 0.
22     do {
23         id = sk_atomic_fetch_add(&gID, 2u) + 2;  // Never set the low bit.
24     } while (0 == id);
25     return id;
26 }
27 
28 ///////////////////////////////////////////////////////////////////////////////
29 
30 // just need a > 0 value, so pick a funny one to aid in debugging
31 #define SKPIXELREF_PRELOCKED_LOCKCOUNT     123456789
32 
validate_info(const SkImageInfo & info)33 static SkImageInfo validate_info(const SkImageInfo& info) {
34     SkAlphaType newAlphaType = info.alphaType();
35     SkAssertResult(SkColorTypeValidateAlphaType(info.colorType(), info.alphaType(), &newAlphaType));
36     return info.makeAlphaType(newAlphaType);
37 }
38 
39 #ifdef SK_TRACE_PIXELREF_LIFETIME
40     static int32_t gInstCounter;
41 #endif
42 
SkPixelRef(const SkImageInfo & info)43 SkPixelRef::SkPixelRef(const SkImageInfo& info)
44     : fInfo(validate_info(info))
45 #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
46     , fStableID(SkNextID::ImageID())
47 #endif
48 
49 {
50 #ifdef SK_TRACE_PIXELREF_LIFETIME
51     SkDebugf(" pixelref %d\n", sk_atomic_inc(&gInstCounter));
52 #endif
53     fRec.zero();
54     fLockCount = 0;
55     this->needsNewGenID();
56     fMutability = kMutable;
57     fPreLocked = false;
58     fAddedToCache.store(false);
59 }
60 
~SkPixelRef()61 SkPixelRef::~SkPixelRef() {
62 #ifndef SK_SUPPORT_LEGACY_UNBALANCED_PIXELREF_LOCKCOUNT
63     SkASSERT(SKPIXELREF_PRELOCKED_LOCKCOUNT == fLockCount || 0 == fLockCount);
64 #endif
65 
66 #ifdef SK_TRACE_PIXELREF_LIFETIME
67     SkDebugf("~pixelref %d\n", sk_atomic_dec(&gInstCounter) - 1);
68 #endif
69     this->callGenIDChangeListeners();
70 }
71 
needsNewGenID()72 void SkPixelRef::needsNewGenID() {
73     fTaggedGenID.store(0);
74     SkASSERT(!this->genIDIsUnique()); // This method isn't threadsafe, so the assert should be fine.
75 }
76 
cloneGenID(const SkPixelRef & that)77 void SkPixelRef::cloneGenID(const SkPixelRef& that) {
78     // This is subtle.  We must call that.getGenerationID() to make sure its genID isn't 0.
79     uint32_t genID = that.getGenerationID();
80 
81     // Neither ID is unique any more.
82     // (These & ~1u are actually redundant.  that.getGenerationID() just did it for us.)
83     this->fTaggedGenID.store(genID & ~1u);
84     that. fTaggedGenID.store(genID & ~1u);
85 
86     // This method isn't threadsafe, so these asserts should be fine.
87     SkASSERT(!this->genIDIsUnique());
88     SkASSERT(!that. genIDIsUnique());
89 }
90 
validate_pixels_ctable(const SkImageInfo & info,const SkColorTable * ctable)91 static void validate_pixels_ctable(const SkImageInfo& info, const SkColorTable* ctable) {
92     if (info.isEmpty()) {
93         return; // can't require ctable if the dimensions are empty
94     }
95     if (kIndex_8_SkColorType == info.colorType()) {
96         SkASSERT(ctable);
97     } else {
98         SkASSERT(nullptr == ctable);
99     }
100 }
101 
setPreLocked(void * pixels,size_t rowBytes,SkColorTable * ctable)102 void SkPixelRef::setPreLocked(void* pixels, size_t rowBytes, SkColorTable* ctable) {
103     SkASSERT(pixels);
104     validate_pixels_ctable(fInfo, ctable);
105     // only call me in your constructor, otherwise fLockCount tracking can get
106     // out of sync.
107     fRec.fPixels = pixels;
108     fRec.fColorTable = ctable;
109     fRec.fRowBytes = rowBytes;
110     fLockCount = SKPIXELREF_PRELOCKED_LOCKCOUNT;
111     fPreLocked = true;
112 }
113 
114 // Increments fLockCount only on success
lockPixelsInsideMutex()115 bool SkPixelRef::lockPixelsInsideMutex() {
116     fMutex.assertHeld();
117 
118     if (1 == ++fLockCount) {
119         SkASSERT(fRec.isZero());
120         if (!this->onNewLockPixels(&fRec)) {
121             fRec.zero();
122             fLockCount -= 1;    // we return fLockCount unchanged if we fail.
123             return false;
124         }
125     }
126     if (fRec.fPixels) {
127         validate_pixels_ctable(fInfo, fRec.fColorTable);
128         return true;
129     }
130     // no pixels, so we failed (somehow)
131     --fLockCount;
132     return false;
133 }
134 
135 // For historical reasons, we always inc fLockCount, even if we return false.
136 // It would be nice to change this (it seems), and only inc if we actually succeed...
lockPixels()137 bool SkPixelRef::lockPixels() {
138     SkASSERT(!fPreLocked || SKPIXELREF_PRELOCKED_LOCKCOUNT == fLockCount);
139 
140     if (!fPreLocked) {
141         TRACE_EVENT_BEGIN0("skia", "SkPixelRef::lockPixelsMutex");
142         SkAutoMutexAcquire  ac(fMutex);
143         TRACE_EVENT_END0("skia", "SkPixelRef::lockPixelsMutex");
144         SkDEBUGCODE(int oldCount = fLockCount;)
145         bool success = this->lockPixelsInsideMutex();
146         // lockPixelsInsideMutex only increments the count if it succeeds.
147         SkASSERT(oldCount + (int)success == fLockCount);
148 
149         if (!success) {
150             // For compatibility with SkBitmap calling lockPixels, we still want to increment
151             // fLockCount even if we failed. If we updated SkBitmap we could remove this oddity.
152             fLockCount += 1;
153             return false;
154         }
155     }
156     if (fRec.fPixels) {
157         validate_pixels_ctable(fInfo, fRec.fColorTable);
158         return true;
159     }
160     return false;
161 }
162 
lockPixels(LockRec * rec)163 bool SkPixelRef::lockPixels(LockRec* rec) {
164     if (this->lockPixels()) {
165         *rec = fRec;
166         return true;
167     }
168     return false;
169 }
170 
unlockPixels()171 void SkPixelRef::unlockPixels() {
172     SkASSERT(!fPreLocked || SKPIXELREF_PRELOCKED_LOCKCOUNT == fLockCount);
173 
174     if (!fPreLocked) {
175         SkAutoMutexAcquire  ac(fMutex);
176 
177         SkASSERT(fLockCount > 0);
178         if (0 == --fLockCount) {
179             // don't call onUnlockPixels unless onLockPixels succeeded
180             if (fRec.fPixels) {
181                 this->onUnlockPixels();
182                 fRec.zero();
183             } else {
184                 SkASSERT(fRec.isZero());
185             }
186         }
187     }
188 }
189 
requestLock(const LockRequest & request,LockResult * result)190 bool SkPixelRef::requestLock(const LockRequest& request, LockResult* result) {
191     SkASSERT(result);
192     if (request.fSize.isEmpty()) {
193         return false;
194     }
195     // until we support subsets, we have to check this...
196     if (request.fSize.width() != fInfo.width() || request.fSize.height() != fInfo.height()) {
197         return false;
198     }
199 
200     if (fPreLocked) {
201         result->fUnlockProc = nullptr;
202         result->fUnlockContext = nullptr;
203         result->fCTable = fRec.fColorTable;
204         result->fPixels = fRec.fPixels;
205         result->fRowBytes = fRec.fRowBytes;
206         result->fSize.set(fInfo.width(), fInfo.height());
207     } else {
208         SkAutoMutexAcquire  ac(fMutex);
209         if (!this->onRequestLock(request, result)) {
210             return false;
211         }
212     }
213     if (result->fPixels) {
214         validate_pixels_ctable(fInfo, result->fCTable);
215         return true;
216     }
217     return false;
218 }
219 
lockPixelsAreWritable() const220 bool SkPixelRef::lockPixelsAreWritable() const {
221     return this->onLockPixelsAreWritable();
222 }
223 
onLockPixelsAreWritable() const224 bool SkPixelRef::onLockPixelsAreWritable() const {
225     return true;
226 }
227 
getGenerationID() const228 uint32_t SkPixelRef::getGenerationID() const {
229     uint32_t id = fTaggedGenID.load();
230     if (0 == id) {
231         uint32_t next = SkNextID::ImageID() | 1u;
232         if (fTaggedGenID.compare_exchange(&id, next)) {
233             id = next;  // There was no race or we won the race.  fTaggedGenID is next now.
234         } else {
235             // We lost a race to set fTaggedGenID. compare_exchange() filled id with the winner.
236         }
237         // We can't quite SkASSERT(this->genIDIsUnique()). It could be non-unique
238         // if we got here via the else path (pretty unlikely, but possible).
239     }
240     return id & ~1u;  // Mask off bottom unique bit.
241 }
242 
addGenIDChangeListener(GenIDChangeListener * listener)243 void SkPixelRef::addGenIDChangeListener(GenIDChangeListener* listener) {
244     if (nullptr == listener || !this->genIDIsUnique()) {
245         // No point in tracking this if we're not going to call it.
246         delete listener;
247         return;
248     }
249     *fGenIDChangeListeners.append() = listener;
250 }
251 
252 // we need to be called *before* the genID gets changed or zerod
callGenIDChangeListeners()253 void SkPixelRef::callGenIDChangeListeners() {
254     // We don't invalidate ourselves if we think another SkPixelRef is sharing our genID.
255     if (this->genIDIsUnique()) {
256         for (int i = 0; i < fGenIDChangeListeners.count(); i++) {
257             fGenIDChangeListeners[i]->onChange();
258         }
259 
260         // TODO: SkAtomic could add "old_value = atomic.xchg(new_value)" to make this clearer.
261         if (fAddedToCache.load()) {
262             SkNotifyBitmapGenIDIsStale(this->getGenerationID());
263             fAddedToCache.store(false);
264         }
265     }
266     // Listeners get at most one shot, so whether these triggered or not, blow them away.
267     fGenIDChangeListeners.deleteAll();
268 }
269 
notifyPixelsChanged()270 void SkPixelRef::notifyPixelsChanged() {
271 #ifdef SK_DEBUG
272     if (this->isImmutable()) {
273         SkDebugf("========== notifyPixelsChanged called on immutable pixelref");
274     }
275 #endif
276     this->callGenIDChangeListeners();
277     this->needsNewGenID();
278     this->onNotifyPixelsChanged();
279 }
280 
changeAlphaType(SkAlphaType at)281 void SkPixelRef::changeAlphaType(SkAlphaType at) {
282     *const_cast<SkImageInfo*>(&fInfo) = fInfo.makeAlphaType(at);
283 }
284 
setImmutable()285 void SkPixelRef::setImmutable() {
286     fMutability = kImmutable;
287 }
288 
setImmutableWithID(uint32_t genID)289 void SkPixelRef::setImmutableWithID(uint32_t genID) {
290     /*
291      *  We are forcing the genID to match an external value. The caller must ensure that this
292      *  value does not conflict with other content.
293      *
294      *  One use is to force this pixelref's id to match an SkImage's id
295      */
296     fMutability = kImmutable;
297     fTaggedGenID.store(genID);
298 }
299 
setTemporarilyImmutable()300 void SkPixelRef::setTemporarilyImmutable() {
301     SkASSERT(fMutability != kImmutable);
302     fMutability = kTemporarilyImmutable;
303 }
304 
restoreMutability()305 void SkPixelRef::restoreMutability() {
306     SkASSERT(fMutability != kImmutable);
307     fMutability = kMutable;
308 }
309 
310 ///////////////////////////////////////////////////////////////////////////////////////////////////
311 
312 
onNotifyPixelsChanged()313 void SkPixelRef::onNotifyPixelsChanged() { }
314 
getAllocatedSizeInBytes() const315 size_t SkPixelRef::getAllocatedSizeInBytes() const {
316     return 0;
317 }
318 
unlock_legacy_result(void * ctx)319 static void unlock_legacy_result(void* ctx) {
320     SkPixelRef* pr = (SkPixelRef*)ctx;
321     pr->unlockPixels();
322     pr->unref();    // balancing the Ref in onRequestLoc
323 }
324 
onRequestLock(const LockRequest & request,LockResult * result)325 bool SkPixelRef::onRequestLock(const LockRequest& request, LockResult* result) {
326     if (!this->lockPixelsInsideMutex()) {
327         return false;
328     }
329 
330     result->fUnlockProc = unlock_legacy_result;
331     result->fUnlockContext = SkRef(this);   // this is balanced in our fUnlockProc
332     result->fCTable = fRec.fColorTable;
333     result->fPixels = fRec.fPixels;
334     result->fRowBytes = fRec.fRowBytes;
335     result->fSize.set(fInfo.width(), fInfo.height());
336     return true;
337 }
338