blob: 99c58e068f292d5a076fa34697785bfd00059b9d [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();
scroggo@google.comc4013c12012-12-13 22:07:08 +0000133 if (!fViewport.isEmpty()) {
134 config.appendf("_viewport_%ix%i", fViewport.width(), fViewport.height());
135 }
scroggo@google.com0a049b82012-11-02 22:01:26 +0000136 if (kRTree_BBoxHierarchyType == fBBoxHierarchyType) {
137 config.append("_rtree");
junov@chromium.org7b537062012-11-06 18:58:43 +0000138 } else if (kTileGrid_BBoxHierarchyType == fBBoxHierarchyType) {
139 config.append("_grid");
scroggo@google.com0a049b82012-11-02 22:01:26 +0000140 }
141#if SK_SUPPORT_GPU
142 if (this->isUsingGpuDevice()) {
143 config.append("_gpu");
144 }
145#endif
caryclark@google.coma3622372012-11-06 21:26:13 +0000146 config.append(fDrawFiltersConfig.c_str());
scroggo@google.com0a049b82012-11-02 22:01:26 +0000147 return config;
148 }
149
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000150#if SK_SUPPORT_GPU
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000151 bool isUsingGpuDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +0000152 return kGPU_DeviceType == fDeviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +0000153 }
keyar@chromium.org77a55222012-08-20 15:03:47 +0000154
155 SkGLContext* getGLContext() {
156 if (this->isUsingGpuDevice()) {
157 return fGrContextFactory.getGLContext(GrContextFactory::kNative_GLContextType);
158 } else {
159 return NULL;
160 }
161 }
robertphillips@google.com163c84b2012-09-13 15:40:37 +0000162
163 GrContext* getGrContext() {
164 return fGrContext;
165 }
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000166#endif
167
keyar@chromium.org02dfb122012-08-20 15:03:36 +0000168 PictureRenderer()
keyar@chromium.org06125642012-08-20 15:03:33 +0000169 : fPicture(NULL)
170 , fDeviceType(kBitmap_DeviceType)
junov@chromium.org50ff9bd2012-11-02 19:16:22 +0000171 , fBBoxHierarchyType(kNone_BBoxHierarchyType)
junov@chromium.org7b537062012-11-06 18:58:43 +0000172 , fGridWidth(0)
173 , fGridHeight(0)
keyar@chromium.org06125642012-08-20 15:03:33 +0000174#if SK_SUPPORT_GPU
175 , fGrContext(fGrContextFactory.get(GrContextFactory::kNative_GLContextType))
176#endif
caryclark@google.come3e940c2012-11-07 16:42:17 +0000177 {
178 sk_bzero(fDrawFilters, sizeof(fDrawFilters));
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000179 fViewport.set(0, 0);
caryclark@google.come3e940c2012-11-07 16:42:17 +0000180 }
keyar@chromium.org06125642012-08-20 15:03:33 +0000181
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000182protected:
183 SkAutoTUnref<SkCanvas> fCanvas;
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000184 SkPicture* fPicture;
185 SkDeviceTypes fDeviceType;
186 BBoxHierarchyType fBBoxHierarchyType;
187 DrawFilterFlags fDrawFilters[SkDrawFilter::kTypeCount];
188 SkString fDrawFiltersConfig;
189 int fGridWidth, fGridHeight; // used when fBBoxHierarchyType is TileGrid
keyar@chromium.org06125642012-08-20 15:03:33 +0000190
191#if SK_SUPPORT_GPU
192 GrContextFactory fGrContextFactory;
193 GrContext* fGrContext;
194#endif
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000195
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000196 void buildBBoxHierarchy();
197
198 /**
199 * Return the total width that should be drawn. If the viewport width has been set greater than
200 * 0, this will be the minimum of the current SkPicture's width and the viewport's width.
201 */
202 int getViewWidth();
203
204 /**
205 * Return the total height that should be drawn. If the viewport height has been set greater
206 * than 0, this will be the minimum of the current SkPicture's height and the viewport's height.
207 */
208 int getViewHeight();
209
210 SkPicture* createPicture();
211 uint32_t recordFlags();
212 SkCanvas* setupCanvas();
213 virtual SkCanvas* setupCanvas(int width, int height);
214
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000215private:
scroggo@google.comc0d5e542012-12-13 21:40:48 +0000216 SkISize fViewport;
217
scroggo@google.com0a049b82012-11-02 22:01:26 +0000218 virtual SkString getConfigNameInternal() = 0;
219
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000220 typedef SkRefCnt INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000221};
222
scroggo@google.com9a412522012-09-07 15:21:18 +0000223/**
224 * This class does not do any rendering, but its render function executes recording, which we want
225 * to time.
226 */
227class RecordPictureRenderer : public PictureRenderer {
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000228 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000229
230 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
231
232 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
scroggo@google.com0a049b82012-11-02 22:01:26 +0000233
djsollen@google.comfd9720c2012-11-06 16:54:40 +0000234protected:
235 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
236
scroggo@google.com0a049b82012-11-02 22:01:26 +0000237private:
238 virtual SkString getConfigNameInternal() SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000239};
240
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000241class PipePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000242public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000243 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000244
245private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000246 virtual SkString getConfigNameInternal() SK_OVERRIDE;
247
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000248 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000249};
250
251class SimplePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000252public:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000253 virtual void init(SkPicture* pict) SK_OVERRIDE;
254
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000255 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000256
257private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000258 virtual SkString getConfigNameInternal() SK_OVERRIDE;
259
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000260 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000261};
262
263class TiledPictureRenderer : public PictureRenderer {
264public:
265 TiledPictureRenderer();
266
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000267 virtual void init(SkPicture* pict) SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000268
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000269 /**
270 * Renders to tiles, rather than a single canvas. If a path is provided, a separate file is
271 * created for each tile, named "path0.png", "path1.png", etc.
272 * Multithreaded mode currently does not support writing to a file.
273 */
274 virtual bool render(const SkString* path) SK_OVERRIDE;
275
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000276 virtual void end() SK_OVERRIDE;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000277
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000278 void setTileWidth(int width) {
279 fTileWidth = width;
280 }
281
282 int getTileWidth() const {
283 return fTileWidth;
284 }
285
286 void setTileHeight(int height) {
287 fTileHeight = height;
288 }
289
290 int getTileHeight() const {
291 return fTileHeight;
292 }
293
294 void setTileWidthPercentage(double percentage) {
295 fTileWidthPercentage = percentage;
296 }
297
keyar@chromium.org163b5672012-08-01 17:53:29 +0000298 double getTileWidthPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000299 return fTileWidthPercentage;
300 }
301
302 void setTileHeightPercentage(double percentage) {
303 fTileHeightPercentage = percentage;
304 }
305
keyar@chromium.org163b5672012-08-01 17:53:29 +0000306 double getTileHeightPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000307 return fTileHeightPercentage;
308 }
309
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000310 void setTileMinPowerOf2Width(int width) {
311 SkASSERT(SkIsPow2(width) && width > 0);
312 if (!SkIsPow2(width) || width <= 0) {
313 return;
314 }
315
316 fTileMinPowerOf2Width = width;
317 }
318
319 int getTileMinPowerOf2Width() const {
320 return fTileMinPowerOf2Width;
321 }
322
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000323protected:
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000324 SkTDArray<SkRect> fTileRects;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000325
scroggo@google.com0a049b82012-11-02 22:01:26 +0000326 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
327 virtual SkString getConfigNameInternal() SK_OVERRIDE;
328
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000329private:
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000330 int fTileWidth;
331 int fTileHeight;
332 double fTileWidthPercentage;
333 double fTileHeightPercentage;
334 int fTileMinPowerOf2Width;
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000335
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000336 void setupTiles();
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000337 void setupPowerOf2Tiles();
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000338
339 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000340};
341
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000342class CloneData;
343
344class MultiCorePictureRenderer : public TiledPictureRenderer {
345public:
346 explicit MultiCorePictureRenderer(int threadCount);
347
348 ~MultiCorePictureRenderer();
349
350 virtual void init(SkPicture* pict) SK_OVERRIDE;
351
352 /**
353 * Behaves like TiledPictureRenderer::render(), only using multiple threads.
354 */
355 virtual bool render(const SkString* path) SK_OVERRIDE;
356
357 virtual void end() SK_OVERRIDE;
358
359private:
scroggo@google.com0a049b82012-11-02 22:01:26 +0000360 virtual SkString getConfigNameInternal() SK_OVERRIDE;
361
scroggo@google.coma62da2f2012-11-02 21:28:12 +0000362 const int fNumThreads;
363 SkTDArray<SkCanvas*> fCanvasPool;
364 SkThreadPool fThreadPool;
365 SkPicture* fPictureClones;
366 CloneData** fCloneData;
367 SkCountdown fCountdown;
368
369 typedef TiledPictureRenderer INHERITED;
370};
371
scroggo@google.com9a412522012-09-07 15:21:18 +0000372/**
373 * This class does not do any rendering, but its render function executes turning an SkPictureRecord
374 * into an SkPicturePlayback, which we want to time.
375 */
376class PlaybackCreationRenderer : public PictureRenderer {
377public:
378 virtual void setup() SK_OVERRIDE;
379
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000380 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000381
382 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
383
384 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
385
386private:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000387 SkAutoTUnref<SkPicture> fReplayer;
scroggo@google.com0a049b82012-11-02 22:01:26 +0000388
389 virtual SkString getConfigNameInternal() SK_OVERRIDE;
390
scroggo@google.com9a412522012-09-07 15:21:18 +0000391 typedef PictureRenderer INHERITED;
392};
393
reed@google.comfe7b1ed2012-11-29 21:00:39 +0000394extern PictureRenderer* CreateGatherPixelRefsRenderer();
reed@google.com5a34fd32012-12-10 16:05:09 +0000395extern PictureRenderer* CreatePictureCloneRenderer();
reed@google.comfe7b1ed2012-11-29 21:00:39 +0000396
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000397}
398
399#endif // PictureRenderer_DEFINED