xref: /aosp_15_r20/external/drm_hwcomposer/drm/DrmPlane.cpp (revision 0a9764fe0a15e71ebbeb85e87e10990c23aab47f)
1*0a9764feSAndroid Build Coastguard Worker /*
2*0a9764feSAndroid Build Coastguard Worker  * Copyright (C) 2015 The Android Open Source Project
3*0a9764feSAndroid Build Coastguard Worker  *
4*0a9764feSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*0a9764feSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*0a9764feSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*0a9764feSAndroid Build Coastguard Worker  *
8*0a9764feSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*0a9764feSAndroid Build Coastguard Worker  *
10*0a9764feSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*0a9764feSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*0a9764feSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*0a9764feSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*0a9764feSAndroid Build Coastguard Worker  * limitations under the License.
15*0a9764feSAndroid Build Coastguard Worker  */
16*0a9764feSAndroid Build Coastguard Worker 
17*0a9764feSAndroid Build Coastguard Worker #define LOG_TAG "drmhwc"
18*0a9764feSAndroid Build Coastguard Worker 
19*0a9764feSAndroid Build Coastguard Worker #include "DrmPlane.h"
20*0a9764feSAndroid Build Coastguard Worker 
21*0a9764feSAndroid Build Coastguard Worker #include <algorithm>
22*0a9764feSAndroid Build Coastguard Worker #include <cerrno>
23*0a9764feSAndroid Build Coastguard Worker #include <cinttypes>
24*0a9764feSAndroid Build Coastguard Worker #include <cstdint>
25*0a9764feSAndroid Build Coastguard Worker 
26*0a9764feSAndroid Build Coastguard Worker #include "DrmDevice.h"
27*0a9764feSAndroid Build Coastguard Worker #include "bufferinfo/BufferInfoGetter.h"
28*0a9764feSAndroid Build Coastguard Worker #include "utils/log.h"
29*0a9764feSAndroid Build Coastguard Worker 
30*0a9764feSAndroid Build Coastguard Worker namespace android {
31*0a9764feSAndroid Build Coastguard Worker 
CreateInstance(DrmDevice & dev,uint32_t plane_id)32*0a9764feSAndroid Build Coastguard Worker auto DrmPlane::CreateInstance(DrmDevice &dev, uint32_t plane_id)
33*0a9764feSAndroid Build Coastguard Worker     -> std::unique_ptr<DrmPlane> {
34*0a9764feSAndroid Build Coastguard Worker   auto p = MakeDrmModePlaneUnique(*dev.GetFd(), plane_id);
35*0a9764feSAndroid Build Coastguard Worker   if (!p) {
36*0a9764feSAndroid Build Coastguard Worker     ALOGE("Failed to get plane %d", plane_id);
37*0a9764feSAndroid Build Coastguard Worker     return {};
38*0a9764feSAndroid Build Coastguard Worker   }
39*0a9764feSAndroid Build Coastguard Worker 
40*0a9764feSAndroid Build Coastguard Worker   auto plane = std::unique_ptr<DrmPlane>(new DrmPlane(dev, std::move(p)));
41*0a9764feSAndroid Build Coastguard Worker 
42*0a9764feSAndroid Build Coastguard Worker   if (plane->Init() != 0) {
43*0a9764feSAndroid Build Coastguard Worker     ALOGE("Failed to init plane %d", plane_id);
44*0a9764feSAndroid Build Coastguard Worker     return {};
45*0a9764feSAndroid Build Coastguard Worker   }
46*0a9764feSAndroid Build Coastguard Worker 
47*0a9764feSAndroid Build Coastguard Worker   return plane;
48*0a9764feSAndroid Build Coastguard Worker }
49*0a9764feSAndroid Build Coastguard Worker 
Init()50*0a9764feSAndroid Build Coastguard Worker int DrmPlane::Init() {
51*0a9764feSAndroid Build Coastguard Worker   // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-pointer-arithmetic)
52*0a9764feSAndroid Build Coastguard Worker   formats_ = {plane_->formats, plane_->formats + plane_->count_formats};
53*0a9764feSAndroid Build Coastguard Worker 
54*0a9764feSAndroid Build Coastguard Worker   DrmProperty p;
55*0a9764feSAndroid Build Coastguard Worker 
56*0a9764feSAndroid Build Coastguard Worker   if (!GetPlaneProperty("type", p)) {
57*0a9764feSAndroid Build Coastguard Worker     return -ENOTSUP;
58*0a9764feSAndroid Build Coastguard Worker   }
59*0a9764feSAndroid Build Coastguard Worker 
60*0a9764feSAndroid Build Coastguard Worker   auto type = p.GetValue();
61*0a9764feSAndroid Build Coastguard Worker   if (!type) {
62*0a9764feSAndroid Build Coastguard Worker     ALOGE("Failed to get plane type property value");
63*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
64*0a9764feSAndroid Build Coastguard Worker   }
65*0a9764feSAndroid Build Coastguard Worker   switch (*type) {
66*0a9764feSAndroid Build Coastguard Worker     case DRM_PLANE_TYPE_OVERLAY:
67*0a9764feSAndroid Build Coastguard Worker     case DRM_PLANE_TYPE_PRIMARY:
68*0a9764feSAndroid Build Coastguard Worker     case DRM_PLANE_TYPE_CURSOR:
69*0a9764feSAndroid Build Coastguard Worker       type_ = (uint32_t)*type;
70*0a9764feSAndroid Build Coastguard Worker       break;
71*0a9764feSAndroid Build Coastguard Worker     default:
72*0a9764feSAndroid Build Coastguard Worker       ALOGE("Invalid plane type %" PRIu64, *type);
73*0a9764feSAndroid Build Coastguard Worker       return -EINVAL;
74*0a9764feSAndroid Build Coastguard Worker   }
75*0a9764feSAndroid Build Coastguard Worker 
76*0a9764feSAndroid Build Coastguard Worker   if (!GetPlaneProperty("CRTC_ID", crtc_property_) ||
77*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("FB_ID", fb_property_) ||
78*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("CRTC_X", crtc_x_property_) ||
79*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("CRTC_Y", crtc_y_property_) ||
80*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("CRTC_W", crtc_w_property_) ||
81*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("CRTC_H", crtc_h_property_) ||
82*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("SRC_X", src_x_property_) ||
83*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("SRC_Y", src_y_property_) ||
84*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("SRC_W", src_w_property_) ||
85*0a9764feSAndroid Build Coastguard Worker       !GetPlaneProperty("SRC_H", src_h_property_)) {
86*0a9764feSAndroid Build Coastguard Worker     return -ENOTSUP;
87*0a9764feSAndroid Build Coastguard Worker   }
88*0a9764feSAndroid Build Coastguard Worker 
89*0a9764feSAndroid Build Coastguard Worker   GetPlaneProperty("zpos", zpos_property_, Presence::kOptional);
90*0a9764feSAndroid Build Coastguard Worker 
91*0a9764feSAndroid Build Coastguard Worker   /* DRM/KMS uses counter-clockwise rotations, while HWC API uses
92*0a9764feSAndroid Build Coastguard Worker    * clockwise. That's why 90 and 270 are swapped here.
93*0a9764feSAndroid Build Coastguard Worker    */
94*0a9764feSAndroid Build Coastguard Worker   if (GetPlaneProperty("rotation", rotation_property_, Presence::kOptional)) {
95*0a9764feSAndroid Build Coastguard Worker     rotation_property_.AddEnumToMap("rotate-0", LayerTransform::kIdentity,
96*0a9764feSAndroid Build Coastguard Worker                                     transform_enum_map_);
97*0a9764feSAndroid Build Coastguard Worker     rotation_property_.AddEnumToMap("rotate-90", LayerTransform::kRotate270,
98*0a9764feSAndroid Build Coastguard Worker                                     transform_enum_map_);
99*0a9764feSAndroid Build Coastguard Worker     rotation_property_.AddEnumToMap("rotate-180", LayerTransform::kRotate180,
100*0a9764feSAndroid Build Coastguard Worker                                     transform_enum_map_);
101*0a9764feSAndroid Build Coastguard Worker     rotation_property_.AddEnumToMap("rotate-270", LayerTransform::kRotate90,
102*0a9764feSAndroid Build Coastguard Worker                                     transform_enum_map_);
103*0a9764feSAndroid Build Coastguard Worker     rotation_property_.AddEnumToMap("reflect-x", LayerTransform::kFlipH,
104*0a9764feSAndroid Build Coastguard Worker                                     transform_enum_map_);
105*0a9764feSAndroid Build Coastguard Worker     rotation_property_.AddEnumToMap("reflect-y", LayerTransform::kFlipV,
106*0a9764feSAndroid Build Coastguard Worker                                     transform_enum_map_);
107*0a9764feSAndroid Build Coastguard Worker   }
108*0a9764feSAndroid Build Coastguard Worker 
109*0a9764feSAndroid Build Coastguard Worker   GetPlaneProperty("alpha", alpha_property_, Presence::kOptional);
110*0a9764feSAndroid Build Coastguard Worker 
111*0a9764feSAndroid Build Coastguard Worker   if (GetPlaneProperty("pixel blend mode", blend_property_,
112*0a9764feSAndroid Build Coastguard Worker                        Presence::kOptional)) {
113*0a9764feSAndroid Build Coastguard Worker     blend_property_.AddEnumToMap("Pre-multiplied", BufferBlendMode::kPreMult,
114*0a9764feSAndroid Build Coastguard Worker                                  blending_enum_map_);
115*0a9764feSAndroid Build Coastguard Worker     blend_property_.AddEnumToMap("Coverage", BufferBlendMode::kCoverage,
116*0a9764feSAndroid Build Coastguard Worker                                  blending_enum_map_);
117*0a9764feSAndroid Build Coastguard Worker     blend_property_.AddEnumToMap("None", BufferBlendMode::kNone,
118*0a9764feSAndroid Build Coastguard Worker                                  blending_enum_map_);
119*0a9764feSAndroid Build Coastguard Worker   }
120*0a9764feSAndroid Build Coastguard Worker 
121*0a9764feSAndroid Build Coastguard Worker   GetPlaneProperty("IN_FENCE_FD", in_fence_fd_property_, Presence::kOptional);
122*0a9764feSAndroid Build Coastguard Worker 
123*0a9764feSAndroid Build Coastguard Worker   if (HasNonRgbFormat()) {
124*0a9764feSAndroid Build Coastguard Worker     if (GetPlaneProperty("COLOR_ENCODING", color_encoding_propery_,
125*0a9764feSAndroid Build Coastguard Worker                          Presence::kOptional)) {
126*0a9764feSAndroid Build Coastguard Worker       color_encoding_propery_.AddEnumToMap("ITU-R BT.709 YCbCr",
127*0a9764feSAndroid Build Coastguard Worker                                            BufferColorSpace::kItuRec709,
128*0a9764feSAndroid Build Coastguard Worker                                            color_encoding_enum_map_);
129*0a9764feSAndroid Build Coastguard Worker       color_encoding_propery_.AddEnumToMap("ITU-R BT.601 YCbCr",
130*0a9764feSAndroid Build Coastguard Worker                                            BufferColorSpace::kItuRec601,
131*0a9764feSAndroid Build Coastguard Worker                                            color_encoding_enum_map_);
132*0a9764feSAndroid Build Coastguard Worker       color_encoding_propery_.AddEnumToMap("ITU-R BT.2020 YCbCr",
133*0a9764feSAndroid Build Coastguard Worker                                            BufferColorSpace::kItuRec2020,
134*0a9764feSAndroid Build Coastguard Worker                                            color_encoding_enum_map_);
135*0a9764feSAndroid Build Coastguard Worker     }
136*0a9764feSAndroid Build Coastguard Worker 
137*0a9764feSAndroid Build Coastguard Worker     if (GetPlaneProperty("COLOR_RANGE", color_range_property_,
138*0a9764feSAndroid Build Coastguard Worker                          Presence::kOptional)) {
139*0a9764feSAndroid Build Coastguard Worker       color_range_property_.AddEnumToMap("YCbCr full range",
140*0a9764feSAndroid Build Coastguard Worker                                          BufferSampleRange::kFullRange,
141*0a9764feSAndroid Build Coastguard Worker                                          color_range_enum_map_);
142*0a9764feSAndroid Build Coastguard Worker       color_range_property_.AddEnumToMap("YCbCr limited range",
143*0a9764feSAndroid Build Coastguard Worker                                          BufferSampleRange::kLimitedRange,
144*0a9764feSAndroid Build Coastguard Worker                                          color_range_enum_map_);
145*0a9764feSAndroid Build Coastguard Worker     }
146*0a9764feSAndroid Build Coastguard Worker   }
147*0a9764feSAndroid Build Coastguard Worker 
148*0a9764feSAndroid Build Coastguard Worker   return 0;
149*0a9764feSAndroid Build Coastguard Worker }
150*0a9764feSAndroid Build Coastguard Worker 
IsCrtcSupported(const DrmCrtc & crtc) const151*0a9764feSAndroid Build Coastguard Worker bool DrmPlane::IsCrtcSupported(const DrmCrtc &crtc) const {
152*0a9764feSAndroid Build Coastguard Worker   auto crtc_prop_optval = crtc_property_.GetValue();
153*0a9764feSAndroid Build Coastguard Worker   auto crtc_prop_val = crtc_prop_optval ? *crtc_prop_optval : 0;
154*0a9764feSAndroid Build Coastguard Worker 
155*0a9764feSAndroid Build Coastguard Worker   if (crtc_prop_val != 0 && crtc_prop_val != crtc.GetId() &&
156*0a9764feSAndroid Build Coastguard Worker       GetType() == DRM_PLANE_TYPE_PRIMARY) {
157*0a9764feSAndroid Build Coastguard Worker     // Some DRM driver such as omap_drm allows sharing primary plane between
158*0a9764feSAndroid Build Coastguard Worker     // CRTCs, but the primary plane could not be shared if it has been used by
159*0a9764feSAndroid Build Coastguard Worker     // any CRTC already, which is protected by the plane_switching_crtc function
160*0a9764feSAndroid Build Coastguard Worker     // in the kernel drivers/gpu/drm/drm_atomic.c file.
161*0a9764feSAndroid Build Coastguard Worker     // The current drm_hwc design is not ready to support such scenario yet,
162*0a9764feSAndroid Build Coastguard Worker     // so adding the CRTC status check here to workaorund for now.
163*0a9764feSAndroid Build Coastguard Worker     return false;
164*0a9764feSAndroid Build Coastguard Worker   }
165*0a9764feSAndroid Build Coastguard Worker 
166*0a9764feSAndroid Build Coastguard Worker   return ((1 << crtc.GetIndexInResArray()) & plane_->possible_crtcs) != 0;
167*0a9764feSAndroid Build Coastguard Worker }
168*0a9764feSAndroid Build Coastguard Worker 
IsValidForLayer(LayerData * layer)169*0a9764feSAndroid Build Coastguard Worker bool DrmPlane::IsValidForLayer(LayerData *layer) {
170*0a9764feSAndroid Build Coastguard Worker   if (layer == nullptr || !layer->bi) {
171*0a9764feSAndroid Build Coastguard Worker     ALOGE("%s: Invalid parameters", __func__);
172*0a9764feSAndroid Build Coastguard Worker     return false;
173*0a9764feSAndroid Build Coastguard Worker   }
174*0a9764feSAndroid Build Coastguard Worker 
175*0a9764feSAndroid Build Coastguard Worker   if (!rotation_property_) {
176*0a9764feSAndroid Build Coastguard Worker     if (layer->pi.transform != LayerTransform::kIdentity) {
177*0a9764feSAndroid Build Coastguard Worker       ALOGV("No rotation property on plane %d", GetId());
178*0a9764feSAndroid Build Coastguard Worker       return false;
179*0a9764feSAndroid Build Coastguard Worker     }
180*0a9764feSAndroid Build Coastguard Worker   } else {
181*0a9764feSAndroid Build Coastguard Worker     if (transform_enum_map_.count(layer->pi.transform) == 0) {
182*0a9764feSAndroid Build Coastguard Worker       ALOGV("Transform is not supported on plane %d", GetId());
183*0a9764feSAndroid Build Coastguard Worker       return false;
184*0a9764feSAndroid Build Coastguard Worker     }
185*0a9764feSAndroid Build Coastguard Worker   }
186*0a9764feSAndroid Build Coastguard Worker 
187*0a9764feSAndroid Build Coastguard Worker   if (!alpha_property_ && layer->pi.alpha != UINT16_MAX) {
188*0a9764feSAndroid Build Coastguard Worker     ALOGV("Alpha is not supported on plane %d", GetId());
189*0a9764feSAndroid Build Coastguard Worker     return false;
190*0a9764feSAndroid Build Coastguard Worker   }
191*0a9764feSAndroid Build Coastguard Worker 
192*0a9764feSAndroid Build Coastguard Worker   if (blending_enum_map_.count(layer->bi->blend_mode) == 0 &&
193*0a9764feSAndroid Build Coastguard Worker       layer->bi->blend_mode != BufferBlendMode::kNone &&
194*0a9764feSAndroid Build Coastguard Worker       layer->bi->blend_mode != BufferBlendMode::kPreMult) {
195*0a9764feSAndroid Build Coastguard Worker     ALOGV("Blending is not supported on plane %d", GetId());
196*0a9764feSAndroid Build Coastguard Worker     return false;
197*0a9764feSAndroid Build Coastguard Worker   }
198*0a9764feSAndroid Build Coastguard Worker 
199*0a9764feSAndroid Build Coastguard Worker   auto format = layer->bi->format;
200*0a9764feSAndroid Build Coastguard Worker   if (!IsFormatSupported(format)) {
201*0a9764feSAndroid Build Coastguard Worker     ALOGV("Plane %d does not supports %c%c%c%c format", GetId(), format,
202*0a9764feSAndroid Build Coastguard Worker           format >> 8, format >> 16, format >> 24);
203*0a9764feSAndroid Build Coastguard Worker     return false;
204*0a9764feSAndroid Build Coastguard Worker   }
205*0a9764feSAndroid Build Coastguard Worker 
206*0a9764feSAndroid Build Coastguard Worker   return true;
207*0a9764feSAndroid Build Coastguard Worker }
208*0a9764feSAndroid Build Coastguard Worker 
IsFormatSupported(uint32_t format) const209*0a9764feSAndroid Build Coastguard Worker bool DrmPlane::IsFormatSupported(uint32_t format) const {
210*0a9764feSAndroid Build Coastguard Worker   return std::find(std::begin(formats_), std::end(formats_), format) !=
211*0a9764feSAndroid Build Coastguard Worker          std::end(formats_);
212*0a9764feSAndroid Build Coastguard Worker }
213*0a9764feSAndroid Build Coastguard Worker 
HasNonRgbFormat() const214*0a9764feSAndroid Build Coastguard Worker bool DrmPlane::HasNonRgbFormat() const {
215*0a9764feSAndroid Build Coastguard Worker   return std::find_if_not(std::begin(formats_), std::end(formats_),
216*0a9764feSAndroid Build Coastguard Worker                           [](uint32_t format) {
217*0a9764feSAndroid Build Coastguard Worker                             return BufferInfoGetter::IsDrmFormatRgb(format);
218*0a9764feSAndroid Build Coastguard Worker                           }) != std::end(formats_);
219*0a9764feSAndroid Build Coastguard Worker }
220*0a9764feSAndroid Build Coastguard Worker 
ToDrmRotation(LayerTransform transform)221*0a9764feSAndroid Build Coastguard Worker static uint64_t ToDrmRotation(LayerTransform transform) {
222*0a9764feSAndroid Build Coastguard Worker   uint64_t rotation = 0;
223*0a9764feSAndroid Build Coastguard Worker   /* DRM/KMS uses counter-clockwise rotations, while HWC API uses
224*0a9764feSAndroid Build Coastguard Worker    * clockwise. That's why 90 and 270 are swapped here.
225*0a9764feSAndroid Build Coastguard Worker    */
226*0a9764feSAndroid Build Coastguard Worker   if ((transform & LayerTransform::kFlipH) != 0)
227*0a9764feSAndroid Build Coastguard Worker     rotation |= DRM_MODE_REFLECT_X;
228*0a9764feSAndroid Build Coastguard Worker   if ((transform & LayerTransform::kFlipV) != 0)
229*0a9764feSAndroid Build Coastguard Worker     rotation |= DRM_MODE_REFLECT_Y;
230*0a9764feSAndroid Build Coastguard Worker   if ((transform & LayerTransform::kRotate90) != 0)
231*0a9764feSAndroid Build Coastguard Worker     rotation |= DRM_MODE_ROTATE_270;
232*0a9764feSAndroid Build Coastguard Worker   else if ((transform & LayerTransform::kRotate180) != 0)
233*0a9764feSAndroid Build Coastguard Worker     rotation |= DRM_MODE_ROTATE_180;
234*0a9764feSAndroid Build Coastguard Worker   else if ((transform & LayerTransform::kRotate270) != 0)
235*0a9764feSAndroid Build Coastguard Worker     rotation |= DRM_MODE_ROTATE_90;
236*0a9764feSAndroid Build Coastguard Worker   else
237*0a9764feSAndroid Build Coastguard Worker     rotation |= DRM_MODE_ROTATE_0;
238*0a9764feSAndroid Build Coastguard Worker 
239*0a9764feSAndroid Build Coastguard Worker   return rotation;
240*0a9764feSAndroid Build Coastguard Worker }
241*0a9764feSAndroid Build Coastguard Worker 
242*0a9764feSAndroid Build Coastguard Worker /* Convert float to 16.16 fixed point */
To1616FixPt(float in)243*0a9764feSAndroid Build Coastguard Worker static int To1616FixPt(float in) {
244*0a9764feSAndroid Build Coastguard Worker   constexpr int kBitShift = 16;
245*0a9764feSAndroid Build Coastguard Worker   return int(in * (1 << kBitShift));
246*0a9764feSAndroid Build Coastguard Worker }
247*0a9764feSAndroid Build Coastguard Worker 
AtomicSetState(drmModeAtomicReq & pset,LayerData & layer,uint32_t zpos,uint32_t crtc_id)248*0a9764feSAndroid Build Coastguard Worker auto DrmPlane::AtomicSetState(drmModeAtomicReq &pset, LayerData &layer,
249*0a9764feSAndroid Build Coastguard Worker                               uint32_t zpos, uint32_t crtc_id) -> int {
250*0a9764feSAndroid Build Coastguard Worker   if (!layer.fb || !layer.bi) {
251*0a9764feSAndroid Build Coastguard Worker     ALOGE("%s: Invalid arguments", __func__);
252*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
253*0a9764feSAndroid Build Coastguard Worker   }
254*0a9764feSAndroid Build Coastguard Worker 
255*0a9764feSAndroid Build Coastguard Worker   if (zpos_property_ && !zpos_property_.IsImmutable()) {
256*0a9764feSAndroid Build Coastguard Worker     uint64_t min_zpos = 0;
257*0a9764feSAndroid Build Coastguard Worker 
258*0a9764feSAndroid Build Coastguard Worker     // Ignore ret and use min_zpos as 0 by default
259*0a9764feSAndroid Build Coastguard Worker     std::tie(std::ignore, min_zpos) = zpos_property_.RangeMin();
260*0a9764feSAndroid Build Coastguard Worker 
261*0a9764feSAndroid Build Coastguard Worker     if (!zpos_property_.AtomicSet(pset, zpos + min_zpos)) {
262*0a9764feSAndroid Build Coastguard Worker       return -EINVAL;
263*0a9764feSAndroid Build Coastguard Worker     }
264*0a9764feSAndroid Build Coastguard Worker   }
265*0a9764feSAndroid Build Coastguard Worker 
266*0a9764feSAndroid Build Coastguard Worker   if (layer.acquire_fence &&
267*0a9764feSAndroid Build Coastguard Worker       !in_fence_fd_property_.AtomicSet(pset, *layer.acquire_fence)) {
268*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
269*0a9764feSAndroid Build Coastguard Worker   }
270*0a9764feSAndroid Build Coastguard Worker 
271*0a9764feSAndroid Build Coastguard Worker   auto &disp = layer.pi.display_frame;
272*0a9764feSAndroid Build Coastguard Worker   auto &src = layer.pi.source_crop;
273*0a9764feSAndroid Build Coastguard Worker   if (!crtc_property_.AtomicSet(pset, crtc_id) ||
274*0a9764feSAndroid Build Coastguard Worker       !fb_property_.AtomicSet(pset, layer.fb->GetFbId()) ||
275*0a9764feSAndroid Build Coastguard Worker       !crtc_x_property_.AtomicSet(pset, disp.left) ||
276*0a9764feSAndroid Build Coastguard Worker       !crtc_y_property_.AtomicSet(pset, disp.top) ||
277*0a9764feSAndroid Build Coastguard Worker       !crtc_w_property_.AtomicSet(pset, disp.right - disp.left) ||
278*0a9764feSAndroid Build Coastguard Worker       !crtc_h_property_.AtomicSet(pset, disp.bottom - disp.top) ||
279*0a9764feSAndroid Build Coastguard Worker       !src_x_property_.AtomicSet(pset, To1616FixPt(src.left)) ||
280*0a9764feSAndroid Build Coastguard Worker       !src_y_property_.AtomicSet(pset, To1616FixPt(src.top)) ||
281*0a9764feSAndroid Build Coastguard Worker       !src_w_property_.AtomicSet(pset, To1616FixPt(src.right - src.left)) ||
282*0a9764feSAndroid Build Coastguard Worker       !src_h_property_.AtomicSet(pset, To1616FixPt(src.bottom - src.top))) {
283*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
284*0a9764feSAndroid Build Coastguard Worker   }
285*0a9764feSAndroid Build Coastguard Worker 
286*0a9764feSAndroid Build Coastguard Worker   if (rotation_property_ &&
287*0a9764feSAndroid Build Coastguard Worker       !rotation_property_.AtomicSet(pset, ToDrmRotation(layer.pi.transform))) {
288*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
289*0a9764feSAndroid Build Coastguard Worker   }
290*0a9764feSAndroid Build Coastguard Worker 
291*0a9764feSAndroid Build Coastguard Worker   if (alpha_property_ && !alpha_property_.AtomicSet(pset, layer.pi.alpha)) {
292*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
293*0a9764feSAndroid Build Coastguard Worker   }
294*0a9764feSAndroid Build Coastguard Worker 
295*0a9764feSAndroid Build Coastguard Worker   if (blending_enum_map_.count(layer.bi->blend_mode) != 0 &&
296*0a9764feSAndroid Build Coastguard Worker       !blend_property_.AtomicSet(pset,
297*0a9764feSAndroid Build Coastguard Worker                                  blending_enum_map_[layer.bi->blend_mode])) {
298*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
299*0a9764feSAndroid Build Coastguard Worker   }
300*0a9764feSAndroid Build Coastguard Worker 
301*0a9764feSAndroid Build Coastguard Worker   if (color_encoding_enum_map_.count(layer.bi->color_space) != 0 &&
302*0a9764feSAndroid Build Coastguard Worker       !color_encoding_propery_
303*0a9764feSAndroid Build Coastguard Worker            .AtomicSet(pset, color_encoding_enum_map_[layer.bi->color_space])) {
304*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
305*0a9764feSAndroid Build Coastguard Worker   }
306*0a9764feSAndroid Build Coastguard Worker 
307*0a9764feSAndroid Build Coastguard Worker   if (color_range_enum_map_.count(layer.bi->sample_range) != 0 &&
308*0a9764feSAndroid Build Coastguard Worker       !color_range_property_
309*0a9764feSAndroid Build Coastguard Worker            .AtomicSet(pset, color_range_enum_map_[layer.bi->sample_range])) {
310*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
311*0a9764feSAndroid Build Coastguard Worker   }
312*0a9764feSAndroid Build Coastguard Worker 
313*0a9764feSAndroid Build Coastguard Worker   return 0;
314*0a9764feSAndroid Build Coastguard Worker }
315*0a9764feSAndroid Build Coastguard Worker 
AtomicDisablePlane(drmModeAtomicReq & pset)316*0a9764feSAndroid Build Coastguard Worker auto DrmPlane::AtomicDisablePlane(drmModeAtomicReq &pset) -> int {
317*0a9764feSAndroid Build Coastguard Worker   if (!crtc_property_.AtomicSet(pset, 0) || !fb_property_.AtomicSet(pset, 0)) {
318*0a9764feSAndroid Build Coastguard Worker     return -EINVAL;
319*0a9764feSAndroid Build Coastguard Worker   }
320*0a9764feSAndroid Build Coastguard Worker 
321*0a9764feSAndroid Build Coastguard Worker   return 0;
322*0a9764feSAndroid Build Coastguard Worker }
323*0a9764feSAndroid Build Coastguard Worker 
GetPlaneProperty(const char * prop_name,DrmProperty & property,Presence presence)324*0a9764feSAndroid Build Coastguard Worker auto DrmPlane::GetPlaneProperty(const char *prop_name, DrmProperty &property,
325*0a9764feSAndroid Build Coastguard Worker                                 Presence presence) -> bool {
326*0a9764feSAndroid Build Coastguard Worker   auto err = drm_->GetProperty(GetId(), DRM_MODE_OBJECT_PLANE, prop_name,
327*0a9764feSAndroid Build Coastguard Worker                                &property);
328*0a9764feSAndroid Build Coastguard Worker   if (err != 0) {
329*0a9764feSAndroid Build Coastguard Worker     if (presence == Presence::kMandatory) {
330*0a9764feSAndroid Build Coastguard Worker       ALOGE("Could not get mandatory property \"%s\" from plane %d", prop_name,
331*0a9764feSAndroid Build Coastguard Worker             GetId());
332*0a9764feSAndroid Build Coastguard Worker     } else {
333*0a9764feSAndroid Build Coastguard Worker       ALOGV("Could not get optional property \"%s\" from plane %d", prop_name,
334*0a9764feSAndroid Build Coastguard Worker             GetId());
335*0a9764feSAndroid Build Coastguard Worker     }
336*0a9764feSAndroid Build Coastguard Worker     return false;
337*0a9764feSAndroid Build Coastguard Worker   }
338*0a9764feSAndroid Build Coastguard Worker 
339*0a9764feSAndroid Build Coastguard Worker   return true;
340*0a9764feSAndroid Build Coastguard Worker }
341*0a9764feSAndroid Build Coastguard Worker 
342*0a9764feSAndroid Build Coastguard Worker }  // namespace android
343