blob: 9cafdc8c5416453d828989ed7ee7d1fcdd71ace1 [file] [log] [blame]
keyar@chromium.org451bb9f2012-07-26 17:27:57 +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
8#ifndef PictureRenderer_DEFINED
9#define PictureRenderer_DEFINED
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +000010#include "SkMath.h"
scroggo@google.com9a412522012-09-07 15:21:18 +000011#include "SkPicture.h"
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000012#include "SkTypes.h"
13#include "SkTDArray.h"
scroggo@google.comacfb30e2012-09-18 14:32:35 +000014#include "SkRect.h"
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000015#include "SkRefCnt.h"
scroggo@google.com9a412522012-09-07 15:21:18 +000016#include "SkString.h"
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000017
keyar@chromium.org06125642012-08-20 15:03:33 +000018#if SK_SUPPORT_GPU
19#include "GrContextFactory.h"
20#include "GrContext.h"
21#endif
22
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000023class SkBitmap;
24class SkCanvas;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000025class SkGLContext;
scroggo@google.combcdf2ec2012-09-20 14:42:33 +000026class ThreadSafePipeController;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +000027
28namespace sk_tools {
29
30class PictureRenderer : public SkRefCnt {
31public:
keyar@chromium.orgc81686c2012-08-20 15:04:04 +000032 enum SkDeviceTypes {
33 kBitmap_DeviceType,
34#if SK_SUPPORT_GPU
35 kGPU_DeviceType
36#endif
37 };
38
junov@chromium.org9313ca42012-11-02 18:11:49 +000039 enum BBoxHierarchyType {
40 kNone_BBoxHierarchyType = 0,
41 kRTree_BBoxHierarchyType,
42 };
43
keyar@chromium.org9d696c02012-08-07 17:11:33 +000044 virtual void init(SkPicture* pict);
scroggo@google.com9a412522012-09-07 15:21:18 +000045
46 /**
47 * Perform any setup that should done prior to each iteration of render() which should not be
48 * timed.
49 */
50 virtual void setup() {}
51
52 /**
53 * Perform work that is to be timed. Typically this is rendering, but is also used for recording
54 * and preparing picture for playback by the subclasses which do those.
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000055 * If path is non-null, subclass implementations should call write().
56 * @param path If non-null, also write the output to the file specified by path. path should
57 * have no extension; it will be added by write().
borenet@google.com070d3542012-10-26 13:26:55 +000058 * @return bool True if rendering succeeded and, if path is non-null, the output was
59 * successfully written to a file.
scroggo@google.com9a412522012-09-07 15:21:18 +000060 */
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000061 virtual bool render(const SkString* path) = 0;
scroggo@google.com9a412522012-09-07 15:21:18 +000062
keyar@chromium.org9d696c02012-08-07 17:11:33 +000063 virtual void end();
keyar@chromium.org28136b32012-08-20 15:04:15 +000064 void resetState();
keyar@chromium.org9d696c02012-08-07 17:11:33 +000065
keyar@chromium.orgc81686c2012-08-20 15:04:04 +000066 void setDeviceType(SkDeviceTypes deviceType) {
67 fDeviceType = deviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000068 }
69
junov@chromium.org9313ca42012-11-02 18:11:49 +000070 void setBBoxHierarchyType(BBoxHierarchyType bbhType) {
71 fBBoxHierarchyType = bbhType;
72 }
73
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000074 bool isUsingBitmapDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +000075 return kBitmap_DeviceType == fDeviceType;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000076 }
77
scroggo@google.com9a412522012-09-07 15:21:18 +000078 virtual SkString getPerIterTimeFormat() { return SkString("%.2f"); }
79
80 virtual SkString getNormalTimeFormat() { return SkString("%6.2f"); }
81
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000082#if SK_SUPPORT_GPU
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000083 bool isUsingGpuDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +000084 return kGPU_DeviceType == fDeviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000085 }
keyar@chromium.org77a55222012-08-20 15:03:47 +000086
87 SkGLContext* getGLContext() {
88 if (this->isUsingGpuDevice()) {
89 return fGrContextFactory.getGLContext(GrContextFactory::kNative_GLContextType);
90 } else {
91 return NULL;
92 }
93 }
robertphillips@google.com163c84b2012-09-13 15:40:37 +000094
95 GrContext* getGrContext() {
96 return fGrContext;
97 }
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000098#endif
99
keyar@chromium.org02dfb122012-08-20 15:03:36 +0000100 PictureRenderer()
keyar@chromium.org06125642012-08-20 15:03:33 +0000101 : fPicture(NULL)
102 , fDeviceType(kBitmap_DeviceType)
junov@chromium.org50ff9bd2012-11-02 19:16:22 +0000103 , fBBoxHierarchyType(kNone_BBoxHierarchyType)
keyar@chromium.org06125642012-08-20 15:03:33 +0000104#if SK_SUPPORT_GPU
105 , fGrContext(fGrContextFactory.get(GrContextFactory::kNative_GLContextType))
106#endif
107 {}
108
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000109protected:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000110 void buildBBoxHierarchy();
111 SkPicture* createPicture();
112 uint32_t recordFlags();
keyar@chromium.orga474ce32012-08-20 15:03:57 +0000113 SkCanvas* setupCanvas();
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000114 virtual SkCanvas* setupCanvas(int width, int height);
keyar@chromium.orga474ce32012-08-20 15:03:57 +0000115
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000116 SkAutoTUnref<SkCanvas> fCanvas;
117 SkPicture* fPicture;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000118 SkDeviceTypes fDeviceType;
junov@chromium.org9313ca42012-11-02 18:11:49 +0000119 BBoxHierarchyType fBBoxHierarchyType;
120
keyar@chromium.org06125642012-08-20 15:03:33 +0000121
122#if SK_SUPPORT_GPU
123 GrContextFactory fGrContextFactory;
124 GrContext* fGrContext;
125#endif
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000126
127private:
128 typedef SkRefCnt INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000129};
130
scroggo@google.com9a412522012-09-07 15:21:18 +0000131/**
132 * This class does not do any rendering, but its render function executes recording, which we want
133 * to time.
134 */
135class RecordPictureRenderer : public PictureRenderer {
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000136 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000137
138 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
139
140 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
141};
142
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000143class PipePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000144public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000145 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000146
147private:
148 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000149};
150
151class SimplePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000152public:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000153 virtual void init(SkPicture* pict) SK_OVERRIDE;
154
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000155 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000156
157private:
158 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000159};
160
161class TiledPictureRenderer : public PictureRenderer {
162public:
163 TiledPictureRenderer();
164
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000165 virtual void init(SkPicture* pict) SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000166
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000167 virtual void setup() SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000168
169 /**
170 * Renders to tiles, rather than a single canvas. If a path is provided, a separate file is
171 * created for each tile, named "path0.png", "path1.png", etc.
172 * Multithreaded mode currently does not support writing to a file.
173 */
174 virtual bool render(const SkString* path) SK_OVERRIDE;
175
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000176 virtual void end() SK_OVERRIDE;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000177
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000178 void setTileWidth(int width) {
179 fTileWidth = width;
180 }
181
182 int getTileWidth() const {
183 return fTileWidth;
184 }
185
186 void setTileHeight(int height) {
187 fTileHeight = height;
188 }
189
190 int getTileHeight() const {
191 return fTileHeight;
192 }
193
194 void setTileWidthPercentage(double percentage) {
195 fTileWidthPercentage = percentage;
196 }
197
keyar@chromium.org163b5672012-08-01 17:53:29 +0000198 double getTileWidthPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000199 return fTileWidthPercentage;
200 }
201
202 void setTileHeightPercentage(double percentage) {
203 fTileHeightPercentage = percentage;
204 }
205
keyar@chromium.org163b5672012-08-01 17:53:29 +0000206 double getTileHeightPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000207 return fTileHeightPercentage;
208 }
209
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000210 void setTileMinPowerOf2Width(int width) {
211 SkASSERT(SkIsPow2(width) && width > 0);
212 if (!SkIsPow2(width) || width <= 0) {
213 return;
214 }
215
216 fTileMinPowerOf2Width = width;
217 }
218
219 int getTileMinPowerOf2Width() const {
220 return fTileMinPowerOf2Width;
221 }
222
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000223 /**
224 * Set the number of threads to use for drawing. Non-positive numbers will set it to 1.
225 */
226 void setNumberOfThreads(int num) {
227 fNumThreads = SkMax32(num, 1);
scroggo@google.com58b4ead2012-08-31 16:15:22 +0000228 }
229
scroggo@google.com58b4ead2012-08-31 16:15:22 +0000230 void setUsePipe(bool usePipe) {
231 fUsePipe = usePipe;
232 }
233
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000234 ~TiledPictureRenderer();
235
236private:
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000237 bool fUsePipe;
238 int fTileWidth;
239 int fTileHeight;
240 double fTileWidthPercentage;
241 double fTileHeightPercentage;
242 int fTileMinPowerOf2Width;
scroggo@google.comacfb30e2012-09-18 14:32:35 +0000243 SkTDArray<SkRect> fTileRects;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000244
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000245 // These are only used for multithreaded rendering
246 int32_t fTileCounter;
247 int fNumThreads;
248 SkTDArray<SkCanvas*> fCanvasPool;
249 SkPicture* fPictureClones;
250 ThreadSafePipeController* fPipeController;
251
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000252 void setupTiles();
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000253 void setupPowerOf2Tiles();
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000254 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
255 bool multiThreaded() { return fNumThreads > 1; }
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000256
257 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000258};
259
scroggo@google.com9a412522012-09-07 15:21:18 +0000260/**
261 * This class does not do any rendering, but its render function executes turning an SkPictureRecord
262 * into an SkPicturePlayback, which we want to time.
263 */
264class PlaybackCreationRenderer : public PictureRenderer {
265public:
266 virtual void setup() SK_OVERRIDE;
267
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000268 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000269
270 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
271
272 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
273
274private:
junov@chromium.org9313ca42012-11-02 18:11:49 +0000275 SkAutoTUnref<SkPicture> fReplayer;
scroggo@google.com9a412522012-09-07 15:21:18 +0000276 typedef PictureRenderer INHERITED;
277};
278
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000279}
280
281#endif // PictureRenderer_DEFINED