1/*
2 * Copyright 2011 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 SkPictureRecord_DEFINED
9#define SkPictureRecord_DEFINED
10
11#include "include/core/SkCanvas.h"
12#include "include/core/SkCanvasVirtualEnforcer.h"
13#include "include/core/SkFlattenable.h"
14#include "include/core/SkPicture.h"
15#include "include/core/SkVertices.h"
16#include "include/private/SkTArray.h"
17#include "include/private/SkTDArray.h"
18#include "include/private/SkTHash.h"
19#include "include/private/SkTo.h"
20#include "src/core/SkPictureData.h"
21#include "src/core/SkWriter32.h"
22
23// These macros help with packing and unpacking a single byte value and
24// a 3 byte value into/out of a uint32_t
25#define MASK_24 0x00FFFFFF
26#define UNPACK_8_24(combined, small, large) \
27 small = (combined >> 24) & 0xFF; \
28 large = combined & MASK_24
29#define PACK_8_24(small, large) ((small << 24) | large)
30
31
32class SkPictureRecord : public SkCanvasVirtualEnforcer<SkCanvas> {
33public:
34 SkPictureRecord(const SkISize& dimensions, uint32_t recordFlags);
35
36 SkPictureRecord(const SkIRect& dimensions, uint32_t recordFlags);
37
38 const SkTArray<sk_sp<const SkPicture>>& getPictures() const {
39 return fPictures;
40 }
41
42 const SkTArray<sk_sp<SkDrawable>>& getDrawables() const {
43 return fDrawables;
44 }
45
46 const SkTArray<sk_sp<const SkTextBlob>>& getTextBlobs() const {
47 return fTextBlobs;
48 }
49
50 const SkTArray<sk_sp<const SkVertices>>& getVertices() const {
51 return fVertices;
52 }
53
54 const SkTArray<sk_sp<const SkImage>>& getImages() const {
55 return fImages;
56 }
57
58 sk_sp<SkData> opData() const {
59 this->validate(fWriter.bytesWritten(), 0);
60
61 if (fWriter.bytesWritten() == 0) {
62 return SkData::MakeEmpty();
63 }
64 return fWriter.snapshotAsData();
65 }
66
67 void setFlags(uint32_t recordFlags) {
68 fRecordFlags = recordFlags;
69 }
70
71 const SkWriter32& writeStream() const {
72 return fWriter;
73 }
74
75 void beginRecording();
76 void endRecording();
77
78protected:
79 void addNoOp();
80
81private:
82 void handleOptimization(int opt);
83 size_t recordRestoreOffsetPlaceholder(SkClipOp);
84 void fillRestoreOffsetPlaceholdersForCurrentStackLevel(uint32_t restoreOffset);
85
86 SkTDArray<int32_t> fRestoreOffsetStack;
87
88 SkTDArray<uint32_t> fCullOffsetStack;
89
90 /*
91 * Write the 'drawType' operation and chunk size to the skp. 'size'
92 * can potentially be increased if the chunk size needs its own storage
93 * location (i.e., it overflows 24 bits).
94 * Returns the start offset of the chunk. This is the location at which
95 * the opcode & size are stored.
96 * TODO: since we are handing the size into here we could call reserve
97 * and then return a pointer to the memory storage. This could decrease
98 * allocation overhead but could lead to more wasted space (the tail
99 * end of blocks could go unused). Possibly add a second addDraw that
100 * operates in this manner.
101 */
102 size_t addDraw(DrawType drawType, size_t* size) {
103 size_t offset = fWriter.bytesWritten();
104
105 this->predrawNotify();
106
107 SkASSERT(0 != *size);
108 SkASSERT(((uint8_t) drawType) == drawType);
109
110 if (0 != (*size & ~MASK_24) || *size == MASK_24) {
111 fWriter.writeInt(PACK_8_24(drawType, MASK_24));
112 *size += 1;
113 fWriter.writeInt(SkToU32(*size));
114 } else {
115 fWriter.writeInt(PACK_8_24(drawType, SkToU32(*size)));
116 }
117
118 return offset;
119 }
120
121 void addInt(int value) {
122 fWriter.writeInt(value);
123 }
124 void addScalar(SkScalar scalar) {
125 fWriter.writeScalar(scalar);
126 }
127
128 void addImage(const SkImage*);
129 void addMatrix(const SkMatrix& matrix);
130 void addPaint(const SkPaint& paint) { this->addPaintPtr(&paint); }
131 void addPaintPtr(const SkPaint* paint);
132 void addPatch(const SkPoint cubics[12]);
133 void addPath(const SkPath& path);
134 void addPicture(const SkPicture* picture);
135 void addDrawable(SkDrawable* picture);
136 void addPoint(const SkPoint& point);
137 void addPoints(const SkPoint pts[], int count);
138 void addRect(const SkRect& rect);
139 void addRectPtr(const SkRect* rect);
140 void addIRect(const SkIRect& rect);
141 void addIRectPtr(const SkIRect* rect);
142 void addRRect(const SkRRect&);
143 void addRegion(const SkRegion& region);
144 void addText(const void* text, size_t byteLength);
145 void addTextBlob(const SkTextBlob* blob);
146 void addVertices(const SkVertices*);
147
148 int find(const SkBitmap& bitmap);
149
150protected:
151 void validate(size_t initialOffset, size_t size) const {
152 SkASSERT(fWriter.bytesWritten() == initialOffset + size);
153 }
154
155 sk_sp<SkSurface> onNewSurface(const SkImageInfo&, const SkSurfaceProps&) override;
156 bool onPeekPixels(SkPixmap*) override { return false; }
157
158 void onFlush() override;
159
160 void willSave() override;
161 SaveLayerStrategy getSaveLayerStrategy(const SaveLayerRec&) override;
162 bool onDoSaveBehind(const SkRect*) override;
163 void willRestore() override;
164
165 void onMarkCTM(const char*) override;
166 void didConcat44(const SkM44&) override;
167 void didConcat(const SkMatrix&) override;
168 void didSetMatrix(const SkMatrix&) override;
169 void didScale(SkScalar, SkScalar) override;
170 void didTranslate(SkScalar, SkScalar) override;
171
172 void onDrawDRRect(const SkRRect&, const SkRRect&, const SkPaint&) override;
173
174 void onDrawTextBlob(const SkTextBlob* blob, SkScalar x, SkScalar y,
175 const SkPaint& paint) override;
176
177 void onDrawPatch(const SkPoint cubics[12], const SkColor colors[4],
178 const SkPoint texCoords[4], SkBlendMode, const SkPaint& paint) override;
179 void onDrawAtlas(const SkImage*, const SkRSXform[], const SkRect[], const SkColor[], int,
180 SkBlendMode, const SkRect*, const SkPaint*) override;
181
182 void onDrawPaint(const SkPaint&) override;
183 void onDrawBehind(const SkPaint&) override;
184 void onDrawPoints(PointMode, size_t count, const SkPoint pts[], const SkPaint&) override;
185 void onDrawRect(const SkRect&, const SkPaint&) override;
186 void onDrawRegion(const SkRegion&, const SkPaint&) override;
187 void onDrawOval(const SkRect&, const SkPaint&) override;
188 void onDrawArc(const SkRect&, SkScalar, SkScalar, bool, const SkPaint&) override;
189 void onDrawRRect(const SkRRect&, const SkPaint&) override;
190 void onDrawPath(const SkPath&, const SkPaint&) override;
191 void onDrawImage(const SkImage*, SkScalar left, SkScalar top, const SkPaint*) override;
192 void onDrawImageRect(const SkImage*, const SkRect* src, const SkRect& dst,
193 const SkPaint*, SrcRectConstraint) override;
194 void onDrawImageNine(const SkImage*, const SkIRect& center, const SkRect& dst,
195 const SkPaint*) override;
196 void onDrawImageLattice(const SkImage*, const SkCanvas::Lattice& lattice, const SkRect& dst,
197 const SkPaint*) override;
198
199 void onDrawShadowRec(const SkPath&, const SkDrawShadowRec&) override;
200 void onDrawVerticesObject(const SkVertices*, SkBlendMode, const SkPaint&) override;
201
202 void onClipRect(const SkRect&, SkClipOp, ClipEdgeStyle) override;
203 void onClipRRect(const SkRRect&, SkClipOp, ClipEdgeStyle) override;
204 void onClipPath(const SkPath&, SkClipOp, ClipEdgeStyle) override;
205 void onClipShader(sk_sp<SkShader>, SkClipOp) override;
206 void onClipRegion(const SkRegion&, SkClipOp) override;
207
208 void onDrawPicture(const SkPicture*, const SkMatrix*, const SkPaint*) override;
209
210 void onDrawDrawable(SkDrawable*, const SkMatrix*) override;
211 void onDrawAnnotation(const SkRect&, const char[], SkData*) override;
212
213 void onDrawEdgeAAQuad(const SkRect&, const SkPoint[4], QuadAAFlags, const SkColor4f&,
214 SkBlendMode) override;
215 void onDrawEdgeAAImageSet(const ImageSetEntry[], int count, const SkPoint[], const SkMatrix[],
216 const SkPaint*, SrcRectConstraint) override;
217
218 int addPathToHeap(const SkPath& path); // does not write to ops stream
219
220 // These entry points allow the writing of matrices, clips, saves &
221 // restores to be deferred (e.g., if the MC state is being collapsed and
222 // only written out as needed).
223 void recordConcat(const SkMatrix& matrix);
224 void recordTranslate(const SkMatrix& matrix);
225 void recordScale(const SkMatrix& matrix);
226 size_t recordClipRect(const SkRect& rect, SkClipOp op, bool doAA);
227 size_t recordClipRRect(const SkRRect& rrect, SkClipOp op, bool doAA);
228 size_t recordClipPath(int pathID, SkClipOp op, bool doAA);
229 size_t recordClipRegion(const SkRegion& region, SkClipOp op);
230 void recordSave();
231 void recordSaveLayer(const SaveLayerRec&);
232 void recordRestore(bool fillInSkips = true);
233
234private:
235 SkTArray<SkPaint> fPaints;
236
237 struct PathHash {
238 uint32_t operator()(const SkPath& p) { return p.getGenerationID(); }
239 };
240 SkTHashMap<SkPath, int, PathHash> fPaths;
241
242 SkWriter32 fWriter;
243
244 SkTArray<sk_sp<const SkImage>> fImages;
245 SkTArray<sk_sp<const SkPicture>> fPictures;
246 SkTArray<sk_sp<SkDrawable>> fDrawables;
247 SkTArray<sk_sp<const SkTextBlob>> fTextBlobs;
248 SkTArray<sk_sp<const SkVertices>> fVertices;
249
250 uint32_t fRecordFlags;
251 int fInitialSaveCount;
252
253 friend class SkPictureData; // for SkPictureData's SkPictureRecord-based constructor
254
255 typedef SkCanvasVirtualEnforcer<SkCanvas> INHERITED;
256};
257
258#endif
259