blob: 28f48f3c4a1af6ded56c5a777bedfe06b04978b3 [file] [log] [blame]
Sean Paul6a55e9f2015-04-30 15:31:06 -04001/*
2 * Copyright (C) 2015 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#define LOG_TAG "hwc-drm-plane"
18
Roman Stratiienko13cc3662020-08-29 21:35:39 +030019#include "DrmPlane.h"
Sean Paul6a55e9f2015-04-30 15:31:06 -040020
Roman Stratiienko2640cd82021-02-26 17:49:40 +020021#include <algorithm>
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020022#include <cerrno>
Sean Paulf72cccd2018-08-27 13:59:08 -040023#include <cinttypes>
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020024#include <cstdint>
Sean Paul6a55e9f2015-04-30 15:31:06 -040025
Roman Stratiienko13cc3662020-08-29 21:35:39 +030026#include "DrmDevice.h"
Roman Stratiienkod518a052021-02-25 19:15:14 +020027#include "bufferinfo/BufferInfoGetter.h"
28#include "utils/log.h"
Sean Paul6a55e9f2015-04-30 15:31:06 -040029
30namespace android {
31
Roman Stratiienkob671fab2022-01-29 00:50:22 +020032auto DrmPlane::CreateInstance(DrmDevice &dev, uint32_t plane_id)
33 -> std::unique_ptr<DrmPlane> {
Roman Stratiienko7d899112022-01-31 11:30:27 +020034 auto p = MakeDrmModePlaneUnique(dev.GetFd(), plane_id);
Roman Stratiienkob671fab2022-01-29 00:50:22 +020035 if (!p) {
36 ALOGE("Failed to get plane %d", plane_id);
37 return {};
38 }
39
40 auto plane = std::unique_ptr<DrmPlane>(new DrmPlane(dev, std::move(p)));
41
42 if (plane->Init() != 0) {
43 ALOGE("Failed to init plane %d", plane_id);
44 return {};
45 }
46
47 return plane;
Sean Paul6a55e9f2015-04-30 15:31:06 -040048}
49
Sean Paul6a55e9f2015-04-30 15:31:06 -040050int DrmPlane::Init() {
Roman Stratiienkob671fab2022-01-29 00:50:22 +020051 // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-pointer-arithmetic)
52 formats_ = {plane_->formats, plane_->formats + plane_->count_formats};
53
Sean Paul6a55e9f2015-04-30 15:31:06 -040054 DrmProperty p;
55
Roman Stratiienko0b063882021-09-30 10:15:05 +030056 if (!GetPlaneProperty("type", p)) {
57 return -ENOTSUP;
Sean Paul6a55e9f2015-04-30 15:31:06 -040058 }
59
Roman Stratiienko0b063882021-09-30 10:15:05 +030060 int ret = 0;
Roman Stratiienkob3b5c1e2021-02-15 13:44:19 +020061 uint64_t type = 0;
Sean Paulfc0b1da2019-03-06 09:48:42 -050062 std::tie(ret, type) = p.value();
Roman Stratiienkof815d382021-12-30 19:23:14 +020063 if (ret != 0) {
Sean Paul6a55e9f2015-04-30 15:31:06 -040064 ALOGE("Failed to get plane type property value");
65 return ret;
66 }
67 switch (type) {
68 case DRM_PLANE_TYPE_OVERLAY:
69 case DRM_PLANE_TYPE_PRIMARY:
70 case DRM_PLANE_TYPE_CURSOR:
71 type_ = (uint32_t)type;
72 break;
73 default:
Sean Paulf741c672016-05-11 13:49:38 -040074 ALOGE("Invalid plane type %" PRIu64, type);
Sean Paul6a55e9f2015-04-30 15:31:06 -040075 return -EINVAL;
76 }
77
Roman Stratiienko0b063882021-09-30 10:15:05 +030078 if (!GetPlaneProperty("CRTC_ID", crtc_property_) ||
79 !GetPlaneProperty("FB_ID", fb_property_) ||
80 !GetPlaneProperty("CRTC_X", crtc_x_property_) ||
81 !GetPlaneProperty("CRTC_Y", crtc_y_property_) ||
82 !GetPlaneProperty("CRTC_W", crtc_w_property_) ||
83 !GetPlaneProperty("CRTC_H", crtc_h_property_) ||
84 !GetPlaneProperty("SRC_X", src_x_property_) ||
85 !GetPlaneProperty("SRC_Y", src_y_property_) ||
86 !GetPlaneProperty("SRC_W", src_w_property_) ||
87 !GetPlaneProperty("SRC_H", src_h_property_)) {
88 return -ENOTSUP;
Sean Paul6a55e9f2015-04-30 15:31:06 -040089 }
90
Roman Stratiienko0b063882021-09-30 10:15:05 +030091 GetPlaneProperty("zpos", zpos_property_, Presence::kOptional);
Sean Paul6a55e9f2015-04-30 15:31:06 -040092
Roman Stratiienko0b063882021-09-30 10:15:05 +030093 if (GetPlaneProperty("rotation", rotation_property_, Presence::kOptional)) {
Roman Stratiienko24a4c182021-09-29 12:59:54 +030094 rotation_property_.AddEnumToMap("rotate-0", DrmHwcTransform::kIdentity,
95 transform_enum_map_);
96 rotation_property_.AddEnumToMap("rotate-90", DrmHwcTransform::kRotate90,
97 transform_enum_map_);
98 rotation_property_.AddEnumToMap("rotate-180", DrmHwcTransform::kRotate180,
99 transform_enum_map_);
100 rotation_property_.AddEnumToMap("rotate-270", DrmHwcTransform::kRotate270,
101 transform_enum_map_);
102 rotation_property_.AddEnumToMap("reflect-x", DrmHwcTransform::kFlipH,
103 transform_enum_map_);
104 rotation_property_.AddEnumToMap("reflect-y", DrmHwcTransform::kFlipV,
105 transform_enum_map_);
Roman Stratiienko24a4c182021-09-29 12:59:54 +0300106 }
Sean Paul1c4c3262015-07-14 15:51:52 -0400107
Roman Stratiienko0b063882021-09-30 10:15:05 +0300108 GetPlaneProperty("alpha", alpha_property_, Presence::kOptional);
Sean Pauld8aefb62015-10-15 15:17:31 -0400109
Roman Stratiienko0b063882021-09-30 10:15:05 +0300110 if (GetPlaneProperty("pixel blend mode", blend_property_,
111 Presence::kOptional)) {
Roman Stratiienko0f679aa2021-09-29 12:59:48 +0300112 blend_property_.AddEnumToMap("Pre-multiplied", DrmHwcBlending::kPreMult,
113 blending_enum_map_);
114 blend_property_.AddEnumToMap("Coverage", DrmHwcBlending::kCoverage,
115 blending_enum_map_);
116 blend_property_.AddEnumToMap("None", DrmHwcBlending::kNone,
117 blending_enum_map_);
Roman Stratiienko5063d532021-09-29 12:47:31 +0300118 }
Lowry Li9b6cafd2018-08-28 17:58:21 +0800119
Roman Stratiienko0b063882021-09-30 10:15:05 +0300120 GetPlaneProperty("IN_FENCE_FD", in_fence_fd_property_, Presence::kOptional);
Robert Fossa09220c2016-09-30 10:27:23 -0400121
Matvii Zorin8338c342020-09-08 16:12:51 +0300122 if (HasNonRgbFormat()) {
Roman Stratiienko0b063882021-09-30 10:15:05 +0300123 if (GetPlaneProperty("COLOR_ENCODING", color_encoding_propery_,
124 Presence::kOptional)) {
Roman Stratiienko0f679aa2021-09-29 12:59:48 +0300125 color_encoding_propery_.AddEnumToMap("ITU-R BT.709 YCbCr",
126 DrmHwcColorSpace::kItuRec709,
127 color_encoding_enum_map_);
128 color_encoding_propery_.AddEnumToMap("ITU-R BT.601 YCbCr",
129 DrmHwcColorSpace::kItuRec601,
130 color_encoding_enum_map_);
131 color_encoding_propery_.AddEnumToMap("ITU-R BT.2020 YCbCr",
132 DrmHwcColorSpace::kItuRec2020,
133 color_encoding_enum_map_);
Roman Stratiienko4f1effa2021-09-29 12:47:26 +0300134 }
Matvii Zorin8338c342020-09-08 16:12:51 +0300135
Roman Stratiienko0b063882021-09-30 10:15:05 +0300136 if (GetPlaneProperty("COLOR_RANGE", color_range_property_,
137 Presence::kOptional)) {
Roman Stratiienko0f679aa2021-09-29 12:59:48 +0300138 color_range_property_.AddEnumToMap("YCbCr full range",
139 DrmHwcSampleRange::kFullRange,
140 color_range_enum_map_);
141 color_range_property_.AddEnumToMap("YCbCr limited range",
142 DrmHwcSampleRange::kLimitedRange,
143 color_range_enum_map_);
Roman Stratiienko4f1effa2021-09-29 12:47:26 +0300144 }
Matvii Zorin8338c342020-09-08 16:12:51 +0300145 }
146
Sean Paul6a55e9f2015-04-30 15:31:06 -0400147 return 0;
148}
149
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200150bool DrmPlane::IsCrtcSupported(const DrmCrtc &crtc) const {
Roman Stratiienko10be8752022-01-30 20:28:46 +0200151 return ((1 << crtc.GetIndexInResArray()) & plane_->possible_crtcs) != 0;
Sean Paul6a55e9f2015-04-30 15:31:06 -0400152}
153
Matvii Zorin67a89d32021-01-31 14:45:05 +0200154bool DrmPlane::IsValidForLayer(DrmHwcLayer *layer) {
Roman Stratiienko24a4c182021-09-29 12:59:54 +0300155 if (!rotation_property_) {
Benjamin Lib00b28d2021-05-09 08:52:18 -0700156 if (layer->transform != DrmHwcTransform::kIdentity) {
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200157 ALOGV("No rotation property on plane %d", GetId());
Benjamin Lib00b28d2021-05-09 08:52:18 -0700158 return false;
159 }
160 } else {
Roman Stratiienko24a4c182021-09-29 12:59:54 +0300161 if (transform_enum_map_.count(layer->transform) == 0) {
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200162 ALOGV("Transform is not supported on plane %d", GetId());
Roman Stratiienko24a4c182021-09-29 12:59:54 +0300163 return false;
Benjamin Lib00b28d2021-05-09 08:52:18 -0700164 }
Matvii Zorin67a89d32021-01-31 14:45:05 +0200165 }
166
Roman Stratiienkof815d382021-12-30 19:23:14 +0200167 if (alpha_property_.id() == 0 && layer->alpha != UINT16_MAX) {
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200168 ALOGV("Alpha is not supported on plane %d", GetId());
Matvii Zorin67a89d32021-01-31 14:45:05 +0200169 return false;
170 }
171
Roman Stratiienko5063d532021-09-29 12:47:31 +0300172 if (blending_enum_map_.count(layer->blending) == 0 &&
Roman Stratiienko013d0682021-09-30 19:36:00 +0300173 layer->blending != DrmHwcBlending::kNone &&
174 layer->blending != DrmHwcBlending::kPreMult) {
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200175 ALOGV("Blending is not supported on plane %d", GetId());
Roman Stratiienko5063d532021-09-29 12:47:31 +0300176 return false;
Matvii Zorin67a89d32021-01-31 14:45:05 +0200177 }
178
Roman Stratiienko51287152021-02-10 14:59:52 +0200179 uint32_t format = layer->buffer_info.format;
Matvii Zorin67a89d32021-01-31 14:45:05 +0200180 if (!IsFormatSupported(format)) {
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200181 ALOGV("Plane %d does not supports %c%c%c%c format", GetId(), format,
Matvii Zorin67a89d32021-01-31 14:45:05 +0200182 format >> 8, format >> 16, format >> 24);
183 return false;
184 }
185
186 return true;
187}
188
Roman Kovalivskyi859b6072020-03-26 05:03:39 +0200189bool DrmPlane::IsFormatSupported(uint32_t format) const {
190 return std::find(std::begin(formats_), std::end(formats_), format) !=
191 std::end(formats_);
192}
193
Matvii Zorin8338c342020-09-08 16:12:51 +0300194bool DrmPlane::HasNonRgbFormat() const {
195 return std::find_if_not(std::begin(formats_), std::end(formats_),
196 [](uint32_t format) {
197 return BufferInfoGetter::IsDrmFormatRgb(format);
198 }) != std::end(formats_);
199}
200
Roman Stratiienko6662f712021-09-29 12:52:22 +0300201static uint64_t ToDrmRotation(DrmHwcTransform transform) {
202 uint64_t rotation = 0;
Roman Stratiienkof815d382021-12-30 19:23:14 +0200203 if ((transform & DrmHwcTransform::kFlipH) != 0)
Roman Stratiienko6662f712021-09-29 12:52:22 +0300204 rotation |= DRM_MODE_REFLECT_X;
Roman Stratiienkof815d382021-12-30 19:23:14 +0200205 if ((transform & DrmHwcTransform::kFlipV) != 0)
Roman Stratiienko6662f712021-09-29 12:52:22 +0300206 rotation |= DRM_MODE_REFLECT_Y;
Roman Stratiienkof815d382021-12-30 19:23:14 +0200207 if ((transform & DrmHwcTransform::kRotate90) != 0)
Roman Stratiienko6662f712021-09-29 12:52:22 +0300208 rotation |= DRM_MODE_ROTATE_90;
Roman Stratiienkof815d382021-12-30 19:23:14 +0200209 else if ((transform & DrmHwcTransform::kRotate180) != 0)
Roman Stratiienko6662f712021-09-29 12:52:22 +0300210 rotation |= DRM_MODE_ROTATE_180;
Roman Stratiienkof815d382021-12-30 19:23:14 +0200211 else if ((transform & DrmHwcTransform::kRotate270) != 0)
Roman Stratiienko6662f712021-09-29 12:52:22 +0300212 rotation |= DRM_MODE_ROTATE_270;
213 else
214 rotation |= DRM_MODE_ROTATE_0;
215
216 return rotation;
217}
218
Roman Stratiienko57b9ba32021-12-30 19:20:53 +0200219/* Convert float to 16.16 fixed point */
220static int To1616FixPt(float in) {
221 constexpr int kBitShift = 16;
222 return int(in * (1 << kBitShift));
223}
224
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300225auto DrmPlane::AtomicSetState(drmModeAtomicReq &pset, DrmHwcLayer &layer,
226 uint32_t zpos, uint32_t crtc_id) -> int {
Roman Stratiienkofc014f52021-12-23 19:04:29 +0200227 if (!layer.fb_id_handle) {
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300228 ALOGE("Expected a valid framebuffer for pset");
229 return -EINVAL;
230 }
231
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300232 if (zpos_property_ && !zpos_property_.is_immutable()) {
233 uint64_t min_zpos = 0;
234
235 // Ignore ret and use min_zpos as 0 by default
236 std::tie(std::ignore, min_zpos) = zpos_property_.range_min();
237
238 if (!zpos_property_.AtomicSet(pset, zpos + min_zpos)) {
239 return -EINVAL;
240 }
241 }
242
Roman Stratiienko6662f712021-09-29 12:52:22 +0300243 if (layer.acquire_fence &&
244 !in_fence_fd_property_.AtomicSet(pset, layer.acquire_fence.Get())) {
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300245 return -EINVAL;
246 }
247
Roman Stratiienko6662f712021-09-29 12:52:22 +0300248 if (!crtc_property_.AtomicSet(pset, crtc_id) ||
Roman Stratiienkofc014f52021-12-23 19:04:29 +0200249 !fb_property_.AtomicSet(pset, layer.fb_id_handle->GetFbId()) ||
Roman Stratiienko6662f712021-09-29 12:52:22 +0300250 !crtc_x_property_.AtomicSet(pset, layer.display_frame.left) ||
251 !crtc_y_property_.AtomicSet(pset, layer.display_frame.top) ||
252 !crtc_w_property_.AtomicSet(pset, layer.display_frame.right -
253 layer.display_frame.left) ||
254 !crtc_h_property_.AtomicSet(pset, layer.display_frame.bottom -
255 layer.display_frame.top) ||
Roman Stratiienko57b9ba32021-12-30 19:20:53 +0200256 !src_x_property_.AtomicSet(pset, To1616FixPt(layer.source_crop.left)) ||
257 !src_y_property_.AtomicSet(pset, To1616FixPt(layer.source_crop.top)) ||
258 !src_w_property_.AtomicSet(pset, To1616FixPt(layer.source_crop.right -
259 layer.source_crop.left)) ||
260 !src_h_property_.AtomicSet(pset, To1616FixPt(layer.source_crop.bottom -
261 layer.source_crop.top))) {
Roman Stratiienko6662f712021-09-29 12:52:22 +0300262 return -EINVAL;
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300263 }
264
Roman Stratiienko6662f712021-09-29 12:52:22 +0300265 if (rotation_property_ &&
266 !rotation_property_.AtomicSet(pset, ToDrmRotation(layer.transform))) {
267 return -EINVAL;
268 }
269
270 if (alpha_property_ && !alpha_property_.AtomicSet(pset, layer.alpha)) {
271 return -EINVAL;
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300272 }
273
Roman Stratiienko5063d532021-09-29 12:47:31 +0300274 if (blending_enum_map_.count(layer.blending) != 0 &&
275 !blend_property_.AtomicSet(pset, blending_enum_map_[layer.blending])) {
276 return -EINVAL;
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300277 }
278
Roman Stratiienko4f1effa2021-09-29 12:47:26 +0300279 if (color_encoding_enum_map_.count(layer.color_space) != 0 &&
280 !color_encoding_propery_
281 .AtomicSet(pset, color_encoding_enum_map_[layer.color_space])) {
282 return -EINVAL;
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300283 }
284
Roman Stratiienko4f1effa2021-09-29 12:47:26 +0300285 if (color_range_enum_map_.count(layer.sample_range) != 0 &&
286 !color_range_property_
287 .AtomicSet(pset, color_range_enum_map_[layer.sample_range])) {
288 return -EINVAL;
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300289 }
Roman Stratiienko4f1effa2021-09-29 12:47:26 +0300290
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300291 return 0;
Sean Paul6a55e9f2015-04-30 15:31:06 -0400292}
293
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300294auto DrmPlane::AtomicDisablePlane(drmModeAtomicReq &pset) -> int {
295 if (!crtc_property_.AtomicSet(pset, 0) || !fb_property_.AtomicSet(pset, 0)) {
296 return -EINVAL;
297 }
Sean Paul6a55e9f2015-04-30 15:31:06 -0400298
Roman Stratiienko0dbe6392021-09-29 12:47:16 +0300299 return 0;
Sean Paul6a55e9f2015-04-30 15:31:06 -0400300}
Sean Paul1c4c3262015-07-14 15:51:52 -0400301
Roman Stratiienko0b063882021-09-30 10:15:05 +0300302auto DrmPlane::GetPlaneProperty(const char *prop_name, DrmProperty &property,
303 Presence presence) -> bool {
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200304 int err = drm_->GetProperty(GetId(), DRM_MODE_OBJECT_PLANE, prop_name,
305 &property);
Roman Stratiienko0b063882021-09-30 10:15:05 +0300306 if (err != 0) {
307 if (presence == Presence::kMandatory) {
308 ALOGE("Could not get mandatory property \"%s\" from plane %d", prop_name,
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200309 GetId());
Roman Stratiienko0b063882021-09-30 10:15:05 +0300310 } else {
311 ALOGV("Could not get optional property \"%s\" from plane %d", prop_name,
Roman Stratiienkob671fab2022-01-29 00:50:22 +0200312 GetId());
Roman Stratiienko0b063882021-09-30 10:15:05 +0300313 }
314 return false;
315 }
316
317 return true;
318}
319
Sean Paulf72cccd2018-08-27 13:59:08 -0400320} // namespace android