blob: 0caf0b40ae87fe9f126596141888982c2e6cd6ab [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
reed@google.comac10a2d2010-12-22 21:39:39 +00002/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 * Copyright 2010 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
reed@google.comac10a2d2010-12-22 21:39:39 +00007 */
8
9
epoger@google.comec3ed6a2011-07-28 14:26:00 +000010
reed@google.comac10a2d2010-12-22 21:39:39 +000011#include "SkGrTexturePixelRef.h"
senorblanco@chromium.orgef843cd2011-12-02 19:11:17 +000012#include "GrContext.h"
reed@google.comac10a2d2010-12-22 21:39:39 +000013#include "GrTexture.h"
senorblanco@chromium.orgef843cd2011-12-02 19:11:17 +000014#include "SkGr.h"
bsalomon@google.com669fdc42011-04-05 17:08:27 +000015#include "SkRect.h"
reed@google.com9c49bc32011-07-07 13:42:37 +000016
17// since we call lockPixels recursively on fBitmap, we need a distinct mutex,
18// to avoid deadlock with the default one provided by SkPixelRef.
digit@google.com1771cbf2012-01-26 21:26:40 +000019SK_DECLARE_STATIC_MUTEX(gROLockPixelsPixelRefMutex);
reed@google.com9c49bc32011-07-07 13:42:37 +000020
21SkROLockPixelsPixelRef::SkROLockPixelsPixelRef() : INHERITED(&gROLockPixelsPixelRefMutex) {
22}
23
24SkROLockPixelsPixelRef::~SkROLockPixelsPixelRef() {
25}
26
27void* SkROLockPixelsPixelRef::onLockPixels(SkColorTable** ctable) {
28 if (ctable) {
29 *ctable = NULL;
30 }
31 fBitmap.reset();
32// SkDebugf("---------- calling readpixels in support of lockpixels\n");
33 if (!this->onReadPixels(&fBitmap, NULL)) {
34 SkDebugf("SkROLockPixelsPixelRef::onLockPixels failed!\n");
35 return NULL;
36 }
37 fBitmap.lockPixels();
38 return fBitmap.getPixels();
39}
40
41void SkROLockPixelsPixelRef::onUnlockPixels() {
42 fBitmap.unlockPixels();
43}
44
45bool SkROLockPixelsPixelRef::onLockPixelsAreWritable() const {
46 return false;
47}
48
49///////////////////////////////////////////////////////////////////////////////
bsalomon@google.com669fdc42011-04-05 17:08:27 +000050
senorblanco@chromium.orgef843cd2011-12-02 19:11:17 +000051static SkGrTexturePixelRef* copyToTexturePixelRef(GrTexture* texture,
52 SkBitmap::Config dstConfig) {
53 if (NULL == texture) {
54 return NULL;
55 }
56 GrContext* context = texture->getContext();
57 if (NULL == context) {
58 return NULL;
59 }
60 GrTextureDesc desc;
61
62 desc.fWidth = texture->width();
63 desc.fHeight = texture->height();
64 desc.fFlags = kRenderTarget_GrTextureFlagBit | kNoStencil_GrTextureFlagBit;
robertphillips@google.coma1e57952012-06-04 20:05:28 +000065 desc.fConfig = SkGr::BitmapConfig2PixelConfig(dstConfig);
senorblanco@chromium.orgef843cd2011-12-02 19:11:17 +000066
67 GrTexture* dst = context->createUncachedTexture(desc, NULL, 0);
68 if (NULL == dst) {
69 return NULL;
70 }
71
72 context->copyTexture(texture, dst->asRenderTarget());
73 SkGrTexturePixelRef* pixelRef = new SkGrTexturePixelRef(dst);
74 GrSafeUnref(dst);
75 return pixelRef;
76}
77
78///////////////////////////////////////////////////////////////////////////////
79
reed@google.comac10a2d2010-12-22 21:39:39 +000080SkGrTexturePixelRef::SkGrTexturePixelRef(GrTexture* tex) {
81 fTexture = tex;
82 GrSafeRef(tex);
83}
84
85SkGrTexturePixelRef::~SkGrTexturePixelRef() {
86 GrSafeUnref(fTexture);
87}
88
reed@google.com9c49bc32011-07-07 13:42:37 +000089SkGpuTexture* SkGrTexturePixelRef::getTexture() {
90 return (SkGpuTexture*)fTexture;
91}
92
senorblanco@chromium.orgef843cd2011-12-02 19:11:17 +000093SkPixelRef* SkGrTexturePixelRef::deepCopy(SkBitmap::Config dstConfig) {
94 return copyToTexturePixelRef(fTexture, dstConfig);
95}
96
reed@google.com50dfa012011-04-01 19:05:36 +000097bool SkGrTexturePixelRef::onReadPixels(SkBitmap* dst, const SkIRect* subset) {
bsalomon@google.com669fdc42011-04-05 17:08:27 +000098 if (NULL != fTexture && fTexture->isValid()) {
99 int left, top, width, height;
100 if (NULL != subset) {
101 left = subset->fLeft;
102 width = subset->width();
103 top = subset->fTop;
104 height = subset->height();
105 } else {
106 left = 0;
107 width = fTexture->width();
108 top = 0;
109 height = fTexture->height();
110 }
111 dst->setConfig(SkBitmap::kARGB_8888_Config, width, height);
112 dst->allocPixels();
113 SkAutoLockPixels al(*dst);
114 void* buffer = dst->getPixels();
115 return fTexture->readPixels(left, top, width, height,
bsalomon@google.comc4364992011-11-07 15:54:49 +0000116 kSkia8888_PM_GrPixelConfig,
bsalomon@google.com6f379512011-11-16 20:36:03 +0000117 buffer, dst->rowBytes());
bsalomon@google.com669fdc42011-04-05 17:08:27 +0000118 } else {
119 return false;
120 }
reed@google.com50dfa012011-04-01 19:05:36 +0000121}
reed@google.comac10a2d2010-12-22 21:39:39 +0000122
reed@google.com9c49bc32011-07-07 13:42:37 +0000123///////////////////////////////////////////////////////////////////////////////
bsalomon@google.com669fdc42011-04-05 17:08:27 +0000124
125SkGrRenderTargetPixelRef::SkGrRenderTargetPixelRef(GrRenderTarget* rt) {
126 fRenderTarget = rt;
127 GrSafeRef(fRenderTarget);
128}
129
130SkGrRenderTargetPixelRef::~SkGrRenderTargetPixelRef() {
131 GrSafeUnref(fRenderTarget);
132}
133
134SkGpuTexture* SkGrRenderTargetPixelRef::getTexture() {
135 if (NULL != fRenderTarget) {
136 return (SkGpuTexture*) fRenderTarget->asTexture();
137 }
138 return NULL;
139}
140
senorblanco@chromium.orgef843cd2011-12-02 19:11:17 +0000141SkPixelRef* SkGrRenderTargetPixelRef::deepCopy(SkBitmap::Config dstConfig) {
142 if (NULL == fRenderTarget) {
143 return NULL;
144 }
145 // Note that when copying an SkGrRenderTargetPixelRef, we actually
146 // return an SkGrTexturePixelRef instead. This is because
147 // SkGrRenderTargetPixelRef is usually created in conjunction with
148 // GrTexture owned elsewhere (e.g., SkGpuDevice), and cannot live
149 // independently of that texture. SkGrTexturePixelRef, on the other
150 // hand, owns its own GrTexture, and is thus self-contained.
151 return copyToTexturePixelRef(fRenderTarget->asTexture(), dstConfig);
152}
153
bsalomon@google.com669fdc42011-04-05 17:08:27 +0000154bool SkGrRenderTargetPixelRef::onReadPixels(SkBitmap* dst, const SkIRect* subset) {
155 if (NULL != fRenderTarget && fRenderTarget->isValid()) {
156 int left, top, width, height;
157 if (NULL != subset) {
158 left = subset->fLeft;
159 width = subset->width();
160 top = subset->fTop;
161 height = subset->height();
162 } else {
163 left = 0;
164 width = fRenderTarget->width();
165 top = 0;
166 height = fRenderTarget->height();
167 }
168 dst->setConfig(SkBitmap::kARGB_8888_Config, width, height);
169 dst->allocPixels();
170 SkAutoLockPixels al(*dst);
171 void* buffer = dst->getPixels();
172 return fRenderTarget->readPixels(left, top, width, height,
bsalomon@google.comc4364992011-11-07 15:54:49 +0000173 kSkia8888_PM_GrPixelConfig,
bsalomon@google.com6f379512011-11-16 20:36:03 +0000174 buffer, dst->rowBytes());
bsalomon@google.com669fdc42011-04-05 17:08:27 +0000175 } else {
176 return false;
177 }
reed@google.com4281d652011-04-08 18:54:20 +0000178}
reed@google.com9c49bc32011-07-07 13:42:37 +0000179