blob: 4e59c97bb2e9c91c29235050f4bd8b8699b76682 [file] [log] [blame]
Lloyd Piquef58625d2017-12-19 13:22:33 -08001/*
2 * Copyright (C) 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#pragma once
18
19#include "DisplayDevice.h"
20#include "SurfaceFlinger.h"
21
22namespace android {
23
Lloyd Piquee39cad22017-12-20 17:01:29 -080024class EventThread;
25
26namespace RE {
27class RenderEngine;
28}
29
30namespace Hwc2 {
31class Composer;
32}
33
Lloyd Piquef58625d2017-12-19 13:22:33 -080034class TestableSurfaceFlinger {
35public:
36 // Extend this as needed for accessing SurfaceFlinger private (and public)
37 // functions.
38
Lloyd Piquee39cad22017-12-20 17:01:29 -080039 void setupRenderEngine(std::unique_ptr<RE::RenderEngine> renderEngine) {
40 mFlinger->getBE().mRenderEngine = std::move(renderEngine);
41 }
42
43 void setupComposer(std::unique_ptr<Hwc2::Composer> composer) {
44 mFlinger->getBE().mHwc.reset(new HWComposer(std::move(composer)));
45 }
46
Lloyd Pique5b36f3f2018-01-17 11:57:07 -080047 using CreateBufferQueueFunction = SurfaceFlinger::CreateBufferQueueFunction;
Lloyd Pique5b36f3f2018-01-17 11:57:07 -080048 void setCreateBufferQueueFunction(CreateBufferQueueFunction f) {
49 mFlinger->mCreateBufferQueue = f;
50 }
51
Lloyd Pique0b1fe702018-01-22 18:03:16 -080052 using CreateNativeWindowSurfaceFunction = SurfaceFlinger::CreateNativeWindowSurfaceFunction;
53 void setCreateNativeWindowSurface(CreateNativeWindowSurfaceFunction f) {
54 mFlinger->mCreateNativeWindowSurface = f;
55 }
56
57 using HotplugEvent = SurfaceFlinger::HotplugEvent;
58
Lloyd Piquef58625d2017-12-19 13:22:33 -080059 /* ------------------------------------------------------------------------
60 * Forwarding for functions being tested
61 */
Lloyd Pique0b1fe702018-01-22 18:03:16 -080062
Lloyd Piquea482f992018-01-22 19:00:34 -080063 auto createDisplay(const String8& displayName, bool secure) {
64 return mFlinger->createDisplay(displayName, secure);
65 }
66
Dominik Laskowskieecd6592018-05-29 10:25:41 -070067 auto destroyDisplay(const sp<IBinder>& displayToken) {
68 return mFlinger->destroyDisplay(displayToken);
69 }
Lloyd Piquea482f992018-01-22 19:00:34 -080070
Lloyd Piqued6fbb8a2018-01-22 19:08:36 -080071 auto resetDisplayState() { return mFlinger->resetDisplayState(); }
72
Dominik Laskowski7e045462018-05-30 13:02:02 -070073 auto setupNewDisplayDeviceInternal(const wp<IBinder>& displayToken, int32_t displayId,
Lloyd Piquec11e0d32018-01-22 18:44:59 -080074 const DisplayDeviceState& state,
75 const sp<DisplaySurface>& dispSurface,
76 const sp<IGraphicBufferProducer>& producer) {
Dominik Laskowski7e045462018-05-30 13:02:02 -070077 return mFlinger->setupNewDisplayDeviceInternal(displayToken, displayId, state, dispSurface,
Lloyd Piquec11e0d32018-01-22 18:44:59 -080078 producer);
79 }
80
Lloyd Pique0b1fe702018-01-22 18:03:16 -080081 auto handleTransactionLocked(uint32_t transactionFlags) {
82 return mFlinger->handleTransactionLocked(transactionFlags);
83 }
Lloyd Piquef58625d2017-12-19 13:22:33 -080084
Lloyd Pique6cf11032018-01-22 18:57:44 -080085 auto onHotplugReceived(int32_t sequenceId, hwc2_display_t display,
86 HWC2::Connection connection) {
87 return mFlinger->onHotplugReceived(sequenceId, display, connection);
88 }
89
Lloyd Pique9d9cf402018-02-16 17:47:13 -080090 auto setDisplayStateLocked(const DisplayState& s) { return mFlinger->setDisplayStateLocked(s); }
91
Lloyd Pique86016da2018-03-01 16:09:38 -080092 auto onInitializeDisplays() { return mFlinger->onInitializeDisplays(); }
93
Dominik Laskowskia2edf612018-06-01 13:15:16 -070094 auto setPowerModeInternal(const sp<DisplayDevice>& display, int mode,
95 bool stateLockHeld = false) {
96 return mFlinger->setPowerModeInternal(display, mode, stateLockHeld);
Lloyd Pique7d4aa6c2018-03-01 16:36:35 -080097 }
98
Lloyd Pique86016da2018-03-01 16:09:38 -080099 /* ------------------------------------------------------------------------
100 * Read-only access to private data to assert post-conditions.
101 */
102
103 const auto& getAnimFrameTracker() const { return mFlinger->mAnimFrameTracker; }
Lloyd Pique7d4aa6c2018-03-01 16:36:35 -0800104 const auto& getHasPoweredOff() const { return mFlinger->mHasPoweredOff; }
105 const auto& getHWVsyncAvailable() const { return mFlinger->mHWVsyncAvailable; }
106 const auto& getVisibleRegionsDirty() const { return mFlinger->mVisibleRegionsDirty; }
107
Lloyd Pique86016da2018-03-01 16:09:38 -0800108 const auto& getCompositorTiming() const { return mFlinger->getBE().mCompositorTiming; }
109
Lloyd Piquef58625d2017-12-19 13:22:33 -0800110 /* ------------------------------------------------------------------------
111 * Read-write access to private data to set up preconditions and assert
112 * post-conditions.
113 */
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800114
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800115 auto& mutableHasWideColorDisplay() { return SurfaceFlinger::hasWideColorDisplay; }
Chia-I Wu304d9cd2018-08-27 14:38:14 -0700116 auto& mutablePrimaryDisplayOrientation() { return SurfaceFlinger::primaryDisplayOrientation; }
Peiyong Lin13effd12018-07-24 17:01:47 -0700117 auto& mutableUseColorManagement() { return SurfaceFlinger::useColorManagement; }
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800118
Dominik Laskowskieecd6592018-05-29 10:25:41 -0700119 auto& mutableDisplayTokens() { return mFlinger->mDisplayTokens; }
Lloyd Piquee39cad22017-12-20 17:01:29 -0800120 auto& mutableCurrentState() { return mFlinger->mCurrentState; }
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800121 auto& mutableDisplays() { return mFlinger->mDisplays; }
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800122 auto& mutableDisplayColorSetting() { return mFlinger->mDisplayColorSetting; }
Lloyd Piquee39cad22017-12-20 17:01:29 -0800123 auto& mutableDrawingState() { return mFlinger->mDrawingState; }
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800124 auto& mutableEventControlThread() { return mFlinger->mEventControlThread; }
Lloyd Piquee39cad22017-12-20 17:01:29 -0800125 auto& mutableEventQueue() { return mFlinger->mEventQueue; }
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800126 auto& mutableEventThread() { return mFlinger->mEventThread; }
Lloyd Piqued6fbb8a2018-01-22 19:08:36 -0800127 auto& mutableHWVsyncAvailable() { return mFlinger->mHWVsyncAvailable; }
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800128 auto& mutableInterceptor() { return mFlinger->mInterceptor; }
Lloyd Pique6cf11032018-01-22 18:57:44 -0800129 auto& mutableMainThreadId() { return mFlinger->mMainThreadId; }
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800130 auto& mutablePendingHotplugEvents() { return mFlinger->mPendingHotplugEvents; }
Lloyd Pique41be5d22018-06-21 13:11:48 -0700131 auto& mutablePrimaryDispSync() { return mFlinger->mPrimaryDispSync; }
Lloyd Piqued6fbb8a2018-01-22 19:08:36 -0800132 auto& mutablePrimaryHWVsyncEnabled() { return mFlinger->mPrimaryHWVsyncEnabled; }
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800133 auto& mutableTransactionFlags() { return mFlinger->mTransactionFlags; }
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800134 auto& mutableUseHwcVirtualDisplays() { return mFlinger->mUseHwcVirtualDisplays; }
Lloyd Piquee39cad22017-12-20 17:01:29 -0800135
Lloyd Pique6cf11032018-01-22 18:57:44 -0800136 auto& mutableComposerSequenceId() { return mFlinger->getBE().mComposerSequenceId; }
Lloyd Piquea618d852018-01-17 11:52:30 -0800137 auto& mutableHwcDisplayData() { return mFlinger->getBE().mHwc->mDisplayData; }
138 auto& mutableHwcDisplaySlots() { return mFlinger->getBE().mHwc->mHwcDisplaySlots; }
139
Lloyd Piquee39cad22017-12-20 17:01:29 -0800140 ~TestableSurfaceFlinger() {
141 // All these pointer and container clears help ensure that GMock does
142 // not report a leaked object, since the SurfaceFlinger instance may
143 // still be referenced by something despite our best efforts to destroy
144 // it after each test is done.
145 mutableDisplays().clear();
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800146 mutableEventControlThread().reset();
147 mutableEventQueue().reset();
Lloyd Piquee39cad22017-12-20 17:01:29 -0800148 mutableEventThread().reset();
Lloyd Pique0b1fe702018-01-22 18:03:16 -0800149 mutableInterceptor().reset();
Lloyd Pique41be5d22018-06-21 13:11:48 -0700150 mutablePrimaryDispSync().reset();
Lloyd Piquee39cad22017-12-20 17:01:29 -0800151 mFlinger->getBE().mHwc.reset();
152 mFlinger->getBE().mRenderEngine.reset();
153 }
Lloyd Piquef58625d2017-12-19 13:22:33 -0800154
Lloyd Piquea618d852018-01-17 11:52:30 -0800155 /* ------------------------------------------------------------------------
156 * Wrapper classes for Read-write access to private data to set up
157 * preconditions and assert post-conditions.
158 */
Michael Wright5d22d4f2018-06-21 02:50:34 +0100159 class FakePowerAdvisor : public Hwc2::PowerAdvisor {
160 public:
161 FakePowerAdvisor() = default;
162 ~FakePowerAdvisor() override = default;
163 void setExpensiveRenderingExpected(hwc2_display_t, bool) override { }
164 };
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800165
Lloyd Piquea618d852018-01-17 11:52:30 -0800166 struct HWC2Display : public HWC2::Display {
Michael Wright5d22d4f2018-06-21 02:50:34 +0100167 HWC2Display(Hwc2::Composer& composer, Hwc2::PowerAdvisor& advisor,
Lloyd Piquea618d852018-01-17 11:52:30 -0800168 const std::unordered_set<HWC2::Capability>& capabilities, hwc2_display_t id,
169 HWC2::DisplayType type)
Michael Wright5d22d4f2018-06-21 02:50:34 +0100170 : HWC2::Display(composer, advisor, capabilities, id, type) {}
Lloyd Piquea618d852018-01-17 11:52:30 -0800171 ~HWC2Display() {
172 // Prevents a call to disable vsyncs.
173 mType = HWC2::DisplayType::Invalid;
174 }
175
176 auto& mutableIsConnected() { return this->mIsConnected; }
177 auto& mutableConfigs() { return this->mConfigs; }
178 };
179
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800180 class FakeHwcDisplayInjector {
181 public:
182 static constexpr hwc2_display_t DEFAULT_HWC_DISPLAY_ID = 1000;
183 static constexpr int32_t DEFAULT_WIDTH = 1920;
184 static constexpr int32_t DEFAULT_HEIGHT = 1280;
185 static constexpr int32_t DEFAULT_REFRESH_RATE = 16'666'666;
186 static constexpr int32_t DEFAULT_DPI = 320;
187 static constexpr int32_t DEFAULT_ACTIVE_CONFIG = 0;
188
189 FakeHwcDisplayInjector(DisplayDevice::DisplayType type, HWC2::DisplayType hwcDisplayType)
190 : mType(type), mHwcDisplayType(hwcDisplayType) {}
191
192 auto& setHwcDisplayId(hwc2_display_t displayId) {
193 mHwcDisplayId = displayId;
194 return *this;
195 }
196
197 auto& setWidth(int32_t width) {
198 mWidth = width;
199 return *this;
200 }
201
202 auto& setHeight(int32_t height) {
203 mHeight = height;
204 return *this;
205 }
206
207 auto& setRefreshRate(int32_t refreshRate) {
208 mRefreshRate = refreshRate;
209 return *this;
210 }
211
212 auto& setDpiX(int32_t dpi) {
213 mDpiX = dpi;
214 return *this;
215 }
216
217 auto& setDpiY(int32_t dpi) {
218 mDpiY = dpi;
219 return *this;
220 }
221
222 auto& setActiveConfig(int32_t config) {
223 mActiveConfig = config;
224 return *this;
225 }
226
Lloyd Piquee22f0332018-07-16 16:35:56 -0700227 auto& setCapabilities(const std::unordered_set<HWC2::Capability>* capabilities) {
228 mCapabilities = capabilities;
229 return *this;
230 }
231
232 auto& setPowerAdvisor(Hwc2::PowerAdvisor* powerAdvisor) {
233 mPowerAdvisor = powerAdvisor;
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800234 return *this;
235 }
236
237 void inject(TestableSurfaceFlinger* flinger, Hwc2::Composer* composer) {
Lloyd Piquee22f0332018-07-16 16:35:56 -0700238 static FakePowerAdvisor defaultPowerAdvisor;
239 if (mPowerAdvisor == nullptr) mPowerAdvisor = &defaultPowerAdvisor;
240 static const std::unordered_set<HWC2::Capability> defaultCapabilities;
241 if (mCapabilities == nullptr) mCapabilities = &defaultCapabilities;
242
243 // Caution - Make sure that any values passed by reference here do
244 // not refer to an instance owned by FakeHwcDisplayInjector. This
245 // class has temporary lifetime, while the constructed HWC2::Display
246 // is much longer lived.
247 auto display = std::make_unique<HWC2Display>(*composer, *mPowerAdvisor, *mCapabilities,
Michael Wright5d22d4f2018-06-21 02:50:34 +0100248 mHwcDisplayId, mHwcDisplayType);
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800249
250 auto config = HWC2::Display::Config::Builder(*display, mActiveConfig);
251 config.setWidth(mWidth);
252 config.setHeight(mHeight);
253 config.setVsyncPeriod(mRefreshRate);
254 config.setDpiX(mDpiX);
255 config.setDpiY(mDpiY);
256 display->mutableConfigs().emplace(mActiveConfig, config.build());
257 display->mutableIsConnected() = true;
258
259 ASSERT_TRUE(flinger->mutableHwcDisplayData().size() > static_cast<size_t>(mType));
Dominik Laskowskif9750f22018-06-06 12:24:53 -0700260 flinger->mutableHwcDisplayData()[mType] = HWComposer::DisplayData();
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800261 flinger->mutableHwcDisplayData()[mType].hwcDisplay = display.get();
262 flinger->mutableHwcDisplaySlots().emplace(mHwcDisplayId, mType);
263
264 flinger->mFakeHwcDisplays.push_back(std::move(display));
265 }
266
267 private:
268 DisplayDevice::DisplayType mType;
269 HWC2::DisplayType mHwcDisplayType;
270 hwc2_display_t mHwcDisplayId = DEFAULT_HWC_DISPLAY_ID;
271 int32_t mWidth = DEFAULT_WIDTH;
272 int32_t mHeight = DEFAULT_HEIGHT;
273 int32_t mRefreshRate = DEFAULT_REFRESH_RATE;
274 int32_t mDpiX = DEFAULT_DPI;
275 int32_t mDpiY = DEFAULT_DPI;
276 int32_t mActiveConfig = DEFAULT_ACTIVE_CONFIG;
Lloyd Piquee22f0332018-07-16 16:35:56 -0700277 const std::unordered_set<HWC2::Capability>* mCapabilities = nullptr;
278 Hwc2::PowerAdvisor* mPowerAdvisor = nullptr;
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800279 };
280
281 class FakeDisplayDeviceInjector {
282 public:
283 FakeDisplayDeviceInjector(TestableSurfaceFlinger& flinger, DisplayDevice::DisplayType type,
Dominik Laskowski7e045462018-05-30 13:02:02 -0700284 int32_t displayId)
285 : mFlinger(flinger), mType(type), mDisplayId(displayId) {}
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800286
287 sp<IBinder> token() const { return mDisplayToken; }
288
289 DisplayDeviceState& mutableDrawingDisplayState() {
290 return mFlinger.mutableDrawingState().displays.editValueFor(mDisplayToken);
291 }
292
293 DisplayDeviceState& mutableCurrentDisplayState() {
294 return mFlinger.mutableCurrentState().displays.editValueFor(mDisplayToken);
295 }
296
Lloyd Pique9d9cf402018-02-16 17:47:13 -0800297 const auto& getDrawingDisplayState() {
298 return mFlinger.mutableDrawingState().displays.valueFor(mDisplayToken);
299 }
300
301 const auto& getCurrentDisplayState() {
302 return mFlinger.mutableCurrentState().displays.valueFor(mDisplayToken);
303 }
304
Dominik Laskowski9fae1022018-05-29 13:17:40 -0700305 auto& mutableDisplayDevice() { return mFlinger.mutableDisplays()[mDisplayToken]; }
Lloyd Pique9d9cf402018-02-16 17:47:13 -0800306
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800307 auto& setNativeWindow(const sp<ANativeWindow>& nativeWindow) {
308 mNativeWindow = nativeWindow;
309 return *this;
310 }
311
312 auto& setDisplaySurface(const sp<DisplaySurface>& displaySurface) {
313 mDisplaySurface = displaySurface;
314 return *this;
315 }
316
317 auto& setRenderSurface(std::unique_ptr<RE::Surface> renderSurface) {
318 mRenderSurface = std::move(renderSurface);
319 return *this;
320 }
321
322 auto& setSecure(bool secure) {
323 mSecure = secure;
324 return *this;
325 }
326
327 sp<DisplayDevice> inject() {
Peiyong Lin136fbbc2018-04-17 15:09:44 -0700328 std::unordered_map<ui::ColorMode, std::vector<ui::RenderIntent>> hdrAndRenderIntents;
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800329 sp<DisplayDevice> device =
Dominik Laskowski7e045462018-05-30 13:02:02 -0700330 new DisplayDevice(mFlinger.mFlinger.get(), mType, mDisplayId, mSecure,
331 mDisplayToken, mNativeWindow, mDisplaySurface,
Chia-I Wua02871c2018-08-27 14:38:23 -0700332 std::move(mRenderSurface), 0, 0,
333 DisplayState::eOrientationDefault, false, HdrCapabilities(),
334 0, hdrAndRenderIntents, HWC_POWER_MODE_NORMAL);
Dominik Laskowski9fae1022018-05-29 13:17:40 -0700335 mFlinger.mutableDisplays().emplace(mDisplayToken, device);
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800336
337 DisplayDeviceState state;
338 state.type = mType;
339 state.isSecure = mSecure;
340 mFlinger.mutableCurrentState().displays.add(mDisplayToken, state);
341 mFlinger.mutableDrawingState().displays.add(mDisplayToken, state);
342
Lloyd Pique9d9cf402018-02-16 17:47:13 -0800343 if (mType >= DisplayDevice::DISPLAY_PRIMARY && mType < DisplayDevice::DISPLAY_VIRTUAL) {
Dominik Laskowskieecd6592018-05-29 10:25:41 -0700344 mFlinger.mutableDisplayTokens()[mType] = mDisplayToken;
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800345 }
346
347 return device;
348 }
349
350 private:
351 TestableSurfaceFlinger& mFlinger;
352 sp<BBinder> mDisplayToken = new BBinder();
353 DisplayDevice::DisplayType mType;
Dominik Laskowski7e045462018-05-30 13:02:02 -0700354 const int32_t mDisplayId;
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800355 sp<ANativeWindow> mNativeWindow;
356 sp<DisplaySurface> mDisplaySurface;
357 std::unique_ptr<RE::Surface> mRenderSurface;
358 bool mSecure = false;
359 };
360
Lloyd Piqueac648ee2018-01-17 13:42:24 -0800361 sp<SurfaceFlinger> mFlinger = new SurfaceFlinger(SurfaceFlinger::SkipInitialization);
Lloyd Piquec11e0d32018-01-22 18:44:59 -0800362
363 // We need to keep a reference to these so they are properly destroyed.
364 std::vector<std::unique_ptr<HWC2Display>> mFakeHwcDisplays;
Lloyd Piquef58625d2017-12-19 13:22:33 -0800365};
366
367} // namespace android