/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <gtest/gtest.h> #include <BakedOpState.h> #include <DeferredLayerUpdater.h> #include <FrameBuilder.h> #include <GlLayer.h> #include <LayerUpdateQueue.h> #include <RecordedOp.h> #include <RecordingCanvas.h> #include <tests/common/TestUtils.h> #include <unordered_map> namespace android { namespace uirenderer { const FrameBuilder::LightGeometry sLightGeometry = {{100, 100, 100}, 50}; /** * Virtual class implemented by each test to redirect static operation / state transitions to * virtual methods. * * Virtual dispatch allows for default behaviors to be specified (very common case in below tests), * and allows Renderer vs Dispatching behavior to be merged. * * onXXXOp methods fail by default - tests should override ops they expect * startRepaintLayer fails by default - tests should override if expected * startFrame/endFrame do nothing by default - tests should override to intercept */ class TestRendererBase { public: virtual ~TestRendererBase() {} virtual OffscreenBuffer* startTemporaryLayer(uint32_t, uint32_t) { ADD_FAILURE() << "Temporary layers not expected in this test"; return nullptr; } virtual void recycleTemporaryLayer(OffscreenBuffer*) { ADD_FAILURE() << "Temporary layers not expected in this test"; } virtual void startRepaintLayer(OffscreenBuffer*, const Rect& repaintRect) { ADD_FAILURE() << "Layer repaint not expected in this test"; } virtual void endLayer() { ADD_FAILURE() << "Layer updates not expected in this test"; } virtual void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) {} virtual void endFrame(const Rect& repaintRect) {} // define virtual defaults for single draw methods #define X(Type) \ virtual void on##Type(const Type&, const BakedOpState&) { \ ADD_FAILURE() << #Type " not expected in this test"; \ } MAP_RENDERABLE_OPS(X) #undef X // define virtual defaults for merged draw methods #define X(Type) \ virtual void onMerged##Type##s(const MergedBakedOpList& opList) { \ ADD_FAILURE() << "Merged " #Type "s not expected in this test"; \ } MAP_MERGEABLE_OPS(X) #undef X int getIndex() { return mIndex; } protected: int mIndex = 0; }; /** * Dispatches all static methods to similar formed methods on renderer, which fail by default but * are overridden by subclasses per test. */ class TestDispatcher { public: // define single op methods, which redirect to TestRendererBase #define X(Type) \ static void on##Type(TestRendererBase& renderer, const Type& op, const BakedOpState& state) { \ renderer.on##Type(op, state); \ } MAP_RENDERABLE_OPS(X); #undef X // define merged op methods, which redirect to TestRendererBase #define X(Type) \ static void onMerged##Type##s(TestRendererBase& renderer, const MergedBakedOpList& opList) { \ renderer.onMerged##Type##s(opList); \ } MAP_MERGEABLE_OPS(X); #undef X }; class FailRenderer : public TestRendererBase {}; RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, simple) { class SimpleTestRenderer : public TestRendererBase { public: void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(100u, width); EXPECT_EQ(200u, height); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); } void onBitmapOp(const BitmapOp& op, const BakedOpState& state) override { EXPECT_EQ(2, mIndex++); } void endFrame(const Rect& repaintRect) override { EXPECT_EQ(3, mIndex++); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 200, [](RenderProperties& props, RecordingCanvas& canvas) { sk_sp<Bitmap> bitmap(TestUtils::createBitmap(25, 25)); canvas.drawRect(0, 0, 100, 200, SkPaint()); canvas.drawBitmap(*bitmap, 10, 10, nullptr); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 200), 100, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SimpleTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); // 2 ops + start + end } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, simpleStroke) { class SimpleStrokeTestRenderer : public TestRendererBase { public: void onPointsOp(const PointsOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); // even though initial bounds are empty... EXPECT_TRUE(op.unmappedBounds.isEmpty()) << "initial bounds should be empty, since they're unstroked"; EXPECT_EQ(Rect(45, 45, 55, 55), state.computedState.clippedBounds) << "final bounds should account for stroke"; } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 200, [](RenderProperties& props, RecordingCanvas& canvas) { SkPaint strokedPaint; strokedPaint.setStrokeWidth(10); canvas.drawPoint(50, 50, strokedPaint); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 200), 100, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SimpleStrokeTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, arcStrokeClip) { class ArcStrokeClipTestRenderer : public TestRendererBase { public: void onArcOp(const ArcOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(25, 25, 175, 175), op.unmappedBounds); EXPECT_EQ(Rect(25, 25, 175, 175), state.computedState.clippedBounds); EXPECT_EQ(OpClipSideFlags::Full, state.computedState.clipSideFlags) << "Arc op clipped conservatively, since path texture may be expanded"; } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.clipRect(25, 25, 175, 175, SkClipOp::kIntersect); SkPaint aaPaint; aaPaint.setAntiAlias(true); canvas.drawArc(25, 25, 175, 175, 40, 180, true, aaPaint); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); ArcStrokeClipTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, simpleRejection) { auto node = TestUtils::createNode<RecordingCanvas>(0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.save(SaveFlags::MatrixClip); canvas.clipRect(200, 200, 400, 400, SkClipOp::kIntersect); // intersection should be empty canvas.drawRect(0, 0, 400, 400, SkPaint()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); FailRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, simpleBatching) { const int LOOPS = 5; class SimpleBatchingTestRenderer : public TestRendererBase { public: void onBitmapOp(const BitmapOp& op, const BakedOpState& state) override { EXPECT_TRUE(mIndex++ >= LOOPS) << "Bitmaps should be above all rects"; } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_TRUE(mIndex++ < LOOPS) << "Rects should be below all bitmaps"; } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { sk_sp<Bitmap> bitmap(TestUtils::createBitmap( 10, 10, kAlpha_8_SkColorType)); // Disable merging by using alpha 8 bitmap // Alternate between drawing rects and bitmaps, with bitmaps overlapping rects. // Rects don't overlap bitmaps, so bitmaps should be brought to front as a group. canvas.save(SaveFlags::MatrixClip); for (int i = 0; i < LOOPS; i++) { canvas.translate(0, 10); canvas.drawRect(0, 0, 10, 10, SkPaint()); canvas.drawBitmap(*bitmap, 5, 0, nullptr); } canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SimpleBatchingTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2 * LOOPS, renderer.getIndex()) << "Expect number of ops = 2 * loop count"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, deferRenderNode_translateClip) { class DeferRenderNodeTranslateClipTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(5, 10, 55, 60), state.computedState.clippedBounds); EXPECT_EQ(OpClipSideFlags::Right | OpClipSideFlags::Bottom, state.computedState.clipSideFlags); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.drawRect(0, 0, 100, 100, SkPaint()); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(5, 10, Rect(50, 50), // translate + clip node *TestUtils::getSyncedNode(node)); DeferRenderNodeTranslateClipTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, deferRenderNodeScene) { class DeferRenderNodeSceneTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { const Rect& clippedBounds = state.computedState.clippedBounds; Matrix4 expected; switch (mIndex++) { case 0: // background - left side EXPECT_EQ(Rect(600, 100, 700, 500), clippedBounds); expected.loadTranslate(100, 100, 0); break; case 1: // background - top side EXPECT_EQ(Rect(100, 400, 600, 500), clippedBounds); expected.loadTranslate(100, 100, 0); break; case 2: // content EXPECT_EQ(Rect(100, 100, 700, 500), clippedBounds); expected.loadTranslate(-50, -50, 0); break; case 3: // overlay EXPECT_EQ(Rect(0, 0, 800, 200), clippedBounds); break; default: ADD_FAILURE() << "Too many rects observed"; } EXPECT_EQ(expected, state.computedState.transform); } }; std::vector<sp<RenderNode>> nodes; SkPaint transparentPaint; transparentPaint.setAlpha(128); // backdrop nodes.push_back(TestUtils::createNode<RecordingCanvas>( 100, 100, 700, 500, // 600x400 [&transparentPaint](RenderProperties& props, RecordingCanvas& canvas) { canvas.drawRect(0, 0, 600, 400, transparentPaint); })); // content Rect contentDrawBounds(150, 150, 650, 450); // 500x300 nodes.push_back(TestUtils::createNode<RecordingCanvas>( 0, 0, 800, 600, [&transparentPaint](RenderProperties& props, RecordingCanvas& canvas) { canvas.drawRect(0, 0, 800, 600, transparentPaint); })); // overlay nodes.push_back(TestUtils::createNode<RecordingCanvas>( 0, 0, 800, 600, [&transparentPaint](RenderProperties& props, RecordingCanvas& canvas) { canvas.drawRect(0, 0, 800, 200, transparentPaint); })); for (auto& node : nodes) { TestUtils::syncHierarchyPropertiesAndDisplayList(node); } { FrameBuilder frameBuilder(SkRect::MakeWH(800, 600), 800, 600, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNodeScene(nodes, contentDrawBounds); DeferRenderNodeSceneTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } for (auto& node : nodes) { EXPECT_TRUE(node->isValid()); EXPECT_FALSE(node->nothingToDraw()); node->setStagingDisplayList(nullptr); EXPECT_FALSE(node->isValid()); EXPECT_FALSE(node->nothingToDraw()); node->destroyHardwareResources(); EXPECT_TRUE(node->nothingToDraw()); EXPECT_FALSE(node->isValid()); } { // Validate no crashes if any nodes are missing DisplayLists FrameBuilder frameBuilder(SkRect::MakeWH(800, 600), 800, 600, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNodeScene(nodes, contentDrawBounds); FailRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); } } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, empty_noFbo0) { class EmptyNoFbo0TestRenderer : public TestRendererBase { public: void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { ADD_FAILURE() << "Primary frame draw not expected in this test"; } void endFrame(const Rect& repaintRect) override { ADD_FAILURE() << "Primary frame draw not expected in this test"; } }; // Use layer update constructor, so no work is enqueued for Fbo0 LayerUpdateQueue emptyLayerUpdateQueue; FrameBuilder frameBuilder(emptyLayerUpdateQueue, sLightGeometry, Caches::getInstance()); EmptyNoFbo0TestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, empty_withFbo0) { class EmptyWithFbo0TestRenderer : public TestRendererBase { public: void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { EXPECT_EQ(0, mIndex++); } void endFrame(const Rect& repaintRect) override { EXPECT_EQ(1, mIndex++); } }; auto node = TestUtils::createNode<RecordingCanvas>( 10, 10, 110, 110, [](RenderProperties& props, RecordingCanvas& canvas) { // no drawn content }); // Draw, but pass node without draw content, so no work is done for primary frame FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); EmptyWithFbo0TestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()) << "No drawing content produced," " but fbo0 update lifecycle should still be observed"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, avoidOverdraw_rects) { class AvoidOverdrawRectsTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(mIndex++, 0) << "Should be one rect"; EXPECT_EQ(Rect(10, 10, 190, 190), op.unmappedBounds) << "Last rect should occlude others."; } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.drawRect(0, 0, 200, 200, SkPaint()); canvas.drawRect(0, 0, 200, 200, SkPaint()); canvas.drawRect(10, 10, 190, 190, SkPaint()); }); // Damage (and therefore clip) is same as last draw, subset of renderable area. // This means last op occludes other contents, and they'll be rejected to avoid overdraw. FrameBuilder frameBuilder(SkRect::MakeLTRB(10, 10, 190, 190), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); EXPECT_EQ(3u, node->getDisplayList()->getOps().size()) << "Recording must not have rejected ops, in order for this test to be valid"; AvoidOverdrawRectsTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()) << "Expect exactly one op"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, avoidOverdraw_bitmaps) { static sk_sp<Bitmap> opaqueBitmap( TestUtils::createBitmap(50, 50, SkColorType::kRGB_565_SkColorType)); static sk_sp<Bitmap> transpBitmap( TestUtils::createBitmap(50, 50, SkColorType::kAlpha_8_SkColorType)); class AvoidOverdrawBitmapsTestRenderer : public TestRendererBase { public: void onBitmapOp(const BitmapOp& op, const BakedOpState& state) override { switch (mIndex++) { case 0: EXPECT_EQ(opaqueBitmap.get(), op.bitmap); break; case 1: EXPECT_EQ(transpBitmap.get(), op.bitmap); break; default: ADD_FAILURE() << "Only two ops expected."; } } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 50, 50, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.drawRect(0, 0, 50, 50, SkPaint()); canvas.drawRect(0, 0, 50, 50, SkPaint()); canvas.drawBitmap(*transpBitmap, 0, 0, nullptr); // only the below draws should remain, since they're canvas.drawBitmap(*opaqueBitmap, 0, 0, nullptr); canvas.drawBitmap(*transpBitmap, 0, 0, nullptr); }); FrameBuilder frameBuilder(SkRect::MakeWH(50, 50), 50, 50, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); EXPECT_EQ(5u, node->getDisplayList()->getOps().size()) << "Recording must not have rejected ops, in order for this test to be valid"; AvoidOverdrawBitmapsTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()) << "Expect exactly two ops"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, clippedMerging) { class ClippedMergingTestRenderer : public TestRendererBase { public: void onMergedBitmapOps(const MergedBakedOpList& opList) override { EXPECT_EQ(0, mIndex); mIndex += opList.count; EXPECT_EQ(4u, opList.count); EXPECT_EQ(Rect(10, 10, 90, 90), opList.clip); EXPECT_EQ(OpClipSideFlags::Left | OpClipSideFlags::Top | OpClipSideFlags::Right, opList.clipSideFlags); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { sk_sp<Bitmap> bitmap(TestUtils::createBitmap(20, 20)); // left side clipped (to inset left half) canvas.clipRect(10, 0, 50, 100, SkClipOp::kReplace_deprecated); canvas.drawBitmap(*bitmap, 0, 40, nullptr); // top side clipped (to inset top half) canvas.clipRect(0, 10, 100, 50, SkClipOp::kReplace_deprecated); canvas.drawBitmap(*bitmap, 40, 0, nullptr); // right side clipped (to inset right half) canvas.clipRect(50, 0, 90, 100, SkClipOp::kReplace_deprecated); canvas.drawBitmap(*bitmap, 80, 40, nullptr); // bottom not clipped, just abutting (inset bottom half) canvas.clipRect(0, 50, 100, 90, SkClipOp::kReplace_deprecated); canvas.drawBitmap(*bitmap, 40, 70, nullptr); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); ClippedMergingTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, regionClipStopsMerge) { class RegionClipStopsMergeTestRenderer : public TestRendererBase { public: void onTextOp(const TextOp& op, const BakedOpState& state) override { mIndex++; } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 400, [](RenderProperties& props, RecordingCanvas& canvas) { SkPath path; path.addCircle(200, 200, 200, SkPath::kCW_Direction); canvas.save(SaveFlags::MatrixClip); canvas.clipPath(&path, SkClipOp::kIntersect); SkPaint paint; paint.setAntiAlias(true); paint.setTextSize(50); TestUtils::drawUtf8ToCanvas(&canvas, "Test string1", paint, 100, 100); TestUtils::drawUtf8ToCanvas(&canvas, "Test string1", paint, 100, 200); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(400, 400), 400, 400, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); RegionClipStopsMergeTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, textMerging) { class TextMergingTestRenderer : public TestRendererBase { public: void onMergedTextOps(const MergedBakedOpList& opList) override { EXPECT_EQ(0, mIndex); mIndex += opList.count; EXPECT_EQ(2u, opList.count); EXPECT_EQ(OpClipSideFlags::Top, opList.clipSideFlags); EXPECT_EQ(OpClipSideFlags::Top, opList.states[0]->computedState.clipSideFlags); EXPECT_EQ(OpClipSideFlags::None, opList.states[1]->computedState.clipSideFlags); } }; auto node = TestUtils::createNode<RecordingCanvas>(0, 0, 400, 400, [](RenderProperties& props, RecordingCanvas& canvas) { SkPaint paint; paint.setAntiAlias(true); paint.setTextSize(50); TestUtils::drawUtf8ToCanvas(&canvas, "Test string1", paint, 100, 0); // will be top clipped TestUtils::drawUtf8ToCanvas(&canvas, "Test string1", paint, 100, 100); // not clipped }); FrameBuilder frameBuilder(SkRect::MakeWH(400, 400), 400, 400, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); TextMergingTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()) << "Expect 2 ops"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, textStrikethrough) { const int LOOPS = 5; class TextStrikethroughTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_TRUE(mIndex++ >= LOOPS) << "Strikethrough rects should be above all text"; } void onMergedTextOps(const MergedBakedOpList& opList) override { EXPECT_EQ(0, mIndex); mIndex += opList.count; EXPECT_EQ(5u, opList.count); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 2000, [](RenderProperties& props, RecordingCanvas& canvas) { SkPaint textPaint; textPaint.setAntiAlias(true); textPaint.setTextSize(20); textPaint.setFlags(textPaint.getFlags() | SkPaint::kStrikeThruText_ReserveFlag); for (int i = 0; i < LOOPS; i++) { TestUtils::drawUtf8ToCanvas(&canvas, "test text", textPaint, 10, 100 * (i + 1)); } }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 2000), 200, 2000, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); TextStrikethroughTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2 * LOOPS, renderer.getIndex()) << "Expect number of ops = 2 * loop count"; } static auto styles = {SkPaint::kFill_Style, SkPaint::kStroke_Style, SkPaint::kStrokeAndFill_Style}; RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, textStyle) { class TextStyleTestRenderer : public TestRendererBase { public: void onMergedTextOps(const MergedBakedOpList& opList) override { ASSERT_EQ(0, mIndex); ASSERT_EQ(3u, opList.count); mIndex += opList.count; int index = 0; for (auto style : styles) { auto state = opList.states[index++]; ASSERT_EQ(style, state->op->paint->getStyle()) << "Remainder of validation relies upon stable merged order"; ASSERT_EQ(0, state->computedState.clipSideFlags) << "Clipped bounds validation requires unclipped ops"; } Rect fill = opList.states[0]->computedState.clippedBounds; Rect stroke = opList.states[1]->computedState.clippedBounds; EXPECT_EQ(stroke, opList.states[2]->computedState.clippedBounds) << "Stroke+Fill should be same as stroke"; EXPECT_TRUE(stroke.contains(fill)); EXPECT_FALSE(fill.contains(stroke)); // outset by half the stroke width Rect outsetFill(fill); outsetFill.outset(5); EXPECT_EQ(stroke, outsetFill); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 400, [](RenderProperties& props, RecordingCanvas& canvas) { SkPaint paint; paint.setAntiAlias(true); paint.setTextSize(50); paint.setStrokeWidth(10); // draw 3 copies of the same text overlapping, each with a different style. // They'll get merged, but with for (auto style : styles) { paint.setStyle(style); TestUtils::drawUtf8ToCanvas(&canvas, "Test string1", paint, 100, 100); } }); FrameBuilder frameBuilder(SkRect::MakeWH(400, 400), 400, 400, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); TextStyleTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()) << "Expect 3 ops"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, textureLayer_clipLocalMatrix) { class TextureLayerClipLocalMatrixTestRenderer : public TestRendererBase { public: void onTextureLayerOp(const TextureLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(50, 50, 150, 150), state.computedState.clipRect()); EXPECT_EQ(Rect(50, 50, 105, 105), state.computedState.clippedBounds); Matrix4 expected; expected.loadTranslate(5, 5, 0); EXPECT_MATRIX_APPROX_EQ(expected, state.computedState.transform); } }; auto layerUpdater = TestUtils::createTextureLayerUpdater(renderThread, 100, 100, SkMatrix::MakeTrans(5, 5)); auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [&layerUpdater](RenderProperties& props, RecordingCanvas& canvas) { canvas.save(SaveFlags::MatrixClip); canvas.clipRect(50, 50, 150, 150, SkClipOp::kIntersect); canvas.drawLayer(layerUpdater.get()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); TextureLayerClipLocalMatrixTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, textureLayer_combineMatrices) { class TextureLayerCombineMatricesTestRenderer : public TestRendererBase { public: void onTextureLayerOp(const TextureLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); Matrix4 expected; expected.loadTranslate(35, 45, 0); EXPECT_MATRIX_APPROX_EQ(expected, state.computedState.transform); } }; auto layerUpdater = TestUtils::createTextureLayerUpdater(renderThread, 100, 100, SkMatrix::MakeTrans(5, 5)); auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [&layerUpdater](RenderProperties& props, RecordingCanvas& canvas) { canvas.save(SaveFlags::MatrixClip); canvas.translate(30, 40); canvas.drawLayer(layerUpdater.get()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); TextureLayerCombineMatricesTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, textureLayer_reject) { auto layerUpdater = TestUtils::createTextureLayerUpdater(renderThread, 100, 100, SkMatrix::MakeTrans(5, 5)); EXPECT_EQ(Layer::Api::OpenGL, layerUpdater->backingLayer()->getApi()); GlLayer* glLayer = static_cast<GlLayer*>(layerUpdater->backingLayer()); glLayer->setRenderTarget(GL_NONE); // Should be rejected auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [&layerUpdater](RenderProperties& props, RecordingCanvas& canvas) { canvas.drawLayer(layerUpdater.get()); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); FailRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, functor_reject) { class FunctorTestRenderer : public TestRendererBase { public: void onFunctorOp(const FunctorOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); } }; Functor noopFunctor; // 1 million pixel tall view, scrolled down 80% auto scrolledFunctorView = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 1000000, [&noopFunctor](RenderProperties& props, RecordingCanvas& canvas) { canvas.translate(0, -800000); canvas.callDrawGLFunction(&noopFunctor, nullptr); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(scrolledFunctorView)); FunctorTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()) << "Functor should not be rejected"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, deferColorOp_unbounded) { class ColorTestRenderer : public TestRendererBase { public: void onColorOp(const ColorOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(200, 200), state.computedState.clippedBounds) << "Color op should be expanded to bounds of surrounding"; } }; auto unclippedColorView = TestUtils::createNode<RecordingCanvas>( 0, 0, 10, 10, [](RenderProperties& props, RecordingCanvas& canvas) { props.setClipToBounds(false); canvas.drawColor(SK_ColorWHITE, SkBlendMode::kSrcOver); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(unclippedColorView)); ColorTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()) << "ColorOp should not be rejected"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderNode) { class RenderNodeTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { switch (mIndex++) { case 0: EXPECT_EQ(Rect(200, 200), state.computedState.clippedBounds); EXPECT_EQ(SK_ColorDKGRAY, op.paint->getColor()); break; case 1: EXPECT_EQ(Rect(50, 50, 150, 150), state.computedState.clippedBounds); EXPECT_EQ(SK_ColorWHITE, op.paint->getColor()); break; default: ADD_FAILURE(); } } }; auto child = TestUtils::createNode<RecordingCanvas>( 10, 10, 110, 110, [](RenderProperties& props, RecordingCanvas& canvas) { SkPaint paint; paint.setColor(SK_ColorWHITE); canvas.drawRect(0, 0, 100, 100, paint); }); auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [&child](RenderProperties& props, RecordingCanvas& canvas) { SkPaint paint; paint.setColor(SK_ColorDKGRAY); canvas.drawRect(0, 0, 200, 200, paint); canvas.save(SaveFlags::MatrixClip); canvas.translate(40, 40); canvas.drawRenderNode(child.get()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); RenderNodeTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, clipped) { class ClippedTestRenderer : public TestRendererBase { public: void onBitmapOp(const BitmapOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(10, 20, 30, 40), state.computedState.clippedBounds); EXPECT_EQ(Rect(10, 20, 30, 40), state.computedState.clipRect()); EXPECT_TRUE(state.computedState.transform.isIdentity()); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { sk_sp<Bitmap> bitmap(TestUtils::createBitmap(200, 200)); canvas.drawBitmap(*bitmap, 0, 0, nullptr); }); // clip to small area, should see in receiver FrameBuilder frameBuilder(SkRect::MakeLTRB(10, 20, 30, 40), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); ClippedTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayer_simple) { class SaveLayerSimpleTestRenderer : public TestRendererBase { public: OffscreenBuffer* startTemporaryLayer(uint32_t width, uint32_t height) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(180u, width); EXPECT_EQ(180u, height); return nullptr; } void endLayer() override { EXPECT_EQ(2, mIndex++); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); EXPECT_EQ(Rect(10, 10, 190, 190), op.unmappedBounds); EXPECT_EQ(Rect(180, 180), state.computedState.clippedBounds); EXPECT_EQ(Rect(180, 180), state.computedState.clipRect()); Matrix4 expectedTransform; expectedTransform.loadTranslate(-10, -10, 0); EXPECT_MATRIX_APPROX_EQ(expectedTransform, state.computedState.transform); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { EXPECT_EQ(3, mIndex++); EXPECT_EQ(Rect(10, 10, 190, 190), state.computedState.clippedBounds); EXPECT_EQ(Rect(200, 200), state.computedState.clipRect()); EXPECT_TRUE(state.computedState.transform.isIdentity()); } void recycleTemporaryLayer(OffscreenBuffer* offscreenBuffer) override { EXPECT_EQ(4, mIndex++); EXPECT_EQ(nullptr, offscreenBuffer); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.saveLayerAlpha(10, 10, 190, 190, 128, SaveFlags::ClipToLayer); canvas.drawRect(10, 10, 190, 190, SkPaint()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SaveLayerSimpleTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(5, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayer_nested) { /* saveLayer1 { rect1, saveLayer2 { rect2 } } will play back as: * - startTemporaryLayer2, rect2 endLayer2 * - startTemporaryLayer1, rect1, drawLayer2, endLayer1 * - startFrame, layerOp1, endFrame */ class SaveLayerNestedTestRenderer : public TestRendererBase { public: OffscreenBuffer* startTemporaryLayer(uint32_t width, uint32_t height) override { const int index = mIndex++; if (index == 0) { EXPECT_EQ(400u, width); EXPECT_EQ(400u, height); return (OffscreenBuffer*)0x400; } else if (index == 3) { EXPECT_EQ(800u, width); EXPECT_EQ(800u, height); return (OffscreenBuffer*)0x800; } else { ADD_FAILURE(); } return (OffscreenBuffer*)nullptr; } void endLayer() override { int index = mIndex++; EXPECT_TRUE(index == 2 || index == 6); } void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { EXPECT_EQ(7, mIndex++); } void endFrame(const Rect& repaintRect) override { EXPECT_EQ(9, mIndex++); } void onRectOp(const RectOp& op, const BakedOpState& state) override { const int index = mIndex++; if (index == 1) { EXPECT_EQ(Rect(400, 400), op.unmappedBounds); // inner rect } else if (index == 4) { EXPECT_EQ(Rect(800, 800), op.unmappedBounds); // outer rect } else { ADD_FAILURE(); } } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { const int index = mIndex++; if (index == 5) { EXPECT_EQ((OffscreenBuffer*)0x400, *op.layerHandle); EXPECT_EQ(Rect(400, 400), op.unmappedBounds); // inner layer } else if (index == 8) { EXPECT_EQ((OffscreenBuffer*)0x800, *op.layerHandle); EXPECT_EQ(Rect(800, 800), op.unmappedBounds); // outer layer } else { ADD_FAILURE(); } } void recycleTemporaryLayer(OffscreenBuffer* offscreenBuffer) override { const int index = mIndex++; // order isn't important, but we need to see both if (index == 10) { EXPECT_EQ((OffscreenBuffer*)0x400, offscreenBuffer); } else if (index == 11) { EXPECT_EQ((OffscreenBuffer*)0x800, offscreenBuffer); } else { ADD_FAILURE(); } } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 800, 800, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.saveLayerAlpha(0, 0, 800, 800, 128, SaveFlags::ClipToLayer); { canvas.drawRect(0, 0, 800, 800, SkPaint()); canvas.saveLayerAlpha(0, 0, 400, 400, 128, SaveFlags::ClipToLayer); { canvas.drawRect(0, 0, 400, 400, SkPaint()); } canvas.restore(); } canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(800, 800), 800, 800, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SaveLayerNestedTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(12, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayer_contentRejection) { auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.save(SaveFlags::MatrixClip); canvas.clipRect(200, 200, 400, 400, SkClipOp::kIntersect); canvas.saveLayerAlpha(200, 200, 400, 400, 128, SaveFlags::ClipToLayer); // draw within save layer may still be recorded, but shouldn't be drawn canvas.drawRect(200, 200, 400, 400, SkPaint()); canvas.restore(); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); FailRenderer renderer; // should see no ops, even within the layer, since the layer should be rejected frameBuilder.replayBakedOps<TestDispatcher>(renderer); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayerUnclipped_simple) { class SaveLayerUnclippedSimpleTestRenderer : public TestRendererBase { public: void onCopyToLayerOp(const CopyToLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(10, 10, 190, 190), state.computedState.clippedBounds); EXPECT_CLIP_RECT(Rect(200, 200), state.computedState.clipState); EXPECT_TRUE(state.computedState.transform.isIdentity()); } void onSimpleRectsOp(const SimpleRectsOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); ASSERT_NE(nullptr, op.paint); ASSERT_EQ(SkBlendMode::kClear, PaintUtils::getBlendModeDirect(op.paint)); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(2, mIndex++); EXPECT_EQ(Rect(200, 200), op.unmappedBounds); EXPECT_EQ(Rect(200, 200), state.computedState.clippedBounds); EXPECT_EQ(Rect(200, 200), state.computedState.clipRect()); EXPECT_TRUE(state.computedState.transform.isIdentity()); } void onCopyFromLayerOp(const CopyFromLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(3, mIndex++); EXPECT_EQ(Rect(10, 10, 190, 190), state.computedState.clippedBounds); EXPECT_CLIP_RECT(Rect(200, 200), state.computedState.clipState); EXPECT_TRUE(state.computedState.transform.isIdentity()); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.saveLayerAlpha(10, 10, 190, 190, 128, (SaveFlags::Flags)(0)); canvas.drawRect(0, 0, 200, 200, SkPaint()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SaveLayerUnclippedSimpleTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayerUnclipped_round) { class SaveLayerUnclippedRoundTestRenderer : public TestRendererBase { public: void onCopyToLayerOp(const CopyToLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(10, 10, 190, 190), state.computedState.clippedBounds) << "Bounds rect should round out"; } void onSimpleRectsOp(const SimpleRectsOp& op, const BakedOpState& state) override {} void onRectOp(const RectOp& op, const BakedOpState& state) override {} void onCopyFromLayerOp(const CopyFromLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); EXPECT_EQ(Rect(10, 10, 190, 190), state.computedState.clippedBounds) << "Bounds rect should round out"; } }; auto node = TestUtils::createNode<RecordingCanvas>(0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.saveLayerAlpha(10.95f, 10.5f, 189.75f, 189.25f, // values should all round out 128, (SaveFlags::Flags)(0)); canvas.drawRect(0, 0, 200, 200, SkPaint()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SaveLayerUnclippedRoundTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayerUnclipped_mergedClears) { class SaveLayerUnclippedMergedClearsTestRenderer : public TestRendererBase { public: void onCopyToLayerOp(const CopyToLayerOp& op, const BakedOpState& state) override { int index = mIndex++; EXPECT_GT(4, index); EXPECT_EQ(5, op.unmappedBounds.getWidth()); EXPECT_EQ(5, op.unmappedBounds.getHeight()); if (index == 0) { EXPECT_EQ(Rect(10, 10), state.computedState.clippedBounds); } else if (index == 1) { EXPECT_EQ(Rect(190, 0, 200, 10), state.computedState.clippedBounds); } else if (index == 2) { EXPECT_EQ(Rect(0, 190, 10, 200), state.computedState.clippedBounds); } else if (index == 3) { EXPECT_EQ(Rect(190, 190, 200, 200), state.computedState.clippedBounds); } } void onSimpleRectsOp(const SimpleRectsOp& op, const BakedOpState& state) override { EXPECT_EQ(4, mIndex++); ASSERT_EQ(op.vertexCount, 16u); for (size_t i = 0; i < op.vertexCount; i++) { auto v = op.vertices[i]; EXPECT_TRUE(v.x == 0 || v.x == 10 || v.x == 190 || v.x == 200); EXPECT_TRUE(v.y == 0 || v.y == 10 || v.y == 190 || v.y == 200); } } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(5, mIndex++); } void onCopyFromLayerOp(const CopyFromLayerOp& op, const BakedOpState& state) override { EXPECT_LT(5, mIndex++); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { int restoreTo = canvas.save(SaveFlags::MatrixClip); canvas.scale(2, 2); canvas.saveLayerAlpha(0, 0, 5, 5, 128, SaveFlags::MatrixClip); canvas.saveLayerAlpha(95, 0, 100, 5, 128, SaveFlags::MatrixClip); canvas.saveLayerAlpha(0, 95, 5, 100, 128, SaveFlags::MatrixClip); canvas.saveLayerAlpha(95, 95, 100, 100, 128, SaveFlags::MatrixClip); canvas.drawRect(0, 0, 100, 100, SkPaint()); canvas.restoreToCount(restoreTo); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SaveLayerUnclippedMergedClearsTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(10, renderer.getIndex()) << "Expect 4 copyTos, 4 copyFroms, 1 clear SimpleRects, and 1 rect."; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayerUnclipped_clearClip) { class SaveLayerUnclippedClearClipTestRenderer : public TestRendererBase { public: void onCopyToLayerOp(const CopyToLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); } void onSimpleRectsOp(const SimpleRectsOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); ASSERT_NE(nullptr, op.paint); EXPECT_EQ(SkBlendMode::kClear, PaintUtils::getBlendModeDirect(op.paint)); EXPECT_EQ(Rect(50, 50, 150, 150), state.computedState.clippedBounds) << "Expect dirty rect as clip"; ASSERT_NE(nullptr, state.computedState.clipState); EXPECT_EQ(Rect(50, 50, 150, 150), state.computedState.clipState->rect); EXPECT_EQ(ClipMode::Rectangle, state.computedState.clipState->mode); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(2, mIndex++); } void onCopyFromLayerOp(const CopyFromLayerOp& op, const BakedOpState& state) override { EXPECT_EQ(3, mIndex++); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { // save smaller than clip, so we get unclipped behavior canvas.saveLayerAlpha(10, 10, 190, 190, 128, (SaveFlags::Flags)(0)); canvas.drawRect(0, 0, 200, 200, SkPaint()); canvas.restore(); }); // draw with partial screen dirty, and assert we see that rect later FrameBuilder frameBuilder(SkRect::MakeLTRB(50, 50, 150, 150), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SaveLayerUnclippedClearClipTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayerUnclipped_reject) { auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { // unclipped savelayer + rect both in area that won't intersect with dirty canvas.saveLayerAlpha(100, 100, 200, 200, 128, (SaveFlags::Flags)(0)); canvas.drawRect(100, 100, 200, 200, SkPaint()); canvas.restore(); }); // draw with partial screen dirty that doesn't intersect with savelayer FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); FailRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); } /* saveLayerUnclipped { saveLayer { saveLayerUnclipped { rect } } } will play back as: * - startTemporaryLayer, onCopyToLayer, onSimpleRects, onRect, onCopyFromLayer, endLayer * - startFrame, onCopyToLayer, onSimpleRects, drawLayer, onCopyFromLayer, endframe */ RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, saveLayerUnclipped_complex) { class SaveLayerUnclippedComplexTestRenderer : public TestRendererBase { public: OffscreenBuffer* startTemporaryLayer(uint32_t width, uint32_t height) { EXPECT_EQ(0, mIndex++); // savelayer first return (OffscreenBuffer*)0xabcd; } void onCopyToLayerOp(const CopyToLayerOp& op, const BakedOpState& state) override { int index = mIndex++; EXPECT_TRUE(index == 1 || index == 7); } void onSimpleRectsOp(const SimpleRectsOp& op, const BakedOpState& state) override { int index = mIndex++; EXPECT_TRUE(index == 2 || index == 8); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(3, mIndex++); Matrix4 expected; expected.loadTranslate(-100, -100, 0); EXPECT_EQ(Rect(100, 100, 200, 200), state.computedState.clippedBounds); EXPECT_MATRIX_APPROX_EQ(expected, state.computedState.transform); } void onCopyFromLayerOp(const CopyFromLayerOp& op, const BakedOpState& state) override { int index = mIndex++; EXPECT_TRUE(index == 4 || index == 10); } void endLayer() override { EXPECT_EQ(5, mIndex++); } void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { EXPECT_EQ(6, mIndex++); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { EXPECT_EQ(9, mIndex++); EXPECT_EQ((OffscreenBuffer*)0xabcd, *op.layerHandle); } void endFrame(const Rect& repaintRect) override { EXPECT_EQ(11, mIndex++); } void recycleTemporaryLayer(OffscreenBuffer* offscreenBuffer) override { EXPECT_EQ(12, mIndex++); EXPECT_EQ((OffscreenBuffer*)0xabcd, offscreenBuffer); } }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 600, 600, // 500x500 triggers clipping [](RenderProperties& props, RecordingCanvas& canvas) { canvas.saveLayerAlpha(0, 0, 500, 500, 128, (SaveFlags::Flags)0); // unclipped canvas.saveLayerAlpha(100, 100, 400, 400, 128, SaveFlags::ClipToLayer); // clipped canvas.saveLayerAlpha(200, 200, 300, 300, 128, (SaveFlags::Flags)0); // unclipped canvas.drawRect(200, 200, 300, 300, SkPaint()); canvas.restore(); canvas.restore(); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(600, 600), 600, 600, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); SaveLayerUnclippedComplexTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(13, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, hwLayer_simple) { class HwLayerSimpleTestRenderer : public TestRendererBase { public: void startRepaintLayer(OffscreenBuffer* offscreenBuffer, const Rect& repaintRect) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(100u, offscreenBuffer->viewportWidth); EXPECT_EQ(100u, offscreenBuffer->viewportHeight); EXPECT_EQ(Rect(25, 25, 75, 75), repaintRect); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); EXPECT_TRUE(state.computedState.transform.isIdentity()) << "Transform should be reset within layer"; EXPECT_EQ(Rect(25, 25, 75, 75), state.computedState.clipRect()) << "Damage rect should be used to clip layer content"; } void endLayer() override { EXPECT_EQ(2, mIndex++); } void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { EXPECT_EQ(3, mIndex++); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { EXPECT_EQ(4, mIndex++); } void endFrame(const Rect& repaintRect) override { EXPECT_EQ(5, mIndex++); } }; auto node = TestUtils::createNode<RecordingCanvas>( 10, 10, 110, 110, [](RenderProperties& props, RecordingCanvas& canvas) { props.mutateLayerProperties().setType(LayerType::RenderLayer); SkPaint paint; paint.setColor(SK_ColorWHITE); canvas.drawRect(0, 0, 100, 100, paint); }); OffscreenBuffer** layerHandle = node->getLayerHandle(); // create RenderNode's layer here in same way prepareTree would OffscreenBuffer layer(renderThread.renderState(), Caches::getInstance(), 100, 100); *layerHandle = &layer; auto syncedNode = TestUtils::getSyncedNode(node); // only enqueue partial damage LayerUpdateQueue layerUpdateQueue; // Note: enqueue damage post-sync, so bounds are valid layerUpdateQueue.enqueueLayerWithDamage(node.get(), Rect(25, 25, 75, 75)); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferLayers(layerUpdateQueue); frameBuilder.deferRenderNode(*syncedNode); HwLayerSimpleTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(6, renderer.getIndex()); // clean up layer pointer, so we can safely destruct RenderNode *layerHandle = nullptr; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, hwLayer_complex) { /* parentLayer { greyRect, saveLayer { childLayer { whiteRect } } } will play back as: * - startRepaintLayer(child), rect(grey), endLayer * - startTemporaryLayer, drawLayer(child), endLayer * - startRepaintLayer(parent), rect(white), drawLayer(saveLayer), endLayer * - startFrame, drawLayer(parent), endLayerb */ class HwLayerComplexTestRenderer : public TestRendererBase { public: OffscreenBuffer* startTemporaryLayer(uint32_t width, uint32_t height) { EXPECT_EQ(3, mIndex++); // savelayer first return (OffscreenBuffer*)0xabcd; } void startRepaintLayer(OffscreenBuffer* offscreenBuffer, const Rect& repaintRect) override { int index = mIndex++; if (index == 0) { // starting inner layer EXPECT_EQ(100u, offscreenBuffer->viewportWidth); EXPECT_EQ(100u, offscreenBuffer->viewportHeight); } else if (index == 6) { // starting outer layer EXPECT_EQ(200u, offscreenBuffer->viewportWidth); EXPECT_EQ(200u, offscreenBuffer->viewportHeight); } else { ADD_FAILURE(); } } void onRectOp(const RectOp& op, const BakedOpState& state) override { int index = mIndex++; if (index == 1) { // inner layer's rect (white) EXPECT_EQ(SK_ColorWHITE, op.paint->getColor()); } else if (index == 7) { // outer layer's rect (grey) EXPECT_EQ(SK_ColorDKGRAY, op.paint->getColor()); } else { ADD_FAILURE(); } } void endLayer() override { int index = mIndex++; EXPECT_TRUE(index == 2 || index == 5 || index == 9); } void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { EXPECT_EQ(10, mIndex++); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { OffscreenBuffer* layer = *op.layerHandle; int index = mIndex++; if (index == 4) { EXPECT_EQ(100u, layer->viewportWidth); EXPECT_EQ(100u, layer->viewportHeight); } else if (index == 8) { EXPECT_EQ((OffscreenBuffer*)0xabcd, *op.layerHandle); } else if (index == 11) { EXPECT_EQ(200u, layer->viewportWidth); EXPECT_EQ(200u, layer->viewportHeight); } else { ADD_FAILURE(); } } void endFrame(const Rect& repaintRect) override { EXPECT_EQ(12, mIndex++); } void recycleTemporaryLayer(OffscreenBuffer* offscreenBuffer) override { EXPECT_EQ(13, mIndex++); } }; auto child = TestUtils::createNode<RecordingCanvas>( 50, 50, 150, 150, [](RenderProperties& props, RecordingCanvas& canvas) { props.mutateLayerProperties().setType(LayerType::RenderLayer); SkPaint paint; paint.setColor(SK_ColorWHITE); canvas.drawRect(0, 0, 100, 100, paint); }); OffscreenBuffer childLayer(renderThread.renderState(), Caches::getInstance(), 100, 100); *(child->getLayerHandle()) = &childLayer; RenderNode* childPtr = child.get(); auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [childPtr](RenderProperties& props, RecordingCanvas& canvas) { props.mutateLayerProperties().setType(LayerType::RenderLayer); SkPaint paint; paint.setColor(SK_ColorDKGRAY); canvas.drawRect(0, 0, 200, 200, paint); canvas.saveLayerAlpha(50, 50, 150, 150, 128, SaveFlags::ClipToLayer); canvas.drawRenderNode(childPtr); canvas.restore(); }); OffscreenBuffer parentLayer(renderThread.renderState(), Caches::getInstance(), 200, 200); *(parent->getLayerHandle()) = &parentLayer; auto syncedNode = TestUtils::getSyncedNode(parent); LayerUpdateQueue layerUpdateQueue; // Note: enqueue damage post-sync, so bounds are valid layerUpdateQueue.enqueueLayerWithDamage(child.get(), Rect(100, 100)); layerUpdateQueue.enqueueLayerWithDamage(parent.get(), Rect(200, 200)); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferLayers(layerUpdateQueue); frameBuilder.deferRenderNode(*syncedNode); HwLayerComplexTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(14, renderer.getIndex()); // clean up layer pointers, so we can safely destruct RenderNodes *(child->getLayerHandle()) = nullptr; *(parent->getLayerHandle()) = nullptr; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, buildLayer) { class BuildLayerTestRenderer : public TestRendererBase { public: void startRepaintLayer(OffscreenBuffer* offscreenBuffer, const Rect& repaintRect) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(100u, offscreenBuffer->viewportWidth); EXPECT_EQ(100u, offscreenBuffer->viewportHeight); EXPECT_EQ(Rect(25, 25, 75, 75), repaintRect); } void onColorOp(const ColorOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); EXPECT_TRUE(state.computedState.transform.isIdentity()) << "Transform should be reset within layer"; EXPECT_EQ(Rect(25, 25, 75, 75), state.computedState.clipRect()) << "Damage rect should be used to clip layer content"; } void endLayer() override { EXPECT_EQ(2, mIndex++); } void startFrame(uint32_t width, uint32_t height, const Rect& repaintRect) override { ADD_FAILURE() << "Primary frame draw not expected in this test"; } void endFrame(const Rect& repaintRect) override { ADD_FAILURE() << "Primary frame draw not expected in this test"; } }; auto node = TestUtils::createNode<RecordingCanvas>( 10, 10, 110, 110, [](RenderProperties& props, RecordingCanvas& canvas) { props.mutateLayerProperties().setType(LayerType::RenderLayer); canvas.drawColor(SK_ColorWHITE, SkBlendMode::kSrcOver); }); OffscreenBuffer** layerHandle = node->getLayerHandle(); // create RenderNode's layer here in same way prepareTree would OffscreenBuffer layer(renderThread.renderState(), Caches::getInstance(), 100, 100); *layerHandle = &layer; TestUtils::syncHierarchyPropertiesAndDisplayList(node); // only enqueue partial damage LayerUpdateQueue layerUpdateQueue; // Note: enqueue damage post-sync, so bounds are valid layerUpdateQueue.enqueueLayerWithDamage(node.get(), Rect(25, 25, 75, 75)); // Draw, but pass empty node list, so no work is done for primary frame FrameBuilder frameBuilder(layerUpdateQueue, sLightGeometry, Caches::getInstance()); BuildLayerTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()); // clean up layer pointer, so we can safely destruct RenderNode *layerHandle = nullptr; } namespace { static void drawOrderedRect(Canvas* canvas, uint8_t expectedDrawOrder) { SkPaint paint; // order put in blue channel, transparent so overlapped content doesn't get rejected paint.setColor(SkColorSetARGB(1, 0, 0, expectedDrawOrder)); canvas->drawRect(0, 0, 100, 100, paint); } static void drawOrderedNode(Canvas* canvas, uint8_t expectedDrawOrder, float z) { auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [expectedDrawOrder](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedRect(&canvas, expectedDrawOrder); }); node->mutateStagingProperties().setTranslationZ(z); node->setPropertyFieldsDirty(RenderNode::TRANSLATION_Z); canvas->drawRenderNode(node.get()); // canvas takes reference/sole ownership } static void drawOrderedNode( Canvas* canvas, uint8_t expectedDrawOrder, std::function<void(RenderProperties& props, RecordingCanvas& canvas)> setup) { auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [expectedDrawOrder, setup](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedRect(&canvas, expectedDrawOrder); if (setup) { setup(props, canvas); } }); canvas->drawRenderNode(node.get()); // canvas takes reference/sole ownership } class ZReorderTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { int expectedOrder = SkColorGetB(op.paint->getColor()); // extract order from blue channel EXPECT_EQ(expectedOrder, mIndex++) << "An op was drawn out of order"; } }; } // end anonymous namespace RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, zReorder) { auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.insertReorderBarrier(true); canvas.insertReorderBarrier(false); drawOrderedNode(&canvas, 0, 10.0f); // in reorder=false at this point, so played inorder drawOrderedRect(&canvas, 1); canvas.insertReorderBarrier(true); drawOrderedNode(&canvas, 6, 2.0f); drawOrderedRect(&canvas, 3); drawOrderedNode(&canvas, 4, 0.0f); drawOrderedRect(&canvas, 5); drawOrderedNode(&canvas, 2, -2.0f); drawOrderedNode(&canvas, 7, 2.0f); canvas.insertReorderBarrier(false); drawOrderedRect(&canvas, 8); drawOrderedNode(&canvas, 9, -10.0f); // in reorder=false at this point, so played inorder canvas.insertReorderBarrier(true); // reorder a node ahead of drawrect op drawOrderedRect(&canvas, 11); drawOrderedNode(&canvas, 10, -1.0f); canvas.insertReorderBarrier(false); canvas.insertReorderBarrier(true); // test with two empty reorder sections canvas.insertReorderBarrier(true); canvas.insertReorderBarrier(false); drawOrderedRect(&canvas, 12); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(13, renderer.getIndex()); }; RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorder) { static const int scrollX = 5; static const int scrollY = 10; class ProjectionReorderTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { const int index = mIndex++; Matrix4 expectedMatrix; switch (index) { case 0: EXPECT_EQ(Rect(100, 100), op.unmappedBounds); EXPECT_EQ(SK_ColorWHITE, op.paint->getColor()); expectedMatrix.loadIdentity(); EXPECT_EQ(nullptr, state.computedState.localProjectionPathMask); break; case 1: EXPECT_EQ(Rect(-10, -10, 60, 60), op.unmappedBounds); EXPECT_EQ(SK_ColorDKGRAY, op.paint->getColor()); expectedMatrix.loadTranslate(50 - scrollX, 50 - scrollY, 0); ASSERT_NE(nullptr, state.computedState.localProjectionPathMask); EXPECT_EQ(Rect(-35, -30, 45, 50), Rect(state.computedState.localProjectionPathMask->getBounds())); break; case 2: EXPECT_EQ(Rect(100, 50), op.unmappedBounds); EXPECT_EQ(SK_ColorBLUE, op.paint->getColor()); expectedMatrix.loadTranslate(-scrollX, 50 - scrollY, 0); EXPECT_EQ(nullptr, state.computedState.localProjectionPathMask); break; default: ADD_FAILURE(); } EXPECT_EQ(expectedMatrix, state.computedState.transform); } }; /** * Construct a tree of nodes, where the root (A) has a receiver background (B), and a child (C) * with a projecting child (P) of its own. P would normally draw between B and C's "background" * draw, but because it is projected backwards, it's drawn in between B and C. * * The parent is scrolled by scrollX/scrollY, but this does not affect the background * (which isn't affected by scroll). */ auto receiverBackground = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& properties, RecordingCanvas& canvas) { properties.setProjectionReceiver(true); // scroll doesn't apply to background, so undone via translationX/Y // NOTE: translationX/Y only! no other transform properties may be set for a proj // receiver! properties.setTranslationX(scrollX); properties.setTranslationY(scrollY); SkPaint paint; paint.setColor(SK_ColorWHITE); canvas.drawRect(0, 0, 100, 100, paint); }); auto projectingRipple = TestUtils::createNode<RecordingCanvas>( 50, 0, 100, 50, [](RenderProperties& properties, RecordingCanvas& canvas) { properties.setProjectBackwards(true); properties.setClipToBounds(false); SkPaint paint; paint.setColor(SK_ColorDKGRAY); canvas.drawRect(-10, -10, 60, 60, paint); }); auto child = TestUtils::createNode<RecordingCanvas>( 0, 50, 100, 100, [&projectingRipple](RenderProperties& properties, RecordingCanvas& canvas) { SkPaint paint; paint.setColor(SK_ColorBLUE); canvas.drawRect(0, 0, 100, 50, paint); canvas.drawRenderNode(projectingRipple.get()); }); auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [&receiverBackground, &child](RenderProperties& properties, RecordingCanvas& canvas) { // Set a rect outline for the projecting ripple to be masked against. properties.mutableOutline().setRoundRect(10, 10, 90, 90, 5, 1.0f); canvas.save(SaveFlags::MatrixClip); canvas.translate(-scrollX, -scrollY); // Apply scroll (note: bg undoes this internally) canvas.drawRenderNode(receiverBackground.get()); canvas.drawRenderNode(child.get()); canvas.restore(); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); ProjectionReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionHwLayer) { static const int scrollX = 5; static const int scrollY = 10; class ProjectionHwLayerTestRenderer : public TestRendererBase { public: void startRepaintLayer(OffscreenBuffer* offscreenBuffer, const Rect& repaintRect) override { EXPECT_EQ(0, mIndex++); } void onArcOp(const ArcOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); ASSERT_EQ(nullptr, state.computedState.localProjectionPathMask); } void endLayer() override { EXPECT_EQ(2, mIndex++); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(3, mIndex++); ASSERT_EQ(nullptr, state.computedState.localProjectionPathMask); } void onOvalOp(const OvalOp& op, const BakedOpState& state) override { EXPECT_EQ(4, mIndex++); ASSERT_NE(nullptr, state.computedState.localProjectionPathMask); Matrix4 expected; expected.loadTranslate(100 - scrollX, 100 - scrollY, 0); EXPECT_EQ(expected, state.computedState.transform); EXPECT_EQ(Rect(-85, -80, 295, 300), Rect(state.computedState.localProjectionPathMask->getBounds())); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { EXPECT_EQ(5, mIndex++); ASSERT_EQ(nullptr, state.computedState.localProjectionPathMask); } }; auto receiverBackground = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 400, [](RenderProperties& properties, RecordingCanvas& canvas) { properties.setProjectionReceiver(true); // scroll doesn't apply to background, so undone via translationX/Y // NOTE: translationX/Y only! no other transform properties may be set for a proj // receiver! properties.setTranslationX(scrollX); properties.setTranslationY(scrollY); canvas.drawRect(0, 0, 400, 400, SkPaint()); }); auto projectingRipple = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& properties, RecordingCanvas& canvas) { properties.setProjectBackwards(true); properties.setClipToBounds(false); canvas.drawOval(100, 100, 300, 300, SkPaint()); // drawn mostly out of layer bounds }); auto child = TestUtils::createNode<RecordingCanvas>( 100, 100, 300, 300, [&projectingRipple](RenderProperties& properties, RecordingCanvas& canvas) { properties.mutateLayerProperties().setType(LayerType::RenderLayer); canvas.drawRenderNode(projectingRipple.get()); canvas.drawArc(0, 0, 200, 200, 0.0f, 280.0f, true, SkPaint()); }); auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 400, [&receiverBackground, &child](RenderProperties& properties, RecordingCanvas& canvas) { // Set a rect outline for the projecting ripple to be masked against. properties.mutableOutline().setRoundRect(10, 10, 390, 390, 0, 1.0f); canvas.translate(-scrollX, -scrollY); // Apply scroll (note: bg undoes this internally) canvas.drawRenderNode(receiverBackground.get()); canvas.drawRenderNode(child.get()); }); OffscreenBuffer** layerHandle = child->getLayerHandle(); // create RenderNode's layer here in same way prepareTree would, setting windowTransform OffscreenBuffer layer(renderThread.renderState(), Caches::getInstance(), 200, 200); Matrix4 windowTransform; windowTransform.loadTranslate(100, 100, 0); // total transform of layer's origin layer.setWindowTransform(windowTransform); *layerHandle = &layer; auto syncedNode = TestUtils::getSyncedNode(parent); LayerUpdateQueue layerUpdateQueue; // Note: enqueue damage post-sync, so bounds are valid layerUpdateQueue.enqueueLayerWithDamage(child.get(), Rect(200, 200)); FrameBuilder frameBuilder(SkRect::MakeWH(400, 400), 400, 400, sLightGeometry, Caches::getInstance()); frameBuilder.deferLayers(layerUpdateQueue); frameBuilder.deferRenderNode(*syncedNode); ProjectionHwLayerTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(6, renderer.getIndex()); // clean up layer pointer, so we can safely destruct RenderNode *layerHandle = nullptr; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionChildScroll) { static const int scrollX = 500000; static const int scrollY = 0; class ProjectionChildScrollTestRenderer : public TestRendererBase { public: void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_TRUE(state.computedState.transform.isIdentity()); } void onOvalOp(const OvalOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); ASSERT_NE(nullptr, state.computedState.clipState); ASSERT_EQ(ClipMode::Rectangle, state.computedState.clipState->mode); ASSERT_EQ(Rect(400, 400), state.computedState.clipState->rect); EXPECT_TRUE(state.computedState.transform.isIdentity()); } }; auto receiverBackground = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 400, [](RenderProperties& properties, RecordingCanvas& canvas) { properties.setProjectionReceiver(true); canvas.drawRect(0, 0, 400, 400, SkPaint()); }); auto projectingRipple = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& properties, RecordingCanvas& canvas) { // scroll doesn't apply to background, so undone via translationX/Y // NOTE: translationX/Y only! no other transform properties may be set for a proj // receiver! properties.setTranslationX(scrollX); properties.setTranslationY(scrollY); properties.setProjectBackwards(true); properties.setClipToBounds(false); canvas.drawOval(0, 0, 200, 200, SkPaint()); }); auto child = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 400, [&projectingRipple](RenderProperties& properties, RecordingCanvas& canvas) { // Record time clip will be ignored by projectee canvas.clipRect(100, 100, 300, 300, SkClipOp::kIntersect); canvas.translate(-scrollX, -scrollY); // Apply scroll (note: bg undoes this internally) canvas.drawRenderNode(projectingRipple.get()); }); auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 400, 400, [&receiverBackground, &child](RenderProperties& properties, RecordingCanvas& canvas) { canvas.drawRenderNode(receiverBackground.get()); canvas.drawRenderNode(child.get()); }); FrameBuilder frameBuilder(SkRect::MakeWH(400, 400), 400, 400, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); ProjectionChildScrollTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } // creates a 100x100 shadow casting node with provided translationZ static sp<RenderNode> createWhiteRectShadowCaster(float translationZ) { return TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [translationZ](RenderProperties& properties, RecordingCanvas& canvas) { properties.setTranslationZ(translationZ); properties.mutableOutline().setRoundRect(0, 0, 100, 100, 0.0f, 1.0f); SkPaint paint; paint.setColor(SK_ColorWHITE); canvas.drawRect(0, 0, 100, 100, paint); }); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, shadow) { class ShadowTestRenderer : public TestRendererBase { public: void onShadowOp(const ShadowOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_FLOAT_EQ(1.0f, op.casterAlpha); EXPECT_TRUE(op.shadowTask->casterPerimeter.isRect(nullptr)); EXPECT_MATRIX_APPROX_EQ(Matrix4::identity(), op.shadowTask->transformXY); Matrix4 expectedZ; expectedZ.loadTranslate(0, 0, 5); EXPECT_MATRIX_APPROX_EQ(expectedZ, op.shadowTask->transformZ); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); } }; auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.insertReorderBarrier(true); canvas.drawRenderNode(createWhiteRectShadowCaster(5.0f).get()); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); ShadowTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, shadowSaveLayer) { class ShadowSaveLayerTestRenderer : public TestRendererBase { public: OffscreenBuffer* startTemporaryLayer(uint32_t width, uint32_t height) override { EXPECT_EQ(0, mIndex++); return nullptr; } void onShadowOp(const ShadowOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); EXPECT_FLOAT_EQ(50, op.shadowTask->lightCenter.x); EXPECT_FLOAT_EQ(40, op.shadowTask->lightCenter.y); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(2, mIndex++); } void endLayer() override { EXPECT_EQ(3, mIndex++); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { EXPECT_EQ(4, mIndex++); } void recycleTemporaryLayer(OffscreenBuffer* offscreenBuffer) override { EXPECT_EQ(5, mIndex++); } }; auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { // save/restore outside of reorderBarrier, so they don't get moved out of place canvas.translate(20, 10); int count = canvas.saveLayerAlpha(30, 50, 130, 150, 128, SaveFlags::ClipToLayer); canvas.insertReorderBarrier(true); canvas.drawRenderNode(createWhiteRectShadowCaster(5.0f).get()); canvas.insertReorderBarrier(false); canvas.restoreToCount(count); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, (FrameBuilder::LightGeometry){{100, 100, 100}, 50}, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); ShadowSaveLayerTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(6, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, shadowHwLayer) { class ShadowHwLayerTestRenderer : public TestRendererBase { public: void startRepaintLayer(OffscreenBuffer* offscreenBuffer, const Rect& repaintRect) override { EXPECT_EQ(0, mIndex++); } void onShadowOp(const ShadowOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); EXPECT_FLOAT_EQ(50, op.shadowTask->lightCenter.x); EXPECT_FLOAT_EQ(40, op.shadowTask->lightCenter.y); EXPECT_FLOAT_EQ(30, op.shadowTask->lightRadius); } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(2, mIndex++); } void endLayer() override { EXPECT_EQ(3, mIndex++); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { EXPECT_EQ(4, mIndex++); } }; auto parent = TestUtils::createNode<RecordingCanvas>( 50, 60, 150, 160, [](RenderProperties& props, RecordingCanvas& canvas) { props.mutateLayerProperties().setType(LayerType::RenderLayer); canvas.insertReorderBarrier(true); canvas.save(SaveFlags::MatrixClip); canvas.translate(20, 10); canvas.drawRenderNode(createWhiteRectShadowCaster(5.0f).get()); canvas.restore(); }); OffscreenBuffer** layerHandle = parent->getLayerHandle(); // create RenderNode's layer here in same way prepareTree would, setting windowTransform OffscreenBuffer layer(renderThread.renderState(), Caches::getInstance(), 100, 100); Matrix4 windowTransform; windowTransform.loadTranslate(50, 60, 0); // total transform of layer's origin layer.setWindowTransform(windowTransform); *layerHandle = &layer; auto syncedNode = TestUtils::getSyncedNode(parent); LayerUpdateQueue layerUpdateQueue; // Note: enqueue damage post-sync, so bounds are valid layerUpdateQueue.enqueueLayerWithDamage(parent.get(), Rect(100, 100)); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, (FrameBuilder::LightGeometry){{100, 100, 100}, 30}, Caches::getInstance()); frameBuilder.deferLayers(layerUpdateQueue); frameBuilder.deferRenderNode(*syncedNode); ShadowHwLayerTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(5, renderer.getIndex()); // clean up layer pointer, so we can safely destruct RenderNode *layerHandle = nullptr; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, shadowLayering) { class ShadowLayeringTestRenderer : public TestRendererBase { public: void onShadowOp(const ShadowOp& op, const BakedOpState& state) override { int index = mIndex++; EXPECT_TRUE(index == 0 || index == 1); } void onRectOp(const RectOp& op, const BakedOpState& state) override { int index = mIndex++; EXPECT_TRUE(index == 2 || index == 3); } }; auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 200, 200, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.insertReorderBarrier(true); canvas.drawRenderNode(createWhiteRectShadowCaster(5.0f).get()); canvas.drawRenderNode(createWhiteRectShadowCaster(5.0001f).get()); }); FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, (FrameBuilder::LightGeometry){{100, 100, 100}, 50}, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); ShadowLayeringTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, shadowClipping) { class ShadowClippingTestRenderer : public TestRendererBase { public: void onShadowOp(const ShadowOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_EQ(Rect(25, 25, 75, 75), state.computedState.clipState->rect) << "Shadow must respect pre-barrier canvas clip value."; } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); } }; auto parent = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { // Apply a clip before the reorder barrier/shadow casting child is drawn. // This clip must be applied to the shadow cast by the child. canvas.clipRect(25, 25, 75, 75, SkClipOp::kIntersect); canvas.insertReorderBarrier(true); canvas.drawRenderNode(createWhiteRectShadowCaster(5.0f).get()); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, (FrameBuilder::LightGeometry){{100, 100, 100}, 50}, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(parent)); ShadowClippingTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } static void testProperty( std::function<void(RenderProperties&)> propSetupCallback, std::function<void(const RectOp&, const BakedOpState&)> opValidateCallback) { class PropertyTestRenderer : public TestRendererBase { public: explicit PropertyTestRenderer( std::function<void(const RectOp&, const BakedOpState&)> callback) : mCallback(callback) {} void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(mIndex++, 0); mCallback(op, state); } std::function<void(const RectOp&, const BakedOpState&)> mCallback; }; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [propSetupCallback](RenderProperties& props, RecordingCanvas& canvas) { propSetupCallback(props); SkPaint paint; paint.setColor(SK_ColorWHITE); canvas.drawRect(0, 0, 100, 100, paint); }); FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); PropertyTestRenderer renderer(opValidateCallback); frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()) << "Should have seen one op"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropOverlappingRenderingAlpha) { testProperty( [](RenderProperties& properties) { properties.setAlpha(0.5f); properties.setHasOverlappingRendering(false); }, [](const RectOp& op, const BakedOpState& state) { EXPECT_EQ(0.5f, state.alpha) << "Alpha should be applied directly to op"; }); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropClipping) { testProperty( [](RenderProperties& properties) { properties.setClipToBounds(true); properties.setClipBounds(Rect(10, 20, 300, 400)); }, [](const RectOp& op, const BakedOpState& state) { EXPECT_EQ(Rect(10, 20, 100, 100), state.computedState.clippedBounds) << "Clip rect should be intersection of node bounds and clip bounds"; }); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropRevealClip) { testProperty( [](RenderProperties& properties) { properties.mutableRevealClip().set(true, 50, 50, 25); }, [](const RectOp& op, const BakedOpState& state) { ASSERT_NE(nullptr, state.roundRectClipState); EXPECT_TRUE(state.roundRectClipState->highPriority); EXPECT_EQ(25, state.roundRectClipState->radius); EXPECT_EQ(Rect(50, 50, 50, 50), state.roundRectClipState->innerRect); }); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropOutlineClip) { testProperty( [](RenderProperties& properties) { properties.mutableOutline().setShouldClip(true); properties.mutableOutline().setRoundRect(10, 20, 30, 40, 5.0f, 0.5f); }, [](const RectOp& op, const BakedOpState& state) { ASSERT_NE(nullptr, state.roundRectClipState); EXPECT_FALSE(state.roundRectClipState->highPriority); EXPECT_EQ(5, state.roundRectClipState->radius); EXPECT_EQ(Rect(15, 25, 25, 35), state.roundRectClipState->innerRect); }); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropTransform) { testProperty( [](RenderProperties& properties) { properties.setLeftTopRightBottom(10, 10, 110, 110); SkMatrix staticMatrix = SkMatrix::MakeScale(1.2f, 1.2f); properties.setStaticMatrix(&staticMatrix); // ignored, since static overrides animation SkMatrix animationMatrix = SkMatrix::MakeTrans(15, 15); properties.setAnimationMatrix(&animationMatrix); properties.setTranslationX(10); properties.setTranslationY(20); properties.setScaleX(0.5f); properties.setScaleY(0.7f); }, [](const RectOp& op, const BakedOpState& state) { Matrix4 matrix; matrix.loadTranslate(10, 10, 0); // left, top matrix.scale(1.2f, 1.2f, 1); // static matrix // ignore animation matrix, since static overrides it // translation xy matrix.translate(10, 20); // scale xy (from default pivot - center) matrix.translate(50, 50); matrix.scale(0.5f, 0.7f, 1); matrix.translate(-50, -50); EXPECT_MATRIX_APPROX_EQ(matrix, state.computedState.transform) << "Op draw matrix must match expected combination of transformation " "properties"; }); } struct SaveLayerAlphaData { uint32_t layerWidth = 0; uint32_t layerHeight = 0; Rect rectClippedBounds; Matrix4 rectMatrix; Matrix4 drawLayerMatrix; }; /** * Constructs a view to hit the temporary layer alpha property implementation: * a) 0 < alpha < 1 * b) too big for layer (larger than maxTextureSize) * c) overlapping rendering content * returning observed data about layer size and content clip/transform. * * Used to validate clipping behavior of temporary layer, where requested layer size is reduced * (for efficiency, and to fit in layer size constraints) based on parent clip. */ void testSaveLayerAlphaClip(SaveLayerAlphaData* outObservedData, std::function<void(RenderProperties&)> propSetupCallback) { class SaveLayerAlphaClipTestRenderer : public TestRendererBase { public: explicit SaveLayerAlphaClipTestRenderer(SaveLayerAlphaData* outData) : mOutData(outData) {} OffscreenBuffer* startTemporaryLayer(uint32_t width, uint32_t height) override { EXPECT_EQ(0, mIndex++); mOutData->layerWidth = width; mOutData->layerHeight = height; return nullptr; } void onRectOp(const RectOp& op, const BakedOpState& state) override { EXPECT_EQ(1, mIndex++); mOutData->rectClippedBounds = state.computedState.clippedBounds; mOutData->rectMatrix = state.computedState.transform; } void endLayer() override { EXPECT_EQ(2, mIndex++); } void onLayerOp(const LayerOp& op, const BakedOpState& state) override { EXPECT_EQ(3, mIndex++); mOutData->drawLayerMatrix = state.computedState.transform; } void recycleTemporaryLayer(OffscreenBuffer* offscreenBuffer) override { EXPECT_EQ(4, mIndex++); } private: SaveLayerAlphaData* mOutData; }; ASSERT_GT(10000, DeviceInfo::get()->maxTextureSize()) << "Node must be bigger than max texture size to exercise saveLayer codepath"; auto node = TestUtils::createNode<RecordingCanvas>( 0, 0, 10000, 10000, [&propSetupCallback](RenderProperties& properties, RecordingCanvas& canvas) { properties.setHasOverlappingRendering(true); properties.setAlpha(0.5f); // force saveLayer, since too big for HW layer // apply other properties propSetupCallback(properties); SkPaint paint; paint.setColor(SK_ColorWHITE); canvas.drawRect(0, 0, 10000, 10000, paint); }); auto syncedNode = TestUtils::getSyncedNode(node); // sync before querying height FrameBuilder frameBuilder(SkRect::MakeWH(200, 200), 200, 200, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*syncedNode); SaveLayerAlphaClipTestRenderer renderer(outObservedData); frameBuilder.replayBakedOps<TestDispatcher>(renderer); // assert, since output won't be valid if we haven't seen a save layer triggered ASSERT_EQ(5, renderer.getIndex()) << "Test must trigger saveLayer alpha behavior."; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropSaveLayerAlphaClipBig) { SaveLayerAlphaData observedData; testSaveLayerAlphaClip(&observedData, [](RenderProperties& properties) { properties.setTranslationX(10); // offset rendering content properties.setTranslationY(-2000); // offset rendering content }); EXPECT_EQ(190u, observedData.layerWidth); EXPECT_EQ(200u, observedData.layerHeight); EXPECT_EQ(Rect(190, 200), observedData.rectClippedBounds) << "expect content to be clipped to screen area"; Matrix4 expected; expected.loadTranslate(0, -2000, 0); EXPECT_MATRIX_APPROX_EQ(expected, observedData.rectMatrix) << "expect content to be translated as part of being clipped"; expected.loadTranslate(10, 0, 0); EXPECT_MATRIX_APPROX_EQ(expected, observedData.drawLayerMatrix) << "expect drawLayer to be translated as part of being clipped"; } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropSaveLayerAlphaRotate) { SaveLayerAlphaData observedData; testSaveLayerAlphaClip(&observedData, [](RenderProperties& properties) { // Translate and rotate the view so that the only visible part is the top left corner of // the view. It will form an isosceles right triangle with a long side length of 200 at the // bottom of the viewport. properties.setTranslationX(100); properties.setTranslationY(100); properties.setPivotX(0); properties.setPivotY(0); properties.setRotation(45); }); // ceil(sqrt(2) / 2 * 200) = 142 EXPECT_EQ(142u, observedData.layerWidth); EXPECT_EQ(142u, observedData.layerHeight); EXPECT_EQ(Rect(142, 142), observedData.rectClippedBounds); EXPECT_MATRIX_APPROX_EQ(Matrix4::identity(), observedData.rectMatrix); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, renderPropSaveLayerAlphaScale) { SaveLayerAlphaData observedData; testSaveLayerAlphaClip(&observedData, [](RenderProperties& properties) { properties.setPivotX(0); properties.setPivotY(0); properties.setScaleX(2); properties.setScaleY(0.5f); }); EXPECT_EQ(100u, observedData.layerWidth); EXPECT_EQ(400u, observedData.layerHeight); EXPECT_EQ(Rect(100, 400), observedData.rectClippedBounds); EXPECT_MATRIX_APPROX_EQ(Matrix4::identity(), observedData.rectMatrix); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, clip_replace) { class ClipReplaceTestRenderer : public TestRendererBase { public: void onColorOp(const ColorOp& op, const BakedOpState& state) override { EXPECT_EQ(0, mIndex++); EXPECT_TRUE(op.localClip->intersectWithRoot); EXPECT_EQ(Rect(20, 10, 30, 40), state.computedState.clipState->rect) << "Expect resolved clip to be intersection of viewport clip and clip op"; } }; auto node = TestUtils::createNode<RecordingCanvas>( 20, 20, 30, 30, [](RenderProperties& props, RecordingCanvas& canvas) { canvas.clipRect(0, -20, 10, 30, SkClipOp::kReplace_deprecated); canvas.drawColor(SK_ColorWHITE, SkBlendMode::kSrcOver); }); FrameBuilder frameBuilder(SkRect::MakeLTRB(10, 10, 40, 40), 50, 50, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(node)); ClipReplaceTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(1, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderProjectedInMiddle) { /* R is backward projected on B A / \ B C | R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectionReceiver(true); }); // nodeB drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeC }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderProjectLast) { /* R is backward projected on E A / | \ / | \ B C E | R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, nullptr); // nodeB drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 3, [](RenderProperties& props, RecordingCanvas& canvas) { // drawn as 2 props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeC drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { // drawn as 3 props.setProjectionReceiver(true); }); // nodeE }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderNoReceivable) { /* R is backward projected without receiver A / \ B C | R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, nullptr); // nodeB drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 255, [](RenderProperties& props, RecordingCanvas& canvas) { // not having a projection receiver is an undefined behavior props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeC }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderParentReceivable) { /* R is backward projected on C A / \ B C | R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, nullptr); // nodeB drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectionReceiver(true); drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeC }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderSameNodeReceivable) { auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, nullptr); // nodeB drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 255, [](RenderProperties& props, RecordingCanvas& canvas) { // having a node that is projected on itself is an // undefined/unexpected behavior props.setProjectionReceiver(true); props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeC }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(2, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderProjectedSibling) { // TODO: this test together with the next "projectionReorderProjectedSibling2" likely expose a // bug in HWUI. First test draws R, while the second test does not draw R for a nearly identical // tree setup. The correct behaviour is to not draw R, because the receiver cannot be a sibling /* R is backward projected on B. R is not expected to be drawn (see Sibling2 outcome below), but for some reason it is drawn. A /|\ / | \ B C R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectionReceiver(true); }); // nodeB drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) {}); // nodeC drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderProjectedSibling2) { /* R is set to project on B, but R is not drawn because projecting on a sibling is not allowed. A | G /|\ / | \ B C R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, [](RenderProperties& props, RecordingCanvas& canvas) { // G drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { // B props.setProjectionReceiver(true); }); // nodeB drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { // C }); // nodeC drawOrderedNode(&canvas, 255, [](RenderProperties& props, RecordingCanvas& canvas) { // R props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeG }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderGrandparentReceivable) { /* R is backward projected on B A | B | C | R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectionReceiver(true); drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeC }); // nodeB }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(3, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderTwoReceivables) { /* B and G are receivables, R is backward projected A / \ B C / \ G R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, [](RenderProperties& props, RecordingCanvas& canvas) { // B props.setProjectionReceiver(true); }); // nodeB drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { // C drawOrderedNode(&canvas, 3, [](RenderProperties& props, RecordingCanvas& canvas) { // G props.setProjectionReceiver(true); }); // nodeG drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { // R props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeC }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderTwoReceivablesLikelyScenario) { /* B and G are receivables, G is backward projected A / \ B C / \ G R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, [](RenderProperties& props, RecordingCanvas& canvas) { // B props.setProjectionReceiver(true); }); // nodeB drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { // C drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { // G props.setProjectionReceiver(true); props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeG drawOrderedNode(&canvas, 3, [](RenderProperties& props, RecordingCanvas& canvas) { // R }); // nodeR }); // nodeC }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(4, renderer.getIndex()); } RENDERTHREAD_OPENGL_PIPELINE_TEST(FrameBuilder, projectionReorderTwoReceivablesDeeper) { /* B and G are receivables, R is backward projected A / \ B C / \ G D | R */ auto nodeA = TestUtils::createNode<RecordingCanvas>( 0, 0, 100, 100, [](RenderProperties& props, RecordingCanvas& canvas) { drawOrderedNode(&canvas, 0, [](RenderProperties& props, RecordingCanvas& canvas) { // B props.setProjectionReceiver(true); }); // nodeB drawOrderedNode(&canvas, 1, [](RenderProperties& props, RecordingCanvas& canvas) { // C drawOrderedNode(&canvas, 2, [](RenderProperties& props, RecordingCanvas& canvas) { // G props.setProjectionReceiver(true); }); // nodeG drawOrderedNode( &canvas, 4, [](RenderProperties& props, RecordingCanvas& canvas) { // D drawOrderedNode(&canvas, 3, [](RenderProperties& props, RecordingCanvas& canvas) { // R props.setProjectBackwards(true); props.setClipToBounds(false); }); // nodeR }); // nodeD }); // nodeC }); // nodeA FrameBuilder frameBuilder(SkRect::MakeWH(100, 100), 100, 100, sLightGeometry, Caches::getInstance()); frameBuilder.deferRenderNode(*TestUtils::getSyncedNode(nodeA)); ZReorderTestRenderer renderer; frameBuilder.replayBakedOps<TestDispatcher>(renderer); EXPECT_EQ(5, renderer.getIndex()); } } // namespace uirenderer } // namespace android