| /* | 
 |  * tsd1.c | 
 |  * | 
 |  * Test Thread Specific Data (TSD) key creation and destruction. | 
 |  * | 
 |  * | 
 |  * -------------------------------------------------------------------------- | 
 |  * | 
 |  *      Pthreads-win32 - POSIX Threads Library for Win32 | 
 |  *      Copyright(C) 1998 John E. Bossom | 
 |  *      Copyright(C) 1999,2005 Pthreads-win32 contributors | 
 |  *  | 
 |  *      Contact Email: rpj@callisto.canberra.edu.au | 
 |  *  | 
 |  *      The current list of contributors is contained | 
 |  *      in the file CONTRIBUTORS included with the source | 
 |  *      code distribution. The list can also be seen at the | 
 |  *      following World Wide Web location: | 
 |  *      http://sources.redhat.com/pthreads-win32/contributors.html | 
 |  *  | 
 |  *      This 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 of the License, or (at your option) any later version. | 
 |  *  | 
 |  *      This 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 this library in the file COPYING.LIB; | 
 |  *      if not, write to the Free Software Foundation, Inc., | 
 |  *      59 Temple Place - Suite 330, Boston, MA 02111-1307, USA | 
 |  * | 
 |  * | 
 |  * -------------------------------------------------------------------------- | 
 |  * | 
 |  * Description: | 
 |  * -  | 
 |  * | 
 |  * Test Method (validation or falsification): | 
 |  * - validation | 
 |  * | 
 |  * Requirements Tested: | 
 |  * - keys are created for each existing thread including the main thread | 
 |  * - keys are created for newly created threads | 
 |  * - keys are thread specific | 
 |  * - destroy routine is called on each thread exit including the main thread | 
 |  * | 
 |  * Features Tested: | 
 |  * -  | 
 |  * | 
 |  * Cases Tested: | 
 |  * -  | 
 |  * | 
 |  * Environment: | 
 |  * -  | 
 |  * | 
 |  * Input: | 
 |  * - none | 
 |  * | 
 |  * Output: | 
 |  * - text to stdout | 
 |  * | 
 |  * Assumptions: | 
 |  * - already validated:     pthread_create() | 
 |  *                          pthread_once() | 
 |  * - main thread also has a POSIX thread identity | 
 |  * | 
 |  * Pass Criteria: | 
 |  * - stdout matches file reference/tsd1.out | 
 |  * | 
 |  * Fail Criteria: | 
 |  * - fails to match file reference/tsd1.out | 
 |  * - output identifies failed component | 
 |  */ | 
 |  | 
 | //#include <sched.h> | 
 | #include "test.h" | 
 |  | 
 | enum { | 
 |   NUM_THREADS = 100 | 
 | }; | 
 |  | 
 | static pthread_key_t key /*= NULL*/; | 
 | static int accesscount[NUM_THREADS]; | 
 | static int thread_set[NUM_THREADS]; | 
 | static int thread_destroyed[NUM_THREADS]; | 
 | static pthread_barrier_t startBarrier; | 
 |  | 
 | static void | 
 | destroy_key(void * arg) | 
 | { | 
 |   int * j = (int *) arg; | 
 |  | 
 |   (*j)++; | 
 |  | 
 |   assert(*j == 2); | 
 |  | 
 |   thread_destroyed[j - accesscount] = 1; | 
 | } | 
 |  | 
 | static void | 
 | setkey(void * arg) | 
 | { | 
 |   int * j = (int *) arg; | 
 |  | 
 |   thread_set[j - accesscount] = 1; | 
 |  | 
 |   assert(*j == 0); | 
 |  | 
 |   assert(pthread_getspecific(key) == NULL); | 
 |  | 
 |   assert(pthread_setspecific(key, arg) == 0); | 
 |   assert(pthread_setspecific(key, arg) == 0); | 
 |   assert(pthread_setspecific(key, arg) == 0); | 
 |  | 
 |   assert(pthread_getspecific(key) == arg); | 
 |  | 
 |   (*j)++; | 
 |  | 
 |   assert(*j == 1); | 
 | } | 
 |  | 
 | static void * | 
 | mythread(void * arg) | 
 | { | 
 |   (void) pthread_barrier_wait(&startBarrier); | 
 |  | 
 |   setkey(arg); | 
 |  | 
 |   return 0; | 
 |  | 
 |   /* Exiting the thread will call the key destructor. */ | 
 | } | 
 |  | 
 | int | 
 | main() | 
 | { | 
 |   int i; | 
 |   int fail = 0; | 
 |   pthread_t thread[NUM_THREADS]; | 
 |  | 
 |   assert(pthread_barrier_init(&startBarrier, NULL, NUM_THREADS/2) == 0); | 
 |  | 
 |   for (i = 1; i < NUM_THREADS/2; i++) | 
 |     { | 
 |       accesscount[i] = thread_set[i] = thread_destroyed[i] = 0; | 
 |       assert(pthread_create(&thread[i], NULL, mythread, (void *)&accesscount[i]) == 0); | 
 |     } | 
 |  | 
 |   /* | 
 |    * Here we test that existing threads will get a key created | 
 |    * for them. | 
 |    */ | 
 |   assert(pthread_key_create(&key, destroy_key) == 0); | 
 |  | 
 |   (void) pthread_barrier_wait(&startBarrier); | 
 |  | 
 |   /* | 
 |    * Test main thread key. | 
 |    */ | 
 |   accesscount[0] = 0; | 
 |   setkey((void *) &accesscount[0]); | 
 |  | 
 |   /* | 
 |    * Here we test that new threads will get a key created | 
 |    * for them. | 
 |    */ | 
 |   for (i = NUM_THREADS/2; i < NUM_THREADS; i++) | 
 |     { | 
 |       accesscount[i] = thread_set[i] = thread_destroyed[i] = 0; | 
 |       assert(pthread_create(&thread[i], NULL, mythread, (void *)&accesscount[i]) == 0); | 
 |     } | 
 |  | 
 |   /* | 
 |    * Wait for all threads to complete. | 
 |    */ | 
 |   for (i = 1; i < NUM_THREADS; i++) | 
 |     { | 
 | 	intptr_t result = 0; | 
 |  | 
 | 	assert(pthread_join(thread[i], (void **) &result) == 0); | 
 |     } | 
 |  | 
 |   assert(pthread_key_delete(key) == 0); | 
 |  | 
 |   assert(pthread_barrier_destroy(&startBarrier) == 0); | 
 |  | 
 |   for (i = 1; i < NUM_THREADS; i++) | 
 |     { | 
 | 	/* | 
 | 	 * The counter is incremented once when the key is set to | 
 | 	 * a value, and again when the key is destroyed. If the key | 
 | 	 * doesn't get set for some reason then it will still be | 
 | 	 * NULL and the destroy function will not be called, and | 
 | 	 * hence accesscount will not equal 2. | 
 | 	 */ | 
 | 	if (accesscount[i] != 2) | 
 | 	  { | 
 | 	    fail++; | 
 | 	    fprintf(stderr, "Thread %d key, set = %d, destroyed = %d\n", | 
 | 			i, thread_set[i], thread_destroyed[i]); | 
 | 	  } | 
 |     } | 
 |  | 
 |   fflush(stderr); | 
 |  | 
 |   return (fail); | 
 | } |