The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 1 | /* |
| 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 | |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 17 | #ifndef ANDROID_DISPLAY_DEVICE_H |
| 18 | #define ANDROID_DISPLAY_DEVICE_H |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 19 | |
| 20 | #include <stdlib.h> |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 21 | |
| 22 | #include <memory> |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 23 | #include <optional> |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 24 | #include <string> |
Peiyong Lin | 136fbbc | 2018-04-17 15:09:44 -0700 | [diff] [blame] | 25 | #include <unordered_map> |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 26 | |
Dan Stoza | 9e56aa0 | 2015-11-02 13:00:03 -0800 | [diff] [blame] | 27 | #include <binder/IBinder.h> |
Chia-I Wu | f2aa311 | 2018-08-27 14:54:37 -0700 | [diff] [blame] | 28 | #include <gui/LayerState.h> |
Peiyong Lin | 0ac5f4e | 2018-04-19 22:06:34 -0700 | [diff] [blame] | 29 | #include <hardware/hwcomposer_defs.h> |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 30 | #include <math/mat4.h> |
Alec Mouri | 0587464 | 2018-11-14 22:34:02 +0000 | [diff] [blame] | 31 | #include <renderengine/Surface.h> |
Peiyong Lin | 0ac5f4e | 2018-04-19 22:06:34 -0700 | [diff] [blame] | 32 | #include <ui/GraphicTypes.h> |
Peiyong Lin | fb06930 | 2018-04-25 14:34:31 -0700 | [diff] [blame] | 33 | #include <ui/HdrCapabilities.h> |
Peiyong Lin | 0ac5f4e | 2018-04-19 22:06:34 -0700 | [diff] [blame] | 34 | #include <ui/Region.h> |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 35 | #include <ui/Transform.h> |
Alec Mouri | 8147b0e | 2018-10-09 20:57:19 -0700 | [diff] [blame] | 36 | #include <utils/RefBase.h> |
Alec Mouri | 8d4f90a | 2018-11-14 22:33:24 +0000 | [diff] [blame^] | 37 | #include <utils/Mutex.h> |
Mathias Agopian | 4f4f094 | 2013-08-19 17:26:18 -0700 | [diff] [blame] | 38 | #include <utils/String8.h> |
Mathias Agopian | 8630320 | 2012-07-24 22:46:10 -0700 | [diff] [blame] | 39 | #include <utils/Timers.h> |
| 40 | |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 41 | #include "DisplayHardware/DisplayIdentification.h" |
chaviw | a76b271 | 2017-09-20 12:02:26 -0700 | [diff] [blame] | 42 | #include "RenderArea.h" |
Mathias Agopian | f435863 | 2012-08-22 17:16:19 -0700 | [diff] [blame] | 43 | |
Alec Mouri | 0587464 | 2018-11-14 22:34:02 +0000 | [diff] [blame] | 44 | struct ANativeWindow; |
| 45 | |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 46 | namespace android { |
| 47 | |
Jesse Hall | 99c7dbb | 2013-03-14 14:29:29 -0700 | [diff] [blame] | 48 | class DisplaySurface; |
Dan Stoza | 9e56aa0 | 2015-11-02 13:00:03 -0800 | [diff] [blame] | 49 | class Fence; |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 50 | class HWComposer; |
Jesse Hall | 9e663de | 2013-08-16 14:28:37 -0700 | [diff] [blame] | 51 | class IGraphicBufferProducer; |
Mathias Agopian | 13127d8 | 2013-03-05 17:47:11 -0800 | [diff] [blame] | 52 | class Layer; |
Mathias Agopian | 8630320 | 2012-07-24 22:46:10 -0700 | [diff] [blame] | 53 | class SurfaceFlinger; |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 54 | |
David Sodman | fb95bcc | 2017-12-22 15:45:30 -0800 | [diff] [blame] | 55 | struct CompositionInfo; |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 56 | struct DisplayDeviceCreationArgs; |
| 57 | struct DisplayInfo; |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 58 | |
Mathias Agopian | 028a757 | 2012-08-05 01:23:51 -0700 | [diff] [blame] | 59 | class DisplayDevice : public LightRefBase<DisplayDevice> |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 60 | { |
| 61 | public: |
Peiyong Lin | fb06930 | 2018-04-25 14:34:31 -0700 | [diff] [blame] | 62 | constexpr static float sDefaultMinLumiance = 0.0; |
| 63 | constexpr static float sDefaultMaxLumiance = 500.0; |
| 64 | |
Mathias Agopian | 87baae1 | 2012-07-31 12:38:26 -0700 | [diff] [blame] | 65 | // region in layer-stack space |
| 66 | mutable Region dirtyRegion; |
| 67 | // region in screen space |
Mathias Agopian | 87baae1 | 2012-07-31 12:38:26 -0700 | [diff] [blame] | 68 | Region undefinedRegion; |
Jesse Hall | b7a0549 | 2014-08-14 15:45:06 -0700 | [diff] [blame] | 69 | bool lastCompositionHadVisibleLayers; |
Mathias Agopian | 3eb38cb | 2012-04-03 22:09:52 -0700 | [diff] [blame] | 70 | |
Mathias Agopian | 92a979a | 2012-08-02 18:32:23 -0700 | [diff] [blame] | 71 | enum { |
Jesse Hall | 01e2905 | 2013-02-19 16:13:35 -0800 | [diff] [blame] | 72 | NO_LAYER_STACK = 0xFFFFFFFF, |
| 73 | }; |
| 74 | |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 75 | explicit DisplayDevice(DisplayDeviceCreationArgs&& args); |
Mathias Agopian | 028a757 | 2012-08-05 01:23:51 -0700 | [diff] [blame] | 76 | ~DisplayDevice(); |
Mathias Agopian | 92a979a | 2012-08-02 18:32:23 -0700 | [diff] [blame] | 77 | |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 78 | bool isVirtual() const { return mIsVirtual; } |
| 79 | bool isPrimary() const { return mIsPrimary; } |
Mathias Agopian | 92a979a | 2012-08-02 18:32:23 -0700 | [diff] [blame] | 80 | |
Jamie Gennis | dd3cb84 | 2012-10-19 18:19:11 -0700 | [diff] [blame] | 81 | // isSecure indicates whether this display can be trusted to display |
| 82 | // secure surfaces. |
| 83 | bool isSecure() const { return mIsSecure; } |
| 84 | |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 85 | // Flip the front and back buffers if the back buffer is "dirty". Might |
| 86 | // be instantaneous, might involve copying the frame buffer around. |
Chia-I Wu | b02087d | 2017-11-09 10:19:54 -0800 | [diff] [blame] | 87 | void flip() const; |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 88 | |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 89 | int getWidth() const; |
| 90 | int getHeight() const; |
Chia-I Wu | a02871c | 2018-08-27 14:38:23 -0700 | [diff] [blame] | 91 | int getInstallOrientation() const { return mDisplayInstallOrientation; } |
Mathias Agopian | 3eb38cb | 2012-04-03 22:09:52 -0700 | [diff] [blame] | 92 | |
Mathias Agopian | 13127d8 | 2013-03-05 17:47:11 -0800 | [diff] [blame] | 93 | void setVisibleLayersSortedByZ(const Vector< sp<Layer> >& layers); |
| 94 | const Vector< sp<Layer> >& getVisibleLayersSortedByZ() const; |
Chia-I Wu | 8380689 | 2017-11-16 10:50:20 -0800 | [diff] [blame] | 95 | void setLayersNeedingFences(const Vector< sp<Layer> >& layers); |
| 96 | const Vector< sp<Layer> >& getLayersNeedingFences() const; |
Mathias Agopian | cd60f99 | 2012-08-16 16:28:27 -0700 | [diff] [blame] | 97 | Region getDirtyRegion(bool repaintEverything) const; |
Mathias Agopian | 3b1d2b6 | 2012-07-11 13:48:17 -0700 | [diff] [blame] | 98 | |
Mathias Agopian | 28947d7 | 2012-08-08 18:51:15 -0700 | [diff] [blame] | 99 | void setLayerStack(uint32_t stack); |
Michael Lentine | 47e4540 | 2014-07-18 15:34:25 -0700 | [diff] [blame] | 100 | void setDisplaySize(const int newWidth, const int newHeight); |
Mathias Agopian | 00e8c7a | 2012-09-04 19:30:46 -0700 | [diff] [blame] | 101 | void setProjection(int orientation, const Rect& viewport, const Rect& frame); |
Mathias Agopian | 28947d7 | 2012-08-08 18:51:15 -0700 | [diff] [blame] | 102 | |
Mathias Agopian | 1b03149 | 2012-06-20 17:51:20 -0700 | [diff] [blame] | 103 | int getOrientation() const { return mOrientation; } |
Mathias Agopian | c1c05de | 2013-09-17 23:45:22 -0700 | [diff] [blame] | 104 | uint32_t getOrientationTransform() const; |
Pablo Ceballos | 021623b | 2016-04-15 17:31:51 -0700 | [diff] [blame] | 105 | static uint32_t getPrimaryDisplayOrientationTransform(); |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 106 | const ui::Transform& getTransform() const { return mGlobalTransform; } |
Mathias Agopian | f5f714a | 2013-02-26 16:54:05 -0800 | [diff] [blame] | 107 | const Rect getViewport() const { return mViewport; } |
| 108 | const Rect getFrame() const { return mFrame; } |
Mathias Agopian | 766dc49 | 2012-10-30 18:08:06 -0700 | [diff] [blame] | 109 | const Rect& getScissor() const { return mScissor; } |
Mathias Agopian | eba8c68 | 2012-09-19 23:14:45 -0700 | [diff] [blame] | 110 | bool needsFiltering() const { return mNeedsFiltering; } |
Mathias Agopian | da8d0a5 | 2012-09-04 15:05:38 -0700 | [diff] [blame] | 111 | |
Mathias Agopian | 87baae1 | 2012-07-31 12:38:26 -0700 | [diff] [blame] | 112 | uint32_t getLayerStack() const { return mLayerStack; } |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 113 | |
| 114 | const std::optional<DisplayId>& getId() const { return mId; } |
| 115 | const wp<IBinder>& getDisplayToken() const { return mDisplayToken; } |
Mathias Agopian | 87baae1 | 2012-07-31 12:38:26 -0700 | [diff] [blame] | 116 | |
Peiyong Lin | 0ac5f4e | 2018-04-19 22:06:34 -0700 | [diff] [blame] | 117 | int32_t getSupportedPerFrameMetadata() const { return mSupportedPerFrameMetadata; } |
| 118 | |
Dan Stoza | 7143316 | 2014-02-04 16:22:36 -0800 | [diff] [blame] | 119 | // We pass in mustRecompose so we can keep VirtualDisplaySurface's state |
| 120 | // machine happy without actually queueing a buffer if nothing has changed |
| 121 | status_t beginFrame(bool mustRecompose) const; |
David Sodman | fb95bcc | 2017-12-22 15:45:30 -0800 | [diff] [blame] | 122 | status_t prepareFrame(HWComposer& hwc, std::vector<CompositionInfo>& compositionInfo); |
Chia-I Wu | be02ec0 | 2018-05-18 10:59:36 -0700 | [diff] [blame] | 123 | |
Peiyong Lin | dd9b2ae | 2018-03-01 16:22:45 -0800 | [diff] [blame] | 124 | bool hasWideColorGamut() const { return mHasWideColorGamut; } |
Peiyong Lin | 136fbbc | 2018-04-17 15:09:44 -0700 | [diff] [blame] | 125 | // Whether h/w composer has native support for specific HDR type. |
Peiyong Lin | 6266589 | 2018-04-16 11:07:44 -0700 | [diff] [blame] | 126 | bool hasHDR10Support() const { return mHasHdr10; } |
| 127 | bool hasHLGSupport() const { return mHasHLG; } |
| 128 | bool hasDolbyVisionSupport() const { return mHasDolbyVision; } |
Chia-I Wu | be02ec0 | 2018-05-18 10:59:36 -0700 | [diff] [blame] | 129 | |
Peiyong Lin | dfde511 | 2018-06-05 10:58:41 -0700 | [diff] [blame] | 130 | // Return true if the HDR dataspace is supported but |
| 131 | // there is no corresponding color mode. |
| 132 | bool hasLegacyHdrSupport(ui::Dataspace dataspace) const; |
Chia-I Wu | be02ec0 | 2018-05-18 10:59:36 -0700 | [diff] [blame] | 133 | |
Peiyong Lin | fb06930 | 2018-04-25 14:34:31 -0700 | [diff] [blame] | 134 | // The returned HdrCapabilities is the combination of HDR capabilities from |
| 135 | // hardware composer and RenderEngine. When the DisplayDevice supports wide |
| 136 | // color gamut, RenderEngine is able to simulate HDR support in Display P3 |
| 137 | // color space for both PQ and HLG HDR contents. The minimum and maximum |
| 138 | // luminance will be set to sDefaultMinLumiance and sDefaultMaxLumiance |
| 139 | // respectively if hardware composer doesn't return meaningful values. |
| 140 | const HdrCapabilities& getHdrCapabilities() const { return mHdrCapabilities; } |
Jesse Hall | 38efe86 | 2013-04-06 23:12:29 -0700 | [diff] [blame] | 141 | |
Chia-I Wu | be02ec0 | 2018-05-18 10:59:36 -0700 | [diff] [blame] | 142 | // Return true if intent is supported by the display. |
| 143 | bool hasRenderIntent(ui::RenderIntent intent) const; |
Peiyong Lin | 136fbbc | 2018-04-17 15:09:44 -0700 | [diff] [blame] | 144 | |
Chia-I Wu | be02ec0 | 2018-05-18 10:59:36 -0700 | [diff] [blame] | 145 | void getBestColorMode(ui::Dataspace dataspace, ui::RenderIntent intent, |
| 146 | ui::Dataspace* outDataspace, ui::ColorMode* outMode, |
| 147 | ui::RenderIntent* outIntent) const; |
Peiyong Lin | 136fbbc | 2018-04-17 15:09:44 -0700 | [diff] [blame] | 148 | |
Alec Mouri | 8d4f90a | 2018-11-14 22:33:24 +0000 | [diff] [blame^] | 149 | void swapBuffers(HWComposer& hwc) const; |
Jesse Hall | 01e2905 | 2013-02-19 16:13:35 -0800 | [diff] [blame] | 150 | |
Mathias Agopian | da27af9 | 2012-09-13 18:17:13 -0700 | [diff] [blame] | 151 | // called after h/w composer has completed its set() call |
Alec Mouri | 8d4f90a | 2018-11-14 22:33:24 +0000 | [diff] [blame^] | 152 | void onSwapBuffersCompleted() const; |
Mathias Agopian | da27af9 | 2012-09-13 18:17:13 -0700 | [diff] [blame] | 153 | |
Mathias Agopian | 9c6e297 | 2011-09-20 17:21:56 -0700 | [diff] [blame] | 154 | Rect getBounds() const { |
Mathias Agopian | 1b03149 | 2012-06-20 17:51:20 -0700 | [diff] [blame] | 155 | return Rect(mDisplayWidth, mDisplayHeight); |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 156 | } |
Mathias Agopian | 9c6e297 | 2011-09-20 17:21:56 -0700 | [diff] [blame] | 157 | inline Rect bounds() const { return getBounds(); } |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 158 | |
Dominik Laskowski | bf170d9 | 2018-04-19 15:08:05 -0700 | [diff] [blame] | 159 | void setDisplayName(const std::string& displayName); |
| 160 | const std::string& getDisplayName() const { return mDisplayName; } |
Andy McFadden | 8dfa92f | 2012-09-17 18:27:17 -0700 | [diff] [blame] | 161 | |
Alec Mouri | 0587464 | 2018-11-14 22:34:02 +0000 | [diff] [blame] | 162 | bool makeCurrent() const; |
Mathias Agopian | 875d8e1 | 2013-06-07 15:35:48 -0700 | [diff] [blame] | 163 | void setViewportAndProjection() const; |
Mathias Agopian | bae92d0 | 2012-09-28 01:00:47 -0700 | [diff] [blame] | 164 | |
Dan Stoza | 9e56aa0 | 2015-11-02 13:00:03 -0800 | [diff] [blame] | 165 | const sp<Fence>& getClientTargetAcquireFence() const; |
Dan Stoza | 9e56aa0 | 2015-11-02 13:00:03 -0800 | [diff] [blame] | 166 | |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 167 | /* ------------------------------------------------------------------------ |
Prashant Malani | 2c9b11f | 2014-05-25 01:36:31 -0700 | [diff] [blame] | 168 | * Display power mode management. |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 169 | */ |
Prashant Malani | 2c9b11f | 2014-05-25 01:36:31 -0700 | [diff] [blame] | 170 | int getPowerMode() const; |
| 171 | void setPowerMode(int mode); |
Dominik Laskowski | eecd659 | 2018-05-29 10:25:41 -0700 | [diff] [blame] | 172 | bool isPoweredOn() const; |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 173 | |
Peiyong Lin | fd997e0 | 2018-03-28 15:29:00 -0700 | [diff] [blame] | 174 | ui::ColorMode getActiveColorMode() const; |
| 175 | void setActiveColorMode(ui::ColorMode mode); |
Peiyong Lin | dd9b2ae | 2018-03-01 16:22:45 -0800 | [diff] [blame] | 176 | ui::RenderIntent getActiveRenderIntent() const; |
| 177 | void setActiveRenderIntent(ui::RenderIntent renderIntent); |
Yiwei Zhang | 7c64f17 | 2018-03-07 14:52:28 -0800 | [diff] [blame] | 178 | android_color_transform_t getColorTransform() const; |
| 179 | void setColorTransform(const mat4& transform); |
Peiyong Lin | 34beb7a | 2018-03-28 11:57:12 -0700 | [diff] [blame] | 180 | void setCompositionDataSpace(ui::Dataspace dataspace); |
Peiyong Lin | dd9b2ae | 2018-03-01 16:22:45 -0800 | [diff] [blame] | 181 | ui::Dataspace getCompositionDataSpace() const; |
Michael Wright | 28f24d0 | 2016-07-12 13:30:53 -0700 | [diff] [blame] | 182 | |
Michael Lentine | 6c9e34a | 2014-07-14 13:48:55 -0700 | [diff] [blame] | 183 | /* ------------------------------------------------------------------------ |
| 184 | * Display active config management. |
| 185 | */ |
| 186 | int getActiveConfig() const; |
| 187 | void setActiveConfig(int mode); |
| 188 | |
Jesse Hall | 02d8656 | 2013-03-25 14:43:23 -0700 | [diff] [blame] | 189 | // release HWC resources (if any) for removable displays |
| 190 | void disconnect(HWComposer& hwc); |
| 191 | |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 192 | /* ------------------------------------------------------------------------ |
| 193 | * Debugging |
| 194 | */ |
| 195 | uint32_t getPageFlipCount() const; |
Dominik Laskowski | 9b17b2c2 | 2018-11-01 14:49:03 -0700 | [diff] [blame] | 196 | std::string getDebugName() const; |
Mathias Agopian | 74d211a | 2013-04-22 16:55:35 +0200 | [diff] [blame] | 197 | void dump(String8& result) const; |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 198 | |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 199 | private: |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 200 | const sp<SurfaceFlinger> mFlinger; |
| 201 | const wp<IBinder> mDisplayToken; |
| 202 | |
| 203 | std::optional<DisplayId> mId; |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 204 | |
Mathias Agopian | a491260 | 2012-07-12 14:25:33 -0700 | [diff] [blame] | 205 | // ANativeWindow this display is rendering into |
Mathias Agopian | d8552d7 | 2012-08-04 21:39:11 -0700 | [diff] [blame] | 206 | sp<ANativeWindow> mNativeWindow; |
Jesse Hall | 99c7dbb | 2013-03-14 14:29:29 -0700 | [diff] [blame] | 207 | sp<DisplaySurface> mDisplaySurface; |
Mathias Agopian | a491260 | 2012-07-12 14:25:33 -0700 | [diff] [blame] | 208 | |
Alec Mouri | 0587464 | 2018-11-14 22:34:02 +0000 | [diff] [blame] | 209 | std::unique_ptr<renderengine::Surface> mSurface; |
Mathias Agopian | 1b03149 | 2012-06-20 17:51:20 -0700 | [diff] [blame] | 210 | int mDisplayWidth; |
| 211 | int mDisplayHeight; |
Chia-I Wu | a02871c | 2018-08-27 14:38:23 -0700 | [diff] [blame] | 212 | const int mDisplayInstallOrientation; |
Mathias Agopian | 076b1cc | 2009-04-10 14:24:30 -0700 | [diff] [blame] | 213 | mutable uint32_t mPageFlipCount; |
Dominik Laskowski | bf170d9 | 2018-04-19 15:08:05 -0700 | [diff] [blame] | 214 | std::string mDisplayName; |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 215 | |
| 216 | const bool mIsVirtual; |
| 217 | const bool mIsSecure; |
Mathias Agopian | 03e4072 | 2012-04-26 16:11:59 -0700 | [diff] [blame] | 218 | |
Mathias Agopian | a491260 | 2012-07-12 14:25:33 -0700 | [diff] [blame] | 219 | /* |
| 220 | * Can only accessed from the main thread, these members |
| 221 | * don't need synchronization. |
| 222 | */ |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 223 | |
Mathias Agopian | 3b1d2b6 | 2012-07-11 13:48:17 -0700 | [diff] [blame] | 224 | // list of visible layers on that display |
Mathias Agopian | 13127d8 | 2013-03-05 17:47:11 -0800 | [diff] [blame] | 225 | Vector< sp<Layer> > mVisibleLayersSortedByZ; |
Chia-I Wu | 8380689 | 2017-11-16 10:50:20 -0800 | [diff] [blame] | 226 | // list of layers needing fences |
| 227 | Vector< sp<Layer> > mLayersNeedingFences; |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 228 | |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 229 | /* |
| 230 | * Transaction state |
| 231 | */ |
Mathias Agopian | da8d0a5 | 2012-09-04 15:05:38 -0700 | [diff] [blame] | 232 | static status_t orientationToTransfrom(int orientation, |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 233 | int w, int h, ui::Transform* tr); |
Mathias Agopian | da8d0a5 | 2012-09-04 15:05:38 -0700 | [diff] [blame] | 234 | |
Fabien Sanglard | f0c53d6 | 2017-03-03 18:58:50 -0800 | [diff] [blame] | 235 | // The identifier of the active layer stack for this display. Several displays |
| 236 | // can use the same layer stack: A z-ordered group of layers (sometimes called |
| 237 | // "surfaces"). Any given layer can only be on a single layer stack. |
Mathias Agopian | 87baae1 | 2012-07-31 12:38:26 -0700 | [diff] [blame] | 238 | uint32_t mLayerStack; |
Fabien Sanglard | f0c53d6 | 2017-03-03 18:58:50 -0800 | [diff] [blame] | 239 | |
Mathias Agopian | da8d0a5 | 2012-09-04 15:05:38 -0700 | [diff] [blame] | 240 | int mOrientation; |
Pablo Ceballos | 021623b | 2016-04-15 17:31:51 -0700 | [diff] [blame] | 241 | static uint32_t sPrimaryDisplayOrientation; |
Mathias Agopian | 766dc49 | 2012-10-30 18:08:06 -0700 | [diff] [blame] | 242 | // user-provided visible area of the layer stack |
Mathias Agopian | da8d0a5 | 2012-09-04 15:05:38 -0700 | [diff] [blame] | 243 | Rect mViewport; |
Mathias Agopian | 766dc49 | 2012-10-30 18:08:06 -0700 | [diff] [blame] | 244 | // user-provided rectangle where mViewport gets mapped to |
Mathias Agopian | da8d0a5 | 2012-09-04 15:05:38 -0700 | [diff] [blame] | 245 | Rect mFrame; |
Mathias Agopian | 766dc49 | 2012-10-30 18:08:06 -0700 | [diff] [blame] | 246 | // pre-computed scissor to apply to the display |
| 247 | Rect mScissor; |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 248 | ui::Transform mGlobalTransform; |
Mathias Agopian | eba8c68 | 2012-09-19 23:14:45 -0700 | [diff] [blame] | 249 | bool mNeedsFiltering; |
Prashant Malani | 2c9b11f | 2014-05-25 01:36:31 -0700 | [diff] [blame] | 250 | // Current power mode |
| 251 | int mPowerMode; |
Michael Lentine | 6c9e34a | 2014-07-14 13:48:55 -0700 | [diff] [blame] | 252 | // Current active config |
| 253 | int mActiveConfig; |
Michael Wright | 28f24d0 | 2016-07-12 13:30:53 -0700 | [diff] [blame] | 254 | // current active color mode |
Chia-I Wu | 614e142 | 2018-05-23 02:17:03 -0700 | [diff] [blame] | 255 | ui::ColorMode mActiveColorMode = ui::ColorMode::NATIVE; |
Peiyong Lin | dd9b2ae | 2018-03-01 16:22:45 -0800 | [diff] [blame] | 256 | // Current active render intent. |
Chia-I Wu | 614e142 | 2018-05-23 02:17:03 -0700 | [diff] [blame] | 257 | ui::RenderIntent mActiveRenderIntent = ui::RenderIntent::COLORIMETRIC; |
| 258 | ui::Dataspace mCompositionDataSpace = ui::Dataspace::UNKNOWN; |
Yiwei Zhang | 7c64f17 | 2018-03-07 14:52:28 -0800 | [diff] [blame] | 259 | // Current color transform |
| 260 | android_color_transform_t mColorTransform; |
Courtney Goeltzenleuchter | 5d94389 | 2017-03-22 13:46:46 -0600 | [diff] [blame] | 261 | |
| 262 | // Need to know if display is wide-color capable or not. |
| 263 | // Initialized by SurfaceFlinger when the DisplayDevice is created. |
| 264 | // Fed to RenderEngine during composition. |
Peiyong Lin | dd9b2ae | 2018-03-01 16:22:45 -0800 | [diff] [blame] | 265 | bool mHasWideColorGamut; |
| 266 | bool mHasHdr10; |
Peiyong Lin | 6266589 | 2018-04-16 11:07:44 -0700 | [diff] [blame] | 267 | bool mHasHLG; |
| 268 | bool mHasDolbyVision; |
Peiyong Lin | fb06930 | 2018-04-25 14:34:31 -0700 | [diff] [blame] | 269 | HdrCapabilities mHdrCapabilities; |
Peiyong Lin | 0ac5f4e | 2018-04-19 22:06:34 -0700 | [diff] [blame] | 270 | const int32_t mSupportedPerFrameMetadata; |
Chia-I Wu | be02ec0 | 2018-05-18 10:59:36 -0700 | [diff] [blame] | 271 | |
| 272 | // Mappings from desired Dataspace/RenderIntent to the supported |
| 273 | // Dataspace/ColorMode/RenderIntent. |
| 274 | using ColorModeKey = uint64_t; |
| 275 | struct ColorModeValue { |
| 276 | ui::Dataspace dataspace; |
| 277 | ui::ColorMode colorMode; |
| 278 | ui::RenderIntent renderIntent; |
| 279 | }; |
| 280 | |
| 281 | static ColorModeKey getColorModeKey(ui::Dataspace dataspace, ui::RenderIntent intent) { |
| 282 | return (static_cast<uint64_t>(dataspace) << 32) | static_cast<uint32_t>(intent); |
| 283 | } |
| 284 | void populateColorModes( |
| 285 | const std::unordered_map<ui::ColorMode, std::vector<ui::RenderIntent>>& hwcColorModes); |
| 286 | void addColorMode( |
| 287 | const std::unordered_map<ui::ColorMode, std::vector<ui::RenderIntent>>& hwcColorModes, |
| 288 | const ui::ColorMode mode, const ui::RenderIntent intent); |
| 289 | |
| 290 | std::unordered_map<ColorModeKey, ColorModeValue> mColorModes; |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 291 | |
| 292 | // TODO(b/74619554): Remove special cases for primary display. |
| 293 | const bool mIsPrimary; |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 294 | }; |
| 295 | |
Irvel | ffc9efc | 2016-07-27 15:16:37 -0700 | [diff] [blame] | 296 | struct DisplayDeviceState { |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 297 | bool isVirtual() const { return !displayId.has_value(); } |
Irvel | ffc9efc | 2016-07-27 15:16:37 -0700 | [diff] [blame] | 298 | |
Dominik Laskowski | 663bd28 | 2018-04-19 15:26:54 -0700 | [diff] [blame] | 299 | int32_t sequenceId = sNextSequenceId++; |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 300 | std::optional<DisplayId> displayId; |
Irvel | ffc9efc | 2016-07-27 15:16:37 -0700 | [diff] [blame] | 301 | sp<IGraphicBufferProducer> surface; |
| 302 | uint32_t layerStack = DisplayDevice::NO_LAYER_STACK; |
| 303 | Rect viewport; |
| 304 | Rect frame; |
| 305 | uint8_t orientation = 0; |
| 306 | uint32_t width = 0; |
| 307 | uint32_t height = 0; |
Dominik Laskowski | bf170d9 | 2018-04-19 15:08:05 -0700 | [diff] [blame] | 308 | std::string displayName; |
Irvel | ffc9efc | 2016-07-27 15:16:37 -0700 | [diff] [blame] | 309 | bool isSecure = false; |
Dominik Laskowski | 663bd28 | 2018-04-19 15:26:54 -0700 | [diff] [blame] | 310 | |
| 311 | private: |
| 312 | static std::atomic<int32_t> sNextSequenceId; |
Irvel | ffc9efc | 2016-07-27 15:16:37 -0700 | [diff] [blame] | 313 | }; |
| 314 | |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 315 | struct DisplayDeviceCreationArgs { |
| 316 | // We use a constructor to ensure some of the values are set, without |
| 317 | // assuming a default value. |
| 318 | DisplayDeviceCreationArgs(const sp<SurfaceFlinger>& flinger, const wp<IBinder>& displayToken, |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 319 | const std::optional<DisplayId>& displayId); |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 320 | |
| 321 | const sp<SurfaceFlinger> flinger; |
| 322 | const wp<IBinder> displayToken; |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 323 | const std::optional<DisplayId> displayId; |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 324 | |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 325 | bool isVirtual{false}; |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 326 | bool isSecure{false}; |
| 327 | sp<ANativeWindow> nativeWindow; |
| 328 | sp<DisplaySurface> displaySurface; |
Alec Mouri | 0587464 | 2018-11-14 22:34:02 +0000 | [diff] [blame] | 329 | std::unique_ptr<renderengine::Surface> renderSurface; |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 330 | int displayInstallOrientation{DisplayState::eOrientationDefault}; |
| 331 | bool hasWideColorGamut{false}; |
| 332 | HdrCapabilities hdrCapabilities; |
| 333 | int32_t supportedPerFrameMetadata{0}; |
| 334 | std::unordered_map<ui::ColorMode, std::vector<ui::RenderIntent>> hwcColorModes; |
| 335 | int initialPowerMode{HWC_POWER_MODE_NORMAL}; |
Dominik Laskowski | 075d317 | 2018-05-24 15:50:06 -0700 | [diff] [blame] | 336 | bool isPrimary{false}; |
Lloyd Pique | 2eef1d2 | 2018-09-18 21:30:04 -0700 | [diff] [blame] | 337 | }; |
| 338 | |
chaviw | a76b271 | 2017-09-20 12:02:26 -0700 | [diff] [blame] | 339 | class DisplayRenderArea : public RenderArea { |
| 340 | public: |
| 341 | DisplayRenderArea(const sp<const DisplayDevice> device, |
Chia-I Wu | c80dcbb | 2018-08-24 15:34:02 -0700 | [diff] [blame] | 342 | ui::Transform::orientation_flags rotation = ui::Transform::ROT_0) |
Chia-I Wu | 9d1abea | 2018-08-23 13:44:43 -0700 | [diff] [blame] | 343 | : DisplayRenderArea(device, device->getBounds(), device->getWidth(), device->getHeight(), |
Peiyong Lin | 0e003c9 | 2018-09-17 11:09:51 -0700 | [diff] [blame] | 344 | device->getCompositionDataSpace(), rotation) {} |
Chia-I Wu | 9d1abea | 2018-08-23 13:44:43 -0700 | [diff] [blame] | 345 | DisplayRenderArea(const sp<const DisplayDevice> device, Rect sourceCrop, uint32_t reqWidth, |
Peiyong Lin | 0e003c9 | 2018-09-17 11:09:51 -0700 | [diff] [blame] | 346 | uint32_t reqHeight, ui::Dataspace reqDataSpace, |
| 347 | ui::Transform::orientation_flags rotation) |
| 348 | : RenderArea(reqWidth, reqHeight, CaptureFill::OPAQUE, reqDataSpace, |
Chia-I Wu | f2aa311 | 2018-08-27 14:54:37 -0700 | [diff] [blame] | 349 | getDisplayRotation(rotation, device->getInstallOrientation())), |
| 350 | mDevice(device), |
| 351 | mSourceCrop(sourceCrop) {} |
chaviw | a76b271 | 2017-09-20 12:02:26 -0700 | [diff] [blame] | 352 | |
Peiyong Lin | efefaac | 2018-08-17 12:27:51 -0700 | [diff] [blame] | 353 | const ui::Transform& getTransform() const override { return mDevice->getTransform(); } |
chaviw | a76b271 | 2017-09-20 12:02:26 -0700 | [diff] [blame] | 354 | Rect getBounds() const override { return mDevice->getBounds(); } |
| 355 | int getHeight() const override { return mDevice->getHeight(); } |
| 356 | int getWidth() const override { return mDevice->getWidth(); } |
| 357 | bool isSecure() const override { return mDevice->isSecure(); } |
Chia-I Wu | 5f6664c | 2018-08-28 11:01:44 -0700 | [diff] [blame] | 358 | |
| 359 | bool needsFiltering() const override { |
Chia-I Wu | 8730ba8 | 2018-08-29 09:25:59 -0700 | [diff] [blame] | 360 | // check if the projection from the logical display to the physical |
| 361 | // display needs filtering |
Chia-I Wu | 5f6664c | 2018-08-28 11:01:44 -0700 | [diff] [blame] | 362 | if (mDevice->needsFiltering()) { |
| 363 | return true; |
| 364 | } |
| 365 | |
Chia-I Wu | 8730ba8 | 2018-08-29 09:25:59 -0700 | [diff] [blame] | 366 | // check if the projection from the logical render area (i.e., the |
| 367 | // physical display) to the physical render area requires filtering |
Chia-I Wu | 5f6664c | 2018-08-28 11:01:44 -0700 | [diff] [blame] | 368 | const Rect sourceCrop = getSourceCrop(); |
| 369 | int width = sourceCrop.width(); |
| 370 | int height = sourceCrop.height(); |
| 371 | if (getRotationFlags() & ui::Transform::ROT_90) { |
| 372 | std::swap(width, height); |
| 373 | } |
| 374 | return width != getReqWidth() || height != getReqHeight(); |
| 375 | } |
Chia-I Wu | f2aa311 | 2018-08-27 14:54:37 -0700 | [diff] [blame] | 376 | |
| 377 | Rect getSourceCrop() const override { |
Chia-I Wu | 8730ba8 | 2018-08-29 09:25:59 -0700 | [diff] [blame] | 378 | // use the (projected) logical display viewport by default |
| 379 | if (mSourceCrop.isEmpty()) { |
| 380 | return mDevice->getScissor(); |
| 381 | } |
| 382 | |
Chia-I Wu | f2aa311 | 2018-08-27 14:54:37 -0700 | [diff] [blame] | 383 | const int orientation = mDevice->getInstallOrientation(); |
| 384 | if (orientation == DisplayState::eOrientationDefault) { |
| 385 | return mSourceCrop; |
| 386 | } |
| 387 | |
Chia-I Wu | 8730ba8 | 2018-08-29 09:25:59 -0700 | [diff] [blame] | 388 | // Install orientation is transparent to the callers. Apply it now. |
Chia-I Wu | f2aa311 | 2018-08-27 14:54:37 -0700 | [diff] [blame] | 389 | uint32_t flags = 0x00; |
| 390 | switch (orientation) { |
| 391 | case DisplayState::eOrientation90: |
| 392 | flags = ui::Transform::ROT_90; |
| 393 | break; |
| 394 | case DisplayState::eOrientation180: |
| 395 | flags = ui::Transform::ROT_180; |
| 396 | break; |
| 397 | case DisplayState::eOrientation270: |
| 398 | flags = ui::Transform::ROT_270; |
| 399 | break; |
| 400 | } |
| 401 | ui::Transform tr; |
| 402 | tr.set(flags, getWidth(), getHeight()); |
| 403 | return tr.transform(mSourceCrop); |
| 404 | } |
chaviw | a76b271 | 2017-09-20 12:02:26 -0700 | [diff] [blame] | 405 | |
| 406 | private: |
Chia-I Wu | 8730ba8 | 2018-08-29 09:25:59 -0700 | [diff] [blame] | 407 | // Install orientation is transparent to the callers. We need to cancel |
| 408 | // it out by modifying rotation flags. |
Chia-I Wu | f2aa311 | 2018-08-27 14:54:37 -0700 | [diff] [blame] | 409 | static ui::Transform::orientation_flags getDisplayRotation( |
| 410 | ui::Transform::orientation_flags rotation, int orientation) { |
| 411 | if (orientation == DisplayState::eOrientationDefault) { |
| 412 | return rotation; |
| 413 | } |
| 414 | |
| 415 | // convert hw orientation into flag presentation |
| 416 | // here inverse transform needed |
| 417 | uint8_t hw_rot_90 = 0x00; |
| 418 | uint8_t hw_flip_hv = 0x00; |
| 419 | switch (orientation) { |
| 420 | case DisplayState::eOrientation90: |
| 421 | hw_rot_90 = ui::Transform::ROT_90; |
| 422 | hw_flip_hv = ui::Transform::ROT_180; |
| 423 | break; |
| 424 | case DisplayState::eOrientation180: |
| 425 | hw_flip_hv = ui::Transform::ROT_180; |
| 426 | break; |
| 427 | case DisplayState::eOrientation270: |
| 428 | hw_rot_90 = ui::Transform::ROT_90; |
| 429 | break; |
| 430 | } |
| 431 | |
| 432 | // transform flags operation |
| 433 | // 1) flip H V if both have ROT_90 flag |
| 434 | // 2) XOR these flags |
| 435 | uint8_t rotation_rot_90 = rotation & ui::Transform::ROT_90; |
| 436 | uint8_t rotation_flip_hv = rotation & ui::Transform::ROT_180; |
| 437 | if (rotation_rot_90 & hw_rot_90) { |
| 438 | rotation_flip_hv = (~rotation_flip_hv) & ui::Transform::ROT_180; |
| 439 | } |
| 440 | |
| 441 | return static_cast<ui::Transform::orientation_flags>( |
| 442 | (rotation_rot_90 ^ hw_rot_90) | (rotation_flip_hv ^ hw_flip_hv)); |
| 443 | } |
| 444 | |
chaviw | a76b271 | 2017-09-20 12:02:26 -0700 | [diff] [blame] | 445 | const sp<const DisplayDevice> mDevice; |
| 446 | const Rect mSourceCrop; |
| 447 | }; |
| 448 | |
The Android Open Source Project | edbf3b6 | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 449 | }; // namespace android |
| 450 | |
Mathias Agopian | d3ee231 | 2012-08-02 14:01:42 -0700 | [diff] [blame] | 451 | #endif // ANDROID_DISPLAY_DEVICE_H |