blob: 3ca5747252bd147da14c18ec8f5494bca99291c7 [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
reed@google.com9ea5a3b2012-07-30 21:03:46 +00008#include "SkSurface_Base.h"
reed@google.com889b09e2012-07-27 21:10:42 +00009#include "SkImagePriv.h"
10#include "SkCanvas.h"
11
reed4a8126e2014-09-22 07:29:03 -070012#include "SkFontLCDConfig.h"
13static SkPixelGeometry compute_default_geometry() {
14 SkFontLCDConfig::LCDOrder order = SkFontLCDConfig::GetSubpixelOrder();
15 if (SkFontLCDConfig::kNONE_LCDOrder == order) {
16 return kUnknown_SkPixelGeometry;
17 } else {
18 // Bit0 is RGB(0), BGR(1)
19 // Bit1 is H(0), V(1)
20 const SkPixelGeometry gGeo[] = {
21 kRGB_H_SkPixelGeometry,
22 kBGR_H_SkPixelGeometry,
23 kRGB_V_SkPixelGeometry,
24 kBGR_V_SkPixelGeometry,
25 };
26 int index = 0;
27 if (SkFontLCDConfig::kBGR_LCDOrder == order) {
28 index |= 1;
29 }
30 if (SkFontLCDConfig::kVertical_LCDOrientation == SkFontLCDConfig::GetSubpixelOrientation()){
31 index |= 2;
32 }
33 return gGeo[index];
34 }
35}
36
37SkSurfaceProps::SkSurfaceProps() : fFlags(0), fPixelGeometry(kUnknown_SkPixelGeometry) {}
38
39SkSurfaceProps::SkSurfaceProps(InitType) : fFlags(0), fPixelGeometry(compute_default_geometry()) {}
40
41SkSurfaceProps::SkSurfaceProps(uint32_t flags, InitType)
42 : fFlags(flags)
43 , fPixelGeometry(compute_default_geometry())
44{}
45
46SkSurfaceProps::SkSurfaceProps(uint32_t flags, SkPixelGeometry pg)
47 : fFlags(flags), fPixelGeometry(pg)
48{}
49
reed4af267b2014-11-21 08:46:37 -080050SkSurfaceProps::SkSurfaceProps(const SkSurfaceProps& other)
51 : fFlags(other.fFlags)
52 , fPixelGeometry(other.fPixelGeometry)
53{}
54
reed@google.com889b09e2012-07-27 21:10:42 +000055///////////////////////////////////////////////////////////////////////////////
56
reed4a8126e2014-09-22 07:29:03 -070057SkSurface_Base::SkSurface_Base(int width, int height, const SkSurfaceProps* props)
58 : INHERITED(width, height, props)
59{
reed@google.com9ea5a3b2012-07-30 21:03:46 +000060 fCachedCanvas = NULL;
reed@google.com97af1a62012-08-28 12:19:02 +000061 fCachedImage = NULL;
reed@google.com9ea5a3b2012-07-30 21:03:46 +000062}
reed@google.com889b09e2012-07-27 21:10:42 +000063
reed4a8126e2014-09-22 07:29:03 -070064SkSurface_Base::SkSurface_Base(const SkImageInfo& info, const SkSurfaceProps* props)
65 : INHERITED(info, props)
66{
reed@google.com1360c522014-01-08 21:25:26 +000067 fCachedCanvas = NULL;
68 fCachedImage = NULL;
69}
70
reed@google.com9ea5a3b2012-07-30 21:03:46 +000071SkSurface_Base::~SkSurface_Base() {
reed@google.com97af1a62012-08-28 12:19:02 +000072 // in case the canvas outsurvives us, we null the callback
73 if (fCachedCanvas) {
74 fCachedCanvas->setSurfaceBase(NULL);
75 }
76
77 SkSafeUnref(fCachedImage);
reed@google.com9ea5a3b2012-07-30 21:03:46 +000078 SkSafeUnref(fCachedCanvas);
79}
reed@google.com889b09e2012-07-27 21:10:42 +000080
reed4a8126e2014-09-22 07:29:03 -070081void SkSurface_Base::onDraw(SkCanvas* canvas, SkScalar x, SkScalar y, const SkPaint* paint) {
junov@chromium.org5ee449a2013-04-12 20:20:50 +000082 SkImage* image = this->newImageSnapshot();
reed@google.com889b09e2012-07-27 21:10:42 +000083 if (image) {
piotaixrb5fae932014-09-24 13:03:30 -070084 canvas->drawImage(image, x, y, paint);
reed@google.com889b09e2012-07-27 21:10:42 +000085 image->unref();
86 }
87}
88
commit-bot@chromium.orgc4c98702013-04-22 14:28:01 +000089void SkSurface_Base::aboutToDraw(ContentChangeMode mode) {
reed@google.com97af1a62012-08-28 12:19:02 +000090 this->dirtyGenerationID();
91
commit-bot@chromium.org28361fa2014-03-28 16:08:05 +000092 SkASSERT(!fCachedCanvas || fCachedCanvas->getSurfaceBase() == this);
reed@google.com97af1a62012-08-28 12:19:02 +000093
bsalomon49f085d2014-09-05 13:34:00 -070094 if (fCachedImage) {
reed@google.com97af1a62012-08-28 12:19:02 +000095 // the surface may need to fork its backend, if its sharing it with
96 // the cached image. Note: we only call if there is an outstanding owner
97 // on the image (besides us).
bungeman@google.comf64c6842013-07-19 23:18:52 +000098 if (!fCachedImage->unique()) {
commit-bot@chromium.orgc4c98702013-04-22 14:28:01 +000099 this->onCopyOnWrite(mode);
reed@google.com97af1a62012-08-28 12:19:02 +0000100 }
101
102 // regardless of copy-on-write, we must drop our cached image now, so
103 // that the next request will get our new contents.
104 fCachedImage->unref();
105 fCachedImage = NULL;
commit-bot@chromium.org28361fa2014-03-28 16:08:05 +0000106 } else if (kDiscard_ContentChangeMode == mode) {
107 this->onDiscard();
reed@google.com97af1a62012-08-28 12:19:02 +0000108 }
109}
110
111uint32_t SkSurface_Base::newGenerationID() {
commit-bot@chromium.org28361fa2014-03-28 16:08:05 +0000112 SkASSERT(!fCachedCanvas || fCachedCanvas->getSurfaceBase() == this);
reed@google.com97af1a62012-08-28 12:19:02 +0000113 static int32_t gID;
114 return sk_atomic_inc(&gID) + 1;
115}
116
reed@google.com889b09e2012-07-27 21:10:42 +0000117static SkSurface_Base* asSB(SkSurface* surface) {
118 return static_cast<SkSurface_Base*>(surface);
119}
120
121///////////////////////////////////////////////////////////////////////////////
122
reed4a8126e2014-09-22 07:29:03 -0700123SkSurface::SkSurface(int width, int height, const SkSurfaceProps* props)
124 : fProps(SkSurfacePropsCopyOrDefault(props)), fWidth(width), fHeight(height)
125{
reed@google.com1360c522014-01-08 21:25:26 +0000126 SkASSERT(fWidth >= 0);
127 SkASSERT(fHeight >= 0);
128 fGenerationID = 0;
129}
130
reed4a8126e2014-09-22 07:29:03 -0700131SkSurface::SkSurface(const SkImageInfo& info, const SkSurfaceProps* props)
132 : fProps(SkSurfacePropsCopyOrDefault(props)), fWidth(info.width()), fHeight(info.height())
133{
reed@google.com1360c522014-01-08 21:25:26 +0000134 SkASSERT(fWidth >= 0);
135 SkASSERT(fHeight >= 0);
reed@google.com889b09e2012-07-27 21:10:42 +0000136 fGenerationID = 0;
137}
138
reed@google.com97af1a62012-08-28 12:19:02 +0000139uint32_t SkSurface::generationID() {
140 if (0 == fGenerationID) {
141 fGenerationID = asSB(this)->newGenerationID();
142 }
143 return fGenerationID;
144}
145
commit-bot@chromium.orgc4c98702013-04-22 14:28:01 +0000146void SkSurface::notifyContentWillChange(ContentChangeMode mode) {
147 asSB(this)->aboutToDraw(mode);
reed@google.com97af1a62012-08-28 12:19:02 +0000148}
149
reed@google.com9ea5a3b2012-07-30 21:03:46 +0000150SkCanvas* SkSurface::getCanvas() {
151 return asSB(this)->getCachedCanvas();
reed@google.com889b09e2012-07-27 21:10:42 +0000152}
153
junov@chromium.org5ee449a2013-04-12 20:20:50 +0000154SkImage* SkSurface::newImageSnapshot() {
reed@google.com97af1a62012-08-28 12:19:02 +0000155 SkImage* image = asSB(this)->getCachedImage();
156 SkSafeRef(image); // the caller will call unref() to balance this
157 return image;
reed@google.com889b09e2012-07-27 21:10:42 +0000158}
159
reed@google.com2bd8b812013-11-01 13:46:54 +0000160SkSurface* SkSurface::newSurface(const SkImageInfo& info) {
mike@reedtribe.orgb9476252012-11-15 02:37:45 +0000161 return asSB(this)->onNewSurface(info);
reed@google.com889b09e2012-07-27 21:10:42 +0000162}
163
164void SkSurface::draw(SkCanvas* canvas, SkScalar x, SkScalar y,
165 const SkPaint* paint) {
166 return asSB(this)->onDraw(canvas, x, y, paint);
167}
commit-bot@chromium.orgc3bd8af2014-02-13 17:14:46 +0000168
169const void* SkSurface::peekPixels(SkImageInfo* info, size_t* rowBytes) {
170 return this->getCanvas()->peekPixels(info, rowBytes);
171}
reed4a8126e2014-09-22 07:29:03 -0700172
173//////////////////////////////////////////////////////////////////////////////////////
174#ifdef SK_SUPPORT_LEGACY_TEXTRENDERMODE
175
176static SkSurfaceProps make_props(SkSurface::TextRenderMode trm) {
177 uint32_t propsFlags = 0;
178 if (SkSurface::kDistanceField_TextRenderMode == trm) {
179 propsFlags |= SkSurfaceProps::kUseDistanceFieldFonts_Flag;
180 }
181 return SkSurfaceProps(propsFlags, SkSurfaceProps::kLegacyFontHost_InitType);
182}
183
184SkSurface* SkSurface::NewRenderTargetDirect(GrRenderTarget* target, TextRenderMode trm) {
185 SkSurfaceProps props = make_props(trm);
186 return NewRenderTargetDirect(target, &props);
187}
188
189SkSurface* SkSurface::NewRenderTarget(GrContext* gr, const SkImageInfo& info, int sampleCount,
190 TextRenderMode trm) {
191 SkSurfaceProps props = make_props(trm);
192 return NewRenderTarget(gr, info, sampleCount, &props);
193}
194
195SkSurface* SkSurface::NewScratchRenderTarget(GrContext* gr, const SkImageInfo& info, int sampleCount,
196 TextRenderMode trm) {
197 SkSurfaceProps props = make_props(trm);
198 return NewScratchRenderTarget(gr, info, sampleCount, &props);
199}
200
201#endif
202
203#if !SK_SUPPORT_GPU
204
205SkSurface* SkSurface::NewRenderTargetDirect(GrRenderTarget*, const SkSurfaceProps*) {
206 return NULL;
207}
208
209SkSurface* SkSurface::NewRenderTarget(GrContext*, const SkImageInfo&, int, const SkSurfaceProps*) {
210 return NULL;
211}
212
213SkSurface* SkSurface::NewScratchRenderTarget(GrContext*, const SkImageInfo&, int sampleCount,
214 const SkSurfaceProps*) {
215 return NULL;
216}
217
218#endif