blob: eea11bff7d8a463ee3befc2393b59dbbe1373d99 [file] [log] [blame]
Chris Craik5ea17242016-01-11 14:07:59 -08001/*
2 * Copyright (C) 2016 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
Chris Craikf158b492016-01-12 14:45:08 -080017#include "LayerBuilder.h"
Chris Craik5ea17242016-01-11 14:07:59 -080018
19#include "BakedOpState.h"
20#include "RenderNode.h"
21#include "utils/PaintUtils.h"
22#include "utils/TraceUtils.h"
23
24#include <utils/TypeHelpers.h>
25
26namespace android {
27namespace uirenderer {
28
29class BatchBase {
30public:
31 BatchBase(batchid_t batchId, BakedOpState* op, bool merging)
32 : mBatchId(batchId)
33 , mMerging(merging) {
34 mBounds = op->computedState.clippedBounds;
35 mOps.push_back(op);
36 }
37
38 bool intersects(const Rect& rect) const {
39 if (!rect.intersects(mBounds)) return false;
40
41 for (const BakedOpState* op : mOps) {
42 if (rect.intersects(op->computedState.clippedBounds)) {
43 return true;
44 }
45 }
46 return false;
47 }
48
49 batchid_t getBatchId() const { return mBatchId; }
50 bool isMerging() const { return mMerging; }
51
52 const std::vector<BakedOpState*>& getOps() const { return mOps; }
53
54 void dump() const {
55 ALOGD(" Batch %p, id %d, merging %d, count %d, bounds " RECT_STRING,
Chris Craikb250a832016-01-11 19:28:17 -080056 this, mBatchId, mMerging, (int) mOps.size(), RECT_ARGS(mBounds));
Chris Craik5ea17242016-01-11 14:07:59 -080057 }
58protected:
59 batchid_t mBatchId;
60 Rect mBounds;
61 std::vector<BakedOpState*> mOps;
62 bool mMerging;
63};
64
65class OpBatch : public BatchBase {
66public:
Chris Craik5ea17242016-01-11 14:07:59 -080067 OpBatch(batchid_t batchId, BakedOpState* op)
68 : BatchBase(batchId, op, false) {
69 }
70
71 void batchOp(BakedOpState* op) {
72 mBounds.unionWith(op->computedState.clippedBounds);
73 mOps.push_back(op);
74 }
75};
76
77class MergingOpBatch : public BatchBase {
78public:
Chris Craik5ea17242016-01-11 14:07:59 -080079 MergingOpBatch(batchid_t batchId, BakedOpState* op)
80 : BatchBase(batchId, op, true)
81 , mClipSideFlags(op->computedState.clipSideFlags) {
82 }
83
84 /*
85 * Helper for determining if a new op can merge with a MergingDrawBatch based on their bounds
86 * and clip side flags. Positive bounds delta means new bounds fit in old.
87 */
88 static inline bool checkSide(const int currentFlags, const int newFlags, const int side,
89 float boundsDelta) {
90 bool currentClipExists = currentFlags & side;
91 bool newClipExists = newFlags & side;
92
93 // if current is clipped, we must be able to fit new bounds in current
94 if (boundsDelta > 0 && currentClipExists) return false;
95
96 // if new is clipped, we must be able to fit current bounds in new
97 if (boundsDelta < 0 && newClipExists) return false;
98
99 return true;
100 }
101
102 static bool paintIsDefault(const SkPaint& paint) {
103 return paint.getAlpha() == 255
104 && paint.getColorFilter() == nullptr
105 && paint.getShader() == nullptr;
106 }
107
108 static bool paintsAreEquivalent(const SkPaint& a, const SkPaint& b) {
109 // Note: don't check color, since all currently mergeable ops can merge across colors
110 return a.getAlpha() == b.getAlpha()
111 && a.getColorFilter() == b.getColorFilter()
112 && a.getShader() == b.getShader();
113 }
114
115 /*
116 * Checks if a (mergeable) op can be merged into this batch
117 *
118 * If true, the op's multiDraw must be guaranteed to handle both ops simultaneously, so it is
119 * important to consider all paint attributes used in the draw calls in deciding both a) if an
120 * op tries to merge at all, and b) if the op can merge with another set of ops
121 *
122 * False positives can lead to information from the paints of subsequent merged operations being
123 * dropped, so we make simplifying qualifications on the ops that can merge, per op type.
124 */
125 bool canMergeWith(BakedOpState* op) const {
126 bool isTextBatch = getBatchId() == OpBatchType::Text
127 || getBatchId() == OpBatchType::ColorText;
128
129 // Overlapping other operations is only allowed for text without shadow. For other ops,
130 // multiDraw isn't guaranteed to overdraw correctly
131 if (!isTextBatch || PaintUtils::hasTextShadow(op->op->paint)) {
132 if (intersects(op->computedState.clippedBounds)) return false;
133 }
134
135 const BakedOpState* lhs = op;
136 const BakedOpState* rhs = mOps[0];
137
138 if (!MathUtils::areEqual(lhs->alpha, rhs->alpha)) return false;
139
140 // Identical round rect clip state means both ops will clip in the same way, or not at all.
141 // As the state objects are const, we can compare their pointers to determine mergeability
142 if (lhs->roundRectClipState != rhs->roundRectClipState) return false;
Chris Craik678ff812016-03-01 13:27:54 -0800143
144 // Local masks prevent merge, since they're potentially in different coordinate spaces
145 if (lhs->computedState.localProjectionPathMask
146 || rhs->computedState.localProjectionPathMask) return false;
Chris Craik5ea17242016-01-11 14:07:59 -0800147
148 /* Clipping compatibility check
149 *
150 * Exploits the fact that if a op or batch is clipped on a side, its bounds will equal its
151 * clip for that side.
152 */
153 const int currentFlags = mClipSideFlags;
154 const int newFlags = op->computedState.clipSideFlags;
155 if (currentFlags != OpClipSideFlags::None || newFlags != OpClipSideFlags::None) {
156 const Rect& opBounds = op->computedState.clippedBounds;
157 float boundsDelta = mBounds.left - opBounds.left;
158 if (!checkSide(currentFlags, newFlags, OpClipSideFlags::Left, boundsDelta)) return false;
159 boundsDelta = mBounds.top - opBounds.top;
160 if (!checkSide(currentFlags, newFlags, OpClipSideFlags::Top, boundsDelta)) return false;
161
162 // right and bottom delta calculation reversed to account for direction
163 boundsDelta = opBounds.right - mBounds.right;
164 if (!checkSide(currentFlags, newFlags, OpClipSideFlags::Right, boundsDelta)) return false;
165 boundsDelta = opBounds.bottom - mBounds.bottom;
166 if (!checkSide(currentFlags, newFlags, OpClipSideFlags::Bottom, boundsDelta)) return false;
167 }
168
169 const SkPaint* newPaint = op->op->paint;
170 const SkPaint* oldPaint = mOps[0]->op->paint;
171
172 if (newPaint == oldPaint) {
173 // if paints are equal, then modifiers + paint attribs don't need to be compared
174 return true;
175 } else if (newPaint && !oldPaint) {
176 return paintIsDefault(*newPaint);
177 } else if (!newPaint && oldPaint) {
178 return paintIsDefault(*oldPaint);
179 }
180 return paintsAreEquivalent(*newPaint, *oldPaint);
181 }
182
183 void mergeOp(BakedOpState* op) {
184 mBounds.unionWith(op->computedState.clippedBounds);
185 mOps.push_back(op);
186
187 // Because a new op must have passed canMergeWith(), we know it's passed the clipping compat
188 // check, and doesn't extend past a side of the clip that's in use by the merged batch.
189 // Therefore it's safe to simply always merge flags, and use the bounds as the clip rect.
190 mClipSideFlags |= op->computedState.clipSideFlags;
191 }
192
193 int getClipSideFlags() const { return mClipSideFlags; }
194 const Rect& getClipRect() const { return mBounds; }
195
196private:
197 int mClipSideFlags;
198};
199
Chris Craikf158b492016-01-12 14:45:08 -0800200LayerBuilder::LayerBuilder(uint32_t width, uint32_t height,
Chris Craik5ea17242016-01-11 14:07:59 -0800201 const Rect& repaintRect, const BeginLayerOp* beginLayerOp, RenderNode* renderNode)
202 : width(width)
203 , height(height)
204 , repaintRect(repaintRect)
Chris Craik4876de12016-02-25 16:54:08 -0800205 , repaintClip(repaintRect)
Chris Craik5ea17242016-01-11 14:07:59 -0800206 , offscreenBuffer(renderNode ? renderNode->getLayer() : nullptr)
207 , beginLayerOp(beginLayerOp)
Chris Craik4876de12016-02-25 16:54:08 -0800208 , renderNode(renderNode) {}
Chris Craik5ea17242016-01-11 14:07:59 -0800209
210// iterate back toward target to see if anything drawn since should overlap the new op
211// if no target, merging ops still iterate to find similar batch to insert after
Chris Craikf158b492016-01-12 14:45:08 -0800212void LayerBuilder::locateInsertIndex(int batchId, const Rect& clippedBounds,
Chris Craik5ea17242016-01-11 14:07:59 -0800213 BatchBase** targetBatch, size_t* insertBatchIndex) const {
214 for (int i = mBatches.size() - 1; i >= 0; i--) {
215 BatchBase* overBatch = mBatches[i];
216
217 if (overBatch == *targetBatch) break;
218
219 // TODO: also consider shader shared between batch types
220 if (batchId == overBatch->getBatchId()) {
221 *insertBatchIndex = i + 1;
222 if (!*targetBatch) break; // found insert position, quit
223 }
224
225 if (overBatch->intersects(clippedBounds)) {
226 // NOTE: it may be possible to optimize for special cases where two operations
227 // of the same batch/paint could swap order, such as with a non-mergeable
228 // (clipped) and a mergeable text operation
229 *targetBatch = nullptr;
230 break;
231 }
232 }
233}
234
Chris Craikf158b492016-01-12 14:45:08 -0800235void LayerBuilder::deferLayerClear(const Rect& rect) {
Chris Craik5ea17242016-01-11 14:07:59 -0800236 mClearRects.push_back(rect);
237}
238
Chris Craik80d2ade2016-03-28 12:54:07 -0700239void LayerBuilder::onDeferOp(LinearAllocator& allocator, const BakedOpState* bakedState) {
240 if (bakedState->op->opId != RecordedOpId::CopyToLayerOp) {
241 // First non-CopyToLayer, so stop stashing up layer clears for unclipped save layers,
242 // and issue them together in one draw.
243 flushLayerClears(allocator);
244
245 if (CC_UNLIKELY(activeUnclippedSaveLayers.empty()
246 && bakedState->computedState.opaqueOverClippedBounds
247 && bakedState->computedState.clippedBounds.contains(repaintRect))) {
248 // discard all deferred drawing ops, since new one will occlude them
249 clear();
250 }
251 }
252}
253
Chris Craikf158b492016-01-12 14:45:08 -0800254void LayerBuilder::flushLayerClears(LinearAllocator& allocator) {
Chris Craik5ea17242016-01-11 14:07:59 -0800255 if (CC_UNLIKELY(!mClearRects.empty())) {
256 const int vertCount = mClearRects.size() * 4;
257 // put the verts in the frame allocator, since
258 // 1) SimpleRectsOps needs verts, not rects
259 // 2) even if mClearRects stored verts, std::vectors will move their contents
Chris Craik7a896002016-02-19 15:51:02 -0800260 Vertex* const verts = (Vertex*) allocator.create_trivial_array<Vertex>(vertCount);
Chris Craik5ea17242016-01-11 14:07:59 -0800261
262 Vertex* currentVert = verts;
263 Rect bounds = mClearRects[0];
264 for (auto&& rect : mClearRects) {
265 bounds.unionWith(rect);
266 Vertex::set(currentVert++, rect.left, rect.top);
267 Vertex::set(currentVert++, rect.right, rect.top);
268 Vertex::set(currentVert++, rect.left, rect.bottom);
269 Vertex::set(currentVert++, rect.right, rect.bottom);
270 }
271 mClearRects.clear(); // discard rects before drawing so this method isn't reentrant
272
273 // One or more unclipped saveLayers have been enqueued, with deferred clears.
274 // Flush all of these clears with a single draw
275 SkPaint* paint = allocator.create<SkPaint>();
276 paint->setXfermodeMode(SkXfermode::kClear_Mode);
John Reck7df9ff22016-02-10 16:08:08 -0800277 SimpleRectsOp* op = allocator.create_trivial<SimpleRectsOp>(bounds,
Chris Craik5ea17242016-01-11 14:07:59 -0800278 Matrix4::identity(), nullptr, paint,
279 verts, vertCount);
280 BakedOpState* bakedState = BakedOpState::directConstruct(allocator,
Chris Craik4876de12016-02-25 16:54:08 -0800281 &repaintClip, bounds, *op);
Chris Craik5ea17242016-01-11 14:07:59 -0800282 deferUnmergeableOp(allocator, bakedState, OpBatchType::Vertices);
283 }
284}
285
Chris Craikf158b492016-01-12 14:45:08 -0800286void LayerBuilder::deferUnmergeableOp(LinearAllocator& allocator,
Chris Craik5ea17242016-01-11 14:07:59 -0800287 BakedOpState* op, batchid_t batchId) {
Chris Craik80d2ade2016-03-28 12:54:07 -0700288 onDeferOp(allocator, op);
Chris Craik5ea17242016-01-11 14:07:59 -0800289 OpBatch* targetBatch = mBatchLookup[batchId];
290
291 size_t insertBatchIndex = mBatches.size();
292 if (targetBatch) {
293 locateInsertIndex(batchId, op->computedState.clippedBounds,
294 (BatchBase**)(&targetBatch), &insertBatchIndex);
295 }
296
297 if (targetBatch) {
298 targetBatch->batchOp(op);
299 } else {
300 // new non-merging batch
John Reck7df9ff22016-02-10 16:08:08 -0800301 targetBatch = allocator.create<OpBatch>(batchId, op);
Chris Craik5ea17242016-01-11 14:07:59 -0800302 mBatchLookup[batchId] = targetBatch;
303 mBatches.insert(mBatches.begin() + insertBatchIndex, targetBatch);
304 }
305}
306
Chris Craikf158b492016-01-12 14:45:08 -0800307void LayerBuilder::deferMergeableOp(LinearAllocator& allocator,
Chris Craik5ea17242016-01-11 14:07:59 -0800308 BakedOpState* op, batchid_t batchId, mergeid_t mergeId) {
Chris Craik80d2ade2016-03-28 12:54:07 -0700309 onDeferOp(allocator, op);
Chris Craik5ea17242016-01-11 14:07:59 -0800310 MergingOpBatch* targetBatch = nullptr;
311
312 // Try to merge with any existing batch with same mergeId
313 auto getResult = mMergingBatchLookup[batchId].find(mergeId);
314 if (getResult != mMergingBatchLookup[batchId].end()) {
315 targetBatch = getResult->second;
316 if (!targetBatch->canMergeWith(op)) {
317 targetBatch = nullptr;
318 }
319 }
320
321 size_t insertBatchIndex = mBatches.size();
322 locateInsertIndex(batchId, op->computedState.clippedBounds,
323 (BatchBase**)(&targetBatch), &insertBatchIndex);
324
325 if (targetBatch) {
326 targetBatch->mergeOp(op);
327 } else {
328 // new merging batch
John Reck7df9ff22016-02-10 16:08:08 -0800329 targetBatch = allocator.create<MergingOpBatch>(batchId, op);
Chris Craik5ea17242016-01-11 14:07:59 -0800330 mMergingBatchLookup[batchId].insert(std::make_pair(mergeId, targetBatch));
331
332 mBatches.insert(mBatches.begin() + insertBatchIndex, targetBatch);
333 }
334}
335
Chris Craikf158b492016-01-12 14:45:08 -0800336void LayerBuilder::replayBakedOpsImpl(void* arg,
Chris Craik5ea17242016-01-11 14:07:59 -0800337 BakedOpReceiver* unmergedReceivers, MergedOpReceiver* mergedReceivers) const {
338 ATRACE_NAME("flush drawing commands");
339 for (const BatchBase* batch : mBatches) {
340 size_t size = batch->getOps().size();
341 if (size > 1 && batch->isMerging()) {
342 int opId = batch->getOps()[0]->op->opId;
343 const MergingOpBatch* mergingBatch = static_cast<const MergingOpBatch*>(batch);
344 MergedBakedOpList data = {
345 batch->getOps().data(),
346 size,
347 mergingBatch->getClipSideFlags(),
348 mergingBatch->getClipRect()
349 };
350 mergedReceivers[opId](arg, data);
351 } else {
352 for (const BakedOpState* op : batch->getOps()) {
353 unmergedReceivers[op->op->opId](arg, *op);
354 }
355 }
356 }
357}
358
Chris Craik80d2ade2016-03-28 12:54:07 -0700359void LayerBuilder::clear() {
360 mBatches.clear();
361 for (int i = 0; i < OpBatchType::Count; i++) {
362 mBatchLookup[i] = nullptr;
363 mMergingBatchLookup[i].clear();
364 }
365}
366
Chris Craikf158b492016-01-12 14:45:08 -0800367void LayerBuilder::dump() const {
Chris Craik02806282016-03-11 19:16:21 -0800368 ALOGD("LayerBuilder %p, %ux%u buffer %p, blo %p, rn %p (%s)",
369 this, width, height, offscreenBuffer, beginLayerOp,
370 renderNode, renderNode ? renderNode->getName() : "-");
Chris Craik5ea17242016-01-11 14:07:59 -0800371 for (const BatchBase* batch : mBatches) {
372 batch->dump();
373 }
374}
375
376} // namespace uirenderer
377} // namespace android