#include <pthread.h> #include <semaphore.h> #include <stdlib.h> #include <stdio.h> #include <unistd.h> static volatile bool destr_called; static volatile bool except_caught; static pthread_barrier_t b; struct monitor { // gcc is broken and would generate a warning without this dummy // constructor. monitor () { } ~monitor() { destr_called = true; } }; static void * tf (void *arg) { sem_t *s = static_cast<sem_t *> (arg); try { monitor m; pthread_barrier_wait (&b); while (1) sem_wait (s); } catch (...) { except_caught = true; throw; } return NULL; } static int do_test () { if (pthread_barrier_init (&b, NULL, 2) != 0) { puts ("barrier_init failed"); return 1; } sem_t s; if (sem_init (&s, 0, 0) != 0) { puts ("sem_init failed"); return 1; } pthread_t th; if (pthread_create (&th, NULL, tf, &s) != 0) { puts ("pthread_create failed"); return 1; } pthread_barrier_wait (&b); /* There is unfortunately no better method to try to assure the child thread reached the sem_wait call and is actually waiting than to sleep here. */ sleep (1); if (pthread_cancel (th) != 0) { puts ("cancel failed"); return 1; } void *res; if (pthread_join (th, &res) != 0) { puts ("join failed"); return 1; } if (res != PTHREAD_CANCELED) { puts ("thread was not canceled"); return 1; } if (! except_caught) { puts ("exception not caught"); return 1; } if (! destr_called) { puts ("destructor not called"); return 1; } return 0; } #define TEST_FUNCTION do_test () #define TIMEOUT 3 #include "../test-skeleton.c"