| /** Initialize several kinds of mutexes and lock each mutex twice. |
| * Note: locking a regular mutex twice causes a deadlock. |
| */ |
| |
| #define _GNU_SOURCE |
| |
| #include <stdio.h> |
| #include <unistd.h> |
| #include <pthread.h> |
| #include "../../config.h" |
| |
| |
| static void lock_twice(pthread_mutex_t* const p) |
| { |
| if (pthread_mutex_trylock(p) != 0) |
| fprintf(stderr, "first lock call failed !\n"); |
| if (pthread_mutex_trylock(p) != 0) |
| fprintf(stderr, "second lock call failed !\n"); |
| if (pthread_mutex_unlock(p) != 0) |
| fprintf(stderr, "first unlock call failed !\n"); |
| if (pthread_mutex_unlock(p) != 0) |
| fprintf(stderr, "second unlock call failed !\n"); |
| } |
| |
| int main(int argc, char** argv) |
| { |
| #if defined(HAVE_PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP) |
| { |
| pthread_mutex_t m = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP; |
| |
| fprintf(stderr, "Recursive mutex (statically initialized).\n"); |
| lock_twice(&m); |
| pthread_mutex_destroy(&m); |
| } |
| #endif |
| #if defined(HAVE_PTHREAD_MUTEX_RECURSIVE_NP) |
| { |
| pthread_mutex_t m; |
| pthread_mutexattr_t attr; |
| |
| fprintf(stderr, "\nRecursive mutex (initialized via mutex attributes).\n"); |
| pthread_mutexattr_init(&attr); |
| pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE_NP); |
| pthread_mutex_init(&m, &attr); |
| pthread_mutexattr_destroy(&attr); |
| lock_twice(&m); |
| pthread_mutex_destroy(&m); |
| } |
| #endif |
| #if defined(HAVE_PTHREAD_MUTEX_ERRORCHECK_NP) |
| { |
| pthread_mutex_t m; |
| pthread_mutexattr_t attr; |
| |
| fprintf(stderr, "\nError checking mutex.\n"); |
| pthread_mutexattr_init(&attr); |
| pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK_NP); |
| pthread_mutex_init(&m, &attr); |
| pthread_mutexattr_destroy(&attr); |
| lock_twice(&m); |
| pthread_mutex_destroy(&m); |
| } |
| #endif |
| |
| { |
| pthread_mutex_t m = PTHREAD_MUTEX_INITIALIZER; |
| |
| fprintf(stderr, "\nNon-recursive mutex.\n"); |
| lock_twice(&m); |
| } |
| |
| fprintf(stderr, "\nDone.\n"); |
| |
| return 0; |
| } |