blob: 5d32203b3c11b79774f203d4efaf4a72126039cc [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
keyar@chromium.org9d696c02012-08-07 17:11:33 +000039 virtual void init(SkPicture* pict);
scroggo@google.com9a412522012-09-07 15:21:18 +000040
41 /**
42 * Perform any setup that should done prior to each iteration of render() which should not be
43 * timed.
44 */
45 virtual void setup() {}
46
47 /**
48 * Perform work that is to be timed. Typically this is rendering, but is also used for recording
49 * and preparing picture for playback by the subclasses which do those.
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000050 * If path is non-null, subclass implementations should call write().
51 * @param path If non-null, also write the output to the file specified by path. path should
52 * have no extension; it will be added by write().
borenet@google.com070d3542012-10-26 13:26:55 +000053 * @return bool True if rendering succeeded and, if path is non-null, the output was
54 * successfully written to a file.
scroggo@google.com9a412522012-09-07 15:21:18 +000055 */
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000056 virtual bool render(const SkString* path) = 0;
scroggo@google.com9a412522012-09-07 15:21:18 +000057
keyar@chromium.org9d696c02012-08-07 17:11:33 +000058 virtual void end();
keyar@chromium.org28136b32012-08-20 15:04:15 +000059 void resetState();
keyar@chromium.org9d696c02012-08-07 17:11:33 +000060
keyar@chromium.orgc81686c2012-08-20 15:04:04 +000061 void setDeviceType(SkDeviceTypes deviceType) {
62 fDeviceType = deviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000063 }
64
65 bool isUsingBitmapDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +000066 return kBitmap_DeviceType == fDeviceType;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000067 }
68
scroggo@google.com9a412522012-09-07 15:21:18 +000069 virtual SkString getPerIterTimeFormat() { return SkString("%.2f"); }
70
71 virtual SkString getNormalTimeFormat() { return SkString("%6.2f"); }
72
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000073#if SK_SUPPORT_GPU
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000074 bool isUsingGpuDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +000075 return kGPU_DeviceType == fDeviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000076 }
keyar@chromium.org77a55222012-08-20 15:03:47 +000077
78 SkGLContext* getGLContext() {
79 if (this->isUsingGpuDevice()) {
80 return fGrContextFactory.getGLContext(GrContextFactory::kNative_GLContextType);
81 } else {
82 return NULL;
83 }
84 }
robertphillips@google.com163c84b2012-09-13 15:40:37 +000085
86 GrContext* getGrContext() {
87 return fGrContext;
88 }
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000089#endif
90
keyar@chromium.org02dfb122012-08-20 15:03:36 +000091 PictureRenderer()
keyar@chromium.org06125642012-08-20 15:03:33 +000092 : fPicture(NULL)
93 , fDeviceType(kBitmap_DeviceType)
94#if SK_SUPPORT_GPU
95 , fGrContext(fGrContextFactory.get(GrContextFactory::kNative_GLContextType))
96#endif
97 {}
98
keyar@chromium.org9d696c02012-08-07 17:11:33 +000099protected:
keyar@chromium.orga474ce32012-08-20 15:03:57 +0000100 SkCanvas* setupCanvas();
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000101 virtual SkCanvas* setupCanvas(int width, int height);
keyar@chromium.orga474ce32012-08-20 15:03:57 +0000102
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000103 SkAutoTUnref<SkCanvas> fCanvas;
104 SkPicture* fPicture;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000105 SkDeviceTypes fDeviceType;
keyar@chromium.org06125642012-08-20 15:03:33 +0000106
107#if SK_SUPPORT_GPU
108 GrContextFactory fGrContextFactory;
109 GrContext* fGrContext;
110#endif
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000111
112private:
113 typedef SkRefCnt INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000114};
115
scroggo@google.com9a412522012-09-07 15:21:18 +0000116/**
117 * This class does not do any rendering, but its render function executes recording, which we want
118 * to time.
119 */
120class RecordPictureRenderer : public PictureRenderer {
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000121 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000122
123 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
124
125 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
126};
127
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000128class PipePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000129public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000130 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000131
132private:
133 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000134};
135
136class SimplePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000137public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000138 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000139
140private:
141 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000142};
143
144class TiledPictureRenderer : public PictureRenderer {
145public:
146 TiledPictureRenderer();
147
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000148 virtual void init(SkPicture* pict) SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000149
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000150 virtual void setup() SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000151
152 /**
153 * Renders to tiles, rather than a single canvas. If a path is provided, a separate file is
154 * created for each tile, named "path0.png", "path1.png", etc.
155 * Multithreaded mode currently does not support writing to a file.
156 */
157 virtual bool render(const SkString* path) SK_OVERRIDE;
158
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000159 virtual void end() SK_OVERRIDE;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000160
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000161 void setTileWidth(int width) {
162 fTileWidth = width;
163 }
164
165 int getTileWidth() const {
166 return fTileWidth;
167 }
168
169 void setTileHeight(int height) {
170 fTileHeight = height;
171 }
172
173 int getTileHeight() const {
174 return fTileHeight;
175 }
176
177 void setTileWidthPercentage(double percentage) {
178 fTileWidthPercentage = percentage;
179 }
180
keyar@chromium.org163b5672012-08-01 17:53:29 +0000181 double getTileWidthPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000182 return fTileWidthPercentage;
183 }
184
185 void setTileHeightPercentage(double percentage) {
186 fTileHeightPercentage = percentage;
187 }
188
keyar@chromium.org163b5672012-08-01 17:53:29 +0000189 double getTileHeightPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000190 return fTileHeightPercentage;
191 }
192
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000193 void setTileMinPowerOf2Width(int width) {
194 SkASSERT(SkIsPow2(width) && width > 0);
195 if (!SkIsPow2(width) || width <= 0) {
196 return;
197 }
198
199 fTileMinPowerOf2Width = width;
200 }
201
202 int getTileMinPowerOf2Width() const {
203 return fTileMinPowerOf2Width;
204 }
205
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000206 /**
207 * Set the number of threads to use for drawing. Non-positive numbers will set it to 1.
208 */
209 void setNumberOfThreads(int num) {
210 fNumThreads = SkMax32(num, 1);
scroggo@google.com58b4ead2012-08-31 16:15:22 +0000211 }
212
scroggo@google.com58b4ead2012-08-31 16:15:22 +0000213 void setUsePipe(bool usePipe) {
214 fUsePipe = usePipe;
215 }
216
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000217 ~TiledPictureRenderer();
218
219private:
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000220 bool fUsePipe;
221 int fTileWidth;
222 int fTileHeight;
223 double fTileWidthPercentage;
224 double fTileHeightPercentage;
225 int fTileMinPowerOf2Width;
scroggo@google.comacfb30e2012-09-18 14:32:35 +0000226 SkTDArray<SkRect> fTileRects;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000227
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000228 // These are only used for multithreaded rendering
229 int32_t fTileCounter;
230 int fNumThreads;
231 SkTDArray<SkCanvas*> fCanvasPool;
232 SkPicture* fPictureClones;
233 ThreadSafePipeController* fPipeController;
234
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000235 void setupTiles();
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000236 void setupPowerOf2Tiles();
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000237 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
238 bool multiThreaded() { return fNumThreads > 1; }
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000239
240 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000241};
242
scroggo@google.com9a412522012-09-07 15:21:18 +0000243/**
244 * This class does not do any rendering, but its render function executes turning an SkPictureRecord
245 * into an SkPicturePlayback, which we want to time.
246 */
247class PlaybackCreationRenderer : public PictureRenderer {
248public:
249 virtual void setup() SK_OVERRIDE;
250
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000251 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000252
253 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
254
255 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
256
257private:
258 SkPicture fReplayer;
259 typedef PictureRenderer INHERITED;
260};
261
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000262}
263
264#endif // PictureRenderer_DEFINED