efl/src/tests/ecore/ecore_test_ecore.c

491 lines
11 KiB
C

#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <stdio.h>
#include <unistd.h>
#ifdef _WIN32
# include <evil_private.h> /* pipe */
#endif
#include <Eina.h>
#include <Ecore.h>
#include "ecore_suite.h"
#define ECORE_EVENT_CUSTOM_1 1
#define ECORE_EVENT_CUSTOM_2 2
static Eina_Bool
_quit_cb(void *data)
{
Eina_Bool *val = data;
if (val) *val = EINA_TRUE;
ecore_main_loop_quit();
return EINA_FALSE;
}
EFL_START_TEST(ecore_test_ecore_init)
{
}
EFL_END_TEST
EFL_START_TEST(ecore_test_ecore_main_loop)
{
Eina_Bool did = EINA_FALSE;
Ecore_Timer *timer;
timer = ecore_timer_add(0.0, _quit_cb, &did);
fail_if(timer == NULL);
/* ensure that this does not auto-cancel main loop */
ecore_main_loop_quit();
ecore_main_loop_begin();
fail_if(did == EINA_FALSE);
}
EFL_END_TEST
// Disabled tests: inner main loops are not supposed to work!
#if 0
static Eina_Bool _timer3(void *data EINA_UNUSED)
{
/* timer 3, do nothing */
return EINA_FALSE;
}
static Eina_Bool _timer2(void *data EINA_UNUSED)
{
/* timer 2, quit inner mainloop */
ecore_main_loop_quit();
return EINA_FALSE;
}
static Eina_Bool _timer1(void *data)
{
/* timer 1, begin inner mainloop */
int *times = data;
(*times)++;
ecore_timer_add(0.3, _timer2, NULL);
ecore_timer_add(0.1, _timer3, NULL);
ecore_main_loop_begin();
ecore_main_loop_quit();
return EINA_FALSE;
}
EFL_START_TEST(ecore_test_ecore_main_loop_timer_inner)
{
Ecore_Timer *timer;
int times = 0;
timer = ecore_timer_add(1.0, _timer1, &times);
fail_if(timer == NULL);
/* BEGIN: outer mainloop */
ecore_main_loop_begin();
/*END: outer mainloop */
fail_if(times != 1);
}
EFL_END_TEST
#endif
static Eina_Bool
_fd_handler_cb(void *data, Ecore_Fd_Handler *handler EINA_UNUSED)
{
/* FIXME: why setting val if it is overwritten just after and what is its purpose ??? */
Eina_Bool *val = data;
*val = EINA_TRUE;
ecore_main_loop_quit();
return EINA_FALSE;
}
EFL_START_TEST(ecore_test_ecore_main_loop_fd_handler)
{
Eina_Bool did = EINA_FALSE;
Ecore_Fd_Handler *fd_handler;
int comm[2];
int ret;
ret = pipe(comm);
fail_if(ret != 0);
fd_handler = ecore_main_fd_handler_add
(comm[0], ECORE_FD_READ, _fd_handler_cb, &did, NULL, NULL);
fail_if(fd_handler == NULL);
ret = write(comm[1], &did, 1);
fail_if(ret != 1);
ecore_main_loop_begin();
close(comm[0]);
close(comm[1]);
fail_if(did == EINA_FALSE);
}
EFL_END_TEST
EFL_START_TEST(ecore_test_ecore_main_loop_fd_handler_valid_flags)
{
Ecore_Fd_Handler *fd_handler;
int comm[2];
int ret;
ret = pipe(comm);
fail_if(ret != 0);
fd_handler = ecore_main_fd_handler_add
(comm[0], 0, _fd_handler_cb, NULL, NULL, NULL);
fail_if(fd_handler != NULL);
if (fd_handler)
ecore_main_fd_handler_del(fd_handler);
close(comm[0]);
close(comm[1]);
}
EFL_END_TEST
EFL_START_TEST(ecore_test_ecore_main_loop_fd_handler_activate_modify)
{
Eina_Bool did = EINA_FALSE;
Ecore_Fd_Handler *fd_handler;
int comm[2];
int ret;
ret = pipe(comm);
fail_if(ret != 0);
fd_handler = ecore_main_fd_handler_add
(comm[0], ECORE_FD_ERROR, _fd_handler_cb, &did, NULL, NULL);
fail_if(fd_handler == NULL);
ecore_main_fd_handler_active_set(fd_handler, ECORE_FD_READ);
ret = write(comm[1], "e", 1);
fail_if(ret != 1);
ecore_main_loop_begin();
close(comm[0]);
close(comm[1]);
fail_if(did != EINA_TRUE);
}
EFL_END_TEST
static Eina_Bool
_event_handler_cb(void *data, int type, void *event)
{
int *did = data;
int t1 = type;
int *e1 = event;
int t2 = ecore_event_current_type_get();
int *e2 = ecore_event_current_event_get();
if (t1 == t2)
(*did)++;
if (*e1 == *e2 && (ECORE_EVENT_CUSTOM_1 == *e1 || ECORE_EVENT_CUSTOM_2 == *e1))
(*did)++;
return EINA_TRUE;
}
static Eina_Bool
_event_handler_cb2(void *data, int type EINA_UNUSED, void *event EINA_UNUSED)
{
int *did = data;
(*did)++;
return EINA_TRUE;
}
static Eina_Bool
_event_handler_cb3(void *data, int type EINA_UNUSED, void *event EINA_UNUSED)
{
int *did = data;
(*did)++;
return EINA_TRUE;
}
static int *
_event_new(int id)
{
int *ev = malloc(sizeof(int));
switch(id)
{
case ECORE_EVENT_CUSTOM_1:
*ev = ECORE_EVENT_CUSTOM_1;
break;
case ECORE_EVENT_CUSTOM_2:
*ev = ECORE_EVENT_CUSTOM_2;
break;
default:
*ev = ECORE_EVENT_NONE;
}
return ev;
}
static void
_event_free(void *user_data, void *func_data)
{
int *did = user_data;
int *ev = func_data;
if (ECORE_EVENT_CUSTOM_1 == *ev || ECORE_EVENT_CUSTOM_2 == *ev)
(*did)++;
free(ev);
}
static void*
_filter_start(void *data)
{
int *fdid = data;
(*fdid)++;
return NULL;
}
static Eina_Bool
_filter(void *data, void *loop_data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Eina_Bool res = EINA_TRUE;
int *fdid = data;
int *ev = event;
if (NULL != event)
{
/* Ignore second event */
if (ECORE_EVENT_CUSTOM_2 == *ev)
{
res = EINA_FALSE;
}
}
(*fdid)++;
return res;
}
static void
_filter_end(void *user_data, void *func_data EINA_UNUSED)
{
int *fdid = user_data;
(*fdid)++;
}
EFL_START_TEST(ecore_test_ecore_main_loop_event)
{
Ecore_Event_Handler *handler, *handler2, *handler3;
Ecore_Event_Filter *filter_handler;
Ecore_Event *event;
Ecore_Event *event2;
int type, type2;
int *ev = NULL;
int did = 0;
int filter = 0;
/* Create 2 new event types */
type = ecore_event_type_new();
fail_if(type < 1);
type2 = ecore_event_type_new();
fail_if(type < 1);
/* Add handler for new type of event */
handler = ecore_event_handler_add(type, _event_handler_cb, &did);
fail_if(handler == NULL);
/* Add another handler for event which will be deleted in next step */
handler2 = ecore_event_handler_add(type, _event_handler_cb2, &did);
fail_if(handler2 == NULL);
/* Add handler for event which will be filtered */
handler3 = ecore_event_handler_add(type2, _event_handler_cb3, &did);
fail_if(handler3 == NULL);
/* Add filtering mechanism */
filter_handler = ecore_event_filter_add(_filter_start, _filter, _filter_end, &filter);
/* Add into main loop three events: one to process, one to filter, one to quit */
ev = _event_new(ECORE_EVENT_CUSTOM_1);
event = ecore_event_add(type, ev, _event_free, &did);
fail_if(event == NULL);
ev = _event_new(ECORE_EVENT_CUSTOM_2);
event = ecore_event_add(type2, ev, _event_free, &did);
fail_if(event == NULL);
event = ecore_event_add(ECORE_EVENT_SIGNAL_EXIT, NULL, NULL, NULL);
fail_if(event == NULL);
/* Add one more events: to check a type */
event2 = ecore_event_add(ECORE_EVENT_NONE, NULL, NULL, NULL);
fail_if(event2 != NULL);
ecore_main_loop_begin();
/*
Check internal fail cases:
event_cbx - 3 increments (4th should be ignored)
free - 2 increments
*/
fail_if(did != 3 + 2); // 5
/*
Check filter procedures calls:
start - 1 call
filter - 3 calls
end - 1 call
*/
fail_if(filter != 1 + 3 + 1); // 5
/* New loop but with new data and without filter and one callback procedure */
int did2 = 0;
filter = 0;
int *old = ecore_event_handler_data_set(handler, &did2);
int *new = ecore_event_handler_data_get(handler);
ecore_event_handler_del(handler2);
ecore_event_filter_del(filter_handler);
fail_if(*old != did);
fail_if(*new != did2);
ev = _event_new(ECORE_EVENT_CUSTOM_1);
event = ecore_event_add(type, ev, _event_free, &did2);
fail_if(event == NULL);
event = ecore_event_add(ECORE_EVENT_SIGNAL_EXIT, NULL, NULL, NULL);
fail_if(event == NULL);
ecore_main_loop_begin();
/*
Check internal fail cases:
event_cb - 2 increments in first callback (another one was deleted)
free - 1 increment
*/
fail_if(did2 != 2 + 1); // 3
/* Filter counter shouldn't change */
fail_if(filter != 0); // 0
}
EFL_END_TEST
#if 0
static int _log_dom;
#define INF(...) EINA_LOG_DOM_INFO(_log_dom, __VA_ARGS__)
static Eina_Bool
_timer_quit_recursive(void *data EINA_UNUSED)
{
INF(" _timer_quit_recursive: begin");
ecore_main_loop_quit(); /* quits inner main loop */
INF(" _timer_quit_recursive: end");
return EINA_FALSE;
}
static Eina_Bool
_event_recursive_cb(void *data EINA_UNUSED, int type EINA_UNUSED, void *event EINA_UNUSED)
{
static int guard = 0;
/* If we enter this callback more than once, it's wrong! */
fail_if(guard != 0);
guard++;
INF(" event_recursive_cb: begin");
ecore_timer_add(1.0, _timer_quit_recursive, NULL);
INF(" add 1.0s timer (once) to trigger _timer_quit_recursive");
INF(" inner main loop begin (recurse)");
ecore_main_loop_begin();
INF(" inner main loop end (recurse)");
ecore_main_loop_quit(); /* quits outer main loop */
INF(" guard = %d", guard);
INF(" event_recursive_cb: end");
return EINA_FALSE;
}
EFL_START_TEST(ecore_test_ecore_main_loop_event_recursive)
{
/* This test tests if the event handlers are really called only once when
* recursive main loops are used and any number of events may have occurred
* between the beginning and the end of recursive main loop.
*/
Ecore_Event *e;
int type;
_log_dom = eina_log_domain_register("test", EINA_COLOR_CYAN);
INF("main: begin");
type = ecore_event_type_new();
ecore_event_handler_add(type, _event_recursive_cb, NULL);
e = ecore_event_add(type, NULL, NULL, NULL);
INF(" add event to trigger cb1: event=%p", e);
INF(" main loop begin");
ecore_main_loop_begin();
INF(" main loop end");
INF("main: end");
}
EFL_END_TEST
#endif
EFL_START_TEST(ecore_test_ecore_app)
{
int argc_in = 2;
const char *argv_in[] = {"arg_str_1", "arg_str2"};
int argc_out = 0;
char **argv_out = NULL;
ecore_app_args_set(argc_in, argv_in);
ecore_app_args_get(&argc_out, &argv_out);
fail_if(argc_in != argc_out);
int i;
for (i = 0; i < argc_out; i++)
{
fail_if( 0 != strcmp(argv_in[i], argv_out[i]) );
}
}
EFL_END_TEST
void ecore_test_ecore(TCase *tc)
{
tcase_add_test(tc, ecore_test_ecore_init);
tcase_add_test(tc, ecore_test_ecore_main_loop);
tcase_add_test(tc, ecore_test_ecore_main_loop_fd_handler);
tcase_add_test(tc, ecore_test_ecore_main_loop_fd_handler_valid_flags);
tcase_add_test(tc, ecore_test_ecore_main_loop_fd_handler_activate_modify);
tcase_add_test(tc, ecore_test_ecore_main_loop_event);
#if 0
tcase_add_test(tc, ecore_test_ecore_main_loop_timer_inner);
tcase_add_test(tc, ecore_test_ecore_main_loop_event_recursive);
#endif
tcase_add_test(tc, ecore_test_ecore_app);
}