/* Test whether detached threads are handled properly. */ #include <assert.h> #include <pthread.h> #include <stdio.h> #include <stdlib.h> #include <unistd.h> static int s_finished_count; static pthread_mutex_t s_mutex; static void increment_finished_count() { pthread_mutex_lock(&s_mutex); s_finished_count++; pthread_mutex_unlock(&s_mutex); } static int get_finished_count() { int result; pthread_mutex_lock(&s_mutex); result = s_finished_count; pthread_mutex_unlock(&s_mutex); return result; } static void* thread_func1(void* arg) { write(STDOUT_FILENO, ".", 1); increment_finished_count(); return 0; } static void* thread_func2(void* arg) { pthread_detach(pthread_self()); write(STDOUT_FILENO, ".", 1); increment_finished_count(); return 0; } int main(int argc, char** argv) { const int count1 = argc > 1 ? atoi(argv[1]) : 100; const int count2 = argc > 2 ? atoi(argv[2]) : 100; int thread_arg[count1 > count2 ? count1 : count2]; int i; int detachstate; pthread_attr_t attr; for (i = 0; i < count1 || i < count2; i++) thread_arg[i] = i; pthread_mutex_init(&s_mutex, 0); pthread_attr_init(&attr); pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); assert(pthread_attr_getdetachstate(&attr, &detachstate) == 0); assert(detachstate == PTHREAD_CREATE_DETACHED); pthread_attr_setstacksize(&attr, 16384); // Create count1 detached threads by setting the "detached" property via // thread attributes. for (i = 0; i < count1; i++) { pthread_t thread; pthread_create(&thread, &attr, thread_func1, &thread_arg[i]); } // Create count2 detached threads by letting the threads detach themselves. pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE); assert(pthread_attr_getdetachstate(&attr, &detachstate) == 0); assert(detachstate == PTHREAD_CREATE_JOINABLE); for (i = 0; i < count2; i++) { pthread_t thread; pthread_create(&thread, &attr, thread_func2, &thread_arg[i]); } pthread_attr_destroy(&attr); // Wait until all detached threads have written their output to stdout. while (get_finished_count() < count1 + count2) { struct timespec delay = { 0, 1 * 1000 * 1000 }; nanosleep(&delay, 0); } write(STDOUT_FILENO, "\n", 1); pthread_mutex_destroy(&s_mutex); sleep(1); return 0; }