blob: f04bb846d6af67f7a9835382650785e2937f9761 [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,
jvanverth@google.com39768252013-02-14 15:25:44 +000081 const GrRect* srcRect,
82 const SkMatrix* srcMatrix,
83 int stage) SK_OVERRIDE;
bsalomon@google.com0b335c12011-04-25 19:17:44 +000084
bsalomon@google.com97805382012-03-13 14:32:07 +000085protected:
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +000086 virtual void clipWillBeSet(const GrClipData* newClip) SK_OVERRIDE;
87
reed@google.comac10a2d2010-12-22 21:39:39 +000088private:
bsalomon@google.coma4f6b102012-06-26 21:04:22 +000089 enum Cmd {
bsalomon@google.comded4f4b2012-06-28 18:48:06 +000090 kDraw_Cmd = 1,
91 kStencilPath_Cmd = 2,
92 kSetState_Cmd = 3,
93 kSetClip_Cmd = 4,
94 kClear_Cmd = 5,
bsalomon@google.coma4f6b102012-06-26 21:04:22 +000095 };
96
bsalomon@google.comd62e88e2013-02-01 14:19:27 +000097 class DrawRecord : public DrawInfo {
98 public:
99 DrawRecord(const DrawInfo& info) : DrawInfo(info) {}
bsalomon@google.com1c13c962011-02-14 16:51:21 +0000100 const GrVertexBuffer* fVertexBuffer;
101 const GrIndexBuffer* fIndexBuffer;
reed@google.comac10a2d2010-12-22 21:39:39 +0000102 };
103
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000104 struct StencilPath {
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000105 StencilPath();
106
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000107 SkAutoTUnref<const GrPath> fPath;
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000108 SkStrokeRec fStroke;
sugoi@google.com12b4e272012-12-06 20:13:11 +0000109 SkPath::FillType fFill;
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000110 };
111
bsalomon@google.com0b335c12011-04-25 19:17:44 +0000112 struct Clear {
robertphillips@google.comc82a8b72012-06-21 20:15:48 +0000113 Clear() : fRenderTarget(NULL) {}
114 ~Clear() { GrSafeUnref(fRenderTarget); }
115
robertphillips@google.comc82a8b72012-06-21 20:15:48 +0000116 GrIRect fRect;
117 GrColor fColor;
118 GrRenderTarget* fRenderTarget;
bsalomon@google.com0b335c12011-04-25 19:17:44 +0000119 };
120
bsalomon@google.com25fb21f2011-06-21 18:17:25 +0000121 // overrides from GrDrawTarget
bsalomon@google.com74749cd2013-01-30 16:12:41 +0000122 virtual void onDraw(const DrawInfo&) SK_OVERRIDE;
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000123 virtual void onStencilPath(const GrPath*, const SkStrokeRec& stroke, SkPath::FillType) SK_OVERRIDE;
jvanverth@google.coma6338982013-01-31 21:34:25 +0000124 virtual bool onReserveVertexSpace(size_t vertexSize,
bsalomon@google.com25fb21f2011-06-21 18:17:25 +0000125 int vertexCount,
bsalomon@google.com13f1b6f2012-05-31 12:52:43 +0000126 void** vertices) SK_OVERRIDE;
127 virtual bool onReserveIndexSpace(int indexCount,
128 void** indices) SK_OVERRIDE;
129 virtual void releaseReservedVertexSpace() SK_OVERRIDE;
130 virtual void releaseReservedIndexSpace() SK_OVERRIDE;
bsalomon@google.combcdbbe62011-04-12 15:40:00 +0000131 virtual void onSetVertexSourceToArray(const void* vertexArray,
bsalomon@google.com13f1b6f2012-05-31 12:52:43 +0000132 int vertexCount) SK_OVERRIDE;
bsalomon@google.combcdbbe62011-04-12 15:40:00 +0000133 virtual void onSetIndexSourceToArray(const void* indexArray,
bsalomon@google.com13f1b6f2012-05-31 12:52:43 +0000134 int indexCount) SK_OVERRIDE;
135 virtual void releaseVertexArray() SK_OVERRIDE;
136 virtual void releaseIndexArray() SK_OVERRIDE;
137 virtual void geometrySourceWillPush() SK_OVERRIDE;
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +0000138 virtual void geometrySourceWillPop(const GeometrySrcState& restoredState) SK_OVERRIDE;
jvanverth@google.comb75b0a02013-02-05 20:33:30 +0000139 virtual void willReserveVertexAndIndexSpace(int vertexCount,
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +0000140 int indexCount) SK_OVERRIDE;
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000141 bool quickInsideClip(const SkRect& devBounds);
bsalomon@google.com02ddc8b2013-01-28 15:35:28 +0000142
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000143 // Attempts to concat instances from info onto the previous draw. info must represent an
144 // instanced draw. The caller must have already recorded a new draw state and clip if necessary.
145 int concatInstancedDraw(const DrawInfo& info);
bsalomon@google.com1c13c962011-02-14 16:51:21 +0000146
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000147 // we lazily record state and clip changes in order to skip clips and states that have no
148 // effect.
bsalomon@google.com86afc2a2011-02-16 16:12:19 +0000149 bool needsNewState() const;
150 bool needsNewClip() const;
reed@google.comac10a2d2010-12-22 21:39:39 +0000151
bsalomon@google.coma4f6b102012-06-26 21:04:22 +0000152 // these functions record a command
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000153 void recordState();
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000154 void recordClip();
bsalomon@google.com74749cd2013-01-30 16:12:41 +0000155 DrawRecord* recordDraw(const DrawInfo&);
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000156 StencilPath* recordStencilPath();
157 Clear* recordClear();
bsalomon@google.com934c5702012-03-20 21:17:58 +0000158
bsalomon@google.com4b90c622011-09-28 17:52:15 +0000159 enum {
bsalomon@google.coma4f6b102012-06-26 21:04:22 +0000160 kCmdPreallocCnt = 32,
bsalomon@google.com4b90c622011-09-28 17:52:15 +0000161 kDrawPreallocCnt = 8,
bsalomon@google.comded4f4b2012-06-28 18:48:06 +0000162 kStencilPathPreallocCnt = 8,
bsalomon@google.com4b90c622011-09-28 17:52:15 +0000163 kStatePreallocCnt = 8,
164 kClipPreallocCnt = 8,
165 kClearPreallocCnt = 4,
166 kGeoPoolStatePreAllocCnt = 4,
167 };
reed@google.comac10a2d2010-12-22 21:39:39 +0000168
bsalomon@google.comca432082013-01-23 19:53:46 +0000169 SkSTArray<kCmdPreallocCnt, uint8_t, true> fCmds;
bsalomon@google.com74749cd2013-01-30 16:12:41 +0000170 GrSTAllocator<kDrawPreallocCnt, DrawRecord> fDraws;
bsalomon@google.comca432082013-01-23 19:53:46 +0000171 GrSTAllocator<kStatePreallocCnt, StencilPath> fStencilPaths;
172 GrSTAllocator<kStatePreallocCnt, GrDrawState::DeferredState> fStates;
173 GrSTAllocator<kClearPreallocCnt, Clear> fClears;
robertphillips@google.combeb1af72012-07-26 18:52:16 +0000174
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000175 GrSTAllocator<kClipPreallocCnt, SkClipStack> fClips;
robertphillips@google.combeb1af72012-07-26 18:52:16 +0000176 GrSTAllocator<kClipPreallocCnt, SkIPoint> fClipOrigins;
bsalomon@google.com97805382012-03-13 14:32:07 +0000177
bsalomon@google.com6e4e6502013-02-25 20:12:45 +0000178 GrDrawTarget* fDstGpu;
bsalomon@google.com97805382012-03-13 14:32:07 +0000179
bsalomon@google.com86afc2a2011-02-16 16:12:19 +0000180 bool fClipSet;
181
bsalomon@google.comd62e88e2013-02-01 14:19:27 +0000182 enum ClipProxyState {
183 kUnknown_ClipProxyState,
184 kValid_ClipProxyState,
185 kInvalid_ClipProxyState
186 };
187 ClipProxyState fClipProxyState;
188 SkRect fClipProxy;
189
bsalomon@google.com934c5702012-03-20 21:17:58 +0000190 GrVertexBufferAllocPool& fVertexPool;
191
192 GrIndexBufferAllocPool& fIndexPool;
193
bsalomon@google.com25fb21f2011-06-21 18:17:25 +0000194 struct GeometryPoolState {
195 const GrVertexBuffer* fPoolVertexBuffer;
196 int fPoolStartVertex;
197 const GrIndexBuffer* fPoolIndexBuffer;
198 int fPoolStartIndex;
199 // caller may conservatively over reserve vertices / indices.
200 // we release unused space back to allocator if possible
201 // can only do this if there isn't an intervening pushGeometrySource()
202 size_t fUsedPoolVertexBytes;
203 size_t fUsedPoolIndexBytes;
204 };
bsalomon@google.com92669012011-09-27 19:10:05 +0000205 SkSTArray<kGeoPoolStatePreAllocCnt, GeometryPoolState> fGeoPoolStateStack;
reed@google.comac10a2d2010-12-22 21:39:39 +0000206
robertphillips@google.comc82a8b72012-06-21 20:15:48 +0000207 bool fFlushing;
208
bsalomon@google.com86afc2a2011-02-16 16:12:19 +0000209 typedef GrDrawTarget INHERITED;
reed@google.comac10a2d2010-12-22 21:39:39 +0000210};
211
212#endif