1 /**************************************************************************
5 * Copyright 2018 Samsung Electronics co., Ltd. All Rights Reserved.
7 * Contact: SooChan Lim <sc1.lim@samsung.com>,
8 * Boram Park <boram1288.park@samsung.com>,
9 * Changyeon Lee <cyeon.lee@samsung.com>,
10 * Sangjin Lee <lsj119@samsung.com>
12 * Permission is hereby granted, free of charge, to any person obtaining a
13 * copy of this software and associated documentation files (the
14 * "Software"), to deal in the Software without restriction, including
15 * without limitation the rights to use, copy, modify, merge, publish,
16 * distribute, sub license, and/or sell copies of the Software, and to
17 * permit persons to whom the Software is furnished to do so, subject to
18 * the following conditions:
20 * The above copyright notice and this permission notice (including the
21 * next paragraph) shall be included in all copies or substantial portions
24 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
25 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
26 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
27 * IN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR
28 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
29 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
30 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
32 **************************************************************************/
38 #include "tdm_private.h"
40 #define HWC_FUNC_ENTRY() \
41 tdm_private_display *private_display; \
42 tdm_private_output *private_output; \
43 tdm_private_hwc *private_hwc; \
44 tdm_error ret = TDM_ERROR_NONE; /* default TDM_ERROR_NONE */\
45 TDM_RETURN_VAL_IF_FAIL(hwc != NULL, TDM_ERROR_INVALID_PARAMETER); \
46 private_hwc = (tdm_private_hwc*)hwc; \
47 private_output = private_hwc->private_output; \
48 TDM_RETURN_VAL_IF_FAIL(private_output != NULL, TDM_ERROR_INVALID_PARAMETER); \
49 private_display = private_output->private_display
51 #define HWC_FUNC_ENTRY_ERROR() \
52 tdm_private_display *private_display; \
53 tdm_private_output *private_output; \
54 tdm_private_hwc *private_hwc; \
55 tdm_error ret = TDM_ERROR_NONE; /* default TDM_ERROR_NONE */\
56 TDM_RETURN_VAL_IF_FAIL_WITH_ERROR(hwc != NULL, TDM_ERROR_INVALID_PARAMETER, NULL); \
57 private_hwc = (tdm_private_hwc*)hwc; \
58 private_output = private_hwc->private_output; \
59 TDM_RETURN_VAL_IF_FAIL_WITH_ERROR(private_output != NULL, TDM_ERROR_INVALID_PARAMETER, NULL); \
60 private_display = private_output->private_display
62 #define HWC_FUNC_ENTRY_VOID_RETURN() \
63 tdm_private_display *private_display; \
64 tdm_private_output *private_output; \
65 tdm_private_hwc *private_hwc; \
66 tdm_error ret = TDM_ERROR_NONE; /* default TDM_ERROR_NONE */\
67 TDM_RETURN_IF_FAIL(hwc != NULL); \
68 private_hwc = (tdm_private_hwc*)hwc; \
69 private_output = private_hwc->private_output; \
70 TDM_RETURN_IF_FAIL(private_output != NULL); \
71 private_display = private_output->private_display
73 static int hwc_use_vblank;
74 static unsigned int hwc_vblank_fps;
76 static tdm_private_hwc_window *
77 _tdm_hwc_find_private_hwc_window(tdm_private_hwc *private_hwc, tdm_hwc_window *hwc_window_backend)
79 tdm_private_hwc_window *private_hwc_window = NULL;
81 LIST_FOR_EACH_ENTRY(private_hwc_window, &private_hwc->hwc_window_list, link) {
82 if (private_hwc_window->hwc_window_backend == hwc_window_backend)
83 return private_hwc_window;
90 _tdm_hwc_check_hwc_commit_handler_validation(tdm_private_hwc *private_hwc, tdm_private_hwc_commit_handler *hwc_commit_handler)
92 tdm_private_hwc_commit_handler *commit_handler = NULL;
94 if (LIST_IS_EMPTY(&private_hwc->hwc_commit_handler_list))
95 return TDM_ERROR_INVALID_PARAMETER;
97 LIST_FOR_EACH_ENTRY(commit_handler, &private_hwc->hwc_commit_handler_list, link) {
98 if (commit_handler == hwc_commit_handler)
99 return TDM_ERROR_NONE;
102 return TDM_ERROR_INVALID_PARAMETER;
106 _tdm_hwc_thread_cb_commit(tdm_private_display *private_display, void *object,
107 tdm_thread_cb_base *cb_base, void *user_data)
109 tdm_thread_cb_hwc_commit *hwc_commit = (tdm_thread_cb_hwc_commit *)cb_base;
110 tdm_private_hwc_commit_handler *hwc_commit_handler = hwc_commit->base.data;
111 tdm_private_hwc *private_hwc = object;
113 TDM_RETURN_IF_FAIL(TDM_MUTEX_IS_LOCKED());
115 if (!hwc_commit_handler)
118 assert(hwc_commit_handler->owner_tid == syscall(SYS_gettid));
120 tdm_thread_cb_remove(private_hwc, TDM_THREAD_CB_HWC_COMMIT, hwc_commit_handler,
121 _tdm_hwc_thread_cb_commit, NULL);
123 TDM_RETURN_IF_FAIL(_tdm_hwc_check_hwc_commit_handler_validation(private_hwc, hwc_commit_handler) == TDM_ERROR_NONE)
125 LIST_DEL(&hwc_commit_handler->link);
127 if (tdm_debug_module & TDM_DEBUG_COMMIT) {
128 TDM_INFO("----------------------------------------- hwc(%d) committed", private_hwc->index);
129 TDM_INFO("handler(%p)", hwc_commit_handler);
132 /* LCOV_EXCL_START */
133 if (private_display->print_fps) {
134 double curr = tdm_helper_get_time();
135 if (private_hwc->fps_stamp == 0) {
136 private_hwc->fps_stamp = curr;
137 } else if ((curr - private_hwc->fps_stamp) > 1.0) {
138 TDM_INFO("hwc(%p,%d) fps: %d",
139 private_hwc, private_hwc->index, private_hwc->fps_count);
140 private_hwc->fps_count = 0;
141 private_hwc->fps_stamp = curr;
143 private_hwc->fps_count++;
144 } else if (private_hwc->fps_stamp != 0) {
145 private_hwc->fps_stamp = 0;
146 private_hwc->fps_count = 0;
150 if (hwc_commit_handler->func) {
151 _pthread_mutex_unlock(&private_display->lock);
152 hwc_commit_handler->func(private_hwc,
153 hwc_commit->sequence,
156 hwc_commit_handler->user_data);
157 _pthread_mutex_lock(&private_display->lock);
160 free(hwc_commit_handler);
162 if (tdm_debug_module & TDM_DEBUG_COMMIT)
163 TDM_INFO("-----------------------------------------...");
167 _tdm_hwc_cb_commit(tdm_hwc *hwc_backend, unsigned int sequence,
168 unsigned int tv_sec, unsigned int tv_usec, void *user_data)
170 tdm_private_hwc_commit_handler *hwc_commit_handler = user_data;
171 tdm_private_hwc *private_hwc;
172 tdm_thread_cb_hwc_commit hwc_commit;
175 if (hwc_commit_handler && hwc_commit_handler->use_vblank)
178 if (hwc_commit_handler)
179 private_hwc = hwc_commit_handler->private_hwc;
181 private_hwc = tdm_display_find_private_hwc(tdm_display_get(), hwc_backend);
183 memset(&hwc_commit, 0, sizeof hwc_commit);
184 hwc_commit.base.type = TDM_THREAD_CB_HWC_COMMIT;
185 hwc_commit.base.length = sizeof hwc_commit;
186 hwc_commit.base.object_stamp = private_hwc->stamp;
187 hwc_commit.base.data = hwc_commit_handler;
188 hwc_commit.base.sync = 0;
189 hwc_commit.sequence = sequence;
190 hwc_commit.tv_sec = tv_sec;
191 hwc_commit.tv_usec = tv_usec;
193 ret = tdm_thread_cb_call(private_hwc, &hwc_commit.base, 1);
194 TDM_WARNING_IF_FAIL(ret == TDM_ERROR_NONE);
198 _tdm_hwc_got_wait_vblank(unsigned int sequence,
199 unsigned int tv_sec, unsigned int tv_usec, void *user_data)
201 tdm_private_hwc_commit_handler *hwc_commit_handler = user_data;
202 tdm_private_hwc *private_hwc;
203 tdm_thread_cb_hwc_commit hwc_commit;
205 private_hwc = hwc_commit_handler->private_hwc;
206 private_hwc->private_output->layer_waiting_vblank = 0;
208 memset(&hwc_commit, 0, sizeof hwc_commit);
209 hwc_commit.base.type = TDM_THREAD_CB_HWC_COMMIT;
210 hwc_commit.base.length = sizeof hwc_commit;
211 hwc_commit.base.object_stamp = private_hwc->stamp;
212 hwc_commit.base.data = hwc_commit_handler;
213 hwc_commit.base.sync = 0;
214 hwc_commit.sequence = sequence;
215 hwc_commit.tv_sec = tv_sec;
216 hwc_commit.tv_usec = tv_usec;
218 _tdm_hwc_thread_cb_commit(private_hwc->private_output->private_display, private_hwc, &hwc_commit.base, user_data);
222 _tdm_hwc_cb_wait_vblank(tdm_vblank *vblank, tdm_error error, unsigned int sequence,
223 unsigned int tv_sec, unsigned int tv_usec, void *user_data)
225 tdm_private_hwc_commit_handler *hwc_commit_handler = user_data;
226 tdm_private_output *private_output = NULL;
227 tdm_private_display *private_display;
229 if (!hwc_commit_handler->use_vblank)
232 TDM_RETURN_IF_FAIL(hwc_commit_handler != NULL);
233 TDM_RETURN_IF_FAIL(hwc_commit_handler->private_hwc != NULL);
235 private_output = hwc_commit_handler->private_hwc->private_output;
236 TDM_RETURN_IF_FAIL(private_output != NULL);
238 private_display = private_output->private_display;
240 _pthread_mutex_lock(&private_display->lock);
242 _tdm_hwc_got_wait_vblank(sequence, tv_sec, tv_usec, user_data);
244 _pthread_mutex_unlock(&private_display->lock);
248 _tdm_hwc_vblank(tdm_private_hwc *private_hwc, tdm_private_hwc_commit_handler *hwc_commit_handler)
250 tdm_private_display *private_display;
251 tdm_private_output *private_output;
252 tdm_error ret = TDM_ERROR_NONE;
254 private_output = private_hwc->private_output;
255 TDM_RETURN_VAL_IF_FAIL(private_output != NULL, TDM_ERROR_INVALID_PARAMETER);
256 private_display = private_output->private_display;
258 if (!private_output->vblank) {
259 /* tdm_vblank APIs is for server. it should be called in unlock status*/
260 _pthread_mutex_unlock(&private_display->lock);
261 private_output->vblank = tdm_vblank_create(private_display, private_output, NULL);
262 _pthread_mutex_lock(&private_display->lock);
263 TDM_RETURN_VAL_IF_FAIL(private_output->vblank != NULL, TDM_ERROR_OPERATION_FAILED);
266 if (!private_output->layer_waiting_vblank) {
267 ret = tdm_vblank_set_fps(private_output->vblank, hwc_vblank_fps);
268 if (ret != TDM_ERROR_NONE)
271 private_output->layer_waiting_vblank = 1;
273 /* tdm_vblank APIs is for server. it should be called in unlock status*/
274 _pthread_mutex_unlock(&private_display->lock);
275 ret = tdm_vblank_wait(private_output->vblank, 0, 0, 1, _tdm_hwc_cb_wait_vblank, hwc_commit_handler);
276 _pthread_mutex_lock(&private_display->lock);
277 if (ret != TDM_ERROR_NONE) {
278 if (!TDM_OUTPUT_DPMS_VSYNC_IS_OFF(private_output->current_dpms_value)) {
279 private_output->layer_waiting_vblank = 0;
289 tdm_hwc_init(tdm_private_display *private_display)
291 tdm_thread_cb_set_find_func(TDM_THREAD_CB_HWC_COMMIT, tdm_display_find_hwc_stamp);
293 return TDM_ERROR_NONE;
297 tdm_hwc_set_vblank(unsigned int fps)
300 hwc_vblank_fps = fps;
304 tdm_hwc_unset_vblank(void)
310 EXTERN tdm_hwc_window *
311 tdm_hwc_create_window(tdm_hwc *hwc, tdm_error *error)
313 tdm_hwc_window *hwc_window = NULL;
315 HWC_FUNC_ENTRY_ERROR();
317 _pthread_mutex_lock(&private_display->lock);
319 hwc_window = (tdm_hwc_window *)tdm_hwc_window_create_internal(private_hwc, error);
321 _pthread_mutex_unlock(&private_display->lock);
327 tdm_hwc_get_video_supported_formats(tdm_hwc *hwc, const tbm_format **formats, int *count)
329 tdm_private_module *private_module;
330 tdm_func_hwc *func_hwc;
334 TDM_RETURN_VAL_IF_FAIL(formats != NULL, TDM_ERROR_INVALID_PARAMETER);
335 TDM_RETURN_VAL_IF_FAIL(count != NULL, TDM_ERROR_INVALID_PARAMETER);
337 _pthread_mutex_lock(&private_display->lock);
339 private_module = private_output->private_module;
340 func_hwc = &private_module->func_hwc;
342 if (!func_hwc->hwc_get_video_supported_formats) {
343 /* LCOV_EXCL_START */
344 _pthread_mutex_unlock(&private_display->lock);
345 TDM_WRN("not implemented!!");
346 return TDM_ERROR_NOT_IMPLEMENTED;
350 ret = func_hwc->hwc_get_video_supported_formats(private_hwc->hwc_backend, formats, count);
352 _pthread_mutex_unlock(&private_display->lock);
358 tdm_hwc_get_video_available_properties(tdm_hwc *hwc, const tdm_prop **props, int *count)
360 tdm_private_module *private_module;
361 tdm_func_hwc *func_hwc = NULL;
365 TDM_RETURN_VAL_IF_FAIL(props != NULL, TDM_ERROR_INVALID_PARAMETER);
366 TDM_RETURN_VAL_IF_FAIL(count != NULL, TDM_ERROR_INVALID_PARAMETER);
368 _pthread_mutex_lock(&private_display->lock);
370 private_module = private_output->private_module;
371 func_hwc = &private_module->func_hwc;
373 if (!func_hwc->hwc_get_video_available_properties) {
374 /* LCOV_EXCL_START */
375 _pthread_mutex_unlock(&private_display->lock);
376 TDM_WRN("not implemented!!");
377 return TDM_ERROR_NOT_IMPLEMENTED;
381 ret = func_hwc->hwc_get_video_available_properties(private_hwc->hwc_backend, props, count);
383 _pthread_mutex_unlock(&private_display->lock);
389 tdm_hwc_get_capabilities(tdm_hwc *hwc, tdm_hwc_capability *capabilities)
391 tdm_private_module *private_module;
392 tdm_func_hwc *func_hwc;
396 _pthread_mutex_lock(&private_display->lock);
398 private_module = private_output->private_module;
399 func_hwc = &private_module->func_hwc;
401 if (!func_hwc->hwc_get_capabilities) {
402 _pthread_mutex_unlock(&private_display->lock);
403 TDM_WRN("not implemented!!");
404 return TDM_ERROR_NOT_IMPLEMENTED;
407 ret = func_hwc->hwc_get_capabilities(private_hwc->hwc_backend, capabilities);
409 _pthread_mutex_unlock(&private_display->lock);
415 tdm_hwc_get_available_properties(tdm_hwc *hwc, const tdm_prop **props, int *count)
417 tdm_private_module *private_module;
418 tdm_func_hwc *func_hwc = NULL;
422 TDM_RETURN_VAL_IF_FAIL(props != NULL, TDM_ERROR_INVALID_PARAMETER);
423 TDM_RETURN_VAL_IF_FAIL(count != NULL, TDM_ERROR_INVALID_PARAMETER);
425 _pthread_mutex_lock(&private_display->lock);
427 private_module = private_output->private_module;
428 func_hwc = &private_module->func_hwc;
430 if (!func_hwc->hwc_get_available_properties) {
431 /* LCOV_EXCL_START */
432 _pthread_mutex_unlock(&private_display->lock);
433 TDM_WRN("not implemented!!");
434 return TDM_ERROR_NOT_IMPLEMENTED;
438 ret = func_hwc->hwc_get_available_properties(private_hwc->hwc_backend, props, count);
440 _pthread_mutex_unlock(&private_display->lock);
445 EXTERN tbm_surface_queue_h
446 tdm_hwc_get_client_target_buffer_queue(tdm_hwc *hwc, tdm_error *error)
448 tdm_private_module *private_module;
449 tdm_func_hwc *func_hwc = NULL;
450 tbm_surface_queue_h queue = NULL;
452 HWC_FUNC_ENTRY_ERROR();
454 _pthread_mutex_lock(&private_display->lock);
456 private_module = private_hwc->private_module;
457 func_hwc = &private_module->func_hwc;
459 if (!func_hwc->hwc_get_client_target_buffer_queue) {
460 /* LCOV_EXCL_START */
461 _pthread_mutex_unlock(&private_display->lock);
462 TDM_WRN("not implemented!!");
467 queue = func_hwc->hwc_get_client_target_buffer_queue(private_hwc->hwc_backend, error);
469 _pthread_mutex_unlock(&private_display->lock);
475 tdm_hwc_set_client_target_buffer(tdm_hwc *hwc, tbm_surface_h target_buffer, tdm_region damage)
477 tdm_private_module *private_module;
478 tdm_func_hwc *func_hwc = NULL;
482 _pthread_mutex_lock(&private_display->lock);
484 if (tdm_debug_dump & TDM_DUMP_FLAG_WINDOW) {
485 /* LCOV_EXCL_START */
486 char str[TDM_PATH_LEN];
488 snprintf(str, TDM_PATH_LEN, "target_window_%03d", i++);
489 tdm_helper_dump_buffer_str(target_buffer, tdm_debug_dump_dir, str);
493 private_module = private_hwc->private_module;
494 func_hwc = &private_module->func_hwc;
496 if (!func_hwc->hwc_set_client_target_buffer) {
497 /* LCOV_EXCL_START */
498 _pthread_mutex_unlock(&private_display->lock);
499 TDM_WRN("not implemented!!");
500 return TDM_ERROR_NOT_IMPLEMENTED;
504 ret = func_hwc->hwc_set_client_target_buffer(private_hwc->hwc_backend, target_buffer, damage);
506 if (private_hwc->display_target_buffer) {
507 if (private_hwc->display_target_buffer != target_buffer) {
508 tbm_surface_internal_unref(private_hwc->display_target_buffer);
509 private_hwc->display_target_buffer = target_buffer;
511 tbm_surface_internal_ref(private_hwc->display_target_buffer);
515 private_hwc->display_target_buffer = target_buffer;
516 tbm_surface_internal_ref(private_hwc->display_target_buffer);
520 _pthread_mutex_unlock(&private_display->lock);
526 tdm_hwc_set_client_target_acquire_fence(tdm_hwc *hwc, int acquire_fence)
528 tdm_private_module *private_module;
529 tdm_func_hwc *func_hwc = NULL;
533 _pthread_mutex_lock(&private_display->lock);
535 private_module = private_hwc->private_module;
536 func_hwc = &private_module->func_hwc;
538 if (!func_hwc->hwc_set_client_target_acquire_fence) {
539 /* LCOV_EXCL_START */
540 _pthread_mutex_unlock(&private_display->lock);
541 TDM_WRN("not implemented!!");
542 return TDM_ERROR_NOT_IMPLEMENTED;
546 ret = func_hwc->hwc_set_client_target_acquire_fence(private_hwc->hwc_backend, acquire_fence);
548 _pthread_mutex_unlock(&private_display->lock);
554 tdm_hwc_validate(tdm_hwc *hwc, tdm_hwc_window **composited_wnds, uint32_t num_wnds, uint32_t *num_types)
556 tdm_private_module *private_module;
557 tdm_func_hwc *func_hwc = NULL;
558 tdm_private_hwc_window **composited_wnds_frontend = NULL;
559 tdm_hwc_window **composited_wnds_backend = NULL;
564 TDM_RETURN_VAL_IF_FAIL(num_types != NULL, TDM_ERROR_INVALID_PARAMETER);
566 _pthread_mutex_lock(&private_display->lock);
568 private_module = private_hwc->private_module;
569 func_hwc = &private_module->func_hwc;
571 if (!func_hwc->hwc_validate) {
572 /* LCOV_EXCL_START */
573 _pthread_mutex_unlock(&private_display->lock);
574 TDM_WRN("not implemented!!");
575 return TDM_ERROR_NOT_IMPLEMENTED;
580 ret = func_hwc->hwc_validate(private_hwc->hwc_backend, NULL, 0, num_types);
582 _pthread_mutex_unlock(&private_display->lock);
586 composited_wnds_backend = calloc(num_wnds, sizeof(tdm_hwc_window *));
587 if (!composited_wnds_backend) {
588 /* LCOV_EXCL_START */
589 _pthread_mutex_unlock(&private_display->lock);
590 return TDM_ERROR_OUT_OF_MEMORY;
594 composited_wnds_frontend = (tdm_private_hwc_window **)composited_wnds;
596 for (i = 0; i < num_wnds; i++)
597 composited_wnds_backend[i] = composited_wnds_frontend[i]->hwc_window_backend;
599 ret = func_hwc->hwc_validate(private_hwc->hwc_backend, composited_wnds_backend,
600 num_wnds, num_types);
602 free(composited_wnds_backend);
604 _pthread_mutex_unlock(&private_display->lock);
610 tdm_hwc_get_changed_composition_types(tdm_hwc *hwc, uint32_t *num_elements,
611 tdm_hwc_window **hwc_window,
612 tdm_hwc_window_composition *composition_types)
614 tdm_private_module *private_module;
615 tdm_func_hwc *func_hwc = NULL;
616 tdm_private_hwc_window * private_hwc_window = NULL;
621 TDM_RETURN_VAL_IF_FAIL(num_elements != NULL, TDM_ERROR_INVALID_PARAMETER);
623 _pthread_mutex_lock(&private_display->lock);
625 private_module = private_hwc->private_module;
626 func_hwc = &private_module->func_hwc;
628 if (!func_hwc->hwc_get_changed_composition_types) {
629 /* LCOV_EXCL_START */
630 _pthread_mutex_unlock(&private_display->lock);
631 TDM_WRN("not implemented!!");
632 return TDM_ERROR_NOT_IMPLEMENTED;
636 ret = func_hwc->hwc_get_changed_composition_types(private_hwc->hwc_backend,
637 num_elements, hwc_window, composition_types);
638 if (ret != TDM_ERROR_NONE) {
639 /* LCOV_EXCL_START */
640 _pthread_mutex_unlock(&private_display->lock);
645 if (hwc_window == NULL || composition_types == NULL) {
646 _pthread_mutex_unlock(&private_display->lock);
647 return TDM_ERROR_NONE;
650 for (i = 0; i < *num_elements; i++) {
651 private_hwc_window = _tdm_hwc_find_private_hwc_window(private_hwc, hwc_window[i]);
652 if (private_hwc_window == NULL) {
653 /* LCOV_EXCL_START */
654 TDM_ERR("failed! This should never happen!");
655 tdm_hwc_window_destroy_internal(private_hwc_window);
657 _pthread_mutex_unlock(&private_display->lock);
658 return TDM_ERROR_OPERATION_FAILED;
662 hwc_window[i] = (tdm_hwc_window*)private_hwc_window;
665 _pthread_mutex_unlock(&private_display->lock);
671 tdm_hwc_accept_validation(tdm_hwc *hwc)
673 tdm_private_module *private_module;
674 tdm_func_hwc *func_hwc = NULL;
678 _pthread_mutex_lock(&private_display->lock);
680 private_module = private_hwc->private_module;
681 func_hwc = &private_module->func_hwc;
683 if (!func_hwc->hwc_validate) {
684 /* LCOV_EXCL_START */
685 _pthread_mutex_unlock(&private_display->lock);
686 TDM_WRN("not implemented!!");
687 return TDM_ERROR_NOT_IMPLEMENTED;
691 ret = func_hwc->hwc_accept_validation(private_hwc->hwc_backend);
693 _pthread_mutex_unlock(&private_display->lock);
699 tdm_hwc_commit(tdm_hwc *hwc, int sync, tdm_hwc_commit_handler func, void *user_data)
701 tdm_private_module *private_module;
702 tdm_func_hwc *func_hwc = NULL;
703 tdm_private_hwc_commit_handler *hwc_commit_handler = NULL;
704 tdm_private_voutput *private_voutput = NULL;
705 tdm_private_voutput_commit_handler *voutput_commit_handler = NULL;
709 _pthread_mutex_lock(&private_display->lock);
711 private_module = private_hwc->private_module;
712 func_hwc = &private_module->func_hwc;
714 if (!func_hwc->hwc_commit) {
715 /* LCOV_EXCL_START */
716 TDM_WRN("not implemented!!");
717 _pthread_mutex_unlock(&private_display->lock);
718 return TDM_ERROR_NOT_IMPLEMENTED;
722 //TODO: I am not sure yet whether we have to check the dpms at hwc_commit.
724 /* check if the dpms is off */
725 if (TDM_OUTPUT_DPMS_VSYNC_IS_OFF(private_output->current_dpms_value)) {
726 TDM_ERR("hwc(%d) dpms: %s", private_hwc->index,
727 tdm_dpms_str(private_output->current_dpms_value));
728 _pthread_mutex_unlock(&private_display->lock);
729 return TDM_ERROR_DPMS_OFF;
733 if (tdm_debug_module & TDM_DEBUG_COMMIT)
734 TDM_INFO("hwc(%d) commit", private_hwc->index);
736 if (private_module == private_display->virtual_module) {
737 if (!private_output->private_voutput) {
738 TDM_ERR("virtual module but don't have voutput");
739 _pthread_mutex_unlock(&private_display->lock);
740 return TDM_ERROR_BAD_MODULE;
744 if (!private_hwc->regist_commit_cb) {
745 private_hwc->regist_commit_cb = 1;
746 ret = func_hwc->hwc_set_commit_handler(private_hwc->hwc_backend, _tdm_hwc_cb_commit);
747 /* LCOV_EXCL_START */
748 if (ret != TDM_ERROR_NONE) {
749 private_hwc->regist_commit_cb = 0;
750 TDM_ERR("hwc(%d) fail to set hwc_set_commit_handler", private_hwc->index);
751 _pthread_mutex_unlock(&private_display->lock);
757 hwc_commit_handler = calloc(1, sizeof(tdm_private_hwc_commit_handler));
758 if (!hwc_commit_handler) {
759 /* LCOV_EXCL_START */
760 TDM_ERR("failed: alloc memory");
761 _pthread_mutex_unlock(&private_display->lock);
762 return TDM_ERROR_OUT_OF_MEMORY;
766 ret = tdm_thread_cb_add(private_hwc, TDM_THREAD_CB_HWC_COMMIT, hwc_commit_handler,
767 _tdm_hwc_thread_cb_commit, NULL);
768 if (ret != TDM_ERROR_NONE) {
769 TDM_ERR("tdm_thread_cb_add failed");
770 free(hwc_commit_handler);
771 _pthread_mutex_unlock(&private_display->lock);
775 LIST_ADDTAIL(&hwc_commit_handler->link, &private_hwc->hwc_commit_handler_list);
776 hwc_commit_handler->private_hwc = private_hwc;
777 hwc_commit_handler->func = func;
778 hwc_commit_handler->user_data = user_data;
779 hwc_commit_handler->owner_tid = syscall(SYS_gettid);
781 hwc_commit_handler->use_vblank = 1;
783 if (private_module == private_display->virtual_module) {
784 private_voutput = private_output->private_voutput;
786 if (LIST_LENGTH(&private_voutput->voutput_commit_handler_list) != 0) {
787 voutput_commit_handler = LIST_FIRST_ENTRY(&private_voutput->voutput_commit_handler_list, tdm_private_voutput_commit_handler, link);
788 voutput_commit_handler->user_data = private_hwc->display_target_buffer;
792 ret = func_hwc->hwc_commit(private_hwc->hwc_backend, sync, hwc_commit_handler);
793 TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, commit_failed);
795 if (hwc_use_vblank) {
796 ret = _tdm_hwc_vblank(private_hwc, hwc_commit_handler);
797 if (ret == TDM_ERROR_NONE) {
798 if (tdm_debug_module & TDM_DEBUG_COMMIT)
799 TDM_INFO("hwc(%d) backend commit: wait vblank handle(%p) func(%p) user_data(%p)",
800 private_hwc->index, hwc_commit_handler, func, user_data);
804 if (tdm_debug_module & TDM_DEBUG_COMMIT)
805 TDM_INFO("hwc(%d) backend commit: handle(%p) func(%p) user_data(%p)",
806 private_hwc->index, hwc_commit_handler, func, user_data);
809 _pthread_mutex_unlock(&private_display->lock);
814 /* LCOV_EXCL_START */
815 if (hwc_commit_handler) {
816 tdm_thread_cb_remove(private_hwc, TDM_THREAD_CB_HWC_COMMIT, hwc_commit_handler,
817 _tdm_hwc_thread_cb_commit, NULL);
818 LIST_DEL(&hwc_commit_handler->link);
819 free(hwc_commit_handler);
822 _pthread_mutex_unlock(&private_display->lock);
829 tdm_hwc_get_commit_fence(tdm_hwc *hwc, int *commit_fence)
831 tdm_private_module *private_module;
832 tdm_func_hwc *func_hwc = NULL;
836 _pthread_mutex_lock(&private_display->lock);
838 private_module = private_hwc->private_module;
839 func_hwc = &private_module->func_hwc;
841 if (!func_hwc->hwc_get_commit_fence) {
842 /* LCOV_EXCL_START */
843 _pthread_mutex_unlock(&private_display->lock);
844 TDM_WRN("not implemented!!");
845 return TDM_ERROR_NOT_IMPLEMENTED;
849 ret = func_hwc->hwc_get_commit_fence(private_hwc->hwc_backend, commit_fence);
851 _pthread_mutex_unlock(&private_display->lock);
857 tdm_hwc_get_release_fences(tdm_hwc *hwc, uint32_t *num_elements,
858 tdm_hwc_window **hwc_windows, int *fences)
860 tdm_private_module *private_module;
861 tdm_func_hwc *func_hwc = NULL;
862 tdm_private_hwc_window *private_hwc_window = NULL;
867 _pthread_mutex_lock(&private_display->lock);
869 private_module = private_hwc->private_module;
870 func_hwc = &private_module->func_hwc;
872 if (!func_hwc->hwc_get_release_fences) {
873 /* LCOV_EXCL_START */
874 _pthread_mutex_unlock(&private_display->lock);
875 TDM_WRN("not implemented!!");
876 return TDM_ERROR_NOT_IMPLEMENTED;
880 ret = func_hwc->hwc_get_release_fences(private_hwc->hwc_backend, num_elements,
881 hwc_windows, fences);
883 if (hwc_windows == NULL || fences == NULL) {
884 _pthread_mutex_unlock(&private_display->lock);
885 return TDM_ERROR_NONE;
888 for (i = 0; i < *num_elements; i++) {
889 private_hwc_window = _tdm_hwc_find_private_hwc_window(private_hwc, hwc_windows[i]);
890 if (private_hwc_window == NULL) {
891 /* LCOV_EXCL_START */
892 TDM_ERR("failed! This should never happen!");
893 tdm_hwc_window_destroy_internal(private_hwc_window);
895 _pthread_mutex_unlock(&private_display->lock);
896 return TDM_ERROR_OPERATION_FAILED;
900 hwc_windows[i] = (tdm_hwc_window*)private_hwc_window;
903 _pthread_mutex_unlock(&private_display->lock);
909 tdm_hwc_set_property(tdm_hwc *hwc, uint32_t id, tdm_value value)
911 tdm_private_module *private_module;
912 tdm_func_hwc *func_hwc = NULL;
916 _pthread_mutex_lock(&private_display->lock);
918 private_module = private_hwc->private_module;
919 func_hwc = &private_module->func_hwc;
921 if (!func_hwc->hwc_set_property) {
922 /* LCOV_EXCL_START */
923 _pthread_mutex_unlock(&private_display->lock);
924 TDM_WRN("not implemented!!");
925 return TDM_ERROR_NOT_IMPLEMENTED;
929 ret = func_hwc->hwc_set_property(private_hwc->hwc_backend, id, value);
931 _pthread_mutex_unlock(&private_display->lock);
937 tdm_hwc_get_property(tdm_hwc *hwc, uint32_t id, tdm_value *value)
939 tdm_private_module *private_module;
940 tdm_func_hwc *func_hwc = NULL;
944 _pthread_mutex_lock(&private_display->lock);
946 private_module = private_hwc->private_module;
947 func_hwc = &private_module->func_hwc;
949 if (!func_hwc->hwc_get_property) {
950 /* LCOV_EXCL_START */
951 _pthread_mutex_unlock(&private_display->lock);
952 TDM_WRN("not implemented!!");
953 return TDM_ERROR_NOT_IMPLEMENTED;
957 ret = func_hwc->hwc_get_property(private_hwc->hwc_backend, id, value);
959 _pthread_mutex_unlock(&private_display->lock);