• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * File: exit5.c
3  *
4  *
5  * --------------------------------------------------------------------------
6  *
7  *      Pthreads-win32 - POSIX Threads Library for Win32
8  *      Copyright(C) 1998 John E. Bossom
9  *      Copyright(C) 1999,2005 Pthreads-win32 contributors
10  *
11  *      Contact Email: rpj@callisto.canberra.edu.au
12  *
13  *      The current list of contributors is contained
14  *      in the file CONTRIBUTORS included with the source
15  *      code distribution. The list can also be seen at the
16  *      following World Wide Web location:
17  *      http://sources.redhat.com/pthreads-win32/contributors.html
18  *
19  *      This library is free software; you can redistribute it and/or
20  *      modify it under the terms of the GNU Lesser General Public
21  *      License as published by the Free Software Foundation; either
22  *      version 2 of the License, or (at your option) any later version.
23  *
24  *      This library is distributed in the hope that it will be useful,
25  *      but WITHOUT ANY WARRANTY; without even the implied warranty of
26  *      MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
27  *      Lesser General Public License for more details.
28  *
29  *      You should have received a copy of the GNU Lesser General Public
30  *      License along with this library in the file COPYING.LIB;
31  *      if not, write to the Free Software Foundation, Inc.,
32  *      59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
33  *
34  * --------------------------------------------------------------------------
35  *
36  * Test Synopsis: Test calling pthread_exit from a Win32 thread
37  *                having created an implicit POSIX handle for it.
38  *
39  * Test Method (Validation or Falsification):
40  * - Validate return value and that POSIX handle is created and destroyed.
41  *
42  * Requirements Tested:
43  * -
44  *
45  * Features Tested:
46  * -
47  *
48  * Cases Tested:
49  * -
50  *
51  * Description:
52  * -
53  *
54  * Environment:
55  * -
56  *
57  * Input:
58  * - None.
59  *
60  * Output:
61  * - File name, Line number, and failed expression on failure.
62  * - No output on success.
63  *
64  * Assumptions:
65  * - have working pthread_create, pthread_self, pthread_mutex_lock/unlock
66  *   pthread_testcancel, pthread_cancel, pthread_join
67  *
68  * Pass Criteria:
69  * - Process returns zero exit status.
70  *
71  * Fail Criteria:
72  * - Process returns non-zero exit status.
73  */
74 
75 #include "test.h"
76 #ifndef _UWIN
77 #include <process.h>
78 #endif
79 
80 /*
81  * Create NUMTHREADS threads in addition to the Main thread.
82  */
83 enum {
84   NUMTHREADS = 4
85 };
86 
87 typedef struct bag_t_ bag_t;
88 struct bag_t_ {
89   int threadnum;
90   int started;
91   /* Add more per-thread state variables here */
92   int count;
93   pthread_t self;
94 };
95 
96 static bag_t threadbag[NUMTHREADS + 1];
97 
98 unsigned int __stdcall
Win32thread(void * arg)99 Win32thread(void * arg)
100 {
101   int result = 1;
102   bag_t * bag = (bag_t *) arg;
103 
104   assert(bag == &threadbag[bag->threadnum]);
105   assert(bag->started == 0);
106   bag->started = 1;
107 
108   assert((bag->self = pthread_self()) != 0);
109   assert(pthread_gethandle (bag->self) != NULL);
110   assert(pthread_kill(bag->self, 0) == 0);
111 
112   /*
113    * Doesn't return.
114    */
115   pthread_exit(((void *) (size_t) result));
116 
117   return 0;
118 }
119 
120 int
main()121 main()
122 {
123   int failed = 0;
124   int i;
125   HANDLE h[NUMTHREADS + 1];
126   unsigned thrAddr; /* Dummy variable to pass a valid location to _beginthreadex (Win98). */
127 
128   for (i = 1; i <= NUMTHREADS; i++)
129     {
130       threadbag[i].started = 0;
131       threadbag[i].threadnum = i;
132       h[i] = (HANDLE) _beginthreadex(NULL, 0, Win32thread, (void *) &threadbag[i], 0, &thrAddr);
133     }
134 
135   /*
136    * Code to control or munipulate child threads should probably go here.
137    */
138   Sleep(500);
139 
140   /*
141    * Give threads time to run.
142    */
143   Sleep(NUMTHREADS * 100);
144 
145   /*
146    * Standard check that all threads started.
147    */
148   for (i = 1; i <= NUMTHREADS; i++)
149     {
150       if (!threadbag[i].started)
151 	{
152 	  failed |= !threadbag[i].started;
153 	  fprintf(stderr, "Thread %d: started %d\n", i, threadbag[i].started);
154 	}
155     }
156 
157   assert(!failed);
158 
159   /*
160    * Check any results here. Set "failed" and only print output on failure.
161    */
162   failed = 0;
163   for (i = 1; i <= NUMTHREADS; i++)
164     {
165       int fail = 0;
166       int result = 0;
167 
168       assert(GetExitCodeThread(h[i], (LPDWORD) &result) == TRUE);
169 
170       assert(pthread_kill(threadbag[i].self, 0) == ESRCH);
171 
172       fail = (result != 1);
173 
174       if (fail)
175 	{
176 	  fprintf(stderr, "Thread %d: started %d: count %d\n",
177 		  i,
178 		  threadbag[i].started,
179 		  threadbag[i].count);
180 	}
181       failed = (failed || fail);
182     }
183 
184   assert(!failed);
185 
186   /*
187    * Success.
188    */
189   return 0;
190 }
191 
192