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