blob: 2cebecb8d304fe1bea99043877bf15babdffda90 [file] [log] [blame]
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007 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
Ady Abrahamb0dbdaa2020-01-06 16:19:42 -080017// TODO(b/129481165): remove the #pragma below and fix conversion issues
18#pragma clang diagnostic push
19#pragma clang diagnostic ignored "-Wconversion"
20
Dan Stoza9e56aa02015-11-02 13:00:03 -080021// #define LOG_NDEBUG 0
22#undef LOG_TAG
23#define LOG_TAG "DisplayDevice"
24
Dominik Laskowski9b17b2c22018-11-01 14:49:03 -070025#include <android-base/stringprintf.h>
Lloyd Pique45a165a2018-10-19 11:54:47 -070026#include <compositionengine/CompositionEngine.h>
27#include <compositionengine/Display.h>
Lloyd Pique3d0c02e2018-10-19 18:38:12 -070028#include <compositionengine/DisplayColorProfile.h>
29#include <compositionengine/DisplayColorProfileCreationArgs.h>
Lloyd Pique45a165a2018-10-19 11:54:47 -070030#include <compositionengine/DisplayCreationArgs.h>
Lloyd Pique542307f2018-10-19 13:24:08 -070031#include <compositionengine/DisplaySurface.h>
Lloyd Pique31cb2942018-10-19 17:23:03 -070032#include <compositionengine/RenderSurface.h>
33#include <compositionengine/RenderSurfaceCreationArgs.h>
Lloyd Pique32cbe282018-10-19 13:09:22 -070034#include <compositionengine/impl/OutputCompositionState.h>
Peiyong Lincbc184f2018-08-22 13:24:10 -070035#include <configstore/Utils.h>
Lloyd Pique3d0c02e2018-10-19 18:38:12 -070036#include <log/log.h>
Alec Mouri0a9c7b82018-11-16 13:05:25 -080037#include <system/window.h>
Lloyd Pique3d0c02e2018-10-19 18:38:12 -070038#include <ui/GraphicTypes.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080039
Lloyd Pique3d0c02e2018-10-19 18:38:12 -070040#include "DisplayDevice.h"
Mathias Agopian13127d82013-03-05 17:47:11 -080041#include "Layer.h"
Lloyd Pique3d0c02e2018-10-19 18:38:12 -070042#include "SurfaceFlinger.h"
Mathias Agopian1f7bec62010-06-25 18:02:21 -070043
Peiyong Linfd997e02018-03-28 15:29:00 -070044namespace android {
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080045
Peiyong Lin65248e02020-04-18 21:15:07 -070046namespace hal = hardware::graphics::composer::hal;
47
Yiwei Zhang5434a782018-12-05 18:06:32 -080048using android::base::StringAppendF;
Jaesoo Lee720a7242017-01-31 15:26:18 +090049
Dominik Laskowski718f9602019-11-09 20:01:35 -080050ui::Transform::RotationFlags DisplayDevice::sPrimaryDisplayRotationFlags = ui::Transform::ROT_0;
Pablo Ceballos021623b2016-04-15 17:31:51 -070051
Lloyd Piqueaad4ebf2019-10-03 17:58:30 -070052DisplayDeviceCreationArgs::DisplayDeviceCreationArgs(
53 const sp<SurfaceFlinger>& flinger, const wp<IBinder>& displayToken,
54 std::shared_ptr<compositionengine::Display> compositionDisplay)
55 : flinger(flinger), displayToken(displayToken), compositionDisplay(compositionDisplay) {}
Chia-I Wube02ec02018-05-18 10:59:36 -070056
Lloyd Piqueaad4ebf2019-10-03 17:58:30 -070057DisplayDevice::DisplayDevice(DisplayDeviceCreationArgs& args)
Lloyd Pique32cbe282018-10-19 13:09:22 -070058 : mFlinger(args.flinger),
Lloyd Pique2eef1d22018-09-18 21:30:04 -070059 mDisplayToken(args.displayToken),
Dominik Laskowskie9774092018-12-11 10:04:24 -080060 mSequenceId(args.sequenceId),
Dominik Laskowski55c85402020-01-21 16:25:47 -080061 mConnectionType(args.connectionType),
Lloyd Piqueaad4ebf2019-10-03 17:58:30 -070062 mCompositionDisplay{args.compositionDisplay},
Dominik Laskowski718f9602019-11-09 20:01:35 -080063 mPhysicalOrientation(args.physicalOrientation),
Dominik Laskowski075d3172018-05-24 15:50:06 -070064 mIsPrimary(args.isPrimary) {
Lloyd Piquea38ea7e2019-04-16 18:10:26 -070065 mCompositionDisplay->editState().isSecure = args.isSecure;
Lloyd Pique31cb2942018-10-19 17:23:03 -070066 mCompositionDisplay->createRenderSurface(
67 compositionengine::RenderSurfaceCreationArgs{ANativeWindow_getWidth(
68 args.nativeWindow.get()),
69 ANativeWindow_getHeight(
70 args.nativeWindow.get()),
71 args.nativeWindow, args.displaySurface});
72
Vishnu Nair9b079a22020-01-21 14:36:08 -080073 if (!mFlinger->mDisableClientCompositionCache &&
74 SurfaceFlinger::maxFrameBufferAcquiredBuffers > 0) {
75 mCompositionDisplay->createClientCompositionCache(
76 static_cast<uint32_t>(SurfaceFlinger::maxFrameBufferAcquiredBuffers));
77 }
78
Lloyd Pique3d0c02e2018-10-19 18:38:12 -070079 mCompositionDisplay->createDisplayColorProfile(
80 compositionengine::DisplayColorProfileCreationArgs{args.hasWideColorGamut,
81 std::move(args.hdrCapabilities),
82 args.supportedPerFrameMetadata,
83 args.hwcColorModes});
84
Lloyd Pique32cbe282018-10-19 13:09:22 -070085 if (!mCompositionDisplay->isValid()) {
86 ALOGE("Composition Display did not validate!");
87 }
88
Lloyd Pique31cb2942018-10-19 17:23:03 -070089 mCompositionDisplay->getRenderSurface()->initialize();
Lloyd Pique2eef1d22018-09-18 21:30:04 -070090
Lloyd Pique32cbe282018-10-19 13:09:22 -070091 setPowerMode(args.initialPowerMode);
Alec Mouriba013fa2018-10-16 12:43:11 -070092
Jesse Hallffe1f192013-03-22 15:13:48 -070093 // initialize the display orientation transform.
Dominik Laskowski718f9602019-11-09 20:01:35 -080094 setProjection(ui::ROTATION_0, Rect::INVALID_RECT, Rect::INVALID_RECT);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080095}
96
Lloyd Pique09594832018-01-22 17:48:03 -080097DisplayDevice::~DisplayDevice() = default;
Mathias Agopian92a979a2012-08-02 18:32:23 -070098
Lloyd Pique45a165a2018-10-19 11:54:47 -070099void DisplayDevice::disconnect() {
100 mCompositionDisplay->disconnect();
Jesse Hall02d86562013-03-25 14:43:23 -0700101}
102
Mathias Agopian0f2f5ff2012-07-31 23:09:07 -0700103int DisplayDevice::getWidth() const {
Lloyd Pique32cbe282018-10-19 13:09:22 -0700104 return mCompositionDisplay->getState().bounds.getWidth();
Mathias Agopiana4912602012-07-12 14:25:33 -0700105}
106
Mathias Agopian0f2f5ff2012-07-31 23:09:07 -0700107int DisplayDevice::getHeight() const {
Lloyd Pique32cbe282018-10-19 13:09:22 -0700108 return mCompositionDisplay->getState().bounds.getHeight();
Mathias Agopiana4912602012-07-12 14:25:33 -0700109}
110
Dominik Laskowskibf170d92018-04-19 15:08:05 -0700111void DisplayDevice::setDisplayName(const std::string& displayName) {
112 if (!displayName.empty()) {
Mathias Agopian9e2463e2012-09-21 18:26:16 -0700113 // never override the name with an empty name
114 mDisplayName = displayName;
Lloyd Pique32cbe282018-10-19 13:09:22 -0700115 mCompositionDisplay->setName(displayName);
Mathias Agopian9e2463e2012-09-21 18:26:16 -0700116 }
117}
118
Marin Shalamanove5cceea2020-04-30 18:13:10 +0200119void DisplayDevice::setDeviceProductInfo(std::optional<DeviceProductInfo> info) {
120 mDeviceProductInfo = std::move(info);
121}
122
Mathias Agopian0f2f5ff2012-07-31 23:09:07 -0700123uint32_t DisplayDevice::getPageFlipCount() const {
Lloyd Pique31cb2942018-10-19 17:23:03 -0700124 return mCompositionDisplay->getRenderSurface()->getPageFlipCount();
Dan Stoza9e56aa02015-11-02 13:00:03 -0800125}
Dan Stoza9e56aa02015-11-02 13:00:03 -0800126
Mathias Agopian1b031492012-06-20 17:51:20 -0700127// ----------------------------------------------------------------------------
Peiyong Lin65248e02020-04-18 21:15:07 -0700128void DisplayDevice::setPowerMode(hal::PowerMode mode) {
Prashant Malani2c9b11f2014-05-25 01:36:31 -0700129 mPowerMode = mode;
Peiyong Lin65248e02020-04-18 21:15:07 -0700130 getCompositionDisplay()->setCompositionEnabled(mPowerMode != hal::PowerMode::OFF);
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700131}
132
Peiyong Lin65248e02020-04-18 21:15:07 -0700133hal::PowerMode DisplayDevice::getPowerMode() const {
Prashant Malani2c9b11f2014-05-25 01:36:31 -0700134 return mPowerMode;
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700135}
136
Dominik Laskowskieecd6592018-05-29 10:25:41 -0700137bool DisplayDevice::isPoweredOn() const {
Peiyong Lin65248e02020-04-18 21:15:07 -0700138 return mPowerMode != hal::PowerMode::OFF;
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700139}
140
Ady Abraham2139f732019-11-13 18:56:40 -0800141void DisplayDevice::setActiveConfig(HwcConfigIndexType mode) {
Michael Lentine6c9e34a2014-07-14 13:48:55 -0700142 mActiveConfig = mode;
143}
144
Ady Abraham2139f732019-11-13 18:56:40 -0800145HwcConfigIndexType DisplayDevice::getActiveConfig() const {
Michael Lentine6c9e34a2014-07-14 13:48:55 -0700146 return mActiveConfig;
147}
148
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800149ui::Dataspace DisplayDevice::getCompositionDataSpace() const {
Lloyd Pique32cbe282018-10-19 13:09:22 -0700150 return mCompositionDisplay->getState().dataspace;
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800151}
152
Dominik Laskowski3cb3d4e2019-11-21 11:14:45 -0800153void DisplayDevice::setLayerStack(ui::LayerStack stack) {
Lloyd Piqueef36b002019-01-23 17:52:04 -0800154 mCompositionDisplay->setLayerStackFilter(stack, isPrimary());
Mathias Agopian28947d72012-08-08 18:51:15 -0700155}
156
Dominik Laskowski3cb3d4e2019-11-21 11:14:45 -0800157void DisplayDevice::setDisplaySize(int width, int height) {
158 mCompositionDisplay->setBounds(ui::Size(width, height));
Michael Lentine47e45402014-07-18 15:34:25 -0700159}
160
Dominik Laskowski718f9602019-11-09 20:01:35 -0800161void DisplayDevice::setProjection(ui::Rotation orientation, Rect viewport, Rect frame) {
Lloyd Pique32cbe282018-10-19 13:09:22 -0700162 mOrientation = orientation;
163
164 const Rect& displayBounds = getCompositionDisplay()->getState().bounds;
165 const int w = displayBounds.width();
166 const int h = displayBounds.height();
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800167
Peiyong Linefefaac2018-08-17 12:27:51 -0700168 ui::Transform R;
Dominik Laskowski718f9602019-11-09 20:01:35 -0800169 if (const auto flags = ui::Transform::toRotationFlags(orientation);
170 flags != ui::Transform::ROT_INVALID) {
171 R.set(flags, w, h);
172 }
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800173
174 if (!frame.isValid()) {
175 // the destination frame can be invalid if it has never been set,
176 // in that case we assume the whole display frame.
177 frame = Rect(w, h);
178 }
179
180 if (viewport.isEmpty()) {
181 // viewport can be invalid if it has never been set, in that case
182 // we assume the whole display size.
183 // it's also invalid to have an empty viewport, so we handle that
184 // case in the same way.
185 viewport = Rect(w, h);
Peiyong Linefefaac2018-08-17 12:27:51 -0700186 if (R.getOrientation() & ui::Transform::ROT_90) {
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800187 // viewport is always specified in the logical orientation
188 // of the display (ie: post-rotation).
Peiyong Lin3db42342018-08-16 09:15:59 -0700189 std::swap(viewport.right, viewport.bottom);
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800190 }
191 }
192
Peiyong Linefefaac2018-08-17 12:27:51 -0700193 ui::Transform TL, TP, S;
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800194 float src_width = viewport.width();
195 float src_height = viewport.height();
196 float dst_width = frame.width();
197 float dst_height = frame.height();
198 if (src_width != dst_width || src_height != dst_height) {
199 float sx = dst_width / src_width;
200 float sy = dst_height / src_height;
201 S.set(sx, 0, 0, sy);
202 }
203
204 float src_x = viewport.left;
205 float src_y = viewport.top;
206 float dst_x = frame.left;
207 float dst_y = frame.top;
208 TL.set(-src_x, -src_y);
209 TP.set(dst_x, dst_y);
210
Lloyd Pique32cbe282018-10-19 13:09:22 -0700211 // need to take care of primary display rotation for globalTransform
Iris Chang7501ed62018-04-30 14:45:42 +0800212 // for case if the panel is not installed aligned with device orientation
Dominik Laskowski075d3172018-05-24 15:50:06 -0700213 if (isPrimary()) {
Dominik Laskowski718f9602019-11-09 20:01:35 -0800214 if (const auto flags = ui::Transform::toRotationFlags(orientation + mPhysicalOrientation);
215 flags != ui::Transform::ROT_INVALID) {
216 R.set(flags, w, h);
217 }
Iris Chang7501ed62018-04-30 14:45:42 +0800218 }
219
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800220 // The viewport and frame are both in the logical orientation.
221 // Apply the logical translation, scale to physical size, apply the
222 // physical translation and finally rotate to the physical orientation.
Lloyd Pique32cbe282018-10-19 13:09:22 -0700223 ui::Transform globalTransform = R * TP * S * TL;
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800224
Lloyd Pique32cbe282018-10-19 13:09:22 -0700225 const uint8_t type = globalTransform.getType();
226 const bool needsFiltering =
227 (!globalTransform.preserveRects() || (type >= ui::Transform::SCALE));
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800228
Alec Mouri5a6d8572020-03-23 23:56:15 -0700229 const Rect& sourceClip = viewport;
230 Rect destinationClip = globalTransform.transform(viewport);
231 if (destinationClip.isEmpty()) {
232 destinationClip = displayBounds;
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800233 }
234
LuK1337a4adfb12019-09-18 18:44:49 +0200235 uint32_t transformOrientation;
236
Dominik Laskowski281644e2018-04-19 15:47:35 -0700237 if (isPrimary()) {
Dominik Laskowski718f9602019-11-09 20:01:35 -0800238 sPrimaryDisplayRotationFlags = ui::Transform::toRotationFlags(orientation);
239 transformOrientation = ui::Transform::toRotationFlags(orientation + mPhysicalOrientation);
LuK1337a4adfb12019-09-18 18:44:49 +0200240 } else {
Dominik Laskowski718f9602019-11-09 20:01:35 -0800241 transformOrientation = ui::Transform::toRotationFlags(orientation);
Pablo Ceballos021623b2016-04-15 17:31:51 -0700242 }
Lloyd Pique32cbe282018-10-19 13:09:22 -0700243
Lloyd Piquee8fe4742020-01-21 15:26:18 -0800244 getCompositionDisplay()->setProjection(globalTransform, transformOrientation, frame, viewport,
245 sourceClip, destinationClip, needsFiltering);
Mathias Agopian1b031492012-06-20 17:51:20 -0700246}
Mathias Agopian1d12d8a2012-09-18 01:38:00 -0700247
Dominik Laskowski718f9602019-11-09 20:01:35 -0800248ui::Transform::RotationFlags DisplayDevice::getPrimaryDisplayRotationFlags() {
249 return sPrimaryDisplayRotationFlags;
Pablo Ceballos021623b2016-04-15 17:31:51 -0700250}
251
Dominik Laskowski9b17b2c22018-11-01 14:49:03 -0700252std::string DisplayDevice::getDebugName() const {
Dominik Laskowski55c85402020-01-21 16:25:47 -0800253 std::string displayId;
254 if (const auto id = getId()) {
255 displayId = to_string(*id) + ", ";
256 }
257
258 const char* type = "virtual";
259 if (mConnectionType) {
260 type = *mConnectionType == DisplayConnectionType::Internal ? "internal" : "external";
261 }
262
263 return base::StringPrintf("DisplayDevice{%s%s%s, \"%s\"}", displayId.c_str(), type,
264 isPrimary() ? ", primary" : "", mDisplayName.c_str());
Dominik Laskowski9b17b2c22018-11-01 14:49:03 -0700265}
266
Yiwei Zhang5434a782018-12-05 18:06:32 -0800267void DisplayDevice::dump(std::string& result) const {
Yiwei Zhang5434a782018-12-05 18:06:32 -0800268 StringAppendF(&result, "+ %s\n", getDebugName().c_str());
Lloyd Pique32cbe282018-10-19 13:09:22 -0700269
270 result.append(" ");
Peiyong Lin65248e02020-04-18 21:15:07 -0700271 StringAppendF(&result, "powerMode=%s (%d), ", to_string(mPowerMode).c_str(),
272 static_cast<int32_t>(mPowerMode));
Ady Abraham2139f732019-11-13 18:56:40 -0800273 StringAppendF(&result, "activeConfig=%d, ", mActiveConfig.value());
Marin Shalamanove5cceea2020-04-30 18:13:10 +0200274 StringAppendF(&result, "deviceProductInfo=");
275 if (mDeviceProductInfo) {
276 mDeviceProductInfo->dump(result);
277 } else {
278 result.append("{}");
279 }
Lloyd Pique32cbe282018-10-19 13:09:22 -0700280 getCompositionDisplay()->dump(result);
Mathias Agopian1d12d8a2012-09-18 01:38:00 -0700281}
Irvelffc9efc2016-07-27 15:16:37 -0700282
Lloyd Pique3d0c02e2018-10-19 18:38:12 -0700283bool DisplayDevice::hasRenderIntent(ui::RenderIntent intent) const {
284 return mCompositionDisplay->getDisplayColorProfile()->hasRenderIntent(intent);
Peiyong Lin136fbbc2018-04-17 15:09:44 -0700285}
286
Lloyd Pique45a165a2018-10-19 11:54:47 -0700287// ----------------------------------------------------------------------------
288
289const std::optional<DisplayId>& DisplayDevice::getId() const {
290 return mCompositionDisplay->getId();
291}
292
293bool DisplayDevice::isSecure() const {
294 return mCompositionDisplay->isSecure();
295}
296
Lloyd Pique32cbe282018-10-19 13:09:22 -0700297const Rect& DisplayDevice::getBounds() const {
298 return mCompositionDisplay->getState().bounds;
299}
300
301const Region& DisplayDevice::getUndefinedRegion() const {
302 return mCompositionDisplay->getState().undefinedRegion;
303}
304
305bool DisplayDevice::needsFiltering() const {
306 return mCompositionDisplay->getState().needsFiltering;
307}
308
Dominik Laskowski3cb3d4e2019-11-21 11:14:45 -0800309ui::LayerStack DisplayDevice::getLayerStack() const {
Lloyd Piqueef36b002019-01-23 17:52:04 -0800310 return mCompositionDisplay->getState().layerStackId;
Lloyd Pique32cbe282018-10-19 13:09:22 -0700311}
312
313const ui::Transform& DisplayDevice::getTransform() const {
314 return mCompositionDisplay->getState().transform;
315}
316
317const Rect& DisplayDevice::getViewport() const {
318 return mCompositionDisplay->getState().viewport;
319}
320
321const Rect& DisplayDevice::getFrame() const {
322 return mCompositionDisplay->getState().frame;
323}
324
Lloyd Piquee8fe4742020-01-21 15:26:18 -0800325const Rect& DisplayDevice::getSourceClip() const {
326 return mCompositionDisplay->getState().sourceClip;
Lloyd Pique32cbe282018-10-19 13:09:22 -0700327}
328
Lloyd Pique3d0c02e2018-10-19 18:38:12 -0700329bool DisplayDevice::hasWideColorGamut() const {
330 return mCompositionDisplay->getDisplayColorProfile()->hasWideColorGamut();
331}
332
333bool DisplayDevice::hasHDR10PlusSupport() const {
334 return mCompositionDisplay->getDisplayColorProfile()->hasHDR10PlusSupport();
335}
336
337bool DisplayDevice::hasHDR10Support() const {
338 return mCompositionDisplay->getDisplayColorProfile()->hasHDR10Support();
339}
340
341bool DisplayDevice::hasHLGSupport() const {
342 return mCompositionDisplay->getDisplayColorProfile()->hasHLGSupport();
343}
344
345bool DisplayDevice::hasDolbyVisionSupport() const {
346 return mCompositionDisplay->getDisplayColorProfile()->hasDolbyVisionSupport();
347}
348
349int DisplayDevice::getSupportedPerFrameMetadata() const {
350 return mCompositionDisplay->getDisplayColorProfile()->getSupportedPerFrameMetadata();
351}
352
353const HdrCapabilities& DisplayDevice::getHdrCapabilities() const {
354 return mCompositionDisplay->getDisplayColorProfile()->getHdrCapabilities();
355}
356
Dominik Laskowski663bd282018-04-19 15:26:54 -0700357std::atomic<int32_t> DisplayDeviceState::sNextSequenceId(1);
Peiyong Linfd997e02018-03-28 15:29:00 -0700358
359} // namespace android
Ady Abrahamb0dbdaa2020-01-06 16:19:42 -0800360
361// TODO(b/129481165): remove the #pragma below and fix conversion issues
362#pragma clang diagnostic pop // ignored "-Wconversion"