blob: 3ca6e92dc585105ffc91b8b918953c01b1f9dfaf [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"
Matvii Zorinef3c7972020-08-11 15:15:44 +030023
24namespace android {
25
Roman Stratiienko3627beb2022-01-04 16:02:55 +020026HWC2::Error Backend::ValidateDisplay(HwcDisplay *display, uint32_t *num_types,
Matvii Zorinef3c7972020-08-11 15:15:44 +030027 uint32_t *num_requests) {
28 *num_types = 0;
29 *num_requests = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030030
Matvii Zorined90ef92021-01-29 18:32:06 +020031 auto layers = display->GetOrderLayersByZPos();
Matvii Zorinef3c7972020-08-11 15:15:44 +030032
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020033 int client_start = -1;
34 size_t client_size = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030035
Roman Stratiienko22fe9612023-01-17 21:22:29 +020036 auto flatcon = display->GetFlatCon();
37 if (flatcon) {
38 bool should_flatten = false;
39 if (layers.size() <= 1)
40 flatcon->Disable();
41 else
42 should_flatten = flatcon->NewFrame();
43
44 if (should_flatten) {
45 display->total_stats().frames_flattened_++;
46 MarkValidated(layers, 0, layers.size());
47 *num_types = layers.size();
48 return HWC2::Error::HasChanges;
49 }
50 }
51
52 std::tie(client_start, client_size) = GetClientLayers(display, layers);
53
54 MarkValidated(layers, client_start, client_size);
55
56 auto testing_needed = client_start != 0 || client_size != layers.size();
57
58 AtomicCommitArgs a_args = {.test_only = true};
59
60 if (testing_needed &&
61 display->CreateComposition(a_args) != HWC2::Error::None) {
62 ++display->total_stats().failed_kms_validate_;
Matvii Zorinef3c7972020-08-11 15:15:44 +030063 client_start = 0;
Matvii Zorined90ef92021-01-29 18:32:06 +020064 client_size = layers.size();
Roman Stratiienko22fe9612023-01-17 21:22:29 +020065 MarkValidated(layers, 0, client_size);
Matvii Zorinef3c7972020-08-11 15:15:44 +030066 }
67
68 *num_types = client_size;
69
Matvii Zorined90ef92021-01-29 18:32:06 +020070 display->total_stats().gpu_pixops_ += CalcPixOps(layers, client_start,
Matvii Zorinb3b15162021-01-29 15:48:20 +020071 client_size);
Matvii Zorined90ef92021-01-29 18:32:06 +020072 display->total_stats().total_pixops_ += CalcPixOps(layers, 0, layers.size());
Matvii Zorinef3c7972020-08-11 15:15:44 +030073
Roman Stratiienkofc014f52021-12-23 19:04:29 +020074 return *num_types != 0 ? HWC2::Error::HasChanges : HWC2::Error::None;
Matvii Zorinef3c7972020-08-11 15:15:44 +030075}
76
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020077std::tuple<int, size_t> Backend::GetClientLayers(
Roman Stratiienko3627beb2022-01-04 16:02:55 +020078 HwcDisplay *display, const std::vector<HwcLayer *> &layers) {
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020079 int client_start = -1;
80 size_t client_size = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030081
Roman Stratiienkodeb77352021-12-06 12:59:26 +020082 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorined90ef92021-01-29 18:32:06 +020083 if (IsClientLayer(display, layers[z_order])) {
Matvii Zorinef3c7972020-08-11 15:15:44 +030084 if (client_start < 0)
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020085 client_start = (int)z_order;
Matvii Zorinef3c7972020-08-11 15:15:44 +030086 client_size = (z_order - client_start) + 1;
87 }
88 }
89
Matvii Zorined90ef92021-01-29 18:32:06 +020090 return GetExtraClientRange(display, layers, client_start, client_size);
Matvii Zorinef3c7972020-08-11 15:15:44 +030091}
92
Roman Stratiienko3627beb2022-01-04 16:02:55 +020093bool Backend::IsClientLayer(HwcDisplay *display, HwcLayer *layer) {
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020094 return !HardwareSupportsLayerType(layer->GetSfType()) ||
Roman Stratiienko0da91bf2023-01-17 18:06:04 +020095 !layer->IsLayerUsableAsDevice() || display->CtmByGpu() ||
Roman Stratiienko4b2cc482022-02-21 14:53:58 +020096 (layer->GetLayerData().pi.RequireScalingOrPhasing() &&
Roman Stratiienko3dacd472022-01-11 19:18:34 +020097 display->GetHwc2()->GetResMan().ForcedScalingWithGpu());
Matvii Zorinef3c7972020-08-11 15:15:44 +030098}
99
Matvii Zorinede54b02021-01-29 16:45:25 +0200100bool Backend::HardwareSupportsLayerType(HWC2::Composition comp_type) {
101 return comp_type == HWC2::Composition::Device ||
102 comp_type == HWC2::Composition::Cursor;
103}
104
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200105uint32_t Backend::CalcPixOps(const std::vector<HwcLayer *> &layers,
Matvii Zorin2c91df52021-01-29 15:39:55 +0200106 size_t first_z, size_t size) {
107 uint32_t pixops = 0;
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200108 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorin2c91df52021-01-29 15:39:55 +0200109 if (z_order >= first_z && z_order < first_z + size) {
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300110 auto &df = layers[z_order]->GetLayerData().pi.display_frame;
Matvii Zorin2c91df52021-01-29 15:39:55 +0200111 pixops += (df.right - df.left) * (df.bottom - df.top);
112 }
113 }
114 return pixops;
115}
116
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200117void Backend::MarkValidated(std::vector<HwcLayer *> &layers,
Matvii Zorined745272021-01-29 16:40:25 +0200118 size_t client_first_z, size_t client_size) {
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200119 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorined90ef92021-01-29 18:32:06 +0200120 if (z_order >= client_first_z && z_order < client_first_z + client_size)
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200121 layers[z_order]->SetValidatedType(HWC2::Composition::Client);
Matvii Zorined745272021-01-29 16:40:25 +0200122 else
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200123 layers[z_order]->SetValidatedType(HWC2::Composition::Device);
Matvii Zorined745272021-01-29 16:40:25 +0200124 }
125}
126
Matvii Zorinb3b15162021-01-29 15:48:20 +0200127std::tuple<int, int> Backend::GetExtraClientRange(
Roman Stratiienko3627beb2022-01-04 16:02:55 +0200128 HwcDisplay *display, const std::vector<HwcLayer *> &layers,
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200129 int client_start, size_t client_size) {
Roman Stratiienko9362cef2022-02-02 09:53:50 +0200130 auto planes = display->GetPipe().GetUsablePlanes();
131 size_t avail_planes = planes.size();
Matvii Zorinb3b15162021-01-29 15:48:20 +0200132
133 /*
134 * If more layers then planes, save one plane
135 * for client composited layers
136 */
137 if (avail_planes < display->layers().size())
138 avail_planes--;
139
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300140 const int extra_client = int(layers.size() - client_size) - int(avail_planes);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200141
142 if (extra_client > 0) {
143 int start = 0;
144 size_t steps = 0;
145 if (client_size != 0) {
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300146 const int prepend = std::min(client_start, extra_client);
147 const int append = std::min(int(layers.size()) -
148 int(client_start + client_size),
149 extra_client);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200150 start = client_start - (int)prepend;
151 client_size += extra_client;
152 steps = 1 + std::min(std::min(append, prepend),
Matvii Zorin4535e532021-04-26 17:24:56 +0300153 int(layers.size()) - int(start + client_size));
Matvii Zorinb3b15162021-01-29 15:48:20 +0200154 } else {
155 client_size = extra_client;
Matvii Zorined90ef92021-01-29 18:32:06 +0200156 steps = 1 + layers.size() - extra_client;
Matvii Zorinb3b15162021-01-29 15:48:20 +0200157 }
158
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200159 uint32_t gpu_pixops = UINT32_MAX;
160 for (size_t i = 0; i < steps; i++) {
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300161 const uint32_t po = CalcPixOps(layers, start + i, client_size);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200162 if (po < gpu_pixops) {
163 gpu_pixops = po;
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200164 client_start = start + int(i);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200165 }
166 }
167 }
168
169 return std::make_tuple(client_start, client_size);
170}
171
Roman Stratiienkod21071f2021-03-09 21:56:50 +0200172// clang-format off
173// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables, cert-err58-cpp)
Matvii Zorinef3c7972020-08-11 15:15:44 +0300174REGISTER_BACKEND("generic", Backend);
Roman Stratiienkod21071f2021-03-09 21:56:50 +0200175// clang-format on
Matvii Zorinef3c7972020-08-11 15:15:44 +0300176
177} // namespace android