eina - redo a lot of the static inlines to have better instr cache usage

this moves a lot of logic that is rare away from the linear/flat asm
path of code so we et fewer l1 cache misses when executing chuncks of
our code. this also reduces the code size and takes some funcs like in
eina_inline_lock_posix.x and makes them real functions to reduce code
size thus better l1 cache usage - only for new/free of locks.
spinlocks, semaphores etc. as these will have no advantage being
inlined but simply bloat out code size instead.

overall this actually reduces efl lib binary sizes 0.4%, so that's a
good sign.

this passes make check and i think i got it right... let me know if i
didn't. i'm also not sure i should just keep the static inlines and
not make the formerly static inline funcs full EAPI ones now... good q.
This commit is contained in:
Carsten Haitzler 2016-09-19 08:58:56 +09:00
parent 3efdd5df54
commit 41c332b726
15 changed files with 643 additions and 605 deletions

View File

@ -136,6 +136,7 @@ lib/eina/eina_inlist.c \
lib/eina/eina_iterator.c \
lib/eina/eina_lalloc.c \
lib/eina/eina_list.c \
lib/eina/eina_lock.c \
lib/eina/eina_log.c \
lib/eina/eina_magic.c \
lib/eina/eina_main.c \

View File

@ -44,29 +44,26 @@ EAPI Eina_Bool eina_array_grow(Eina_Array *array);
static inline Eina_Bool
eina_array_push(Eina_Array *array, const void *data)
{
if (!data) return EINA_FALSE;
if (data)
{
if (EINA_UNLIKELY((array->count + 1) > array->total)) goto do_grow;
do_grow_back:
if (EINA_UNLIKELY((array->count + 1) > array->total))
if (!eina_array_grow(array))
return EINA_FALSE;
array->data[array->count++] = (void*) data;
array->data[array->count++] = (void*) data;
return EINA_TRUE;
return EINA_TRUE;
}
return EINA_FALSE;
do_grow:
if (!eina_array_grow(array)) return EINA_FALSE;
goto do_grow_back;
}
static inline void *
eina_array_pop(Eina_Array *array)
{
void *ret = NULL;
if (array->count <= 0)
goto on_empty;
ret = array->data[--array->count];
on_empty:
return ret;
if (array->count > 0) return array->data[--array->count];
return NULL;
}
static inline void *
@ -104,12 +101,11 @@ eina_array_foreach(Eina_Array *array, Eina_Each_Cb cb, void *fdata)
Eina_Bool ret = EINA_TRUE;
EINA_ARRAY_ITER_NEXT(array, i, data, iterator)
if (cb(array, data, fdata) != EINA_TRUE)
{
ret = EINA_FALSE;
break;
}
{
if (cb(array, data, fdata) == EINA_TRUE) continue;
ret = EINA_FALSE;
break;
}
return ret;
}

View File

@ -33,11 +33,13 @@ eina_hash_djb2(const char *key, int len)
unsigned int hash_num = 5381 ^ eina_seed;
const unsigned char *ptr;
if (!key) return 0;
for (ptr = (unsigned char *)key; len; ptr++, len--)
hash_num = ((hash_num << 5) + hash_num) ^ *ptr; /* hash * 33 ^ c */
return (int)hash_num;
if (key)
{
for (ptr = (unsigned char *)key; len; ptr++, len--)
hash_num = ((hash_num << 5) + hash_num) ^ *ptr; /* hash * 33 ^ c */
return (int)hash_num;
}
return 0;
}
static inline int
@ -47,14 +49,14 @@ eina_hash_djb2_len(const char *key, int *plen)
int len = 0;
const unsigned char *ptr;
if (!key) return 0;
for (ptr = (unsigned char *)key; *ptr; ptr++, len++)
hash_num = ((hash_num << 5) + hash_num) ^ *ptr; /* hash * 33 ^ c */
*plen = len + 1;
return (int)hash_num;
if (key)
{
for (ptr = (unsigned char *)key; *ptr; ptr++, len++)
hash_num = ((hash_num << 5) + hash_num) ^ *ptr; /* hash * 33 ^ c */
*plen = len + 1;
return (int)hash_num;
}
return 0;
}
static inline int

View File

@ -24,11 +24,12 @@ eina_inlist_first(const Eina_Inlist *list)
{
Eina_Inlist *l;
if (!list) return NULL;
for (l = (Eina_Inlist*)list; l->prev; l = l->prev);
return l;
if (list)
{
for (l = (Eina_Inlist*)list; l->prev; l = l->prev);
return l;
}
return NULL;
}
static inline Eina_Inlist *
@ -36,11 +37,12 @@ eina_inlist_last(const Eina_Inlist *list)
{
Eina_Inlist *l;
if (!list) return NULL;
for (l = (Eina_Inlist*)list; l->next; l = l->next);
return l;
if (list)
{
for (l = (Eina_Inlist*)list; l->next; l = l->next);
return l;
}
return NULL;
}
#endif /* EINA_INLIST_INLINE_H_ */

View File

@ -22,53 +22,55 @@
static inline Eina_List *
eina_list_last(const Eina_List *list)
{
if (!list) return NULL;
return list->accounting->last;
if (list) return list->accounting->last;
return NULL;
}
static inline Eina_List *
eina_list_next(const Eina_List *list)
{
if (!list) return NULL;
return list->next;
if (list) return list->next;
return NULL;
}
static inline Eina_List *
eina_list_prev(const Eina_List *list)
{
if (!list) return NULL;
return list->prev;
if (list) return list->prev;
return NULL;
}
static inline void *
eina_list_data_get(const Eina_List *list)
{
if (!list) return NULL;
return list->data;
if (list) return list->data;
return NULL;
}
static inline void *
eina_list_data_set(Eina_List *list, const void *data)
{
void *tmp;
if (!list) return NULL;
tmp = list->data;
list->data = (void*) data;
return tmp;
if (list)
{
void *tmp = list->data;
list->data = (void *) data;
return tmp;
}
return NULL;
}
static inline unsigned int
eina_list_count(const Eina_List *list)
{
if (!list) return 0;
return list->accounting->count;
if (list) return list->accounting->count;
return 0;
}
static inline void *
eina_list_last_data_get(const Eina_List *list)
{
if (!list) return NULL;
return eina_list_data_get(eina_list_last(list));
if (list) return eina_list_data_get(eina_list_last(list));
return NULL;
}
#endif /* EINA_LIST_INLINE_H_ */

View File

