blob: 93a54445a033d36b4f0c267db61089204c58c15d [file] [log] [blame]
Marissa Wallf6a73fa2018-12-10 10:41:08 -08001/*
2 * Copyright 2018 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
Marissa Wall1be5a102019-01-18 16:14:04 -080017#include <android/hardware/configstore/1.0/ISurfaceFlingerConfigs.h>
Marissa Wallf6a73fa2018-12-10 10:41:08 -080018#include <android/native_window.h>
19#include <android/surface_control.h>
Jorim Jaggi71db8892021-02-03 23:19:29 +010020#include <surface_control_private.h>
Marissa Wallf6a73fa2018-12-10 10:41:08 -080021
Marissa Wall1be5a102019-01-18 16:14:04 -080022#include <configstore/Utils.h>
23
24#include <gui/HdrMetadata.h>
25#include <gui/ISurfaceComposer.h>
Marissa Wallf6a73fa2018-12-10 10:41:08 -080026#include <gui/Surface.h>
27#include <gui/SurfaceComposerClient.h>
28#include <gui/SurfaceControl.h>
29
Marin Shalamanov463ad8e2021-01-28 22:58:37 +010030#include <ui/DynamicDisplayInfo.h>
Marissa Wall1be5a102019-01-18 16:14:04 -080031
32#include <utils/Timers.h>
33
34using namespace android::hardware::configstore;
35using namespace android::hardware::configstore::V1_0;
Marissa Wallf6a73fa2018-12-10 10:41:08 -080036using namespace android;
37
38using Transaction = SurfaceComposerClient::Transaction;
39
40#define CHECK_NOT_NULL(name) \
41 LOG_ALWAYS_FATAL_IF(name == nullptr, "nullptr passed as " #name " argument");
42
43#define CHECK_VALID_RECT(name) \
44 LOG_ALWAYS_FATAL_IF(!static_cast<const Rect&>(name).isValid(), \
45 "invalid arg passed as " #name " argument");
46
Valerie Hau5bbfd512019-01-22 17:39:43 -080047static bool getWideColorSupport(const sp<SurfaceControl>& surfaceControl) {
48 sp<SurfaceComposerClient> client = surfaceControl->getClient();
Dominik Laskowski3316a0a2019-01-25 02:56:41 -080049
50 const sp<IBinder> display = client->getInternalDisplayToken();
51 if (display == nullptr) {
52 ALOGE("unable to get wide color support for disconnected internal display");
53 return false;
54 }
55
Peiyong Lin0d04f322019-01-24 17:49:46 -080056 bool isWideColorDisplay = false;
57 status_t err = client->isWideColorDisplay(display, &isWideColorDisplay);
Valerie Hau5bbfd512019-01-22 17:39:43 -080058 if (err) {
59 ALOGE("unable to get wide color support");
60 return false;
61 }
Peiyong Lin0d04f322019-01-24 17:49:46 -080062 return isWideColorDisplay;
Valerie Hau5bbfd512019-01-22 17:39:43 -080063}
64
65static bool getHdrSupport(const sp<SurfaceControl>& surfaceControl) {
66 sp<SurfaceComposerClient> client = surfaceControl->getClient();
Dominik Laskowski3316a0a2019-01-25 02:56:41 -080067
68 const sp<IBinder> display = client->getInternalDisplayToken();
69 if (display == nullptr) {
70 ALOGE("unable to get hdr capabilities for disconnected internal display");
71 return false;
72 }
Valerie Hau5bbfd512019-01-22 17:39:43 -080073
Marin Shalamanov463ad8e2021-01-28 22:58:37 +010074 ui::DynamicDisplayInfo info;
75 if (status_t err = client->getDynamicDisplayInfo(display, &info); err != NO_ERROR) {
Valerie Hau5bbfd512019-01-22 17:39:43 -080076 ALOGE("unable to get hdr capabilities");
Marin Shalamanov463ad8e2021-01-28 22:58:37 +010077 return err;
Valerie Hau5bbfd512019-01-22 17:39:43 -080078 }
79
Marin Shalamanov463ad8e2021-01-28 22:58:37 +010080 return !info.hdrCapabilities.getSupportedHdrTypes().empty();
Valerie Hau5bbfd512019-01-22 17:39:43 -080081}
82
83static bool isDataSpaceValid(const sp<SurfaceControl>& surfaceControl, ADataSpace dataSpace) {
84 static_assert(static_cast<int>(ADATASPACE_UNKNOWN) == static_cast<int>(HAL_DATASPACE_UNKNOWN));
85 static_assert(static_cast<int>(ADATASPACE_SCRGB_LINEAR) == static_cast<int>(HAL_DATASPACE_V0_SCRGB_LINEAR));
86 static_assert(static_cast<int>(ADATASPACE_SRGB) == static_cast<int>(HAL_DATASPACE_V0_SRGB));
87 static_assert(static_cast<int>(ADATASPACE_SCRGB) == static_cast<int>(HAL_DATASPACE_V0_SCRGB));
88 static_assert(static_cast<int>(ADATASPACE_DISPLAY_P3) == static_cast<int>(HAL_DATASPACE_DISPLAY_P3));
89 static_assert(static_cast<int>(ADATASPACE_BT2020_PQ) == static_cast<int>(HAL_DATASPACE_BT2020_PQ));
90
91 switch (static_cast<android_dataspace_t>(dataSpace)) {
92 case HAL_DATASPACE_UNKNOWN:
93 case HAL_DATASPACE_V0_SRGB:
94 return true;
95 // These data space need wide gamut support.
96 case HAL_DATASPACE_V0_SCRGB_LINEAR:
97 case HAL_DATASPACE_V0_SCRGB:
98 case HAL_DATASPACE_DISPLAY_P3:
99 return getWideColorSupport(surfaceControl);
100 // These data space need HDR support.
101 case HAL_DATASPACE_BT2020_PQ:
Valerie Hau5e18c432019-08-26 12:48:49 -0700102 if (!getHdrSupport(surfaceControl)) {
103 ALOGE("Invalid dataspace - device does not support hdr");
104 return false;
105 }
106 return true;
Valerie Hau5bbfd512019-01-22 17:39:43 -0800107 default:
108 return false;
109 }
110}
111
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800112Transaction* ASurfaceTransaction_to_Transaction(ASurfaceTransaction* aSurfaceTransaction) {
113 return reinterpret_cast<Transaction*>(aSurfaceTransaction);
114}
115
116SurfaceControl* ASurfaceControl_to_SurfaceControl(ASurfaceControl* aSurfaceControl) {
117 return reinterpret_cast<SurfaceControl*>(aSurfaceControl);
118}
119
120void SurfaceControl_acquire(SurfaceControl* surfaceControl) {
121 // incStrong/decStrong token must be the same, doesn't matter what it is
122 surfaceControl->incStrong((void*)SurfaceControl_acquire);
123}
124
125void SurfaceControl_release(SurfaceControl* surfaceControl) {
126 // incStrong/decStrong token must be the same, doesn't matter what it is
127 surfaceControl->decStrong((void*)SurfaceControl_acquire);
128}
129
130ASurfaceControl* ASurfaceControl_createFromWindow(ANativeWindow* window, const char* debug_name) {
131 CHECK_NOT_NULL(window);
132 CHECK_NOT_NULL(debug_name);
133
134 sp<SurfaceComposerClient> client = new SurfaceComposerClient();
135 if (client->initCheck() != NO_ERROR) {
136 return nullptr;
137 }
138
Vishnu Nairce1a6482020-10-22 17:41:30 -0700139 Surface* surface = static_cast<Surface*>(window);
140 sp<IBinder> parentHandle = surface->getSurfaceControlHandle();
141
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800142 uint32_t flags = ISurfaceComposerClient::eFXSurfaceBufferState;
Vishnu Nairce1a6482020-10-22 17:41:30 -0700143 sp<SurfaceControl> surfaceControl;
144 if (parentHandle) {
145 surfaceControl =
146 client->createSurface(String8(debug_name), 0 /* width */, 0 /* height */,
147 // Format is only relevant for buffer queue layers.
148 PIXEL_FORMAT_UNKNOWN /* format */, flags, parentHandle);
149 } else {
150 surfaceControl =
151 client->createWithSurfaceParent(String8(debug_name), 0 /* width */, 0 /* height */,
152 // Format is only relevant for buffer queue layers.
153 PIXEL_FORMAT_UNKNOWN /* format */, flags,
154 static_cast<Surface*>(window));
155 }
156
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800157 if (!surfaceControl) {
158 return nullptr;
159 }
160
161 SurfaceControl_acquire(surfaceControl.get());
162 return reinterpret_cast<ASurfaceControl*>(surfaceControl.get());
163}
164
165ASurfaceControl* ASurfaceControl_create(ASurfaceControl* parent, const char* debug_name) {
166 CHECK_NOT_NULL(parent);
167 CHECK_NOT_NULL(debug_name);
168
169 SurfaceComposerClient* client = ASurfaceControl_to_SurfaceControl(parent)->getClient().get();
170
171 SurfaceControl* surfaceControlParent = ASurfaceControl_to_SurfaceControl(parent);
172
173 uint32_t flags = ISurfaceComposerClient::eFXSurfaceBufferState;
174 sp<SurfaceControl> surfaceControl =
175 client->createSurface(String8(debug_name), 0 /* width */, 0 /* height */,
176 // Format is only relevant for buffer queue layers.
177 PIXEL_FORMAT_UNKNOWN /* format */, flags,
Vishnu Nairce1a6482020-10-22 17:41:30 -0700178 surfaceControlParent->getHandle());
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800179 if (!surfaceControl) {
180 return nullptr;
181 }
182
183 SurfaceControl_acquire(surfaceControl.get());
184 return reinterpret_cast<ASurfaceControl*>(surfaceControl.get());
185}
186
Huihong Luo91697e12021-01-28 15:24:19 -0800187void ASurfaceControl_acquire(ASurfaceControl* aSurfaceControl) {
188 SurfaceControl* surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800189
Huihong Luo91697e12021-01-28 15:24:19 -0800190 SurfaceControl_acquire(surfaceControl);
191}
192
193void ASurfaceControl_release(ASurfaceControl* aSurfaceControl) {
194 SurfaceControl* surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
195
196 SurfaceControl_release(surfaceControl);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800197}
198
Jorim Jaggi71db8892021-02-03 23:19:29 +0100199struct ASurfaceControlStats {
200 int64_t acquireTime;
201 sp<Fence> previousReleaseFence;
202 uint64_t frameNumber;
203};
204
205void ASurfaceControl_registerSurfaceStatsListener(ASurfaceControl* control, void* context,
206 ASurfaceControl_SurfaceStatsListener func) {
207 SurfaceStatsCallback callback = [func](void* callback_context,
208 nsecs_t,
209 const sp<Fence>&,
210 const SurfaceStats& surfaceStats) {
211
212 ASurfaceControlStats aSurfaceControlStats;
213
214 ASurfaceControl* aSurfaceControl =
215 reinterpret_cast<ASurfaceControl*>(surfaceStats.surfaceControl.get());
216 aSurfaceControlStats.acquireTime = surfaceStats.acquireTime;
217 aSurfaceControlStats.previousReleaseFence = surfaceStats.previousReleaseFence;
218 aSurfaceControlStats.frameNumber = surfaceStats.eventStats.frameNumber;
219
220 (*func)(callback_context, aSurfaceControl, &aSurfaceControlStats);
221 };
222 TransactionCompletedListener::getInstance()->addSurfaceStatsListener(context,
223 reinterpret_cast<void*>(func), ASurfaceControl_to_SurfaceControl(control), callback);
224}
225
226
227void ASurfaceControl_unregisterSurfaceStatsListener(void* context,
228 ASurfaceControl_SurfaceStatsListener func) {
229 TransactionCompletedListener::getInstance()->removeSurfaceStatsListener(context,
230 reinterpret_cast<void*>(func));
231}
232
233int64_t ASurfaceControlStats_getAcquireTime(ASurfaceControlStats* stats) {
234 return stats->acquireTime;
235}
236
237uint64_t ASurfaceControlStats_getFrameNumber(ASurfaceControlStats* stats) {
238 return stats->frameNumber;
239}
240
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800241ASurfaceTransaction* ASurfaceTransaction_create() {
242 Transaction* transaction = new Transaction;
243 return reinterpret_cast<ASurfaceTransaction*>(transaction);
244}
245
246void ASurfaceTransaction_delete(ASurfaceTransaction* aSurfaceTransaction) {
247 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
248 delete transaction;
249}
250
251void ASurfaceTransaction_apply(ASurfaceTransaction* aSurfaceTransaction) {
252 CHECK_NOT_NULL(aSurfaceTransaction);
253
254 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
255
256 transaction->apply();
257}
258
Marissa Wall1be5a102019-01-18 16:14:04 -0800259struct ASurfaceTransactionStats {
260 std::unordered_map<ASurfaceControl*, ASurfaceControlStats> aSurfaceControlStats;
261 int64_t latchTime;
262 sp<Fence> presentFence;
Vishnu Nairbeb3b482021-04-21 08:31:27 -0700263 bool transactionCompleted;
Marissa Wall1be5a102019-01-18 16:14:04 -0800264};
265
266int64_t ASurfaceTransactionStats_getLatchTime(ASurfaceTransactionStats* aSurfaceTransactionStats) {
267 CHECK_NOT_NULL(aSurfaceTransactionStats);
268 return aSurfaceTransactionStats->latchTime;
269}
270
271int ASurfaceTransactionStats_getPresentFenceFd(ASurfaceTransactionStats* aSurfaceTransactionStats) {
272 CHECK_NOT_NULL(aSurfaceTransactionStats);
Vishnu Nairbeb3b482021-04-21 08:31:27 -0700273 LOG_ALWAYS_FATAL_IF(!aSurfaceTransactionStats->transactionCompleted,
274 "ASurfaceTransactionStats queried from an incomplete transaction callback");
275
Marissa Wall1be5a102019-01-18 16:14:04 -0800276 auto& presentFence = aSurfaceTransactionStats->presentFence;
277 return (presentFence) ? presentFence->dup() : -1;
278}
279
280void ASurfaceTransactionStats_getASurfaceControls(ASurfaceTransactionStats* aSurfaceTransactionStats,
281 ASurfaceControl*** outASurfaceControls,
282 size_t* outASurfaceControlsSize) {
283 CHECK_NOT_NULL(aSurfaceTransactionStats);
284 CHECK_NOT_NULL(outASurfaceControls);
285 CHECK_NOT_NULL(outASurfaceControlsSize);
286
287 size_t size = aSurfaceTransactionStats->aSurfaceControlStats.size();
288
289 SurfaceControl** surfaceControls = new SurfaceControl*[size];
290 ASurfaceControl** aSurfaceControls = reinterpret_cast<ASurfaceControl**>(surfaceControls);
291
292 size_t i = 0;
293 for (auto& [aSurfaceControl, aSurfaceControlStats] : aSurfaceTransactionStats->aSurfaceControlStats) {
294 aSurfaceControls[i] = aSurfaceControl;
295 i++;
296 }
297
298 *outASurfaceControls = aSurfaceControls;
299 *outASurfaceControlsSize = size;
300}
301
302int64_t ASurfaceTransactionStats_getAcquireTime(ASurfaceTransactionStats* aSurfaceTransactionStats,
303 ASurfaceControl* aSurfaceControl) {
304 CHECK_NOT_NULL(aSurfaceTransactionStats);
305 CHECK_NOT_NULL(aSurfaceControl);
306
307 const auto& aSurfaceControlStats =
308 aSurfaceTransactionStats->aSurfaceControlStats.find(aSurfaceControl);
309 LOG_ALWAYS_FATAL_IF(
310 aSurfaceControlStats == aSurfaceTransactionStats->aSurfaceControlStats.end(),
311 "ASurfaceControl not found");
312
313 return aSurfaceControlStats->second.acquireTime;
314}
315
316int ASurfaceTransactionStats_getPreviousReleaseFenceFd(
317 ASurfaceTransactionStats* aSurfaceTransactionStats, ASurfaceControl* aSurfaceControl) {
318 CHECK_NOT_NULL(aSurfaceTransactionStats);
319 CHECK_NOT_NULL(aSurfaceControl);
Vishnu Nairbeb3b482021-04-21 08:31:27 -0700320 LOG_ALWAYS_FATAL_IF(!aSurfaceTransactionStats->transactionCompleted,
321 "ASurfaceTransactionStats queried from an incomplete transaction callback");
Marissa Wall1be5a102019-01-18 16:14:04 -0800322
323 const auto& aSurfaceControlStats =
324 aSurfaceTransactionStats->aSurfaceControlStats.find(aSurfaceControl);
325 LOG_ALWAYS_FATAL_IF(
326 aSurfaceControlStats == aSurfaceTransactionStats->aSurfaceControlStats.end(),
327 "ASurfaceControl not found");
328
329 auto& previousReleaseFence = aSurfaceControlStats->second.previousReleaseFence;
330 return (previousReleaseFence) ? previousReleaseFence->dup() : -1;
331}
332
333void ASurfaceTransactionStats_releaseASurfaceControls(ASurfaceControl** aSurfaceControls) {
334 CHECK_NOT_NULL(aSurfaceControls);
335
336 SurfaceControl** surfaceControls = reinterpret_cast<SurfaceControl**>(aSurfaceControls);
337 delete[] surfaceControls;
338}
339
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800340void ASurfaceTransaction_setOnComplete(ASurfaceTransaction* aSurfaceTransaction, void* context,
341 ASurfaceTransaction_OnComplete func) {
342 CHECK_NOT_NULL(aSurfaceTransaction);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800343 CHECK_NOT_NULL(func);
344
345 TransactionCompletedCallbackTakesContext callback = [func](void* callback_context,
Marissa Wall1be5a102019-01-18 16:14:04 -0800346 nsecs_t latchTime,
347 const sp<Fence>& presentFence,
348 const std::vector<SurfaceControlStats>& surfaceControlStats) {
349 ASurfaceTransactionStats aSurfaceTransactionStats;
350
351 aSurfaceTransactionStats.latchTime = latchTime;
352 aSurfaceTransactionStats.presentFence = presentFence;
Vishnu Nairbeb3b482021-04-21 08:31:27 -0700353 aSurfaceTransactionStats.transactionCompleted = true;
Marissa Wall1be5a102019-01-18 16:14:04 -0800354
355 auto& aSurfaceControlStats = aSurfaceTransactionStats.aSurfaceControlStats;
356
Valerie Haud6a222e2020-01-29 14:27:09 -0800357 for (const auto& [surfaceControl, latchTime, acquireTime, presentFence, previousReleaseFence, transformHint, frameEvents] : surfaceControlStats) {
Marissa Wall1be5a102019-01-18 16:14:04 -0800358 ASurfaceControl* aSurfaceControl = reinterpret_cast<ASurfaceControl*>(surfaceControl.get());
359 aSurfaceControlStats[aSurfaceControl].acquireTime = acquireTime;
360 aSurfaceControlStats[aSurfaceControl].previousReleaseFence = previousReleaseFence;
361 }
362
363 (*func)(callback_context, &aSurfaceTransactionStats);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800364 };
365
366 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
367
368 transaction->addTransactionCompletedCallback(callback, context);
369}
370
Marissa Wall1be5a102019-01-18 16:14:04 -0800371void ASurfaceTransaction_reparent(ASurfaceTransaction* aSurfaceTransaction,
372 ASurfaceControl* aSurfaceControl,
373 ASurfaceControl* newParentASurfaceControl) {
374 CHECK_NOT_NULL(aSurfaceTransaction);
375 CHECK_NOT_NULL(aSurfaceControl);
376
377 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
378 sp<SurfaceControl> newParentSurfaceControl = ASurfaceControl_to_SurfaceControl(
379 newParentASurfaceControl);
Marissa Wall1be5a102019-01-18 16:14:04 -0800380 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
381
Pablo Gamito117040c2020-09-14 08:24:41 +0000382 transaction->reparent(surfaceControl, newParentSurfaceControl);
Marissa Wall1be5a102019-01-18 16:14:04 -0800383}
384
385void ASurfaceTransaction_setVisibility(ASurfaceTransaction* aSurfaceTransaction,
386 ASurfaceControl* aSurfaceControl,
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800387 int8_t visibility) {
388 CHECK_NOT_NULL(aSurfaceTransaction);
389 CHECK_NOT_NULL(aSurfaceControl);
390
391 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
392 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
393
394 switch (visibility) {
395 case ASURFACE_TRANSACTION_VISIBILITY_SHOW:
396 transaction->show(surfaceControl);
397 break;
398 case ASURFACE_TRANSACTION_VISIBILITY_HIDE:
399 transaction->hide(surfaceControl);
400 break;
401 default:
402 LOG_ALWAYS_FATAL("invalid visibility %d", visibility);
403 }
404}
405
Marissa Wall1be5a102019-01-18 16:14:04 -0800406void ASurfaceTransaction_setZOrder(ASurfaceTransaction* aSurfaceTransaction,
407 ASurfaceControl* aSurfaceControl,
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800408 int32_t z_order) {
409 CHECK_NOT_NULL(aSurfaceTransaction);
410 CHECK_NOT_NULL(aSurfaceControl);
411
412 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
413 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
414
415 transaction->setLayer(surfaceControl, z_order);
416}
417
Marissa Wall1be5a102019-01-18 16:14:04 -0800418void ASurfaceTransaction_setBuffer(ASurfaceTransaction* aSurfaceTransaction,
419 ASurfaceControl* aSurfaceControl,
420 AHardwareBuffer* buffer, int acquire_fence_fd) {
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800421 CHECK_NOT_NULL(aSurfaceTransaction);
422 CHECK_NOT_NULL(aSurfaceControl);
423
424 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
425 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
426
427 sp<GraphicBuffer> graphic_buffer(reinterpret_cast<GraphicBuffer*>(buffer));
428
429 transaction->setBuffer(surfaceControl, graphic_buffer);
Marissa Wall1be5a102019-01-18 16:14:04 -0800430 if (acquire_fence_fd != -1) {
431 sp<Fence> fence = new Fence(acquire_fence_fd);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800432 transaction->setAcquireFence(surfaceControl, fence);
433 }
434}
435
436void ASurfaceTransaction_setGeometry(ASurfaceTransaction* aSurfaceTransaction,
437 ASurfaceControl* aSurfaceControl, const ARect& source,
438 const ARect& destination, int32_t transform) {
439 CHECK_NOT_NULL(aSurfaceTransaction);
440 CHECK_NOT_NULL(aSurfaceControl);
chaviw87a07ea2021-04-29 09:04:41 -0500441 CHECK_VALID_RECT(source);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800442 CHECK_VALID_RECT(destination);
443
444 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
445 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
446
chaviw87a07ea2021-04-29 09:04:41 -0500447 Rect sourceRect = static_cast<const Rect&>(source);
448 Rect destRect = static_cast<const Rect&>(destination);
449 // Adjust the source so its top and left are not negative
450 sourceRect.left = std::max(sourceRect.left, 0);
451 sourceRect.top = std::max(sourceRect.top, 0);
452
453 if (!sourceRect.isValid()) {
454 sourceRect.makeInvalid();
455 }
chaviw9b2ac242021-04-27 15:52:09 -0500456 transaction->setBufferCrop(surfaceControl, sourceRect);
Vishnu Nair0d7aff72021-05-10 15:01:20 -0700457 transaction->setDestinationFrame(surfaceControl, destRect);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800458 transaction->setTransform(surfaceControl, transform);
Vishnu Nair1ad69542019-05-23 16:27:45 +0800459 bool transformToInverseDisplay = (NATIVE_WINDOW_TRANSFORM_INVERSE_DISPLAY & transform) ==
460 NATIVE_WINDOW_TRANSFORM_INVERSE_DISPLAY;
461 transaction->setTransformToDisplayInverse(surfaceControl, transformToInverseDisplay);
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800462}
463
chaviwccf3e8b2021-03-25 15:28:44 -0500464void ASurfaceTransaction_setCrop(ASurfaceTransaction* aSurfaceTransaction,
465 ASurfaceControl* aSurfaceControl, const ARect& crop) {
Vasiliy Telezhnikov5ead3aa2021-03-13 19:55:00 -0500466 CHECK_NOT_NULL(aSurfaceTransaction);
467 CHECK_NOT_NULL(aSurfaceControl);
chaviwccf3e8b2021-03-25 15:28:44 -0500468 CHECK_VALID_RECT(crop);
Vasiliy Telezhnikov5ead3aa2021-03-13 19:55:00 -0500469
470 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
471 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
472
chaviwccf3e8b2021-03-25 15:28:44 -0500473 transaction->setCrop(surfaceControl, static_cast<const Rect&>(crop));
Vasiliy Telezhnikov5ead3aa2021-03-13 19:55:00 -0500474}
475
chaviwccf3e8b2021-03-25 15:28:44 -0500476void ASurfaceTransaction_setPosition(ASurfaceTransaction* aSurfaceTransaction,
477 ASurfaceControl* aSurfaceControl, int32_t x, int32_t y) {
478 CHECK_NOT_NULL(aSurfaceTransaction);
Vasiliy Telezhnikov5ead3aa2021-03-13 19:55:00 -0500479 CHECK_NOT_NULL(aSurfaceControl);
Vasiliy Telezhnikov5ead3aa2021-03-13 19:55:00 -0500480
481 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
482 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
483
chaviwccf3e8b2021-03-25 15:28:44 -0500484 transaction->setPosition(surfaceControl, x, y);
Vasiliy Telezhnikov5ead3aa2021-03-13 19:55:00 -0500485}
486
chaviwccf3e8b2021-03-25 15:28:44 -0500487void ASurfaceTransaction_setBufferTransform(ASurfaceTransaction* aSurfaceTransaction,
488 ASurfaceControl* aSurfaceControl, int32_t transform) {
Vasiliy Telezhnikov5ead3aa2021-03-13 19:55:00 -0500489 CHECK_NOT_NULL(aSurfaceTransaction);
490 CHECK_NOT_NULL(aSurfaceControl);
491
492 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
493 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
494
495 transaction->setTransform(surfaceControl, transform);
496 bool transformToInverseDisplay = (NATIVE_WINDOW_TRANSFORM_INVERSE_DISPLAY & transform) ==
497 NATIVE_WINDOW_TRANSFORM_INVERSE_DISPLAY;
498 transaction->setTransformToDisplayInverse(surfaceControl, transformToInverseDisplay);
499}
500
chaviwccf3e8b2021-03-25 15:28:44 -0500501void ASurfaceTransaction_setScale(ASurfaceTransaction* aSurfaceTransaction,
502 ASurfaceControl* aSurfaceControl, float xScale, float yScale) {
503 CHECK_NOT_NULL(aSurfaceTransaction);
504 CHECK_NOT_NULL(aSurfaceControl);
505 LOG_ALWAYS_FATAL_IF(xScale < 0, "negative value passed in for xScale");
506 LOG_ALWAYS_FATAL_IF(yScale < 0, "negative value passed in for yScale");
507
508 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
509 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
510
511 transaction->setMatrix(surfaceControl, xScale, 0, 0, yScale);
512}
513
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800514void ASurfaceTransaction_setBufferTransparency(ASurfaceTransaction* aSurfaceTransaction,
515 ASurfaceControl* aSurfaceControl,
516 int8_t transparency) {
517 CHECK_NOT_NULL(aSurfaceTransaction);
518 CHECK_NOT_NULL(aSurfaceControl);
519
520 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
521 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
522
523 uint32_t flags = (transparency == ASURFACE_TRANSACTION_TRANSPARENCY_OPAQUE) ?
524 layer_state_t::eLayerOpaque : 0;
525 transaction->setFlags(surfaceControl, flags, layer_state_t::eLayerOpaque);
526}
527
Marissa Wall1be5a102019-01-18 16:14:04 -0800528void ASurfaceTransaction_setDamageRegion(ASurfaceTransaction* aSurfaceTransaction,
529 ASurfaceControl* aSurfaceControl,
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800530 const ARect rects[], uint32_t count) {
531 CHECK_NOT_NULL(aSurfaceTransaction);
532 CHECK_NOT_NULL(aSurfaceControl);
533
534 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
535 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
536
537 Region region;
538 for (uint32_t i = 0; i < count; ++i) {
Marissa Wallbb9b14f2019-04-23 14:10:15 -0700539 region.orSelf(static_cast<const Rect&>(rects[i]));
540 }
541
542 // Hardware composer interprets a DamageRegion with a single Rect of {0,0,0,0} to be an
543 // undamaged region and {0,0,-1,-1} to be a fully damaged buffer. This is a confusing
544 // distinction for a public api. Instead, default both cases to be a fully damaged buffer.
545 if (count == 1 && region.getBounds().isEmpty()) {
546 transaction->setSurfaceDamageRegion(surfaceControl, Region::INVALID_REGION);
547 return;
Marissa Wallf6a73fa2018-12-10 10:41:08 -0800548 }
549
550 transaction->setSurfaceDamageRegion(surfaceControl, region);
551}
Marissa Wall1be5a102019-01-18 16:14:04 -0800552
553void ASurfaceTransaction_setDesiredPresentTime(ASurfaceTransaction* aSurfaceTransaction,
554 int64_t desiredPresentTime) {
555 CHECK_NOT_NULL(aSurfaceTransaction);
556
557 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
558
559 transaction->setDesiredPresentTime(static_cast<nsecs_t>(desiredPresentTime));
560}
561
562void ASurfaceTransaction_setBufferAlpha(ASurfaceTransaction* aSurfaceTransaction,
563 ASurfaceControl* aSurfaceControl,
564 float alpha) {
565 CHECK_NOT_NULL(aSurfaceTransaction);
566 CHECK_NOT_NULL(aSurfaceControl);
567
568 LOG_ALWAYS_FATAL_IF(alpha < 0.0 || alpha > 1.0, "invalid alpha");
569
570 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
571 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
572
573 transaction->setAlpha(surfaceControl, alpha);
574}
575
Marissa Wall7f24f792019-02-07 14:06:04 -0800576void ASurfaceTransaction_setBufferDataSpace(ASurfaceTransaction* aSurfaceTransaction,
577 ASurfaceControl* aSurfaceControl,
578 ADataSpace aDataSpace) {
579 CHECK_NOT_NULL(aSurfaceTransaction);
580 CHECK_NOT_NULL(aSurfaceControl);
581
582 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
Valerie Hau5e18c432019-08-26 12:48:49 -0700583 if (!isDataSpaceValid(surfaceControl, aDataSpace)) {
584 ALOGE("Failed to set buffer dataspace - invalid dataspace");
585 return;
586 }
Marissa Wall7f24f792019-02-07 14:06:04 -0800587 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
Marissa Wall7f24f792019-02-07 14:06:04 -0800588 transaction->setDataspace(surfaceControl, static_cast<ui::Dataspace>(aDataSpace));
589}
590
Marissa Wall1be5a102019-01-18 16:14:04 -0800591void ASurfaceTransaction_setHdrMetadata_smpte2086(ASurfaceTransaction* aSurfaceTransaction,
592 ASurfaceControl* aSurfaceControl,
593 struct AHdrMetadata_smpte2086* metadata) {
594 CHECK_NOT_NULL(aSurfaceTransaction);
595 CHECK_NOT_NULL(aSurfaceControl);
596
597 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
598 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
599
600 HdrMetadata hdrMetadata;
601
602 if (metadata) {
603 hdrMetadata.smpte2086.displayPrimaryRed.x = metadata->displayPrimaryRed.x;
604 hdrMetadata.smpte2086.displayPrimaryRed.y = metadata->displayPrimaryRed.y;
605 hdrMetadata.smpte2086.displayPrimaryGreen.x = metadata->displayPrimaryGreen.x;
606 hdrMetadata.smpte2086.displayPrimaryGreen.y = metadata->displayPrimaryGreen.y;
607 hdrMetadata.smpte2086.displayPrimaryBlue.x = metadata->displayPrimaryBlue.x;
608 hdrMetadata.smpte2086.displayPrimaryBlue.y = metadata->displayPrimaryBlue.y;
609 hdrMetadata.smpte2086.whitePoint.x = metadata->whitePoint.x;
610 hdrMetadata.smpte2086.whitePoint.y = metadata->whitePoint.y;
611 hdrMetadata.smpte2086.minLuminance = metadata->minLuminance;
612 hdrMetadata.smpte2086.maxLuminance = metadata->maxLuminance;
613
614 hdrMetadata.validTypes |= HdrMetadata::SMPTE2086;
615 } else {
616 hdrMetadata.validTypes &= ~HdrMetadata::SMPTE2086;
617 }
618
619 transaction->setHdrMetadata(surfaceControl, hdrMetadata);
620}
621
622void ASurfaceTransaction_setHdrMetadata_cta861_3(ASurfaceTransaction* aSurfaceTransaction,
623 ASurfaceControl* aSurfaceControl,
624 struct AHdrMetadata_cta861_3* metadata) {
625 CHECK_NOT_NULL(aSurfaceTransaction);
626 CHECK_NOT_NULL(aSurfaceControl);
627
628 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
629 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
630
631 HdrMetadata hdrMetadata;
632
633 if (metadata) {
634 hdrMetadata.cta8613.maxContentLightLevel = metadata->maxContentLightLevel;
635 hdrMetadata.cta8613.maxFrameAverageLightLevel = metadata->maxFrameAverageLightLevel;
636
637 hdrMetadata.validTypes |= HdrMetadata::CTA861_3;
638 } else {
639 hdrMetadata.validTypes &= ~HdrMetadata::CTA861_3;
640 }
641
642 transaction->setHdrMetadata(surfaceControl, hdrMetadata);
643}
Valerie Hau5bbfd512019-01-22 17:39:43 -0800644
645void ASurfaceTransaction_setColor(ASurfaceTransaction* aSurfaceTransaction,
646 ASurfaceControl* aSurfaceControl,
647 float r, float g, float b, float alpha,
648 ADataSpace dataspace) {
649 CHECK_NOT_NULL(aSurfaceTransaction);
650 CHECK_NOT_NULL(aSurfaceControl);
651
652 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
Valerie Hau5e18c432019-08-26 12:48:49 -0700653 if (!isDataSpaceValid(surfaceControl, dataspace)) {
654 ALOGE("Failed to set buffer dataspace - invalid dataspace");
655 return;
656 }
Valerie Hau5bbfd512019-01-22 17:39:43 -0800657 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
658
659 half3 color;
660 color.r = r;
661 color.g = g;
662 color.b = b;
663
Marin Shalamanov511f9142021-03-16 18:03:30 +0100664 transaction->setBackgroundColor(surfaceControl, color, alpha,
665 static_cast<ui::Dataspace>(dataspace));
Valerie Hau5bbfd512019-01-22 17:39:43 -0800666}
Steven Thomas6cf051e2020-01-14 11:37:21 -0800667
668void ASurfaceTransaction_setFrameRate(ASurfaceTransaction* aSurfaceTransaction,
Steven Thomasdd7bf2f2020-01-31 18:50:02 -0800669 ASurfaceControl* aSurfaceControl, float frameRate,
670 int8_t compatibility) {
Marin Shalamanov511f9142021-03-16 18:03:30 +0100671 ASurfaceTransaction_setFrameRateWithChangeStrategy(
672 aSurfaceTransaction, aSurfaceControl, frameRate, compatibility,
673 ANATIVEWINDOW_CHANGE_FRAME_RATE_ONLY_IF_SEAMLESS);
Marin Shalamanov41ffa8d2020-10-13 12:35:20 +0200674}
675
Marin Shalamanov511f9142021-03-16 18:03:30 +0100676void ASurfaceTransaction_setFrameRateWithChangeStrategy(ASurfaceTransaction* aSurfaceTransaction,
677 ASurfaceControl* aSurfaceControl,
678 float frameRate, int8_t compatibility,
679 int8_t changeFrameRateStrategy) {
Steven Thomas6cf051e2020-01-14 11:37:21 -0800680 CHECK_NOT_NULL(aSurfaceTransaction);
681 CHECK_NOT_NULL(aSurfaceControl);
Steven Thomas6cf051e2020-01-14 11:37:21 -0800682 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
Steven Thomasdd7bf2f2020-01-31 18:50:02 -0800683 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
Marin Shalamanov511f9142021-03-16 18:03:30 +0100684 transaction->setFrameRate(surfaceControl, frameRate, compatibility, changeFrameRateStrategy);
Steven Thomas6cf051e2020-01-14 11:37:21 -0800685}
Robert Carrf57c0162021-03-24 15:48:25 -0700686
687void ASurfaceTransaction_setEnableBackPressure(ASurfaceTransaction* aSurfaceTransaction,
688 ASurfaceControl* aSurfaceControl,
689 bool enableBackpressure) {
690 CHECK_NOT_NULL(aSurfaceControl);
691 CHECK_NOT_NULL(aSurfaceTransaction);
692
693 sp<SurfaceControl> surfaceControl = ASurfaceControl_to_SurfaceControl(aSurfaceControl);
694 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
695
696 const uint32_t flags = enableBackpressure ?
697 layer_state_t::eEnableBackpressure : 0;
698 transaction->setFlags(surfaceControl, flags, layer_state_t::eEnableBackpressure);
699}
Vishnu Nairbeb3b482021-04-21 08:31:27 -0700700
701void ASurfaceTransaction_setOnCommit(ASurfaceTransaction* aSurfaceTransaction, void* context,
702 ASurfaceTransaction_OnCommit func) {
703 CHECK_NOT_NULL(aSurfaceTransaction);
704 CHECK_NOT_NULL(func);
705
706 TransactionCompletedCallbackTakesContext callback =
707 [func](void* callback_context, nsecs_t latchTime, const sp<Fence>& /* presentFence */,
708 const std::vector<SurfaceControlStats>& surfaceControlStats) {
709 ASurfaceTransactionStats aSurfaceTransactionStats;
710 aSurfaceTransactionStats.latchTime = latchTime;
711 aSurfaceTransactionStats.transactionCompleted = false;
712
713 auto& aSurfaceControlStats = aSurfaceTransactionStats.aSurfaceControlStats;
714 for (const auto&
715 [surfaceControl, latchTime, acquireTime, presentFence,
716 previousReleaseFence, transformHint,
717 frameEvents] : surfaceControlStats) {
718 ASurfaceControl* aSurfaceControl =
719 reinterpret_cast<ASurfaceControl*>(surfaceControl.get());
720 aSurfaceControlStats[aSurfaceControl].acquireTime = acquireTime;
721 }
722
723 (*func)(callback_context, &aSurfaceTransactionStats);
724 };
725
726 Transaction* transaction = ASurfaceTransaction_to_Transaction(aSurfaceTransaction);
727
728 transaction->addTransactionCommittedCallback(callback, context);
729}