blob: f0d7affdc8fe985a62629e784cff84d9e7e361b9 [file] [log] [blame]
halcanary@google.comad04eb42013-11-21 15:32:08 +00001/*
2 * Copyright 2013 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 "SkDiscardablePixelRef.h"
9#include "SkDiscardableMemory.h"
halcanary@google.comedd370f2013-12-10 21:11:12 +000010#include "SkImageGenerator.h"
halcanary@google.comad04eb42013-11-21 15:32:08 +000011
reed@google.combf790232013-12-13 19:45:58 +000012SkDiscardablePixelRef::SkDiscardablePixelRef(const SkImageInfo& info,
13 SkImageGenerator* generator,
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000014 size_t rowBytes,
15 SkDiscardableMemory::Factory* fact)
reed@google.combf790232013-12-13 19:45:58 +000016 : INHERITED(info)
17 , fGenerator(generator)
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000018 , fDMFactory(fact)
halcanary@google.comad04eb42013-11-21 15:32:08 +000019 , fRowBytes(rowBytes)
reed@google.combf790232013-12-13 19:45:58 +000020 , fDiscardableMemory(NULL)
21{
halcanary@google.comad04eb42013-11-21 15:32:08 +000022 SkASSERT(fGenerator != NULL);
halcanary@google.comad04eb42013-11-21 15:32:08 +000023 SkASSERT(fRowBytes > 0);
commit-bot@chromium.orgdd597992013-12-02 22:32:54 +000024 // The SkImageGenerator contract requires fGenerator to always
25 // decode the same image on each call to getPixels().
26 this->setImmutable();
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000027 SkSafeRef(fDMFactory);
halcanary@google.comad04eb42013-11-21 15:32:08 +000028}
29
30SkDiscardablePixelRef::~SkDiscardablePixelRef() {
reed@google.com1d0654f2013-12-12 22:37:32 +000031 if (this->isLocked()) {
32 fDiscardableMemory->unlock();
33 }
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000034 SkDELETE(fDiscardableMemory);
35 SkSafeUnref(fDMFactory);
halcanary@google.comad04eb42013-11-21 15:32:08 +000036 SkDELETE(fGenerator);
37}
38
reed@google.comd0419b12014-01-06 17:08:27 +000039bool SkDiscardablePixelRef::onNewLockPixels(LockRec* rec) {
halcanary@google.comad04eb42013-11-21 15:32:08 +000040 if (fDiscardableMemory != NULL) {
41 if (fDiscardableMemory->lock()) {
reed@google.comd0419b12014-01-06 17:08:27 +000042 rec->fPixels = fDiscardableMemory->data();
43 rec->fColorTable = NULL;
44 rec->fRowBytes = fRowBytes;
45 return true;
halcanary@google.comad04eb42013-11-21 15:32:08 +000046 }
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000047 SkDELETE(fDiscardableMemory);
halcanary@google.comad04eb42013-11-21 15:32:08 +000048 fDiscardableMemory = NULL;
49 }
skia.committer@gmail.com98272d92014-01-04 07:01:40 +000050
reed@google.combf790232013-12-13 19:45:58 +000051 const size_t size = this->info().getSafeSize(fRowBytes);
52
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000053 if (fDMFactory != NULL) {
reed@google.combf790232013-12-13 19:45:58 +000054 fDiscardableMemory = fDMFactory->create(size);
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000055 } else {
reed@google.combf790232013-12-13 19:45:58 +000056 fDiscardableMemory = SkDiscardableMemory::Create(size);
halcanary@google.com2c7c7ee2013-12-05 18:31:42 +000057 }
halcanary@google.comad04eb42013-11-21 15:32:08 +000058 if (NULL == fDiscardableMemory) {
reed@google.comd0419b12014-01-06 17:08:27 +000059 return false; // Memory allocation failed.
halcanary@google.comad04eb42013-11-21 15:32:08 +000060 }
reed@google.comd0419b12014-01-06 17:08:27 +000061
halcanary@google.comad04eb42013-11-21 15:32:08 +000062 void* pixels = fDiscardableMemory->data();
commit-bot@chromium.org00f8d6c2014-05-29 15:57:20 +000063 const SkImageInfo& info = this->info();
64 SkPMColor colors[256];
65 int colorCount = 0;
66
reed@google.com672e7cf2014-05-29 17:10:54 +000067#ifdef SK_SUPPORT_LEGACY_IMAGEGENERATORAPI
68 if (!fGenerator->getPixels(info, pixels, fRowBytes)) {
69#else
commit-bot@chromium.org00f8d6c2014-05-29 15:57:20 +000070 if (!fGenerator->getPixels(info, pixels, fRowBytes, colors, &colorCount)) {
reed@google.com672e7cf2014-05-29 17:10:54 +000071#endif
reed@google.comc83a91f2013-12-13 13:41:14 +000072 fDiscardableMemory->unlock();
73 SkDELETE(fDiscardableMemory);
74 fDiscardableMemory = NULL;
reed@google.comd0419b12014-01-06 17:08:27 +000075 return false;
halcanary@google.comad04eb42013-11-21 15:32:08 +000076 }
reed@google.comd0419b12014-01-06 17:08:27 +000077
commit-bot@chromium.org00f8d6c2014-05-29 15:57:20 +000078 // Note: our ctable is not purgable, as it is not stored in the discardablememory block.
79 // This is because SkColorTable is refcntable, and therefore our caller could hold onto it
80 // beyond the scope of a lock/unlock. If we change the API/lifecycle for SkColorTable, we
81 // could move it into the block, but then again perhaps it is small enough that this doesn't
82 // really matter.
83 if (colorCount > 0) {
84 fCTable.reset(SkNEW_ARGS(SkColorTable, (colors, colorCount)));
85 } else {
86 fCTable.reset(NULL);
87 }
88
reed@google.comd0419b12014-01-06 17:08:27 +000089 rec->fPixels = pixels;
commit-bot@chromium.org00f8d6c2014-05-29 15:57:20 +000090 rec->fColorTable = fCTable.get();
reed@google.comd0419b12014-01-06 17:08:27 +000091 rec->fRowBytes = fRowBytes;
92 return true;
halcanary@google.comad04eb42013-11-21 15:32:08 +000093}
reed@google.comd0419b12014-01-06 17:08:27 +000094
halcanary@google.comad04eb42013-11-21 15:32:08 +000095void SkDiscardablePixelRef::onUnlockPixels() {
reed@google.comc83a91f2013-12-13 13:41:14 +000096 fDiscardableMemory->unlock();
halcanary@google.comad04eb42013-11-21 15:32:08 +000097}
98
commit-bot@chromium.org2d970b52014-05-27 14:14:22 +000099bool SkInstallDiscardablePixelRef(SkImageGenerator* generator, SkBitmap* dst,
halcanary@google.comedd370f2013-12-10 21:11:12 +0000100 SkDiscardableMemory::Factory* factory) {
halcanary@google.comad04eb42013-11-21 15:32:08 +0000101 SkImageInfo info;
halcanary@google.com3d50ea12014-01-02 13:15:13 +0000102 SkAutoTDelete<SkImageGenerator> autoGenerator(generator);
103 if ((NULL == autoGenerator.get())
104 || (!autoGenerator->getInfo(&info))
commit-bot@chromium.org986d6812014-05-29 20:47:53 +0000105 || (!dst->setInfo(info))) {
halcanary@google.comad04eb42013-11-21 15:32:08 +0000106 return false;
107 }
reed@google.com900ecf22014-02-20 20:55:37 +0000108 SkASSERT(dst->colorType() != kUnknown_SkColorType);
halcanary@google.com3d50ea12014-01-02 13:15:13 +0000109 if (dst->empty()) { // Use a normal pixelref.
reed@google.com9ebcac52014-01-24 18:53:42 +0000110 return dst->allocPixels();
halcanary@google.comedd370f2013-12-10 21:11:12 +0000111 }
halcanary@google.com3d50ea12014-01-02 13:15:13 +0000112 SkAutoTUnref<SkDiscardablePixelRef> ref(
113 SkNEW_ARGS(SkDiscardablePixelRef,
114 (info, autoGenerator.detach(), dst->rowBytes(), factory)));
commit-bot@chromium.orge13af712014-01-13 20:39:14 +0000115 dst->setPixelRef(ref);
halcanary@google.comad04eb42013-11-21 15:32:08 +0000116 return true;
117}
commit-bot@chromium.org2d970b52014-05-27 14:14:22 +0000118
119// This is the public API
120bool SkInstallDiscardablePixelRef(SkImageGenerator* generator, SkBitmap* dst) {
121 return SkInstallDiscardablePixelRef(generator, dst, NULL);
122}