TestUtils.h revision 2de950d5a8b47c7b4648ada1b1260ce4b7342798
1b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik/*
2b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * Copyright (C) 2015 The Android Open Source Project
3b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *
4b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * Licensed under the Apache License, Version 2.0 (the "License");
5b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * you may not use this file except in compliance with the License.
6b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * You may obtain a copy of the License at
7b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *
8b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *      http://www.apache.org/licenses/LICENSE-2.0
9b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik *
10b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * Unless required by applicable law or agreed to in writing, software
11b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * distributed under the License is distributed on an "AS IS" BASIS,
12b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * See the License for the specific language governing permissions and
14b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik * limitations under the License.
15b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik */
165e00c7ce063116c11315639f0035aca8ad73e8ccChris Craik
175e00c7ce063116c11315639f0035aca8ad73e8ccChris Craik#pragma once
18b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
1976caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik#include <DeviceInfo.h>
20161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik#include <DisplayList.h>
21b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include <Matrix.h>
2298c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel#include <Properties.h>
230a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <Rect.h>
24b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include <RenderNode.h>
25c1c54062f8cc9d47bdea820ae5ab6aef260b4488sergeyv#include <hwui/Bitmap.h>
26500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev#include <pipeline/skia/SkiaRecordingCanvas.h>
270a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <renderstate/RenderState.h>
280a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <renderthread/RenderThread.h>
290a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <Snapshot.h>
30b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
31161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik#include <RecordedOp.h>
3216c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck#include <RecordingCanvas.h>
33161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik
34b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include <memory>
35b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
36b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craiknamespace android {
37b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craiknamespace uirenderer {
38b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
39b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#define EXPECT_MATRIX_APPROX_EQ(a, b) \
40b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    EXPECT_TRUE(TestUtils::matricesAreApproxEqual(a, b))
41b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
426fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik#define EXPECT_RECT_APPROX_EQ(a, b) \
43474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh    EXPECT_TRUE(MathUtils::areEqual((a).left, (b).left) \
44474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh            && MathUtils::areEqual((a).top, (b).top) \
45474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh            && MathUtils::areEqual((a).right, (b).right) \
46474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh            && MathUtils::areEqual((a).bottom, (b).bottom));
476fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik
487435eb148e72382126e9073183e881357bb38a8bChris Craik#define EXPECT_CLIP_RECT(expRect, clipStatePtr) \
497435eb148e72382126e9073183e881357bb38a8bChris Craik        EXPECT_NE(nullptr, (clipStatePtr)) << "Op is unclipped"; \
507435eb148e72382126e9073183e881357bb38a8bChris Craik        if ((clipStatePtr)->mode == ClipMode::Rectangle) { \
517435eb148e72382126e9073183e881357bb38a8bChris Craik            EXPECT_EQ((expRect), reinterpret_cast<const ClipRect*>(clipStatePtr)->rect); \
527435eb148e72382126e9073183e881357bb38a8bChris Craik        } else { \
537435eb148e72382126e9073183e881357bb38a8bChris Craik            ADD_FAILURE() << "ClipState not a rect"; \
547435eb148e72382126e9073183e881357bb38a8bChris Craik        }
5598c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel
5698c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel#define INNER_PIPELINE_TEST(test_case_name, test_name, pipeline, functionCall) \
5798c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    TEST(test_case_name, test_name##_##pipeline) { \
5898c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel        RenderPipelineType oldType = Properties::getRenderPipelineType(); \
5998c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel        Properties::overrideRenderPipelineType(RenderPipelineType::pipeline); \
6098c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel        functionCall; \
6198c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel        Properties::overrideRenderPipelineType(oldType); \
6298c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    };
6398c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel
6498c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel/**
6598c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel * Like gtests' TEST, but only runs with the OpenGL RenderPipelineType
6698c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel */
6798c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel#define OPENGL_PIPELINE_TEST(test_case_name, test_name) \
6898c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    class test_case_name##_##test_name##_HwuiTest { \
6998c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    public: \
7098c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel        static void doTheThing(); \
7198c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    }; \
7298c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_TEST(test_case_name, test_name, OpenGL, \
7398c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel            test_case_name##_##test_name##_HwuiTest::doTheThing()) \
7498c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    void test_case_name##_##test_name##_HwuiTest::doTheThing()
7598c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel
7698c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel#define INNER_PIPELINE_RENDERTHREAD_TEST(test_case_name, test_name, pipeline) \
7798c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_TEST(test_case_name, test_name, pipeline, \
7898c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel            TestUtils::runOnRenderThread(test_case_name##_##test_name##_RenderThreadTest::doTheThing))
7998c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel
8098787e6c9b2c10b1ab7820bdac168686025b924aChris Craik/**
8198787e6c9b2c10b1ab7820bdac168686025b924aChris Craik * Like gtest's TEST, but runs on the RenderThread, and 'renderThread' is passed, in top level scope
8298787e6c9b2c10b1ab7820bdac168686025b924aChris Craik * (for e.g. accessing its RenderState)
8398787e6c9b2c10b1ab7820bdac168686025b924aChris Craik */
8498787e6c9b2c10b1ab7820bdac168686025b924aChris Craik#define RENDERTHREAD_TEST(test_case_name, test_name) \
8598787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    class test_case_name##_##test_name##_RenderThreadTest { \
8698787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    public: \
8798787e6c9b2c10b1ab7820bdac168686025b924aChris Craik        static void doTheThing(renderthread::RenderThread& renderThread); \
8898787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    }; \
8998c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_RENDERTHREAD_TEST(test_case_name, test_name, OpenGL); \
9098c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_RENDERTHREAD_TEST(test_case_name, test_name, SkiaGL); \
9198c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_RENDERTHREAD_TEST(test_case_name, test_name, SkiaVulkan); \
9298c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    void test_case_name##_##test_name##_RenderThreadTest::doTheThing(renderthread::RenderThread& renderThread)
9398c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel
9498c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel/**
9598c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel * Like RENDERTHREAD_TEST, but only runs with the OpenGL RenderPipelineType
9698c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel */
9798c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel#define RENDERTHREAD_OPENGL_PIPELINE_TEST(test_case_name, test_name) \
9898c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    class test_case_name##_##test_name##_RenderThreadTest { \
9998c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    public: \
10098c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel        static void doTheThing(renderthread::RenderThread& renderThread); \
10198c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    }; \
10298c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_RENDERTHREAD_TEST(test_case_name, test_name, OpenGL); \
10398c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    void test_case_name##_##test_name##_RenderThreadTest::doTheThing(renderthread::RenderThread& renderThread)
10498c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel
10598c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel/**
10698c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel * Like RENDERTHREAD_TEST, but only runs with the Skia RenderPipelineTypes
10798c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel */
10898c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel#define RENDERTHREAD_SKIA_PIPELINE_TEST(test_case_name, test_name) \
10998c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    class test_case_name##_##test_name##_RenderThreadTest { \
11098c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    public: \
11198c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel        static void doTheThing(renderthread::RenderThread& renderThread); \
11298787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    }; \
11398c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_RENDERTHREAD_TEST(test_case_name, test_name, SkiaGL); \
11498c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    INNER_PIPELINE_RENDERTHREAD_TEST(test_case_name, test_name, SkiaVulkan); \
11598787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    void test_case_name##_##test_name##_RenderThreadTest::doTheThing(renderthread::RenderThread& renderThread)
11698787e6c9b2c10b1ab7820bdac168686025b924aChris Craik
11737413289478a965336239c731ebfea37ac4dde28Chris Craik/**
11837413289478a965336239c731ebfea37ac4dde28Chris Craik * Sets a property value temporarily, generally for the duration of a test, restoring the previous
11937413289478a965336239c731ebfea37ac4dde28Chris Craik * value when going out of scope.
12037413289478a965336239c731ebfea37ac4dde28Chris Craik *
12137413289478a965336239c731ebfea37ac4dde28Chris Craik * Can be used e.g. to test behavior only active while Properties::debugOverdraw is enabled.
12237413289478a965336239c731ebfea37ac4dde28Chris Craik */
12337413289478a965336239c731ebfea37ac4dde28Chris Craiktemplate <typename T>
12437413289478a965336239c731ebfea37ac4dde28Chris Craikclass ScopedProperty {
12537413289478a965336239c731ebfea37ac4dde28Chris Craikpublic:
12637413289478a965336239c731ebfea37ac4dde28Chris Craik    ScopedProperty(T& property, T newValue)
12737413289478a965336239c731ebfea37ac4dde28Chris Craik        : mPropertyPtr(&property)
12837413289478a965336239c731ebfea37ac4dde28Chris Craik        , mOldValue(property) {
12937413289478a965336239c731ebfea37ac4dde28Chris Craik        property = newValue;
13037413289478a965336239c731ebfea37ac4dde28Chris Craik    }
13137413289478a965336239c731ebfea37ac4dde28Chris Craik    ~ScopedProperty() {
13237413289478a965336239c731ebfea37ac4dde28Chris Craik        *mPropertyPtr = mOldValue;
13337413289478a965336239c731ebfea37ac4dde28Chris Craik    }
13437413289478a965336239c731ebfea37ac4dde28Chris Craikprivate:
13537413289478a965336239c731ebfea37ac4dde28Chris Craik    T* mPropertyPtr;
13637413289478a965336239c731ebfea37ac4dde28Chris Craik    T mOldValue;
13737413289478a965336239c731ebfea37ac4dde28Chris Craik};
13837413289478a965336239c731ebfea37ac4dde28Chris Craik
139b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikclass TestUtils {
140b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikpublic:
14176ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    class SignalingDtor {
14276ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    public:
14376ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        SignalingDtor()
14476ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik                : mSignal(nullptr) {}
145a619ec70cf765d9166f0862e74653711b87307b3Chih-Hung Hsieh        explicit SignalingDtor(int* signal)
14676ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik                : mSignal(signal) {}
14776ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        void setSignal(int* signal) {
14876ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik            mSignal = signal;
14976ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        }
15076ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        ~SignalingDtor() {
15176ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik            if (mSignal) {
15276ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik                (*mSignal)++;
15376ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik            }
15476ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        }
15576ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    private:
15676ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        int* mSignal;
15776ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    };
15876ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik
1592de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck    class MockTreeObserver : public TreeObserver {
1602de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck    public:
1612de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck        virtual void onMaybeRemovedFromTree(RenderNode* node) {}
1622de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck    };
1632de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck
164b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    static bool matricesAreApproxEqual(const Matrix4& a, const Matrix4& b) {
165b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        for (int i = 0; i < 16; i++) {
166b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik            if (!MathUtils::areEqual(a[i], b[i])) {
167b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik                return false;
168b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik            }
169b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        }
170b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        return true;
171b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
172b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
173b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    static std::unique_ptr<Snapshot> makeSnapshot(const Matrix4& transform, const Rect& clip) {
174b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        std::unique_ptr<Snapshot> snapshot(new Snapshot());
1756e49c9f007c879f05b035c40c0ba543c00f9d0d0Mike Reed        // store clip first, so it isn't transformed
1766e49c9f007c879f05b035c40c0ba543c00f9d0d0Mike Reed        snapshot->setClip(clip.left, clip.top, clip.right, clip.bottom);
177b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        *(snapshot->transform) = transform;
178b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        return snapshot;
179b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
180b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
181aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    static sk_sp<Bitmap> createBitmap(int width, int height,
182aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv            SkColorType colorType = kN32_SkColorType) {
183aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        SkImageInfo info = SkImageInfo::Make(width, height, colorType, kPremul_SkAlphaType);
184fc9999505a36c66892d7ccce85187936105f4f36sergeyv        return Bitmap::allocateHeapBitmap(info);
185aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    }
186aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv
187aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    static sk_sp<Bitmap> createBitmap(int width, int height, SkBitmap* outBitmap) {
188aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        SkImageInfo info = SkImageInfo::Make(width, height, kN32_SkColorType, kPremul_SkAlphaType);
189aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        outBitmap->setInfo(info);
190aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        return Bitmap::allocateHeapBitmap(outBitmap, nullptr);
191aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    }
192aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv
193d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik    static sp<DeferredLayerUpdater> createTextureLayerUpdater(
19498c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel            renderthread::RenderThread& renderThread);
19598c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel
19698c78dad1969e2321cfee2085faa55d95bba7e29Greg Daniel    static sp<DeferredLayerUpdater> createTextureLayerUpdater(
197d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik            renderthread::RenderThread& renderThread, uint32_t width, uint32_t height,
198243e85b2e443def1ef47a180e824b36f513c8db8Chris Craik            const SkMatrix& transform);
199d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik
200b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    template<class CanvasType>
201003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik    static std::unique_ptr<DisplayList> createDisplayList(int width, int height,
202b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik            std::function<void(CanvasType& canvas)> canvasCallback) {
203b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        CanvasType canvas(width, height);
204b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        canvasCallback(canvas);
205003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik        return std::unique_ptr<DisplayList>(canvas.finishRecording());
206b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
207b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
208d3daa3198e2212c985c634821682d5819346b653Chris Craik    static sp<RenderNode> createNode(int left, int top, int right, int bottom,
20906152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::function<void(RenderProperties& props, Canvas& canvas)> setup) {
21006152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev#if HWUI_NULL_GPU
21106152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        // if RenderNodes are being sync'd/used, device info will be needed, since
21206152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        // DeviceInfo::maxTextureSize() affects layer property
21306152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        DeviceInfo::initialize();
21406152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev#endif
21506152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev
21606152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        sp<RenderNode> node = new RenderNode();
21706152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        RenderProperties& props = node->mutateStagingProperties();
21806152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        props.setLeftTopRightBottom(left, top, right, bottom);
21906152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        if (setup) {
22006152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::unique_ptr<Canvas> canvas(Canvas::create_recording_canvas(props.getWidth(),
22106152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev                    props.getHeight()));
22206152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            setup(props, *canvas.get());
2232de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck            node->setStagingDisplayList(canvas->finishRecording());
22406152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        }
22506152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        node->setPropertyFieldsDirty(0xFFFFFFFF);
22606152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        return node;
22706152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev    }
22806152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev
22906152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev    template<class RecordingCanvasType>
23006152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev    static sp<RenderNode> createNode(int left, int top, int right, int bottom,
23106152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::function<void(RenderProperties& props, RecordingCanvasType& canvas)> setup) {
2329fded232a9548a304e0145011df8849fba0dcda7Chris Craik#if HWUI_NULL_GPU
23376caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik        // if RenderNodes are being sync'd/used, device info will be needed, since
23476caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik        // DeviceInfo::maxTextureSize() affects layer property
23576caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik        DeviceInfo::initialize();
2369fded232a9548a304e0145011df8849fba0dcda7Chris Craik#endif
23776caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik
238b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        sp<RenderNode> node = new RenderNode();
23916c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        RenderProperties& props = node->mutateStagingProperties();
24016c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        props.setLeftTopRightBottom(left, top, right, bottom);
24116c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        if (setup) {
24206152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            RecordingCanvasType canvas(props.getWidth(), props.getHeight());
24316c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck            setup(props, canvas);
2442de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck            node->setStagingDisplayList(canvas.finishRecording());
2450b7e8245db728d127ada698be63d78b33fc6e4daChris Craik        }
24616c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        node->setPropertyFieldsDirty(0xFFFFFFFF);
2470b7e8245db728d127ada698be63d78b33fc6e4daChris Craik        return node;
2480b7e8245db728d127ada698be63d78b33fc6e4daChris Craik    }
2490b7e8245db728d127ada698be63d78b33fc6e4daChris Craik
25016c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck    static void recordNode(RenderNode& node,
25106152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::function<void(Canvas&)> contentCallback) {
25206152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev       std::unique_ptr<Canvas> canvas(Canvas::create_recording_canvas(
25306152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            node.stagingProperties().getWidth(), node.stagingProperties().getHeight()));
25406152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev       contentCallback(*canvas.get());
2552de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck       node.setStagingDisplayList(canvas->finishRecording());
256b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
257b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
258500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev    static sp<RenderNode> createSkiaNode(int left, int top, int right, int bottom,
259500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev            std::function<void(RenderProperties& props, skiapipeline::SkiaRecordingCanvas& canvas)> setup,
260500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev            const char* name = nullptr, skiapipeline::SkiaDisplayList* displayList = nullptr) {
261500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev    #if HWUI_NULL_GPU
262500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        // if RenderNodes are being sync'd/used, device info will be needed, since
263500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        // DeviceInfo::maxTextureSize() affects layer property
264500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        DeviceInfo::initialize();
265500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev    #endif
266500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        sp<RenderNode> node = new RenderNode();
267500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        if (name) {
268500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev            node->setName(name);
269500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        }
270500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        RenderProperties& props = node->mutateStagingProperties();
271500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        props.setLeftTopRightBottom(left, top, right, bottom);
272500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        if (displayList) {
2732de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck            node->setStagingDisplayList(displayList);
274500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        }
275500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        if (setup) {
276500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev            std::unique_ptr<skiapipeline::SkiaRecordingCanvas> canvas(
277500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev                new skiapipeline::SkiaRecordingCanvas(nullptr,
278500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev                props.getWidth(), props.getHeight()));
279500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev            setup(props, *canvas.get());
2802de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck            node->setStagingDisplayList(canvas->finishRecording());
281500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        }
282500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        node->setPropertyFieldsDirty(0xFFFFFFFF);
283500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        TestUtils::syncHierarchyPropertiesAndDisplayList(node);
284500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev        return node;
285500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev    }
286500a0c30d4dcd012218c3e44a62926a1c34a259fStan Iliev
2878d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik    /**
2888d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * Forces a sync of a tree of RenderNode, such that every descendant will have its staging
2898d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * properties and DisplayList moved to the render copies.
2908d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     *
2918d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * Note: does not check dirtiness bits, so any non-staging DisplayLists will be discarded.
2928d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * For this reason, this should generally only be called once on a tree.
2938d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     */
294161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik    static void syncHierarchyPropertiesAndDisplayList(sp<RenderNode>& node) {
295161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        syncHierarchyPropertiesAndDisplayListImpl(node.get());
296b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
2970a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik
2989cd1bbe5c9e14472e631d8cc10005613925f34afChris Craik    static sp<RenderNode>& getSyncedNode(sp<RenderNode>& node) {
2999cd1bbe5c9e14472e631d8cc10005613925f34afChris Craik        syncHierarchyPropertiesAndDisplayList(node);
3009cd1bbe5c9e14472e631d8cc10005613925f34afChris Craik        return node;
3017db5ffb7dbd30202468459e2ef4426e91d4fcbb3John Reck    }
3027db5ffb7dbd30202468459e2ef4426e91d4fcbb3John Reck
3030b7e8245db728d127ada698be63d78b33fc6e4daChris Craik    typedef std::function<void(renderthread::RenderThread& thread)> RtCallback;
3040a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik
3050a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    class TestTask : public renderthread::RenderTask {
3060a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    public:
307a619ec70cf765d9166f0862e74653711b87307b3Chih-Hung Hsieh        explicit TestTask(RtCallback rtCallback)
3080a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik                : rtCallback(rtCallback) {}
3090a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        virtual ~TestTask() {}
310e5da4ef971258193cd1e89737a12b95b6ac244bbJohn Reck        virtual void run() override;
3110a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        RtCallback rtCallback;
3120a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    };
3130a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik
3140a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    /**
3150a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik     * NOTE: requires surfaceflinger to run, otherwise this method will wait indefinitely.
3160a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik     */
3170a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    static void runOnRenderThread(RtCallback rtCallback) {
3180a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        TestTask task(rtCallback);
3190a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        renderthread::RenderThread::getInstance().queueAndWait(&task);
3200a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    }
32116c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck
32238e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck    static bool isRenderThreadRunning() {
32338e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck        return renderthread::RenderThread::hasInstance();
32438e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck    }
32538e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck
32616c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck    static SkColor interpolateColor(float fraction, SkColor start, SkColor end);
32716c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck
328e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik    static void layoutTextUnscaled(const SkPaint& paint, const char* text,
329e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik            std::vector<glyph_t>* outGlyphs, std::vector<float>* outPositions,
330e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik            float* outTotalAdvance, Rect* outBounds);
331e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik
33279abbf22d4f672208327546661e694d837f564a9Derek Sollenberger    static void drawUtf8ToCanvas(Canvas* canvas, const char* text,
33342a5407f2c6403ea7aa7a64eaf19948dc4050df5Chris Craik            const SkPaint& paint, float x, float y);
334a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik
33579abbf22d4f672208327546661e694d837f564a9Derek Sollenberger    static void drawUtf8ToCanvas(Canvas* canvas, const char* text,
336d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik            const SkPaint& paint, const SkPath& path);
337d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik
33879abbf22d4f672208327546661e694d837f564a9Derek Sollenberger    static std::unique_ptr<uint16_t[]> asciiToUtf16(const char* str);
339dccca44ffda4836b56a21da95a046c9708ffd49csergeyv
340835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger    class MockFunctor : public Functor {
341835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger     public:
342835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger         virtual status_t operator ()(int what, void* data) {
343835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger             mLastMode = what;
344835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger             return DrawGlInfo::kStatusDone;
345835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger         }
346835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger         int getLastMode() const { return mLastMode; }
347835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger     private:
348835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger         int mLastMode = -1;
349835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger     };
350835b3a69c345d2b58a0774daeb2e717a8a878059Derek Sollenberger
351021693b967a2c5556dddd183eb0247df4079e1adStan Iliev    static SkColor getColor(const sk_sp<SkSurface>& surface, int x, int y);
352021693b967a2c5556dddd183eb0247df4079e1adStan Iliev
35352771272f4f018f4fc6846224bf047497e784af1Stan Iliev    static SkRect getClipBounds(const SkCanvas* canvas);
35452771272f4f018f4fc6846224bf047497e784af1Stan Iliev    static SkRect getLocalClipBounds(const SkCanvas* canvas);
35552771272f4f018f4fc6846224bf047497e784af1Stan Iliev
356161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craikprivate:
357161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik    static void syncHierarchyPropertiesAndDisplayListImpl(RenderNode* node) {
3582de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck        MarkAndSweepRemoved observer(nullptr);
359161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        node->syncProperties();
3602de950d5a8b47c7b4648ada1b1260ce4b7342798John Reck        node->syncDisplayList(observer, nullptr);
361161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        auto displayList = node->getDisplayList();
362161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        if (displayList) {
363161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik            for (auto&& childOp : displayList->getChildren()) {
364161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik                syncHierarchyPropertiesAndDisplayListImpl(childOp->renderNode);
365161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik            }
366161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        }
367161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik    }
368161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik
369b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik}; // class TestUtils
370b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
371b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} /* namespace uirenderer */
372b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} /* namespace android */
373