TestUtils.h revision fc9999505a36c66892d7ccce85187936105f4f36
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>
220a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <Rect.h>
23b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include <RenderNode.h>
24c1c54062f8cc9d47bdea820ae5ab6aef260b4488sergeyv#include <hwui/Bitmap.h>
250a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <renderstate/RenderState.h>
260a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <renderthread/RenderThread.h>
270a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik#include <Snapshot.h>
28b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
29161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik#include <RecordedOp.h>
3016c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck#include <RecordingCanvas.h>
31161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik
32b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#include <memory>
33b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
34b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craiknamespace android {
35b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craiknamespace uirenderer {
36b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
37b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik#define EXPECT_MATRIX_APPROX_EQ(a, b) \
38b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    EXPECT_TRUE(TestUtils::matricesAreApproxEqual(a, b))
39b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
406fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik#define EXPECT_RECT_APPROX_EQ(a, b) \
41474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh    EXPECT_TRUE(MathUtils::areEqual((a).left, (b).left) \
42474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh            && MathUtils::areEqual((a).top, (b).top) \
43474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh            && MathUtils::areEqual((a).right, (b).right) \
44474081eee117025d343372f6cad99562914bc387Chih-Hung Hsieh            && MathUtils::areEqual((a).bottom, (b).bottom));
456fe991e5e76f9af9dab960100d5768d96d5f4daaChris Craik
467435eb148e72382126e9073183e881357bb38a8bChris Craik#define EXPECT_CLIP_RECT(expRect, clipStatePtr) \
477435eb148e72382126e9073183e881357bb38a8bChris Craik        EXPECT_NE(nullptr, (clipStatePtr)) << "Op is unclipped"; \
487435eb148e72382126e9073183e881357bb38a8bChris Craik        if ((clipStatePtr)->mode == ClipMode::Rectangle) { \
497435eb148e72382126e9073183e881357bb38a8bChris Craik            EXPECT_EQ((expRect), reinterpret_cast<const ClipRect*>(clipStatePtr)->rect); \
507435eb148e72382126e9073183e881357bb38a8bChris Craik        } else { \
517435eb148e72382126e9073183e881357bb38a8bChris Craik            ADD_FAILURE() << "ClipState not a rect"; \
527435eb148e72382126e9073183e881357bb38a8bChris Craik        }
5398787e6c9b2c10b1ab7820bdac168686025b924aChris Craik/**
5498787e6c9b2c10b1ab7820bdac168686025b924aChris Craik * Like gtest's TEST, but runs on the RenderThread, and 'renderThread' is passed, in top level scope
5598787e6c9b2c10b1ab7820bdac168686025b924aChris Craik * (for e.g. accessing its RenderState)
5698787e6c9b2c10b1ab7820bdac168686025b924aChris Craik */
5798787e6c9b2c10b1ab7820bdac168686025b924aChris Craik#define RENDERTHREAD_TEST(test_case_name, test_name) \
5898787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    class test_case_name##_##test_name##_RenderThreadTest { \
5998787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    public: \
6098787e6c9b2c10b1ab7820bdac168686025b924aChris Craik        static void doTheThing(renderthread::RenderThread& renderThread); \
6198787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    }; \
6298787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    TEST(test_case_name, test_name) { \
6398787e6c9b2c10b1ab7820bdac168686025b924aChris Craik        TestUtils::runOnRenderThread(test_case_name##_##test_name##_RenderThreadTest::doTheThing); \
6498787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    }; \
6598787e6c9b2c10b1ab7820bdac168686025b924aChris Craik    void test_case_name##_##test_name##_RenderThreadTest::doTheThing(renderthread::RenderThread& renderThread)
6698787e6c9b2c10b1ab7820bdac168686025b924aChris Craik
6737413289478a965336239c731ebfea37ac4dde28Chris Craik/**
6837413289478a965336239c731ebfea37ac4dde28Chris Craik * Sets a property value temporarily, generally for the duration of a test, restoring the previous
6937413289478a965336239c731ebfea37ac4dde28Chris Craik * value when going out of scope.
7037413289478a965336239c731ebfea37ac4dde28Chris Craik *
7137413289478a965336239c731ebfea37ac4dde28Chris Craik * Can be used e.g. to test behavior only active while Properties::debugOverdraw is enabled.
7237413289478a965336239c731ebfea37ac4dde28Chris Craik */
7337413289478a965336239c731ebfea37ac4dde28Chris Craiktemplate <typename T>
7437413289478a965336239c731ebfea37ac4dde28Chris Craikclass ScopedProperty {
7537413289478a965336239c731ebfea37ac4dde28Chris Craikpublic:
7637413289478a965336239c731ebfea37ac4dde28Chris Craik    ScopedProperty(T& property, T newValue)
7737413289478a965336239c731ebfea37ac4dde28Chris Craik        : mPropertyPtr(&property)
7837413289478a965336239c731ebfea37ac4dde28Chris Craik        , mOldValue(property) {
7937413289478a965336239c731ebfea37ac4dde28Chris Craik        property = newValue;
8037413289478a965336239c731ebfea37ac4dde28Chris Craik    }
8137413289478a965336239c731ebfea37ac4dde28Chris Craik    ~ScopedProperty() {
8237413289478a965336239c731ebfea37ac4dde28Chris Craik        *mPropertyPtr = mOldValue;
8337413289478a965336239c731ebfea37ac4dde28Chris Craik    }
8437413289478a965336239c731ebfea37ac4dde28Chris Craikprivate:
8537413289478a965336239c731ebfea37ac4dde28Chris Craik    T* mPropertyPtr;
8637413289478a965336239c731ebfea37ac4dde28Chris Craik    T mOldValue;
8737413289478a965336239c731ebfea37ac4dde28Chris Craik};
8837413289478a965336239c731ebfea37ac4dde28Chris Craik
89b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikclass TestUtils {
90b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craikpublic:
9176ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    class SignalingDtor {
9276ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    public:
9376ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        SignalingDtor()
9476ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik                : mSignal(nullptr) {}
95a619ec70cf765d9166f0862e74653711b87307b3Chih-Hung Hsieh        explicit SignalingDtor(int* signal)
9676ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik                : mSignal(signal) {}
9776ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        void setSignal(int* signal) {
9876ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik            mSignal = signal;
9976ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        }
10076ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        ~SignalingDtor() {
10176ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik            if (mSignal) {
10276ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik                (*mSignal)++;
10376ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik            }
10476ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        }
10576ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    private:
10676ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik        int* mSignal;
10776ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik    };
10876ace115f7870fed9899a9db7d3852e21b5fb258Chris Craik
109b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    static bool matricesAreApproxEqual(const Matrix4& a, const Matrix4& b) {
110b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        for (int i = 0; i < 16; i++) {
111b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik            if (!MathUtils::areEqual(a[i], b[i])) {
112b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik                return false;
113b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik            }
114b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        }
115b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        return true;
116b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
117b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
118b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    static std::unique_ptr<Snapshot> makeSnapshot(const Matrix4& transform, const Rect& clip) {
119b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        std::unique_ptr<Snapshot> snapshot(new Snapshot());
120a2a70723b8cbda4354d23f901f995623e819012cChris Craik        snapshot->clip(clip, SkRegion::kReplace_Op); // store clip first, so it isn't transformed
121b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        *(snapshot->transform) = transform;
122b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        return snapshot;
123b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
124b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
125aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    static sk_sp<Bitmap> createBitmap(int width, int height,
126aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv            SkColorType colorType = kN32_SkColorType) {
127aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        SkImageInfo info = SkImageInfo::Make(width, height, colorType, kPremul_SkAlphaType);
128fc9999505a36c66892d7ccce85187936105f4f36sergeyv        return Bitmap::allocateHeapBitmap(info);
129aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    }
130aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv
131aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    static sk_sp<Bitmap> createBitmap(int width, int height, SkBitmap* outBitmap) {
132aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        SkImageInfo info = SkImageInfo::Make(width, height, kN32_SkColorType, kPremul_SkAlphaType);
133aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        outBitmap->setInfo(info);
134aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv        return Bitmap::allocateHeapBitmap(outBitmap, nullptr);
135aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv    }
136aed7f58fb05a25ce2112829e77c0eb5dd268e8a7sergeyv
13715c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik    static SkBitmap createSkBitmap(int width, int height,
13815c3f19a445b8df575911a16e8a6dba755a084b5Chris Craik            SkColorType colorType = kN32_SkColorType) {
139ddf2215d9807b641dbcb304779ef6b530f876ac7Chris Craik        SkBitmap bitmap;
140253f2c213f6ecda63b6872aee77bd30d5ec07c82Romain Guy        sk_sp<SkColorSpace> colorSpace = SkColorSpace::NewNamed(SkColorSpace::kSRGB_Named);
14116c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        SkImageInfo info = SkImageInfo::Make(width, height,
142253f2c213f6ecda63b6872aee77bd30d5ec07c82Romain Guy                colorType, kPremul_SkAlphaType, colorSpace);
1430a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        bitmap.setInfo(info);
144c1c54062f8cc9d47bdea820ae5ab6aef260b4488sergeyv        Bitmap::allocateHeapBitmap(&bitmap, nullptr);
145ddf2215d9807b641dbcb304779ef6b530f876ac7Chris Craik        return bitmap;
146ddf2215d9807b641dbcb304779ef6b530f876ac7Chris Craik    }
147ddf2215d9807b641dbcb304779ef6b530f876ac7Chris Craik
148d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik    static sp<DeferredLayerUpdater> createTextureLayerUpdater(
149d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik            renderthread::RenderThread& renderThread, uint32_t width, uint32_t height,
150243e85b2e443def1ef47a180e824b36f513c8db8Chris Craik            const SkMatrix& transform);
151d2dfd8f128b632ed99418ab2b32949c939a9a369Chris Craik
152b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    template<class CanvasType>
153003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik    static std::unique_ptr<DisplayList> createDisplayList(int width, int height,
154b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik            std::function<void(CanvasType& canvas)> canvasCallback) {
155b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        CanvasType canvas(width, height);
156b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        canvasCallback(canvas);
157003cc3dec8e2a92e51086fbcd5ee1bb236efa701Chris Craik        return std::unique_ptr<DisplayList>(canvas.finishRecording());
158b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
159b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
160d3daa3198e2212c985c634821682d5819346b653Chris Craik    static sp<RenderNode> createNode(int left, int top, int right, int bottom,
16106152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::function<void(RenderProperties& props, Canvas& canvas)> setup) {
16206152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev#if HWUI_NULL_GPU
16306152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        // if RenderNodes are being sync'd/used, device info will be needed, since
16406152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        // DeviceInfo::maxTextureSize() affects layer property
16506152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        DeviceInfo::initialize();
16606152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev#endif
16706152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev
16806152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        sp<RenderNode> node = new RenderNode();
16906152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        RenderProperties& props = node->mutateStagingProperties();
17006152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        props.setLeftTopRightBottom(left, top, right, bottom);
17106152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        if (setup) {
17206152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::unique_ptr<Canvas> canvas(Canvas::create_recording_canvas(props.getWidth(),
17306152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev                    props.getHeight()));
17406152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            setup(props, *canvas.get());
17506152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            node->setStagingDisplayList(canvas->finishRecording(), nullptr);
17606152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        }
17706152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        node->setPropertyFieldsDirty(0xFFFFFFFF);
17806152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev        return node;
17906152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev    }
18006152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev
18106152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev    template<class RecordingCanvasType>
18206152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev    static sp<RenderNode> createNode(int left, int top, int right, int bottom,
18306152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::function<void(RenderProperties& props, RecordingCanvasType& canvas)> setup) {
1849fded232a9548a304e0145011df8849fba0dcda7Chris Craik#if HWUI_NULL_GPU
18576caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik        // if RenderNodes are being sync'd/used, device info will be needed, since
18676caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik        // DeviceInfo::maxTextureSize() affects layer property
18776caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik        DeviceInfo::initialize();
1889fded232a9548a304e0145011df8849fba0dcda7Chris Craik#endif
18976caecf421b42e9b8294a65f62ff2d90b55a337bChris Craik
190b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik        sp<RenderNode> node = new RenderNode();
19116c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        RenderProperties& props = node->mutateStagingProperties();
19216c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        props.setLeftTopRightBottom(left, top, right, bottom);
19316c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        if (setup) {
19406152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            RecordingCanvasType canvas(props.getWidth(), props.getHeight());
19516c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck            setup(props, canvas);
19651f2d606dcbfba3cc5b03dfea37c1304b91c232fJohn Reck            node->setStagingDisplayList(canvas.finishRecording(), nullptr);
1970b7e8245db728d127ada698be63d78b33fc6e4daChris Craik        }
19816c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck        node->setPropertyFieldsDirty(0xFFFFFFFF);
1990b7e8245db728d127ada698be63d78b33fc6e4daChris Craik        return node;
2000b7e8245db728d127ada698be63d78b33fc6e4daChris Craik    }
2010b7e8245db728d127ada698be63d78b33fc6e4daChris Craik
20216c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck    static void recordNode(RenderNode& node,
20306152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            std::function<void(Canvas&)> contentCallback) {
20406152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev       std::unique_ptr<Canvas> canvas(Canvas::create_recording_canvas(
20506152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev            node.stagingProperties().getWidth(), node.stagingProperties().getHeight()));
20606152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev       contentCallback(*canvas.get());
20706152cdd06da50762716cd455dcf7ab0117f25b0Stan Iliev       node.setStagingDisplayList(canvas->finishRecording(), nullptr);
208b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
209b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
2108d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik    /**
2118d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * Forces a sync of a tree of RenderNode, such that every descendant will have its staging
2128d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * properties and DisplayList moved to the render copies.
2138d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     *
2148d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * Note: does not check dirtiness bits, so any non-staging DisplayLists will be discarded.
2158d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     * For this reason, this should generally only be called once on a tree.
2168d1f2120fe80b23ab03c7168e3b6b2d13bafe2e7Chris Craik     */
217161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik    static void syncHierarchyPropertiesAndDisplayList(sp<RenderNode>& node) {
218161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        syncHierarchyPropertiesAndDisplayListImpl(node.get());
219b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik    }
2200a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik
2219cd1bbe5c9e14472e631d8cc10005613925f34afChris Craik    static sp<RenderNode>& getSyncedNode(sp<RenderNode>& node) {
2229cd1bbe5c9e14472e631d8cc10005613925f34afChris Craik        syncHierarchyPropertiesAndDisplayList(node);
2239cd1bbe5c9e14472e631d8cc10005613925f34afChris Craik        return node;
2247db5ffb7dbd30202468459e2ef4426e91d4fcbb3John Reck    }
2257db5ffb7dbd30202468459e2ef4426e91d4fcbb3John Reck
2260b7e8245db728d127ada698be63d78b33fc6e4daChris Craik    typedef std::function<void(renderthread::RenderThread& thread)> RtCallback;
2270a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik
2280a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    class TestTask : public renderthread::RenderTask {
2290a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    public:
230a619ec70cf765d9166f0862e74653711b87307b3Chih-Hung Hsieh        explicit TestTask(RtCallback rtCallback)
2310a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik                : rtCallback(rtCallback) {}
2320a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        virtual ~TestTask() {}
233e5da4ef971258193cd1e89737a12b95b6ac244bbJohn Reck        virtual void run() override;
2340a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        RtCallback rtCallback;
2350a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    };
2360a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik
2370a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    /**
2380a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik     * NOTE: requires surfaceflinger to run, otherwise this method will wait indefinitely.
2390a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik     */
2400a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    static void runOnRenderThread(RtCallback rtCallback) {
2410a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        TestTask task(rtCallback);
2420a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik        renderthread::RenderThread::getInstance().queueAndWait(&task);
2430a24b146cd3dacf372ce98424044423a5b2fbf2aChris Craik    }
24416c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck
24538e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck    static bool isRenderThreadRunning() {
24638e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck        return renderthread::RenderThread::hasInstance();
24738e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck    }
24838e0c32852e3b9d8ca4a9d3791577f52536419cbJohn Reck
24916c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck    static SkColor interpolateColor(float fraction, SkColor start, SkColor end);
25016c9d6a92e1b86d448c00c52a1630f3e71e6df76John Reck
251e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik    static void layoutTextUnscaled(const SkPaint& paint, const char* text,
252e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik            std::vector<glyph_t>* outGlyphs, std::vector<float>* outPositions,
253e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik            float* outTotalAdvance, Rect* outBounds);
254e8c3c813b0e3ac98304b17a751ce6e436e252bd9Chris Craik
25579abbf22d4f672208327546661e694d837f564a9Derek Sollenberger    static void drawUtf8ToCanvas(Canvas* canvas, const char* text,
25642a5407f2c6403ea7aa7a64eaf19948dc4050df5Chris Craik            const SkPaint& paint, float x, float y);
257a1717271caac5e8ea3808c331d4141ac01a42134Chris Craik
25879abbf22d4f672208327546661e694d837f564a9Derek Sollenberger    static void drawUtf8ToCanvas(Canvas* canvas, const char* text,
259d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik            const SkPaint& paint, const SkPath& path);
260d7448e65e243754f31890baef29dff187dc2e5e5Chris Craik
26179abbf22d4f672208327546661e694d837f564a9Derek Sollenberger    static std::unique_ptr<uint16_t[]> asciiToUtf16(const char* str);
262dccca44ffda4836b56a21da95a046c9708ffd49csergeyv
263161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craikprivate:
264161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik    static void syncHierarchyPropertiesAndDisplayListImpl(RenderNode* node) {
265161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        node->syncProperties();
26644b49f070aafe8ad44efae87341121cce49ff11cJohn Reck        node->syncDisplayList(nullptr);
267161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        auto displayList = node->getDisplayList();
268161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        if (displayList) {
269161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik            for (auto&& childOp : displayList->getChildren()) {
270161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik                syncHierarchyPropertiesAndDisplayListImpl(childOp->renderNode);
271161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik            }
272161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik        }
273161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik    }
274161f54b2d4160b8d3f3da9eba5746da5162e4821Chris Craik
275b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik}; // class TestUtils
276b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik
277b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} /* namespace uirenderer */
278b565df13a9e5c7b1d7d93bdfa4a793752d66d3ccChris Craik} /* namespace android */
279