nptl: Move cancel type out of cancelhandling

Now that the thread cancellation type is not accessed concurrently
anymore, it is possible to move it out the cancelhandling.

By removing the cancel state out of the internal thread cancel handling
state there is no need to check if cancelled bit was set in CAS
operation.

It allows simplifing the cancellation wrappers and the
CANCEL_CANCELED_AND_ASYNCHRONOUS is removed.

Checked on x86_64-linux-gnu and aarch64-linux-gnu.
This commit is contained in:
Adhemerval Zanella 2020-03-31 17:24:39 -03:00
parent 2b51742531
commit 8c1c0aae20
8 changed files with 36 additions and 171 deletions

View File

@ -161,6 +161,7 @@ get_cached_stack (size_t *sizep, void **memp)
/* Cancellation handling is back to the default. */ /* Cancellation handling is back to the default. */
result->cancelhandling = 0; result->cancelhandling = 0;
result->cancelstate = PTHREAD_CANCEL_ENABLE; result->cancelstate = PTHREAD_CANCEL_ENABLE;
result->canceltype = PTHREAD_CANCEL_DEFERRED;
result->cleanup = NULL; result->cleanup = NULL;
result->setup_failed = 0; result->setup_failed = 0;

View File

@ -31,31 +31,19 @@ int
__pthread_enable_asynccancel (void) __pthread_enable_asynccancel (void)
{ {
struct pthread *self = THREAD_SELF; struct pthread *self = THREAD_SELF;
int oldval = THREAD_GETMEM (self, cancelhandling);
while (1) int oldval = THREAD_GETMEM (self, canceltype);
THREAD_SETMEM (self, canceltype, PTHREAD_CANCEL_ASYNCHRONOUS);
int ch = THREAD_GETMEM (self, cancelhandling);
if (self->cancelstate == PTHREAD_CANCEL_ENABLE
&& (ch & CANCELED_BITMASK)
&& !(ch & EXITING_BITMASK)
&& !(ch & TERMINATED_BITMASK))
{ {
int newval = oldval | CANCELTYPE_BITMASK; THREAD_SETMEM (self, result, PTHREAD_CANCELED);
__do_cancel ();
if (newval == oldval)
break;
int curval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling, newval,
oldval);
if (__glibc_likely (curval == oldval))
{
if (self->cancelstate == PTHREAD_CANCEL_ENABLE
&& CANCEL_CANCELED_AND_ASYNCHRONOUS (newval))
{
THREAD_SETMEM (self, result, PTHREAD_CANCELED);
__do_cancel ();
}
break;
}
/* Prepare the next round. */
oldval = curval;
} }
return oldval; return oldval;
@ -69,25 +57,10 @@ __pthread_disable_asynccancel (int oldtype)
{ {
/* If asynchronous cancellation was enabled before we do not have /* If asynchronous cancellation was enabled before we do not have
anything to do. */ anything to do. */
if (oldtype & CANCELTYPE_BITMASK) if (oldtype == PTHREAD_CANCEL_ASYNCHRONOUS)
return; return;
struct pthread *self = THREAD_SELF; struct pthread *self = THREAD_SELF;
int newval; self->canceltype = PTHREAD_CANCEL_DEFERRED;
int oldval = THREAD_GETMEM (self, cancelhandling);
while (1)
{
newval = oldval & ~CANCELTYPE_BITMASK;
int curval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling, newval,
oldval);
if (__glibc_likely (curval == oldval))
break;
/* Prepare the next round. */
oldval = curval;
}
} }
libc_hidden_def (__pthread_disable_asynccancel) libc_hidden_def (__pthread_disable_asynccancel)

View File