@ -9,27 +9,6 @@ struct _Eina_Barrier
Eina_Condition cond; /**< The condition variable for the barrier */
};
static inline Eina_Bool
eina_barrier_new(Eina_Barrier *barrier, int needed)
{
barrier->needed = needed;
barrier->called = 0;
if (!eina_lock_new(&(barrier->cond_lock)))
return EINA_FALSE;
if (!eina_condition_new(&(barrier->cond), &(barrier->cond_lock)))
return EINA_FALSE;
return EINA_TRUE;
}
static inline void
eina_barrier_free(Eina_Barrier *barrier)
{
eina_condition_free(&(barrier->cond));
eina_lock_free(&(barrier->cond_lock));
barrier->needed = 0;
barrier->called = 0;
}
static inline Eina_Bool
eina_barrier_wait(Eina_Barrier *barrier)
{

View File

@ -63,21 +63,16 @@
typedef void (*Eina_Lock_Bt_Func) ();
#include "eina_inlist.h"
#define EINA_LOCK_ABORT_DEBUG(err, fn, ptr) do { \
printf("EINA ERROR: '%s' on %s %p\n", strerror(err), #fn, ptr); \
abort(); \
} while (0)
#define EINA_LOCK_DEADLOCK_DEBUG(fn, ptr) do { \
printf("EINA ERROR: DEADLOCK on %s %p\n", #fn, ptr); \
abort(); \
} while (0)
#else
#define EINA_LOCK_ABORT_DEBUG(err, fn, ptr) \
printf("EINA ERROR: '%s' on %s %p\n", strerror(err), #fn, ptr)
#define EINA_LOCK_DEADLOCK_DEBUG(fn, ptr) \
printf("EINA ERROR: DEADLOCK on %s %p\n", #fn, ptr)
#endif
EAPI void _eina_lock_debug_abort(int err, const char *fn, const volatile void *ptr);
EAPI void _eina_lock_debug_deadlock(const char *fn, const volatile void *ptr);
#define EINA_LOCK_ABORT_DEBUG(err, fn, ptr) \
_eina_lock_debug_abort(err, #fn, ptr)
#define EINA_LOCK_DEADLOCK_DEBUG(fn, ptr) \
_eina_lock_debug_deadlock(#fn, ptr)
/* For cond_timedwait */
#include <time.h>
#include <sys/time.h>
@ -103,6 +98,8 @@ typedef semaphore_t Eina_Semaphore;
typedef sem_t Eina_Semaphore;
#endif
EAPI void eina_lock_debug(const Eina_Lock *mutex);
/** @privatesection @{ */
struct _Eina_Lock
{
@ -145,91 +142,34 @@ EAPI extern pthread_mutex_t _eina_tracking_lock;
EAPI extern Eina_Inlist *_eina_tracking;
#endif
static inline void
eina_lock_debug(const Eina_Lock *mutex)
{
#ifdef EINA_HAVE_DEBUG_THREADS
printf("lock %p, locked: %i, by %ti\n",
mutex, (int)mutex->locked, (ptrdiff_t)mutex->lock_thread_id);
backtrace_symbols_fd((void **)mutex->lock_bt, mutex->lock_bt_num, 1);
#else
(void) mutex;
#endif
}
EAPI Eina_Bool _eina_lock_new(Eina_Lock *mutex, Eina_Bool recursive);
EAPI void _eina_lock_free(Eina_Lock *mutex);
EAPI Eina_Bool _eina_condition_new(Eina_Condition *cond, Eina_Lock *mutex);
EAPI void _eina_condition_free(Eina_Condition *cond);
EAPI Eina_Bool _eina_rwlock_new(Eina_RWLock *mutex);
EAPI void _eina_rwlock_free(Eina_RWLock *mutex);
EAPI Eina_Bool _eina_spinlock_new(Eina_Spinlock *spinlock);
EAPI void _eina_spinlock_free(Eina_Spinlock *spinlock);
EAPI Eina_Bool _eina_semaphore_new(Eina_Semaphore *sem, int count_init);
EAPI Eina_Bool _eina_semaphore_free(Eina_Semaphore *sem);
static inline Eina_Bool
eina_lock_new(Eina_Lock *mutex)
{
pthread_mutexattr_t attr;
Eina_Bool ok = EINA_FALSE;
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
if (pthread_mutexattr_init(&attr) != 0)
return EINA_FALSE;
/* NOTE: PTHREAD_MUTEX_RECURSIVE is not allowed at all, you will break on/off
feature for sure with that change. */
#ifdef EINA_HAVE_DEBUG_THREADS
if (pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK) != 0)
goto fail_release;
memset(mutex, 0, sizeof(Eina_Lock));
#endif
if (pthread_mutex_init(&(mutex->mutex), &attr) != 0)
goto fail_release;
ok = EINA_TRUE;
fail_release:
pthread_mutexattr_destroy(&attr);
return ok;
return _eina_lock_new(mutex, EINA_FALSE);
}
static inline Eina_Bool
eina_lock_recursive_new(Eina_Lock *mutex)
{
pthread_mutexattr_t attr;
Eina_Bool ok = EINA_FALSE;
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
if (pthread_mutexattr_init(&attr) != 0)
return EINA_FALSE;
if (pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE) != 0)
goto fail_release;
#ifdef EINA_HAVE_DEBUG_THREADS
if (pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK) != 0)
goto fail_release;
memset(mutex, 0, sizeof(Eina_Lock));
#endif
if (pthread_mutex_init(&(mutex->mutex), &attr) != 0)
goto fail_release;
ok = EINA_TRUE;
fail_release:
pthread_mutexattr_destroy(&attr);
return ok;
return _eina_lock_new(mutex, EINA_TRUE);
}
static inline void
eina_lock_free(Eina_Lock *mutex)
{
int ok;
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
ok = pthread_mutex_destroy(&(mutex->mutex));
if (ok != 0) EINA_LOCK_ABORT_DEBUG(ok, mutex_destroy, mutex);
#ifdef EINA_HAVE_DEBUG_THREADS
memset(mutex, 0, sizeof(Eina_Lock));
#endif
_eina_lock_free(mutex);
}
static inline Eina_Lock_Result
@ -278,9 +218,8 @@ eina_lock_take(Eina_Lock *mutex)
if (ok == 0) ret = EINA_LOCK_SUCCEED;
else if (ok == EDEADLK)
{
printf("EINA ERROR: DEADLOCK on lock %p\n", mutex);
eina_lock_debug(mutex);
ret = EINA_LOCK_DEADLOCK; // magic
ret = EINA_LOCK_DEADLOCK;
#ifdef EINA_HAVE_DEBUG_THREADS
if (_eina_threads_debug) abort();
#endif
@ -326,8 +265,8 @@ eina_lock_take_try(Eina_Lock *mutex)
if (ok == 0) ret = EINA_LOCK_SUCCEED;
else if (ok == EDEADLK)
{
printf("EINA ERROR: DEADLOCK on trylock %p\n", mutex);
ret = EINA_LOCK_DEADLOCK; // magic
eina_lock_debug(mutex);
ret = EINA_LOCK_DEADLOCK;
}
else if (ok != EBUSY) EINA_LOCK_ABORT_DEBUG(ok, trylock, mutex);
#ifdef EINA_HAVE_DEBUG_THREADS
@ -383,64 +322,13 @@ eina_lock_release(Eina_Lock *mutex)
static inline Eina_Bool
eina_condition_new(Eina_Condition *cond, Eina_Lock *mutex)
{
pthread_condattr_t attr;
int ok;
#ifdef EINA_HAVE_DEBUG_THREADS
assert(mutex != NULL);
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
memset(cond, 0, sizeof (Eina_Condition));
#endif
cond->lock = mutex;
pthread_condattr_init(&attr);
/* OSX doesn't provide clockid_t or clock_gettime. */
#if defined(__clockid_t_defined)
cond->clkid = (clockid_t) 0;
/* We try here to chose the best clock for cond_timedwait */
# if defined(CLOCK_MONOTONIC_RAW)
if (!pthread_condattr_setclock(&attr, CLOCK_MONOTONIC_RAW))
cond->clkid = CLOCK_MONOTONIC_RAW;
# endif
# if defined(CLOCK_MONOTONIC)
if (!cond->clkid && !pthread_condattr_setclock(&attr, CLOCK_MONOTONIC))
cond->clkid = CLOCK_MONOTONIC;
# endif
# if defined(CLOCK_REALTIME)
if (!cond->clkid && !pthread_condattr_setclock(&attr, CLOCK_REALTIME))
cond->clkid = CLOCK_REALTIME;
# endif
#endif
ok = pthread_cond_init(&cond->condition, &attr);
if (ok != 0)
{
pthread_condattr_destroy(&attr);
#ifdef EINA_HAVE_DEBUG_THREADS
if (ok == EBUSY)
printf("eina_condition_new on already initialized Eina_Condition\n");
#endif
return EINA_FALSE;
}
pthread_condattr_destroy(&attr);
return EINA_TRUE;
return _eina_condition_new(cond, mutex);
}
static inline void
eina_condition_free(Eina_Condition *cond)
{
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
pthread_cond_destroy(&(cond->condition));
#ifdef EINA_HAVE_DEBUG_THREADS
memset(cond, 0, sizeof (Eina_Condition));
#endif
_eina_condition_free(cond);
}
static inline Eina_Bool
@ -455,7 +343,7 @@ eina_condition_wait(Eina_Condition *cond)
pthread_mutex_lock(&_eina_tracking_lock);
_eina_tracking = eina_inlist_remove(_eina_tracking,
EINA_INLIST_GET(cond->lock));
EINA_INLIST_GET(cond->lock));
pthread_mutex_unlock(&_eina_tracking_lock);
#endif
@ -467,7 +355,7 @@ eina_condition_wait(Eina_Condition *cond)
#ifdef EINA_HAVE_DEBUG_THREADS
pthread_mutex_lock(&_eina_tracking_lock);
_eina_tracking = eina_inlist_append(_eina_tracking,
EINA_INLIST_GET(cond->lock));
EINA_INLIST_GET(cond->lock));
pthread_mutex_unlock(&_eina_tracking_lock);
#endif
@ -483,69 +371,60 @@ eina_condition_timedwait(Eina_Condition *cond, double t)
int err;
Eina_Bool r = EINA_FALSE;
if (t < 0)
if (t >= 0.0)
{
errno = EINVAL;
return EINA_FALSE;
}
#if defined(__clockid_t_defined)
if (cond->clkid)
{
if (clock_gettime(cond->clkid, &ts) != 0)
return EINA_FALSE;
}
else
if (cond->clkid)
{
if (clock_gettime(cond->clkid, &ts) != 0) return EINA_FALSE;
}
else
#endif
{
/* Obsolete for Linux.
* TODO: use pthread_cond_timedwait_relative_np for OSX. */
struct timeval tv;
if (gettimeofday(&tv, NULL) != 0)
return EINA_FALSE;
ts.tv_sec = tv.tv_sec;
ts.tv_nsec = tv.tv_usec * 1000L;
}
{
/* Obsolete for Linux.
* TODO: use pthread_cond_timedwait_relative_np for OSX. */
struct timeval tv;
if (gettimeofday(&tv, NULL) != 0) return EINA_FALSE;
ts.tv_sec = tv.tv_sec;
ts.tv_nsec = tv.tv_usec * 1000L;
}
#ifdef EINA_HAVE_DEBUG_THREADS
assert(_eina_threads_activated);
assert(cond->lock != NULL);
assert(_eina_threads_activated);
assert(cond->lock != NULL);
pthread_mutex_lock(&_eina_tracking_lock);
_eina_tracking = eina_inlist_remove(_eina_tracking,
EINA_INLIST_GET(cond->lock));
pthread_mutex_unlock(&_eina_tracking_lock);
pthread_mutex_lock(&_eina_tracking_lock);
_eina_tracking = eina_inlist_remove(_eina_tracking,
EINA_INLIST_GET(cond->lock));
pthread_mutex_unlock(&_eina_tracking_lock);
#endif
sec = (time_t)t;
nsec = (t - (double) sec) * 1000000000L;
ts.tv_sec += sec;
ts.tv_nsec += nsec;
if (ts.tv_nsec > 1000000000L)
{
ts.tv_sec++;
ts.tv_nsec -= 1000000000L;
}
sec = (time_t)t;
nsec = (t - (double)sec) * 1000000000L;
ts.tv_sec += sec;
ts.tv_nsec += nsec;
if (ts.tv_nsec > 1000000000L)
{
ts.tv_sec++;
ts.tv_nsec -= 1000000000L;
}
err = pthread_cond_timedwait(&(cond->condition),
&(cond->lock->mutex),
&ts);
if (err == 0)
r = EINA_TRUE;
else if (err == EPERM)
eina_error_set(EPERM);
else if (err == ETIMEDOUT)
eina_error_set(ETIMEDOUT);
else EINA_LOCK_ABORT_DEBUG(err, cond_timedwait, cond);
err = pthread_cond_timedwait(&(cond->condition),
&(cond->lock->mutex),
&ts);
if (err == 0) r = EINA_TRUE;
else if (err == EPERM) eina_error_set(EPERM);
else if (err == ETIMEDOUT) eina_error_set(ETIMEDOUT);
else EINA_LOCK_ABORT_DEBUG(err, cond_timedwait, cond);
#ifdef EINA_HAVE_DEBUG_THREADS
pthread_mutex_lock(&_eina_tracking_lock);
_eina_tracking = eina_inlist_append(_eina_tracking,
EINA_INLIST_GET(cond->lock));
pthread_mutex_unlock(&_eina_tracking_lock);
pthread_mutex_lock(&_eina_tracking_lock);
_eina_tracking = eina_inlist_append(_eina_tracking,
EINA_INLIST_GET(cond->lock));
pthread_mutex_unlock(&_eina_tracking_lock);
#endif
}
errno = EINVAL;
return r;
}
@ -584,29 +463,13 @@ eina_condition_signal(Eina_Condition *cond)
static inline Eina_Bool
eina_rwlock_new(Eina_RWLock *mutex)
{
int ok;
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
ok = pthread_rwlock_init(&(mutex->mutex), NULL);
if (ok == 0) return EINA_TRUE;
else if (ok == EAGAIN || ok == ENOMEM) return EINA_FALSE;
else EINA_LOCK_ABORT_DEBUG(ok, rwlock_init, mutex);
return EINA_FALSE;
return _eina_rwlock_new(mutex);
}
static inline void
eina_rwlock_free(Eina_RWLock *mutex)
{
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
pthread_rwlock_destroy(&(mutex->mutex));
_eina_rwlock_free(mutex);
}
static inline Eina_Lock_Result
@ -680,9 +543,8 @@ eina_rwlock_release(Eina_RWLock *mutex)
static inline Eina_Bool
eina_tls_cb_new(Eina_TLS *key, Eina_TLS_Delete_Cb delete_cb)
{
if (pthread_key_create(key, delete_cb) != 0)
return EINA_FALSE;
return EINA_TRUE;
if (pthread_key_create(key, delete_cb) == 0) return EINA_TRUE;
return EINA_FALSE;
}
static inline Eina_Bool
@ -691,24 +553,23 @@ eina_tls_new(Eina_TLS *key)
return eina_tls_cb_new(key, NULL);
}
static inline void
static inline void
eina_tls_free(Eina_TLS key)
{
pthread_key_delete(key);
}
static inline void *
static inline void *
eina_tls_get(Eina_TLS key)
{
return pthread_getspecific(key);
}
static inline Eina_Bool
static inline Eina_Bool
eina_tls_set(Eina_TLS key, const void *data)
{
if (pthread_setspecific(key, data) != 0)
return EINA_FALSE;
return EINA_TRUE;
if (pthread_setspecific(key, data) == 0) return EINA_TRUE;
return EINA_FALSE;
}
@ -720,31 +581,12 @@ struct _Eina_Barrier
pthread_barrier_t barrier;
};
static inline Eina_Bool
eina_barrier_new(Eina_Barrier *barrier, int needed)
{
int ok;
ok = pthread_barrier_init(&(barrier->barrier), NULL, needed);
if (ok == 0) return EINA_TRUE;
else if (ok == EAGAIN || ok == ENOMEM) return EINA_FALSE;
else EINA_LOCK_ABORT_DEBUG(ok, barrier_init, barrier);
return EINA_FALSE;
}
static inline void
eina_barrier_free(Eina_Barrier *barrier)
{
int ok;
ok = pthread_barrier_destroy(&(barrier->barrier));
if (ok != 0) EINA_LOCK_ABORT_DEBUG(ok, barrier_destroy, barrier);
}
static inline Eina_Bool
eina_barrier_wait(Eina_Barrier *barrier)
{
int ok;
ok = pthread_barrier_wait(&(barrier->barrier));
if (ok != 0) EINA_LOCK_ABORT_DEBUG(ok, barrier_wait, barrier);
int ok = pthread_barrier_wait(&(barrier->barrier));
if (ok == 0) return EINA_TRUE;
else EINA_LOCK_ABORT_DEBUG(ok, barrier_wait, barrier);
return EINA_TRUE;
}
@ -752,24 +594,32 @@ eina_barrier_wait(Eina_Barrier *barrier)
#include "eina_inline_lock_barrier.x"
#endif
EAPI Eina_Bool _eina_barrier_new(Eina_Barrier *barrier, int needed);
EAPI void _eina_barrier_free(Eina_Barrier *barrier);
static inline Eina_Bool
eina_barrier_new(Eina_Barrier *barrier, int needed)
{
return _eina_barrier_new(barrier, needed);
}
static inline void
eina_barrier_free(Eina_Barrier *barrier)
{
_eina_barrier_free(barrier);
}
static inline Eina_Bool
eina_spinlock_new(Eina_Spinlock *spinlock)
{
#if defined(EINA_HAVE_POSIX_SPINLOCK)
int ok;
ok = pthread_spin_init(spinlock, PTHREAD_PROCESS_PRIVATE);
if (ok == 0) return EINA_TRUE;
else if (ok == EAGAIN || ok == ENOMEM) return EINA_FALSE;
else EINA_LOCK_ABORT_DEBUG(ok, spin_init, spinlock);
return EINA_FALSE;
#elif defined(EINA_HAVE_OSX_SPINLOCK)
/* OSSpinLock is an integer type. The convention is that unlocked is
* zero, and locked is nonzero. */
*spinlock = 0;
return EINA_TRUE;
#else
return eina_lock_new(spinlock);
#endif
return _eina_spinlock_new(spinlock);
}
static inline void
eina_spinlock_free(Eina_Spinlock *spinlock)
{
_eina_spinlock_free(spinlock);
}
static inline Eina_Lock_Result
@ -778,12 +628,11 @@ eina_spinlock_take(Eina_Spinlock *spinlock)
#if defined(EINA_HAVE_POSIX_SPINLOCK)
int t;
while (EINA_TRUE)
for (;;)
{
t = pthread_spin_trylock(spinlock);
t = pthread_spin_lock(spinlock);
if (t == 0) break;
else if (t == EBUSY) sched_yield();
else EINA_LOCK_ABORT_DEBUG(t, spin_trylock, spinlock);
else EINA_LOCK_ABORT_DEBUG(t, spin_lock, spinlock);
}
return EINA_LOCK_SUCCEED;
@ -802,9 +651,7 @@ static inline Eina_Lock_Result
eina_spinlock_take_try(Eina_Spinlock *spinlock)
{
#if defined(EINA_HAVE_POSIX_SPINLOCK)
int t;
t = pthread_spin_trylock(spinlock);
int t = pthread_spin_trylock(spinlock);
if (t == 0) return EINA_LOCK_SUCCEED;
else if (t == EBUSY) return EINA_LOCK_FAIL;
else EINA_LOCK_ABORT_DEBUG(t, spin_trylock, spinlock);
@ -823,8 +670,7 @@ static inline Eina_Lock_Result
eina_spinlock_release(Eina_Spinlock *spinlock)
{
#if defined(EINA_HAVE_POSIX_SPINLOCK)
int ok;
ok = pthread_spin_unlock(spinlock);
int ok = pthread_spin_unlock(spinlock);
if (ok == 0) return EINA_LOCK_SUCCEED;
else if (ok == EPERM) return EINA_LOCK_FAIL;
else EINA_LOCK_ABORT_DEBUG(ok, spin_unlock, spinlock);
@ -839,96 +685,52 @@ eina_spinlock_release(Eina_Spinlock *spinlock)
#endif
}
static inline void
eina_spinlock_free(Eina_Spinlock *spinlock)
{
#if defined(EINA_HAVE_POSIX_SPINLOCK)
int ok;
ok = pthread_spin_destroy(spinlock);
if (ok != 0) EINA_LOCK_ABORT_DEBUG(ok, spin_destroy, spinlock);
#elif defined(EINA_HAVE_OSX_SPINLOCK)
/* Not applicable */
(void) spinlock;
#else
eina_lock_free(spinlock);
#endif
}
static inline Eina_Bool
eina_semaphore_new(Eina_Semaphore *sem, int count_init)
{
if (!sem || (count_init < 0))
return EINA_FALSE;
#if defined(EINA_HAVE_OSX_SEMAPHORE)
kern_return_t kr;
kr = semaphore_create(mach_task_self(), sem, SYNC_POLICY_FIFO, count_init);
return (kr == KERN_SUCCESS) ? EINA_TRUE : EINA_FALSE;
#else
return (sem_init(sem, 0, count_init) == 0) ? EINA_TRUE : EINA_FALSE;
#endif
return _eina_semaphore_new(sem, count_init);
}
static inline Eina_Bool
eina_semaphore_free(Eina_Semaphore *sem)
{
if (!sem)
return EINA_FALSE;
#if defined(EINA_HAVE_OSX_SEMAPHORE)
return (semaphore_destroy(*sem, mach_task_self()) == KERN_SUCCESS)
? EINA_TRUE : EINA_FALSE;
#else
return (sem_destroy(sem) == 0) ? EINA_TRUE : EINA_FALSE;
#endif
return _eina_semaphore_free(sem);
}
static inline Eina_Bool
eina_semaphore_lock(Eina_Semaphore *sem)
{
Eina_Bool ok = EINA_FALSE;
if (!sem)
return EINA_FALSE;
for (;;)
if (sem)
{
if (
for (;;)
{
if (
#if defined(EINA_HAVE_OSX_SEMAPHORE)
semaphore_wait(*sem)
semaphore_wait(*sem)
#else
sem_wait(sem)
sem_wait(sem)
#endif
== 0)
{
ok = EINA_TRUE;
break;
}
else
{
if (errno != EINTR)
{
if (errno == EDEADLK)
EINA_LOCK_DEADLOCK_DEBUG(sem_wait, sem);
break;
}
== 0)
return EINA_TRUE;
else if (errno != EINTR) goto err;
}
}
return ok;
return EINA_FALSE;
err:
if (errno == EDEADLK) EINA_LOCK_DEADLOCK_DEBUG(sem_wait, sem);
return EINA_FALSE;
}
static inline Eina_Bool
eina_semaphore_release(Eina_Semaphore *sem, int count_release EINA_UNUSED)
{
if (!sem)
return EINA_FALSE;
if (sem)
#if defined(EINA_HAVE_OSX_SEMAPHORE)
return (semaphore_signal(*sem) == KERN_SUCCESS) ? EINA_TRUE : EINA_FALSE;
return (semaphore_signal(*sem) == KERN_SUCCESS) ? EINA_TRUE : EINA_FALSE;
#else
return (sem_post(sem) == 0) ? EINA_TRUE : EINA_FALSE;
return (sem_post(sem) == 0) ? EINA_TRUE : EINA_FALSE;
#endif
return EINA_FALSE;
}
#undef _XOPEN_SOURCE

View File

@ -84,9 +84,9 @@ static inline Eina_Bool
eina_log_domain_level_check(int domain, int level)
{
int dom_level = eina_log_domain_registered_level_get(domain);
if (EINA_UNLIKELY(dom_level == EINA_LOG_LEVEL_UNKNOWN))
return EINA_FALSE;
return dom_level >= level;
if (EINA_LIKELY(dom_level != EINA_LOG_LEVEL_UNKNOWN))
return dom_level >= level;
return EINA_FALSE;
}
/**

View File

@ -94,16 +94,14 @@ static inline void *
eina_mempool_calloc(Eina_Mempool *mp, unsigned int size)
{
void *r = mp->backend.alloc(mp->backend_data, size);
if (!r) return NULL;
memset(r, 0, size);
if (r) memset(r, 0, size);
return r;
}
static inline void
eina_mempool_free(Eina_Mempool *mp, void *element)
{
if (!element) return ;
mp->backend.free(mp->backend_data, element);
if (element) mp->backend.free(mp->backend_data, element);
}
static inline unsigned int
@ -111,31 +109,33 @@ eina_mempool_alignof(unsigned int size)
{
unsigned int align;
if (EINA_UNLIKELY(size <= 2))
{
align = 2;
}
else if (EINA_UNLIKELY(size < 8))
{
align = 4;
}
else
#if __WORDSIZE == 32
if (size >= 8)
{
align = 8;
calc:
return ((size / align) + (size % align ? 1 : 0)) * align;
}
#else
if (EINA_UNLIKELY(size < 16))
{
align = 8;
}
else
#else // __WORDSIZE == 64
if (size >= 16)
{
align = 16;
calc:
return ((size / align) + (size % align ? 1 : 0)) * align;
}
else if (size >= 8)
{
align = 8;
goto calc;
}
#endif
return ((size / align) + (size % align ? 1 : 0)) * align;
else if (size >= 4)
{
align = 4;
goto calc;
}
align = 2;
goto calc;
}
#endif

View File

@ -34,13 +34,11 @@ eina_rbtree_inline_lookup(const Eina_Rbtree *root, const void *key, int length,
while (root)
{
result = cmp(root, key, length, (void*) data);
if (result == 0) return (Eina_Rbtree*) root;
root = root->son[result < 0 ? 0 : 1];
result = cmp(root, key, length, (void *)data);
if (result != 0) root = root->son[result < 0 ? 0 : 1];
else break;
}
return NULL;
return (Eina_Rbtree *)root;
}
/**

View File

@ -22,9 +22,8 @@
static inline Eina_Bool
eina_rectangle_is_valid(const Eina_Rectangle *r)
{
if (r->w <= 0 || r->h <= 0)
return EINA_FALSE;
return EINA_TRUE;
if ((r->w > 0) && (r->h > 0)) return EINA_TRUE;
return EINA_FALSE;
}
static inline int
@ -36,7 +35,7 @@ eina_spans_intersect(int c1, int l1, int c2, int l2)
static inline Eina_Bool
eina_rectangle_is_empty(const Eina_Rectangle *r)
{
return ((r->w < 1) || (r->h < 1)) ? EINA_TRUE : EINA_FALSE;
return !eina_rectangle_is_valid(r);
}
static inline void
@ -98,39 +97,34 @@ eina_rectangle_union(Eina_Rectangle *dst, const Eina_Rectangle *src)
static inline Eina_Bool
eina_rectangle_intersection(Eina_Rectangle *dst, const Eina_Rectangle *src)
{
if (!(eina_rectangle_is_valid(dst)) || !(eina_rectangle_is_valid(src)))
return EINA_FALSE;
if (eina_rectangle_is_valid(dst) && eina_rectangle_is_valid(src) &&
eina_rectangles_intersect(dst, src))
{
/* left */
if (dst->x < src->x)
{
dst->w += dst->x - src->x;
dst->x = src->x;
if (dst->w < 0) dst->w = 0;
}
/* right */
if ((dst->x + dst->w) > (src->x + src->w))
dst->w = src->x + src->w - dst->x;
/* top */
if (dst->y < src->y)
{
dst->h += dst->y - src->y;
dst->y = src->y;
if (dst->h < 0) dst->h = 0;
}
/* bottom */
if ((dst->y + dst->h) > (src->y + src->h))
dst->h = src->y + src->h - dst->y;
if (!(eina_rectangles_intersect(dst, src)))
return EINA_FALSE;
/* left */
if (dst->x < src->x)
{
dst->w += dst->x - src->x;
dst->x = src->x;
if (dst->w < 0)
dst->w = 0;
}
/* right */
if ((dst->x + dst->w) > (src->x + src->w))
dst->w = src->x + src->w - dst->x;
/* top */
if (dst->y < src->y)
{
dst->h += dst->y - src->y;
dst->y = src->y;
if (dst->h < 0)
dst->h = 0;
}
/* bottom */
if ((dst->y + dst->h) > (src->y + src->h))
dst->h = src->y + src->h - dst->y;
if (dst->w == 0 || dst->h == 0)
return EINA_FALSE;
return EINA_TRUE;
if ((dst->w == 0) || (dst->h == 0)) return EINA_FALSE;
return EINA_TRUE;
}
return EINA_FALSE;
}
static inline void
@ -218,57 +212,57 @@ eina_rectangle_height_cut(Eina_Rectangle *thiz, Eina_Rectangle *slice, Eina_Rect
static inline Eina_Bool
eina_rectangle_subtract(Eina_Rectangle *thiz, Eina_Rectangle *other, Eina_Rectangle out[4])
{
Eina_Rectangle intersection;
Eina_Rectangle leftover = EINA_RECTANGLE_INIT;
Eina_Rectangle tmp;
int cut = 0;
Eina_Rectangle intersection;
Eina_Rectangle leftover = EINA_RECTANGLE_INIT;
Eina_Rectangle tmp;
int cut = 0;
if (!eina_rectangle_is_valid(thiz))
return EINA_FALSE;
if (eina_rectangle_is_valid(thiz))
{
eina_rectangle_coords_from(&out[0], 0, 0, 0, 0);
eina_rectangle_coords_from(&out[1], 0, 0, 0, 0);
eina_rectangle_coords_from(&out[2], 0, 0, 0, 0);
eina_rectangle_coords_from(&out[3], 0, 0, 0, 0);
intersection = *thiz;
if (!eina_rectangle_intersection(&intersection, other))
{
out[0] = *thiz;
return EINA_TRUE;
}
eina_rectangle_coords_from(&out[0], 0, 0, 0, 0);
eina_rectangle_coords_from(&out[1], 0, 0, 0, 0);
eina_rectangle_coords_from(&out[2], 0, 0, 0, 0);
eina_rectangle_coords_from(&out[3], 0, 0, 0, 0);
intersection = *thiz;
if (!eina_rectangle_intersection(&intersection, other))
{
out[0] = *thiz;
return EINA_TRUE;
}
/* cut in height */
{
cut = thiz->h - (intersection.y - thiz->y);
if (cut > thiz->h) { cut = thiz->h; }
eina_rectangle_height_cut(thiz, &leftover, &out[0], cut);
}
/* cut in y */
tmp = leftover;
if (eina_rectangle_intersection(&tmp, &intersection))
{
cut = leftover.h - (eina_rectangle_max_y(&leftover) - eina_rectangle_max_y(&tmp));
if (cut > leftover.h) { cut = leftover.h; }
eina_rectangle_y_cut(&leftover, &leftover, &out[1], cut);
}
/* cut in width */
tmp = leftover;
if (eina_rectangle_intersection(&tmp, &intersection))
{
cut = leftover.w - (tmp.x - leftover.x);
if (cut > leftover.w) { cut = leftover.w; }
eina_rectangle_width_cut(&leftover, &leftover, &out[2], cut);
}
/* cut in x */
tmp = leftover;
if (eina_rectangle_intersection(&tmp, &intersection))
{
cut = leftover.w - (eina_rectangle_max_x(&leftover) - eina_rectangle_max_x(&tmp));
if (cut > leftover.w) { cut = leftover.w; }
eina_rectangle_x_cut(&leftover, &leftover, &out[3], cut);
}
return EINA_TRUE;
/* cut in height */
{
cut = thiz->h - (intersection.y - thiz->y);
if (cut > thiz->h) cut = thiz->h;
eina_rectangle_height_cut(thiz, &leftover, &out[0], cut);
}
/* cut in y */
tmp = leftover;
if (eina_rectangle_intersection(&tmp, &intersection))
{
cut = leftover.h - (eina_rectangle_max_y(&leftover) - eina_rectangle_max_y(&tmp));
if (cut > leftover.h) cut = leftover.h;
eina_rectangle_y_cut(&leftover, &leftover, &out[1], cut);
}
/* cut in width */
tmp = leftover;
if (eina_rectangle_intersection(&tmp, &intersection))
{
cut = leftover.w - (tmp.x - leftover.x);
if (cut > leftover.w) cut = leftover.w;
eina_rectangle_width_cut(&leftover, &leftover, &out[2], cut);
}
/* cut in x */
tmp = leftover;
if (eina_rectangle_intersection(&tmp, &intersection))
{
cut = leftover.w - (eina_rectangle_max_x(&leftover) - eina_rectangle_max_x(&tmp));
if (cut > leftover.w) cut = leftover.w;
eina_rectangle_x_cut(&leftover, &leftover, &out[3], cut);
}
return EINA_TRUE;
}
return EINA_FALSE;
}
#endif

View File

@ -38,10 +38,8 @@ eina_slice_dup(const Eina_Slice slice)
if (ret.len == 0) return ret;
ret.mem = malloc(ret.len);
if (!ret.mem)
ret.len = 0;
else
memcpy(ret.mem, slice.mem, ret.len);
if (ret.mem) memcpy(ret.mem, slice.mem, ret.len);
else ret.len = 0;
return ret;
}
@ -99,16 +97,13 @@ eina_slice_seek(const Eina_Slice slice, ssize_t offset, int whence)
offset += slice.len;
}
if (whence != SEEK_SET)
return ret;
if (offset < 0)
offset = 0;
else if ((size_t)offset > slice.len)
offset = slice.len;
ret.len = slice.len - offset;
ret.mem = (const void *)(slice.bytes + offset);
if (whence == SEEK_SET)
{
if ((size_t)offset > slice.len) offset = slice.len;
else if (offset < 0) offset = 0;
ret.len = slice.len - offset;
ret.mem = (const void *)(slice.bytes + offset);
}
return ret;
}
@ -126,88 +121,88 @@ eina_rw_slice_seek(const Eina_Rw_Slice rw_slice, ssize_t offset, int whence)
offset += rw_slice.len;
}
if (whence != SEEK_SET)
return ret;
if (offset < 0)
offset = 0;
else if ((size_t)offset > rw_slice.len)
offset = rw_slice.len;
ret.len = rw_slice.len - offset;
ret.mem = (void *)(rw_slice.bytes + offset);
if (whence == SEEK_SET)
{
if ((size_t)offset > rw_slice.len) offset = rw_slice.len;
else if (offset < 0) offset = 0;
ret.len = rw_slice.len - offset;
ret.mem = (void *)(rw_slice.bytes + offset);
}
return ret;
}
static inline const void *
eina_slice_strchr(const Eina_Slice slice, int c)
{
if (slice.len == 0) return NULL;
return memchr(slice.mem, c, slice.len);
if (slice.len != 0) return memchr(slice.mem, c, slice.len);
return NULL;
}
static inline const void *
eina_slice_find(const Eina_Slice slice, const Eina_Slice needle)
{
Eina_Slice s, n;
uint8_t c;
if (slice.len == 0) return NULL;
if (needle.len == 0) return NULL;
if (slice.len < needle.len) return NULL;
if (needle.len == 1) return eina_slice_strchr(slice, needle.bytes[0]);
if ((slice.len == needle.len) &&
(memcmp(slice.mem, needle.mem, needle.len) == 0))
return slice.mem;
s.mem = slice.mem;
s.len = slice.len - (needle.len - 1);
c = needle.bytes[0];
n.mem = (const void *)(needle.bytes + 1);
n.len = needle.len - 1;
while (s.len > 0)
if ((slice.len != 0) && (needle.len != 0) && (slice.len >= needle.len))
{
const uint8_t *p = (const uint8_t *)eina_slice_strchr(s, c);
size_t offset;
if (needle.len == 1) return eina_slice_strchr(slice, needle.bytes[0]);
if ((slice.len == needle.len) &&
(memcmp(slice.mem, needle.mem, needle.len) == 0))
return slice.mem;
if (!p) return NULL;
s.mem = slice.mem;
s.len = slice.len - (needle.len - 1);
p++;
if (memcmp(p, n.mem, n.len) == 0)
return (const void *)(p - 1);
c = needle.bytes[0];
n.mem = (const void *)(needle.bytes + 1);
n.len = needle.len - 1;
offset = p - s.bytes;
s.bytes += offset;
s.len -= offset;
while (s.len > 0)
{
const uint8_t *p = (const uint8_t *)eina_slice_strchr(s, c);
size_t offset;
if (p)
{
p++;
if (memcmp(p, n.mem, n.len) == 0)
return (const void *)(p - 1);
offset = p - s.bytes;
s.bytes += offset;
s.len -= offset;
continue;
}
break;
}
}
return NULL;
}
static inline Eina_Bool
eina_slice_startswith(const Eina_Slice slice, const Eina_Slice prefix)
{
if (prefix.len == 0) return EINA_FALSE;
if (slice.len < prefix.len) return EINA_FALSE;
return memcmp(slice.mem, prefix.mem, prefix.len) == 0;
if ((prefix.len != 0) && (slice.len >= prefix.len))
return memcmp(slice.mem, prefix.mem, prefix.len) == 0;
return EINA_FALSE;
}
static inline Eina_Bool
eina_slice_endswith(const Eina_Slice slice, const Eina_Slice suffix)
{
if (suffix.len == 0) return EINA_FALSE;
if (slice.len < suffix.len) return EINA_FALSE;
return memcmp(slice.bytes + slice.len - suffix.len,
suffix.mem, suffix.len) == 0;
if ((suffix.len != 0) && (slice.len > suffix.len))
return memcmp(slice.bytes + slice.len - suffix.len,
suffix.mem, suffix.len) == 0;
return EINA_FALSE;
}
static inline void *
eina_rw_slice_strchr(const Eina_Rw_Slice rw_slice, int c)
{
if (rw_slice.len == 0) return NULL;
return memchr(rw_slice.mem, c, rw_slice.len);
if (rw_slice.len != 0) return memchr(rw_slice.mem, c, rw_slice.len);
return NULL;
}
static inline void *
@ -219,18 +214,18 @@ eina_rw_slice_find(const Eina_Rw_Slice rw_slice, const Eina_Slice needle)
static inline Eina_Bool
eina_rw_slice_startswith(const Eina_Rw_Slice rw_slice, const Eina_Slice prefix)
{
if (prefix.len == 0) return EINA_FALSE;
if (rw_slice.len < prefix.len) return EINA_FALSE;
return memcmp(rw_slice.mem, prefix.mem, prefix.len) == 0;
if ((prefix.len != 0) && (rw_slice.len >= prefix.len))
return memcmp(rw_slice.mem, prefix.mem, prefix.len) == 0;
return EINA_FALSE;
}
static inline Eina_Bool
eina_rw_slice_endswith(const Eina_Rw_Slice rw_slice, const Eina_Slice suffix)
{
if (suffix.len == 0) return EINA_FALSE;
if (rw_slice.len < suffix.len) return EINA_FALSE;
return memcmp(rw_slice.bytes + rw_slice.len - suffix.len,
suffix.mem, suffix.len) == 0;
if ((suffix.len != 0) && (rw_slice.len >= suffix.len))
return memcmp(rw_slice.bytes + rw_slice.len - suffix.len,
suffix.mem, suffix.len) == 0;
return EINA_FALSE;
}
static inline const void *
@ -248,19 +243,17 @@ eina_rw_slice_end_get(const Eina_Rw_Slice rw_slice)
static inline char *
eina_slice_strdup(const Eina_Slice slice)
{
if (slice.len == 0)
return strdup("");
else
if (slice.len != 0)
return strndup((const char *)slice.mem, slice.len);
return strdup("");
}
static inline char *
eina_rw_slice_strdup(const Eina_Rw_Slice rw_slice)
{
if (rw_slice.len == 0)
return strdup("");
else
if (rw_slice.len != 0)
return strndup((const char *)rw_slice.mem, rw_slice.len);
return strdup("");
}
#endif /* _EINA_INLINE_SLICE_H */

