blob: 4829e1df8750223f84942db85d98b5fd972d2aba [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().
53 * @return bool True if path is non-null and the output is successfully written to a file.
scroggo@google.com9a412522012-09-07 15:21:18 +000054 */
scroggo@google.com81f9d2e2012-09-20 14:54:21 +000055 virtual bool render(const SkString* path) = 0;
scroggo@google.com9a412522012-09-07 15:21:18 +000056
keyar@chromium.org9d696c02012-08-07 17:11:33 +000057 virtual void end();
keyar@chromium.org28136b32012-08-20 15:04:15 +000058 void resetState();
keyar@chromium.org9d696c02012-08-07 17:11:33 +000059
keyar@chromium.orgc81686c2012-08-20 15:04:04 +000060 void setDeviceType(SkDeviceTypes deviceType) {
61 fDeviceType = deviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000062 }
63
64 bool isUsingBitmapDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +000065 return kBitmap_DeviceType == fDeviceType;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000066 }
67
scroggo@google.com9a412522012-09-07 15:21:18 +000068 virtual SkString getPerIterTimeFormat() { return SkString("%.2f"); }
69
70 virtual SkString getNormalTimeFormat() { return SkString("%6.2f"); }
71
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000072#if SK_SUPPORT_GPU
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000073 bool isUsingGpuDevice() {
keyar@chromium.org78a35c52012-08-20 15:03:44 +000074 return kGPU_DeviceType == fDeviceType;
keyar@chromium.orgfe6391a2012-08-20 15:03:41 +000075 }
keyar@chromium.org77a55222012-08-20 15:03:47 +000076
77 SkGLContext* getGLContext() {
78 if (this->isUsingGpuDevice()) {
79 return fGrContextFactory.getGLContext(GrContextFactory::kNative_GLContextType);
80 } else {
81 return NULL;
82 }
83 }
robertphillips@google.com163c84b2012-09-13 15:40:37 +000084
85 GrContext* getGrContext() {
86 return fGrContext;
87 }
keyar@chromium.org4ea96c52012-08-20 15:03:29 +000088#endif
89
keyar@chromium.org02dfb122012-08-20 15:03:36 +000090 PictureRenderer()
keyar@chromium.org06125642012-08-20 15:03:33 +000091 : fPicture(NULL)
92 , fDeviceType(kBitmap_DeviceType)
93#if SK_SUPPORT_GPU
94 , fGrContext(fGrContextFactory.get(GrContextFactory::kNative_GLContextType))
95#endif
96 {}
97
keyar@chromium.org9d696c02012-08-07 17:11:33 +000098protected:
keyar@chromium.orga474ce32012-08-20 15:03:57 +000099 SkCanvas* setupCanvas();
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000100 virtual SkCanvas* setupCanvas(int width, int height);
keyar@chromium.orga474ce32012-08-20 15:03:57 +0000101
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000102 SkAutoTUnref<SkCanvas> fCanvas;
103 SkPicture* fPicture;
keyar@chromium.org4ea96c52012-08-20 15:03:29 +0000104 SkDeviceTypes fDeviceType;
keyar@chromium.org06125642012-08-20 15:03:33 +0000105
106#if SK_SUPPORT_GPU
107 GrContextFactory fGrContextFactory;
108 GrContext* fGrContext;
109#endif
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000110
111private:
112 typedef SkRefCnt INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000113};
114
scroggo@google.com9a412522012-09-07 15:21:18 +0000115/**
116 * This class does not do any rendering, but its render function executes recording, which we want
117 * to time.
118 */
119class RecordPictureRenderer : public PictureRenderer {
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000120 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000121
122 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
123
124 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
125};
126
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000127class PipePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000128public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000129 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000130
131private:
132 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000133};
134
135class SimplePictureRenderer : public PictureRenderer {
keyar@chromium.org163b5672012-08-01 17:53:29 +0000136public:
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000137 virtual bool render(const SkString*) SK_OVERRIDE;
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000138
139private:
140 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000141};
142
143class TiledPictureRenderer : public PictureRenderer {
144public:
145 TiledPictureRenderer();
146
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000147 virtual void init(SkPicture* pict) SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000148
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000149 virtual void setup() SK_OVERRIDE;
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000150
151 /**
152 * Renders to tiles, rather than a single canvas. If a path is provided, a separate file is
153 * created for each tile, named "path0.png", "path1.png", etc.
154 * Multithreaded mode currently does not support writing to a file.
155 */
156 virtual bool render(const SkString* path) SK_OVERRIDE;
157
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000158 virtual void end() SK_OVERRIDE;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000159
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000160 void setTileWidth(int width) {
161 fTileWidth = width;
162 }
163
164 int getTileWidth() const {
165 return fTileWidth;
166 }
167
168 void setTileHeight(int height) {
169 fTileHeight = height;
170 }
171
172 int getTileHeight() const {
173 return fTileHeight;
174 }
175
176 void setTileWidthPercentage(double percentage) {
177 fTileWidthPercentage = percentage;
178 }
179
keyar@chromium.org163b5672012-08-01 17:53:29 +0000180 double getTileWidthPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000181 return fTileWidthPercentage;
182 }
183
184 void setTileHeightPercentage(double percentage) {
185 fTileHeightPercentage = percentage;
186 }
187
keyar@chromium.org163b5672012-08-01 17:53:29 +0000188 double getTileHeightPercentage() const {
keyar@chromium.orgcc6e5ef2012-07-27 20:09:26 +0000189 return fTileHeightPercentage;
190 }
191
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000192 void setTileMinPowerOf2Width(int width) {
193 SkASSERT(SkIsPow2(width) && width > 0);
194 if (!SkIsPow2(width) || width <= 0) {
195 return;
196 }
197
198 fTileMinPowerOf2Width = width;
199 }
200
201 int getTileMinPowerOf2Width() const {
202 return fTileMinPowerOf2Width;
203 }
204
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000205 /**
206 * Set the number of threads to use for drawing. Non-positive numbers will set it to 1.
207 */
208 void setNumberOfThreads(int num) {
209 fNumThreads = SkMax32(num, 1);
scroggo@google.com58b4ead2012-08-31 16:15:22 +0000210 }
211
scroggo@google.com58b4ead2012-08-31 16:15:22 +0000212 void setUsePipe(bool usePipe) {
213 fUsePipe = usePipe;
214 }
215
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000216 ~TiledPictureRenderer();
217
218private:
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000219 bool fUsePipe;
220 int fTileWidth;
221 int fTileHeight;
222 double fTileWidthPercentage;
223 double fTileHeightPercentage;
224 int fTileMinPowerOf2Width;
scroggo@google.comacfb30e2012-09-18 14:32:35 +0000225 SkTDArray<SkRect> fTileRects;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000226
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000227 // These are only used for multithreaded rendering
228 int32_t fTileCounter;
229 int fNumThreads;
230 SkTDArray<SkCanvas*> fCanvasPool;
231 SkPicture* fPictureClones;
232 ThreadSafePipeController* fPipeController;
233
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000234 void setupTiles();
keyar@chromium.orgf4959ab2012-08-23 20:53:25 +0000235 void setupPowerOf2Tiles();
scroggo@google.combcdf2ec2012-09-20 14:42:33 +0000236 virtual SkCanvas* setupCanvas(int width, int height) SK_OVERRIDE;
237 bool multiThreaded() { return fNumThreads > 1; }
keyar@chromium.org9d696c02012-08-07 17:11:33 +0000238
239 typedef PictureRenderer INHERITED;
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000240};
241
scroggo@google.com9a412522012-09-07 15:21:18 +0000242/**
243 * This class does not do any rendering, but its render function executes turning an SkPictureRecord
244 * into an SkPicturePlayback, which we want to time.
245 */
246class PlaybackCreationRenderer : public PictureRenderer {
247public:
248 virtual void setup() SK_OVERRIDE;
249
scroggo@google.com81f9d2e2012-09-20 14:54:21 +0000250 virtual bool render(const SkString*) SK_OVERRIDE;
scroggo@google.com9a412522012-09-07 15:21:18 +0000251
252 virtual SkString getPerIterTimeFormat() SK_OVERRIDE { return SkString("%.4f"); }
253
254 virtual SkString getNormalTimeFormat() SK_OVERRIDE { return SkString("%6.4f"); }
255
256private:
257 SkPicture fReplayer;
258 typedef PictureRenderer INHERITED;
259};
260
keyar@chromium.org451bb9f2012-07-26 17:27:57 +0000261}
262
263#endif // PictureRenderer_DEFINED