blob: 8f6e3089765f6bf2d19bed86ad36e14b69f8b8f6 [file] [log] [blame]
reed@google.comc9062042012-07-30 18:06:00 +00001/*
2 * Copyright 2012 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
mtklein59c12e32016-05-02 07:19:41 -07008#include "SkAtomics.h"
reed@google.com9ea5a3b2012-07-30 21:03:46 +00009#include "SkSurface_Base.h"
reed@google.com889b09e2012-07-27 21:10:42 +000010#include "SkImagePriv.h"
11#include "SkCanvas.h"
12
reed4a8126e2014-09-22 07:29:03 -070013#include "SkFontLCDConfig.h"
14static SkPixelGeometry compute_default_geometry() {
15 SkFontLCDConfig::LCDOrder order = SkFontLCDConfig::GetSubpixelOrder();
16 if (SkFontLCDConfig::kNONE_LCDOrder == order) {
17 return kUnknown_SkPixelGeometry;
18 } else {
19 // Bit0 is RGB(0), BGR(1)
20 // Bit1 is H(0), V(1)
21 const SkPixelGeometry gGeo[] = {
22 kRGB_H_SkPixelGeometry,
23 kBGR_H_SkPixelGeometry,
24 kRGB_V_SkPixelGeometry,
25 kBGR_V_SkPixelGeometry,
26 };
27 int index = 0;
28 if (SkFontLCDConfig::kBGR_LCDOrder == order) {
29 index |= 1;
30 }
31 if (SkFontLCDConfig::kVertical_LCDOrientation == SkFontLCDConfig::GetSubpixelOrientation()){
32 index |= 2;
33 }
34 return gGeo[index];
35 }
36}
37
38SkSurfaceProps::SkSurfaceProps() : fFlags(0), fPixelGeometry(kUnknown_SkPixelGeometry) {}
39
40SkSurfaceProps::SkSurfaceProps(InitType) : fFlags(0), fPixelGeometry(compute_default_geometry()) {}
41
42SkSurfaceProps::SkSurfaceProps(uint32_t flags, InitType)
43 : fFlags(flags)
44 , fPixelGeometry(compute_default_geometry())
45{}
46
47SkSurfaceProps::SkSurfaceProps(uint32_t flags, SkPixelGeometry pg)
48 : fFlags(flags), fPixelGeometry(pg)
49{}
50
reed4af267b2014-11-21 08:46:37 -080051SkSurfaceProps::SkSurfaceProps(const SkSurfaceProps& other)
52 : fFlags(other.fFlags)
53 , fPixelGeometry(other.fPixelGeometry)
54{}
55
reed@google.com889b09e2012-07-27 21:10:42 +000056///////////////////////////////////////////////////////////////////////////////
57
reed4a8126e2014-09-22 07:29:03 -070058SkSurface_Base::SkSurface_Base(int width, int height, const SkSurfaceProps* props)
59 : INHERITED(width, height, props)
60{
halcanary96fcdcc2015-08-27 07:41:13 -070061 fCachedCanvas = nullptr;
62 fCachedImage = nullptr;
reed@google.com9ea5a3b2012-07-30 21:03:46 +000063}
reed@google.com889b09e2012-07-27 21:10:42 +000064
reed4a8126e2014-09-22 07:29:03 -070065SkSurface_Base::SkSurface_Base(const SkImageInfo& info, const SkSurfaceProps* props)
66 : INHERITED(info, props)
67{
halcanary96fcdcc2015-08-27 07:41:13 -070068 fCachedCanvas = nullptr;
69 fCachedImage = nullptr;
reed@google.com1360c522014-01-08 21:25:26 +000070}
71
reed@google.com9ea5a3b2012-07-30 21:03:46 +000072SkSurface_Base::~SkSurface_Base() {
reed@google.com97af1a62012-08-28 12:19:02 +000073 // in case the canvas outsurvives us, we null the callback
74 if (fCachedCanvas) {
halcanary96fcdcc2015-08-27 07:41:13 -070075 fCachedCanvas->setSurfaceBase(nullptr);
reed@google.com97af1a62012-08-28 12:19:02 +000076 }
77
78 SkSafeUnref(fCachedImage);
reed@google.com9ea5a3b2012-07-30 21:03:46 +000079 SkSafeUnref(fCachedCanvas);
80}
reed@google.com889b09e2012-07-27 21:10:42 +000081
reed4a8126e2014-09-22 07:29:03 -070082void SkSurface_Base::onDraw(SkCanvas* canvas, SkScalar x, SkScalar y, const SkPaint* paint) {
reed9ce9d672016-03-17 10:51:11 -070083 auto image = this->makeImageSnapshot(SkBudgeted::kYes);
reed@google.com889b09e2012-07-27 21:10:42 +000084 if (image) {
piotaixrb5fae932014-09-24 13:03:30 -070085 canvas->drawImage(image, x, y, paint);
reed@google.com889b09e2012-07-27 21:10:42 +000086 }
87}
88
reedc83a2972015-07-16 07:40:45 -070089bool SkSurface_Base::outstandingImageSnapshot() const {
90 return fCachedImage && !fCachedImage->unique();
91}
92
commit-bot@chromium.orgc4c98702013-04-22 14:28:01 +000093void SkSurface_Base::aboutToDraw(ContentChangeMode mode) {
reed@google.com97af1a62012-08-28 12:19:02 +000094 this->dirtyGenerationID();
95
commit-bot@chromium.org28361fa2014-03-28 16:08:05 +000096 SkASSERT(!fCachedCanvas || fCachedCanvas->getSurfaceBase() == this);
reed@google.com97af1a62012-08-28 12:19:02 +000097
bsalomon49f085d2014-09-05 13:34:00 -070098 if (fCachedImage) {
reed@google.com97af1a62012-08-28 12:19:02 +000099 // the surface may need to fork its backend, if its sharing it with
100 // the cached image. Note: we only call if there is an outstanding owner
101 // on the image (besides us).
reed26e0e582015-07-29 11:44:52 -0700102 bool unique = fCachedImage->unique();
103 if (!unique) {
commit-bot@chromium.orgc4c98702013-04-22 14:28:01 +0000104 this->onCopyOnWrite(mode);
reed@google.com97af1a62012-08-28 12:19:02 +0000105 }
106
107 // regardless of copy-on-write, we must drop our cached image now, so
108 // that the next request will get our new contents.
109 fCachedImage->unref();
halcanary96fcdcc2015-08-27 07:41:13 -0700110 fCachedImage = nullptr;
reed26e0e582015-07-29 11:44:52 -0700111
112 if (unique) {
113 // Our content isn't held by any image now, so we can consider that content mutable.
114 // Raster surfaces need to be told it's safe to consider its pixels mutable again.
115 // We make this call after the ->unref() so the subclass can assert there are no images.
116 this->onRestoreBackingMutability();
117 }
commit-bot@chromium.org28361fa2014-03-28 16:08:05 +0000118 } else if (kDiscard_ContentChangeMode == mode) {
119 this->onDiscard();
reed@google.com97af1a62012-08-28 12:19:02 +0000120 }
121}
122
123uint32_t SkSurface_Base::newGenerationID() {
commit-bot@chromium.org28361fa2014-03-28 16:08:05 +0000124 SkASSERT(!fCachedCanvas || fCachedCanvas->getSurfaceBase() == this);
reed@google.com97af1a62012-08-28 12:19:02 +0000125 static int32_t gID;
126 return sk_atomic_inc(&gID) + 1;
127}
128
reed@google.com889b09e2012-07-27 21:10:42 +0000129static SkSurface_Base* asSB(SkSurface* surface) {
130 return static_cast<SkSurface_Base*>(surface);
131}
132
133///////////////////////////////////////////////////////////////////////////////
134
reed4a8126e2014-09-22 07:29:03 -0700135SkSurface::SkSurface(int width, int height, const SkSurfaceProps* props)
136 : fProps(SkSurfacePropsCopyOrDefault(props)), fWidth(width), fHeight(height)
137{
reedb2497c22014-12-31 12:31:43 -0800138 SkASSERT(fWidth > 0);
139 SkASSERT(fHeight > 0);
reed@google.com1360c522014-01-08 21:25:26 +0000140 fGenerationID = 0;
141}
142
reed4a8126e2014-09-22 07:29:03 -0700143SkSurface::SkSurface(const SkImageInfo& info, const SkSurfaceProps* props)
144 : fProps(SkSurfacePropsCopyOrDefault(props)), fWidth(info.width()), fHeight(info.height())
145{
reedb2497c22014-12-31 12:31:43 -0800146 SkASSERT(fWidth > 0);
147 SkASSERT(fHeight > 0);
reed@google.com889b09e2012-07-27 21:10:42 +0000148 fGenerationID = 0;
149}
150
reed@google.com97af1a62012-08-28 12:19:02 +0000151uint32_t SkSurface::generationID() {
152 if (0 == fGenerationID) {
153 fGenerationID = asSB(this)->newGenerationID();
154 }
155 return fGenerationID;
156}
157
commit-bot@chromium.orgc4c98702013-04-22 14:28:01 +0000158void SkSurface::notifyContentWillChange(ContentChangeMode mode) {
159 asSB(this)->aboutToDraw(mode);
reed@google.com97af1a62012-08-28 12:19:02 +0000160}
161
reed@google.com9ea5a3b2012-07-30 21:03:46 +0000162SkCanvas* SkSurface::getCanvas() {
163 return asSB(this)->getCachedCanvas();
reed@google.com889b09e2012-07-27 21:10:42 +0000164}
165
reed9ce9d672016-03-17 10:51:11 -0700166sk_sp<SkImage> SkSurface::makeImageSnapshot(SkBudgeted budgeted) {
bsalomonf47b9a32016-02-22 11:02:58 -0800167 // the caller will call unref() to balance this
168 return asSB(this)->refCachedImage(budgeted, kNo_ForceUnique);
169}
170
reed9ce9d672016-03-17 10:51:11 -0700171sk_sp<SkImage> SkSurface::makeImageSnapshot(SkBudgeted budgeted, ForceUnique unique) {
bsalomonf47b9a32016-02-22 11:02:58 -0800172 // the caller will call unref() to balance this
173 return asSB(this)->refCachedImage(budgeted, unique);
reed@google.com889b09e2012-07-27 21:10:42 +0000174}
175
reede8f30622016-03-23 18:59:25 -0700176sk_sp<SkSurface> SkSurface::makeSurface(const SkImageInfo& info) {
mike@reedtribe.orgb9476252012-11-15 02:37:45 +0000177 return asSB(this)->onNewSurface(info);
reed@google.com889b09e2012-07-27 21:10:42 +0000178}
179
180void SkSurface::draw(SkCanvas* canvas, SkScalar x, SkScalar y,
181 const SkPaint* paint) {
182 return asSB(this)->onDraw(canvas, x, y, paint);
183}
commit-bot@chromium.orgc3bd8af2014-02-13 17:14:46 +0000184
reed6ceeebd2016-03-09 14:26:26 -0800185bool SkSurface::peekPixels(SkPixmap* pmap) {
186 return this->getCanvas()->peekPixels(pmap);
commit-bot@chromium.orgc3bd8af2014-02-13 17:14:46 +0000187}
reed4a8126e2014-09-22 07:29:03 -0700188
reed6ceeebd2016-03-09 14:26:26 -0800189#ifdef SK_SUPPORT_LEGACY_PEEKPIXELS_PARMS
190const void* SkSurface::peekPixels(SkImageInfo* info, size_t* rowBytes) {
191 SkPixmap pm;
192 if (this->peekPixels(&pm)) {
193 if (info) {
194 *info = pm.info();
195 }
196 if (rowBytes) {
197 *rowBytes = pm.rowBytes();
198 }
199 return pm.addr();
200 }
201 return nullptr;
202}
203#endif
204
reed7543aa22014-12-09 14:39:44 -0800205bool SkSurface::readPixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRowBytes,
206 int srcX, int srcY) {
207 return this->getCanvas()->readPixels(dstInfo, dstPixels, dstRowBytes, srcX, srcY);
208}
209
joshualitt81793412015-07-08 12:54:04 -0700210GrBackendObject SkSurface::getTextureHandle(BackendHandleAccess access) {
reedfa5e68e2015-06-29 07:37:01 -0700211 return asSB(this)->onGetTextureHandle(access);
212}
213
joshualitt81793412015-07-08 12:54:04 -0700214bool SkSurface::getRenderTargetHandle(GrBackendObject* obj, BackendHandleAccess access) {
215 return asSB(this)->onGetRenderTargetHandle(obj, access);
216}
217
ericrkf7b8b8a2016-02-24 14:49:51 -0800218void SkSurface::prepareForExternalIO() {
219 asSB(this)->onPrepareForExternalIO();
220}
221
reed4a8126e2014-09-22 07:29:03 -0700222//////////////////////////////////////////////////////////////////////////////////////
reed4a8126e2014-09-22 07:29:03 -0700223
224#if !SK_SUPPORT_GPU
225
reede8f30622016-03-23 18:59:25 -0700226sk_sp<SkSurface> SkSurface::MakeRenderTargetDirect(GrRenderTarget*, const SkSurfaceProps*) {
halcanary96fcdcc2015-08-27 07:41:13 -0700227 return nullptr;
reed4a8126e2014-09-22 07:29:03 -0700228}
229
reede8f30622016-03-23 18:59:25 -0700230sk_sp<SkSurface> SkSurface::MakeRenderTarget(GrContext*, SkBudgeted, const SkImageInfo&, int,
bsalomon7e68ab72016-04-13 14:29:25 -0700231 const SkSurfaceProps*) {
halcanary96fcdcc2015-08-27 07:41:13 -0700232 return nullptr;
reed4a8126e2014-09-22 07:29:03 -0700233}
234
reede8f30622016-03-23 18:59:25 -0700235sk_sp<SkSurface> SkSurface::MakeFromBackendTexture(GrContext*, const GrBackendTextureDesc&,
236 const SkSurfaceProps*) {
halcanary96fcdcc2015-08-27 07:41:13 -0700237 return nullptr;
bsalomone4579ad2015-04-08 08:38:40 -0700238}
reed4a8126e2014-09-22 07:29:03 -0700239
reede8f30622016-03-23 18:59:25 -0700240sk_sp<SkSurface> SkSurface::MakeFromBackendRenderTarget(GrContext*,
241 const GrBackendRenderTargetDesc&,
242 const SkSurfaceProps*) {
halcanary96fcdcc2015-08-27 07:41:13 -0700243 return nullptr;
bsalomond3e259a2015-06-30 12:04:40 -0700244}
245
reede8f30622016-03-23 18:59:25 -0700246sk_sp<SkSurface> MakeFromBackendTextureAsRenderTarget(GrContext*, const GrBackendTextureDesc&,
247 const SkSurfaceProps*) {
ericrkf7b8b8a2016-02-24 14:49:51 -0800248 return nullptr;
249}
250
reed4a8126e2014-09-22 07:29:03 -0700251#endif