greatly improve SIGEV_THREAD timers

calling pthread_exit from, or pthread_cancel on, the timer callback
thread will no longer destroy the timer.
This commit is contained in:
Rich Felker 2011-04-09 02:23:33 -04:00
parent 2d1d62ed47
commit 82171d6ac0
3 changed files with 21 additions and 15 deletions

View File

@ -42,6 +42,7 @@ struct pthread {
void *pending;
} robust_list;
int unblock_cancel;
int delete_timer;
};
struct __timer {

View File

@ -13,21 +13,24 @@ struct start_args {
struct sigevent *sev;
};
static void cleanup_fromsig(void *p)
{
pthread_t self = __pthread_self();
self->cancel = 0;
self->cancelbuf = 0;
longjmp(p, 1);
}
void __sigtimer_handler(pthread_t self)
{
int st;
jmp_buf jb;
void (*notify)(union sigval) = (void (*)(union sigval))self->start;
union sigval val = { .sival_ptr = self->start_arg };
pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &st);
if (setjmp(jb)) return;
pthread_cleanup_push(cleanup_fromsig, jb);
notify(val);
pthread_setcancelstate(st, 0);
}
static void cleanup(void *p)
{
pthread_t self = p;
__syscall(SYS_timer_delete, self->result);
pthread_cleanup_pop(0);
}
static void *start(void *arg)
@ -41,12 +44,9 @@ static void *start(void *arg)
self->start_arg = args->sev->sigev_value.sival_ptr;
self->result = (void *)-1;
pthread_cleanup_push(cleanup, self);
pthread_barrier_wait(&args->b);
pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, 0);
/* Loop on async-signal-safe cancellation point */
for (;;) sleep(1000000000);
pthread_cleanup_pop(0);
__wait(&self->delete_timer, 0, 0, 1);
__syscall(SYS_timer_delete, self->result);
return 0;
}

View File

@ -3,6 +3,11 @@
int timer_delete(timer_t t)
{
if ((uintptr_t)t >= 0x100000) return pthread_cancel(t);
if ((uintptr_t)t >= 0x100000) {
pthread_t td = t;
td->delete_timer = 1;
__wake(&td->delete_timer, 1, 1);
return 0;
}
return __syscall(SYS_timer_delete, (long)t);
}