Update copyright to LGPL on all files
[dyninst.git] / testsuite / src / dyninst / test_thread.c
1 /*
2  * Copyright (c) 1996-2009 Barton P. Miller
3  * 
4  * We provide the Paradyn Parallel Performance Tools (below
5  * described as "Paradyn") on an AS IS basis, and do not warrant its
6  * validity or performance.  We reserve the right to update, modify,
7  * or discontinue this software at any time.  We shall have no
8  * obligation to supply such updates or modifications or any other
9  * form of support to you.
10  * 
11  * By your use of Paradyn, you understand and agree that we (or any
12  * other person or entity with proprietary rights in Paradyn) are
13  * under no obligation to provide either maintenance services,
14  * update services, notices of latent defects, or correction of
15  * defects for Paradyn.
16  * 
17  * This library is free software; you can redistribute it and/or
18  * modify it under the terms of the GNU Lesser General Public
19  * License as published by the Free Software Foundation; either
20  * version 2.1 of the License, or (at your option) any later version.
21  * 
22  * This library is distributed in the hope that it will be useful,
23  * but WITHOUT ANY WARRANTY; without even the implied warranty of
24  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
25  * Lesser General Public License for more details.
26  * 
27  * You should have received a copy of the GNU Lesser General Public
28  * License along with this library; if not, write to the Free Software
29  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
30  */
31 #include <assert.h>
32 #include <errno.h>
33 #include <pthread.h>
34 #include <stdlib.h>
35
36 #include "mutatee_util.h"
37 #include "test_thread.h"
38
39 void sleep_ms(int ms) 
40 {
41   struct timespec ts,rem;
42   ts.tv_sec = 0;
43   ts.tv_nsec = ms * 1000 /*us*/ * 1000/*ms*/;
44   assert(ms < 1000);
45
46  sleep:
47  if (0 != nanosleep(&ts, &rem)) {
48     if (errno == EINTR) {
49       dprintf("%s[%d]:  sleep interrupted\n", __FILE__, __LINE__);
50       ts.tv_sec = rem.tv_sec;
51       ts.tv_nsec = rem.tv_nsec;
52       goto sleep;
53     }
54     sleep(1);
55  }
56
57 }
58
59 #if defined (os_windows_test)
60 #error
61 #else
62   /*
63     createThreads()
64     createThreads creates specified number of threads and returns
65     a pointer to an allocated buffer that contains their handles
66     caller is responsible for free'ing the result
67   */
68
69 pthread_attr_t attr;
70
71 Thread_t *createThreads(unsigned int num, ThreadMain_t tmain, Thread_t *tbuf)
72 {
73   unsigned int i;
74   int err = 0;
75   Thread_t *threads;
76   if (tbuf == NULL)
77     threads = (Thread_t *)malloc(num * sizeof(Thread_t));
78   else 
79     threads = tbuf;
80     
81   if (!threads) {
82     logerror("%s[%d]:  could not alloc space for %d thread handles\n",
83             __FILE__, __LINE__, num);
84     return NULL;
85   }
86
87   if (0 != pthread_attr_init(&attr)) {
88     err = 1;
89     perror("pthread_attr_init");
90     goto cleanup;
91   }
92
93   if (0 != pthread_attr_setdetachstate( &attr, PTHREAD_CREATE_DETACHED)) {
94     err = 1;
95     perror("pthread_attr_setdetachstate");
96     goto cleanup;
97   }
98
99   if (0 != pthread_attr_setscope(&attr, PTHREAD_SCOPE_SYSTEM)) {
100     err = 1;
101     perror("pthread_attr_setscope");
102     goto cleanup;
103   }
104
105   /* start a bunch of threads */
106   for (i = 0; i < num; ++i) {
107
108     if (0 != pthread_create((pthread_t *) &threads[i], 
109                             &attr, (void *(*)(void*))tmain, NULL))    
110     {
111       err = 1;
112       logerror("%s[%d]:pthread_create\n", __FILE__, __LINE__);
113       goto cleanup;
114     }
115     dprintf("%s[%d]:  PTHREAD_CREATE: %lu\n", __FILE__, __LINE__, 
116             (unsigned long)threads[i]); 
117   }
118
119   cleanup:
120
121   if (err) {
122     if (!tbuf) free (threads);
123     return NULL;
124   }
125
126   return threads;
127 }
128
129 int createLock(Lock_t *lock)
130 {
131   if (debugPrint)
132     dprintf("%s[%d]:  creating lock on thread %lu\n", __FILE__, __LINE__, (unsigned long) pthread_self());
133   if (0 != pthread_mutex_init(lock, NULL)) {
134     perror("pthread_mutex_init");
135     return FALSE;
136   }
137   return TRUE;
138 }
139
140 int destroyLock(Lock_t *lock)
141 {
142   if (debugPrint)
143     dprintf("%s[%d]:  destroying lock on thread %lu\n", __FILE__, __LINE__, (unsigned long) pthread_self());
144   if (0 != pthread_mutex_destroy(lock)) {
145     perror("pthread_mutex_destroy");
146     return FALSE;
147   }
148   return TRUE;
149 }
150
151 int lockLock(Lock_t *lock)
152 {
153   if (debugPrint)
154     dprintf("%s[%d]:  locking lock on thread %lu\n", __FILE__, __LINE__, (unsigned long) pthread_self());
155   if (0 != pthread_mutex_lock(lock)) {
156     perror("pthread_mutex_lock");
157     return FALSE;
158   }
159   return TRUE;
160 }
161
162 int unlockLock(Lock_t *lock)
163 {
164   if (debugPrint)
165     dprintf("%s[%d]:  unlocking lock on thread %lu\n", __FILE__, __LINE__, (unsigned long) pthread_self());
166   if (0 != pthread_mutex_unlock(lock)) {
167     perror("pthread_mutex_unlock");
168     return FALSE;
169   }
170   return TRUE;
171 }
172
173 #endif