1 /**************************************************************************
5 Copyright 2015 Samsung Electronics co., Ltd. All Rights Reserved.
7 Contact: Eunchul Kim <chulspro.kim@samsung.com>,
8 JinYoung Jeon <jy0.jeon@samsung.com>,
9 Taeheon Kim <th908.kim@samsung.com>,
10 YoungJun Cho <yj44.cho@samsung.com>,
11 SooChan Lim <sc1.lim@samsung.com>,
12 Boram Park <sc1.lim@samsung.com>
14 Permission is hereby granted, free of charge, to any person obtaining a
15 copy of this software and associated documentation files (the
16 "Software"), to deal in the Software without restriction, including
17 without limitation the rights to use, copy, modify, merge, publish,
18 distribute, sub license, and/or sell copies of the Software, and to
19 permit persons to whom the Software is furnished to do so, subject to
20 the following conditions:
22 The above copyright notice and this permission notice (including the
23 next paragraph) shall be included in all copies or substantial portions
26 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
27 OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
28 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
29 IN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR
30 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
31 TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
32 SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
34 **************************************************************************/
36 #ifndef _TDM_PRIVATE_H_
37 #define _TDM_PRIVATE_H_
46 #include <sys/types.h>
52 #include <sys/syscall.h>
53 #include <sys/types.h>
55 #include <tbm_bufmgr.h>
56 #include <tbm_surface_queue.h>
58 #include "tdm_backend.h"
61 #include "tdm_macro.h"
71 * @brief The private header file for a frontend library
74 extern int tdm_debug_buffer;
75 extern int tdm_debug_mutex;
76 extern int tdm_debug_thread;
80 #define TDM_TRACE_BEGIN(NAME) traceBegin(TTRACE_TAG_GRAPHICS, "TDM:"#NAME)
81 #define TDM_TRACE_END() traceEnd(TTRACE_TAG_GRAPHICS)
83 #define TDM_TRACE_BEGIN(NAME)
84 #define TDM_TRACE_END()
87 #define prototype_name_fn(res) const char * res##_str(int type)
89 prototype_name_fn(dpms);
90 prototype_name_fn(status);
93 TDM_CAPTURE_TARGET_OUTPUT,
94 TDM_CAPTURE_TARGET_LAYER,
97 typedef struct _tdm_private_display tdm_private_display;
98 typedef struct _tdm_private_output tdm_private_output;
99 typedef struct _tdm_private_layer tdm_private_layer;
100 typedef struct _tdm_private_pp tdm_private_pp;
101 typedef struct _tdm_private_capture tdm_private_capture;
102 typedef struct _tdm_private_loop tdm_private_loop;
103 typedef struct _tdm_private_server tdm_private_server;
104 typedef struct _tdm_private_thread tdm_private_thread;
105 typedef struct _tdm_private_vblank_handler tdm_private_vblank_handler;
106 typedef struct _tdm_private_commit_handler tdm_private_commit_handler;
107 typedef struct _tdm_private_change_handler tdm_private_change_handler;
109 struct _tdm_private_display {
110 pthread_mutex_t lock;
111 unsigned int init_count;
113 /* backend module info */
115 tdm_backend_module *module_data;
116 tdm_backend_data *bdata;
122 /* backend function */
123 tdm_display_capability capabilities;
124 tdm_func_display func_display;
125 tdm_func_output func_output;
126 tdm_func_layer func_layer;
128 tdm_func_capture func_capture;
130 /* backend capability */
131 tdm_caps_display caps_display;
133 tdm_caps_capture caps_capture;
135 /* output, pp list */
136 struct list_head output_list;
137 struct list_head pp_list;
138 struct list_head capture_list;
142 /* for event handling */
143 tdm_private_loop *private_loop;
146 tdm_output **outputs;
149 struct _tdm_private_output {
150 struct list_head link;
154 tdm_private_display *private_display;
156 tdm_caps_output caps;
157 tdm_output *output_backend;
160 tdm_output_dpms current_dpms_value;
162 int regist_vblank_cb;
163 int regist_commit_cb;
164 int regist_change_cb;
166 struct list_head layer_list;
167 struct list_head capture_list;
168 struct list_head vblank_handler_list;
169 struct list_head commit_handler_list;
171 /* seperate list for multi-thread*/
172 struct list_head change_handler_list_main;
173 struct list_head change_handler_list_sub;
178 struct _tdm_private_layer {
179 struct list_head link;
181 tdm_private_display *private_display;
182 tdm_private_output *private_output;
185 tdm_layer *layer_backend;
187 tbm_surface_h pending_buffer;
188 tbm_surface_h waiting_buffer;
189 tbm_surface_h showing_buffer;
190 tbm_surface_queue_h buffer_queue;
192 struct list_head capture_list;
197 struct _tdm_private_pp {
198 struct list_head link;
202 tdm_private_display *private_display;
206 struct list_head src_pending_buffer_list;
207 struct list_head dst_pending_buffer_list;
208 struct list_head src_buffer_list;
209 struct list_head dst_buffer_list;
214 struct _tdm_private_capture {
215 struct list_head link;
216 struct list_head display_link;
220 tdm_capture_target target;
222 tdm_private_display *private_display;
223 tdm_private_output *private_output;
224 tdm_private_layer *private_layer;
226 tdm_capture *capture_backend;
228 struct list_head pending_buffer_list;
229 struct list_head buffer_list;
235 * Note that we don't need to (un)lock mutex to use this structure. If there is
236 * no TDM thread, all TDM resources are protected by private_display's mutex.
237 * If there is a TDM thread, this struct will be used only in a TDM thread.
238 * So, we don't need to protect this structure by mutex. Not thread-safe.
240 struct _tdm_private_loop {
241 /* TDM uses wl_event_loop to handle various event sources including the TDM
244 struct wl_display *wl_display;
245 struct wl_event_loop *wl_loop;
248 tdm_event_loop_source *backend_source;
250 /* In event loop, all resources are accessed by this dpy.
252 * - DO NOT include other private structure in this structure because this
253 * struct is not protected by mutex.
257 /* for handling TDM client requests */
258 tdm_private_server *private_server;
260 /* To have a TDM event thread. If TDM_THREAD enviroment variable is not set
261 * private_thread is NULL.
263 tdm_private_thread *private_thread;
266 struct _tdm_private_vblank_handler {
267 struct list_head link;
269 tdm_private_output *private_output;
270 tdm_output_vblank_handler func;
276 struct _tdm_private_commit_handler {
277 struct list_head link;
279 tdm_private_output *private_output;
280 tdm_output_commit_handler func;
286 struct _tdm_private_change_handler {
287 struct list_head link;
289 tdm_private_output *private_output;
290 tdm_output_change_handler func;
296 typedef struct _tdm_buffer_info {
297 tbm_surface_h buffer;
299 /* ref_count for backend */
300 int backend_ref_count;
302 struct list_head release_funcs;
303 struct list_head destroy_funcs;
305 struct list_head *list;
306 struct list_head link;
310 tdm_get_dpms_str(tdm_output_dpms dpms_value);
313 tdm_display_check_module_abi(tdm_private_display *private_display, int abimaj, int abimin);
316 tdm_display_find_output_stamp(tdm_private_display *private_display,
317 unsigned long stamp);
319 tdm_pp_find_stamp(tdm_private_display *private_display, unsigned long stamp);
320 tdm_private_capture *
321 tdm_capture_find_stamp(tdm_private_display *private_display, unsigned long stamp);
324 tdm_output_cb_vblank(tdm_output *output_backend, unsigned int sequence,
325 unsigned int tv_sec, unsigned int tv_usec, void *user_data);
327 tdm_output_cb_commit(tdm_output *output_backend, unsigned int sequence,
328 unsigned int tv_sec, unsigned int tv_usec, void *user_data);
330 tdm_output_cb_status(tdm_output *output_backend, tdm_output_conn_status status,
333 tdm_pp_cb_done(tdm_pp *pp_backend, tbm_surface_h src, tbm_surface_h dst,
336 tdm_capture_cb_done(tdm_capture *capture_backend, tbm_surface_h buffer,
340 tdm_output_call_change_handler_internal(tdm_private_output *private_output,
341 struct list_head *change_handler_list,
342 tdm_output_change_type type,
346 tdm_pp_create_internal(tdm_private_display *private_display, tdm_error *error);
348 tdm_pp_destroy_internal(tdm_private_pp *private_pp);
350 tdm_private_capture *
351 tdm_capture_create_output_internal(tdm_private_output *private_output,
353 tdm_private_capture *
354 tdm_capture_create_layer_internal(tdm_private_layer *private_layer,
357 tdm_capture_destroy_internal(tdm_private_capture *private_capture);
359 /* utility buffer functions for private */
361 tdm_buffer_get_info(tbm_surface_h buffer);
363 tdm_buffer_list_get_first_entry(struct list_head *list);
365 tdm_buffer_list_dump(struct list_head *list);
367 /* event functions for private */
369 tdm_event_loop_init(tdm_private_display *private_display);
371 tdm_event_loop_deinit(tdm_private_display *private_display);
373 tdm_event_loop_create_backend_source(tdm_private_display *private_display);
375 tdm_event_loop_get_fd(tdm_private_display *private_display);
377 tdm_event_loop_dispatch(tdm_private_display *private_display);
379 tdm_event_loop_flush(tdm_private_display *private_display);
383 TDM_THREAD_CB_OUTPUT_COMMIT,
384 TDM_THREAD_CB_OUTPUT_VBLANK,
385 TDM_THREAD_CB_OUTPUT_STATUS,
386 TDM_THREAD_CB_PP_DONE,
387 TDM_THREAD_CB_CAPTURE_DONE,
388 } tdm_thread_cb_type;
390 typedef struct _tdm_thread_cb_base tdm_thread_cb_base;
391 typedef struct _tdm_thread_cb_output_vblank tdm_thread_cb_output_commit;
392 typedef struct _tdm_thread_cb_output_vblank tdm_thread_cb_output_vblank;
393 typedef struct _tdm_thread_cb_output_status tdm_thread_cb_output_status;
394 typedef struct _tdm_thread_cb_pp_done tdm_thread_cb_pp_done;
395 typedef struct _tdm_thread_cb_capture_done tdm_thread_cb_capture_done;
397 struct _tdm_thread_cb_base {
398 tdm_thread_cb_type type;
402 struct _tdm_thread_cb_output_vblank {
403 tdm_thread_cb_base base;
404 unsigned long output_stamp;
405 unsigned int sequence;
407 unsigned int tv_usec;
411 struct _tdm_thread_cb_output_status {
412 tdm_thread_cb_base base;
413 unsigned long output_stamp;
414 tdm_output_conn_status status;
418 struct _tdm_thread_cb_pp_done {
419 tdm_thread_cb_base base;
420 unsigned long pp_stamp;
426 struct _tdm_thread_cb_capture_done {
427 tdm_thread_cb_base base;
428 unsigned long capture_stamp;
429 tbm_surface_h buffer;
434 tdm_thread_init(tdm_private_loop *private_loop);
436 tdm_thread_deinit(tdm_private_loop *private_loop);
438 tdm_thread_get_fd(tdm_private_loop *private_loop);
440 tdm_thread_send_cb(tdm_private_loop *private_loop, tdm_thread_cb_base *base);
442 tdm_thread_handle_cb(tdm_private_loop *private_loop);
444 tdm_thread_in_display_thread(pid_t tid);
446 tdm_thread_is_running(void);
449 tdm_server_init(tdm_private_loop *private_loop);
451 tdm_server_deinit(tdm_private_loop *private_loop);
454 tdm_helper_get_time_in_millis(void);
456 tdm_helper_get_time_in_micros(void);
458 extern pthread_mutex_t tdm_mutex_check_lock;
459 extern int tdm_mutex_locked;
460 extern int tdm_dump_enable;
462 #define _pthread_mutex_unlock(l) \
464 if (tdm_debug_mutex) \
465 TDM_INFO("mutex unlock"); \
466 pthread_mutex_lock(&tdm_mutex_check_lock); \
467 tdm_mutex_locked = 0; \
468 pthread_mutex_unlock(&tdm_mutex_check_lock); \
469 pthread_mutex_unlock(l); \
471 #ifdef TDM_CONFIG_MUTEX_TIMEOUT
472 #define MUTEX_TIMEOUT_SEC 5
473 #define _pthread_mutex_lock(l) \
475 if (tdm_debug_mutex) \
476 TDM_INFO("mutex lock"); \
477 struct timespec rtime; \
478 clock_gettime(CLOCK_REALTIME, &rtime); \
479 rtime.tv_sec += MUTEX_TIMEOUT_SEC; \
480 if (pthread_mutex_timedlock(l, &rtime)) { \
481 TDM_ERR("Mutex lock failed PID %d", getpid()); \
482 _pthread_mutex_unlock(l); \
485 pthread_mutex_lock(&tdm_mutex_check_lock); \
486 tdm_mutex_locked = 1; \
487 pthread_mutex_unlock(&tdm_mutex_check_lock); \
490 #else //TDM_CONFIG_MUTEX_TIMEOUT
491 #define _pthread_mutex_lock(l) \
493 if (tdm_debug_mutex) \
494 TDM_INFO("mutex lock"); \
495 pthread_mutex_lock(l); \
496 pthread_mutex_lock(&tdm_mutex_check_lock); \
497 tdm_mutex_locked = 1; \
498 pthread_mutex_unlock(&tdm_mutex_check_lock); \
500 #endif //TDM_CONFIG_MUTEX_TIMEOUT
501 //#define TDM_MUTEX_IS_LOCKED() (tdm_mutex_locked == 1)
502 static inline int TDM_MUTEX_IS_LOCKED(void)
505 pthread_mutex_lock(&tdm_mutex_check_lock);
506 ret = (tdm_mutex_locked == 1);
507 pthread_mutex_unlock(&tdm_mutex_check_lock);
512 _tdm_display_lock(tdm_display *dpy, const char *func);
514 _tdm_display_unlock(tdm_display *dpy, const char *func);
516 #define tdm_display_lock(dpy) _tdm_display_lock(dpy, __FUNCTION__)
517 #define tdm_display_unlock(dpy) _tdm_display_unlock(dpy, __FUNCTION__)
520 tdm_display_update_output(tdm_private_display *private_display,
521 tdm_output *output_backend, int pipe);
527 #endif /* _TDM_PRIVATE_H_ */