@ -31,27 +31,9 @@ ___pthread_register_cancel_defer (__pthread_unwind_buf_t *buf)
ibuf->priv.data.prev = THREAD_GETMEM (self, cleanup_jmp_buf); ibuf->priv.data.prev = THREAD_GETMEM (self, cleanup_jmp_buf);
ibuf->priv.data.cleanup = THREAD_GETMEM (self, cleanup); ibuf->priv.data.cleanup = THREAD_GETMEM (self, cleanup);
int cancelhandling = THREAD_GETMEM (self, cancelhandling);
/* Disable asynchronous cancellation for now. */ /* Disable asynchronous cancellation for now. */
if (__glibc_unlikely (cancelhandling & CANCELTYPE_BITMASK)) ibuf->priv.data.canceltype = THREAD_GETMEM (self, canceltype);
while (1) THREAD_SETMEM (self, canceltype, PTHREAD_CANCEL_DEFERRED);
{
int curval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling,
cancelhandling
& ~CANCELTYPE_BITMASK,
cancelhandling);
if (__glibc_likely (curval == cancelhandling))
/* Successfully replaced the value. */
break;
/* Prepare for the next round. */
cancelhandling = curval;
}
ibuf->priv.data.canceltype = (cancelhandling & CANCELTYPE_BITMASK
? PTHREAD_CANCEL_ASYNCHRONOUS
: PTHREAD_CANCEL_DEFERRED);
/* Store the new cleanup handler info. */ /* Store the new cleanup handler info. */
THREAD_SETMEM (self, cleanup_jmp_buf, (struct pthread_unwind_buf *) buf); THREAD_SETMEM (self, cleanup_jmp_buf, (struct pthread_unwind_buf *) buf);
@ -73,27 +55,9 @@ ___pthread_unregister_cancel_restore (__pthread_unwind_buf_t *buf)
THREAD_SETMEM (self, cleanup_jmp_buf, ibuf->priv.data.prev); THREAD_SETMEM (self, cleanup_jmp_buf, ibuf->priv.data.prev);
int cancelhandling; THREAD_SETMEM (self, canceltype, ibuf->priv.data.canceltype);
if (ibuf->priv.data.canceltype != PTHREAD_CANCEL_DEFERRED if (ibuf->priv.data.canceltype == PTHREAD_CANCEL_ASYNCHRONOUS)
&& ((cancelhandling = THREAD_GETMEM (self, cancelhandling)) __pthread_testcancel ();
& CANCELTYPE_BITMASK) == 0)
{
while (1)
{
int curval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling,
cancelhandling
| CANCELTYPE_BITMASK,
cancelhandling);
if (__glibc_likely (curval == cancelhandling))
/* Successfully replaced the value. */
break;
/* Prepare for the next round. */
cancelhandling = curval;
}
__pthread_testcancel ();
}
} }
versioned_symbol (libc, ___pthread_unregister_cancel_restore, versioned_symbol (libc, ___pthread_unregister_cancel_restore,
__pthread_unregister_cancel_restore, GLIBC_2_34); __pthread_unregister_cancel_restore, GLIBC_2_34);

View File

@ -277,9 +277,6 @@ struct pthread
/* Flags determining processing of cancellation. */ /* Flags determining processing of cancellation. */
int cancelhandling; int cancelhandling;
/* Bit set if asynchronous cancellation mode is selected. */
#define CANCELTYPE_BIT 1
#define CANCELTYPE_BITMASK (0x01 << CANCELTYPE_BIT)
/* Bit set if canceled. */ /* Bit set if canceled. */
#define CANCELED_BIT 3 #define CANCELED_BIT 3
#define CANCELED_BITMASK (0x01 << CANCELED_BIT) #define CANCELED_BITMASK (0x01 << CANCELED_BIT)
@ -292,13 +289,6 @@ struct pthread
/* Bit set if thread is supposed to change XID. */ /* Bit set if thread is supposed to change XID. */
#define SETXID_BIT 6 #define SETXID_BIT 6
#define SETXID_BITMASK (0x01 << SETXID_BIT) #define SETXID_BITMASK (0x01 << SETXID_BIT)
/* Mask for the rest. Helps the compiler to optimize. */
#define CANCEL_RESTMASK 0xffffff80
#define CANCEL_CANCELED_AND_ASYNCHRONOUS(value) \
(((value) & (CANCELTYPE_BITMASK | CANCELED_BITMASK \
| EXITING_BITMASK | CANCEL_RESTMASK | TERMINATED_BITMASK)) \
== (CANCELTYPE_BITMASK | CANCELED_BITMASK))
/* Flags. Including those copied from the thread attribute. */ /* Flags. Including those copied from the thread attribute. */
int flags; int flags;
@ -402,6 +392,10 @@ struct pthread
PTHREAD_CANCEL_DISABLE). */ PTHREAD_CANCEL_DISABLE). */
unsigned char cancelstate; unsigned char cancelstate;
/* Thread cancel type (PTHREAD_CANCEL_DEFERRED or
PTHREAD_CANCEL_ASYNCHRONOUS). */
unsigned char canceltype;
/* Used on strsignal. */ /* Used on strsignal. */
struct tls_internal_t tls_state; struct tls_internal_t tls_state;

View File

