summaryrefslogtreecommitdiffstats
path: root/libpthread/nptl/sysdeps/pthread/timer_routines.c
diff options
context:
space:
mode:
author"Steven J. Hill" <sjhill@realitydiluted.com>2005-05-26 02:17:02 +0000
committer"Steven J. Hill" <sjhill@realitydiluted.com>2005-05-26 02:17:02 +0000
commit6321510368dffba1e4748233e7143f315d150484 (patch)
treef4a7037ba4086c5fa56bf3c6fdcf328345f16b55 /libpthread/nptl/sysdeps/pthread/timer_routines.c
parent8028f35dbe29c5ace3883005e08ac91d873553b2 (diff)
downloaduClibc-alpine-6321510368dffba1e4748233e7143f315d150484.tar.bz2
uClibc-alpine-6321510368dffba1e4748233e7143f315d150484.tar.xz
libpthread/nptl/sysdeps/unix/sysv/linux/libc_multiple_threads.c
libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/bits libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc32 libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc32/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc32/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc32/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc32/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/Versions libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/pt-longjmp.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/not-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64 libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/timer_gettime.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/timer_delete.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/timer_create.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/Versions libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/timer_settime.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/powerpc64/timer_getoverr.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/sem_post.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/createthread.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/pthread_once.c libpthread/nptl/sysdeps/unix/sysv/linux/powerpc/Makefile libpthread/nptl/sysdeps/unix/sysv/linux/configure libpthread/nptl/sysdeps/unix/sysv/linux/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_kill.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/timer_gettime.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/alpha/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/timer_delete.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/timer_create.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/alpha/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/alpha/timer_settime.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/aio_cancel.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/bits libpthread/nptl/sysdeps/unix/sysv/linux/alpha/bits/local_lim.h libpthread/nptl/sysdeps/unix/sysv/linux/alpha/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/alpha/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/alpha/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/alpha/Versions libpthread/nptl/sysdeps/unix/sysv/linux/alpha/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/alpha/sem_post.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/pthread_once.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/createthread.c libpthread/nptl/sysdeps/unix/sysv/linux/alpha/Makefile libpthread/nptl/sysdeps/unix/sysv/linux/alpha/timer_getoverr.c libpthread/nptl/sysdeps/unix/sysv/linux/smp.h libpthread/nptl/sysdeps/unix/sysv/linux/mq_notify.c libpthread/nptl/sysdeps/unix/sysv/linux/internaltypes.h libpthread/nptl/sysdeps/unix/sysv/linux/fork.h libpthread/nptl/sysdeps/unix/sysv/linux/sem_trywait.c libpthread/nptl/sysdeps/unix/sysv/linux/register-atfork.c libpthread/nptl/sysdeps/unix/sysv/linux/sleep.c libpthread/nptl/sysdeps/unix/sysv/linux/timer_routines.c libpthread/nptl/sysdeps/unix/sysv/linux/getpid.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-32 libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-32/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-32/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-32/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-32/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-32/pt-initfini.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/s390/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/s390/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/timer_gettime.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/timer_delete.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/timer_create.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/Versions libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/pt-initfini.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/timer_settime.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/s390-64/timer_getoverr.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/s390/jmp-unwind.c libpthread/nptl/sysdeps/unix/sysv/linux/s390/not-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/s390/pthread_once.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_yield.c libpthread/nptl/sysdeps/unix/sysv/linux/Implies libpthread/nptl/sysdeps/unix/sysv/linux/ia64/timer_gettime.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/ia64/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/timer_delete.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/clone2.S libpthread/nptl/sysdeps/unix/sysv/linux/ia64/timer_create.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/ia64/__ia64_longjmp.S libpthread/nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/ia64/dl-sysdep.h libpthread/nptl/sysdeps/unix/sysv/linux/ia64/timer_settime.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/bits/local_lim.h libpthread/nptl/sysdeps/unix/sysv/linux/ia64/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/ia64/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/ia64/Versions libpthread/nptl/sysdeps/unix/sysv/linux/ia64/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/ia64/unwind-forcedunwind.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/pt-initfini.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/jmpbuf-unwind.h libpthread/nptl/sysdeps/unix/sysv/linux/ia64/unwind_longjmp.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/pthread_once.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/createthread.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/__sigstack_longjmp.c libpthread/nptl/sysdeps/unix/sysv/linux/ia64/Makefile libpthread/nptl/sysdeps/unix/sysv/linux/ia64/timer_getoverr.c libpthread/nptl/sysdeps/unix/sysv/linux/pt-fork.c libpthread/nptl/sysdeps/unix/sysv/linux/pt-raise.c libpthread/nptl/sysdeps/unix/sysv/linux/kernel-posix-timers.h libpthread/nptl/sysdeps/unix/sysv/linux/lowlevelbarrier.sym libpthread/nptl/sysdeps/unix/sysv/linux/sparc/bits/local_lim.h libpthread/nptl/sysdeps/unix/sysv/linux/sparc/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/sparc/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/sparc/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/timer_gettime.c libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/timer_delete.c libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/timer_create.c libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/Versions libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/timer_settime.c libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc64/timer_getoverr.c libpthread/nptl/sysdeps/unix/sysv/linux/sparc/Versions libpthread/nptl/sysdeps/unix/sysv/linux/sparc/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/sparc/not-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/sparc/pthread_once.c libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc32 libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc32/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc32/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc32/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/sparc/sparc32/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/sparc/aio_cancel.c libpthread/nptl/sysdeps/unix/sysv/linux/mips/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/mips/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/mips/bits libpthread/nptl/sysdeps/unix/sysv/linux/mips/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/mips/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/mips/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/mips/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/mips/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/mips/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/mips/createthread.c libpthread/nptl/sysdeps/unix/sysv/linux/mips/pthread_once.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_getcpuclockid.c libpthread/nptl/sysdeps/unix/sysv/linux/jmp-unwind.c libpthread/nptl/sysdeps/unix/sysv/linux/aio_misc.h libpthread/nptl/sysdeps/unix/sysv/linux/sem_post.c libpthread/nptl/sysdeps/unix/sysv/linux/sigtimedwait.c libpthread/nptl/sysdeps/unix/sysv/linux/createthread.c libpthread/nptl/sysdeps/unix/sysv/linux/Makefile libpthread/nptl/sysdeps/unix/sysv/linux/timer_gettime.c libpthread/nptl/sysdeps/unix/sysv/linux/lowlevelrwlock.sym libpthread/nptl/sysdeps/unix/sysv/linux/lowlevellock.c libpthread/nptl/sysdeps/unix/sysv/linux/timer_delete.c libpthread/nptl/sysdeps/unix/sysv/linux/raise.c libpthread/nptl/sysdeps/unix/sysv/linux/allocrtsig.c libpthread/nptl/sysdeps/unix/sysv/linux/libc_pthread_init.c libpthread/nptl/sysdeps/unix/sysv/linux/timer_create.c libpthread/nptl/sysdeps/unix/sysv/linux/sigwait.c libpthread/nptl/sysdeps/unix/sysv/linux/sem_wait.c libpthread/nptl/sysdeps/unix/sysv/linux/i386/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/i386/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/i386/pthread_spin_init.c libpthread/nptl/sysdeps/unix/sysv/linux/i386/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/i386/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/dl-sysdep.h libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_cond_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_barrier_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_rwlock_unlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_cond_signal.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_rwlock_timedrdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_rwlock_wrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/sem_post.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/sem_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_cond_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_rwlock_rdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/sem_trywait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/sem_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/libc-lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_rwlock_timedwrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i486/pthread_cond_broadcast.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_cond_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_barrier_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_rwlock_unlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_cond_signal.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_rwlock_timedrdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_rwlock_wrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/sem_post.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/sem_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_cond_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_rwlock_rdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/sem_trywait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/sem_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/libc-lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_rwlock_timedwrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i586/pthread_cond_broadcast.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_cond_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_barrier_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_rwlock_unlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_cond_signal.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/dl-sysdep.h libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_rwlock_timedrdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_rwlock_wrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/sem_post.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/sem_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_cond_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_rwlock_rdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/sem_trywait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/sem_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/libc-lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_rwlock_timedwrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/i686/pthread_cond_broadcast.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/pthread_once.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/pthread_spin_unlock.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/i386/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/i386/not-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/i386/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/i386/createthread.c libpthread/nptl/sysdeps/unix/sysv/linux/libc-lowlevellock.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_attr_getaffinity.c libpthread/nptl/sysdeps/unix/sysv/linux/timer_settime.c libpthread/nptl/sysdeps/unix/sysv/linux/unregister-atfork.c libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_barrier_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_cond_signal.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/sem_post.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_rwlock_timedrdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_once.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/sem_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/libc-lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/Makefile libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_cond_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/timer_gettime.c libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/timer_delete.c libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_rwlock_unlock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_spin_init.c libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/timer_create.c libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/compat-timer.h libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_rwlock_wrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/timer_settime.c libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_spin_unlock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/sem_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/bits libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_cond_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_rwlock_rdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/Versions libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/sem_trywait.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/not-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_rwlock_timedwrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/pthread_cond_broadcast.S libpthread/nptl/sysdeps/unix/sysv/linux/x86_64/timer_getoverr.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_getaffinity.c libpthread/nptl/sysdeps/unix/sysv/linux/bits/local_lim.h libpthread/nptl/sysdeps/unix/sysv/linux/bits/posix_opt.h libpthread/nptl/sysdeps/unix/sysv/linux/lowlevelcond.sym libpthread/nptl/sysdeps/unix/sysv/linux/sigwaitinfo.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_mutex_cond_lock.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_attr_setaffinity.c libpthread/nptl/sysdeps/unix/sysv/linux/Versions libpthread/nptl/sysdeps/unix/sysv/linux/unwindbuf.sym libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_cond_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/sysdep-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/fork.c libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_barrier_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_rwlock_unlock.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/smp.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_cond_signal.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/lowlevel-atomic.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/sem_post.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_rwlock_wrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_rwlock_timedrdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/sem_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_once.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/sh4 libpthread/nptl/sysdeps/unix/sysv/linux/sh/sh4/lowlevellock.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/clone.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/bits libpthread/nptl/sysdeps/unix/sysv/linux/sh/bits/pthreadtypes.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/bits/semaphore.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_cond_timedwait.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_rwlock_rdlock.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/sem_trywait.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pt-vfork.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/not-cancel.h libpthread/nptl/sysdeps/unix/sysv/linux/sh/sem_wait.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/libc-lowlevellock.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pt-initfini.c libpthread/nptl/sysdeps/unix/sysv/linux/sh/createthread.c libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_cond_broadcast.S libpthread/nptl/sysdeps/unix/sysv/linux/sh/pthread_rwlock_timedwrlock.S libpthread/nptl/sysdeps/unix/sysv/linux/sem_timedwait.c libpthread/nptl/sysdeps/unix/sysv/linux/pthread_setaffinity.c libpthread/nptl/sysdeps/unix/sysv/linux/timer_getoverr.c libpthread/nptl/sysdeps/pthread/configure libpthread/nptl/sysdeps/pthread/pthread_cond_timedwait.c libpthread/nptl/sysdeps/pthread/unwind-resume.c libpthread/nptl/sysdeps/pthread/sigfillset.c libpthread/nptl/sysdeps/pthread/pthread_rwlock_rdlock.c libpthread/nptl/sysdeps/pthread/timer_routines.c libpthread/nptl/sysdeps/pthread/pthread_cond_broadcast.c libpthread/nptl/sysdeps/pthread/pthread_rwlock_timedwrlock.c libpthread/nptl/sysdeps/pthread/malloc-machine.h libpthread/nptl/sysdeps/pthread/pthread_barrier_wait.c libpthread/nptl/sysdeps/pthread/tcb-offsets.h libpthread/nptl/sysdeps/pthread/pthread_getcpuclockid.c libpthread/nptl/sysdeps/pthread/pt-longjmp.c libpthread/nptl/sysdeps/pthread/pthread_cond_signal.c libpthread/nptl/sysdeps/pthread/unwind-forcedunwind.c libpthread/nptl/sysdeps/pthread/pthread_rwlock_timedrdlock.c libpthread/nptl/sysdeps/pthread/pthread_once.c libpthread/nptl/sysdeps/pthread/createthread.c libpthread/nptl/sysdeps/pthread/Makefile libpthread/nptl/sysdeps/pthread/pthread.h libpthread/nptl/sysdeps/pthread/timer_gettime.c libpthread/nptl/sysdeps/pthread/flockfile.c libpthread/nptl/sysdeps/pthread/funlockfile.c libpthread/nptl/sysdeps/pthread/timer_delete.c libpthread/nptl/sysdeps/pthread/pthread-functions.h libpthread/nptl/sysdeps/pthread/allocalim.h libpthread/nptl/sysdeps/pthread/pthread_spin_init.c libpthread/nptl/sysdeps/pthread/timer_create.c libpthread/nptl/sysdeps/pthread/setxid.h libpthread/nptl/sysdeps/pthread/pthread_sigmask.c libpthread/nptl/sysdeps/pthread/configure.in libpthread/nptl/sysdeps/pthread/librt-cancellation.c libpthread/nptl/sysdeps/pthread/timer_settime.c libpthread/nptl/sysdeps/pthread/pthread_cond_wait.c libpthread/nptl/sysdeps/pthread/sigaction.c libpthread/nptl/sysdeps/pthread/sigprocmask.c libpthread/nptl/sysdeps/pthread/pthread_rwlock_unlock.c libpthread/nptl/sysdeps/pthread/rt-unwind-resume.c libpthread/nptl/sysdeps/pthread/posix-timer.h libpthread/nptl/sysdeps/pthread/pthread_rwlock_wrlock.c libpthread/nptl/sysdeps/pthread/pthread_spin_destroy.c libpthread/nptl/sysdeps/pthread/pt-initfini.c libpthread/nptl/sysdeps/pthread/Subdirs libpthread/nptl/sysdeps/pthread/pthread_spin_unlock.c libpthread/nptl/sysdeps/pthread/ftrylockfile.c libpthread/nptl/sysdeps/pthread/timer_getoverr.c libpthread/nptl/sysdeps/generic/pt-raise.c libpthread/nptl/sysdeps/generic/lowlevellock.h
Diffstat (limited to 'libpthread/nptl/sysdeps/pthread/timer_routines.c')
-rw-r--r--libpthread/nptl/sysdeps/pthread/timer_routines.c578
1 files changed, 578 insertions, 0 deletions
diff --git a/libpthread/nptl/sysdeps/pthread/timer_routines.c b/libpthread/nptl/sysdeps/pthread/timer_routines.c
new file mode 100644
index 000000000..8d5b1d13a
--- /dev/null
+++ b/libpthread/nptl/sysdeps/pthread/timer_routines.c
@@ -0,0 +1,578 @@
+/* Helper code for POSIX timer implementation on NPTL.
+ Copyright (C) 2000, 2001, 2002, 2003, 2004 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+ Contributed by Kaz Kylheku <kaz@ashi.footprints.net>.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public License as
+ published by the Free Software Foundation; either version 2.1 of the
+ License, or (at your option) any later version.
+
+ The GNU C Library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; see the file COPYING.LIB. If not,
+ write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ Boston, MA 02111-1307, USA. */
+
+#include <assert.h>
+#include <errno.h>
+#include <pthread.h>
+#include <stddef.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sysdep.h>
+#include <time.h>
+#include <unistd.h>
+#include <sys/syscall.h>
+
+#include "posix-timer.h"
+#include <pthreadP.h>
+
+
+/* Number of threads used. */
+#define THREAD_MAXNODES 16
+
+/* Array containing the descriptors for the used threads. */
+static struct thread_node thread_array[THREAD_MAXNODES];
+
+/* Static array with the structures for all the timers. */
+struct timer_node __timer_array[TIMER_MAX];
+
+/* Global lock to protect operation on the lists. */
+pthread_mutex_t __timer_mutex = PTHREAD_MUTEX_INITIALIZER;
+
+/* Variable to protext initialization. */
+pthread_once_t __timer_init_once_control = PTHREAD_ONCE_INIT;
+
+/* Nonzero if initialization of timer implementation failed. */
+int __timer_init_failed;
+
+/* Node for the thread used to deliver signals. */
+struct thread_node __timer_signal_thread_rclk;
+
+/* Lists to keep free and used timers and threads. */
+struct list_links timer_free_list;
+struct list_links thread_free_list;
+struct list_links thread_active_list;
+
+
+#ifdef __NR_rt_sigqueueinfo
+extern int __syscall_rt_sigqueueinfo (int, int, siginfo_t *);
+#endif
+
+
+/* List handling functions. */
+static inline void
+list_init (struct list_links *list)
+{
+ list->next = list->prev = list;
+}
+
+static inline void
+list_append (struct list_links *list, struct list_links *newp)
+{
+ newp->prev = list->prev;
+ newp->next = list;
+ list->prev->next = newp;
+ list->prev = newp;
+}
+
+static inline void
+list_insbefore (struct list_links *list, struct list_links *newp)
+{
+ list_append (list, newp);
+}
+
+/*
+ * Like list_unlink_ip, except that calling it on a node that
+ * is already unlinked is disastrous rather than a noop.
+ */
+
+static inline void
+list_unlink (struct list_links *list)
+{
+ struct list_links *lnext = list->next, *lprev = list->prev;
+
+ lnext->prev = lprev;
+ lprev->next = lnext;
+}
+
+static inline struct list_links *
+list_first (struct list_links *list)
+{
+ return list->next;
+}
+
+static inline struct list_links *
+list_null (struct list_links *list)
+{
+ return list;
+}
+
+static inline struct list_links *
+list_next (struct list_links *list)
+{
+ return list->next;
+}
+
+static inline int
+list_isempty (struct list_links *list)
+{
+ return list->next == list;
+}
+
+
+/* Functions build on top of the list functions. */
+static inline struct thread_node *
+thread_links2ptr (struct list_links *list)
+{
+ return (struct thread_node *) ((char *) list
+ - offsetof (struct thread_node, links));
+}
+
+static inline struct timer_node *
+timer_links2ptr (struct list_links *list)
+{
+ return (struct timer_node *) ((char *) list
+ - offsetof (struct timer_node, links));
+}
+
+
+/* Initialize a newly allocated thread structure. */
+static void
+thread_init (struct thread_node *thread, const pthread_attr_t *attr, clockid_t clock_id)
+{
+ if (attr != NULL)
+ thread->attr = *attr;
+ else
+ {
+ pthread_attr_init (&thread->attr);
+ pthread_attr_setdetachstate (&thread->attr, PTHREAD_CREATE_DETACHED);
+ }
+
+ thread->exists = 0;
+ list_init (&thread->timer_queue);
+ pthread_cond_init (&thread->cond, 0);
+ thread->current_timer = 0;
+ thread->captured = pthread_self ();
+ thread->clock_id = clock_id;
+}
+
+
+/* Initialize the global lists, and acquire global resources. Error
+ reporting is done by storing a non-zero value to the global variable
+ timer_init_failed. */
+static void
+init_module (void)
+{
+ int i;
+
+ list_init (&timer_free_list);
+ list_init (&thread_free_list);
+ list_init (&thread_active_list);
+
+ for (i = 0; i < TIMER_MAX; ++i)
+ {
+ list_append (&timer_free_list, &__timer_array[i].links);
+ __timer_array[i].inuse = TIMER_FREE;
+ }
+
+ for (i = 0; i < THREAD_MAXNODES; ++i)
+ list_append (&thread_free_list, &thread_array[i].links);
+
+ thread_init (&__timer_signal_thread_rclk, 0, CLOCK_REALTIME);
+}
+
+
+/* This is a handler executed in a child process after a fork()
+ occurs. It reinitializes the module, resetting all of the data
+ structures to their initial state. The mutex is initialized in
+ case it was locked in the parent process. */
+static void
+reinit_after_fork (void)
+{
+ init_module ();
+ pthread_mutex_init (&__timer_mutex, 0);
+}
+
+
+/* Called once form pthread_once in timer_init. This initializes the
+ module and ensures that reinit_after_fork will be executed in any
+ child process. */
+void
+__timer_init_once (void)
+{
+ init_module ();
+ pthread_atfork (0, 0, reinit_after_fork);
+}
+
+
+/* Deinitialize a thread that is about to be deallocated. */
+static void
+thread_deinit (struct thread_node *thread)
+{
+ assert (list_isempty (&thread->timer_queue));
+ pthread_cond_destroy (&thread->cond);
+}
+
+
+/* Allocate a thread structure from the global free list. Global
+ mutex lock must be held by caller. The thread is moved to
+ the active list. */
+struct thread_node *
+__timer_thread_alloc (const pthread_attr_t *desired_attr, clockid_t clock_id)
+{
+ struct list_links *node = list_first (&thread_free_list);
+
+ if (node != list_null (&thread_free_list))
+ {
+ struct thread_node *thread = thread_links2ptr (node);
+ list_unlink (node);
+ thread_init (thread, desired_attr, clock_id);
+ list_append (&thread_active_list, node);
+ return thread;
+ }
+
+ return 0;
+}
+
+
+/* Return a thread structure to the global free list. Global lock
+ must be held by caller. */
+void
+__timer_thread_dealloc (struct thread_node *thread)
+{
+ thread_deinit (thread);
+ list_unlink (&thread->links);
+ list_append (&thread_free_list, &thread->links);
+}
+
+
+/* Each of our threads which terminates executes this cleanup
+ handler. We never terminate threads ourselves; if a thread gets here
+ it means that the evil application has killed it. If the thread has
+ timers, these require servicing and so we must hire a replacement
+ thread right away. We must also unblock another thread that may
+ have been waiting for this thread to finish servicing a timer (see
+ timer_delete()). */
+
+static void
+thread_cleanup (void *val)
+{
+ if (val != NULL)
+ {
+ struct thread_node *thread = val;
+
+ /* How did the signal thread get killed? */
+ assert (thread != &__timer_signal_thread_rclk);
+
+ pthread_mutex_lock (&__timer_mutex);
+
+ thread->exists = 0;
+
+ /* We are no longer processing a timer event. */
+ thread->current_timer = 0;
+
+ if (list_isempty (&thread->timer_queue))
+ __timer_thread_dealloc (thread);
+ else
+ (void) __timer_thread_start (thread);
+
+ pthread_mutex_unlock (&__timer_mutex);
+
+ /* Unblock potentially blocked timer_delete(). */
+ pthread_cond_broadcast (&thread->cond);
+ }
+}
+
+
+/* Handle a timer which is supposed to go off now. */
+static void
+thread_expire_timer (struct thread_node *self, struct timer_node *timer)
+{
+ self->current_timer = timer; /* Lets timer_delete know timer is running. */
+
+ pthread_mutex_unlock (&__timer_mutex);
+
+ switch (__builtin_expect (timer->event.sigev_notify, SIGEV_SIGNAL))
+ {
+ case SIGEV_NONE:
+ break;
+
+ case SIGEV_SIGNAL:
+#ifdef __NR_rt_sigqueueinfo
+ {
+ siginfo_t info;
+
+ /* First, clear the siginfo_t structure, so that we don't pass our
+ stack content to other tasks. */
+ memset (&info, 0, sizeof (siginfo_t));
+ /* We must pass the information about the data in a siginfo_t
+ value. */
+ info.si_signo = timer->event.sigev_signo;
+ info.si_code = SI_TIMER;
+ info.si_pid = timer->creator_pid;
+ info.si_uid = getuid ();
+ info.si_value = timer->event.sigev_value;
+
+ INLINE_SYSCALL (rt_sigqueueinfo, 3, info.si_pid, info.si_signo, &info);
+ }
+#else
+ if (pthread_kill (self->captured, timer->event.sigev_signo) != 0)
+ {
+ if (pthread_kill (self->id, timer->event.sigev_signo) != 0)
+ abort ();
+ }
+#endif
+ break;
+
+ case SIGEV_THREAD:
+ timer->event.sigev_notify_function (timer->event.sigev_value);
+ break;
+
+ default:
+ assert (! "unknown event");
+ break;
+ }
+
+ pthread_mutex_lock (&__timer_mutex);
+
+ self->current_timer = 0;
+
+ pthread_cond_broadcast (&self->cond);
+}
+
+
+/* Thread function; executed by each timer thread. The job of this
+ function is to wait on the thread's timer queue and expire the
+ timers in chronological order as close to their scheduled time as
+ possible. */
+static void
+__attribute__ ((noreturn))
+thread_func (void *arg)
+{
+ struct thread_node *self = arg;
+
+ /* Register cleanup handler, in case rogue application terminates
+ this thread. (This cannot happen to __timer_signal_thread, which
+ doesn't invoke application callbacks). */
+
+ pthread_cleanup_push (thread_cleanup, self);
+
+ pthread_mutex_lock (&__timer_mutex);
+
+ while (1)
+ {
+ struct list_links *first;
+ struct timer_node *timer = NULL;
+
+ /* While the timer queue is not empty, inspect the first node. */
+ first = list_first (&self->timer_queue);
+ if (first != list_null (&self->timer_queue))
+ {
+ struct timespec now;
+
+ timer = timer_links2ptr (first);
+
+ /* This assumes that the elements of the list of one thread
+ are all for the same clock. */
+ clock_gettime (timer->clock, &now);
+
+ while (1)
+ {
+ /* If the timer is due or overdue, remove it from the queue.
+ If it's a periodic timer, re-compute its new time and
+ requeue it. Either way, perform the timer expiry. */
+ if (timespec_compare (&now, &timer->expirytime) < 0)
+ break;
+
+ list_unlink_ip (first);
+
+ if (__builtin_expect (timer->value.it_interval.tv_sec, 0) != 0
+ || timer->value.it_interval.tv_nsec != 0)
+ {
+ timer->overrun_count = 0;
+ timespec_add (&timer->expirytime, &timer->expirytime,
+ &timer->value.it_interval);
+ while (timespec_compare (&timer->expirytime, &now) < 0)
+ {
+ timespec_add (&timer->expirytime, &timer->expirytime,
+ &timer->value.it_interval);
+ if (timer->overrun_count < DELAYTIMER_MAX)
+ ++timer->overrun_count;
+ }
+ __timer_thread_queue_timer (self, timer);
+ }
+
+ thread_expire_timer (self, timer);
+
+ first = list_first (&self->timer_queue);
+ if (first == list_null (&self->timer_queue))
+ break;
+
+ timer = timer_links2ptr (first);
+ }
+ }
+
+ /* If the queue is not empty, wait until the expiry time of the
+ first node. Otherwise wait indefinitely. Insertions at the
+ head of the queue must wake up the thread by broadcasting
+ this condition variable. */
+ if (timer != NULL)
+ pthread_cond_timedwait (&self->cond, &__timer_mutex,
+ &timer->expirytime);
+ else
+ pthread_cond_wait (&self->cond, &__timer_mutex);
+ }
+ /* This macro will never be executed since the while loop loops
+ forever - but we have to add it for proper nesting. */
+ pthread_cleanup_pop (1);
+}
+
+
+/* Enqueue a timer in wakeup order in the thread's timer queue.
+ Returns 1 if the timer was inserted at the head of the queue,
+ causing the queue's next wakeup time to change. */
+
+int
+__timer_thread_queue_timer (struct thread_node *thread,
+ struct timer_node *insert)
+{
+ struct list_links *iter;
+ int athead = 1;
+
+ for (iter = list_first (&thread->timer_queue);
+ iter != list_null (&thread->timer_queue);
+ iter = list_next (iter))
+ {
+ struct timer_node *timer = timer_links2ptr (iter);
+
+ if (timespec_compare (&insert->expirytime, &timer->expirytime) < 0)
+ break;
+ athead = 0;
+ }
+
+ list_insbefore (iter, &insert->links);
+ return athead;
+}
+
+
+/* Start a thread and associate it with the given thread node. Global
+ lock must be held by caller. */
+int
+__timer_thread_start (struct thread_node *thread)
+{
+ int retval = 1;
+
+ assert (!thread->exists);
+ thread->exists = 1;
+
+ if (pthread_create (&thread->id, &thread->attr,
+ (void *(*) (void *)) thread_func, thread) != 0)
+ {
+ thread->exists = 0;
+ retval = -1;
+ }
+
+ return retval;
+}
+
+
+void
+__timer_thread_wakeup (struct thread_node *thread)
+{
+ pthread_cond_broadcast (&thread->cond);
+}
+
+
+/* Compare two pthread_attr_t thread attributes for exact equality.
+ Returns 1 if they are equal, otherwise zero if they are not equal
+ or contain illegal values. This version is NPTL-specific for
+ performance reason. One could use the access functions to get the
+ values of all the fields of the attribute structure. */
+static int
+thread_attr_compare (const pthread_attr_t *left, const pthread_attr_t *right)
+{
+ struct pthread_attr *ileft = (struct pthread_attr *) left;
+ struct pthread_attr *iright = (struct pthread_attr *) right;
+
+ return (ileft->flags == iright->flags
+ && ileft->schedpolicy == iright->schedpolicy
+ && (ileft->schedparam.sched_priority
+ == iright->schedparam.sched_priority)
+ && ileft->guardsize == iright->guardsize
+ && ileft->stackaddr == iright->stackaddr
+ && ileft->stacksize == iright->stacksize
+ && ((ileft->cpuset == NULL && iright->cpuset == NULL)
+ || (ileft->cpuset != NULL && iright->cpuset != NULL
+ && ileft->cpusetsize == iright->cpusetsize
+ && memcmp (ileft->cpuset, iright->cpuset,
+ ileft->cpusetsize) == 0)));
+}
+
+
+/* Search the list of active threads and find one which has matching
+ attributes. Global mutex lock must be held by caller. */
+struct thread_node *
+__timer_thread_find_matching (const pthread_attr_t *desired_attr,
+ clockid_t desired_clock_id)
+{
+ struct list_links *iter = list_first (&thread_active_list);
+
+ while (iter != list_null (&thread_active_list))
+ {
+ struct thread_node *candidate = thread_links2ptr (iter);
+
+ if (thread_attr_compare (desired_attr, &candidate->attr)
+ && desired_clock_id == candidate->clock_id)
+ return candidate;
+
+ iter = list_next (iter);
+ }
+
+ return NULL;
+}
+
+
+/* Grab a free timer structure from the global free list. The global
+ lock must be held by the caller. */
+struct timer_node *
+__timer_alloc (void)
+{
+ struct list_links *node = list_first (&timer_free_list);
+
+ if (node != list_null (&timer_free_list))
+ {
+ struct timer_node *timer = timer_links2ptr (node);
+ list_unlink_ip (node);
+ timer->inuse = TIMER_INUSE;
+ timer->refcount = 1;
+ return timer;
+ }
+
+ return NULL;
+}
+
+
+/* Return a timer structure to the global free list. The global lock
+ must be held by the caller. */
+void
+__timer_dealloc (struct timer_node *timer)
+{
+ assert (timer->refcount == 0);
+ timer->thread = NULL; /* Break association between timer and thread. */
+ timer->inuse = TIMER_FREE;
+ list_append (&timer_free_list, &timer->links);
+}
+
+
+/* Thread cancellation handler which unlocks a mutex. */
+void
+__timer_mutex_cancel_handler (void *arg)
+{
+ pthread_mutex_unlock (arg);
+}