blob: 12a5fea22f94923ae82aa7bc2687fbe568e410f3 [file] [log] [blame]
Matvii Zorinef3c7972020-08-11 15:15:44 +03001/*
2 * Copyright (C) 2020 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
Roman Stratiienko13cc3662020-08-29 21:35:39 +030017#include "Backend.h"
18
Roman Stratiienkod21071f2021-03-09 21:56:50 +020019#include <climits>
20
Roman Stratiienko13cc3662020-08-29 21:35:39 +030021#include "BackendManager.h"
Roman Stratiienkob2e9fe22020-10-03 10:52:36 +030022#include "bufferinfo/BufferInfoGetter.h"
Andrew Wolfers5c530832024-12-03 16:30:26 +000023#include "hardware/hwcomposer2.h"
Matvii Zorinef3c7972020-08-11 15:15:44 +030024
25namespace android {
26
Andrew Wolfers5c530832024-12-03 16:30:26 +000027namespace {
28
29bool HasCursorLayer(const std::vector<HwcLayer *> &layers) {
30 return std::find_if(layers.begin(), layers.end(), [&](auto *layer) -> bool {
31 return layer->GetSfType() == HWC2::Composition::Cursor;
32 }) != layers.end();
33}
34
35} // namespace
36
Roman Stratiienko3627beb2022-01-04 16:02:55 +020037HWC2::Error Backend::ValidateDisplay(HwcDisplay *display, uint32_t *num_types,
Matvii Zorinef3c7972020-08-11 15:15:44 +030038 uint32_t *num_requests) {
39 *num_types = 0;
40 *num_requests = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030041
Matvii Zorined90ef92021-01-29 18:32:06 +020042 auto layers = display->GetOrderLayersByZPos();
Matvii Zorinef3c7972020-08-11 15:15:44 +030043
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020044 int client_start = -1;
45 size_t client_size = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030046
Roman Stratiienko22fe9612023-01-17 21:22:29 +020047 auto flatcon = display->GetFlatCon();
48 if (flatcon) {
49 bool should_flatten = false;
50 if (layers.size() <= 1)
51 flatcon->Disable();
52 else
53 should_flatten = flatcon->NewFrame();
54
55 if (should_flatten) {
56 display->total_stats().frames_flattened_++;
57 MarkValidated(layers, 0, layers.size());
58 *num_types = layers.size();
59 return HWC2::Error::HasChanges;
60 }
61 }
62
63 std::tie(client_start, client_size) = GetClientLayers(display, layers);
64
65 MarkValidated(layers, client_start, client_size);
66
67 auto testing_needed = client_start != 0 || client_size != layers.size();
68
69 AtomicCommitArgs a_args = {.test_only = true};
70
71 if (testing_needed &&
72 display->CreateComposition(a_args) != HWC2::Error::None) {
73 ++display->total_stats().failed_kms_validate_;
Matvii Zorinef3c7972020-08-11 15:15:44 +030074 client_start = 0;
Matvii Zorined90ef92021-01-29 18:32:06 +020075 client_size = layers.size();
Andrew Wolfers5c530832024-12-03 16:30:26 +000076
77 // Expand the client range to include all layers except the cursor layer (if
78 // there is one) and retry.
79 auto [_, cursor_plane] = display->GetPipe().GetUsablePlanes();
80 if (cursor_plane && HasCursorLayer(layers)) {
81 --client_size;
82 MarkValidated(layers, 0, client_size);
83
84 testing_needed = display->CreateComposition(a_args) != HWC2::Error::None;
85
86 // If testing is still needed, expand the client range to include the
87 // cursor layer for the next retry.
88 if (testing_needed) {
89 ++client_size;
90 ++display->total_stats().failed_kms_validate_;
91 }
92 }
93
94 if (testing_needed) {
95 MarkValidated(layers, 0, client_size);
96 }
Matvii Zorinef3c7972020-08-11 15:15:44 +030097 }
98
99 *num_types = client_size;
100
Matvii Zorined90ef92021-01-29 18:32:06 +0200101 display->total_stats().gpu_pixops_ += CalcPixOps(layers, client_start,
Matvii Zorinb3b15162021-01-29 15:48:20 +0200102 client_size);
Matvii Zorined90ef92021-01-29 18:32:06 +0200103 display->total_stats().total_pixops_ += CalcPixOps(layers, 0, layers.size());
Matvii Zorinef3c7972020-08-11 15:15:44 +0300104
Roman Stratiienkofc014f52021-12-23 19:04:29 +0200105 return *num_types != 0 ? HWC2::Error::HasChanges : HWC2::Error::None;
Matvii Zorinef3c7972020-08-11 15:15:44 +0300106}
107
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +0200108std::tuple<int, size_t> Backend::GetClientLayers(
Roman Stratiienko3627beb2022-01-04 16:02:55 +0200109 HwcDisplay *display, const std::vector<HwcLayer *> &layers) {
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +0200110 int client_start = -1;
111 size_t client_size = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +0300112
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200113 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorined90ef92021-01-29 18:32:06 +0200114 if (IsClientLayer(display, layers[z_order])) {
Matvii Zorinef3c7972020-08-11 15:15:44 +0300115 if (client_start < 0)
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +0200116 client_start = (int)z_order;
Matvii Zorinef3c7972020-08-11 15:15:44 +0300117 client_size = (z_order - client_start) + 1;
118 }
119 }
120
Matvii Zorined90ef92021-01-29 18:32:06 +0200121 return GetExtraClientRange(display, layers, client_start, client_size);
Matvii Zorinef3c7972020-08-11 15:15:44 +0300122}
123
Roman Stratiienko3627beb2022-01-04 16:02:55 +0200124bool Backend::IsClientLayer(HwcDisplay *display, HwcLayer *layer) {
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200125 return !HardwareSupportsLayerType(layer->GetSfType()) ||
Roman Stratiienko0da91bf2023-01-17 18:06:04 +0200126 !layer->IsLayerUsableAsDevice() || display->CtmByGpu() ||
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200127 (layer->GetLayerData().pi.RequireScalingOrPhasing() &&
Drew Davenport93443182023-12-14 09:25:45 +0000128 display->GetHwc()->GetResMan().ForcedScalingWithGpu());
Matvii Zorinef3c7972020-08-11 15:15:44 +0300129}
130
Matvii Zorinede54b02021-01-29 16:45:25 +0200131bool Backend::HardwareSupportsLayerType(HWC2::Composition comp_type) {
132 return comp_type == HWC2::Composition::Device ||
133 comp_type == HWC2::Composition::Cursor;
134}
135
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200136uint32_t Backend::CalcPixOps(const std::vector<HwcLayer *> &layers,
Matvii Zorin2c91df52021-01-29 15:39:55 +0200137 size_t first_z, size_t size) {
138 uint32_t pixops = 0;
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200139 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorin2c91df52021-01-29 15:39:55 +0200140 if (z_order >= first_z && z_order < first_z + size) {
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300141 auto &df = layers[z_order]->GetLayerData().pi.display_frame;
Roman Stratiienko4e15bfc2025-01-23 01:55:21 +0200142 if (df.i_rect) {
143 pixops += (df.i_rect->right - df.i_rect->left) *
144 (df.i_rect->bottom - df.i_rect->top);
145 }
Matvii Zorin2c91df52021-01-29 15:39:55 +0200146 }
147 }
148 return pixops;
149}
150
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200151void Backend::MarkValidated(std::vector<HwcLayer *> &layers,
Matvii Zorined745272021-01-29 16:40:25 +0200152 size_t client_first_z, size_t client_size) {
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200153 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Andrew Wolfers5c530832024-12-03 16:30:26 +0000154 if (z_order >= client_first_z && z_order < client_first_z + client_size) {
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200155 layers[z_order]->SetValidatedType(HWC2::Composition::Client);
Andrew Wolfers5c530832024-12-03 16:30:26 +0000156 } else if (layers[z_order]->GetSfType() == HWC2::Composition::Cursor) {
157 layers[z_order]->SetValidatedType(HWC2::Composition::Cursor);
158 } else {
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200159 layers[z_order]->SetValidatedType(HWC2::Composition::Device);
Andrew Wolfers5c530832024-12-03 16:30:26 +0000160 }
Matvii Zorined745272021-01-29 16:40:25 +0200161 }
162}
163
Matvii Zorinb3b15162021-01-29 15:48:20 +0200164std::tuple<int, int> Backend::GetExtraClientRange(
Roman Stratiienko3627beb2022-01-04 16:02:55 +0200165 HwcDisplay *display, const std::vector<HwcLayer *> &layers,
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200166 int client_start, size_t client_size) {
Andrew Wolfers7979d2a2025-02-27 14:44:41 +0000167 auto [planes, cursor_plane] = display->GetPipe().GetUsablePlanes();
Roman Stratiienko9362cef2022-02-02 09:53:50 +0200168 size_t avail_planes = planes.size();
Andrew Wolfers5c530832024-12-03 16:30:26 +0000169 size_t layers_size = layers.size();
170
171 // |cursor_plane| is not counted among |avail_planes|, so the cursor layer
172 // shouldn't be counted in |layers_size|.
173 if (cursor_plane && HasCursorLayer(layers)) {
174 --layers_size;
175 }
Matvii Zorinb3b15162021-01-29 15:48:20 +0200176
177 /*
Andrew Wolfers5c530832024-12-03 16:30:26 +0000178 * If more layers than planes, save one plane
Matvii Zorinb3b15162021-01-29 15:48:20 +0200179 * for client composited layers
180 */
Andrew Wolfers5c530832024-12-03 16:30:26 +0000181 if (avail_planes < layers_size) {
Matvii Zorinb3b15162021-01-29 15:48:20 +0200182 avail_planes--;
Andrew Wolfers5c530832024-12-03 16:30:26 +0000183 }
Matvii Zorinb3b15162021-01-29 15:48:20 +0200184
Andrew Wolfers5c530832024-12-03 16:30:26 +0000185 const int extra_client = int(layers_size - client_size) - int(avail_planes);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200186
187 if (extra_client > 0) {
188 int start = 0;
189 size_t steps = 0;
190 if (client_size != 0) {
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300191 const int prepend = std::min(client_start, extra_client);
Andrew Wolfers5c530832024-12-03 16:30:26 +0000192 const int append = std::min(int(layers_size) -
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300193 int(client_start + client_size),
194 extra_client);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200195 start = client_start - (int)prepend;
196 client_size += extra_client;
197 steps = 1 + std::min(std::min(append, prepend),
Andrew Wolfers5c530832024-12-03 16:30:26 +0000198 int(layers_size) - int(start + client_size));
Matvii Zorinb3b15162021-01-29 15:48:20 +0200199 } else {
200 client_size = extra_client;
Andrew Wolfers5c530832024-12-03 16:30:26 +0000201 steps = 1 + layers_size - extra_client;
Matvii Zorinb3b15162021-01-29 15:48:20 +0200202 }
203
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200204 uint32_t gpu_pixops = UINT32_MAX;
205 for (size_t i = 0; i < steps; i++) {
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300206 const uint32_t po = CalcPixOps(layers, start + i, client_size);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200207 if (po < gpu_pixops) {
208 gpu_pixops = po;
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200209 client_start = start + int(i);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200210 }
211 }
212 }
213
214 return std::make_tuple(client_start, client_size);
215}
216
Roman Stratiienkod21071f2021-03-09 21:56:50 +0200217// clang-format off
218// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables, cert-err58-cpp)
Matvii Zorinef3c7972020-08-11 15:15:44 +0300219REGISTER_BACKEND("generic", Backend);
Roman Stratiienkod21071f2021-03-09 21:56:50 +0200220// clang-format on
Matvii Zorinef3c7972020-08-11 15:15:44 +0300221
222} // namespace android