| /* Verify that exception table for pthread_cond_wait is correct. | 
 |    Copyright (C) 2012-2016 Free Software Foundation, Inc. | 
 |    This file is part of the GNU C Library. | 
 |  | 
 |    The GNU C Library is free software; you can redistribute it and/or | 
 |    modify it under the terms of the GNU Lesser General Public | 
 |    License as published by the Free Software Foundation; either | 
 |    version 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |    The GNU C Library is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
 |    Lesser General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU Lesser General Public | 
 |    License along with the GNU C Library; if not, see | 
 |    <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include <pthread.h> | 
 | #include <stdio.h> | 
 | #include <stdint.h> | 
 | #include <string.h> | 
 | #include <unistd.h> | 
 |  | 
 | pthread_mutex_t mutex; | 
 | pthread_cond_t cond; | 
 |  | 
 | #define CHECK_RETURN_VAL_OR_FAIL(ret,str) \ | 
 |   ({ if ((ret) != 0) \ | 
 |        { \ | 
 |          printf ("%s failed: %s\n", (str), strerror (ret)); \ | 
 |          ret = 1; \ | 
 |          goto out; \ | 
 |        } \ | 
 |   }) | 
 |  | 
 |  | 
 | void | 
 | clean (void *arg) | 
 | { | 
 |   puts ("clean: Unlocking mutex..."); | 
 |   pthread_mutex_unlock ((pthread_mutex_t *) arg); | 
 |   puts ("clean: Mutex unlocked..."); | 
 | } | 
 |  | 
 | void * | 
 | thr (void *arg) | 
 | { | 
 |   int ret = 0; | 
 |   pthread_mutexattr_t mutexAttr; | 
 |   ret = pthread_mutexattr_init (&mutexAttr); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_mutexattr_init"); | 
 |  | 
 |   ret = pthread_mutexattr_setprotocol (&mutexAttr, PTHREAD_PRIO_INHERIT); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_mutexattr_setprotocol"); | 
 |  | 
 |   ret = pthread_mutex_init (&mutex, &mutexAttr); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_mutex_init"); | 
 |  | 
 |   ret = pthread_cond_init (&cond, 0); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_cond_init"); | 
 |  | 
 |   puts ("th: Init done, entering wait..."); | 
 |  | 
 |   pthread_cleanup_push (clean, (void *) &mutex); | 
 |   ret = pthread_mutex_lock (&mutex); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_mutex_lock"); | 
 |   while (1) | 
 |     { | 
 |       ret = pthread_cond_wait (&cond, &mutex); | 
 |       CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_cond_wait"); | 
 |     } | 
 |   pthread_cleanup_pop (1); | 
 |  | 
 | out: | 
 |   return (void *) (uintptr_t) ret; | 
 | } | 
 |  | 
 | int | 
 | do_test (void) | 
 | { | 
 |   pthread_t thread; | 
 |   int ret = 0; | 
 |   void *thr_ret = 0; | 
 |   ret = pthread_create (&thread, 0, thr, &thr_ret); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_create"); | 
 |  | 
 |   puts ("main: Thread created, waiting a bit..."); | 
 |   sleep (2); | 
 |  | 
 |   puts ("main: Cancelling thread..."); | 
 |   ret = pthread_cancel (thread); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_cancel"); | 
 |  | 
 |   puts ("main: Joining th..."); | 
 |   ret = pthread_join (thread, NULL); | 
 |   CHECK_RETURN_VAL_OR_FAIL (ret, "pthread_join"); | 
 |  | 
 |   if (thr_ret != NULL) | 
 |     return 1; | 
 |  | 
 |   puts ("main: Joined thread, done!"); | 
 |  | 
 | out: | 
 |   return ret; | 
 | } | 
 |  | 
 | #define TEST_FUNCTION do_test () | 
 | #define TIMEOUT 5 | 
 | #include "../test-skeleton.c" |