blob: 9422c9e8ba1e435208ebd3d05a88438b43ec218e [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"
scroggo@google.com9a412522012-09-07 15:21:18 +000014#include "SkPicture.h"
scroggo@google.comacfb30e2012-09-18 14:32:35 +000015#include "SkRect.h"
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000016#include "SkRefCnt.h"
scroggo@google.coma62da2f2012-11-02 21:28:12 +000017#include "SkRunnable.h"
scroggo@google.com9a412522012-09-07 15:21:18 +000018#include "SkString.h"
scroggo@google.coma62da2f2012-11-02 21:28:12 +000019#include "SkTDArray.h"
20#include "SkThreadPool.h"
21#include "SkTypes.h"
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000022
keyar@chromium.org06125642012-08-20 15:03:33 +000023#if SK_SUPPORT_GPU
24#include "GrContextFactory.h"
25#include "GrContext.h"
26#endif
27
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000028class SkBitmap;
29class SkCanvas;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000030class SkGLContext;
scroggo@google.coma62da2f2012-11-02 21:28:12 +000031class SkThread;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000032
33namespace sk_tools {
34
35class PictureRenderer : public SkRefCnt {
36public:
keyar@chromium.orgc81686c2012-08-20 15:04:04 +000037 enum SkDeviceTypes {
38 kBitmap_DeviceType,
39#if SK_SUPPORT_GPU
40 kGPU_DeviceType
41#endif
42 };
43
junov@chromium.org9313ca42012-11-02 18:11:49 +000044 enum BBoxHierarchyType {
45 kNone_BBoxHierarchyType = 0,
46 kRTree_BBoxHierarchyType,
junov@chromium.org7b537062012-11-06 18:58:43 +000047 kTileGrid_BBoxHierarchyType,
junov@chromium.org9313ca42012-11-02 18:11:49 +000048 };
49
caryclark@google.coma3622372012-11-06 21:26:13 +000050 // this uses SkPaint::Flags as a base and adds additional flags
51 enum DrawFilterFlags {
52 kNone_DrawFilterFlag = 0,
53 kBlur_DrawFilterFlag = 0x4000,
54 kHinting_DrawFilterFlag = 0x8000, // toggles between no hinting and normal hinting
55 kSlightHinting_DrawFilterFlag = 0x10000, // toggles between slight and normal hinting
56 };
57
58 SK_COMPILE_ASSERT(!(kBlur_DrawFilterFlag & SkPaint::kAllFlags), blur_flag_must_be_greater);
59 SK_COMPILE_ASSERT(!(kHinting_DrawFilterFlag & SkPaint::kAllFlags),
60 hinting_flag_must_be_greater);
61 SK_COMPILE_ASSERT(!(kSlightHinting_DrawFilterFlag & SkPaint::kAllFlags),
62 slight_hinting_flag_must_be_greater);
63
scroggo@google.coma62da2f2012-11-02 21:28:12 +000064 /**
65 * Called with each new SkPicture to render.
66 */
keyar@chromium.org9d696c02012-08-07 17:11:33 +000067 virtual void init(SkPicture* pict);
scroggo@google.com9a412522012-09-07 15:21:18 +000068
69 /**
70 * Perform any setup that should done prior to each iteration of render() which should not be
71 * timed.
72 */
73 virtual void setup() {}
74
75 /**
76 * Perform work that is to be timed. Typically this is rendering, but is also used for recording
77 * and preparing picture for playback by the subclasses which do those.
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000078 * If path is non-null, subclass implementations should call write().
79 * @param path If non-null, also write the output to the file specified by path. path should
80 * have no extension; it will be added by write().
borenet@google.com070d3542012-10-26 13:26:55 +000081 * @return bool True if rendering succeeded and, if path is non-null, the output was
82 * successfully written to a file.
scroggo@google.com9a412522012-09-07 15:21:18 +000083 */
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000084 virtual bool render(const SkString* path) = 0;
scroggo@google.com9a412522012-09-07 15:21:18 +000085
scroggo@google.coma62da2f2012-11-02 21:28:12 +000086 /**
87 * Called once finished with a particular SkPicture, before calling init again, and before
88 * being done with this Renderer.
89 */
keyar@chromium.org9d696c02012-08-07 17:11:33 +000090 virtual void end();
scroggo@google.coma62da2f2012-11-02 21:28:12 +000091
keyar@chromium.org28136b32012-08-20 15:04:15 +000092 void resetState();
keyar@chromium.org9d696c02012-08-07 17:11:33 +000093
keyar@chromium.orgc81686c2012-08-20 15:04:04 +000094 void setDeviceType(SkDeviceTypes deviceType) {
95 fDeviceType = deviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000096 }
97
caryclark@google.coma3622372012-11-06 21:26:13 +000098 void setDrawFilters(DrawFilterFlags* filters, const SkString& configName) {
99 fDrawFilters = filters;
100 fDrawFiltersConfig = configName;
101 }
102
junov@chromium.org9313ca42012-11-02 18:11:49 +0000103 void setBBoxHierarchyType(BBoxHierarchyType bbhType) {
104 fBBoxHierarchyType = bbhType;
105 }
106
junov@chromium.org7b537062012-11-06 18:58:43 +0000107 void setGridSize(int width, int height) {
108 fGridWidth = width;
109 fGridHeight = height;
110 }
111
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000112 bool isUsingBitmapDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +0000113 return kBitmap_DeviceType == fDeviceType;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000114 }
115
scroggo@google.com9a412522012-09-07 15:21:18 +0000116 virtual SkString getPerIterTimeFormat() { return SkString("%.2f"); }
117
118 virtual SkString getNormalTimeFormat() { return SkString("%6.2f"); }
119
scroggo@google.com0a049b82012-11-02 22:01:26 +0000120 /**
121 * Reports the configuration of this PictureRenderer.
122 */
123 SkString getConfigName() {
124 SkString config = this->getConfigNameInternal();
125 if (kRTree_BBoxHierarchyType == fBBoxHierarchyType) {
126 config.append("_rtree");
junov@chromium.org7b537062012-11-06 18:58:43 +0000127 } else if (kTileGrid_BBoxHierarchyType == fBBoxHierarchyType) {
128 config.append("_grid");
scroggo@google.com0a049b82012-11-02 22:01:26 +0000129 }
130#if SK_SUPPORT_GPU
131 if (this->isUsingGpuDevice()) {
132 config.append("_gpu");
133 }
134#endif
caryclark@google.coma3622372012-11-06 21:26:13 +0000135 config.append(fDrawFiltersConfig.c_str());
scroggo@google.com0a049b82012-11-02 22:01:26 +0000136 return config;
137 }
138
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000139#if SK_SUPPORT_GPU
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000140 bool isUsingGpuDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +0000141 return kGPU_DeviceType == fDeviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000142 }
keyar@chromium.org77a55222012-08-20 15:03:47 +0000143
144 SkGLContext* getGLContext() {
145 if (this->isUsingGpuDevice()) {
146 return fGrContextFactory.getGLContext(GrContextFactory::kNative_GLContextType);
147 } else {
148 return NULL;
149 }
150 }
robertphillips@google.com163c84b2012-09-13 15:40:37 +0000151
152 GrContext* getGrContext() {
153 return fGrContext;
154 }
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000155#endif
156
keyar@chromium.org02dfb122012-08-20 15:03:36 +0000157 PictureRenderer()
keyar@chromium.org06125642012-08-20 15:03:33 +0000158 : fPicture(NULL)
159 , fDeviceType(kBitmap_DeviceType)
junov@chromium.org50ff9bd2012-11-02 19:16:22 +0000160 , fBBoxHierarchyType(kNone_BBoxHierarchyType)
caryclark@google.coma3622372012-11-06 21:26:13 +0000161 , fDrawFilters(NULL)
junov@chromium.org7b537062012-11-06 18:58:43 +0000162 , fGridWidth(0)
163 , fGridHeight(0)
keyar@chromium.org06125642012-08-20 15:03:33 +0000164#if SK_SUPPORT_GPU
165 , fGrContext(fGrContextFactory.get(GrContextFactory::kNative_GLContextType))
166#endif
167 {}
168
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000169protected:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000170 void buildBBoxHierarchy();
171 SkPicture* createPicture();
172 uint32_t recordFlags();
keyar@chromium.orga474ce32012-08-20 15:03:57 +0000173 SkCanvas* setupCanvas();
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000174 virtual SkCanvas* setupCanvas(int width, int height);
keyar@chromium.orga474ce32012-08-20 15:03:57 +0000175
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000176 SkAutoTUnref<SkCanvas> fCanvas;
177 SkPicture* fPicture;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000178 SkDeviceTypes fDeviceType;
junov@chromium.org9313ca42012-11-02 18:11:49 +0000179 BBoxHierarchyType fBBoxHierarchyType;
caryclark@google.coma3622372012-11-06 21:26:13 +0000180 DrawFilterFlags* fDrawFilters;
181 SkString fDrawFiltersConfig;
junov@chromium.org7b537062012-11-06 18:58:43 +0000182 int fGridWidth, fGridHeight; // used when fBBoxHierarchyType is TileGrid
keyar@chromium.org06125642012-08-20 15:03:33 +0000183
184#if SK_SUPPORT_GPU
185 GrContextFactory fGrContextFactory;
186 GrContext* fGrContext;
187#endif
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000188
189private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000190 virtual SkString getConfigNameInternal() = 0;
191
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000192 typedef SkRefCnt INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000193};
194
scroggo@google.com9a412522012-09-07 15:21:18 +0000195/**
196 * This class does not do any rendering, but its render function executes recording, which we want
197 * to time.
198 */
199class RecordPictureRenderer : public PictureRenderer {
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000200 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000201
202 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
203
204 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
scroggo@google.com0a049b82012-11-02 22:01:26 +0000205
djsollen@google.comfd9720c2012-11-06 16:54:40 +0000206protected:
207 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
208
scroggo@google.com0a049b82012-11-02 22:01:26 +0000209private:
210 virtual SkString getConfigNameInternal() SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000211};
212
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000213class PipePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000214public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000215 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000216
217private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000218 virtual SkString getConfigNameInternal() SK_OVERRIDE;
219
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000220 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000221};
222
223class SimplePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000224public:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000225 virtual void init(SkPicture* pict) SK_OVERRIDE;
226
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000227 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000228
229private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000230 virtual SkString getConfigNameInternal() SK_OVERRIDE;
231
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000232 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000233};
234
235class TiledPictureRenderer : public PictureRenderer {
236public:
237 TiledPictureRenderer();
238
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000239 virtual void init(SkPicture* pict) SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000240
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000241 /**
242 * Renders to tiles, rather than a single canvas. If a path is provided, a separate file is
243 * created for each tile, named "path0.png", "path1.png", etc.
244 * Multithreaded mode currently does not support writing to a file.
245 */
246 virtual bool render(const SkString* path) SK_OVERRIDE;
247
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000248 virtual void end() SK_OVERRIDE;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000249
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000250 void setTileWidth(int width) {
251 fTileWidth = width;
252 }
253
254 int getTileWidth() const {
255 return fTileWidth;
256 }
257
258 void setTileHeight(int height) {
259 fTileHeight = height;
260 }
261
262 int getTileHeight() const {
263 return fTileHeight;
264 }
265
266 void setTileWidthPercentage(double percentage) {
267 fTileWidthPercentage = percentage;
268 }
269
keyar@chromium.org163b5672012-08-01 17:53:29 +0000270 double getTileWidthPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000271 return fTileWidthPercentage;
272 }
273
274 void setTileHeightPercentage(double percentage) {
275 fTileHeightPercentage = percentage;
276 }
277
keyar@chromium.org163b5672012-08-01 17:53:29 +0000278 double getTileHeightPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000279 return fTileHeightPercentage;
280 }
281
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000282 void setTileMinPowerOf2Width(int width) {
283 SkASSERT(SkIsPow2(width) && width > 0);
284 if (!SkIsPow2(width) || width <= 0) {
285 return;
286 }
287
288 fTileMinPowerOf2Width = width;
289 }
290
291 int getTileMinPowerOf2Width() const {
292 return fTileMinPowerOf2Width;
293 }
294
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000295protected:
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000296 SkTDArray<SkRect> fTileRects;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000297
scroggo@google.com0a049b82012-11-02 22:01:26 +0000298 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
299 virtual SkString getConfigNameInternal() SK_OVERRIDE;
300
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000301private:
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000302 int fTileWidth;
303 int fTileHeight;
304 double fTileWidthPercentage;
305 double fTileHeightPercentage;
306 int fTileMinPowerOf2Width;
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000307
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000308 void setupTiles();
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000309 void setupPowerOf2Tiles();
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000310
311 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000312};
313
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000314class CloneData;
315
316class MultiCorePictureRenderer : public TiledPictureRenderer {
317public:
318 explicit MultiCorePictureRenderer(int threadCount);
319
320 ~MultiCorePictureRenderer();
321
322 virtual void init(SkPicture* pict) SK_OVERRIDE;
323
324 /**
325 * Behaves like TiledPictureRenderer::render(), only using multiple threads.
326 */
327 virtual bool render(const SkString* path) SK_OVERRIDE;
328
329 virtual void end() SK_OVERRIDE;
330
331private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000332 virtual SkString getConfigNameInternal() SK_OVERRIDE;
333
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000334 const int fNumThreads;
335 SkTDArray<SkCanvas*> fCanvasPool;
336 SkThreadPool fThreadPool;
337 SkPicture* fPictureClones;
338 CloneData** fCloneData;
339 SkCountdown fCountdown;
340
341 typedef TiledPictureRenderer INHERITED;
342};
343
scroggo@google.com9a412522012-09-07 15:21:18 +0000344/**
345 * This class does not do any rendering, but its render function executes turning an SkPictureRecord
346 * into an SkPicturePlayback, which we want to time.
347 */
348class PlaybackCreationRenderer : public PictureRenderer {
349public:
350 virtual void setup() SK_OVERRIDE;
351
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000352 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000353
354 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
355
356 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
357
358private:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000359 SkAutoTUnref<SkPicture> fReplayer;
scroggo@google.com0a049b82012-11-02 22:01:26 +0000360
361 virtual SkString getConfigNameInternal() SK_OVERRIDE;
362
scroggo@google.com9a412522012-09-07 15:21:18 +0000363 typedef PictureRenderer INHERITED;
364};
365
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000366}
367
368#endif // PictureRenderer_DEFINED