Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 1 | /* |
| 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 | |
| 22 | namespace android { |
| 23 | |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 24 | class EventThread; |
| 25 | |
| 26 | namespace RE { |
| 27 | class RenderEngine; |
| 28 | } |
| 29 | |
| 30 | namespace Hwc2 { |
| 31 | class Composer; |
| 32 | } |
| 33 | |
Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 34 | class TestableSurfaceFlinger { |
| 35 | public: |
| 36 | // Extend this as needed for accessing SurfaceFlinger private (and public) |
| 37 | // functions. |
| 38 | |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 39 | 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 Pique | 5b36f3f | 2018-01-17 11:57:07 -0800 | [diff] [blame] | 47 | using CreateBufferQueueFunction = SurfaceFlinger::CreateBufferQueueFunction; |
Lloyd Pique | 5b36f3f | 2018-01-17 11:57:07 -0800 | [diff] [blame] | 48 | void setCreateBufferQueueFunction(CreateBufferQueueFunction f) { |
| 49 | mFlinger->mCreateBufferQueue = f; |
| 50 | } |
| 51 | |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 52 | using CreateNativeWindowSurfaceFunction = SurfaceFlinger::CreateNativeWindowSurfaceFunction; |
| 53 | void setCreateNativeWindowSurface(CreateNativeWindowSurfaceFunction f) { |
| 54 | mFlinger->mCreateNativeWindowSurface = f; |
| 55 | } |
| 56 | |
| 57 | using HotplugEvent = SurfaceFlinger::HotplugEvent; |
| 58 | |
Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 59 | /* ------------------------------------------------------------------------ |
| 60 | * Forwarding for functions being tested |
| 61 | */ |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 62 | |
Lloyd Pique | a482f99 | 2018-01-22 19:00:34 -0800 | [diff] [blame] | 63 | auto createDisplay(const String8& displayName, bool secure) { |
| 64 | return mFlinger->createDisplay(displayName, secure); |
| 65 | } |
| 66 | |
| 67 | auto destroyDisplay(const sp<IBinder>& display) { return mFlinger->destroyDisplay(display); } |
| 68 | |
Lloyd Pique | d6fbb8a | 2018-01-22 19:08:36 -0800 | [diff] [blame] | 69 | auto resetDisplayState() { return mFlinger->resetDisplayState(); } |
| 70 | |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 71 | auto setupNewDisplayDeviceInternal(const wp<IBinder>& display, int hwcId, |
| 72 | const DisplayDeviceState& state, |
| 73 | const sp<DisplaySurface>& dispSurface, |
| 74 | const sp<IGraphicBufferProducer>& producer) { |
| 75 | return mFlinger->setupNewDisplayDeviceInternal(display, hwcId, state, dispSurface, |
| 76 | producer); |
| 77 | } |
| 78 | |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 79 | auto handleTransactionLocked(uint32_t transactionFlags) { |
| 80 | return mFlinger->handleTransactionLocked(transactionFlags); |
| 81 | } |
Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 82 | |
Lloyd Pique | 6cf1103 | 2018-01-22 18:57:44 -0800 | [diff] [blame] | 83 | auto onHotplugReceived(int32_t sequenceId, hwc2_display_t display, |
| 84 | HWC2::Connection connection) { |
| 85 | return mFlinger->onHotplugReceived(sequenceId, display, connection); |
| 86 | } |
| 87 | |
Lloyd Pique | 9d9cf40 | 2018-02-16 17:47:13 -0800 | [diff] [blame^] | 88 | auto setDisplayStateLocked(const DisplayState& s) { return mFlinger->setDisplayStateLocked(s); } |
| 89 | |
Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 90 | /* ------------------------------------------------------------------------ |
| 91 | * Read-write access to private data to set up preconditions and assert |
| 92 | * post-conditions. |
| 93 | */ |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 94 | |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 95 | auto& mutableHasWideColorDisplay() { return SurfaceFlinger::hasWideColorDisplay; } |
| 96 | |
Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 97 | auto& mutableBuiltinDisplays() { return mFlinger->mBuiltinDisplays; } |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 98 | auto& mutableCurrentState() { return mFlinger->mCurrentState; } |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 99 | auto& mutableDisplays() { return mFlinger->mDisplays; } |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 100 | auto& mutableDisplayColorSetting() { return mFlinger->mDisplayColorSetting; } |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 101 | auto& mutableDrawingState() { return mFlinger->mDrawingState; } |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 102 | auto& mutableEventControlThread() { return mFlinger->mEventControlThread; } |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 103 | auto& mutableEventQueue() { return mFlinger->mEventQueue; } |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 104 | auto& mutableEventThread() { return mFlinger->mEventThread; } |
Lloyd Pique | d6fbb8a | 2018-01-22 19:08:36 -0800 | [diff] [blame] | 105 | auto& mutableHWVsyncAvailable() { return mFlinger->mHWVsyncAvailable; } |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 106 | auto& mutableInterceptor() { return mFlinger->mInterceptor; } |
Lloyd Pique | 6cf1103 | 2018-01-22 18:57:44 -0800 | [diff] [blame] | 107 | auto& mutableMainThreadId() { return mFlinger->mMainThreadId; } |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 108 | auto& mutablePendingHotplugEvents() { return mFlinger->mPendingHotplugEvents; } |
Lloyd Pique | d6fbb8a | 2018-01-22 19:08:36 -0800 | [diff] [blame] | 109 | auto& mutablePrimaryHWVsyncEnabled() { return mFlinger->mPrimaryHWVsyncEnabled; } |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 110 | auto& mutableTransactionFlags() { return mFlinger->mTransactionFlags; } |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 111 | auto& mutableUseHwcVirtualDisplays() { return mFlinger->mUseHwcVirtualDisplays; } |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 112 | |
Lloyd Pique | 6cf1103 | 2018-01-22 18:57:44 -0800 | [diff] [blame] | 113 | auto& mutableComposerSequenceId() { return mFlinger->getBE().mComposerSequenceId; } |
Lloyd Pique | bc79209 | 2018-01-17 11:52:30 -0800 | [diff] [blame] | 114 | auto& mutableHwcDisplayData() { return mFlinger->getBE().mHwc->mDisplayData; } |
| 115 | auto& mutableHwcDisplaySlots() { return mFlinger->getBE().mHwc->mHwcDisplaySlots; } |
| 116 | |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 117 | ~TestableSurfaceFlinger() { |
| 118 | // All these pointer and container clears help ensure that GMock does |
| 119 | // not report a leaked object, since the SurfaceFlinger instance may |
| 120 | // still be referenced by something despite our best efforts to destroy |
| 121 | // it after each test is done. |
| 122 | mutableDisplays().clear(); |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 123 | mutableEventControlThread().reset(); |
| 124 | mutableEventQueue().reset(); |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 125 | mutableEventThread().reset(); |
Lloyd Pique | 1fa4d46 | 2018-01-22 18:03:16 -0800 | [diff] [blame] | 126 | mutableInterceptor().reset(); |
Lloyd Pique | e39cad2 | 2017-12-20 17:01:29 -0800 | [diff] [blame] | 127 | mFlinger->getBE().mHwc.reset(); |
| 128 | mFlinger->getBE().mRenderEngine.reset(); |
| 129 | } |
Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 130 | |
Lloyd Pique | bc79209 | 2018-01-17 11:52:30 -0800 | [diff] [blame] | 131 | /* ------------------------------------------------------------------------ |
| 132 | * Wrapper classes for Read-write access to private data to set up |
| 133 | * preconditions and assert post-conditions. |
| 134 | */ |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 135 | |
Lloyd Pique | bc79209 | 2018-01-17 11:52:30 -0800 | [diff] [blame] | 136 | struct HWC2Display : public HWC2::Display { |
| 137 | HWC2Display(Hwc2::Composer& composer, |
| 138 | const std::unordered_set<HWC2::Capability>& capabilities, hwc2_display_t id, |
| 139 | HWC2::DisplayType type) |
| 140 | : HWC2::Display(composer, capabilities, id, type) {} |
| 141 | ~HWC2Display() { |
| 142 | // Prevents a call to disable vsyncs. |
| 143 | mType = HWC2::DisplayType::Invalid; |
| 144 | } |
| 145 | |
| 146 | auto& mutableIsConnected() { return this->mIsConnected; } |
| 147 | auto& mutableConfigs() { return this->mConfigs; } |
| 148 | }; |
| 149 | |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 150 | class FakeHwcDisplayInjector { |
| 151 | public: |
| 152 | static constexpr hwc2_display_t DEFAULT_HWC_DISPLAY_ID = 1000; |
| 153 | static constexpr int32_t DEFAULT_WIDTH = 1920; |
| 154 | static constexpr int32_t DEFAULT_HEIGHT = 1280; |
| 155 | static constexpr int32_t DEFAULT_REFRESH_RATE = 16'666'666; |
| 156 | static constexpr int32_t DEFAULT_DPI = 320; |
| 157 | static constexpr int32_t DEFAULT_ACTIVE_CONFIG = 0; |
| 158 | |
| 159 | FakeHwcDisplayInjector(DisplayDevice::DisplayType type, HWC2::DisplayType hwcDisplayType) |
| 160 | : mType(type), mHwcDisplayType(hwcDisplayType) {} |
| 161 | |
| 162 | auto& setHwcDisplayId(hwc2_display_t displayId) { |
| 163 | mHwcDisplayId = displayId; |
| 164 | return *this; |
| 165 | } |
| 166 | |
| 167 | auto& setWidth(int32_t width) { |
| 168 | mWidth = width; |
| 169 | return *this; |
| 170 | } |
| 171 | |
| 172 | auto& setHeight(int32_t height) { |
| 173 | mHeight = height; |
| 174 | return *this; |
| 175 | } |
| 176 | |
| 177 | auto& setRefreshRate(int32_t refreshRate) { |
| 178 | mRefreshRate = refreshRate; |
| 179 | return *this; |
| 180 | } |
| 181 | |
| 182 | auto& setDpiX(int32_t dpi) { |
| 183 | mDpiX = dpi; |
| 184 | return *this; |
| 185 | } |
| 186 | |
| 187 | auto& setDpiY(int32_t dpi) { |
| 188 | mDpiY = dpi; |
| 189 | return *this; |
| 190 | } |
| 191 | |
| 192 | auto& setActiveConfig(int32_t config) { |
| 193 | mActiveConfig = config; |
| 194 | return *this; |
| 195 | } |
| 196 | |
| 197 | auto& addCapability(HWC2::Capability cap) { |
| 198 | mCapabilities.emplace(cap); |
| 199 | return *this; |
| 200 | } |
| 201 | |
| 202 | void inject(TestableSurfaceFlinger* flinger, Hwc2::Composer* composer) { |
| 203 | auto display = std::make_unique<HWC2Display>(*composer, mCapabilities, mHwcDisplayId, |
| 204 | mHwcDisplayType); |
| 205 | |
| 206 | auto config = HWC2::Display::Config::Builder(*display, mActiveConfig); |
| 207 | config.setWidth(mWidth); |
| 208 | config.setHeight(mHeight); |
| 209 | config.setVsyncPeriod(mRefreshRate); |
| 210 | config.setDpiX(mDpiX); |
| 211 | config.setDpiY(mDpiY); |
| 212 | display->mutableConfigs().emplace(mActiveConfig, config.build()); |
| 213 | display->mutableIsConnected() = true; |
| 214 | |
| 215 | ASSERT_TRUE(flinger->mutableHwcDisplayData().size() > static_cast<size_t>(mType)); |
| 216 | flinger->mutableHwcDisplayData()[mType].reset(); |
| 217 | flinger->mutableHwcDisplayData()[mType].hwcDisplay = display.get(); |
| 218 | flinger->mutableHwcDisplaySlots().emplace(mHwcDisplayId, mType); |
| 219 | |
| 220 | flinger->mFakeHwcDisplays.push_back(std::move(display)); |
| 221 | } |
| 222 | |
| 223 | private: |
| 224 | DisplayDevice::DisplayType mType; |
| 225 | HWC2::DisplayType mHwcDisplayType; |
| 226 | hwc2_display_t mHwcDisplayId = DEFAULT_HWC_DISPLAY_ID; |
| 227 | int32_t mWidth = DEFAULT_WIDTH; |
| 228 | int32_t mHeight = DEFAULT_HEIGHT; |
| 229 | int32_t mRefreshRate = DEFAULT_REFRESH_RATE; |
| 230 | int32_t mDpiX = DEFAULT_DPI; |
| 231 | int32_t mDpiY = DEFAULT_DPI; |
| 232 | int32_t mActiveConfig = DEFAULT_ACTIVE_CONFIG; |
| 233 | std::unordered_set<HWC2::Capability> mCapabilities; |
| 234 | }; |
| 235 | |
| 236 | class FakeDisplayDeviceInjector { |
| 237 | public: |
| 238 | FakeDisplayDeviceInjector(TestableSurfaceFlinger& flinger, DisplayDevice::DisplayType type, |
| 239 | int hwcId) |
| 240 | : mFlinger(flinger), mType(type), mHwcId(hwcId) {} |
| 241 | |
| 242 | sp<IBinder> token() const { return mDisplayToken; } |
| 243 | |
| 244 | DisplayDeviceState& mutableDrawingDisplayState() { |
| 245 | return mFlinger.mutableDrawingState().displays.editValueFor(mDisplayToken); |
| 246 | } |
| 247 | |
| 248 | DisplayDeviceState& mutableCurrentDisplayState() { |
| 249 | return mFlinger.mutableCurrentState().displays.editValueFor(mDisplayToken); |
| 250 | } |
| 251 | |
Lloyd Pique | 9d9cf40 | 2018-02-16 17:47:13 -0800 | [diff] [blame^] | 252 | const auto& getDrawingDisplayState() { |
| 253 | return mFlinger.mutableDrawingState().displays.valueFor(mDisplayToken); |
| 254 | } |
| 255 | |
| 256 | const auto& getCurrentDisplayState() { |
| 257 | return mFlinger.mutableCurrentState().displays.valueFor(mDisplayToken); |
| 258 | } |
| 259 | |
| 260 | auto& mutableDisplayDevice() { return mFlinger.mutableDisplays().valueFor(mDisplayToken); } |
| 261 | |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 262 | auto& setNativeWindow(const sp<ANativeWindow>& nativeWindow) { |
| 263 | mNativeWindow = nativeWindow; |
| 264 | return *this; |
| 265 | } |
| 266 | |
| 267 | auto& setDisplaySurface(const sp<DisplaySurface>& displaySurface) { |
| 268 | mDisplaySurface = displaySurface; |
| 269 | return *this; |
| 270 | } |
| 271 | |
| 272 | auto& setRenderSurface(std::unique_ptr<RE::Surface> renderSurface) { |
| 273 | mRenderSurface = std::move(renderSurface); |
| 274 | return *this; |
| 275 | } |
| 276 | |
| 277 | auto& setSecure(bool secure) { |
| 278 | mSecure = secure; |
| 279 | return *this; |
| 280 | } |
| 281 | |
| 282 | sp<DisplayDevice> inject() { |
| 283 | sp<DisplayDevice> device = |
| 284 | new DisplayDevice(mFlinger.mFlinger.get(), mType, mHwcId, mSecure, mDisplayToken, |
| 285 | mNativeWindow, mDisplaySurface, std::move(mRenderSurface), 0, |
| 286 | 0, false, HdrCapabilities(), 0, HWC_POWER_MODE_NORMAL); |
| 287 | mFlinger.mutableDisplays().add(mDisplayToken, device); |
| 288 | |
| 289 | DisplayDeviceState state(mType, mSecure); |
| 290 | mFlinger.mutableCurrentState().displays.add(mDisplayToken, state); |
| 291 | mFlinger.mutableDrawingState().displays.add(mDisplayToken, state); |
| 292 | |
Lloyd Pique | 9d9cf40 | 2018-02-16 17:47:13 -0800 | [diff] [blame^] | 293 | if (mType >= DisplayDevice::DISPLAY_PRIMARY && mType < DisplayDevice::DISPLAY_VIRTUAL) { |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 294 | mFlinger.mutableBuiltinDisplays()[mType] = mDisplayToken; |
| 295 | } |
| 296 | |
| 297 | return device; |
| 298 | } |
| 299 | |
| 300 | private: |
| 301 | TestableSurfaceFlinger& mFlinger; |
| 302 | sp<BBinder> mDisplayToken = new BBinder(); |
| 303 | DisplayDevice::DisplayType mType; |
| 304 | int mHwcId; |
| 305 | sp<ANativeWindow> mNativeWindow; |
| 306 | sp<DisplaySurface> mDisplaySurface; |
| 307 | std::unique_ptr<RE::Surface> mRenderSurface; |
| 308 | bool mSecure = false; |
| 309 | }; |
| 310 | |
Lloyd Pique | 2d3ee6d | 2018-01-17 13:42:24 -0800 | [diff] [blame] | 311 | sp<SurfaceFlinger> mFlinger = new SurfaceFlinger(SurfaceFlinger::SkipInitialization); |
Lloyd Pique | fc12f56 | 2018-01-22 18:44:59 -0800 | [diff] [blame] | 312 | |
| 313 | // We need to keep a reference to these so they are properly destroyed. |
| 314 | std::vector<std::unique_ptr<HWC2Display>> mFakeHwcDisplays; |
Lloyd Pique | f58625d | 2017-12-19 13:22:33 -0800 | [diff] [blame] | 315 | }; |
| 316 | |
| 317 | } // namespace android |