blob: ce606ddbef34a8f2d035b5b044d374ccb8ff392d [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
26HWC2::Error Backend::ValidateDisplay(DrmHwcTwo::HwcDisplay *display,
27 uint32_t *num_types,
28 uint32_t *num_requests) {
29 *num_types = 0;
30 *num_requests = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030031
Matvii Zorined90ef92021-01-29 18:32:06 +020032 auto layers = display->GetOrderLayersByZPos();
Matvii Zorinef3c7972020-08-11 15:15:44 +030033
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020034 int client_start = -1;
35 size_t client_size = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030036
Roman Stratiienkoe8c06792021-09-30 10:09:31 +030037 if (display->ProcessClientFlatteningState(layers.size() <= 1)) {
38 display->total_stats().frames_flattened_++;
Matvii Zorinef3c7972020-08-11 15:15:44 +030039 client_start = 0;
Matvii Zorined90ef92021-01-29 18:32:06 +020040 client_size = layers.size();
41 MarkValidated(layers, client_start, client_size);
Matvii Zorinef3c7972020-08-11 15:15:44 +030042 } else {
Matvii Zorined90ef92021-01-29 18:32:06 +020043 std::tie(client_start, client_size) = GetClientLayers(display, layers);
Matvii Zorinef3c7972020-08-11 15:15:44 +030044
Matvii Zorined90ef92021-01-29 18:32:06 +020045 MarkValidated(layers, client_start, client_size);
Matvii Zorinef3c7972020-08-11 15:15:44 +030046
Matvii Zorined90ef92021-01-29 18:32:06 +020047 bool testing_needed = !(client_start == 0 && client_size == layers.size());
Matvii Zorinef3c7972020-08-11 15:15:44 +030048
Roman Stratiienko2a1f1ae2021-10-23 17:47:35 +030049 AtomicCommitArgs a_args = {.test_only = true};
50
Matvii Zorinef3c7972020-08-11 15:15:44 +030051 if (testing_needed &&
Roman Stratiienko2a1f1ae2021-10-23 17:47:35 +030052 display->CreateComposition(a_args) != HWC2::Error::None) {
Matvii Zorinef3c7972020-08-11 15:15:44 +030053 ++display->total_stats().failed_kms_validate_;
Matvii Zorinb3b15162021-01-29 15:48:20 +020054 client_start = 0;
Matvii Zorined90ef92021-01-29 18:32:06 +020055 client_size = layers.size();
56 MarkValidated(layers, 0, client_size);
Matvii Zorinef3c7972020-08-11 15:15:44 +030057 }
58 }
59
60 *num_types = client_size;
61
Matvii Zorined90ef92021-01-29 18:32:06 +020062 display->total_stats().gpu_pixops_ += CalcPixOps(layers, client_start,
Matvii Zorinb3b15162021-01-29 15:48:20 +020063 client_size);
Matvii Zorined90ef92021-01-29 18:32:06 +020064 display->total_stats().total_pixops_ += CalcPixOps(layers, 0, layers.size());
Matvii Zorinef3c7972020-08-11 15:15:44 +030065
66 return *num_types ? HWC2::Error::HasChanges : HWC2::Error::None;
67}
68
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020069std::tuple<int, size_t> Backend::GetClientLayers(
Matvii Zorinef3c7972020-08-11 15:15:44 +030070 DrmHwcTwo::HwcDisplay *display,
Matvii Zorinb3b15162021-01-29 15:48:20 +020071 const std::vector<DrmHwcTwo::HwcLayer *> &layers) {
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020072 int client_start = -1;
73 size_t client_size = 0;
Matvii Zorinef3c7972020-08-11 15:15:44 +030074
Roman Stratiienkodeb77352021-12-06 12:59:26 +020075 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorined90ef92021-01-29 18:32:06 +020076 if (IsClientLayer(display, layers[z_order])) {
Matvii Zorinef3c7972020-08-11 15:15:44 +030077 if (client_start < 0)
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020078 client_start = (int)z_order;
Matvii Zorinef3c7972020-08-11 15:15:44 +030079 client_size = (z_order - client_start) + 1;
80 }
81 }
82
Matvii Zorined90ef92021-01-29 18:32:06 +020083 return GetExtraClientRange(display, layers, client_start, client_size);
Matvii Zorinef3c7972020-08-11 15:15:44 +030084}
85
86bool Backend::IsClientLayer(DrmHwcTwo::HwcDisplay *display,
87 DrmHwcTwo::HwcLayer *layer) {
Matvii Zorinede54b02021-01-29 16:45:25 +020088 return !HardwareSupportsLayerType(layer->sf_type()) ||
Roman Stratiienkob2e9fe22020-10-03 10:52:36 +030089 !BufferInfoGetter::GetInstance()->IsHandleUsable(layer->buffer()) ||
Matvii Zorinef3c7972020-08-11 15:15:44 +030090 display->color_transform_hint() != HAL_COLOR_TRANSFORM_IDENTITY ||
91 (layer->RequireScalingOrPhasing() &&
92 display->resource_manager()->ForcedScalingWithGpu());
93}
94
Matvii Zorinede54b02021-01-29 16:45:25 +020095bool Backend::HardwareSupportsLayerType(HWC2::Composition comp_type) {
96 return comp_type == HWC2::Composition::Device ||
97 comp_type == HWC2::Composition::Cursor;
98}
99
Matvii Zorin2c91df52021-01-29 15:39:55 +0200100uint32_t Backend::CalcPixOps(const std::vector<DrmHwcTwo::HwcLayer *> &layers,
101 size_t first_z, size_t size) {
102 uint32_t pixops = 0;
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200103 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorin2c91df52021-01-29 15:39:55 +0200104 if (z_order >= first_z && z_order < first_z + size) {
Matvii Zorined90ef92021-01-29 18:32:06 +0200105 hwc_rect_t df = layers[z_order]->display_frame();
Matvii Zorin2c91df52021-01-29 15:39:55 +0200106 pixops += (df.right - df.left) * (df.bottom - df.top);
107 }
108 }
109 return pixops;
110}
111
Matvii Zorined90ef92021-01-29 18:32:06 +0200112void Backend::MarkValidated(std::vector<DrmHwcTwo::HwcLayer *> &layers,
Matvii Zorined745272021-01-29 16:40:25 +0200113 size_t client_first_z, size_t client_size) {
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200114 for (size_t z_order = 0; z_order < layers.size(); ++z_order) {
Matvii Zorined90ef92021-01-29 18:32:06 +0200115 if (z_order >= client_first_z && z_order < client_first_z + client_size)
116 layers[z_order]->set_validated_type(HWC2::Composition::Client);
Matvii Zorined745272021-01-29 16:40:25 +0200117 else
Matvii Zorined90ef92021-01-29 18:32:06 +0200118 layers[z_order]->set_validated_type(HWC2::Composition::Device);
Matvii Zorined745272021-01-29 16:40:25 +0200119 }
120}
121
Matvii Zorinb3b15162021-01-29 15:48:20 +0200122std::tuple<int, int> Backend::GetExtraClientRange(
123 DrmHwcTwo::HwcDisplay *display,
124 const std::vector<DrmHwcTwo::HwcLayer *> &layers, int client_start,
125 size_t client_size) {
126 size_t avail_planes = display->primary_planes().size() +
127 display->overlay_planes().size();
128
129 /*
130 * If more layers then planes, save one plane
131 * for client composited layers
132 */
133 if (avail_planes < display->layers().size())
134 avail_planes--;
135
Matvii Zorin4535e532021-04-26 17:24:56 +0300136 int extra_client = int(layers.size() - client_size) - int(avail_planes);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200137
138 if (extra_client > 0) {
139 int start = 0;
140 size_t steps = 0;
141 if (client_size != 0) {
Matvii Zorin4535e532021-04-26 17:24:56 +0300142 int prepend = std::min(client_start, extra_client);
143 int append = std::min(int(layers.size()) -
144 int(client_start + client_size),
145 extra_client);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200146 start = client_start - (int)prepend;
147 client_size += extra_client;
148 steps = 1 + std::min(std::min(append, prepend),
Matvii Zorin4535e532021-04-26 17:24:56 +0300149 int(layers.size()) - int(start + client_size));
Matvii Zorinb3b15162021-01-29 15:48:20 +0200150 } else {
151 client_size = extra_client;
Matvii Zorined90ef92021-01-29 18:32:06 +0200152 steps = 1 + layers.size() - extra_client;
Matvii Zorinb3b15162021-01-29 15:48:20 +0200153 }
154
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200155 uint32_t gpu_pixops = UINT32_MAX;
156 for (size_t i = 0; i < steps; i++) {
Matvii Zorined90ef92021-01-29 18:32:06 +0200157 uint32_t po = CalcPixOps(layers, start + i, client_size);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200158 if (po < gpu_pixops) {
159 gpu_pixops = po;
Roman Stratiienkodeb77352021-12-06 12:59:26 +0200160 client_start = start + int(i);
Matvii Zorinb3b15162021-01-29 15:48:20 +0200161 }
162 }
163 }
164
165 return std::make_tuple(client_start, client_size);
166}
167
Roman Stratiienkod21071f2021-03-09 21:56:50 +0200168// clang-format off
169// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables, cert-err58-cpp)
Matvii Zorinef3c7972020-08-11 15:15:44 +0300170REGISTER_BACKEND("generic", Backend);
Roman Stratiienkod21071f2021-03-09 21:56:50 +0200171// clang-format on
Matvii Zorinef3c7972020-08-11 15:15:44 +0300172
173} // namespace android