| /* |
| * Copyright 2013 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SkDiscardablePixelRef.h" |
| #include "SkDiscardableMemory.h" |
| #include "SkImageGenerator.h" |
| |
| SkDiscardablePixelRef::SkDiscardablePixelRef(SkImageGenerator* generator, |
| const SkImageInfo& info, |
| size_t size, |
| size_t rowBytes, |
| SkDiscardableMemory::Factory* fact) |
| : fGenerator(generator) |
| , fDMFactory(fact) |
| , fInfo(info) |
| , fSize(size) |
| , fRowBytes(rowBytes) |
| , fDiscardableMemory(NULL) { |
| SkASSERT(fGenerator != NULL); |
| SkASSERT(fSize > 0); |
| SkASSERT(fRowBytes > 0); |
| // The SkImageGenerator contract requires fGenerator to always |
| // decode the same image on each call to getPixels(). |
| this->setImmutable(); |
| SkSafeRef(fDMFactory); |
| } |
| |
| SkDiscardablePixelRef::~SkDiscardablePixelRef() { |
| if (this->isLocked()) { |
| fDiscardableMemory->unlock(); |
| } |
| SkDELETE(fDiscardableMemory); |
| SkSafeUnref(fDMFactory); |
| SkDELETE(fGenerator); |
| } |
| |
| void* SkDiscardablePixelRef::onLockPixels(SkColorTable**) { |
| if (fDiscardableMemory != NULL) { |
| if (fDiscardableMemory->lock()) { |
| return fDiscardableMemory->data(); |
| } |
| SkDELETE(fDiscardableMemory); |
| fDiscardableMemory = NULL; |
| } |
| if (fDMFactory != NULL) { |
| fDiscardableMemory = fDMFactory->create(fSize); |
| } else { |
| fDiscardableMemory = SkDiscardableMemory::Create(fSize); |
| } |
| if (NULL == fDiscardableMemory) { |
| return NULL; // Memory allocation failed. |
| } |
| void* pixels = fDiscardableMemory->data(); |
| if (!fGenerator->getPixels(fInfo, pixels, fRowBytes)) { |
| fDiscardableMemory->unlock(); |
| SkDELETE(fDiscardableMemory); |
| fDiscardableMemory = NULL; |
| return NULL; |
| } |
| return pixels; |
| } |
| void SkDiscardablePixelRef::onUnlockPixels() { |
| fDiscardableMemory->unlock(); |
| } |
| |
| bool SkInstallDiscardablePixelRef(SkImageGenerator* generator, |
| SkBitmap* dst, |
| SkDiscardableMemory::Factory* factory) { |
| SkImageInfo info; |
| SkASSERT(generator != NULL); |
| if ((NULL == generator) |
| || (!generator->getInfo(&info)) |
| || (!dst->setConfig(info, 0))) { |
| SkDELETE(generator); |
| return false; |
| } |
| SkASSERT(dst->config() != SkBitmap::kNo_Config); |
| if (dst->empty()) { // Use a normal pixelref. |
| SkDELETE(generator); // Do not need this anymore. |
| return dst->allocPixels(NULL, NULL); |
| } |
| SkAutoTUnref<SkDiscardablePixelRef> ref(SkNEW_ARGS(SkDiscardablePixelRef, |
| (generator, info, |
| dst->getSize(), |
| dst->rowBytes(), |
| factory))); |
| dst->setPixelRef(ref); |
| return true; |
| } |