blob: 5a5845af81b96e47dbdee2fae8536af597db08cd [file] [log] [blame]
Chris Craikb565df12015-10-05 13:00:52 -07001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef ANDROID_HWUI_BAKED_OP_STATE_H
18#define ANDROID_HWUI_BAKED_OP_STATE_H
19
20#include "Matrix.h"
21#include "RecordedOp.h"
22#include "Rect.h"
23#include "Snapshot.h"
24
25namespace android {
26namespace uirenderer {
27
28namespace OpClipSideFlags {
29 enum {
30 None = 0x0,
31 Left = 0x1,
32 Top = 0x2,
33 Right = 0x4,
34 Bottom = 0x8,
35 Full = 0xF,
36 // ConservativeFull = 0x1F needed?
37 };
38}
39
40/**
Chris Craik15c3f192015-12-03 12:16:56 -080041 * Holds a list of BakedOpStates of ops that can be drawn together
42 */
43struct MergedBakedOpList {
44 const BakedOpState*const* states;
45 size_t count;
46 int clipSideFlags;
47 Rect clip;
48};
49
50/**
Chris Craikb565df12015-10-05 13:00:52 -070051 * Holds the resolved clip, transform, and bounds of a recordedOp, when replayed with a snapshot
52 */
53class ResolvedRenderState {
54public:
Chris Craike4db79d2015-12-22 16:32:23 -080055 ResolvedRenderState(LinearAllocator& allocator, Snapshot& snapshot,
56 const RecordedOp& recordedOp, bool expandForStroke);
Chris Craikb565df12015-10-05 13:00:52 -070057
Chris Craike4db79d2015-12-22 16:32:23 -080058 // Constructor for unbounded ops without transform/clip (namely shadows)
59 ResolvedRenderState(LinearAllocator& allocator, Snapshot& snapshot);
Chris Craikd3daa312015-11-06 10:59:56 -080060
Chris Craik7435eb12016-01-07 17:41:40 -080061 // Constructor for primitive ops provided clip, and no transform
62 ResolvedRenderState(const ClipRect* viewportRect, const Rect& dstRect);
Chris Craikb87eadd2016-01-06 09:16:05 -080063
Chris Craikd7448e62015-12-15 10:34:36 -080064 Rect computeLocalSpaceClip() const {
65 Matrix4 inverse;
66 inverse.loadInverse(transform);
67
Chris Craike4db79d2015-12-22 16:32:23 -080068 Rect outClip(clipRect());
Chris Craikd7448e62015-12-15 10:34:36 -080069 inverse.mapRect(outClip);
70 return outClip;
71 }
72
Chris Craike4db79d2015-12-22 16:32:23 -080073 const Rect& clipRect() const {
74 return clipState->rect;
75 }
Chris Craikb87eadd2016-01-06 09:16:05 -080076
Chris Craike4db79d2015-12-22 16:32:23 -080077 bool requiresClip() const {
78 return clipSideFlags != OpClipSideFlags::None
Chris Craik7435eb12016-01-07 17:41:40 -080079 || CC_UNLIKELY(clipState->mode != ClipMode::Rectangle);
Chris Craike4db79d2015-12-22 16:32:23 -080080 }
81
82 // returns the clip if it's needed to draw the operation, otherwise nullptr
83 const ClipBase* getClipIfNeeded() const {
84 return requiresClip() ? clipState : nullptr;
85 }
Chris Craikb87eadd2016-01-06 09:16:05 -080086
87 Matrix4 transform;
Chris Craike4db79d2015-12-22 16:32:23 -080088 const ClipBase* clipState = nullptr;
Chris Craikb565df12015-10-05 13:00:52 -070089 Rect clippedBounds;
Chris Craikb87eadd2016-01-06 09:16:05 -080090 int clipSideFlags = 0;
Chris Craikb565df12015-10-05 13:00:52 -070091};
92
93/**
94 * Self-contained op wrapper, containing all resolved state required to draw the op.
95 *
96 * Stashed pointers within all point to longer lived objects, with no ownership implied.
97 */
98class BakedOpState {
99public:
100 static BakedOpState* tryConstruct(LinearAllocator& allocator,
Chris Craike4db79d2015-12-22 16:32:23 -0800101 Snapshot& snapshot, const RecordedOp& recordedOp) {
Chris Craik15f04682016-01-11 17:50:08 -0800102 if (CC_UNLIKELY(snapshot.getRenderTargetClip().isEmpty())) return nullptr;
John Reck7df9ff22016-02-10 16:08:08 -0800103 BakedOpState* bakedState = allocator.create_trivial<BakedOpState>(
Chris Craike4db79d2015-12-22 16:32:23 -0800104 allocator, snapshot, recordedOp, false);
Chris Craik386aa032015-12-07 17:08:25 -0800105 if (bakedState->computedState.clippedBounds.isEmpty()) {
Chris Craikb565df12015-10-05 13:00:52 -0700106 // bounds are empty, so op is rejected
Chris Craik386aa032015-12-07 17:08:25 -0800107 allocator.rewindIfLastAlloc(bakedState);
Chris Craikb565df12015-10-05 13:00:52 -0700108 return nullptr;
109 }
Chris Craik386aa032015-12-07 17:08:25 -0800110 return bakedState;
111 }
112
113 enum class StrokeBehavior {
114 // stroking is forced, regardless of style on paint
115 Forced,
116 // stroking is defined by style on paint
117 StyleDefined,
118 };
119
120 static BakedOpState* tryStrokeableOpConstruct(LinearAllocator& allocator,
Chris Craike4db79d2015-12-22 16:32:23 -0800121 Snapshot& snapshot, const RecordedOp& recordedOp, StrokeBehavior strokeBehavior) {
Chris Craik15f04682016-01-11 17:50:08 -0800122 if (CC_UNLIKELY(snapshot.getRenderTargetClip().isEmpty())) return nullptr;
Chris Craik386aa032015-12-07 17:08:25 -0800123 bool expandForStroke = (strokeBehavior == StrokeBehavior::StyleDefined)
124 ? (recordedOp.paint && recordedOp.paint->getStyle() != SkPaint::kFill_Style)
125 : true;
126
John Reck7df9ff22016-02-10 16:08:08 -0800127 BakedOpState* bakedState = allocator.create_trivial<BakedOpState>(
Chris Craike4db79d2015-12-22 16:32:23 -0800128 allocator, snapshot, recordedOp, expandForStroke);
Chris Craik386aa032015-12-07 17:08:25 -0800129 if (bakedState->computedState.clippedBounds.isEmpty()) {
130 // bounds are empty, so op is rejected
Chris Craik15f04682016-01-11 17:50:08 -0800131 // NOTE: this won't succeed if a clip was allocated
Chris Craik386aa032015-12-07 17:08:25 -0800132 allocator.rewindIfLastAlloc(bakedState);
133 return nullptr;
134 }
135 return bakedState;
Chris Craikb565df12015-10-05 13:00:52 -0700136 }
137
Chris Craikd3daa312015-11-06 10:59:56 -0800138 static BakedOpState* tryShadowOpConstruct(LinearAllocator& allocator,
Chris Craike4db79d2015-12-22 16:32:23 -0800139 Snapshot& snapshot, const ShadowOp* shadowOpPtr) {
Chris Craik15f04682016-01-11 17:50:08 -0800140 if (CC_UNLIKELY(snapshot.getRenderTargetClip().isEmpty())) return nullptr;
Chris Craikd3daa312015-11-06 10:59:56 -0800141
142 // clip isn't empty, so construct the op
John Reck7df9ff22016-02-10 16:08:08 -0800143 return allocator.create_trivial<BakedOpState>(allocator, snapshot, shadowOpPtr);
Chris Craikd3daa312015-11-06 10:59:56 -0800144 }
145
Chris Craikb87eadd2016-01-06 09:16:05 -0800146 static BakedOpState* directConstruct(LinearAllocator& allocator,
Chris Craik7435eb12016-01-07 17:41:40 -0800147 const ClipRect* clip, const Rect& dstRect, const RecordedOp& recordedOp) {
John Reck7df9ff22016-02-10 16:08:08 -0800148 return allocator.create_trivial<BakedOpState>(clip, dstRect, recordedOp);
Chris Craikb565df12015-10-05 13:00:52 -0700149 }
150
151 // computed state:
Chris Craikb87eadd2016-01-06 09:16:05 -0800152 ResolvedRenderState computedState;
Chris Craikb565df12015-10-05 13:00:52 -0700153
154 // simple state (straight pointer/value storage):
155 const float alpha;
156 const RoundRectClipState* roundRectClipState;
157 const ProjectionPathMask* projectionPathMask;
158 const RecordedOp* op;
159
160private:
John Reck7df9ff22016-02-10 16:08:08 -0800161 friend class LinearAllocator;
162
Chris Craike4db79d2015-12-22 16:32:23 -0800163 BakedOpState(LinearAllocator& allocator, Snapshot& snapshot,
164 const RecordedOp& recordedOp, bool expandForStroke)
165 : computedState(allocator, snapshot, recordedOp, expandForStroke)
Chris Craikb565df12015-10-05 13:00:52 -0700166 , alpha(snapshot.alpha)
167 , roundRectClipState(snapshot.roundRectClipState)
168 , projectionPathMask(snapshot.projectionPathMask)
169 , op(&recordedOp) {}
Chris Craikd3daa312015-11-06 10:59:56 -0800170
Chris Craike4db79d2015-12-22 16:32:23 -0800171 BakedOpState(LinearAllocator& allocator, Snapshot& snapshot, const ShadowOp* shadowOpPtr)
172 : computedState(allocator, snapshot)
Chris Craikd3daa312015-11-06 10:59:56 -0800173 , alpha(snapshot.alpha)
174 , roundRectClipState(snapshot.roundRectClipState)
175 , projectionPathMask(snapshot.projectionPathMask)
176 , op(shadowOpPtr) {}
Chris Craikb87eadd2016-01-06 09:16:05 -0800177
Chris Craik7435eb12016-01-07 17:41:40 -0800178 BakedOpState(const ClipRect* viewportRect, const Rect& dstRect, const RecordedOp& recordedOp)
179 : computedState(viewportRect, dstRect)
Chris Craikb87eadd2016-01-06 09:16:05 -0800180 , alpha(1.0f)
181 , roundRectClipState(nullptr)
182 , projectionPathMask(nullptr)
183 , op(&recordedOp) {}
Chris Craikb565df12015-10-05 13:00:52 -0700184};
185
186}; // namespace uirenderer
187}; // namespace android
188
189#endif // ANDROID_HWUI_BAKED_OP_STATE_H