mirror of
https://sourceware.org/git/glibc.git
synced 2025-07-29 11:41:21 +03:00
Fix races in setXid implementation.
This commit is contained in:
committed by
Ulrich Drepper
parent
31c759bf37
commit
25db0f6ca9
@ -1,3 +1,11 @@
|
|||||||
|
2009-10-30 Ulrich Drepper <drepper@redhat.com>
|
||||||
|
|
||||||
|
[BZ #3270]
|
||||||
|
* allocatestack.c (__nptl_setxid): Perform the operation in multiple
|
||||||
|
steps to avoid races with creation and terminations.
|
||||||
|
* nptl-init.c (sighandler_setxid): Adjust.
|
||||||
|
Patch by Daniel Jacobowitz.
|
||||||
|
|
||||||
2009-09-07 Andreas Schwab <schwab@redhat.com>
|
2009-09-07 Andreas Schwab <schwab@redhat.com>
|
||||||
|
|
||||||
* sysdeps/pthread/bits/libc-lock.h (BP_SYM): Remove space before paren.
|
* sysdeps/pthread/bits/libc-lock.h (BP_SYM): Remove space before paren.
|
||||||
|
@ -965,22 +965,53 @@ __find_thread_by_id (pid_t tid)
|
|||||||
|
|
||||||
static void
|
static void
|
||||||
internal_function
|
internal_function
|
||||||
|
setxid_mark_thread (struct xid_command *cmdp, struct pthread *t)
|
||||||
|
{
|
||||||
|
int ch;
|
||||||
|
|
||||||
|
/* Don't let the thread exit before the setxid handler runs. */
|
||||||
|
t->setxid_futex = 0;
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
ch = t->cancelhandling;
|
||||||
|
|
||||||
|
/* If the thread is exiting right now, ignore it. */
|
||||||
|
if ((ch & EXITING_BITMASK) != 0)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
while (atomic_compare_and_exchange_bool_acq (&t->cancelhandling,
|
||||||
|
ch | SETXID_BITMASK, ch));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
internal_function
|
||||||
|
setxid_unmark_thread (struct xid_command *cmdp, struct pthread *t)
|
||||||
|
{
|
||||||
|
int ch;
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
ch = t->cancelhandling;
|
||||||
|
if ((ch & SETXID_BITMASK) == 0)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
while (atomic_compare_and_exchange_bool_acq (&t->cancelhandling,
|
||||||
|
ch & ~SETXID_BITMASK, ch));
|
||||||
|
|
||||||
|
/* Release the futex just in case. */
|
||||||
|
t->setxid_futex = 1;
|
||||||
|
lll_futex_wake (&t->setxid_futex, 1, LLL_PRIVATE);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
internal_function
|
||||||
setxid_signal_thread (struct xid_command *cmdp, struct pthread *t)
|
setxid_signal_thread (struct xid_command *cmdp, struct pthread *t)
|
||||||
{
|
{
|
||||||
if (! IS_DETACHED (t))
|
if ((t->cancelhandling & SETXID_BITMASK) == 0)
|
||||||
{
|
return 0;
|
||||||
int ch;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
ch = t->cancelhandling;
|
|
||||||
|
|
||||||
/* If the thread is exiting right now, ignore it. */
|
|
||||||
if ((ch & EXITING_BITMASK) != 0)
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
while (atomic_compare_and_exchange_bool_acq (&t->cancelhandling,
|
|
||||||
ch | SETXID_BITMASK, ch));
|
|
||||||
}
|
|
||||||
|
|
||||||
int val;
|
int val;
|
||||||
INTERNAL_SYSCALL_DECL (err);
|
INTERNAL_SYSCALL_DECL (err);
|
||||||
@ -997,8 +1028,14 @@ setxid_signal_thread (struct xid_command *cmdp, struct pthread *t)
|
|||||||
val = INTERNAL_SYSCALL (tkill, err, 2, t->tid, SIGSETXID);
|
val = INTERNAL_SYSCALL (tkill, err, 2, t->tid, SIGSETXID);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* If this failed, it must have had not started yet or else exited. */
|
||||||
if (!INTERNAL_SYSCALL_ERROR_P (val, err))
|
if (!INTERNAL_SYSCALL_ERROR_P (val, err))
|
||||||
atomic_increment (&cmdp->cntr);
|
{
|
||||||
|
atomic_increment (&cmdp->cntr);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1006,6 +1043,7 @@ int
|
|||||||
attribute_hidden
|
attribute_hidden
|
||||||
__nptl_setxid (struct xid_command *cmdp)
|
__nptl_setxid (struct xid_command *cmdp)
|
||||||
{
|
{
|
||||||
|
int signalled;
|
||||||
int result;
|
int result;
|
||||||
lll_lock (stack_cache_lock, LLL_PRIVATE);
|
lll_lock (stack_cache_lock, LLL_PRIVATE);
|
||||||
|
|
||||||
@ -1022,7 +1060,7 @@ __nptl_setxid (struct xid_command *cmdp)
|
|||||||
if (t == self)
|
if (t == self)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
setxid_signal_thread (cmdp, t);
|
setxid_mark_thread (cmdp, t);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Now the list with threads using user-allocated stacks. */
|
/* Now the list with threads using user-allocated stacks. */
|
||||||
@ -1032,14 +1070,61 @@ __nptl_setxid (struct xid_command *cmdp)
|
|||||||
if (t == self)
|
if (t == self)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
setxid_signal_thread (cmdp, t);
|
setxid_mark_thread (cmdp, t);
|
||||||
}
|
}
|
||||||
|
|
||||||
int cur = cmdp->cntr;
|
/* Iterate until we don't succeed in signalling anyone. That means
|
||||||
while (cur != 0)
|
we have gotten all running threads, and their children will be
|
||||||
|
automatically correct once started. */
|
||||||
|
do
|
||||||
{
|
{
|
||||||
lll_futex_wait (&cmdp->cntr, cur, LLL_PRIVATE);
|
signalled = 0;
|
||||||
cur = cmdp->cntr;
|
|
||||||
|
list_for_each (runp, &stack_used)
|
||||||
|
{
|
||||||
|
struct pthread *t = list_entry (runp, struct pthread, list);
|
||||||
|
if (t == self)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
signalled += setxid_signal_thread (cmdp, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
list_for_each (runp, &__stack_user)
|
||||||
|
{
|
||||||
|
struct pthread *t = list_entry (runp, struct pthread, list);
|
||||||
|
if (t == self)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
signalled += setxid_signal_thread (cmdp, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
int cur = cmdp->cntr;
|
||||||
|
while (cur != 0)
|
||||||
|
{
|
||||||
|
lll_futex_wait (&cmdp->cntr, cur, LLL_PRIVATE);
|
||||||
|
cur = cmdp->cntr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (signalled != 0);
|
||||||
|
|
||||||
|
/* Clean up flags, so that no thread blocks during exit waiting
|
||||||
|
for a signal which will never come. */
|
||||||
|
list_for_each (runp, &stack_used)
|
||||||
|
{
|
||||||
|
struct pthread *t = list_entry (runp, struct pthread, list);
|
||||||
|
if (t == self)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
setxid_unmark_thread (cmdp, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
list_for_each (runp, &__stack_user)
|
||||||
|
{
|
||||||
|
struct pthread *t = list_entry (runp, struct pthread, list);
|
||||||
|
if (t == self)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
setxid_unmark_thread (cmdp, t);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This must be last, otherwise the current thread might not have
|
/* This must be last, otherwise the current thread might not have
|
||||||
|
@ -240,17 +240,23 @@ sighandler_setxid (int sig, siginfo_t *si, void *ctx)
|
|||||||
INTERNAL_SYSCALL_NCS (__xidcmd->syscall_no, err, 3, __xidcmd->id[0],
|
INTERNAL_SYSCALL_NCS (__xidcmd->syscall_no, err, 3, __xidcmd->id[0],
|
||||||
__xidcmd->id[1], __xidcmd->id[2]);
|
__xidcmd->id[1], __xidcmd->id[2]);
|
||||||
|
|
||||||
if (atomic_decrement_val (&__xidcmd->cntr) == 0)
|
|
||||||
lll_futex_wake (&__xidcmd->cntr, 1, LLL_PRIVATE);
|
|
||||||
|
|
||||||
/* Reset the SETXID flag. */
|
/* Reset the SETXID flag. */
|
||||||
struct pthread *self = THREAD_SELF;
|
struct pthread *self = THREAD_SELF;
|
||||||
int flags = THREAD_GETMEM (self, cancelhandling);
|
int flags, newval;
|
||||||
THREAD_SETMEM (self, cancelhandling, flags & ~SETXID_BITMASK);
|
do
|
||||||
|
{
|
||||||
|
flags = THREAD_GETMEM (self, cancelhandling);
|
||||||
|
newval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling,
|
||||||
|
flags & ~SETXID_BITMASK, flags);
|
||||||
|
}
|
||||||
|
while (flags != newval);
|
||||||
|
|
||||||
/* And release the futex. */
|
/* And release the futex. */
|
||||||
self->setxid_futex = 1;
|
self->setxid_futex = 1;
|
||||||
lll_futex_wake (&self->setxid_futex, 1, LLL_PRIVATE);
|
lll_futex_wake (&self->setxid_futex, 1, LLL_PRIVATE);
|
||||||
|
|
||||||
|
if (atomic_decrement_val (&__xidcmd->cntr) == 0)
|
||||||
|
lll_futex_wake (&__xidcmd->cntr, 1, LLL_PRIVATE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user