blob: 696cd4b59a124ffde12391708da9591738ae28e4 [file] [log] [blame]
bart55df6e72009-02-16 19:42:17 +00001/* Test whether detached threads are handled properly. */
bart25200ff2008-03-08 13:15:03 +00002
sewardjaf44c822007-11-25 14:01:38 +00003
4#include <assert.h>
5#include <pthread.h>
6#include <stdio.h>
7#include <stdlib.h>
8#include <unistd.h>
sewardjaf44c822007-11-25 14:01:38 +00009
bart25200ff2008-03-08 13:15:03 +000010
bart6c87f312008-03-16 07:46:36 +000011static int s_finished_count;
bart6c87f312008-03-16 07:46:36 +000012static pthread_mutex_t s_mutex;
sewardjaf44c822007-11-25 14:01:38 +000013
bart25200ff2008-03-08 13:15:03 +000014
barta547c822008-03-13 17:34:43 +000015static void increment_finished_count()
sewardjaf44c822007-11-25 14:01:38 +000016{
bart6c87f312008-03-16 07:46:36 +000017 pthread_mutex_lock(&s_mutex);
18 s_finished_count++;
19 pthread_mutex_unlock(&s_mutex);
20}
21
22static int get_finished_count()
23{
24 int result;
25 pthread_mutex_lock(&s_mutex);
26 result = s_finished_count;
27 pthread_mutex_unlock(&s_mutex);
28 return result;
sewardjaf44c822007-11-25 14:01:38 +000029}
30
31static void* thread_func1(void* arg)
32{
barta547c822008-03-13 17:34:43 +000033 write(STDOUT_FILENO, ".", 1);
sewardjaf44c822007-11-25 14:01:38 +000034 increment_finished_count();
35 return 0;
36}
37
38static void* thread_func2(void* arg)
39{
sewardjaf44c822007-11-25 14:01:38 +000040 pthread_detach(pthread_self());
barta547c822008-03-13 17:34:43 +000041 write(STDOUT_FILENO, ".", 1);
sewardjaf44c822007-11-25 14:01:38 +000042 increment_finished_count();
43 return 0;
44}
45
46int main(int argc, char** argv)
47{
48 const int count1 = argc > 1 ? atoi(argv[1]) : 100;
49 const int count2 = argc > 2 ? atoi(argv[2]) : 100;
50 int thread_arg[count1 > count2 ? count1 : count2];
51 int i;
52 int detachstate;
53 pthread_attr_t attr;
54
sewardjaf44c822007-11-25 14:01:38 +000055 for (i = 0; i < count1 || i < count2; i++)
56 thread_arg[i] = i;
57
bart6c87f312008-03-16 07:46:36 +000058 pthread_mutex_init(&s_mutex, 0);
bart31b983d2010-02-21 14:52:59 +000059
sewardjaf44c822007-11-25 14:01:38 +000060 pthread_attr_init(&attr);
61 pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
62 assert(pthread_attr_getdetachstate(&attr, &detachstate) == 0);
63 assert(detachstate == PTHREAD_CREATE_DETACHED);
64 pthread_attr_setstacksize(&attr, 16384);
bart31b983d2010-02-21 14:52:59 +000065 // Create count1 detached threads by setting the "detached" property via
sewardjaf44c822007-11-25 14:01:38 +000066 // thread attributes.
67 for (i = 0; i < count1; i++)
68 {
69 pthread_t thread;
70 pthread_create(&thread, &attr, thread_func1, &thread_arg[i]);
71 }
72 // Create count2 detached threads by letting the threads detach themselves.
73 pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
74 assert(pthread_attr_getdetachstate(&attr, &detachstate) == 0);
75 assert(detachstate == PTHREAD_CREATE_JOINABLE);
76 for (i = 0; i < count2; i++)
77 {
78 pthread_t thread;
79 pthread_create(&thread, &attr, thread_func2, &thread_arg[i]);
80 }
81 pthread_attr_destroy(&attr);
82
83 // Wait until all detached threads have written their output to stdout.
bart6c87f312008-03-16 07:46:36 +000084 while (get_finished_count() < count1 + count2)
sewardjaf44c822007-11-25 14:01:38 +000085 {
bart6c87f312008-03-16 07:46:36 +000086 struct timespec delay = { 0, 1 * 1000 * 1000 };
87 nanosleep(&delay, 0);
sewardjaf44c822007-11-25 14:01:38 +000088 }
89
barta547c822008-03-13 17:34:43 +000090 write(STDOUT_FILENO, "\n", 1);
sewardjaf44c822007-11-25 14:01:38 +000091
bart6c87f312008-03-16 07:46:36 +000092 pthread_mutex_destroy(&s_mutex);
sewardjaf44c822007-11-25 14:01:38 +000093
bart1169b792009-04-02 07:19:25 +000094 sleep(1);
95
sewardjaf44c822007-11-25 14:01:38 +000096 return 0;
97}