blob: eda89a07681a6d56670a31bb50214aa673c87d66 [file] [log] [blame]
caryclark@google.com411bb722012-11-06 21:29:16 +00001/*
keyar@chromium.org451bb9f2012-07-26 17:27:57 +00002 * 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
8#ifndef PictureRenderer_DEFINED
9#define PictureRenderer_DEFINED
scroggo@google.coma62da2f2012-11-02 21:28:12 +000010
11#include "SkCountdown.h"
caryclark@google.coma3622372012-11-06 21:26:13 +000012#include "SkDrawFilter.h"
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +000013#include "SkMath.h"
reed@google.comea6a3062012-11-06 22:14:54 +000014#include "SkPaint.h"
scroggo@google.com9a412522012-09-07 15:21:18 +000015#include "SkPicture.h"
scroggo@google.comacfb30e2012-09-18 14:32:35 +000016#include "SkRect.h"
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000017#include "SkRefCnt.h"
scroggo@google.coma62da2f2012-11-02 21:28:12 +000018#include "SkRunnable.h"
scroggo@google.com9a412522012-09-07 15:21:18 +000019#include "SkString.h"
scroggo@google.coma62da2f2012-11-02 21:28:12 +000020#include "SkTDArray.h"
21#include "SkThreadPool.h"
22#include "SkTypes.h"
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000023
keyar@chromium.org06125642012-08-20 15:03:33 +000024#if SK_SUPPORT_GPU
25#include "GrContextFactory.h"
26#include "GrContext.h"
27#endif
28
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000029class SkBitmap;
30class SkCanvas;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000031class SkGLContext;
scroggo@google.coma62da2f2012-11-02 21:28:12 +000032class SkThread;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000033
34namespace sk_tools {
35
36class PictureRenderer : public SkRefCnt {
37public:
keyar@chromium.orgc81686c2012-08-20 15:04:04 +000038 enum SkDeviceTypes {
39 kBitmap_DeviceType,
40#if SK_SUPPORT_GPU
41 kGPU_DeviceType
42#endif
43 };
44
junov@chromium.org9313ca42012-11-02 18:11:49 +000045 enum BBoxHierarchyType {
46 kNone_BBoxHierarchyType = 0,
47 kRTree_BBoxHierarchyType,
junov@chromium.org7b537062012-11-06 18:58:43 +000048 kTileGrid_BBoxHierarchyType,
junov@chromium.org9313ca42012-11-02 18:11:49 +000049 };
50
caryclark@google.coma3622372012-11-06 21:26:13 +000051 // this uses SkPaint::Flags as a base and adds additional flags
52 enum DrawFilterFlags {
53 kNone_DrawFilterFlag = 0,
caryclark@google.come3e940c2012-11-07 16:42:17 +000054 kBlur_DrawFilterFlag = 0x4000, // toggles between blur and no blur
55 kLowBlur_DrawFilterFlag = 0x8000, // toggles between low and high quality blur
56 kHinting_DrawFilterFlag = 0x10000, // toggles between no hinting and normal hinting
57 kSlightHinting_DrawFilterFlag = 0x20000, // toggles between slight and normal hinting
58 kAAClip_DrawFilterFlag = 0x40000, // toggles between soft and hard clip
caryclark@google.coma3622372012-11-06 21:26:13 +000059 };
60
61 SK_COMPILE_ASSERT(!(kBlur_DrawFilterFlag & SkPaint::kAllFlags), blur_flag_must_be_greater);
62 SK_COMPILE_ASSERT(!(kHinting_DrawFilterFlag & SkPaint::kAllFlags),
63 hinting_flag_must_be_greater);
64 SK_COMPILE_ASSERT(!(kSlightHinting_DrawFilterFlag & SkPaint::kAllFlags),
65 slight_hinting_flag_must_be_greater);
66
scroggo@google.coma62da2f2012-11-02 21:28:12 +000067 /**
68 * Called with each new SkPicture to render.
69 */
keyar@chromium.org9d696c02012-08-07 17:11:33 +000070 virtual void init(SkPicture* pict);
scroggo@google.com9a412522012-09-07 15:21:18 +000071
72 /**
scroggo@google.comc0d5e542012-12-13 21:40:48 +000073 * Set the viewport so that only the portion listed gets drawn.
74 */
75 void setViewport(SkISize size) { fViewport = size; }
76
77 /**
scroggo@google.com9a412522012-09-07 15:21:18 +000078 * Perform any setup that should done prior to each iteration of render() which should not be
79 * timed.
80 */
81 virtual void setup() {}
82
83 /**
84 * Perform work that is to be timed. Typically this is rendering, but is also used for recording
85 * and preparing picture for playback by the subclasses which do those.
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000086 * If path is non-null, subclass implementations should call write().
87 * @param path If non-null, also write the output to the file specified by path. path should
88 * have no extension; it will be added by write().
borenet@google.com070d3542012-10-26 13:26:55 +000089 * @return bool True if rendering succeeded and, if path is non-null, the output was
90 * successfully written to a file.
scroggo@google.com9a412522012-09-07 15:21:18 +000091 */
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000092 virtual bool render(const SkString* path) = 0;
scroggo@google.com9a412522012-09-07 15:21:18 +000093
scroggo@google.coma62da2f2012-11-02 21:28:12 +000094 /**
95 * Called once finished with a particular SkPicture, before calling init again, and before
96 * being done with this Renderer.
97 */
keyar@chromium.org9d696c02012-08-07 17:11:33 +000098 virtual void end();
scroggo@google.coma62da2f2012-11-02 21:28:12 +000099
keyar@chromium.org28136b32012-08-20 15:04:15 +0000100 void resetState();
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000101
keyar@chromium.orgc81686c2012-08-20 15:04:04 +0000102 void setDeviceType(SkDeviceTypes deviceType) {
103 fDeviceType = deviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000104 }
105
caryclark@google.come3e940c2012-11-07 16:42:17 +0000106 void setDrawFilters(DrawFilterFlags const * const filters, const SkString& configName) {
107 memcpy(fDrawFilters, filters, sizeof(fDrawFilters));
caryclark@google.coma3622372012-11-06 21:26:13 +0000108 fDrawFiltersConfig = configName;
109 }
110
junov@chromium.org9313ca42012-11-02 18:11:49 +0000111 void setBBoxHierarchyType(BBoxHierarchyType bbhType) {
112 fBBoxHierarchyType = bbhType;
113 }
114
junov@chromium.org7b537062012-11-06 18:58:43 +0000115 void setGridSize(int width, int height) {
116 fGridWidth = width;
117 fGridHeight = height;
118 }
119
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000120 bool isUsingBitmapDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +0000121 return kBitmap_DeviceType == fDeviceType;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000122 }
123
scroggo@google.com9a412522012-09-07 15:21:18 +0000124 virtual SkString getPerIterTimeFormat() { return SkString("%.2f"); }
125
126 virtual SkString getNormalTimeFormat() { return SkString("%6.2f"); }
127
scroggo@google.com0a049b82012-11-02 22:01:26 +0000128 /**
129 * Reports the configuration of this PictureRenderer.
130 */
131 SkString getConfigName() {
132 SkString config = this->getConfigNameInternal();
133 if (kRTree_BBoxHierarchyType == fBBoxHierarchyType) {
134 config.append("_rtree");
junov@chromium.org7b537062012-11-06 18:58:43 +0000135 } else if (kTileGrid_BBoxHierarchyType == fBBoxHierarchyType) {
136 config.append("_grid");
scroggo@google.com0a049b82012-11-02 22:01:26 +0000137 }
138#if SK_SUPPORT_GPU
139 if (this->isUsingGpuDevice()) {
140 config.append("_gpu");
141 }
142#endif
caryclark@google.coma3622372012-11-06 21:26:13 +0000143 config.append(fDrawFiltersConfig.c_str());
scroggo@google.com0a049b82012-11-02 22:01:26 +0000144 return config;
145 }
146
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000147#if SK_SUPPORT_GPU
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000148 bool isUsingGpuDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +0000149 return kGPU_DeviceType == fDeviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000150 }
keyar@chromium.org77a55222012-08-20 15:03:47 +0000151
152 SkGLContext* getGLContext() {
153 if (this->isUsingGpuDevice()) {
154 return fGrContextFactory.getGLContext(GrContextFactory::kNative_GLContextType);
155 } else {
156 return NULL;
157 }
158 }
robertphillips@google.com163c84b2012-09-13 15:40:37 +0000159
160 GrContext* getGrContext() {
161 return fGrContext;
162 }
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000163#endif
164
keyar@chromium.org02dfb122012-08-20 15:03:36 +0000165 PictureRenderer()
keyar@chromium.org06125642012-08-20 15:03:33 +0000166 : fPicture(NULL)
167 , fDeviceType(kBitmap_DeviceType)
junov@chromium.org50ff9bd2012-11-02 19:16:22 +0000168 , fBBoxHierarchyType(kNone_BBoxHierarchyType)
junov@chromium.org7b537062012-11-06 18:58:43 +0000169 , fGridWidth(0)
170 , fGridHeight(0)
keyar@chromium.org06125642012-08-20 15:03:33 +0000171#if SK_SUPPORT_GPU
172 , fGrContext(fGrContextFactory.get(GrContextFactory::kNative_GLContextType))
173#endif
caryclark@google.come3e940c2012-11-07 16:42:17 +0000174 {
175 sk_bzero(fDrawFilters, sizeof(fDrawFilters));
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000176 fViewport.set(0, 0);
caryclark@google.come3e940c2012-11-07 16:42:17 +0000177 }
keyar@chromium.org06125642012-08-20 15:03:33 +0000178
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000179protected:
180 SkAutoTUnref<SkCanvas> fCanvas;
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000181 SkPicture* fPicture;
182 SkDeviceTypes fDeviceType;
183 BBoxHierarchyType fBBoxHierarchyType;
184 DrawFilterFlags fDrawFilters[SkDrawFilter::kTypeCount];
185 SkString fDrawFiltersConfig;
186 int fGridWidth, fGridHeight; // used when fBBoxHierarchyType is TileGrid
keyar@chromium.org06125642012-08-20 15:03:33 +0000187
188#if SK_SUPPORT_GPU
189 GrContextFactory fGrContextFactory;
190 GrContext* fGrContext;
191#endif
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000192
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000193 void buildBBoxHierarchy();
194
195 /**
196 * Return the total width that should be drawn. If the viewport width has been set greater than
197 * 0, this will be the minimum of the current SkPicture's width and the viewport's width.
198 */
199 int getViewWidth();
200
201 /**
202 * Return the total height that should be drawn. If the viewport height has been set greater
203 * than 0, this will be the minimum of the current SkPicture's height and the viewport's height.
204 */
205 int getViewHeight();
206
207 SkPicture* createPicture();
208 uint32_t recordFlags();
209 SkCanvas* setupCanvas();
210 virtual SkCanvas* setupCanvas(int width, int height);
211
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000212private:
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000213 SkISize fViewport;
214
scroggo@google.com0a049b82012-11-02 22:01:26 +0000215 virtual SkString getConfigNameInternal() = 0;
216
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000217 typedef SkRefCnt INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000218};
219
scroggo@google.com9a412522012-09-07 15:21:18 +0000220/**
221 * This class does not do any rendering, but its render function executes recording, which we want
222 * to time.
223 */
224class RecordPictureRenderer : public PictureRenderer {
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000225 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000226
227 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
228
229 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
scroggo@google.com0a049b82012-11-02 22:01:26 +0000230
djsollen@google.comfd9720c2012-11-06 16:54:40 +0000231protected:
232 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
233
scroggo@google.com0a049b82012-11-02 22:01:26 +0000234private:
235 virtual SkString getConfigNameInternal() SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000236};
237
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000238class PipePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000239public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000240 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000241
242private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000243 virtual SkString getConfigNameInternal() SK_OVERRIDE;
244
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000245 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000246};
247
248class SimplePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000249public:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000250 virtual void init(SkPicture* pict) SK_OVERRIDE;
251
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000252 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000253
254private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000255 virtual SkString getConfigNameInternal() SK_OVERRIDE;
256
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000257 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000258};
259
260class TiledPictureRenderer : public PictureRenderer {
261public:
262 TiledPictureRenderer();
263
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000264 virtual void init(SkPicture* pict) SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000265
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000266 /**
267 * Renders to tiles, rather than a single canvas. If a path is provided, a separate file is
268 * created for each tile, named "path0.png", "path1.png", etc.
269 * Multithreaded mode currently does not support writing to a file.
270 */
271 virtual bool render(const SkString* path) SK_OVERRIDE;
272
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000273 virtual void end() SK_OVERRIDE;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000274
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000275 void setTileWidth(int width) {
276 fTileWidth = width;
277 }
278
279 int getTileWidth() const {
280 return fTileWidth;
281 }
282
283 void setTileHeight(int height) {
284 fTileHeight = height;
285 }
286
287 int getTileHeight() const {
288 return fTileHeight;
289 }
290
291 void setTileWidthPercentage(double percentage) {
292 fTileWidthPercentage = percentage;
293 }
294
keyar@chromium.org163b5672012-08-01 17:53:29 +0000295 double getTileWidthPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000296 return fTileWidthPercentage;
297 }
298
299 void setTileHeightPercentage(double percentage) {
300 fTileHeightPercentage = percentage;
301 }
302
keyar@chromium.org163b5672012-08-01 17:53:29 +0000303 double getTileHeightPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000304 return fTileHeightPercentage;
305 }
306
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000307 void setTileMinPowerOf2Width(int width) {
308 SkASSERT(SkIsPow2(width) && width > 0);
309 if (!SkIsPow2(width) || width <= 0) {
310 return;
311 }
312
313 fTileMinPowerOf2Width = width;
314 }
315
316 int getTileMinPowerOf2Width() const {
317 return fTileMinPowerOf2Width;
318 }
319
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000320protected:
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000321 SkTDArray<SkRect> fTileRects;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000322
scroggo@google.com0a049b82012-11-02 22:01:26 +0000323 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
324 virtual SkString getConfigNameInternal() SK_OVERRIDE;
325
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000326private:
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000327 int fTileWidth;
328 int fTileHeight;
329 double fTileWidthPercentage;
330 double fTileHeightPercentage;
331 int fTileMinPowerOf2Width;
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000332
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000333 void setupTiles();
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000334 void setupPowerOf2Tiles();
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000335
336 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000337};
338
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000339class CloneData;
340
341class MultiCorePictureRenderer : public TiledPictureRenderer {
342public:
343 explicit MultiCorePictureRenderer(int threadCount);
344
345 ~MultiCorePictureRenderer();
346
347 virtual void init(SkPicture* pict) SK_OVERRIDE;
348
349 /**
350 * Behaves like TiledPictureRenderer::render(), only using multiple threads.
351 */
352 virtual bool render(const SkString* path) SK_OVERRIDE;
353
354 virtual void end() SK_OVERRIDE;
355
356private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000357 virtual SkString getConfigNameInternal() SK_OVERRIDE;
358
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000359 const int fNumThreads;
360 SkTDArray<SkCanvas*> fCanvasPool;
361 SkThreadPool fThreadPool;
362 SkPicture* fPictureClones;
363 CloneData** fCloneData;
364 SkCountdown fCountdown;
365
366 typedef TiledPictureRenderer INHERITED;
367};
368
scroggo@google.com9a412522012-09-07 15:21:18 +0000369/**
370 * This class does not do any rendering, but its render function executes turning an SkPictureRecord
371 * into an SkPicturePlayback, which we want to time.
372 */
373class PlaybackCreationRenderer : public PictureRenderer {
374public:
375 virtual void setup() SK_OVERRIDE;
376
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000377 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000378
379 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
380
381 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
382
383private:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000384 SkAutoTUnref<SkPicture> fReplayer;
scroggo@google.com0a049b82012-11-02 22:01:26 +0000385
386 virtual SkString getConfigNameInternal() SK_OVERRIDE;
387
scroggo@google.com9a412522012-09-07 15:21:18 +0000388 typedef PictureRenderer INHERITED;
389};
390
reed@google.comfe7b1ed2012-11-29 21:00:39 +0000391extern PictureRenderer* CreateGatherPixelRefsRenderer();
reed@google.com5a34fd32012-12-10 16:05:09 +0000392extern PictureRenderer* CreatePictureCloneRenderer();
reed@google.comfe7b1ed2012-11-29 21:00:39 +0000393
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000394}
395
396#endif // PictureRenderer_DEFINED