| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright 2012 Google Inc. | 2 * Copyright 2012 Google Inc. |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
| 5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
| 6 */ | 6 */ |
| 7 | 7 |
| 8 #include "Sk64.h" | 8 #include "Sk64.h" |
| 9 #include "SkLazyPixelRef.h" | 9 #include "SkLazyPixelRef.h" |
| 10 #include "SkColorTable.h" | 10 #include "SkColorTable.h" |
| 11 #include "SkData.h" | 11 #include "SkData.h" |
| 12 #include "SkImageCache.h" | 12 #include "SkImageCache.h" |
| 13 #include "SkImagePriv.h" | 13 #include "SkImagePriv.h" |
| 14 | 14 |
| 15 #if LAZY_CACHE_STATS | 15 #if LAZY_CACHE_STATS |
| 16 #include "SkThread.h" | 16 #include "SkThread.h" |
| 17 | 17 |
| 18 int32_t SkLazyPixelRef::gCacheHits; | 18 int32_t SkLazyPixelRef::gCacheHits; |
| 19 int32_t SkLazyPixelRef::gCacheMisses; | 19 int32_t SkLazyPixelRef::gCacheMisses; |
| 20 #endif | 20 #endif |
| 21 | 21 |
| 22 SkLazyPixelRef::SkLazyPixelRef(SkData* data, SkBitmapFactory::DecodeProc proc, S
kImageCache* cache) | 22 SkLazyPixelRef::SkLazyPixelRef(SkData* data, SkBitmapFactory::DecodeProc proc, S
kImageCache* cache) |
| 23 // Pass NULL for the Mutex so that the default (ring buffer) will be used. | 23 // Pass NULL for the Mutex so that the default (ring buffer) will be used. |
| 24 : INHERITED(NULL) | 24 : INHERITED(NULL) |
| 25 , fDecodeProc(proc) | 25 , fDecodeProc(proc) |
| 26 , fImageCache(cache) | 26 , fImageCache(cache) |
| 27 , fCacheId(SkImageCache::UNINITIALIZED_ID) { | 27 , fCacheId(SkImageCache::UNINITIALIZED_ID) |
| 28 , fRowBytes(0) { |
| 28 SkASSERT(fDecodeProc != NULL); | 29 SkASSERT(fDecodeProc != NULL); |
| 29 if (NULL == data) { | 30 if (NULL == data) { |
| 30 fData = SkData::NewEmpty(); | 31 fData = SkData::NewEmpty(); |
| 31 fErrorInDecoding = true; | 32 fErrorInDecoding = true; |
| 32 } else { | 33 } else { |
| 33 fData = data; | 34 fData = data; |
| 34 fData->ref(); | 35 fData->ref(); |
| 35 fErrorInDecoding = data->size() == 0; | 36 fErrorInDecoding = data->size() == 0; |
| 36 } | 37 } |
| 37 SkASSERT(cache != NULL); | 38 SkASSERT(cache != NULL); |
| (...skipping 26 matching lines...) Expand all Loading... |
| 64 | 65 |
| 65 void* SkLazyPixelRef::onLockPixels(SkColorTable**) { | 66 void* SkLazyPixelRef::onLockPixels(SkColorTable**) { |
| 66 if (fErrorInDecoding) { | 67 if (fErrorInDecoding) { |
| 67 return NULL; | 68 return NULL; |
| 68 } | 69 } |
| 69 SkBitmapFactory::Target target; | 70 SkBitmapFactory::Target target; |
| 70 // Check to see if the pixels still exist in the cache. | 71 // Check to see if the pixels still exist in the cache. |
| 71 if (SkImageCache::UNINITIALIZED_ID == fCacheId) { | 72 if (SkImageCache::UNINITIALIZED_ID == fCacheId) { |
| 72 target.fAddr = NULL; | 73 target.fAddr = NULL; |
| 73 } else { | 74 } else { |
| 74 target.fAddr = fImageCache->pinCache(fCacheId); | 75 SkImageCache::PurgeStatus status; |
| 75 if (NULL != target.fAddr) { | 76 target.fAddr = fImageCache->pinCache(fCacheId, &status); |
| 77 if (target.fAddr == NULL) { |
| 78 fCacheId = SkImageCache::UNINITIALIZED_ID; |
| 79 } else { |
| 80 if (SkImageCache::kNotPurged_PurgeStatus == status) { |
| 76 #if LAZY_CACHE_STATS | 81 #if LAZY_CACHE_STATS |
| 77 sk_atomic_inc(&gCacheHits); | 82 sk_atomic_inc(&gCacheHits); |
| 78 #endif | 83 #endif |
| 79 return target.fAddr; | 84 return target.fAddr; |
| 85 } |
| 86 SkASSERT(SkImageCache::kPurged_PurgeStatus == status); |
| 80 } | 87 } |
| 88 // Cache miss. Either pinCache returned NULL or it returned a memory add
ress without the old |
| 89 // data |
| 81 #if LAZY_CACHE_STATS | 90 #if LAZY_CACHE_STATS |
| 82 sk_atomic_inc(&gCacheMisses); | 91 sk_atomic_inc(&gCacheMisses); |
| 83 #endif | 92 #endif |
| 84 } | 93 } |
| 85 SkASSERT(NULL == target.fAddr); | |
| 86 SkImage::Info info; | 94 SkImage::Info info; |
| 87 SkASSERT(fData != NULL && fData->size() > 0); | 95 SkASSERT(fData != NULL && fData->size() > 0); |
| 88 // FIXME: As an optimization, only do this part once. | 96 if (NULL == target.fAddr) { |
| 89 fErrorInDecoding = !fDecodeProc(fData->data(), fData->size(), &info, NULL); | 97 // Determine the size of the image in order to determine how much memory
to allocate. |
| 90 if (fErrorInDecoding) { | 98 // FIXME: As an optimization, only do this part once. |
| 91 // In case a previous call to allocAndPinCache succeeded. | 99 fErrorInDecoding = !fDecodeProc(fData->data(), fData->size(), &info, NUL
L); |
| 92 fImageCache->throwAwayCache(fCacheId); | 100 if (fErrorInDecoding) { |
| 93 fCacheId = SkImageCache::UNINITIALIZED_ID; | 101 // We can only reach here if fCacheId was already set to UNINITIALIZ
ED_ID, or if |
| 94 return NULL; | 102 // pinCache returned NULL, in which case it was reset to UNINITIALIZ
ED_ID. |
| 103 SkASSERT(SkImageCache::UNINITIALIZED_ID == fCacheId); |
| 104 return NULL; |
| 105 } |
| 106 |
| 107 size_t bytes = ComputeMinRowBytesAndSize(info, &target.fRowBytes); |
| 108 target.fAddr = fImageCache->allocAndPinCache(bytes, &fCacheId); |
| 109 if (NULL == target.fAddr) { |
| 110 // Space could not be allocated. |
| 111 // Just like the last assert, fCacheId must be UNINITIALIZED_ID. |
| 112 SkASSERT(SkImageCache::UNINITIALIZED_ID == fCacheId); |
| 113 return NULL; |
| 114 } |
| 115 } else { |
| 116 // pinCache returned purged memory to which target.fAddr already points.
Set |
| 117 // target.fRowBytes properly. |
| 118 target.fRowBytes = fRowBytes; |
| 119 // Assume that the size is correct, since it was determined by this same
function |
| 120 // previously. |
| 95 } | 121 } |
| 96 // Allocate the memory. | 122 SkASSERT(target.fAddr != NULL); |
| 97 size_t bytes = ComputeMinRowBytesAndSize(info, &target.fRowBytes); | |
| 98 | |
| 99 target.fAddr = fImageCache->allocAndPinCache(bytes, &fCacheId); | |
| 100 if (NULL == target.fAddr) { | |
| 101 // Space could not be allocated. | |
| 102 fCacheId = SkImageCache::UNINITIALIZED_ID; | |
| 103 return NULL; | |
| 104 } | |
| 105 SkASSERT(SkImageCache::UNINITIALIZED_ID != fCacheId); | 123 SkASSERT(SkImageCache::UNINITIALIZED_ID != fCacheId); |
| 106 fErrorInDecoding = !fDecodeProc(fData->data(), fData->size(), &info, &target
); | 124 fErrorInDecoding = !fDecodeProc(fData->data(), fData->size(), &info, &target
); |
| 107 if (fErrorInDecoding) { | 125 if (fErrorInDecoding) { |
| 108 fImageCache->throwAwayCache(fCacheId); | 126 fImageCache->throwAwayCache(fCacheId); |
| 109 fCacheId = SkImageCache::UNINITIALIZED_ID; | 127 fCacheId = SkImageCache::UNINITIALIZED_ID; |
| 110 return NULL; | 128 return NULL; |
| 111 } | 129 } |
| 130 // Upon success, store fRowBytes so it can be used in case pinCache later re
turns purged memory. |
| 131 fRowBytes = target.fRowBytes; |
| 112 return target.fAddr; | 132 return target.fAddr; |
| 113 } | 133 } |
| 114 | 134 |
| 115 void SkLazyPixelRef::onUnlockPixels() { | 135 void SkLazyPixelRef::onUnlockPixels() { |
| 116 if (fErrorInDecoding) { | 136 if (fErrorInDecoding) { |
| 117 return; | 137 return; |
| 118 } | 138 } |
| 119 if (fCacheId != SkImageCache::UNINITIALIZED_ID) { | 139 if (fCacheId != SkImageCache::UNINITIALIZED_ID) { |
| 120 fImageCache->releaseCache(fCacheId); | 140 fImageCache->releaseCache(fCacheId); |
| 121 } | 141 } |
| 122 } | 142 } |
| 123 | 143 |
| 124 SkData* SkLazyPixelRef::onRefEncodedData() { | 144 SkData* SkLazyPixelRef::onRefEncodedData() { |
| 125 fData->ref(); | 145 fData->ref(); |
| 126 return fData; | 146 return fData; |
| 127 } | 147 } |
| OLD | NEW |