#include <tbm_surface_internal.h>
#include <Elementary.h>
#include <Ecore_Evas.h>
+#include <pthread.h>
#include <Ecore_Wayland.h>
#include <wayland-client.h>
#include <tizen-extension-client-protocol.h>
#include <screenshooter-client-protocol.h>
+#include <efl_util_screenshot_extension.h>
+
#include <dlog.h>
#ifdef LOG_TAG
#undef LOG_TAG
#endif
+/* Determine Tizen profile at runtime */
+#include <system_info.h>
+typedef enum {
+ TIZEN_PROFILE_UNKNOWN = 0,
+ TIZEN_PROFILE_MOBILE = 0x1,
+ TIZEN_PROFILE_WEARABLE = 0x2,
+ TIZEN_PROFILE_TV = 0x4,
+ TIZEN_PROFILE_IVI = 0x8,
+ TIZEN_PROFILE_COMMON = 0x10,
+} tizen_profile_t;
+static tizen_profile_t profile = TIZEN_PROFILE_UNKNOWN;
+static tizen_profile_t _get_tizen_profile()
+{
+ char *profileName;
+ system_info_get_platform_string("http://tizen.org/feature/profile", &profileName);
+ switch (*profileName) {
+ case 'm':
+ case 'M':
+ profile = TIZEN_PROFILE_MOBILE;
+ break;
+ case 'w':
+ case 'W':
+ profile = TIZEN_PROFILE_WEARABLE;
+ break;
+ case 't':
+ case 'T':
+ profile = TIZEN_PROFILE_TV;
+ break;
+ case 'i':
+ case 'I':
+ profile = TIZEN_PROFILE_IVI;
+ break;
+ default: // common or unknown ==> ALL ARE COMMON.
+ profile = TIZEN_PROFILE_COMMON;
+ }
+ free(profileName);
+
+ return profile;
+}
+static inline tizen_profile_t get_tizen_profile()
+{
+ if (__builtin_expect(profile != TIZEN_PROFILE_UNKNOWN, 1))
+ return profile;
+ return _get_tizen_profile();
+}
+
+
#define LOG_TAG "TIZEN_N_EFL_UTIL"
} policy;
struct
{
+ struct wl_event_queue *queue;
struct screenshooter *screenshooter;
+ struct tizen_screenshooter *tz_screenshooter;
struct wayland_tbm_client *tbm_client;
Eina_List *output_list;
+ uint32_t noti;
} shot;
struct
{
EINA_FALSE,
NULL, NULL,
{ NULL, NULL, NULL }, /* tizen_policy protocol */
- { NULL, NULL, NULL }, /* screenshooter protocol */
+ { NULL, NULL, NULL, NULL, NULL, 0 }, /* screenshooter protocol */
{ NULL, -1 } /* tizen_input_device_manager protocol */
},
{
static Eina_Bool _wl_init(void);
static void _cb_wl_reg_global(void *data, struct wl_registry *reg, unsigned int name, const char *interface, unsigned int version);
static void _cb_wl_reg_global_remove(void *data, struct wl_registry *reg, unsigned int name);
+static void _cb_wl_reg_screenshooter_global(void *data, struct wl_registry *reg, unsigned int name, const char *interface, unsigned int version);
+static void _cb_wl_reg_screenshooter_global_remove(void *data, struct wl_registry *reg, unsigned int name);
static Efl_Util_Callback_Info *_cb_info_find_by_wlsurf(void *wlsurf, int idx);
static void _cb_wl_tz_policy_conformant(void *data, struct tizen_policy *tizen_policy, struct wl_surface *surface, uint32_t is_conformant);
static void _cb_wl_tz_policy_conformant_area(void *data, struct tizen_policy *tizen_policy, struct wl_surface *surface, uint32_t conformant_part, uint32_t state, int32_t x, int32_t y, int32_t w, int32_t h);
_cb_wl_reg_global_remove
};
+static const struct wl_registry_listener _wl_reg_screenshooter_listener =
+{
+ _cb_wl_reg_screenshooter_global,
+ _cb_wl_reg_screenshooter_global_remove
+};
+
struct tizen_policy_listener _wl_tz_policy_listener =
{
_cb_wl_tz_policy_conformant,
static Eina_Bool
_wl_init(void)
{
+ struct wl_display *display_wrapper = NULL;
struct wl_registry *reg = NULL;
if (_eflutil.wl.init) return EINA_TRUE;
_eflutil.wl.dpy = ecore_wl_display_get();
EINA_SAFETY_ON_NULL_GOTO(_eflutil.wl.dpy, fail);
+ display_wrapper = wl_proxy_create_wrapper(_eflutil.wl.dpy);
+ EINA_SAFETY_ON_NULL_GOTO(display_wrapper, fail);
+
_eflutil.wl.queue = wl_display_create_queue(_eflutil.wl.dpy);
EINA_SAFETY_ON_NULL_GOTO(_eflutil.wl.queue, fail);
- reg = wl_display_get_registry(_eflutil.wl.dpy);
+ wl_proxy_set_queue((struct wl_proxy *)display_wrapper, _eflutil.wl.queue);
+
+ reg = wl_display_get_registry(display_wrapper);
+ wl_proxy_wrapper_destroy(display_wrapper);
+ display_wrapper = NULL;
EINA_SAFETY_ON_NULL_GOTO(reg, fail);
- wl_proxy_set_queue((struct wl_proxy*)reg, _eflutil.wl.queue);
wl_registry_add_listener(reg, &_wl_reg_listener, NULL);
_eflutil.wl.init = EINA_TRUE;
return EINA_TRUE;
fail:
+ if (display_wrapper)
+ wl_proxy_wrapper_destroy(display_wrapper);
+
if (_eflutil.wl.queue)
{
wl_event_queue_destroy(_eflutil.wl.queue);
};
static void
+_cb_tz_screenshot_format(void *data, struct tizen_screenshooter *tz_screenshooter, uint32_t format)
+{
+}
+
+static void
+_cb_tz_screenshot_noti(void *data, struct tizen_screenshooter *tz_screenshooter, uint32_t noti)
+{
+ _eflutil.wl.shot.noti = noti;
+}
+
+static const struct tizen_screenshooter_listener tz_screenshooter_listener =
+{
+ _cb_tz_screenshot_format,
+ _cb_tz_screenshot_noti
+};
+
+static void
_cb_wl_reg_global(void *data,
struct wl_registry *reg,
unsigned int name,
output->output = wl_registry_bind(reg, name, &wl_output_interface, version);
wl_output_add_listener(output->output, &output_listener, output);
}
- else if (strcmp(interface, "screenshooter") == 0)
- {
- _eflutil.wl.shot.screenshooter = wl_registry_bind(reg, name, &screenshooter_interface, version);
- screenshooter_add_listener(_eflutil.wl.shot.screenshooter, &screenshooter_listener, NULL);
- }
else if (strcmp(interface, "tizen_input_device_manager") == 0)
{
_eflutil.wl.devmgr.devicemgr = wl_registry_bind(reg, name, &tizen_input_device_manager_interface, version);
_eflutil.wl.display_policy.proto = NULL;
eina_hash_free(_eflutil.wl.display_policy.hash_brightness);
}
+/* LCOV_EXCL_STOP */
+
+static void
+_cb_wl_reg_screenshooter_global(void *data,
+ struct wl_registry *reg,
+ unsigned int name,
+ const char *interface,
+ unsigned int version)
+{
+ if (strcmp(interface, "screenshooter") == 0)
+ {
+ _eflutil.wl.shot.screenshooter = wl_registry_bind(reg, name, &screenshooter_interface, version);
+ screenshooter_add_listener(_eflutil.wl.shot.screenshooter, &screenshooter_listener, NULL);
+ }
+ else if (strcmp(interface, "tizen_screenshooter") == 0)
+ {
+ _eflutil.wl.shot.tz_screenshooter = wl_registry_bind(reg, name, &tizen_screenshooter_interface, version);
+ tizen_screenshooter_add_listener(_eflutil.wl.shot.tz_screenshooter, &tz_screenshooter_listener, NULL);
+
+ wl_display_roundtrip_queue(_eflutil.wl.dpy, _eflutil.wl.shot.queue);
+ }
+}
+
+/* LCOV_EXCL_START */
+static void
+_cb_wl_reg_screenshooter_global_remove(void *data,
+ struct wl_registry *reg,
+ unsigned int name)
+{
+}
static Efl_Util_Callback_Info *
_cb_info_find_by_wlsurf(void *wlsurf,
return EFL_UTIL_ERROR_INVALID_PARAMETER;
}
-#ifndef TIZEN_WEARABLE
API int
efl_util_set_window_screen_mode_error_cb(Evas_Object *window,
efl_util_window_screen_mode_error_cb callback,
void *user_data)
{
+ /* Wearable device cannot use this. */
+ if (get_tizen_profile() == TIZEN_PROFILE_WEARABLE)
+ return EFL_UTIL_ERROR_NO_SUCH_DEVICE;
+
dlog_print(DLOG_WARN, LOG_TAG,
"DEPRECATION WARNING: efl_util_set_window_screen_mode_error_cb() is deprecated and will be removed from next release. Use the return value of efl_util_set_window_screen_mode() instead.");
API int
efl_util_unset_window_screen_mode_error_cb(Evas_Object *window)
{
+ /* Wearable device cannot use this. */
+ if (get_tizen_profile() == TIZEN_PROFILE_WEARABLE)
+ return EFL_UTIL_ERROR_NO_SUCH_DEVICE;
+
dlog_print(DLOG_WARN, LOG_TAG,
"DEPRECATION WARNING: efl_util_unset_window_screen_mode_error_cb() is deprecated and will be removed from next release. Use the return value of efl_util_set_window_screen_mode() instead.");
return EFL_UTIL_ERROR_NONE;
}
-#endif
API int
efl_util_set_window_brightness(Evas_Object *window, int brightness)
{
int ret = EFL_UTIL_ERROR_NONE;
efl_util_inputgen_h inputgen_h = NULL;
+ unsigned int clas = 0x0;
if (!dev_type ||
dev_type & ~(EFL_UTIL_INPUT_DEVTYPE_TOUCHSCREEN
goto out;
}
+ if (dev_type & EFL_UTIL_INPUT_DEVTYPE_TOUCHSCREEN)
+ clas |= TIZEN_INPUT_DEVICE_MANAGER_CLAS_TOUCHSCREEN;
+ if (dev_type & EFL_UTIL_INPUT_DEVTYPE_KEYBOARD)
+ clas |= TIZEN_INPUT_DEVICE_MANAGER_CLAS_KEYBOARD;
+ if (dev_type & EFL_UTIL_INPUT_DEVTYPE_POINTER)
+ clas |= TIZEN_INPUT_DEVICE_MANAGER_CLAS_MOUSE;
+
while (!_eflutil.wl.devmgr.devicemgr)
wl_display_dispatch_queue(_eflutil.wl.dpy, _eflutil.wl.queue);
- tizen_input_device_manager_init_generator(_eflutil.wl.devmgr.devicemgr);
+ tizen_input_device_manager_init_generator(_eflutil.wl.devmgr.devicemgr, clas);
while (_eflutil.wl.devmgr.request_notified == -1)
wl_display_dispatch_queue(_eflutil.wl.dpy, _eflutil.wl.queue);
efl_util_input_deinitialize_generator(efl_util_inputgen_h inputgen_h)
{
int ret = EFL_UTIL_ERROR_NONE;
+ unsigned int clas = 0x0;
EINA_SAFETY_ON_NULL_RETURN_VAL(inputgen_h, EFL_UTIL_ERROR_INVALID_PARAMETER);
+ if (inputgen_h->init_type & EFL_UTIL_INPUT_DEVTYPE_TOUCHSCREEN)
+ clas |= TIZEN_INPUT_DEVICE_MANAGER_CLAS_TOUCHSCREEN;
+ if (inputgen_h->init_type & EFL_UTIL_INPUT_DEVTYPE_KEYBOARD)
+ clas |= TIZEN_INPUT_DEVICE_MANAGER_CLAS_KEYBOARD;
+ if (inputgen_h->init_type & EFL_UTIL_INPUT_DEVTYPE_POINTER)
+ clas |= TIZEN_INPUT_DEVICE_MANAGER_CLAS_MOUSE;
+
free(inputgen_h);
inputgen_h = NULL;
EINA_SAFETY_ON_NULL_RETURN_VAL(_eflutil.wl.devmgr.devicemgr, EFL_UTIL_ERROR_INVALID_PARAMETER);
- tizen_input_device_manager_deinit_generator(_eflutil.wl.devmgr.devicemgr);
+ tizen_input_device_manager_deinit_generator(_eflutil.wl.devmgr.devicemgr, clas);
while (_eflutil.wl.devmgr.request_notified == -1)
wl_display_dispatch_queue(_eflutil.wl.dpy, _eflutil.wl.queue);
/* tbm bufmgr */
tbm_bufmgr bufmgr;
+
+ Eina_Bool auto_rotation;
};
/* scrrenshot handle */
static efl_util_screenshot_h g_screenshot;
+static Eina_Bool shot_mutex_init;
+static pthread_mutex_t shot_lock;
+
+static Eina_Bool
+_screenshot_mutex_init(void)
+{
+ if (shot_mutex_init)
+ return EINA_TRUE;
+
+ if (pthread_mutex_init(&shot_lock, NULL))
+ {
+ fprintf(stderr, "[screenshot] fail: mutex init"); /*LCOV_EXCL_LINE*/
+ return EINA_FALSE; /*LCOV_EXCL_LINE*/
+ }
+
+ shot_mutex_init = EINA_TRUE;
+
+ return EINA_TRUE;
+}
+
+static Eina_Bool
+_screenshot_mutex_destory(void)
+{
+ if (!shot_mutex_init)
+ return EINA_TRUE;
+
+ if (pthread_mutex_destroy(&shot_lock))
+ {
+ fprintf(stderr, "[screenshot] fail: mutex destory"); /*LCOV_EXCL_LINE*/
+ return EINA_FALSE; /*LCOV_EXCL_LINE*/
+ }
+
+ shot_mutex_init = EINA_FALSE;
+
+ return EINA_TRUE;
+}
+
+void
+_screenshot_mutex_lock(void)
+{
+ if (!_screenshot_mutex_init())
+ return;
+
+ pthread_mutex_lock(&shot_lock);
+}
+
+void
+_screenshot_mutex_unlock(void)
+{
+ pthread_mutex_unlock(&shot_lock);
+}
API efl_util_screenshot_h
efl_util_screenshot_initialize(int width, int height)
{
efl_util_screenshot_h screenshot = NULL;
+ struct wl_display *display_wrapper = NULL;
+ struct wl_registry *reg = NULL;
+ int ret = 0;
+
+ EINA_SAFETY_ON_FALSE_GOTO(width > 0, fail_param);
+ EINA_SAFETY_ON_FALSE_GOTO(height > 0, fail_param);
+
+ _screenshot_mutex_lock();
if (!_eflutil.wl.shot.screenshooter)
{
- int ret = 0;
- _wl_init();
- while (!_eflutil.wl.shot.screenshooter && ret != -1)
- ret = wl_display_roundtrip_queue(_eflutil.wl.dpy, _eflutil.wl.queue);
+ ret = _wl_init();
+ if (ret == (int)EINA_FALSE)
+ {
+ set_last_result(EFL_UTIL_ERROR_INVALID_PARAMETER);
+ _screenshot_mutex_unlock();
+ return NULL;
+ }
+ wl_display_roundtrip_queue(_eflutil.wl.dpy, _eflutil.wl.queue);
+
+ display_wrapper = wl_proxy_create_wrapper(_eflutil.wl.dpy);
+ EINA_SAFETY_ON_NULL_GOTO(display_wrapper, fail_memory);
+
+ _eflutil.wl.shot.queue = wl_display_create_queue(_eflutil.wl.dpy);
+ EINA_SAFETY_ON_NULL_GOTO(_eflutil.wl.shot.queue, fail_memory);
+
+ wl_proxy_set_queue((struct wl_proxy *)display_wrapper, _eflutil.wl.shot.queue);
+
+ reg = wl_display_get_registry(display_wrapper);
+ wl_proxy_wrapper_destroy(display_wrapper);
+ display_wrapper = NULL;
+ EINA_SAFETY_ON_NULL_GOTO(reg, fail_init);
+
+ wl_registry_add_listener(reg, &_wl_reg_screenshooter_listener, NULL);
+
+ ret = wl_display_roundtrip_queue(_eflutil.wl.dpy, _eflutil.wl.shot.queue);
+ EINA_SAFETY_ON_TRUE_GOTO(ret == -1, fail_init);
EINA_SAFETY_ON_NULL_GOTO(_eflutil.wl.shot.screenshooter, fail_init);
+ EINA_SAFETY_ON_NULL_GOTO(_eflutil.wl.shot.tz_screenshooter, fail_init);
- _eflutil.wl.shot.tbm_client = wayland_tbm_client_init(_eflutil.wl.dpy);
- EINA_SAFETY_ON_NULL_GOTO(_eflutil.wl.shot.tbm_client, fail_init);
+ _eflutil.wl.shot.tbm_client = wayland_tbm_client_init(_eflutil.wl.dpy);
+ EINA_SAFETY_ON_NULL_GOTO(_eflutil.wl.shot.tbm_client, fail_init);
}
- EINA_SAFETY_ON_FALSE_GOTO(width > 0, fail_param);
- EINA_SAFETY_ON_FALSE_GOTO(height > 0, fail_param);
+ if (_eflutil.wl.shot.noti == 0)
+ {
+ fprintf(stderr, "[screenshot] fail: privilege error\n"); /* LCOV_EXCL_LINE */
+ goto fail_init;
+ }
if (g_screenshot)
{
g_screenshot->height = height;
}
+ _screenshot_mutex_unlock();
+
return g_screenshot;
}
screenshot->width = width;
screenshot->height = height;
+ screenshot->auto_rotation = EINA_TRUE;
screenshot->bufmgr = wayland_tbm_client_get_bufmgr(_eflutil.wl.shot.tbm_client);
EINA_SAFETY_ON_NULL_GOTO(screenshot->bufmgr, fail_init);
screenshooter_set_user_data(_eflutil.wl.shot.screenshooter, &screenshot->shot_done);
+ _screenshot_mutex_unlock();
+
return g_screenshot;
fail_param:
return NULL;
fail_memory:
/* LCOV_EXCL_START */
+ if (display_wrapper)
+ wl_proxy_wrapper_destroy(display_wrapper);
set_last_result(EFL_UTIL_ERROR_OUT_OF_MEMORY);
return NULL;
-/* LCOV_EXCL_STOP */
fail_init:
if (screenshot)
efl_util_screenshot_deinitialize(screenshot);
+ _screenshot_mutex_unlock();
set_last_result(EFL_UTIL_ERROR_SCREENSHOT_INIT_FAIL);
return NULL;
+/* LCOV_EXCL_STOP */
}
API int
efl_util_screenshot_deinitialize(efl_util_screenshot_h screenshot)
{
+ _screenshot_mutex_lock();
+
if (!screenshot)
- return EFL_UTIL_ERROR_INVALID_PARAMETER;
+ {
+ _screenshot_mutex_unlock();
+ _screenshot_mutex_destory();
+ return EFL_UTIL_ERROR_INVALID_PARAMETER;
+ }
free(screenshot);
g_screenshot = NULL;
+ if (_eflutil.wl.shot.queue)
+ {
+ wl_event_queue_destroy(_eflutil.wl.shot.queue);
+ _eflutil.wl.shot.queue = NULL;
+ }
+
if (_eflutil.wl.shot.screenshooter)
- screenshooter_set_user_data(_eflutil.wl.shot.screenshooter, NULL);
+ {
+ screenshooter_destroy(_eflutil.wl.shot.screenshooter);
+ _eflutil.wl.shot.screenshooter = NULL;
+ }
+ if (_eflutil.wl.shot.tz_screenshooter)
+ {
+ tizen_screenshooter_destroy(_eflutil.wl.shot.tz_screenshooter);
+ _eflutil.wl.shot.tz_screenshooter = NULL;
+ }
+
+ _screenshot_mutex_unlock();
+ _screenshot_mutex_destory();
return EFL_UTIL_ERROR_NONE;
}
Efl_Util_Wl_Output_Info *output;
int ret = 0;
- if (screenshot != g_screenshot)
+ _screenshot_mutex_lock();
+
+ if (!screenshot || (screenshot != g_screenshot))
{
set_last_result(EFL_UTIL_ERROR_INVALID_PARAMETER);
+ _screenshot_mutex_unlock();
return NULL;
}
screenshot->shot_done = EINA_FALSE;
while (!screenshot->shot_done && ret != -1)
- ret = wl_display_roundtrip_queue(_eflutil.wl.dpy, _eflutil.wl.queue);
+ ret = wl_display_roundtrip_queue(_eflutil.wl.dpy, _eflutil.wl.shot.queue);
if (ret == -1)
{
/* reset shot_done for next screenshot */
screenshot->shot_done = EINA_FALSE;
+ set_last_result(EFL_UTIL_ERROR_NONE);
+
+ _screenshot_mutex_unlock();
+
return t_surface;
fail:
set_last_result(EFL_UTIL_ERROR_SCREENSHOT_EXECUTION_FAIL);
+ _screenshot_mutex_unlock();
+
return NULL;
}
+
+API int
+efl_util_screenshot_set_auto_rotation(efl_util_screenshot_h screenshot, int set)
+{
+ if (!screenshot || (screenshot != g_screenshot))
+ return EFL_UTIL_ERROR_INVALID_PARAMETER;
+
+ if (!(set == 0 || set == 1))
+ return EFL_UTIL_ERROR_INVALID_PARAMETER;
+
+ if (set)
+ g_screenshot->auto_rotation = EINA_TRUE;
+ else
+ g_screenshot->auto_rotation = EINA_FALSE;
+
+ tizen_screenshooter_set_oneshot_auto_rotation(_eflutil.wl.shot.tz_screenshooter, g_screenshot->auto_rotation);
+
+ return EFL_UTIL_ERROR_NONE;
+}
+
+API int
+efl_util_screenshot_get_auto_rotation(efl_util_screenshot_h screenshot, int *set)
+{
+ if (!screenshot || (screenshot != g_screenshot) || !set)
+ return EFL_UTIL_ERROR_INVALID_PARAMETER;
+
+ *set = g_screenshot->auto_rotation;
+
+ return EFL_UTIL_ERROR_NONE;
+}