blob: ded8acd5e0481ac5a30fd47f447de62fd3517132 [file] [log] [blame]
Roman Stratiienko03fd35c2022-01-04 14:30:37 +02001/*
2 * Copyright (C) 2022 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
Sean Paul468a7542024-07-16 19:50:58 +000017#define LOG_TAG "drmhwc"
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020018
19#include "HwcLayer.h"
20
Roman Stratiienko4b2cc482022-02-21 14:53:58 +020021#include "HwcDisplay.h"
22#include "bufferinfo/BufferInfoGetter.h"
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020023#include "utils/log.h"
24
25namespace android {
26
Drew Davenporta241a772024-09-24 11:26:30 -060027void HwcLayer::SetLayerProperties(const LayerProperties& layer_properties) {
28 if (layer_properties.blend_mode) {
29 blend_mode_ = layer_properties.blend_mode.value();
30 }
Drew Davenportac9681e2024-09-24 12:17:34 -060031 if (layer_properties.color_space) {
32 color_space_ = layer_properties.color_space.value();
33 }
34 if (layer_properties.sample_range) {
35 sample_range_ = layer_properties.sample_range.value();
36 }
Drew Davenport1b0d8b72024-09-24 15:31:38 -060037 if (layer_properties.composition_type) {
38 sf_type_ = layer_properties.composition_type.value();
39 }
Drew Davenport22d66b42024-09-24 15:34:57 -060040 if (layer_properties.display_frame) {
41 layer_data_.pi.display_frame = layer_properties.display_frame.value();
42 }
Drew Davenport07b96f02024-09-24 15:37:12 -060043 if (layer_properties.alpha) {
44 layer_data_.pi.alpha = std::lround(layer_properties.alpha.value() *
45 UINT16_MAX);
46 }
Drew Davenport7ab8c182024-09-24 17:04:26 -060047 if (layer_properties.source_crop) {
48 layer_data_.pi.source_crop = layer_properties.source_crop.value();
49 }
Drew Davenport51c61e42024-09-24 17:13:39 -060050 if (layer_properties.transform) {
51 layer_data_.pi.transform = layer_properties.transform.value();
52 }
Drew Davenporta241a772024-09-24 11:26:30 -060053}
54
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020055// NOLINTNEXTLINE(readability-convert-member-functions-to-static)
56HWC2::Error HwcLayer::SetCursorPosition(int32_t /*x*/, int32_t /*y*/) {
57 return HWC2::Error::None;
58}
59
60HWC2::Error HwcLayer::SetLayerBlendMode(int32_t mode) {
61 switch (static_cast<HWC2::BlendMode>(mode)) {
62 case HWC2::BlendMode::None:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +020063 blend_mode_ = BufferBlendMode::kNone;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020064 break;
65 case HWC2::BlendMode::Premultiplied:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +020066 blend_mode_ = BufferBlendMode::kPreMult;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020067 break;
68 case HWC2::BlendMode::Coverage:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +020069 blend_mode_ = BufferBlendMode::kCoverage;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020070 break;
71 default:
John Stultz71d983d2024-02-13 20:40:36 -080072 ALOGE("Unknown blending mode b=%d", mode);
Roman Stratiienko4b2cc482022-02-21 14:53:58 +020073 blend_mode_ = BufferBlendMode::kUndefined;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020074 break;
75 }
76 return HWC2::Error::None;
77}
78
79/* Find API details at:
80 * https://cs.android.com/android/platform/superproject/+/android-11.0.0_r3:hardware/libhardware/include/hardware/hwcomposer2.h;l=2314
81 */
82HWC2::Error HwcLayer::SetLayerBuffer(buffer_handle_t buffer,
83 int32_t acquire_fence) {
Roman Stratiienko359a9d32023-01-16 17:41:07 +020084 layer_data_.acquire_fence = MakeSharedFd(acquire_fence);
Roman Stratiienko4b2cc482022-02-21 14:53:58 +020085 buffer_handle_ = buffer;
86 buffer_handle_updated_ = true;
87
Roman Stratiienko03fd35c2022-01-04 14:30:37 +020088 return HWC2::Error::None;
89}
90
91// NOLINTNEXTLINE(readability-convert-member-functions-to-static)
92HWC2::Error HwcLayer::SetLayerColor(hwc_color_t /*color*/) {
93 // TODO(nobody): Put to client composition here?
94 return HWC2::Error::None;
95}
96
97HWC2::Error HwcLayer::SetLayerCompositionType(int32_t type) {
98 sf_type_ = static_cast<HWC2::Composition>(type);
99 return HWC2::Error::None;
100}
101
102HWC2::Error HwcLayer::SetLayerDataspace(int32_t dataspace) {
103 switch (dataspace & HAL_DATASPACE_STANDARD_MASK) {
104 case HAL_DATASPACE_STANDARD_BT709:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200105 color_space_ = BufferColorSpace::kItuRec709;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200106 break;
107 case HAL_DATASPACE_STANDARD_BT601_625:
108 case HAL_DATASPACE_STANDARD_BT601_625_UNADJUSTED:
109 case HAL_DATASPACE_STANDARD_BT601_525:
110 case HAL_DATASPACE_STANDARD_BT601_525_UNADJUSTED:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200111 color_space_ = BufferColorSpace::kItuRec601;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200112 break;
113 case HAL_DATASPACE_STANDARD_BT2020:
114 case HAL_DATASPACE_STANDARD_BT2020_CONSTANT_LUMINANCE:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200115 color_space_ = BufferColorSpace::kItuRec2020;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200116 break;
117 default:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200118 color_space_ = BufferColorSpace::kUndefined;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200119 }
120
121 switch (dataspace & HAL_DATASPACE_RANGE_MASK) {
122 case HAL_DATASPACE_RANGE_FULL:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200123 sample_range_ = BufferSampleRange::kFullRange;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200124 break;
125 case HAL_DATASPACE_RANGE_LIMITED:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200126 sample_range_ = BufferSampleRange::kLimitedRange;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200127 break;
128 default:
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200129 sample_range_ = BufferSampleRange::kUndefined;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200130 }
131 return HWC2::Error::None;
132}
133
134HWC2::Error HwcLayer::SetLayerDisplayFrame(hwc_rect_t frame) {
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200135 layer_data_.pi.display_frame = frame;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200136 return HWC2::Error::None;
137}
138
139HWC2::Error HwcLayer::SetLayerPlaneAlpha(float alpha) {
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200140 layer_data_.pi.alpha = std::lround(alpha * UINT16_MAX);
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200141 return HWC2::Error::None;
142}
143
144// NOLINTNEXTLINE(readability-convert-member-functions-to-static)
145HWC2::Error HwcLayer::SetLayerSidebandStream(
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200146 const native_handle_t* /*stream*/) {
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200147 // TODO(nobody): We don't support sideband
148 return HWC2::Error::Unsupported;
149}
150
151HWC2::Error HwcLayer::SetLayerSourceCrop(hwc_frect_t crop) {
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200152 layer_data_.pi.source_crop = crop;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200153 return HWC2::Error::None;
154}
155
156// NOLINTNEXTLINE(readability-convert-member-functions-to-static)
157HWC2::Error HwcLayer::SetLayerSurfaceDamage(hwc_region_t /*damage*/) {
158 // TODO(nobody): We don't use surface damage, marking as unsupported
159 return HWC2::Error::None;
160}
161
162HWC2::Error HwcLayer::SetLayerTransform(int32_t transform) {
163 uint32_t l_transform = 0;
164
165 // 270* and 180* cannot be combined with flips. More specifically, they
166 // already contain both horizontal and vertical flips, so those fields are
167 // redundant in this case. 90* rotation can be combined with either horizontal
168 // flip or vertical flip, so treat it differently
169 if (transform == HWC_TRANSFORM_ROT_270) {
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200170 l_transform = LayerTransform::kRotate270;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200171 } else if (transform == HWC_TRANSFORM_ROT_180) {
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200172 l_transform = LayerTransform::kRotate180;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200173 } else {
174 if ((transform & HWC_TRANSFORM_FLIP_H) != 0)
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200175 l_transform |= LayerTransform::kFlipH;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200176 if ((transform & HWC_TRANSFORM_FLIP_V) != 0)
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200177 l_transform |= LayerTransform::kFlipV;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200178 if ((transform & HWC_TRANSFORM_ROT_90) != 0)
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200179 l_transform |= LayerTransform::kRotate90;
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200180 }
181
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200182 layer_data_.pi.transform = static_cast<LayerTransform>(l_transform);
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200183 return HWC2::Error::None;
184}
185
186// NOLINTNEXTLINE(readability-convert-member-functions-to-static)
187HWC2::Error HwcLayer::SetLayerVisibleRegion(hwc_region_t /*visible*/) {
188 // TODO(nobody): We don't use this information, marking as unsupported
189 return HWC2::Error::None;
190}
191
192HWC2::Error HwcLayer::SetLayerZOrder(uint32_t order) {
193 z_order_ = order;
194 return HWC2::Error::None;
195}
196
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200197void HwcLayer::ImportFb() {
198 if (!IsLayerUsableAsDevice() || !buffer_handle_updated_) {
199 return;
200 }
201 buffer_handle_updated_ = false;
202
203 layer_data_.fb = {};
204
Roman Stratiienkoa32f9072022-05-13 12:12:20 +0300205 auto unique_id = BufferInfoGetter::GetInstance()->GetUniqueId(buffer_handle_);
206 if (unique_id && SwChainGetBufferFromCache(*unique_id)) {
207 return;
208 }
209
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200210 layer_data_.bi = BufferInfoGetter::GetInstance()->GetBoInfo(buffer_handle_);
211 if (!layer_data_.bi) {
212 ALOGW("Unable to get buffer information (0x%p)", buffer_handle_);
213 bi_get_failed_ = true;
214 return;
215 }
216
217 layer_data_
218 .fb = parent_->GetPipe().device->GetDrmFbImporter().GetOrCreateFbId(
219 &layer_data_.bi.value());
220
221 if (!layer_data_.fb) {
222 ALOGV("Unable to create framebuffer object for buffer 0x%p",
223 buffer_handle_);
224 fb_import_failed_ = true;
225 return;
226 }
Roman Stratiienkoa32f9072022-05-13 12:12:20 +0300227
228 if (unique_id) {
229 SwChainAddCurrentBuffer(*unique_id);
230 }
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200231}
232
Roman Stratiienko359a9d32023-01-16 17:41:07 +0200233void HwcLayer::PopulateLayerData() {
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200234 ImportFb();
235
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300236 if (!layer_data_.bi) {
237 ALOGE("%s: Invalid state", __func__);
238 return;
239 }
240
Roman Stratiienko4b2cc482022-02-21 14:53:58 +0200241 if (blend_mode_ != BufferBlendMode::kUndefined) {
242 layer_data_.bi->blend_mode = blend_mode_;
243 }
244 if (color_space_ != BufferColorSpace::kUndefined) {
245 layer_data_.bi->color_space = color_space_;
246 }
247 if (sample_range_ != BufferSampleRange::kUndefined) {
248 layer_data_.bi->sample_range = sample_range_;
249 }
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200250}
251
Roman Stratiienkoa32f9072022-05-13 12:12:20 +0300252/* SwapChain Cache */
253
254bool HwcLayer::SwChainGetBufferFromCache(BufferUniqueId unique_id) {
255 if (swchain_lookup_table_.count(unique_id) == 0) {
256 return false;
257 }
258
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300259 auto seq = swchain_lookup_table_[unique_id];
Roman Stratiienkoa32f9072022-05-13 12:12:20 +0300260
261 if (swchain_cache_.count(seq) == 0) {
262 return false;
263 }
264
265 auto& el = swchain_cache_[seq];
266 if (!el.bi) {
267 return false;
268 }
269
270 layer_data_.bi = el.bi;
271 layer_data_.fb = el.fb;
272
273 return true;
274}
275
276void HwcLayer::SwChainReassemble(BufferUniqueId unique_id) {
277 if (swchain_lookup_table_.count(unique_id) != 0) {
278 if (swchain_lookup_table_[unique_id] ==
279 int(swchain_lookup_table_.size()) - 1) {
280 /* Skip same buffer */
281 return;
282 }
283 if (swchain_lookup_table_[unique_id] == 0) {
284 swchain_reassembled_ = true;
285 return;
286 }
287 /* Tracking error */
288 SwChainClearCache();
289 return;
290 }
291
292 swchain_lookup_table_[unique_id] = int(swchain_lookup_table_.size());
293}
294
295void HwcLayer::SwChainAddCurrentBuffer(BufferUniqueId unique_id) {
296 if (!swchain_reassembled_) {
297 SwChainReassemble(unique_id);
298 }
299
300 if (swchain_reassembled_) {
301 if (swchain_lookup_table_.count(unique_id) == 0) {
302 SwChainClearCache();
303 return;
304 }
305
Roman Stratiienkoa7913de2022-10-20 13:18:57 +0300306 auto seq = swchain_lookup_table_[unique_id];
Roman Stratiienkoa32f9072022-05-13 12:12:20 +0300307
308 if (swchain_cache_.count(seq) == 0) {
309 swchain_cache_[seq] = {};
310 }
311
312 swchain_cache_[seq].bi = layer_data_.bi;
313 swchain_cache_[seq].fb = layer_data_.fb;
314 }
315}
316
317void HwcLayer::SwChainClearCache() {
318 swchain_cache_.clear();
319 swchain_lookup_table_.clear();
320 swchain_reassembled_ = false;
321}
322
Roman Stratiienko03fd35c2022-01-04 14:30:37 +0200323} // namespace android