blob: d0ca78e658b74b12c5b053ecbe7decb1e5445bf9 [file] [log] [blame]
Alec Mouri671d0f52019-09-05 13:59:19 -07001/*
2 * Copyright 2019 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include <apex/display.h>
18#include <gui/SurfaceComposerClient.h>
Marin Shalamanova7fe3042021-01-29 21:02:08 +010019#include <ui/DisplayMode.h>
Marin Shalamanov228f46b2021-01-28 21:11:45 +010020#include <ui/DynamicDisplayInfo.h>
Alec Mouri671d0f52019-09-05 13:59:19 -070021#include <ui/GraphicTypes.h>
Alec Mouri46170512019-11-20 11:04:55 -080022#include <ui/PixelFormat.h>
Marin Shalamanov228f46b2021-01-28 21:11:45 +010023#include <ui/StaticDisplayInfo.h>
Alec Mouri671d0f52019-09-05 13:59:19 -070024
25#include <algorithm>
26#include <optional>
27#include <type_traits>
28#include <vector>
29
30namespace android::display::impl {
31
32/**
33 * Implementation of ADisplayConfig
34 */
35struct DisplayConfigImpl {
36 /**
Marin Shalamanov228f46b2021-01-28 21:11:45 +010037 * The ID of the display configuration.
38 */
39 size_t id;
40
41 /**
Alec Mouri671d0f52019-09-05 13:59:19 -070042 * The width in pixels of the display configuration.
43 */
44 int32_t width{0};
45
46 /**
47 * The height in pixels of the display configuration.
48 */
49
50 int32_t height{0};
51
52 /**
Alec Mouri671d0f52019-09-05 13:59:19 -070053 * The refresh rate of the display configuration, in frames per second.
54 */
55 float fps{0.0};
56
57 /**
58 * The vsync offset at which surfaceflinger runs, in nanoseconds.
59 */
60 int64_t sfOffset{0};
61
62 /**
63 * The vsync offset at which applications run, in nanoseconds.
64 */
65 int64_t appOffset{0};
66};
67
68// DisplayConfigImpl allocation is not managed through C++ memory apis, so
69// preventing calling the destructor here.
70static_assert(std::is_trivially_destructible<DisplayConfigImpl>::value);
71
72/**
73 * Implementation of ADisplay
74 */
75struct DisplayImpl {
76 /**
77 * A physical display ID, unique to this display.
78 */
79 PhysicalDisplayId id;
80
81 /**
82 * The type of the display, i.e. whether it is an internal or external
83 * display.
84 */
85 ADisplayType type;
86
87 /**
Alec Mouri46170512019-11-20 11:04:55 -080088 * The preferred WCG dataspace
89 */
90 ADataSpace wcgDataspace;
91
92 /**
93 * The preferred WCG pixel format
94 */
95 AHardwareBuffer_Format wcgPixelFormat;
96
97 /**
Alec Mouri671d0f52019-09-05 13:59:19 -070098 * Number of supported configs
99 */
100 size_t numConfigs;
101
102 /**
103 * Set of supported configs by this display.
104 */
105 DisplayConfigImpl* configs;
106};
107
108// DisplayImpl allocation is not managed through C++ memory apis, so
109// preventing calling the destructor here.
110static_assert(std::is_trivially_destructible<DisplayImpl>::value);
111
112} // namespace android::display::impl
113
114using namespace android;
115using namespace android::display::impl;
116
117#define CHECK_NOT_NULL(name) \
118 LOG_ALWAYS_FATAL_IF(name == nullptr, "nullptr passed as " #name " argument");
119
Alec Mourid9ff3272019-11-19 16:23:59 -0800120namespace android {
121
Alec Mouri671d0f52019-09-05 13:59:19 -0700122int ADisplay_acquirePhysicalDisplays(ADisplay*** outDisplays) {
123 const std::vector<PhysicalDisplayId> ids = SurfaceComposerClient::getPhysicalDisplayIds();
124 const size_t size = ids.size();
125 if (size == 0) {
126 return NO_INIT;
127 }
128
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100129 std::vector<DisplayConfigImpl> modesPerDisplay[size];
Huihong Luo31b5ac22022-08-15 20:38:10 -0700130 ui::DisplayConnectionType displayConnectionTypes[size];
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100131 int numModes = 0;
Jim Shargo2acc6b72024-08-29 16:35:06 +0000132 for (size_t i = 0; i < size; ++i) {
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100133 ui::StaticDisplayInfo staticInfo;
Sally Qi6bb12822022-10-05 11:42:30 -0700134 if (const status_t status =
135 SurfaceComposerClient::getStaticDisplayInfo(ids[i].value, &staticInfo);
Dominik Laskowski3cb3d4e2019-11-21 11:14:45 -0800136 status != OK) {
137 return status;
138 }
Huihong Luo31b5ac22022-08-15 20:38:10 -0700139 displayConnectionTypes[i] = staticInfo.connectionType;
Dominik Laskowski3cb3d4e2019-11-21 11:14:45 -0800140
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100141 ui::DynamicDisplayInfo dynamicInfo;
142 if (const status_t status =
Sally Qi6bb12822022-10-05 11:42:30 -0700143 SurfaceComposerClient::getDynamicDisplayInfoFromId(ids[i].value, &dynamicInfo);
Dominik Laskowski3cb3d4e2019-11-21 11:14:45 -0800144 status != OK) {
Alec Mouri671d0f52019-09-05 13:59:19 -0700145 return status;
146 }
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100147 const auto& modes = dynamicInfo.supportedDisplayModes;
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100148 if (modes.empty()) {
Alec Mouri671d0f52019-09-05 13:59:19 -0700149 return NO_INIT;
150 }
151
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100152 numModes += modes.size();
153 modesPerDisplay[i].reserve(modes.size());
Jim Shargo2acc6b72024-08-29 16:35:06 +0000154 for (size_t j = 0; j < modes.size(); ++j) {
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100155 const ui::DisplayMode& mode = modes[j];
156 modesPerDisplay[i].emplace_back(
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100157 DisplayConfigImpl{static_cast<size_t>(mode.id), mode.resolution.getWidth(),
Alec Mouri55e31032023-10-02 20:34:18 +0000158 mode.resolution.getHeight(), mode.peakRefreshRate,
Brian Lindahl9b416b02022-01-12 10:57:39 +0100159 mode.sfVsyncOffset, mode.appVsyncOffset});
Alec Mouri671d0f52019-09-05 13:59:19 -0700160 }
161 }
162
Alec Mouri46170512019-11-20 11:04:55 -0800163 ui::Dataspace defaultDataspace;
164 ui::PixelFormat defaultPixelFormat;
165 ui::Dataspace wcgDataspace;
166 ui::PixelFormat wcgPixelFormat;
167
168 const status_t status =
169 SurfaceComposerClient::getCompositionPreference(&defaultDataspace, &defaultPixelFormat,
170 &wcgDataspace, &wcgPixelFormat);
171 if (status != NO_ERROR) {
172 return status;
173 }
Alec Mouri671d0f52019-09-05 13:59:19 -0700174
175 // Here we allocate all our required memory in one block. The layout is as
176 // follows:
177 // ------------------------------------------------------------
178 // | DisplayImpl pointers | DisplayImpls | DisplayConfigImpls |
179 // ------------------------------------------------------------
180 //
181 // The caller will be given a DisplayImpl** which points to the beginning of
182 // the block of DisplayImpl pointers.
183 // Each DisplayImpl* points to a DisplayImpl in the second block.
184 // Each DisplayImpl contains a DisplayConfigImpl*, which points to a
185 // contiguous block of DisplayConfigImpls specific to that display.
186 DisplayImpl** const impls = reinterpret_cast<DisplayImpl**>(
187 malloc((sizeof(DisplayImpl) + sizeof(DisplayImpl*)) * size +
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100188 sizeof(DisplayConfigImpl) * numModes));
Alec Mouri671d0f52019-09-05 13:59:19 -0700189 DisplayImpl* const displayData = reinterpret_cast<DisplayImpl*>(impls + size);
190 DisplayConfigImpl* configData = reinterpret_cast<DisplayConfigImpl*>(displayData + size);
191
192 for (size_t i = 0; i < size; ++i) {
193 const PhysicalDisplayId id = ids[i];
Huihong Luo31b5ac22022-08-15 20:38:10 -0700194 const ADisplayType type = (displayConnectionTypes[i] == ui::DisplayConnectionType::Internal)
195 ? ADisplayType::DISPLAY_TYPE_INTERNAL
196 : ADisplayType::DISPLAY_TYPE_EXTERNAL;
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100197 const std::vector<DisplayConfigImpl>& configs = modesPerDisplay[i];
Alec Mouri671d0f52019-09-05 13:59:19 -0700198 memcpy(configData, configs.data(), sizeof(DisplayConfigImpl) * configs.size());
199
Alec Mouri46170512019-11-20 11:04:55 -0800200 displayData[i] = DisplayImpl{id,
201 type,
202 static_cast<ADataSpace>(wcgDataspace),
203 static_cast<AHardwareBuffer_Format>(wcgPixelFormat),
204 configs.size(),
205 configData};
Alec Mouri671d0f52019-09-05 13:59:19 -0700206 impls[i] = displayData + i;
207 // Advance the configData pointer so that future configs are written to
208 // the correct display.
209 configData += configs.size();
210 }
211
212 *outDisplays = reinterpret_cast<ADisplay**>(impls);
213 return size;
214}
215
216void ADisplay_release(ADisplay** displays) {
217 if (displays == nullptr) {
218 return;
219 }
220 free(displays);
221}
222
223float ADisplay_getMaxSupportedFps(ADisplay* display) {
224 CHECK_NOT_NULL(display);
225 DisplayImpl* impl = reinterpret_cast<DisplayImpl*>(display);
226 float maxFps = 0.0;
Jim Shargo2acc6b72024-08-29 16:35:06 +0000227 for (size_t i = 0; i < impl->numConfigs; ++i) {
Alec Mouri671d0f52019-09-05 13:59:19 -0700228 maxFps = std::max(maxFps, impl->configs[i].fps);
229 }
230 return maxFps;
231}
232
233ADisplayType ADisplay_getDisplayType(ADisplay* display) {
234 CHECK_NOT_NULL(display);
235
236 return reinterpret_cast<DisplayImpl*>(display)->type;
237}
238
Alec Mouri46170512019-11-20 11:04:55 -0800239void ADisplay_getPreferredWideColorFormat(ADisplay* display, ADataSpace* outDataspace,
240 AHardwareBuffer_Format* outPixelFormat) {
241 CHECK_NOT_NULL(display);
242 CHECK_NOT_NULL(outDataspace);
243 CHECK_NOT_NULL(outPixelFormat);
244
245 DisplayImpl* impl = reinterpret_cast<DisplayImpl*>(display);
246 *outDataspace = impl->wcgDataspace;
247 *outPixelFormat = impl->wcgPixelFormat;
248}
249
Alec Mouri671d0f52019-09-05 13:59:19 -0700250int ADisplay_getCurrentConfig(ADisplay* display, ADisplayConfig** outConfig) {
251 CHECK_NOT_NULL(display);
252
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100253 ui::DynamicDisplayInfo info;
Sally Qi6bb12822022-10-05 11:42:30 -0700254 DisplayImpl* impl = reinterpret_cast<DisplayImpl*>(display);
255
256 if (const auto status =
257 SurfaceComposerClient::getDynamicDisplayInfoFromId(impl->id.value, &info);
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100258 status != OK) {
259 return status;
Alec Mouri671d0f52019-09-05 13:59:19 -0700260 }
261
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100262 for (size_t i = 0; i < impl->numConfigs; i++) {
263 auto* config = impl->configs + i;
Jim Shargo2acc6b72024-08-29 16:35:06 +0000264 if (info.activeDisplayModeId >= 0 && config->id == (size_t)info.activeDisplayModeId) {
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100265 *outConfig = reinterpret_cast<ADisplayConfig*>(config);
266 return OK;
267 }
268 }
Alec Mouri671d0f52019-09-05 13:59:19 -0700269
Marin Shalamanov228f46b2021-01-28 21:11:45 +0100270 return NAME_NOT_FOUND;
Alec Mouri671d0f52019-09-05 13:59:19 -0700271}
272
Alec Mouri671d0f52019-09-05 13:59:19 -0700273int32_t ADisplayConfig_getWidth(ADisplayConfig* config) {
274 CHECK_NOT_NULL(config);
275
276 return reinterpret_cast<DisplayConfigImpl*>(config)->width;
277}
278
279int32_t ADisplayConfig_getHeight(ADisplayConfig* config) {
280 CHECK_NOT_NULL(config);
281
282 return reinterpret_cast<DisplayConfigImpl*>(config)->height;
283}
284
285float ADisplayConfig_getFps(ADisplayConfig* config) {
286 CHECK_NOT_NULL(config);
287
288 return reinterpret_cast<DisplayConfigImpl*>(config)->fps;
289}
290
291int64_t ADisplayConfig_getCompositorOffsetNanos(ADisplayConfig* config) {
292 CHECK_NOT_NULL(config);
293
294 return reinterpret_cast<DisplayConfigImpl*>(config)->sfOffset;
295}
296
297int64_t ADisplayConfig_getAppVsyncOffsetNanos(ADisplayConfig* config) {
298 CHECK_NOT_NULL(config);
299
300 return reinterpret_cast<DisplayConfigImpl*>(config)->appOffset;
301}
Alec Mourid9ff3272019-11-19 16:23:59 -0800302
303} // namespace android