#include "aux.h" #include #include #include #include #include #include #if 0 #define BSDB(x...) fprintf(stderr, x) #else #define BSDB(x...) ((void)0) #endif struct detach_thread { void (*func)(void *); void *data; }; struct thread_buf { char buf[THREAD_BUF_SIZE]; }; static mutex_t threads_lists_lock = MUTEX_STATIC_INIT; static GList *threads_to_join; static GList *threads_running; static cond_t threads_cond = COND_STATIC_INIT; static struct thread_buf __thread t_bufs[NUM_THREAD_BUFS]; static int __thread t_buf_idx; __thread struct timeval g_now; volatile int g_shutdown; #ifdef NEED_ATOMIC64_MUTEX mutex_t __atomic64_mutex = MUTEX_STATIC_INIT; #endif GList *g_list_link(GList *list, GList *el) { el->prev = NULL; el->next = list; if (list) list->prev = el; return el; } int pcre_multi_match(pcre *re, pcre_extra *ree, const char *s, unsigned int num, parse_func f, void *p, GQueue *q) { unsigned int start, len; int ovec[60]; int *ov; char **el; unsigned int i; void *ins; el = malloc(sizeof(*el) * num); for (start = 0, len = strlen(s); pcre_exec(re, ree, s + start, len - start, 0, 0, ovec, G_N_ELEMENTS(ovec)) > 0; start += ovec[1]) { for (i = 0; i < num; i++) { ov = ovec + 2 + i*2; el[i] = (ov[0] == -1) ? NULL : g_strndup(s + start + ov[0], ov[1] - ov[0]); } if (!f(el, &ins, p)) g_queue_push_tail(q, ins); for (i = 0; i < num; i++) { if (el[i]) free(el[i]); } } free(el); return q ? q->length : 0; } static void thread_join_me() { pthread_t *me; me = g_slice_alloc(sizeof(*me)); *me = pthread_self(); mutex_lock(&threads_lists_lock); threads_to_join = g_list_prepend(threads_to_join, me); cond_broadcast(&threads_cond); mutex_unlock(&threads_lists_lock); } static gint thread_equal(gconstpointer a, gconstpointer b) { const pthread_t *x = a, *y = b; return !pthread_equal(*x, *y); } void threads_join_all(int wait) { pthread_t *t; GList *l; mutex_lock(&threads_lists_lock); while (1) { while (threads_to_join) { t = threads_to_join->data; pthread_join(*t, NULL); threads_to_join = g_list_delete_link(threads_to_join, threads_to_join); l = g_list_find_custom(threads_running, t, thread_equal); if (l) threads_running = g_list_delete_link(threads_running, l); else abort(); g_slice_free1(sizeof(*t), t); } if (!wait) break; if (!threads_running) break; cond_wait(&threads_cond, &threads_lists_lock); } mutex_unlock(&threads_lists_lock); } static void *thread_detach_func(void *d) { struct detach_thread *dt = d; pthread_t *t; t = g_slice_alloc(sizeof(*t)); *t = pthread_self(); mutex_lock(&threads_lists_lock); threads_running = g_list_prepend(threads_running, t); mutex_unlock(&threads_lists_lock); dt->func(dt->data); g_slice_free1(sizeof(*dt), dt); thread_join_me(); return NULL; } static int thread_create(void *(*func)(void *), void *arg, int joinable, pthread_t *handle) { pthread_attr_t att; pthread_t thr; int ret; if (pthread_attr_init(&att)) abort(); if (pthread_attr_setdetachstate(&att, joinable ? PTHREAD_CREATE_JOINABLE : PTHREAD_CREATE_DETACHED)) abort(); ret = pthread_create(&thr, &att, func, arg); pthread_attr_destroy(&att); if (ret) return ret; if (handle) *handle = thr; return 0; } void thread_create_detach(void (*f)(void *), void *d) { struct detach_thread *dt; dt = g_slice_alloc(sizeof(*dt)); dt->func = f; dt->data = d; if (thread_create(thread_detach_func, dt, 1, NULL)) abort(); } unsigned int in6_addr_hash(const void *p) { const struct in6_addr *a = p; return a->s6_addr32[0] ^ a->s6_addr32[3]; } int in6_addr_eq(const void *a, const void *b) { const struct in6_addr *A = a, *B = b; return !memcmp(A, B, sizeof(*A)); } char *get_thread_buf(void) { char *ret; ret = t_bufs[t_buf_idx].buf; t_buf_idx++; if (t_buf_idx >= G_N_ELEMENTS(t_bufs)) t_buf_idx = 0; return ret; } int g_tree_find_first_cmp(void *k, void *v, void *d) { void **p = d; GEqualFunc f = p[1]; if (!f || f(v, p[0])) { p[2] = v; return TRUE; } return FALSE; } int g_tree_find_all_cmp(void *k, void *v, void *d) { void **p = d; GEqualFunc f = p[1]; GQueue *q = p[2]; if (!f || f(v, p[0])) g_queue_push_tail(q, v); return FALSE; } unsigned int uint32_hash(const void *p) { const u_int32_t *a = p; return *a; } int uint32_eq(const void *a, const void *b) { const u_int32_t *A = a, *B = b; return (*A == *B) ? TRUE : FALSE; }