blob: cb51bc46726b8e8f2b5fea96c35efa49bbede30d [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
reed@google.comac10a2d2010-12-22 21:39:39 +00002/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
reed@google.comac10a2d2010-12-22 21:39:39 +00007 */
8
9
epoger@google.comec3ed6a2011-07-28 14:26:00 +000010
reed@google.comac10a2d2010-12-22 21:39:39 +000011#ifndef GrInOrderDrawBuffer_DEFINED
12#define GrInOrderDrawBuffer_DEFINED
13
14#include "GrDrawTarget.h"
15#include "GrAllocPool.h"
16#include "GrAllocator.h"
bsalomon@google.comded4f4b2012-06-28 18:48:06 +000017#include "GrPath.h"
reed@google.comac10a2d2010-12-22 21:39:39 +000018
robertphillips@google.com641f8b12012-07-31 19:15:58 +000019#include "SkClipStack.h"
sugoi@google.com5f74cf82012-12-17 21:16:45 +000020#include "SkStrokeRec.h"
bsalomon@google.comded4f4b2012-06-28 18:48:06 +000021#include "SkTemplates.h"
22
bsalomon@google.com471d4712011-08-23 15:45:25 +000023class GrGpu;
bsalomon@google.com1c13c962011-02-14 16:51:21 +000024class GrIndexBufferAllocPool;
bsalomon@google.com471d4712011-08-23 15:45:25 +000025class GrVertexBufferAllocPool;
reed@google.comac10a2d2010-12-22 21:39:39 +000026
bsalomon@google.com1c13c962011-02-14 16:51:21 +000027/**
bsalomon@google.com55e4a202013-01-11 13:54:21 +000028 * GrInOrderDrawBuffer is an implementation of GrDrawTarget that queues up draws for eventual
29 * playback into a GrGpu. In theory one draw buffer could playback into another. When index or
30 * vertex buffers are used as geometry sources it is the callers the draw buffer only holds
31 * references to the buffers. It is the callers responsibility to ensure that the data is still
32 * valid when the draw buffer is played back into a GrGpu. Similarly, it is the caller's
33 * responsibility to ensure that all referenced textures, buffers, and render-targets are associated
34 * in the GrGpu object that the buffer is played back into. The buffer requires VB and IB pools to
35 * store geometry.
bsalomon@google.com1c13c962011-02-14 16:51:21 +000036 */
reed@google.comac10a2d2010-12-22 21:39:39 +000037class GrInOrderDrawBuffer : public GrDrawTarget {
38public:
39
bsalomon@google.com86afc2a2011-02-16 16:12:19 +000040 /**
41 * Creates a GrInOrderDrawBuffer
42 *
bsalomon@google.com6e4e6502013-02-25 20:12:45 +000043 * @param gpu the gpu object that this draw buffer flushes to.
bsalomon@google.com86afc2a2011-02-16 16:12:19 +000044 * @param vertexPool pool where vertices for queued draws will be saved when
45 * the vertex source is either reserved or array.
46 * @param indexPool pool where indices for queued draws will be saved when
47 * the index source is either reserved or array.
48 */
bsalomon@google.com6e4e6502013-02-25 20:12:45 +000049 GrInOrderDrawBuffer(GrGpu* gpu,
bsalomon@google.com471d4712011-08-23 15:45:25 +000050 GrVertexBufferAllocPool* vertexPool,
bsalomon@google.com1c13c962011-02-14 16:51:21 +000051 GrIndexBufferAllocPool* indexPool);
reed@google.comac10a2d2010-12-22 21:39:39 +000052
53 virtual ~GrInOrderDrawBuffer();
54
bsalomon@google.com86afc2a2011-02-16 16:12:19 +000055 /**
bsalomon@google.com55e4a202013-01-11 13:54:21 +000056 * Empties the draw buffer of any queued up draws. This must not be called while inside an
57 * unbalanced pushGeometrySource(). The current draw state and clip are preserved.
bsalomon@google.com86afc2a2011-02-16 16:12:19 +000058 */
59 void reset();
60
61 /**
bsalomon@google.com6e4e6502013-02-25 20:12:45 +000062 * This plays the queued up draws to its GrGpu target. It also resets this object (i.e. flushing
bsalomon@google.com55e4a202013-01-11 13:54:21 +000063 * is destructive). This buffer must not have an active reserved vertex or index source. Any
64 * reserved geometry on the target will be finalized because it's geometry source will be pushed
65 * before flushing and popped afterwards.
bsalomon@google.com97805382012-03-13 14:32:07 +000066 *
bsalomon@google.com55e4a202013-01-11 13:54:21 +000067 * @return false if the playback trivially drew nothing because nothing was recorded.
bsalomon@google.com86afc2a2011-02-16 16:12:19 +000068 */
bsalomon@google.com6e4e6502013-02-25 20:12:45 +000069 bool flush();
bsalomon@google.com97805382012-03-13 14:32:07 +000070
bsalomon@google.com6e4e6502013-02-25 20:12:45 +000071 bool isFlushing() const { return fFlushing; }
bsalomon@google.com97805382012-03-13 14:32:07 +000072
bsalomon@google.com86afc2a2011-02-16 16:12:19 +000073 // overrides from GrDrawTarget
jvanverth@google.comb75b0a02013-02-05 20:33:30 +000074 virtual bool geometryHints(int* vertexCount,
bsalomon@google.com97805382012-03-13 14:32:07 +000075 int* indexCount) const SK_OVERRIDE;
rmistry@google.comfbfcd562012-08-23 18:09:54 +000076 virtual void clear(const GrIRect* rect,
robertphillips@google.comc82a8b72012-06-21 20:15:48 +000077 GrColor color,
78 GrRenderTarget* renderTarget = NULL) SK_OVERRIDE;
bsalomon@google.comd62e88e2013-02-01 14:19:27 +000079 virtual void drawRect(const GrRect& rect,
80 const SkMatrix* matrix,
bsalomon@google.comc7818882013-03-20 19:19:53 +000081 const GrRect* localRect,
82 const SkMatrix* localMatrix) SK_OVERRIDE;
bsalomon@google.com0b335c12011-04-25 19:17:44 +000083
bsalomon@google.com97805382012-03-13 14:32:07 +000084protected:
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +000085 virtual void clipWillBeSet(const GrClipData* newClip) SK_OVERRIDE;
86
reed@google.comac10a2d2010-12-22 21:39:39 +000087private:
bsalomon@google.coma4f6b102012-06-26 21:04:22 +000088 enum Cmd {
bsalomon@google.comded4f4b2012-06-28 18:48:06 +000089 kDraw_Cmd = 1,
90 kStencilPath_Cmd = 2,
91 kSetState_Cmd = 3,
92 kSetClip_Cmd = 4,
93 kClear_Cmd = 5,
bsalomon@google.coma4f6b102012-06-26 21:04:22 +000094 };
95
bsalomon@google.comd62e88e2013-02-01 14:19:27 +000096 class DrawRecord : public DrawInfo {
97 public:
98 DrawRecord(const DrawInfo& info) : DrawInfo(info) {}
bsalomon@google.com1c13c962011-02-14 16:51:21 +000099 const GrVertexBuffer* fVertexBuffer;
100 const GrIndexBuffer* fIndexBuffer;
reed@google.comac10a2d2010-12-22 21:39:39 +0000101 };
102
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000103 struct StencilPath {
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000104 StencilPath();
105
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000106 SkAutoTUnref<const GrPath> fPath;
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000107 SkStrokeRec fStroke;
sugoi@google.com12b4e272012-12-06 20:13:11 +0000108 SkPath::FillType fFill;
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000109 };
110
bsalomon@google.com0b335c12011-04-25 19:17:44 +0000111 struct Clear {
robertphillips@google.comc82a8b72012-06-21 20:15:48 +0000112 Clear() : fRenderTarget(NULL) {}
113 ~Clear() { GrSafeUnref(fRenderTarget); }
114
robertphillips@google.comc82a8b72012-06-21 20:15:48 +0000115 GrIRect fRect;
116 GrColor fColor;
117 GrRenderTarget* fRenderTarget;
bsalomon@google.com0b335c12011-04-25 19:17:44 +0000118 };
119
bsalomon@google.com25fb21f2011-06-21 18:17:25 +0000120 // overrides from GrDrawTarget
bsalomon@google.com74749cd2013-01-30 16:12:41 +0000121 virtual void onDraw(const DrawInfo&) SK_OVERRIDE;
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000122 virtual void onStencilPath(const GrPath*, const SkStrokeRec& stroke, SkPath::FillType) SK_OVERRIDE;
jvanverth@google.coma6338982013-01-31 21:34:25 +0000123 virtual bool onReserveVertexSpace(size_t vertexSize,
bsalomon@google.com25fb21f2011-06-21 18:17:25 +0000124 int vertexCount,
bsalomon@google.com13f1b6f2012-05-31 12:52:43 +0000125 void** vertices) SK_OVERRIDE;
126 virtual bool onReserveIndexSpace(int indexCount,
127 void** indices) SK_OVERRIDE;
128 virtual void releaseReservedVertexSpace() SK_OVERRIDE;
129 virtual void releaseReservedIndexSpace() SK_OVERRIDE;
bsalomon@google.combcdbbe62011-04-12 15:40:00 +0000130 virtual void onSetVertexSourceToArray(const void* vertexArray,
bsalomon@google.com13f1b6f2012-05-31 12:52:43 +0000131 int vertexCount) SK_OVERRIDE;
bsalomon@google.combcdbbe62011-04-12 15:40:00 +0000132 virtual void onSetIndexSourceToArray(const void* indexArray,
bsalomon@google.com13f1b6f2012-05-31 12:52:43 +0000133 int indexCount) SK_OVERRIDE;
134 virtual void releaseVertexArray() SK_OVERRIDE;
135 virtual void releaseIndexArray() SK_OVERRIDE;
136 virtual void geometrySourceWillPush() SK_OVERRIDE;
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +0000137 virtual void geometrySourceWillPop(const GeometrySrcState& restoredState) SK_OVERRIDE;
jvanverth@google.comb75b0a02013-02-05 20:33:30 +0000138 virtual void willReserveVertexAndIndexSpace(int vertexCount,
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +0000139 int indexCount) SK_OVERRIDE;
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000140 bool quickInsideClip(const SkRect& devBounds);
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +0000141
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000142 // Attempts to concat instances from info onto the previous draw. info must represent an
143 // instanced draw. The caller must have already recorded a new draw state and clip if necessary.
144 int concatInstancedDraw(const DrawInfo& info);
bsalomon@google.com1c13c962011-02-14 16:51:21 +0000145
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000146 // we lazily record state and clip changes in order to skip clips and states that have no
147 // effect.
bsalomon@google.com86afc2a2011-02-16 16:12:19 +0000148 bool needsNewState() const;
149 bool needsNewClip() const;
reed@google.comac10a2d2010-12-22 21:39:39 +0000150
bsalomon@google.coma4f6b102012-06-26 21:04:22 +0000151 // these functions record a command
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000152 void recordState();
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000153 void recordClip();
bsalomon@google.com74749cd2013-01-30 16:12:41 +0000154 DrawRecord* recordDraw(const DrawInfo&);
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000155 StencilPath* recordStencilPath();
156 Clear* recordClear();
bsalomon@google.com934c5702012-03-20 21:17:58 +0000157
bsalomon@google.com4b90c622011-09-28 17:52:15 +0000158 enum {
bsalomon@google.coma4f6b102012-06-26 21:04:22 +0000159 kCmdPreallocCnt = 32,
bsalomon@google.com4b90c622011-09-28 17:52:15 +0000160 kDrawPreallocCnt = 8,
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000161 kStencilPathPreallocCnt = 8,
bsalomon@google.com4b90c622011-09-28 17:52:15 +0000162 kStatePreallocCnt = 8,
163 kClipPreallocCnt = 8,
164 kClearPreallocCnt = 4,
165 kGeoPoolStatePreAllocCnt = 4,
166 };
reed@google.comac10a2d2010-12-22 21:39:39 +0000167
bsalomon@google.comca432082013-01-23 19:53:46 +0000168 SkSTArray<kCmdPreallocCnt, uint8_t, true> fCmds;
bsalomon@google.com74749cd2013-01-30 16:12:41 +0000169 GrSTAllocator<kDrawPreallocCnt, DrawRecord> fDraws;
bsalomon@google.comca432082013-01-23 19:53:46 +0000170 GrSTAllocator<kStatePreallocCnt, StencilPath> fStencilPaths;
171 GrSTAllocator<kStatePreallocCnt, GrDrawState::DeferredState> fStates;
172 GrSTAllocator<kClearPreallocCnt, Clear> fClears;
robertphillips@google.combeb1af72012-07-26 18:52:16 +0000173
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000174 GrSTAllocator<kClipPreallocCnt, SkClipStack> fClips;
robertphillips@google.combeb1af72012-07-26 18:52:16 +0000175 GrSTAllocator<kClipPreallocCnt, SkIPoint> fClipOrigins;
bsalomon@google.com97805382012-03-13 14:32:07 +0000176
bsalomon@google.com6e4e6502013-02-25 20:12:45 +0000177 GrDrawTarget* fDstGpu;
bsalomon@google.com97805382012-03-13 14:32:07 +0000178
bsalomon@google.com86afc2a2011-02-16 16:12:19 +0000179 bool fClipSet;
180
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000181 enum ClipProxyState {
182 kUnknown_ClipProxyState,
183 kValid_ClipProxyState,
184 kInvalid_ClipProxyState
185 };
186 ClipProxyState fClipProxyState;
187 SkRect fClipProxy;
188
bsalomon@google.com934c5702012-03-20 21:17:58 +0000189 GrVertexBufferAllocPool& fVertexPool;
190
191 GrIndexBufferAllocPool& fIndexPool;
192
bsalomon@google.com25fb21f2011-06-21 18:17:25 +0000193 struct GeometryPoolState {
194 const GrVertexBuffer* fPoolVertexBuffer;
195 int fPoolStartVertex;
196 const GrIndexBuffer* fPoolIndexBuffer;
197 int fPoolStartIndex;
198 // caller may conservatively over reserve vertices / indices.
199 // we release unused space back to allocator if possible
200 // can only do this if there isn't an intervening pushGeometrySource()
201 size_t fUsedPoolVertexBytes;
202 size_t fUsedPoolIndexBytes;
203 };
bsalomon@google.com92669012011-09-27 19:10:05 +0000204 SkSTArray<kGeoPoolStatePreAllocCnt, GeometryPoolState> fGeoPoolStateStack;
reed@google.comac10a2d2010-12-22 21:39:39 +0000205
robertphillips@google.comc82a8b72012-06-21 20:15:48 +0000206 bool fFlushing;
207
bsalomon@google.com86afc2a2011-02-16 16:12:19 +0000208 typedef GrDrawTarget INHERITED;
reed@google.comac10a2d2010-12-22 21:39:39 +0000209};
210
211#endif