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