blob: 960fe8afa1e7fdb24ada8b5aefd2c31e3efd1410 [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
Mathias Agopiand3ee2312012-08-02 14:01:42 -070017#ifndef ANDROID_DISPLAY_DEVICE_H
18#define ANDROID_DISPLAY_DEVICE_H
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080019
Dan Stoza9e56aa02015-11-02 13:00:03 -080020#include "Transform.h"
21
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080022#include <stdlib.h>
23
Yiwei Zhang7c64f172018-03-07 14:52:28 -080024#include <math/mat4.h>
25
Dan Stoza9e56aa02015-11-02 13:00:03 -080026#include <binder/IBinder.h>
Peiyong Lin0ac5f4e2018-04-19 22:06:34 -070027#include <gui/ISurfaceComposer.h>
28#include <hardware/hwcomposer_defs.h>
29#include <ui/GraphicTypes.h>
30#include <ui/Region.h>
Dan Stoza9e56aa02015-11-02 13:00:03 -080031#include <utils/RefBase.h>
Mathias Agopian86303202012-07-24 22:46:10 -070032#include <utils/Mutex.h>
Mathias Agopian4f4f0942013-08-19 17:26:18 -070033#include <utils/String8.h>
Mathias Agopian86303202012-07-24 22:46:10 -070034#include <utils/Timers.h>
35
chaviwa76b2712017-09-20 12:02:26 -070036#include "RenderArea.h"
Chia-I Wuf846a352017-11-10 09:22:52 -080037#include "RenderEngine/Surface.h"
Mathias Agopianf4358632012-08-22 17:16:19 -070038
Dan Stoza9e56aa02015-11-02 13:00:03 -080039#include <memory>
Dominik Laskowskibf170d92018-04-19 15:08:05 -070040#include <string>
Mathias Agopian1f7bec62010-06-25 18:02:21 -070041
Mathias Agopiand8552d72012-08-04 21:39:11 -070042struct ANativeWindow;
43
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080044namespace android {
45
Jesse Hall646f5412014-08-07 22:19:07 -070046struct DisplayInfo;
Jesse Hall99c7dbb2013-03-14 14:29:29 -070047class DisplaySurface;
Dan Stoza9e56aa02015-11-02 13:00:03 -080048class Fence;
Jesse Hall9e663de2013-08-16 14:28:37 -070049class IGraphicBufferProducer;
Mathias Agopian13127d82013-03-05 17:47:11 -080050class Layer;
Mathias Agopian86303202012-07-24 22:46:10 -070051class SurfaceFlinger;
Mathias Agopianda27af92012-09-13 18:17:13 -070052class HWComposer;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080053
Mathias Agopian028a7572012-08-05 01:23:51 -070054class DisplayDevice : public LightRefBase<DisplayDevice>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080055{
56public:
Mathias Agopian87baae12012-07-31 12:38:26 -070057 // region in layer-stack space
58 mutable Region dirtyRegion;
59 // region in screen space
Mathias Agopian87baae12012-07-31 12:38:26 -070060 Region undefinedRegion;
Jesse Hallb7a05492014-08-14 15:45:06 -070061 bool lastCompositionHadVisibleLayers;
Mathias Agopian3eb38cb2012-04-03 22:09:52 -070062
Mathias Agopian3ee454a2012-08-27 16:28:24 -070063 enum DisplayType {
64 DISPLAY_ID_INVALID = -1,
65 DISPLAY_PRIMARY = HWC_DISPLAY_PRIMARY,
66 DISPLAY_EXTERNAL = HWC_DISPLAY_EXTERNAL,
Jesse Hall9e663de2013-08-16 14:28:37 -070067 DISPLAY_VIRTUAL = HWC_DISPLAY_VIRTUAL,
68 NUM_BUILTIN_DISPLAY_TYPES = HWC_NUM_PHYSICAL_DISPLAY_TYPES,
Mathias Agopian92a979a2012-08-02 18:32:23 -070069 };
70
71 enum {
Jesse Hall01e29052013-02-19 16:13:35 -080072 NO_LAYER_STACK = 0xFFFFFFFF,
73 };
74
Courtney Goeltzenleuchter5d943892017-03-22 13:46:46 -060075 // clang-format off
Mathias Agopian0f2f5ff2012-07-31 23:09:07 -070076 DisplayDevice(
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080077 const sp<SurfaceFlinger>& flinger,
Jamie Gennisdd3cb842012-10-19 18:19:11 -070078 DisplayType type,
Dan Stoza9e56aa02015-11-02 13:00:03 -080079 int32_t hwcId,
Jamie Gennisdd3cb842012-10-19 18:19:11 -070080 bool isSecure,
81 const wp<IBinder>& displayToken,
Lloyd Pique09594832018-01-22 17:48:03 -080082 const sp<ANativeWindow>& nativeWindow,
Jesse Hall99c7dbb2013-03-14 14:29:29 -070083 const sp<DisplaySurface>& displaySurface,
Lloyd Pique09594832018-01-22 17:48:03 -080084 std::unique_ptr<RE::Surface> renderSurface,
85 int displayWidth,
86 int displayHeight,
Peiyong Lindd9b2ae2018-03-01 16:22:45 -080087 bool hasWideColorGamut,
Peiyong Lin62665892018-04-16 11:07:44 -070088 const HdrCapabilities& hdrCapabilities,
Peiyong Lin0ac5f4e2018-04-19 22:06:34 -070089 const int32_t supportedPerFrameMetadata,
Lloyd Pique09594832018-01-22 17:48:03 -080090 int initialPowerMode);
Courtney Goeltzenleuchter5d943892017-03-22 13:46:46 -060091 // clang-format on
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080092
Mathias Agopian028a7572012-08-05 01:23:51 -070093 ~DisplayDevice();
Mathias Agopian92a979a2012-08-02 18:32:23 -070094
95 // whether this is a valid object. An invalid DisplayDevice is returned
96 // when an non existing id is requested
97 bool isValid() const;
98
Jamie Gennisdd3cb842012-10-19 18:19:11 -070099 // isSecure indicates whether this display can be trusted to display
100 // secure surfaces.
101 bool isSecure() const { return mIsSecure; }
102
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800103 // Flip the front and back buffers if the back buffer is "dirty". Might
104 // be instantaneous, might involve copying the frame buffer around.
Chia-I Wub02087d2017-11-09 10:19:54 -0800105 void flip() const;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800106
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800107 int getWidth() const;
108 int getHeight() const;
Mathias Agopian3eb38cb2012-04-03 22:09:52 -0700109
Mathias Agopian13127d82013-03-05 17:47:11 -0800110 void setVisibleLayersSortedByZ(const Vector< sp<Layer> >& layers);
111 const Vector< sp<Layer> >& getVisibleLayersSortedByZ() const;
Chia-I Wu83806892017-11-16 10:50:20 -0800112 void setLayersNeedingFences(const Vector< sp<Layer> >& layers);
113 const Vector< sp<Layer> >& getLayersNeedingFences() const;
Mathias Agopiancd60f992012-08-16 16:28:27 -0700114 Region getDirtyRegion(bool repaintEverything) const;
Mathias Agopian3b1d2b62012-07-11 13:48:17 -0700115
Mathias Agopian28947d72012-08-08 18:51:15 -0700116 void setLayerStack(uint32_t stack);
Michael Lentine47e45402014-07-18 15:34:25 -0700117 void setDisplaySize(const int newWidth, const int newHeight);
Mathias Agopian00e8c7a2012-09-04 19:30:46 -0700118 void setProjection(int orientation, const Rect& viewport, const Rect& frame);
Mathias Agopian28947d72012-08-08 18:51:15 -0700119
Mathias Agopian1b031492012-06-20 17:51:20 -0700120 int getOrientation() const { return mOrientation; }
Mathias Agopianc1c05de2013-09-17 23:45:22 -0700121 uint32_t getOrientationTransform() const;
Pablo Ceballos021623b2016-04-15 17:31:51 -0700122 static uint32_t getPrimaryDisplayOrientationTransform();
Mathias Agopian1b031492012-06-20 17:51:20 -0700123 const Transform& getTransform() const { return mGlobalTransform; }
Mathias Agopianf5f714a2013-02-26 16:54:05 -0800124 const Rect getViewport() const { return mViewport; }
125 const Rect getFrame() const { return mFrame; }
Mathias Agopian766dc492012-10-30 18:08:06 -0700126 const Rect& getScissor() const { return mScissor; }
Mathias Agopianeba8c682012-09-19 23:14:45 -0700127 bool needsFiltering() const { return mNeedsFiltering; }
Mathias Agopianda8d0a52012-09-04 15:05:38 -0700128
Mathias Agopian87baae12012-07-31 12:38:26 -0700129 uint32_t getLayerStack() const { return mLayerStack; }
Mathias Agopian3ee454a2012-08-27 16:28:24 -0700130 int32_t getDisplayType() const { return mType; }
Chia-I Wuab0c3192017-08-01 11:29:00 -0700131 bool isPrimary() const { return mType == DISPLAY_PRIMARY; }
Dominik Laskowski281644e2018-04-19 15:47:35 -0700132 bool isVirtual() const { return mType == DISPLAY_VIRTUAL; }
Mathias Agopiane60b0682012-08-21 23:34:09 -0700133 int32_t getHwcDisplayId() const { return mHwcDisplayId; }
Mathias Agopian3ee454a2012-08-27 16:28:24 -0700134 const wp<IBinder>& getDisplayToken() const { return mDisplayToken; }
Mathias Agopian87baae12012-07-31 12:38:26 -0700135
Peiyong Lin0ac5f4e2018-04-19 22:06:34 -0700136 int32_t getSupportedPerFrameMetadata() const { return mSupportedPerFrameMetadata; }
137
Dan Stoza71433162014-02-04 16:22:36 -0800138 // We pass in mustRecompose so we can keep VirtualDisplaySurface's state
139 // machine happy without actually queueing a buffer if nothing has changed
140 status_t beginFrame(bool mustRecompose) const;
Dan Stoza9e56aa02015-11-02 13:00:03 -0800141 status_t prepareFrame(HWComposer& hwc);
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800142 bool hasWideColorGamut() const { return mHasWideColorGamut; }
Peiyong Lin62665892018-04-16 11:07:44 -0700143 bool hasHDR10Support() const { return mHasHdr10; }
144 bool hasHLGSupport() const { return mHasHLG; }
145 bool hasDolbyVisionSupport() const { return mHasDolbyVision; }
Jesse Hall38efe862013-04-06 23:12:29 -0700146
Mathias Agopianda27af92012-09-13 18:17:13 -0700147 void swapBuffers(HWComposer& hwc) const;
Jesse Hall01e29052013-02-19 16:13:35 -0800148
Mathias Agopianda27af92012-09-13 18:17:13 -0700149 // called after h/w composer has completed its set() call
Dan Stoza9e56aa02015-11-02 13:00:03 -0800150 void onSwapBuffersCompleted() const;
Mathias Agopianda27af92012-09-13 18:17:13 -0700151
Mathias Agopian9c6e2972011-09-20 17:21:56 -0700152 Rect getBounds() const {
Mathias Agopian1b031492012-06-20 17:51:20 -0700153 return Rect(mDisplayWidth, mDisplayHeight);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800154 }
Mathias Agopian9c6e2972011-09-20 17:21:56 -0700155 inline Rect bounds() const { return getBounds(); }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800156
Dominik Laskowskibf170d92018-04-19 15:08:05 -0700157 void setDisplayName(const std::string& displayName);
158 const std::string& getDisplayName() const { return mDisplayName; }
Andy McFadden8dfa92f2012-09-17 18:27:17 -0700159
Chia-I Wuf846a352017-11-10 09:22:52 -0800160 bool makeCurrent() const;
Mathias Agopian875d8e12013-06-07 15:35:48 -0700161 void setViewportAndProjection() const;
Mathias Agopianbae92d02012-09-28 01:00:47 -0700162
Dan Stoza9e56aa02015-11-02 13:00:03 -0800163 const sp<Fence>& getClientTargetAcquireFence() const;
Dan Stoza9e56aa02015-11-02 13:00:03 -0800164
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700165 /* ------------------------------------------------------------------------
Prashant Malani2c9b11f2014-05-25 01:36:31 -0700166 * Display power mode management.
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700167 */
Prashant Malani2c9b11f2014-05-25 01:36:31 -0700168 int getPowerMode() const;
169 void setPowerMode(int mode);
170 bool isDisplayOn() const;
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700171
Peiyong Linfd997e02018-03-28 15:29:00 -0700172 ui::ColorMode getActiveColorMode() const;
173 void setActiveColorMode(ui::ColorMode mode);
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800174 ui::RenderIntent getActiveRenderIntent() const;
175 void setActiveRenderIntent(ui::RenderIntent renderIntent);
Yiwei Zhang7c64f172018-03-07 14:52:28 -0800176 android_color_transform_t getColorTransform() const;
177 void setColorTransform(const mat4& transform);
Peiyong Lin34beb7a2018-03-28 11:57:12 -0700178 void setCompositionDataSpace(ui::Dataspace dataspace);
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800179 ui::Dataspace getCompositionDataSpace() const;
Michael Wright28f24d02016-07-12 13:30:53 -0700180
Michael Lentine6c9e34a2014-07-14 13:48:55 -0700181 /* ------------------------------------------------------------------------
182 * Display active config management.
183 */
184 int getActiveConfig() const;
185 void setActiveConfig(int mode);
186
Jesse Hall02d86562013-03-25 14:43:23 -0700187 // release HWC resources (if any) for removable displays
188 void disconnect(HWComposer& hwc);
189
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700190 /* ------------------------------------------------------------------------
191 * Debugging
192 */
193 uint32_t getPageFlipCount() const;
Mathias Agopian74d211a2013-04-22 16:55:35 +0200194 void dump(String8& result) const;
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700195
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800196private:
Mathias Agopiana4912602012-07-12 14:25:33 -0700197 /*
198 * Constants, set during initialization
199 */
Mathias Agopianc7d14e22011-08-01 16:32:21 -0700200 sp<SurfaceFlinger> mFlinger;
Mathias Agopian3ee454a2012-08-27 16:28:24 -0700201 DisplayType mType;
Mathias Agopiane60b0682012-08-21 23:34:09 -0700202 int32_t mHwcDisplayId;
Mathias Agopian3ee454a2012-08-27 16:28:24 -0700203 wp<IBinder> mDisplayToken;
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700204
Mathias Agopiana4912602012-07-12 14:25:33 -0700205 // ANativeWindow this display is rendering into
Mathias Agopiand8552d72012-08-04 21:39:11 -0700206 sp<ANativeWindow> mNativeWindow;
Jesse Hall99c7dbb2013-03-14 14:29:29 -0700207 sp<DisplaySurface> mDisplaySurface;
Mathias Agopiana4912602012-07-12 14:25:33 -0700208
Lloyd Pique144e1162017-12-20 16:44:52 -0800209 std::unique_ptr<RE::Surface> mSurface;
Mathias Agopian1b031492012-06-20 17:51:20 -0700210 int mDisplayWidth;
211 int mDisplayHeight;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700212 mutable uint32_t mPageFlipCount;
Dominik Laskowskibf170d92018-04-19 15:08:05 -0700213 std::string mDisplayName;
Jamie Gennisdd3cb842012-10-19 18:19:11 -0700214 bool mIsSecure;
Mathias Agopian03e40722012-04-26 16:11:59 -0700215
Mathias Agopiana4912602012-07-12 14:25:33 -0700216 /*
217 * Can only accessed from the main thread, these members
218 * don't need synchronization.
219 */
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700220
Mathias Agopian3b1d2b62012-07-11 13:48:17 -0700221 // list of visible layers on that display
Mathias Agopian13127d82013-03-05 17:47:11 -0800222 Vector< sp<Layer> > mVisibleLayersSortedByZ;
Chia-I Wu83806892017-11-16 10:50:20 -0800223 // list of layers needing fences
224 Vector< sp<Layer> > mLayersNeedingFences;
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700225
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700226 /*
227 * Transaction state
228 */
Mathias Agopianda8d0a52012-09-04 15:05:38 -0700229 static status_t orientationToTransfrom(int orientation,
230 int w, int h, Transform* tr);
231
Fabien Sanglardf0c53d62017-03-03 18:58:50 -0800232 // The identifier of the active layer stack for this display. Several displays
233 // can use the same layer stack: A z-ordered group of layers (sometimes called
234 // "surfaces"). Any given layer can only be on a single layer stack.
Mathias Agopian87baae12012-07-31 12:38:26 -0700235 uint32_t mLayerStack;
Fabien Sanglardf0c53d62017-03-03 18:58:50 -0800236
Mathias Agopianda8d0a52012-09-04 15:05:38 -0700237 int mOrientation;
Pablo Ceballos021623b2016-04-15 17:31:51 -0700238 static uint32_t sPrimaryDisplayOrientation;
Mathias Agopian766dc492012-10-30 18:08:06 -0700239 // user-provided visible area of the layer stack
Mathias Agopianda8d0a52012-09-04 15:05:38 -0700240 Rect mViewport;
Mathias Agopian766dc492012-10-30 18:08:06 -0700241 // user-provided rectangle where mViewport gets mapped to
Mathias Agopianda8d0a52012-09-04 15:05:38 -0700242 Rect mFrame;
Mathias Agopian766dc492012-10-30 18:08:06 -0700243 // pre-computed scissor to apply to the display
244 Rect mScissor;
Mathias Agopianda8d0a52012-09-04 15:05:38 -0700245 Transform mGlobalTransform;
Mathias Agopianeba8c682012-09-19 23:14:45 -0700246 bool mNeedsFiltering;
Prashant Malani2c9b11f2014-05-25 01:36:31 -0700247 // Current power mode
248 int mPowerMode;
Michael Lentine6c9e34a2014-07-14 13:48:55 -0700249 // Current active config
250 int mActiveConfig;
Michael Wright28f24d02016-07-12 13:30:53 -0700251 // current active color mode
Peiyong Linfd997e02018-03-28 15:29:00 -0700252 ui::ColorMode mActiveColorMode;
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800253 // Current active render intent.
254 ui::RenderIntent mActiveRenderIntent;
255 ui::Dataspace mCompositionDataSpace;
Yiwei Zhang7c64f172018-03-07 14:52:28 -0800256 // Current color transform
257 android_color_transform_t mColorTransform;
Courtney Goeltzenleuchter5d943892017-03-22 13:46:46 -0600258
259 // Need to know if display is wide-color capable or not.
260 // Initialized by SurfaceFlinger when the DisplayDevice is created.
261 // Fed to RenderEngine during composition.
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800262 bool mHasWideColorGamut;
263 bool mHasHdr10;
Peiyong Lin62665892018-04-16 11:07:44 -0700264 bool mHasHLG;
265 bool mHasDolbyVision;
Peiyong Lin0ac5f4e2018-04-19 22:06:34 -0700266
267 const int32_t mSupportedPerFrameMetadata;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800268};
269
Irvelffc9efc2016-07-27 15:16:37 -0700270struct DisplayDeviceState {
Irvelffc9efc2016-07-27 15:16:37 -0700271 bool isValid() const { return type >= 0; }
Dominik Laskowski663bd282018-04-19 15:26:54 -0700272 bool isVirtual() const { return type >= DisplayDevice::DISPLAY_VIRTUAL; }
Irvelffc9efc2016-07-27 15:16:37 -0700273
Dominik Laskowski663bd282018-04-19 15:26:54 -0700274 int32_t sequenceId = sNextSequenceId++;
Irvelffc9efc2016-07-27 15:16:37 -0700275 DisplayDevice::DisplayType type = DisplayDevice::DISPLAY_ID_INVALID;
276 sp<IGraphicBufferProducer> surface;
277 uint32_t layerStack = DisplayDevice::NO_LAYER_STACK;
278 Rect viewport;
279 Rect frame;
280 uint8_t orientation = 0;
281 uint32_t width = 0;
282 uint32_t height = 0;
Dominik Laskowskibf170d92018-04-19 15:08:05 -0700283 std::string displayName;
Irvelffc9efc2016-07-27 15:16:37 -0700284 bool isSecure = false;
Dominik Laskowski663bd282018-04-19 15:26:54 -0700285
286private:
287 static std::atomic<int32_t> sNextSequenceId;
Irvelffc9efc2016-07-27 15:16:37 -0700288};
289
chaviwa76b2712017-09-20 12:02:26 -0700290class DisplayRenderArea : public RenderArea {
291public:
292 DisplayRenderArea(const sp<const DisplayDevice> device,
293 ISurfaceComposer::Rotation rotation = ISurfaceComposer::eRotateNone)
294 : DisplayRenderArea(device, device->getBounds(), device->getHeight(), device->getWidth(),
295 rotation) {}
296 DisplayRenderArea(const sp<const DisplayDevice> device, Rect sourceCrop, uint32_t reqHeight,
297 uint32_t reqWidth, ISurfaceComposer::Rotation rotation)
chaviw50da5042018-04-09 13:49:37 -0700298 : RenderArea(reqHeight, reqWidth, CaptureFill::OPAQUE, rotation), mDevice(device),
299 mSourceCrop(sourceCrop) {}
chaviwa76b2712017-09-20 12:02:26 -0700300
301 const Transform& getTransform() const override { return mDevice->getTransform(); }
302 Rect getBounds() const override { return mDevice->getBounds(); }
303 int getHeight() const override { return mDevice->getHeight(); }
304 int getWidth() const override { return mDevice->getWidth(); }
305 bool isSecure() const override { return mDevice->isSecure(); }
306 bool needsFiltering() const override { return mDevice->needsFiltering(); }
307 Rect getSourceCrop() const override { return mSourceCrop; }
Peiyong Lindd9b2ae2018-03-01 16:22:45 -0800308 bool getWideColorSupport() const override { return mDevice->hasWideColorGamut(); }
309 ui::Dataspace getDataSpace() const override {
310 return mDevice->getCompositionDataSpace();
chaviwa76b2712017-09-20 12:02:26 -0700311 }
chaviwa76b2712017-09-20 12:02:26 -0700312
313private:
314 const sp<const DisplayDevice> mDevice;
315 const Rect mSourceCrop;
316};
317
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800318}; // namespace android
319
Mathias Agopiand3ee2312012-08-02 14:01:42 -0700320#endif // ANDROID_DISPLAY_DEVICE_H