return e;
}
+pui_bool
+_geometry_get(int *width, int *height)
+{
+ if (!width || !height)
+ return 0;
+
+ if (width)
+ *width = DEFAULT_BACKEND_GEOM_WIDTH;
+ if (height)
+ *height = DEFAULT_BACKEND_GEOM_HEIGHT;
+
+ return 1;
+}
+
pui_int_error
_is_ani_supported(pui_id id)
{
}
backend_data->create_ani_collection = _create_ani_collection;
+ backend_data->geometry_get = _geometry_get;
backend_data->ani_create = _ani_create;
backend_data->ani_destroy = _ani_destroy;
}
backend_data->create_ani_collection = NULL;
+ backend_data->geometry_get = NULL;
backend_data->ani_create = NULL;
backend_data->ani_destroy = NULL;
#define ANI_COLLECTION_DIR "/usr/share/pui/"
#define MAX_STR 1024
+#define DEFAULT_BACKEND_GEOM_WIDTH 12
+#define DEFAULT_BACKEND_GEOM_HEIGHT 1
+
#ifndef PI
#define PI 3.1415926545
#endif
PUI_API int
pui_ani_get_repeat(pui_ani_h ani_h);
+PUI_API pui_ani_control_buffer *
+pui_display_get_buffer(pui_h handle);
+
+PUI_API pui_error
+pui_display_set_buffer(pui_h handle, pui_ani_control_buffer *buffer);
+
+PUI_API pui_error
+pui_display_update(pui_h handle);
+
+PUI_API pui_error
+pui_display_manual_render_set(pui_h handle, pui_bool set);
+
+PUI_API pui_bool
+pui_display_manual_render_get(pui_h handle);
+
+PUI_API pui_bool
+pui_display_geometry_get(pui_h handle, int *width, int *height);
+
#ifdef __cplusplus
}
#endif
void *data;
pui_int_error (*create_ani_collection)(void);
+ pui_bool (*geometry_get)(int *width, int *height);
pui_backend_ani_data *(*ani_create)(pui_id id);
void (*ani_destroy)(pui_backend_ani_data *ani_data);
PUI_ERROR_INVALID_ANI_CMD,
PUI_ERROR_INVALID_ANI_REPEAT,
PUI_ERROR_INTERNAL,
+ PUI_ERROR_INVALID_HANDLE,
+ PUI_ERROR_INVALID_SURFACE,
+ PUI_ERROR_INVALID_BUFFER,
+ PUI_ERROR_MANUAL_RENDER_ENABLED,
+ PUI_ERROR_UNABLE_SET_MANUAL_RENDER,
} pui_error;
typedef enum {
PUI_INT_ERROR_NO_ANI_AVAILABLE,
PUI_INT_ERROR_BACKEND_FUNC_ERROR,
PUI_INT_ERROR_INVALID_RESOURCES,
+ PUI_INT_ERROR_UNKNOWN,
} pui_int_error;
typedef unsigned int pui_bool;
int PUI_EVENT_ANI_READY_TO_START = 0;
int PUI_EVENT_ANI_READY_TO_RESUME = 0;
+static int KEY_WL_BUFFER = 0xabcdbeaf;
+static int KEY_CLIENT = 0xdcbabeaf;
+
pui_error_string
pui_error_to_string(pui_error e)
{
str = "PUI_Internal_Error";
break;
+ case PUI_ERROR_MANUAL_RENDER_ENABLED:
+ str = "PUI_Manual_Render_Enabled";
+ break;
+
+ case PUI_ERROR_UNABLE_SET_MANUAL_RENDER:
+ str = "PUI_Unable_To_Set_Manual_Render";
+ break;
+
default:
str = "PUI_Unknown_Error";
}
return str;
}
+static void
+_buffer_release(void *data, struct wl_buffer *buffer)
+{
+ tbm_surface_h surface = (tbm_surface_h)data;
+ pui_h handle;
+
+ tbm_surface_internal_get_user_data(surface, (unsigned long)&KEY_CLIENT, (void **)&handle);
+ tbm_surface_queue_release(handle->tbm_queue, surface);
+
+ pui_info("[UPDATE] release wl_buffer:%p, surface:%p\n", buffer, surface);
+}
+
+static const struct wl_buffer_listener buffer_listener = {
+ _buffer_release
+};
+
+pui_ani_control_buffer *
+pui_display_get_buffer(pui_h handle)
+{
+ tbm_surface_error_e ret;
+ tbm_surface_h surface;
+ pui_ani_control_buffer *buffer = NULL;
+
+ if (!handle)
+ {
+ pui_err("Error : PUI_INT_ERROR_INVALID_HANDLE\n");
+ return NULL;
+ }
+
+ if (handle->current_surface)
+ {
+ pui_warn("Current_surface is not used !\n");
+ }
+
+ if (!tbm_surface_queue_can_dequeue(handle->tbm_queue, 0))
+ {
+ pui_err("[UPDATE] Cannot dequeue (error : PUI_ERROR_INTERNAL)\n");
+ return NULL;
+ }
+
+ ret = tbm_surface_queue_dequeue(handle->tbm_queue, &surface);
+
+ if (ret != TBM_SURFACE_ERROR_NONE)
+ {
+ pui_err("[UPDATE] dequeue err:%d\n", ret);
+ return NULL;
+ }
+
+ tbm_surface_map(surface, TBM_SURF_OPTION_READ | TBM_SURF_OPTION_WRITE, &handle->current_sinfo);
+
+ handle->current_surface = surface;
+ buffer = (pui_ani_control_buffer *)&(handle->current_sinfo.planes[0]);
+
+ return buffer;
+}
+
+pui_error
+pui_display_set_buffer(pui_h handle, pui_ani_control_buffer *buffer)
+{
+ if (!handle)
+ {
+ pui_err("Error : PUI_ERROR_INVALID_HANDLE\n");
+ return PUI_ERROR_INVALID_HANDLE;
+ }
+
+ if (!handle->current_surface)
+ {
+ pui_err("Current_surface is not valid !\n");
+ return PUI_ERROR_INVALID_SURFACE;
+ }
+
+ if (!buffer || !buffer->ptr || !buffer->size)
+ return PUI_ERROR_INVALID_BUFFER;
+
+ handle->is_buffer_set = 1;
+
+ return PUI_ERROR_NONE;
+}
+
+pui_error
+pui_display_update(pui_h handle)
+{
+ tbm_surface_h surface;
+ tbm_surface_error_e ret;
+ struct wl_buffer *wl_buffer = NULL;
+
+ if (!handle)
+ {
+ pui_err("Error : PUI_ERROR_INVALID_HANDLE\n");
+ return PUI_ERROR_INVALID_HANDLE;
+ }
+
+ if (!handle->current_surface)
+ return PUI_ERROR_INVALID_SURFACE;
+
+ if (!handle->is_buffer_set)
+ {
+ pui_err("Buffer is not set !\n");
+ return PUI_ERROR_INVALID_BUFFER;
+ }
+
+ surface = handle->current_surface;
+ handle->current_surface = NULL;
+
+ tbm_surface_unmap(surface);
+
+ ret = tbm_surface_queue_enqueue(handle->tbm_queue, surface);
+
+ if (ret != TBM_SURFACE_ERROR_NONE)
+ {
+ pui_err("[UPDATE] enqueue err:%d\n", ret);
+ return PUI_ERROR_INVALID_SURFACE;
+ }
+
+ ret = tbm_surface_queue_acquire(handle->tbm_queue, &surface);
+
+ if (ret != TBM_SURFACE_ERROR_NONE)
+ {
+ pui_err("[UPDATE] acquire err:%d\n", ret);
+ return PUI_ERROR_INVALID_SURFACE;
+ }
+
+ if (!tbm_surface_internal_get_user_data(surface, (unsigned long)&KEY_WL_BUFFER, (void **)&wl_buffer)) {
+ wl_buffer = wayland_tbm_client_create_buffer(handle->wl_tbm_client, surface);
+
+ if (!wl_buffer)
+ {
+ pui_err("[UPDATE] failed to create wl_buffer tbm_surface:%p\n", surface);
+ return PUI_ERROR_INVALID_BUFFER;
+ }
+
+ wl_buffer_add_listener(wl_buffer, &buffer_listener, surface);
+
+ tbm_surface_internal_add_user_data(surface, (unsigned long)&KEY_WL_BUFFER, NULL);
+ tbm_surface_internal_set_user_data(surface, (unsigned long)&KEY_WL_BUFFER, wl_buffer);
+ tbm_surface_internal_add_user_data(surface, (unsigned long)&KEY_CLIENT, NULL);
+ tbm_surface_internal_set_user_data(surface, (unsigned long)&KEY_CLIENT, handle);
+ }
+
+ if (!wl_buffer)
+ {
+ pui_err("[UPDATE] dequeue err:%d\n", ret);
+ return PUI_ERROR_INVALID_BUFFER;
+ }
+
+ ecore_wl2_window_buffer_attach(handle->win, wl_buffer, 0, 0, 0);
+ ecore_wl2_window_damage(handle->win, NULL, 0);
+ ecore_wl2_window_commit(handle->win, EINA_TRUE);
+
+ pui_info("[UPDATE] commit wl_buffer:%p, surface:%p\n", wl_buffer, surface);
+
+ handle->is_buffer_set = 0;
+
+ return PUI_ERROR_NONE;
+}
+
+pui_error
+pui_display_manual_render_set(pui_h handle, pui_bool set)
+{
+ pui_ani_h ani_h = NULL;
+
+ if (!handle)
+ {
+ pui_err("Error : PUI_ERROR_INVALID_HANDLE\n");
+ return PUI_ERROR_INVALID_HANDLE;
+ }
+
+ ani_h = handle->current_ani_h;
+
+ if (ani_h)
+ {
+ if (pui_ani_status_get(ani_h->ani) == PUI_ANI_STATUS_RUNNING ||
+ pui_ani_status_get(ani_h->ani) == PUI_ANI_STATUS_PAUSED)
+ {
+ pui_err("Please stop the current animation.\n");
+ return PUI_ERROR_UNABLE_SET_MANUAL_RENDER;
+ }
+ }
+
+ handle->manual_render = !!set;
+
+ return PUI_ERROR_NONE;
+}
+
+pui_bool
+pui_display_manual_render_get(pui_h handle)
+{
+ if (!handle)
+ {
+ pui_err("Error : PUI_ERROR_INVALID_HANDLE\n");
+ return 0;
+ }
+
+ return handle->manual_render;
+}
+
+pui_bool
+pui_display_geometry_get(pui_h handle, int *width, int *height)
+{
+ if (!width || !height)
+ return 0;
+
+ *width = 0;
+ *height = 0;
+
+ if (!pui_module || !pui_module->backend_module_data) {
+ pui_err("pui module data is not loaded\n");
+ return 0;
+ }
+
+ if (!pui_module->backend_module_data->geometry_get)
+ {
+ pui_err("Backend doesn't have geometry_get() !\n");
+ return 0;
+ }
+
+ return pui_module->backend_module_data->geometry_get(width, height);
+}
+
pui_h
pui_create(Ecore_Wl2_Window *win)
{
handle->wl_tbm_client = wl_tbm_client;
handle->ani_handles = NULL;
handle->current_ani_h = NULL;
+ handle->manual_render = 0;
handle->backend_module_data = pui_module->backend_module_data;
handle->tbm_queue = wayland_tbm_client_create_surface_queue(handle->wl_tbm_client,
return;
}
+ if (!pui_module->backend_module_data->create_ani_collection)
+ {
+ pui_err("Backend doesn't have create_ani_collection() !\n");
+ return;
+ }
+
ret = pui_module->backend_module_data->create_ani_collection();
if (ret != PUI_INT_ERROR_NONE) {
pui_err("Failed to collect animations data (%s)\n",
#include "PUI.h"
#include <Eina.h>
-static int KEY_WL_BUFFER = 0xabcdbeaf;
-static int KEY_CLIENT = 0xdcbabeaf;
+pui_int_error
+_pui_ani_error_to_int_error(pui_error e)
+{
+ pui_int_error ei;
+
+ switch (e)
+ {
+ case PUI_ERROR_NONE:
+ ei = PUI_INT_ERROR_NONE;
+ break;
+
+ case PUI_ERROR_INVALID_HANDLE:
+ ei = PUI_INT_ERROR_INVALID_HANDLE;
+ break;
+
+ case PUI_ERROR_INVALID_SURFACE:
+ ei = PUI_INT_ERROR_INVALID_SURFACE;
+ break;
+
+ case PUI_ERROR_INVALID_BUFFER:
+ ei = PUI_INT_ERROR_INVALID_BUFFER;
+ break;
+
+ default:
+ ei = PUI_INT_ERROR_UNKNOWN;
+ break;
+ }
+
+ return ei;
+}
static void
_pui_ani_cb_frame_done(Ecore_Wl2_Window *win, uint32_t timestamp EINA_UNUSED, void *data)
return;
}
-static void
-_buffer_release(void *data, struct wl_buffer *buffer)
-{
- tbm_surface_h surface = (tbm_surface_h)data;
- pui_h handle;
-
- tbm_surface_internal_get_user_data(surface, (unsigned long)&KEY_CLIENT, (void **)&handle);
- tbm_surface_queue_release(handle->tbm_queue, surface);
-
- pui_info("[UPDATE] release wl_buffer:%p, surface:%p\n", buffer, surface);
-}
-
-static const struct wl_buffer_listener buffer_listener = {
- _buffer_release
-};
-
pui_ani_control_buffer *
pui_ani_get_buffer(pui_ani_h ani_h)
{
pui_h handle = NULL;
- tbm_surface_error_e ret;
- tbm_surface_h surface;
pui_ani_control_buffer *buffer = NULL;
if (!ani_h)
}
handle = ani_h->pui_handle;
-
- if (!handle)
- {
- pui_err("Error : PUI_INT_ERROR_INVALID_HANDLE\n");
- return NULL;
- }
-
- if (handle->current_surface)
- {
- pui_warn("Current_surface is not used !\n");
- }
-
- if (!tbm_surface_queue_can_dequeue(handle->tbm_queue, 0))
- {
- pui_err("[UPDATE] Cannot dequeue (error : PUI_ERROR_INTERNAL)\n");
- return NULL;
- }
-
- ret = tbm_surface_queue_dequeue(handle->tbm_queue, &surface);
-
- if (ret != TBM_SURFACE_ERROR_NONE)
- {
- pui_err("[UPDATE] dequeue err:%d\n", ret);
- return NULL;
- }
-
- tbm_surface_map(surface, TBM_SURF_OPTION_READ | TBM_SURF_OPTION_WRITE, &handle->current_sinfo);
-
- handle->current_surface = surface;
- buffer = (pui_ani_control_buffer *)&(handle->current_sinfo.planes[0]);
+ buffer = pui_display_get_buffer(handle);
return buffer;
}
pui_int_error
pui_ani_set_buffer(pui_ani_h ani_h, pui_ani_control_buffer *buffer)
{
+ pui_error e = PUI_ERROR_NONE;
pui_h handle = NULL;
- if (!ani_h)
- {
- pui_err("Invalid ani handle !\n");
- return PUI_INT_ERROR_INVALID_HANDLE;
- }
-
handle = ani_h->pui_handle;
+ e = pui_display_set_buffer(handle, buffer);
- if (!handle)
- {
- pui_err("Error : PUI_INT_ERROR_INVALID_HANDLE\n");
- return PUI_INT_ERROR_INVALID_HANDLE;
- }
-
- if (!handle->current_surface)
- {
- pui_err("Current_surface is not valid !\n");
- return PUI_INT_ERROR_INVALID_SURFACE;
- }
-
- if (!buffer || !buffer->ptr || !buffer->size)
- return PUI_INT_ERROR_INVALID_BUFFER;
-
- handle->is_buffer_set = 1;
-
- return PUI_INT_ERROR_NONE;
+ return _pui_ani_error_to_int_error(e);
}
pui_int_error
pui_ani_update(pui_ani_h ani_h)
{
- tbm_surface_h surface;
- tbm_surface_error_e ret;
- struct wl_buffer *wl_buffer = NULL;
-
+ pui_error e = PUI_ERROR_NONE;
pui_h handle = NULL;
if (!ani_h)
}
handle = ani_h->pui_handle;
+ e = pui_display_update(handle);
- if (!handle)
- {
- pui_err("Error : PUI_INT_ERROR_INVALID_HANDLE\n");
- return PUI_INT_ERROR_INVALID_HANDLE;
- }
-
- if (!handle->current_surface)
- return PUI_INT_ERROR_INVALID_SURFACE;
-
- if (!handle->is_buffer_set)
- {
- pui_err("Buffer is not set !\n");
- return PUI_INT_ERROR_INVALID_BUFFER;
- }
-
- surface = handle->current_surface;
- handle->current_surface = NULL;
-
- tbm_surface_unmap(surface);
-
- ret = tbm_surface_queue_enqueue(handle->tbm_queue, surface);
-
- if (ret != TBM_SURFACE_ERROR_NONE)
- {
- pui_err("[UPDATE] enqueue err:%d\n", ret);
- return PUI_INT_ERROR_INVALID_SURFACE;
- }
-
- ret = tbm_surface_queue_acquire(handle->tbm_queue, &surface);
-
- if (ret != TBM_SURFACE_ERROR_NONE)
- {
- pui_err("[UPDATE] acquire err:%d\n", ret);
- return PUI_INT_ERROR_INVALID_SURFACE;
- }
-
- if (!tbm_surface_internal_get_user_data(surface, (unsigned long)&KEY_WL_BUFFER, (void **)&wl_buffer)) {
- wl_buffer = wayland_tbm_client_create_buffer(handle->wl_tbm_client, surface);
-
- if (!wl_buffer)
- {
- pui_err("[UPDATE] failed to create wl_buffer tbm_surface:%p\n", surface);
- return PUI_INT_ERROR_INVALID_BUFFER;
- }
-
- wl_buffer_add_listener(wl_buffer, &buffer_listener, surface);
-
- tbm_surface_internal_add_user_data(surface, (unsigned long)&KEY_WL_BUFFER, NULL);
- tbm_surface_internal_set_user_data(surface, (unsigned long)&KEY_WL_BUFFER, wl_buffer);
- tbm_surface_internal_add_user_data(surface, (unsigned long)&KEY_CLIENT, NULL);
- tbm_surface_internal_set_user_data(surface, (unsigned long)&KEY_CLIENT, handle);
- }
-
- if (!wl_buffer)
- {
- pui_err("[UPDATE] dequeue err:%d\n", ret);
- return PUI_INT_ERROR_INVALID_BUFFER;
- }
-
- ecore_wl2_window_buffer_attach(handle->win, wl_buffer, 0, 0, 0);
- ecore_wl2_window_damage(handle->win, NULL, 0);
- ecore_wl2_window_commit(handle->win, EINA_TRUE);
-
- pui_info("[UPDATE] commit wl_buffer:%p, surface:%p\n", wl_buffer, surface);
-
- handle->is_buffer_set = 0;
-
- return PUI_INT_ERROR_NONE;
+ return _pui_ani_error_to_int_error(e);
}
static Eina_Bool
if (cmd == PUI_ANI_CMD_START)
{
+ if (handle->manual_render)
+ {
+ pui_err("Manual render has been set ! Unable to start animation.\n");
+ return PUI_ERROR_MANUAL_RENDER_ENABLED;
+ }
+
if (handle->current_ani_h && handle->current_ani_h != ani_h)
{
pui_ani_t *current_ani = handle->current_ani_h->ani;
tbm_surface_info_s current_sinfo;
tbm_surface_h current_surface;
int is_buffer_set;
+ pui_bool manual_render;
Eina_List *ani_handles;