mirror of
https://sourceware.org/git/glibc.git
synced 2025-07-28 00:21:52 +03:00
* sysdeps/powerpc/powerpc32/dl-trampoline.S (_dl_runtime_resolve):
Don't clobber caller's LRSAVE. (_dl_prof_resolve): Likewise.
This commit is contained in:
@ -335,6 +335,90 @@ __pthread_mutex_lock (mutex)
|
||||
}
|
||||
break;
|
||||
|
||||
case PTHREAD_MUTEX_PP_RECURSIVE_NP:
|
||||
case PTHREAD_MUTEX_PP_ERRORCHECK_NP:
|
||||
case PTHREAD_MUTEX_PP_NORMAL_NP:
|
||||
case PTHREAD_MUTEX_PP_ADAPTIVE_NP:
|
||||
{
|
||||
int kind = mutex->__data.__kind & PTHREAD_MUTEX_KIND_MASK_NP;
|
||||
|
||||
oldval = mutex->__data.__lock;
|
||||
|
||||
/* Check whether we already hold the mutex. */
|
||||
if (mutex->__data.__owner == id)
|
||||
{
|
||||
if (kind == PTHREAD_MUTEX_ERRORCHECK_NP)
|
||||
return EDEADLK;
|
||||
|
||||
if (kind == PTHREAD_MUTEX_RECURSIVE_NP)
|
||||
{
|
||||
/* Just bump the counter. */
|
||||
if (__builtin_expect (mutex->__data.__count + 1 == 0, 0))
|
||||
/* Overflow of the counter. */
|
||||
return EAGAIN;
|
||||
|
||||
++mutex->__data.__count;
|
||||
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
int oldprio = -1, ceilval;
|
||||
do
|
||||
{
|
||||
int ceiling = (oldval & PTHREAD_MUTEX_PRIO_CEILING_MASK)
|
||||
>> PTHREAD_MUTEX_PRIO_CEILING_SHIFT;
|
||||
|
||||
if (__pthread_current_priority () > ceiling)
|
||||
{
|
||||
if (oldprio != -1)
|
||||
__pthread_tpp_change_priority (oldprio, -1);
|
||||
return EINVAL;
|
||||
}
|
||||
|
||||
retval = __pthread_tpp_change_priority (oldprio, ceiling);
|
||||
if (retval)
|
||||
return retval;
|
||||
|
||||
ceilval = ceiling << PTHREAD_MUTEX_PRIO_CEILING_SHIFT;
|
||||
oldprio = ceiling;
|
||||
|
||||
oldval
|
||||
= atomic_compare_and_exchange_val_acq (&mutex->__data.__lock,
|
||||
#ifdef NO_INCR
|
||||
ceilval | 2,
|
||||
#else
|
||||
ceilval | 1,
|
||||
#endif
|
||||
ceilval);
|
||||
|
||||
if (oldval == ceilval)
|
||||
break;
|
||||
|
||||
do
|
||||
{
|
||||
oldval
|
||||
= atomic_compare_and_exchange_val_acq (&mutex->__data.__lock,
|
||||
ceilval | 2,
|
||||
ceilval | 1);
|
||||
|
||||
if ((oldval & PTHREAD_MUTEX_PRIO_CEILING_MASK) != ceilval)
|
||||
break;
|
||||
|
||||
if (oldval != ceilval)
|
||||
lll_futex_wait (&mutex->__data.__lock, ceilval | 2);
|
||||
}
|
||||
while (atomic_compare_and_exchange_val_acq (&mutex->__data.__lock,
|
||||
ceilval | 2, ceilval)
|
||||
!= ceilval);
|
||||
}
|
||||
while ((oldval & PTHREAD_MUTEX_PRIO_CEILING_MASK) != ceilval);
|
||||
|
||||
assert (mutex->__data.__owner == 0);
|
||||
mutex->__data.__count = 1;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
/* Correct code cannot set any other type. */
|
||||
return EINVAL;
|
||||
|
Reference in New Issue
Block a user