@ -27,27 +27,9 @@ __libc_cleanup_push_defer (struct _pthread_cleanup_buffer *buffer)
buffer->__prev = THREAD_GETMEM (self, cleanup); buffer->__prev = THREAD_GETMEM (self, cleanup);
int cancelhandling = THREAD_GETMEM (self, cancelhandling);
/* Disable asynchronous cancellation for now. */ /* Disable asynchronous cancellation for now. */
if (__glibc_unlikely (cancelhandling & CANCELTYPE_BITMASK)) buffer->__canceltype = THREAD_GETMEM (self, canceltype);
while (1) THREAD_SETMEM (self, canceltype, PTHREAD_CANCEL_DEFERRED);
{
int curval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling,
cancelhandling
& ~CANCELTYPE_BITMASK,
cancelhandling);
if (__glibc_likely (curval == cancelhandling))
/* Successfully replaced the value. */
break;
/* Prepare for the next round. */
cancelhandling = curval;
}
buffer->__canceltype = (cancelhandling & CANCELTYPE_BITMASK
? PTHREAD_CANCEL_ASYNCHRONOUS
: PTHREAD_CANCEL_DEFERRED);
THREAD_SETMEM (self, cleanup, buffer); THREAD_SETMEM (self, cleanup, buffer);
} }
@ -60,26 +42,8 @@ __libc_cleanup_pop_restore (struct _pthread_cleanup_buffer *buffer)
THREAD_SETMEM (self, cleanup, buffer->__prev); THREAD_SETMEM (self, cleanup, buffer->__prev);
int cancelhandling; THREAD_SETMEM (self, canceltype, buffer->__canceltype);
if (__builtin_expect (buffer->__canceltype != PTHREAD_CANCEL_DEFERRED, 0) if (buffer->__canceltype == PTHREAD_CANCEL_ASYNCHRONOUS)
&& ((cancelhandling = THREAD_GETMEM (self, cancelhandling))
& CANCELTYPE_BITMASK) == 0)
{
while (1)
{
int curval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling,
cancelhandling
| CANCELTYPE_BITMASK,
cancelhandling);
if (__glibc_likely (curval == cancelhandling))
/* Successfully replaced the value. */
break;
/* Prepare for the next round. */
cancelhandling = curval;
}
__pthread_testcancel (); __pthread_testcancel ();
}
} }
libc_hidden_def (__libc_cleanup_pop_restore) libc_hidden_def (__libc_cleanup_pop_restore)

View File

@ -53,7 +53,7 @@ sigcancel_handler (int sig, siginfo_t *si, void *ctx)
/* Set the return value. */ /* Set the return value. */
THREAD_SETMEM (self, result, PTHREAD_CANCELED); THREAD_SETMEM (self, result, PTHREAD_CANCELED);
/* Make sure asynchronous cancellation is still enabled. */ /* Make sure asynchronous cancellation is still enabled. */
if ((ch & CANCELTYPE_BITMASK) != 0) if (self->canceltype == PTHREAD_CANCEL_ASYNCHRONOUS)
__do_cancel (); __do_cancel ();
} }
@ -104,8 +104,8 @@ __pthread_cancel (pthread_t th)
#endif #endif
THREAD_SETMEM (pd, result, PTHREAD_CANCELED); THREAD_SETMEM (pd, result, PTHREAD_CANCELED);
if ((oldch & CANCELSTATE_BITMASK) == 0 if (pd->cancelstate == PTHREAD_CANCEL_ENABLE
&& (oldch & CANCELTYPE_BITMASK) != 0) && pd->canceltype == PTHREAD_CANCEL_ASYNCHRONOUS)
__do_cancel (); __do_cancel ();
return 0; return 0;
} }

View File

@ -29,43 +29,11 @@ __pthread_setcanceltype (int type, int *oldtype)
volatile struct pthread *self = THREAD_SELF; volatile struct pthread *self = THREAD_SELF;
int oldval = THREAD_GETMEM (self, cancelhandling); if (oldtype != NULL)
while (1) *oldtype = self->canceltype;
{ self->canceltype = type;
int newval = (type == PTHREAD_CANCEL_ASYNCHRONOUS if (type == PTHREAD_CANCEL_ASYNCHRONOUS)
? oldval | CANCELTYPE_BITMASK __pthread_testcancel ();
: oldval & ~CANCELTYPE_BITMASK);
/* Store the old value. */
if (oldtype != NULL)
*oldtype = ((oldval & CANCELTYPE_BITMASK)
? PTHREAD_CANCEL_ASYNCHRONOUS : PTHREAD_CANCEL_DEFERRED);
/* Avoid doing unnecessary work. The atomic operation can
potentially be expensive if the memory has to be locked and
remote cache lines have to be invalidated. */
if (oldval == newval)
break;
/* Update the cancel handling word. This has to be done
atomically since other bits could be modified as well. */
int curval = THREAD_ATOMIC_CMPXCHG_VAL (self, cancelhandling, newval,
oldval);
if (__glibc_likely (curval == oldval))
{
if (self->cancelstate == PTHREAD_CANCEL_ENABLE
&& CANCEL_CANCELED_AND_ASYNCHRONOUS (newval))
{
THREAD_SETMEM (self, result, PTHREAD_CANCELED);
__do_cancel ();
}
break;
}
/* Prepare for the next round. */
oldval = curval;
}
return 0; return 0;
} }

View File

@ -96,4 +96,5 @@ __tls_init_tp (void)
THREAD_SETMEM (pd, stackblock_size, (size_t) __libc_stack_end); THREAD_SETMEM (pd, stackblock_size, (size_t) __libc_stack_end);
THREAD_SETMEM (pd, cancelstate, PTHREAD_CANCEL_ENABLE); THREAD_SETMEM (pd, cancelstate, PTHREAD_CANCEL_ENABLE);
THREAD_SETMEM (pd, canceltype, PTHREAD_CANCEL_DEFERRED);
} }