Мой рекурсивный мьютекс против pthread_mutex_t (тип: рекурсивный) (repost, push)
Мне было интересно, если бы я мог сделать рекурсивный тип мьютекса самостоятельно с PTHREAD_MUTEX_ERRORCHECK
мьютекс, это результат:
typedef struct {
pthread_mutex_t mutex;
uint32_t deadlocks;
} pthread_recursivemutex_t;
int pthread_recursivemutex_init(pthread_recursivemutex_t *mutex)
{
int ret;
pthread_mutexattr_t attr;
mutex->deadlocks = 0;
ret = pthread_mutexattr_init(&attr);
if (ret != 0) {
return ret;
}
(void)pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK);
ret = pthread_mutex_init(&mutex->mutex, &attr);
(void)pthread_mutexattr_destroy(&attr);
return ret;
}
void pthread_recursivemutex_lock(pthread_recursivemutex_t *mutex)
{
int ret;
ret = pthread_mutex_lock(&mutex->mutex);
if (ret == 0) {
mutex->deadlocks = 0;
} else if (ret == EDEADLK) {
mutex->deadlocks += 1;
}
}
void pthread_recursivemutex_unlock(pthread_recursivemutex_t *mutex)
{
if (mutex->deadlocks == 0) {
(void)pthread_mutex_unlock(&mutex->mutex);
} else {
mutex->deadlocks -= 1;
}
}
void pthread_recursivemutex_destroy(pthread_recursivemutex_t *mutex)
{
(void)pthread_mutex_destroy(&mutex->mutex);
}
Я обнаружил, что этот тип рекурсивного мьютекса намного быстрее, чем мьютекс с PTHREAD_MUTEX_RECURSIVE
атрибут:
iterations : 1000000
pthread_mutex_t : 71757 μSeconds
pthread_recursivemutex_t : 48583 μSeconds
Тестовый код (каждый вызывается 1000000 раз):
void mutex_test()
{
pthread_mutex_lock(&recursiveMutex);
pthread_mutex_lock(&recursiveMutex);
pthread_mutex_unlock(&recursiveMutex);
pthread_mutex_unlock(&recursiveMutex);
}
void recursivemutex_test()
{
pthread_recursivemutex_lock(&myMutex);
pthread_recursivemutex_lock(&myMutex);
pthread_recursivemutex_unlock(&myMutex);
pthread_recursivemutex_unlock(&myMutex);
}
pthread_recursivemutex_t
почти в два раза быстрее, чем pthread_mutex_t
?! Но оба ведут себя одинаково...?
КСТАТИ:
pthread_mutex_t my_mutex;
void *m(void*a)
{
printf("m: %d %d\n", pthread_mutex_lock(&my_mutex), EDEADLK);
usleep(500 * 1000);
printf("m: %d\n", pthread_mutex_lock(&my_mutex));
pthread_mutex_unlock(&my_mutex);
return a;
}
void *m2(void *a)
{
printf("UNLOCKED!!: %d\n", pthread_mutex_lock(&my_mutex));
usleep(500 * 1000);
printf("m2: %d\n", pthread_mutex_lock(&my_mutex));
return a;
}
int main(int argc, const char * argv[])
{
pthread_t t,t2;
pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK);
pthread_mutex_init(&my_mutex, &attr);
pthread_create(&t, NULL, m, NULL);
pthread_create(&t2, NULL, m2, NULL);
pthread_join(t, NULL);
pthread_join(t2, NULL);
return 0;
}
дает
m: 0 11
m: 11
UNLOCKED!!: 0
m2: 11
Это означает, что поток 2 не имеет возможности получить мьютекс, пока он заблокирован.