blob: ffe2901782a9da7a00935d87a1437f8964a6a308 [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 Craik678ff812016-03-01 13:27:54 -080091 const SkPath* localProjectionPathMask = nullptr;
Chris Craikb565df12015-10-05 13:00:52 -070092};
93
94/**
95 * Self-contained op wrapper, containing all resolved state required to draw the op.
96 *
97 * Stashed pointers within all point to longer lived objects, with no ownership implied.
98 */
99class BakedOpState {
100public:
101 static BakedOpState* tryConstruct(LinearAllocator& allocator,
Chris Craike4db79d2015-12-22 16:32:23 -0800102 Snapshot& snapshot, const RecordedOp& recordedOp) {
Chris Craik15f04682016-01-11 17:50:08 -0800103 if (CC_UNLIKELY(snapshot.getRenderTargetClip().isEmpty())) return nullptr;
John Reck7df9ff22016-02-10 16:08:08 -0800104 BakedOpState* bakedState = allocator.create_trivial<BakedOpState>(
Chris Craike4db79d2015-12-22 16:32:23 -0800105 allocator, snapshot, recordedOp, false);
Chris Craik386aa032015-12-07 17:08:25 -0800106 if (bakedState->computedState.clippedBounds.isEmpty()) {
Chris Craikb565df12015-10-05 13:00:52 -0700107 // bounds are empty, so op is rejected
Chris Craik386aa032015-12-07 17:08:25 -0800108 allocator.rewindIfLastAlloc(bakedState);
Chris Craikb565df12015-10-05 13:00:52 -0700109 return nullptr;
110 }
Chris Craik386aa032015-12-07 17:08:25 -0800111 return bakedState;
112 }
113
114 enum class StrokeBehavior {
115 // stroking is forced, regardless of style on paint
116 Forced,
117 // stroking is defined by style on paint
118 StyleDefined,
119 };
120
121 static BakedOpState* tryStrokeableOpConstruct(LinearAllocator& allocator,
Chris Craike4db79d2015-12-22 16:32:23 -0800122 Snapshot& snapshot, const RecordedOp& recordedOp, StrokeBehavior strokeBehavior) {
Chris Craik15f04682016-01-11 17:50:08 -0800123 if (CC_UNLIKELY(snapshot.getRenderTargetClip().isEmpty())) return nullptr;
Chris Craik386aa032015-12-07 17:08:25 -0800124 bool expandForStroke = (strokeBehavior == StrokeBehavior::StyleDefined)
125 ? (recordedOp.paint && recordedOp.paint->getStyle() != SkPaint::kFill_Style)
126 : true;
127
John Reck7df9ff22016-02-10 16:08:08 -0800128 BakedOpState* bakedState = allocator.create_trivial<BakedOpState>(
Chris Craike4db79d2015-12-22 16:32:23 -0800129 allocator, snapshot, recordedOp, expandForStroke);
Chris Craik386aa032015-12-07 17:08:25 -0800130 if (bakedState->computedState.clippedBounds.isEmpty()) {
131 // bounds are empty, so op is rejected
Chris Craik15f04682016-01-11 17:50:08 -0800132 // NOTE: this won't succeed if a clip was allocated
Chris Craik386aa032015-12-07 17:08:25 -0800133 allocator.rewindIfLastAlloc(bakedState);
134 return nullptr;
135 }
136 return bakedState;
Chris Craikb565df12015-10-05 13:00:52 -0700137 }
138
Chris Craikd3daa312015-11-06 10:59:56 -0800139 static BakedOpState* tryShadowOpConstruct(LinearAllocator& allocator,
Chris Craike4db79d2015-12-22 16:32:23 -0800140 Snapshot& snapshot, const ShadowOp* shadowOpPtr) {
Chris Craik15f04682016-01-11 17:50:08 -0800141 if (CC_UNLIKELY(snapshot.getRenderTargetClip().isEmpty())) return nullptr;
Chris Craikd3daa312015-11-06 10:59:56 -0800142
143 // clip isn't empty, so construct the op
John Reck7df9ff22016-02-10 16:08:08 -0800144 return allocator.create_trivial<BakedOpState>(allocator, snapshot, shadowOpPtr);
Chris Craikd3daa312015-11-06 10:59:56 -0800145 }
146
Chris Craikb87eadd2016-01-06 09:16:05 -0800147 static BakedOpState* directConstruct(LinearAllocator& allocator,
Chris Craik7435eb12016-01-07 17:41:40 -0800148 const ClipRect* clip, const Rect& dstRect, const RecordedOp& recordedOp) {
John Reck7df9ff22016-02-10 16:08:08 -0800149 return allocator.create_trivial<BakedOpState>(clip, dstRect, recordedOp);
Chris Craikb565df12015-10-05 13:00:52 -0700150 }
151
152 // computed state:
Chris Craikb87eadd2016-01-06 09:16:05 -0800153 ResolvedRenderState computedState;
Chris Craikb565df12015-10-05 13:00:52 -0700154
155 // simple state (straight pointer/value storage):
156 const float alpha;
157 const RoundRectClipState* roundRectClipState;
Chris Craikb565df12015-10-05 13:00:52 -0700158 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)
Chris Craikb565df12015-10-05 13:00:52 -0700168 , op(&recordedOp) {}
Chris Craikd3daa312015-11-06 10:59:56 -0800169
Chris Craike4db79d2015-12-22 16:32:23 -0800170 BakedOpState(LinearAllocator& allocator, Snapshot& snapshot, const ShadowOp* shadowOpPtr)
171 : computedState(allocator, snapshot)
Chris Craikd3daa312015-11-06 10:59:56 -0800172 , alpha(snapshot.alpha)
173 , roundRectClipState(snapshot.roundRectClipState)
Chris Craikd3daa312015-11-06 10:59:56 -0800174 , op(shadowOpPtr) {}
Chris Craikb87eadd2016-01-06 09:16:05 -0800175
Chris Craik4876de12016-02-25 16:54:08 -0800176 BakedOpState(const ClipRect* clipRect, const Rect& dstRect, const RecordedOp& recordedOp)
177 : computedState(clipRect, dstRect)
Chris Craikb87eadd2016-01-06 09:16:05 -0800178 , alpha(1.0f)
179 , roundRectClipState(nullptr)
Chris Craikb87eadd2016-01-06 09:16:05 -0800180 , op(&recordedOp) {}
Chris Craikb565df12015-10-05 13:00:52 -0700181};
182
183}; // namespace uirenderer
184}; // namespace android
185
186#endif // ANDROID_HWUI_BAKED_OP_STATE_H