blob: e4d7223c7e0cdc807d795da5f40020e5b39bb2e3 [file] [log] [blame]
robertphillips@google.com1e945b72012-04-16 18:03:03 +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 GrClipMaskManager_DEFINED
9#define GrClipMaskManager_DEFINED
10
commit-bot@chromium.orga0b40282013-09-18 13:00:55 +000011#include "GrClipMaskCache.h"
robertphillips@google.comf105b102012-05-14 12:18:26 +000012#include "GrContext.h"
bsalomon@google.comeb6879f2013-06-13 19:34:18 +000013#include "GrDrawState.h"
bsalomon@google.com4c2443e2012-12-06 20:58:57 +000014#include "GrReducedClip.h"
bsalomon@google.com411dad02012-06-05 20:24:20 +000015#include "GrStencil.h"
16#include "GrTexture.h"
17
robertphillips@google.com641f8b12012-07-31 19:15:58 +000018#include "SkClipStack.h"
bsalomon@google.com411dad02012-06-05 20:24:20 +000019#include "SkDeque.h"
20#include "SkPath.h"
21#include "SkRefCnt.h"
bsalomon@google.com8182fa02012-12-04 14:06:06 +000022#include "SkTLList.h"
commit-bot@chromium.orga0b40282013-09-18 13:00:55 +000023#include "SkTypes.h"
robertphillips@google.com1fcc1b82012-08-29 12:52:05 +000024
joshualitt329bf482014-10-29 12:31:28 -070025class GrClipTarget;
robertphillips@google.com1e945b72012-04-16 18:03:03 +000026class GrPathRenderer;
27class GrPathRendererChain;
robertphillips@google.comf294b772012-04-27 14:29:26 +000028class GrTexture;
commit-bot@chromium.orga0b40282013-09-18 13:00:55 +000029class SkPath;
robertphillips@google.com1e945b72012-04-16 18:03:03 +000030
31/**
rmistry@google.comd6176b02012-08-23 18:14:13 +000032 * The clip mask creator handles the generation of the clip mask. If anti
33 * aliasing is requested it will (in the future) generate a single channel
34 * (8bit) mask. If no anti aliasing is requested it will generate a 1-bit
robertphillips@google.com1e945b72012-04-16 18:03:03 +000035 * mask in the stencil buffer. In the non anti-aliasing case, if the clip
36 * mask can be represented as a rectangle then scissoring is used. In all
37 * cases scissoring is used to bound the range of the clip mask.
38 */
commit-bot@chromium.orge3beb6b2014-04-07 19:34:38 +000039class GrClipMaskManager : SkNoncopyable {
robertphillips@google.com1e945b72012-04-16 18:03:03 +000040public:
robertphillips@google.com5d8d1862012-08-15 14:36:41 +000041 GrClipMaskManager()
joshualitt329bf482014-10-29 12:31:28 -070042 : fCurrClipMaskType(kNone_ClipMaskType)
43 , fClipTarget(NULL) {
robertphillips@google.com1e945b72012-04-16 18:03:03 +000044 }
45
joshualitt6db519c2014-10-29 08:48:18 -070046 // The state of the scissor is controlled by the clip manager, no one else should set
47 // Scissor state. This should really be on Gpu itself. We should revist this when GPU
48 // and drawtarget are separate
49 struct ScissorState {
50 ScissorState() : fEnabled(false) {}
51 void set(const SkIRect& rect) { fRect = rect; fEnabled = true; }
52 bool fEnabled;
53 SkIRect fRect;
54 };
55
bsalomon@google.coma3201942012-06-21 19:58:20 +000056 /**
57 * Creates a clip mask if necessary as a stencil buffer or alpha texture
58 * and sets the GrGpu's scissor and stencil state. If the return is false
commit-bot@chromium.org3ae0e6c2014-02-11 18:24:25 +000059 * then the draw can be skipped. The AutoRestoreEffects is initialized by
60 * the manager when it must install additional effects to implement the
61 * clip. devBounds is optional but can help optimize clipping.
bsalomon@google.coma3201942012-06-21 19:58:20 +000062 */
joshualitta58fe352014-10-27 08:39:00 -070063 bool setupClipping(const GrClipData* clipDataIn,
joshualitt77b13072014-10-27 14:51:01 -070064 const SkRect* devBounds,
joshualitta58fe352014-10-27 08:39:00 -070065 GrDrawState::AutoRestoreEffects*,
66 GrDrawState::AutoRestoreStencil*,
joshualitt329bf482014-10-29 12:31:28 -070067 ScissorState*);
robertphillips@google.com1e945b72012-04-16 18:03:03 +000068
bsalomonc8dc1f72014-08-21 13:02:13 -070069 /**
70 * Purge resources to free up memory. TODO: This class shouldn't hold any long lived refs
71 * which will allow ResourceCache2 to automatically purge anything this class has created.
72 */
73 void purgeResources();
robertphillips@google.com1e945b72012-04-16 18:03:03 +000074
bsalomon@google.comc8f7f472012-06-18 13:44:51 +000075 bool isClipInStencil() const {
76 return kStencil_ClipMaskType == fCurrClipMaskType;
77 }
78 bool isClipInAlpha() const {
79 return kAlpha_ClipMaskType == fCurrClipMaskType;
80 }
robertphillips@google.com1e945b72012-04-16 18:03:03 +000081
bsalomon@google.comc8f7f472012-06-18 13:44:51 +000082 void invalidateStencilMask() {
83 if (kStencil_ClipMaskType == fCurrClipMaskType) {
84 fCurrClipMaskType = kNone_ClipMaskType;
85 }
robertphillips@google.com1e945b72012-04-16 18:03:03 +000086 }
87
robertphillips@google.com2c756812012-05-22 20:28:23 +000088 GrContext* getContext() {
89 return fAACache.getContext();
90 }
91
joshualitt329bf482014-10-29 12:31:28 -070092 void setClipTarget(GrClipTarget*);
commit-bot@chromium.orgc4dc0ad2013-10-09 14:11:33 +000093
joshualitt329bf482014-10-29 12:31:28 -070094 void adjustPathStencilParams(GrStencilSettings*);
95
bsalomon@google.coma3201942012-06-21 19:58:20 +000096private:
joshualitt329bf482014-10-29 12:31:28 -070097 enum PrivateDrawStateStateBits {
98 kFirstBit = (GrDrawState::kLastPublicStateBit << 1),
99
100 kModifyStencilClip_StateBit = kFirstBit, // allows draws to modify
101 // stencil bits used for
102 // clipping.
103 };
104
bsalomon@google.com411dad02012-06-05 20:24:20 +0000105 /**
106 * Informs the helper function adjustStencilParams() about how the stencil
107 * buffer clip is being used.
108 */
109 enum StencilClipMode {
110 // Draw to the clip bit of the stencil buffer
111 kModifyClip_StencilClipMode,
112 // Clip against the existing representation of the clip in the high bit
113 // of the stencil buffer.
114 kRespectClip_StencilClipMode,
115 // Neither writing to nor clipping against the clip bit.
116 kIgnoreClip_StencilClipMode,
117 };
118
commit-bot@chromium.orge5a041c2014-03-07 19:43:43 +0000119 // Attempts to install a series of coverage effects to implement the clip. Return indicates
mtklein217daa72014-07-02 12:55:21 -0700120 // whether the element list was successfully converted to effects.
commit-bot@chromium.orge5a041c2014-03-07 19:43:43 +0000121 bool installClipEffects(const GrReducedClip::ElementList&,
122 GrDrawState::AutoRestoreEffects*,
123 const SkVector& clipOffset,
mtklein217daa72014-07-02 12:55:21 -0700124 const SkRect* devBounds);
commit-bot@chromium.orge5a041c2014-03-07 19:43:43 +0000125
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000126 // Draws the clip into the stencil buffer
commit-bot@chromium.orgd3e58422013-11-05 15:03:08 +0000127 bool createStencilClipMask(int32_t elementsGenID,
128 GrReducedClip::InitialState initialState,
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000129 const GrReducedClip::ElementList& elements,
130 const SkIRect& clipSpaceIBounds,
131 const SkIPoint& clipSpaceToStencilOffset);
132 // Creates an alpha mask of the clip. The mask is a rasterization of elements through the
133 // rect specified by clipSpaceIBounds.
commit-bot@chromium.orgd3e58422013-11-05 15:03:08 +0000134 GrTexture* createAlphaClipMask(int32_t elementsGenID,
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000135 GrReducedClip::InitialState initialState,
136 const GrReducedClip::ElementList& elements,
137 const SkIRect& clipSpaceIBounds);
138 // Similar to createAlphaClipMask but it rasterizes in SW and uploads to the result texture.
commit-bot@chromium.orgd3e58422013-11-05 15:03:08 +0000139 GrTexture* createSoftwareClipMask(int32_t elementsGenID,
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000140 GrReducedClip::InitialState initialState,
141 const GrReducedClip::ElementList& elements,
142 const SkIRect& clipSpaceIBounds);
robertphillips@google.comf294b772012-04-27 14:29:26 +0000143
krajcevskiad1dc582014-06-10 15:06:47 -0700144 // Returns the cached mask texture if it matches the elementsGenID and the clipSpaceIBounds.
145 // Returns NULL if not found.
146 GrTexture* getCachedMaskTexture(int32_t elementsGenID, const SkIRect& clipSpaceIBounds);
147
148
149 // Handles allocation (if needed) of a clip alpha-mask texture for both the sw-upload
150 // or gpu-rendered cases.
151 GrTexture* allocMaskTexture(int32_t elementsGenID,
152 const SkIRect& clipSpaceIBounds,
153 bool willUpload);
robertphillips@google.comfa662942012-05-17 12:20:22 +0000154
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000155 bool useSWOnlyPath(const GrReducedClip::ElementList& elements);
156
bsalomon@google.comb68addd2012-12-14 13:36:53 +0000157 // Draws a clip element into the target alpha mask. The caller should have already setup the
robertphillips@google.come79f3202014-02-11 16:30:21 +0000158 // desired blend operation. Optionally if the caller already selected a path renderer it can
159 // be passed. Otherwise the function will select one if the element is a path.
160 bool drawElement(GrTexture* target, const SkClipStack::Element*, GrPathRenderer* = NULL);
bsalomon@google.comb68addd2012-12-14 13:36:53 +0000161
162 // Determines whether it is possible to draw the element to both the stencil buffer and the
163 // alpha mask simultaneously. If so and the element is a path a compatible path renderer is
164 // also returned.
robertphillips@google.come79f3202014-02-11 16:30:21 +0000165 bool canStencilAndDrawElement(GrTexture* target, const SkClipStack::Element*, GrPathRenderer**);
robertphillips@google.comf294b772012-04-27 14:29:26 +0000166
bsalomon@google.com7b7cdd12012-11-07 16:17:24 +0000167 void mergeMask(GrTexture* dstMask,
168 GrTexture* srcMask,
169 SkRegion::Op op,
commit-bot@chromium.orgfd03d4a2013-07-17 21:39:42 +0000170 const SkIRect& dstBound,
171 const SkIRect& srcBound);
robertphillips@google.com1e945b72012-04-16 18:03:03 +0000172
bsalomon427cf282014-10-16 13:41:43 -0700173 GrTexture* createTempMask(int width, int height);
robertphillips@google.comf105b102012-05-14 12:18:26 +0000174
rmistry@google.comd6176b02012-08-23 18:14:13 +0000175 void setupCache(const SkClipStack& clip,
commit-bot@chromium.orgfd03d4a2013-07-17 21:39:42 +0000176 const SkIRect& bounds);
robertphillips@google.com6d62df42012-05-07 18:07:36 +0000177
bsalomon@google.coma3201942012-06-21 19:58:20 +0000178 /**
179 * Called prior to return control back the GrGpu in setupClipping. It
180 * updates the GrGpu with stencil settings that account stencil-based
181 * clipping.
182 */
joshualitta58fe352014-10-27 08:39:00 -0700183 void setDrawStateStencil(GrDrawState::AutoRestoreStencil* asr);
bsalomon@google.coma3201942012-06-21 19:58:20 +0000184
185 /**
186 * Adjusts the stencil settings to account for interaction with stencil
187 * clipping.
188 */
189 void adjustStencilParams(GrStencilSettings* settings,
190 StencilClipMode mode,
191 int stencilBitCnt);
192
joshualitt329bf482014-10-29 12:31:28 -0700193 /**
194 * We may represent the clip as a mask in the stencil buffer or as an alpha
195 * texture. It may be neither because the scissor rect suffices or we
196 * haven't yet examined the clip.
197 */
198 enum ClipMaskType {
199 kNone_ClipMaskType,
200 kStencil_ClipMaskType,
201 kAlpha_ClipMaskType,
202 } fCurrClipMaskType;
203
204 GrClipMaskCache fAACache; // cache for the AA path
205 GrClipTarget* fClipTarget;
206
commit-bot@chromium.orga0b40282013-09-18 13:00:55 +0000207 typedef SkNoncopyable INHERITED;
robertphillips@google.com1e945b72012-04-16 18:03:03 +0000208};
209
robertphillips@google.com1e945b72012-04-16 18:03:03 +0000210#endif // GrClipMaskManager_DEFINED