blob: f3d96d4aa99c2e5e1f76c264bd8097ec313c5874 [file] [log] [blame]
commit-bot@chromium.orgc4b21e62014-04-11 18:33:31 +00001/*
2 * Copyright 2014 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
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +00008#ifndef SkRecords_DEFINED
9#define SkRecords_DEFINED
10
11#include "SkCanvas.h"
reed2347b622014-08-07 12:19:50 -070012#include "SkPicture.h"
fmalita00d5c2c2014-08-21 08:53:26 -070013#include "SkTextBlob.h"
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +000014
15namespace SkRecords {
16
17// A list of all the types of canvas calls we can record.
18// Each of these is reified into a struct below.
19//
20// (We're using the macro-of-macro trick here to do several different things with the same list.)
21//
22// We leave this SK_RECORD_TYPES macro defined for use by code that wants to operate on SkRecords
23// types polymorphically. (See SkRecord::Record::{visit,mutate} for an example.)
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +000024//
25// Order doesn't technically matter here, but the compiler can generally generate better code if
26// you keep them semantically grouped, especially the Draws. It's also nice to leave NoOp at 0.
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +000027#define SK_RECORD_TYPES(M) \
28 M(NoOp) \
29 M(Restore) \
30 M(Save) \
31 M(SaveLayer) \
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +000032 M(PushCull) \
33 M(PopCull) \
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +000034 M(SetMatrix) \
35 M(ClipPath) \
36 M(ClipRRect) \
37 M(ClipRect) \
38 M(ClipRegion) \
39 M(Clear) \
40 M(DrawBitmap) \
41 M(DrawBitmapMatrix) \
42 M(DrawBitmapNine) \
43 M(DrawBitmapRectToRect) \
44 M(DrawDRRect) \
45 M(DrawOval) \
46 M(DrawPaint) \
47 M(DrawPath) \
dandov963137b2014-08-07 07:49:53 -070048 M(DrawPatch) \
reed2347b622014-08-07 12:19:50 -070049 M(DrawPicture) \
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +000050 M(DrawPoints) \
51 M(DrawPosText) \
52 M(DrawPosTextH) \
mtkleinc551d9f2014-08-20 08:09:46 -070053 M(DrawText) \
54 M(DrawTextOnPath) \
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +000055 M(DrawRRect) \
56 M(DrawRect) \
57 M(DrawSprite) \
fmalita00d5c2c2014-08-21 08:53:26 -070058 M(DrawTextBlob) \
mtkleinf4078ad2014-08-08 10:05:19 -070059 M(DrawVertices)
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +000060
61// Defines SkRecords::Type, an enum of all record types.
62#define ENUM(T) T##_Type,
63enum Type { SK_RECORD_TYPES(ENUM) };
64#undef ENUM
65
66// Macros to make it easier to define a record for a draw call with 0 args, 1 args, 2 args, etc.
67// These should be clearer when you look at their use below.
68#define RECORD0(T) \
69struct T { \
70 static const Type kType = T##_Type; \
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +000071};
72
73// We try to be flexible about the types the constructors take. Instead of requring the exact type
74// A here, we take any type Z which implicitly casts to A. This allows the delay_copy() trick to
75// work, allowing the caller to decide whether to pass by value or by const&.
76
77#define RECORD1(T, A, a) \
78struct T { \
79 static const Type kType = T##_Type; \
80 template <typename Z> \
81 T(Z a) : a(a) {} \
82 A a; \
83};
84
85#define RECORD2(T, A, a, B, b) \
86struct T { \
87 static const Type kType = T##_Type; \
88 template <typename Z, typename Y> \
89 T(Z a, Y b) : a(a), b(b) {} \
90 A a; B b; \
91};
92
93#define RECORD3(T, A, a, B, b, C, c) \
94struct T { \
95 static const Type kType = T##_Type; \
96 template <typename Z, typename Y, typename X> \
97 T(Z a, Y b, X c) : a(a), b(b), c(c) {} \
98 A a; B b; C c; \
99};
100
101#define RECORD4(T, A, a, B, b, C, c, D, d) \
102struct T { \
103 static const Type kType = T##_Type; \
104 template <typename Z, typename Y, typename X, typename W> \
105 T(Z a, Y b, X c, W d) : a(a), b(b), c(c), d(d) {} \
106 A a; B b; C c; D d; \
107};
108
109#define RECORD5(T, A, a, B, b, C, c, D, d, E, e) \
110struct T { \
111 static const Type kType = T##_Type; \
112 template <typename Z, typename Y, typename X, typename W, typename V> \
113 T(Z a, Y b, X c, W d, V e) : a(a), b(b), c(c), d(d), e(e) {} \
114 A a; B b; C c; D d; E e; \
115};
116
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +0000117#define ACT_AS_PTR(ptr) \
118 operator T*() { return ptr; } \
119 operator const T*() const { return ptr; } \
120 T* operator->() { return ptr; } \
121 const T* operator->() const { return ptr; }
122
mtklein53fecfb2014-08-21 09:11:37 -0700123template <typename T>
124class RefBox : SkNoncopyable {
125public:
126 RefBox(T* obj) : fObj(SkRef(obj)) {}
127 ~RefBox() { fObj->unref(); }
128
129 ACT_AS_PTR(fObj);
130
131private:
132 T* fObj;
133};
134
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000135// An Optional doesn't own the pointer's memory, but may need to destroy non-POD data.
136template <typename T>
137class Optional : SkNoncopyable {
138public:
139 Optional(T* ptr) : fPtr(ptr) {}
140 ~Optional() { if (fPtr) fPtr->~T(); }
141
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +0000142 ACT_AS_PTR(fPtr);
143private:
144 T* fPtr;
145};
146
147// Like Optional, but ptr must not be NULL.
148template <typename T>
149class Adopted : SkNoncopyable {
150public:
151 Adopted(T* ptr) : fPtr(ptr) { SkASSERT(fPtr); }
commit-bot@chromium.orgf0ae5e42014-04-24 15:33:48 +0000152 Adopted(Adopted* source) {
153 // Transfer ownership from source to this.
154 fPtr = source->fPtr;
155 source->fPtr = NULL;
156 }
157 ~Adopted() { if (fPtr) fPtr->~T(); }
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +0000158
159 ACT_AS_PTR(fPtr);
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000160private:
161 T* fPtr;
162};
163
164// PODArray doesn't own the pointer's memory, and we assume the data is POD.
165template <typename T>
commit-bot@chromium.orgf0ae5e42014-04-24 15:33:48 +0000166class PODArray {
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000167public:
168 PODArray(T* ptr) : fPtr(ptr) {}
commit-bot@chromium.orgf0ae5e42014-04-24 15:33:48 +0000169 // Default copy and assign.
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000170
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +0000171 ACT_AS_PTR(fPtr);
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000172private:
173 T* fPtr;
174};
175
commit-bot@chromium.org16307bd2014-04-22 18:32:58 +0000176#undef ACT_AS_PTR
177
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000178// Like SkBitmap, but deep copies pixels if they're not immutable.
179// Using this, we guarantee the immutability of all bitmaps we record.
180class ImmutableBitmap {
181public:
182 explicit ImmutableBitmap(const SkBitmap& bitmap) {
183 if (bitmap.isImmutable()) {
184 fBitmap = bitmap;
185 } else {
186 bitmap.copyTo(&fBitmap);
187 }
188 fBitmap.setImmutable();
189 }
190
191 operator const SkBitmap& () const { return fBitmap; }
192
193private:
194 SkBitmap fBitmap;
195};
196
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +0000197RECORD0(NoOp);
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000198
mtkleina723b572014-08-15 11:49:49 -0700199RECORD2(Restore, SkIRect, devBounds, SkMatrix, matrix);
Florin Malita5f6102d2014-06-30 10:13:28 -0400200RECORD0(Save);
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000201RECORD3(SaveLayer, Optional<SkRect>, bounds, Optional<SkPaint>, paint, SkCanvas::SaveFlags, flags);
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000202
commit-bot@chromium.org88c3e272014-04-22 16:57:20 +0000203RECORD1(PushCull, SkRect, rect);
commit-bot@chromium.org03a99b82014-04-08 15:17:17 +0000204RECORD0(PopCull);
205
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000206RECORD1(SetMatrix, SkMatrix, matrix);
207
mtkleina723b572014-08-15 11:49:49 -0700208RECORD4(ClipPath, SkIRect, devBounds, SkPath, path, SkRegion::Op, op, bool, doAA);
209RECORD4(ClipRRect, SkIRect, devBounds, SkRRect, rrect, SkRegion::Op, op, bool, doAA);
210RECORD4(ClipRect, SkIRect, devBounds, SkRect, rect, SkRegion::Op, op, bool, doAA);
211RECORD3(ClipRegion, SkIRect, devBounds, SkRegion, region, SkRegion::Op, op);
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000212
213RECORD1(Clear, SkColor, color);
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000214// While not strictly required, if you have an SkPaint, it's fastest to put it first.
215RECORD4(DrawBitmap, Optional<SkPaint>, paint,
216 ImmutableBitmap, bitmap,
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000217 SkScalar, left,
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000218 SkScalar, top);
219RECORD3(DrawBitmapMatrix, Optional<SkPaint>, paint, ImmutableBitmap, bitmap, SkMatrix, matrix);
220RECORD4(DrawBitmapNine, Optional<SkPaint>, paint,
221 ImmutableBitmap, bitmap,
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000222 SkIRect, center,
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000223 SkRect, dst);
224RECORD5(DrawBitmapRectToRect, Optional<SkPaint>, paint,
225 ImmutableBitmap, bitmap,
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000226 Optional<SkRect>, src,
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000227 SkRect, dst,
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000228 SkCanvas::DrawBitmapRectFlags, flags);
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000229RECORD3(DrawDRRect, SkPaint, paint, SkRRect, outer, SkRRect, inner);
230RECORD2(DrawOval, SkPaint, paint, SkRect, oval);
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000231RECORD1(DrawPaint, SkPaint, paint);
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000232RECORD2(DrawPath, SkPaint, paint, SkPath, path);
dandovb3c9d1c2014-08-12 08:34:29 -0700233//RECORD2(DrawPatch, SkPaint, paint, SkPatch, patch);
mtklein53fecfb2014-08-21 09:11:37 -0700234RECORD3(DrawPicture, Optional<SkPaint>, paint,
235 RefBox<const SkPicture>, picture,
236 Optional<SkMatrix>, matrix);
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000237RECORD4(DrawPoints, SkPaint, paint, SkCanvas::PointMode, mode, size_t, count, SkPoint*, pts);
238RECORD4(DrawPosText, SkPaint, paint,
239 PODArray<char>, text,
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000240 size_t, byteLength,
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000241 PODArray<SkPoint>, pos);
242RECORD5(DrawPosTextH, SkPaint, paint,
243 PODArray<char>, text,
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000244 size_t, byteLength,
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000245 PODArray<SkScalar>, xpos,
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000246 SkScalar, y);
247RECORD2(DrawRRect, SkPaint, paint, SkRRect, rrect);
248RECORD2(DrawRect, SkPaint, paint, SkRect, rect);
249RECORD4(DrawSprite, Optional<SkPaint>, paint, ImmutableBitmap, bitmap, int, left, int, top);
250RECORD5(DrawText, SkPaint, paint,
251 PODArray<char>, text,
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000252 size_t, byteLength,
253 SkScalar, x,
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000254 SkScalar, y);
fmalita00d5c2c2014-08-21 08:53:26 -0700255RECORD4(DrawTextBlob, SkPaint, paint,
mtklein53fecfb2014-08-21 09:11:37 -0700256 RefBox<const SkTextBlob>, blob,
fmalita00d5c2c2014-08-21 08:53:26 -0700257 SkScalar, x,
258 SkScalar, y);
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000259RECORD5(DrawTextOnPath, SkPaint, paint,
260 PODArray<char>, text,
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000261 size_t, byteLength,
262 SkPath, path,
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000263 Optional<SkMatrix>, matrix);
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000264
265// This guy is so ugly we just write it manually.
266struct DrawVertices {
267 static const Type kType = DrawVertices_Type;
268
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000269 DrawVertices(const SkPaint& paint,
270 SkCanvas::VertexMode vmode,
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000271 int vertexCount,
272 SkPoint* vertices,
273 SkPoint* texs,
274 SkColor* colors,
275 SkXfermode* xmode,
276 uint16_t* indices,
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000277 int indexCount)
278 : paint(paint)
279 , vmode(vmode)
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000280 , vertexCount(vertexCount)
281 , vertices(vertices)
282 , texs(texs)
283 , colors(colors)
284 , xmode(SkSafeRef(xmode))
285 , indices(indices)
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000286 , indexCount(indexCount) {}
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000287
commit-bot@chromium.org37f6e622014-05-07 22:59:38 +0000288 SkPaint paint;
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000289 SkCanvas::VertexMode vmode;
290 int vertexCount;
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000291 PODArray<SkPoint> vertices;
292 PODArray<SkPoint> texs;
293 PODArray<SkColor> colors;
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000294 SkAutoTUnref<SkXfermode> xmode;
commit-bot@chromium.org653d5182014-04-15 14:27:14 +0000295 PODArray<uint16_t> indices;
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000296 int indexCount;
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000297};
mtklein6cfa73a2014-08-13 13:33:49 -0700298
dandovb3c9d1c2014-08-12 08:34:29 -0700299struct DrawPatch {
300 static const Type kType = DrawPatch_Type;
mtklein6cfa73a2014-08-13 13:33:49 -0700301
dandovb3c9d1c2014-08-12 08:34:29 -0700302 DrawPatch(const SkPaint& paint, SkPoint cubics[12], SkColor colors[4],
303 SkPoint texCoords[4], SkXfermode* xmode)
304 : paint(paint)
305 , cubics(cubics)
306 , colors(colors)
307 , texCoords(texCoords)
308 , xmode(SkSafeRef(xmode)) { }
mtklein6cfa73a2014-08-13 13:33:49 -0700309
dandovb3c9d1c2014-08-12 08:34:29 -0700310 SkPaint paint;
311 PODArray<SkPoint> cubics;
312 PODArray<SkColor> colors;
313 PODArray<SkPoint> texCoords;
314 SkAutoTUnref<SkXfermode> xmode;
315};
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000316
commit-bot@chromium.orge3ff5582014-04-01 16:24:06 +0000317#undef RECORD0
318#undef RECORD1
319#undef RECORD2
320#undef RECORD3
321#undef RECORD4
322#undef RECORD5
323
324} // namespace SkRecords
325
326#endif//SkRecords_DEFINED