View File

@ -92,8 +92,7 @@ static inline Eina_Bool
eina_streq(const char *a, const char *b)
{
if (a == b) return EINA_TRUE;
if (!a) return EINA_FALSE;
if (!b) return EINA_FALSE;
if ((!a) || (!b)) return EINA_FALSE;
return !strcmp(a, b);
}

270
src/lib/eina/eina_lock.c Normal file
View File

@ -0,0 +1,270 @@
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <stdio.h>
#include <string.h>
#include "eina_config.h"
#include "Eina.h"
EAPI void
_eina_lock_debug_abort(int err, const char *fn, const volatile void *ptr)
{
fprintf(stderr, "EINA ERROR: '%s' on %s %p\n", strerror(err), fn, ptr);
#ifdef EINA_HAVE_DEBUG_THREADS
abort();
#endif
}
EAPI void
_eina_lock_debug_deadlock(const char *fn, const volatile void *ptr)
{
fprintf(stderr, "EINA ERROR: DEADLOCK on %s %p\n", fn, ptr);
#ifdef EINA_HAVE_DEBUG_THREADS
abort();
#endif
}
EAPI void
eina_lock_debug(const Eina_Lock *mutex)
{
#ifdef EINA_HAVE_DEBUG_THREADS
fprintf(stderr, "EINA ERROR: DEADLOCK on lock %p, locked: %i, by %ti\n",
mutex, (int)mutex->locked, (ptrdiff_t)mutex->lock_thread_id);
backtrace_symbols_fd((void **)mutex->lock_bt, mutex->lock_bt_num, 1);
#else
fprintf(stderr, "EINA ERROR: DEADLOCK on lock %p\n", mutex);
#endif
}
EAPI Eina_Bool
_eina_lock_new(Eina_Lock *mutex, Eina_Bool recursive)
{
pthread_mutexattr_t attr;
Eina_Bool ok = EINA_FALSE;
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
if (pthread_mutexattr_init(&attr) != 0) return EINA_FALSE;
if (recursive)
{
if (pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE) != 0) goto fail_release;
}
#ifdef EINA_HAVE_DEBUG_THREADS
if (pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK) != 0) goto fail_release;
memset(mutex, 0, sizeof(Eina_Lock));
#endif
if (pthread_mutex_init(&(mutex->mutex), &attr) != 0) goto fail_release;
ok = EINA_TRUE;
fail_release:
pthread_mutexattr_destroy(&attr);
return ok;
}
EAPI void
_eina_lock_free(Eina_Lock *mutex)
{
int ok;
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
ok = pthread_mutex_destroy(&(mutex->mutex));
if (ok != 0) EINA_LOCK_ABORT_DEBUG(ok, mutex_destroy, mutex);
#ifdef EINA_HAVE_DEBUG_THREADS
memset(mutex, 0, sizeof(Eina_Lock));
#endif
}
EAPI Eina_Bool
_eina_condition_new(Eina_Condition *cond, Eina_Lock *mutex)
{
pthread_condattr_t attr;
int ok;
#ifdef EINA_HAVE_DEBUG_THREADS
assert(mutex != NULL);
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
memset(cond, 0, sizeof (Eina_Condition));
#endif
cond->lock = mutex;
pthread_condattr_init(&attr);
/* OSX doesn't provide clockid_t or clock_gettime. */
#if defined(__clockid_t_defined)
cond->clkid = (clockid_t) 0;
/* We try here to chose the best clock for cond_timedwait */
# if defined(CLOCK_MONOTONIC_RAW)
if (!pthread_condattr_setclock(&attr, CLOCK_MONOTONIC_RAW))
cond->clkid = CLOCK_MONOTONIC_RAW;
# endif
# if defined(CLOCK_MONOTONIC)
if (!cond->clkid && !pthread_condattr_setclock(&attr, CLOCK_MONOTONIC))
cond->clkid = CLOCK_MONOTONIC;
# endif
# if defined(CLOCK_REALTIME)
if (!cond->clkid && !pthread_condattr_setclock(&attr, CLOCK_REALTIME))
cond->clkid = CLOCK_REALTIME;
# endif
#endif
ok = pthread_cond_init(&cond->condition, &attr);
if (ok != 0)
{
pthread_condattr_destroy(&attr);
#ifdef EINA_HAVE_DEBUG_THREADS
if (ok == EBUSY)
fprintf(stderr, "EINA ERROR: eina_condition_new on already initialized Eina_Condition\n");
#endif
return EINA_FALSE;
}
pthread_condattr_destroy(&attr);
return EINA_TRUE;
}
EAPI void
_eina_condition_free(Eina_Condition *cond)
{
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
pthread_cond_destroy(&(cond->condition));
#ifdef EINA_HAVE_DEBUG_THREADS
memset(cond, 0, sizeof (Eina_Condition));
#endif
}
EAPI Eina_Bool
_eina_rwlock_new(Eina_RWLock *mutex)
{
int ok;
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
ok = pthread_rwlock_init(&(mutex->mutex), NULL);
if (ok == 0) return EINA_TRUE;
else if ((ok == EAGAIN) || (ok == ENOMEM)) return EINA_FALSE;
else EINA_LOCK_ABORT_DEBUG(ok, rwlock_init, mutex);
return EINA_FALSE;
}
EAPI void
_eina_rwlock_free(Eina_RWLock *mutex)
{
#ifdef EINA_HAVE_DEBUG_THREADS
if (!_eina_threads_activated)
assert(pthread_equal(_eina_main_loop, pthread_self()));
#endif
pthread_rwlock_destroy(&(mutex->mutex));
}
EAPI Eina_Bool
_eina_barrier_new(Eina_Barrier *barrier, int needed)
{
#ifdef EINA_HAVE_PTHREAD_BARRIER
int ok = pthread_barrier_init(&(barrier->barrier), NULL, needed);
if (ok == 0) return EINA_TRUE;
else if ((ok == EAGAIN) || (ok == ENOMEM)) return EINA_FALSE;
else EINA_LOCK_ABORT_DEBUG(ok, barrier_init, barrier);
return EINA_FALSE;
#else
barrier->needed = needed;
barrier->called = 0;
if (eina_lock_new(&(barrier->cond_lock)))
{
if (eina_condition_new(&(barrier->cond), &(barrier->cond_lock)))
return EINA_TRUE;
}
return EINA_FALSE;
#endif
}
EAPI void
_eina_barrier_free(Eina_Barrier *barrier)
{
#ifdef EINA_HAVE_PTHREAD_BARRIER
int ok = pthread_barrier_destroy(&(barrier->barrier));
if (ok != 0) EINA_LOCK_ABORT_DEBUG(ok, barrier_destroy, barrier);
#else
eina_condition_free(&(barrier->cond));
eina_lock_free(&(barrier->cond_lock));
barrier->needed = 0;
barrier->called = 0;
#endif
}
EAPI Eina_Bool
_eina_spinlock_new(Eina_Spinlock *spinlock)
{
#if defined(EINA_HAVE_POSIX_SPINLOCK)
int ok = pthread_spin_init(spinlock, PTHREAD_PROCESS_PRIVATE);
if (ok == 0) return EINA_TRUE;
else if ((ok == EAGAIN) || (ok == ENOMEM)) return EINA_FALSE;
else EINA_LOCK_ABORT_DEBUG(ok, spin_init, spinlock);
return EINA_FALSE;
#elif defined(EINA_HAVE_OSX_SPINLOCK)
/* OSSpinLock is an integer type. The convention is that unlocked is
* * zero, and locked is nonzero. */
*spinlock = 0;
return EINA_TRUE;
#else
return eina_lock_new(spinlock);
#endif
}
EAPI void
_eina_spinlock_free(Eina_Spinlock *spinlock)
{
#if defined(EINA_HAVE_POSIX_SPINLOCK)
int ok = pthread_spin_destroy(spinlock);
if (ok != 0) EINA_LOCK_ABORT_DEBUG(ok, spin_destroy, spinlock);
#elif defined(EINA_HAVE_OSX_SPINLOCK)
/* Not applicable */
(void) spinlock;
#else
eina_lock_free(spinlock);
#endif
}
EAPI Eina_Bool
_eina_semaphore_new(Eina_Semaphore *sem, int count_init)
{
if (sem && (count_init >= 0))
{
#if defined(EINA_HAVE_OSX_SEMAPHORE)
kern_return_t kr = semaphore_create(mach_task_self(), sem, SYNC_POLICY_FIFO, count_init);
return (kr == KERN_SUCCESS) ? EINA_TRUE : EINA_FALSE;
#else
return (sem_init(sem, 0, count_init) == 0) ? EINA_TRUE : EINA_FALSE;
#endif
}
return EINA_FALSE;
}
EAPI Eina_Bool
_eina_semaphore_free(Eina_Semaphore *sem)
{
if (sem)
{
#if defined(EINA_HAVE_OSX_SEMAPHORE)
return (semaphore_destroy(*sem, mach_task_self()) == KERN_SUCCESS)
? EINA_TRUE : EINA_FALSE;
#else
return (sem_destroy(sem) == 0) ? EINA_TRUE : EINA_FALSE;
#endif
}
return EINA_FALSE;
}

View File

@ -211,7 +211,7 @@ static inline Eina_Lock_Result eina_lock_release(Eina_Lock *mutex);
*
* @note If @c EINA_HAVE_DEBUG_THREADS is not defined, this function does nothing.
*/
static inline void eina_lock_debug(const Eina_Lock *mutex);
EAPI void eina_lock_debug(const Eina_Lock *mutex);
/**
* @brief Initializes a new condition variable.