1 /*
2 * Copyright (c) 2004, Bull S.A.. All rights reserved.
3 * Created by: Sebastien Decugis
4
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of version 2 of the GNU General Public License as
7 * published by the Free Software Foundation.
8 *
9 * This program is distributed in the hope that it would be useful, but
10 * WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12 *
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write the Free Software Foundation, Inc.,
15 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
16
17 * This sample test aims to check the following assertion:
18 *
19 * When the abstime parameter is invalid,
20 * the function must return EINVAL and
21 * the mutex state must not have changed during the call.
22
23 * The steps are:
24 * -> parent (for each mutex type and each condvar options, across threads or processes)
25 * -> locks the mutex m
26 * -> sets ctrl = 0
27 * -> creates a bunch of children, which:
28 * -> lock the mutex m
29 * -> if ctrl == 0, test has failed
30 * -> unlock the mutex then exit
31 * -> calls pthread_cond_timedwait with invalid values (nsec > 999999999)
32 * -> sets ctrl = non-zero value
33 * -> unlocks the mutex m
34 */
35
36 #include <pthread.h>
37 #include <stdarg.h>
38 #include <stdio.h>
39 #include <stdlib.h>
40 #include <unistd.h>
41
42 #include <errno.h>
43 #include <sys/wait.h>
44 #include <sys/mman.h>
45 #include <string.h>
46 #include <time.h>
47
48 #include "../testfrmw/testfrmw.h"
49 #include "../testfrmw/testfrmw.c"
50
51 #ifndef VERBOSE
52 #define VERBOSE 1
53 #endif
54
55 #define NCHILDREN (20)
56
57 #ifndef WITHOUT_ALTCLK
58 #define USE_ALTCLK /* make tests with MONOTONIC CLOCK if supported */
59 #endif
60
61 #ifndef WITHOUT_XOPEN
62
63 typedef struct {
64 pthread_mutex_t mtx;
65 int ctrl; /* Control value */
66 int gotit; /* Thread locked the mutex while ctrl == 0 */
67 int status; /* error code */
68 } testdata_t;
69
70 static struct _scenar {
71 int m_type; /* Mutex type to use */
72 int mc_pshared; /* 0: mutex and cond are process-private (default) ~ !0: Both are process-shared, if supported */
73 int c_clock; /* 0: cond uses the default clock. ~ !0: Cond uses monotonic clock, if supported. */
74 int fork; /* 0: Test between threads. ~ !0: Test across processes, if supported (mmap) */
75 char *descr; /* Case description */
76 } scenarii[] = {
77 {
78 PTHREAD_MUTEX_DEFAULT, 0, 0, 0, "Default mutex"}
79 , {
80 PTHREAD_MUTEX_NORMAL, 0, 0, 0, "Normal mutex"}
81 , {
82 PTHREAD_MUTEX_ERRORCHECK, 0, 0, 0, "Errorcheck mutex"}
83 , {
84 PTHREAD_MUTEX_RECURSIVE, 0, 0, 0, "Recursive mutex"}
85
86 , {
87 PTHREAD_MUTEX_DEFAULT, 1, 0, 0, "PShared default mutex"}
88 , {
89 PTHREAD_MUTEX_NORMAL, 1, 0, 0, "Pshared normal mutex"}
90 , {
91 PTHREAD_MUTEX_ERRORCHECK, 1, 0, 0, "Pshared errorcheck mutex"}
92 , {
93 PTHREAD_MUTEX_RECURSIVE, 1, 0, 0, "Pshared recursive mutex"}
94
95 , {
96 PTHREAD_MUTEX_DEFAULT, 1, 0, 1,
97 "Pshared default mutex across processes"}
98 , {
99 PTHREAD_MUTEX_NORMAL, 1, 0, 1,
100 "Pshared normal mutex across processes"}
101 , {
102 PTHREAD_MUTEX_ERRORCHECK, 1, 0, 1,
103 "Pshared errorcheck mutex across processes"}
104 , {
105 PTHREAD_MUTEX_RECURSIVE, 1, 0, 1,
106 "Pshared recursive mutex across processes"}
107
108 #ifdef USE_ALTCLK
109 , {
110 PTHREAD_MUTEX_DEFAULT, 1, 1, 1,
111 "Pshared default mutex and alt clock condvar across processes"}
112 , {
113 PTHREAD_MUTEX_NORMAL, 1, 1, 1,
114 "Pshared normal mutex and alt clock condvar across processes"}
115 , {
116 PTHREAD_MUTEX_ERRORCHECK, 1, 1, 1,
117 "Pshared errorcheck mutex and alt clock condvar across processes"}
118 , {
119 PTHREAD_MUTEX_RECURSIVE, 1, 1, 1,
120 "Pshared recursive mutex and alt clock condvar across processes"}
121
122 , {
123 PTHREAD_MUTEX_DEFAULT, 0, 1, 0,
124 "Default mutex and alt clock condvar"}
125 , {
126 PTHREAD_MUTEX_NORMAL, 0, 1, 0,
127 "Normal mutex and alt clock condvar"}
128 , {
129 PTHREAD_MUTEX_ERRORCHECK, 0, 1, 0,
130 "Errorcheck mutex and alt clock condvar"}
131 , {
132 PTHREAD_MUTEX_RECURSIVE, 0, 1, 0,
133 "Recursive mutex and alt clock condvar"}
134
135 , {
136 PTHREAD_MUTEX_DEFAULT, 1, 1, 0,
137 "PShared default mutex and alt clock condvar"}
138 , {
139 PTHREAD_MUTEX_NORMAL, 1, 1, 0,
140 "Pshared normal mutex and alt clock condvar"}
141 , {
142 PTHREAD_MUTEX_ERRORCHECK, 1, 1, 0,
143 "Pshared errorcheck mutex and alt clock condvar"}
144 , {
145 PTHREAD_MUTEX_RECURSIVE, 1, 1, 0,
146 "Pshared recursive mutex and alt clock condvar"}
147 #endif
148 };
149
150 static struct {
151 long sec_val; /* Value for seconds */
152 short sec_is_offset; /* Seconds value is added to current time or is absolute */
153 long nsec_val; /* Value for nanoseconds */
154 short nsec_is_offset; /* Nanoseconds value is added to current time or is absolute */
155 } junks_ts[] = {
156 {
157 -2, 1, 1000000000, 1}
158 , {
159 -2, 1, -1, 0}
160 , {
161 -3, 1, 2000000000, 0}
162 };
163
tf(void * arg)164 static void *tf(void *arg)
165 {
166 int ret = 0;
167
168 testdata_t *td = (testdata_t *) arg;
169
170 /* Lock the mutex */
171 ret = pthread_mutex_lock(&(td->mtx));
172 if (ret != 0) {
173 td->status = ret;
174 UNRESOLVED(ret, "[child] Unable to lock the mutex");
175 }
176
177 /* Checks whether the parent release the lock inside the timedwait function */
178 if (td->ctrl == 0)
179 td->gotit += 1;
180
181 /* Unlock and exit */
182 ret = pthread_mutex_unlock(&(td->mtx));
183 if (ret != 0) {
184 td->status = ret;
185 UNRESOLVED(ret, "[child] Failed to unlock the mutex.");
186 }
187 return NULL;
188 }
189
main(void)190 int main(void)
191 {
192 int ret, k;
193 unsigned int i, j;
194 pthread_mutexattr_t ma;
195 pthread_condattr_t ca;
196 pthread_cond_t cnd;
197 clockid_t cid = CLOCK_REALTIME;
198 struct timespec ts, ts_junk;
199
200 testdata_t *td;
201 testdata_t alternativ;
202
203 int do_fork;
204
205 pid_t child_pr[NCHILDREN], chkpid;
206 int status;
207 pthread_t child_th[NCHILDREN];
208
209 long pshared, monotonic, cs, mf;
210
211 output_init();
212 pshared = sysconf(_SC_THREAD_PROCESS_SHARED);
213 cs = sysconf(_SC_CLOCK_SELECTION);
214 monotonic = sysconf(_SC_MONOTONIC_CLOCK);
215 mf = sysconf(_SC_MAPPED_FILES);
216
217 #if VERBOSE > 0
218 output("Test starting\n");
219 output("System abilities:\n");
220 output(" TPS : %li\n", pshared);
221 output(" CS : %li\n", cs);
222 output(" MON : %li\n", monotonic);
223 output(" MF : %li\n", mf);
224 if ((mf < 0) || (pshared < 0))
225 output("Process-shared attributes won't be tested\n");
226 if ((cs < 0) || (monotonic < 0))
227 output("Alternative clock won't be tested\n");
228 fflush(stdout);
229 #endif
230
231 /* We are not interested in testing the clock if we have no other clock available.. */
232 if (monotonic < 0)
233 cs = -1;
234
235 #ifndef USE_ALTCLK
236 if (cs > 0)
237 output
238 ("Implementation supports the MONOTONIC CLOCK but option is disabled in test.\n");
239 #endif
240
241 /**********
242 * Allocate space for the testdata structure
243 */
244 if (mf < 0) {
245 /* Cannot mmap a file, we use an alternative method */
246 td = &alternativ;
247 pshared = -1; /* We won't do this testing anyway */
248 #if VERBOSE > 0
249 output("Testdata allocated in the process memory.\n");
250 #endif
251 } else {
252 /* We will place the test data in a mmaped file */
253 char filename[] = "/tmp/cond_timedwait_2-4-XXXXXX";
254 size_t sz;
255 void *mmaped;
256 int fd;
257 char *tmp;
258
259 /* We now create the temp files */
260 fd = mkstemp(filename);
261 if (fd == -1) {
262 UNRESOLVED(errno,
263 "Temporary file could not be created");
264 }
265
266 /* and make sure the file will be deleted when closed */
267 unlink(filename);
268
269 #if VERBOSE > 1
270 output("Temp file created (%s).\n", filename);
271 #endif
272
273 sz = (size_t) sysconf(_SC_PAGESIZE);
274
275 tmp = calloc(1, sz);
276 if (tmp == NULL) {
277 UNRESOLVED(errno, "Memory allocation failed");
278 }
279
280 /* Write the data to the file. */
281 if (write(fd, tmp, sz) != (ssize_t) sz) {
282 UNRESOLVED(sz, "Writting to the file failed");
283 }
284
285 free(tmp);
286
287 /* Now we can map the file in memory */
288 mmaped =
289 mmap(NULL, sz, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
290 if (mmaped == MAP_FAILED) {
291 UNRESOLVED(errno, "mmap failed");
292 }
293
294 td = (testdata_t *) mmaped;
295
296 /* Our datatest structure is now in shared memory */
297 #if VERBOSE > 1
298 output("Testdata allocated in shared memory.\n");
299 #endif
300 }
301
302 /**********
303 * For each test scenario, initialize the attributes and other variables.
304 * Do the whole thing for each time to test.
305 */
306 for (i = 0; i < (sizeof(scenarii) / sizeof(scenarii[0])); i++) {
307 for (j = 0; j < (sizeof(junks_ts) / sizeof(junks_ts[0])); j++) {
308 #if VERBOSE > 1
309 output("[parent] Preparing attributes for: %s\n",
310 scenarii[i].descr);
311 #endif
312 /* set / reset everything */
313 do_fork = 0;
314 ret = pthread_mutexattr_init(&ma);
315 if (ret != 0) {
316 UNRESOLVED(ret,
317 "[parent] Unable to initialize the mutex attribute object");
318 }
319 ret = pthread_condattr_init(&ca);
320 if (ret != 0) {
321 UNRESOLVED(ret,
322 "[parent] Unable to initialize the cond attribute object");
323 }
324
325 /* Set the mutex type */
326 ret =
327 pthread_mutexattr_settype(&ma, scenarii[i].m_type);
328 if (ret != 0) {
329 UNRESOLVED(ret,
330 "[parent] Unable to set mutex type");
331 }
332 #if VERBOSE > 1
333 output("[parent] Mutex type : %i\n",
334 scenarii[i].m_type);
335 #endif
336
337 /* Set the pshared attributes, if supported */
338 if ((pshared > 0) && (scenarii[i].mc_pshared != 0)) {
339 ret =
340 pthread_mutexattr_setpshared(&ma,
341 PTHREAD_PROCESS_SHARED);
342 if (ret != 0) {
343 UNRESOLVED(ret,
344 "[parent] Unable to set the mutex process-shared");
345 }
346 ret =
347 pthread_condattr_setpshared(&ca,
348 PTHREAD_PROCESS_SHARED);
349 if (ret != 0) {
350 UNRESOLVED(ret,
351 "[parent] Unable to set the cond var process-shared");
352 }
353 #if VERBOSE > 1
354 output
355 ("[parent] Mutex & cond are process-shared\n");
356 #endif
357 }
358 #if VERBOSE > 1
359 else {
360 output
361 ("[parent] Mutex & cond are process-private\n");
362 }
363 #endif
364
365 /* Set the alternative clock, if supported */
366 #ifdef USE_ALTCLK
367 if ((cs > 0) && (scenarii[i].c_clock != 0)) {
368 ret =
369 pthread_condattr_setclock(&ca,
370 CLOCK_MONOTONIC);
371 if (ret != 0) {
372 UNRESOLVED(ret,
373 "[parent] Unable to set the monotonic clock for the cond");
374 }
375 #if VERBOSE > 1
376 output
377 ("[parent] Cond uses the Monotonic clock\n");
378 #endif
379 }
380 #if VERBOSE > 1
381 else {
382 output
383 ("[parent] Cond uses the default clock\n");
384 }
385 #endif
386 ret = pthread_condattr_getclock(&ca, &cid);
387 if (ret != 0) {
388 UNRESOLVED(ret,
389 "Unable to get clock from cond attr");
390 }
391 #endif
392
393 /* Tell whether the test will be across processes */
394 if ((pshared > 0) && (scenarii[i].fork != 0)) {
395 do_fork = 1;
396 #if VERBOSE > 1
397 output
398 ("[parent] Child will be a new process\n");
399 #endif
400 }
401 #if VERBOSE > 1
402 else {
403 output("[parent] Child will be a new thread\n");
404 }
405 #endif
406
407 /* initialize the condvar */
408 ret = pthread_cond_init(&cnd, &ca);
409 if (ret != 0) {
410 UNRESOLVED(ret, "[parent] Cond init failed");
411 }
412
413 /**********
414 * Initialize the testdata_t structure with the previously defined attributes
415 */
416 /* Initialize the mutex */
417 ret = pthread_mutex_init(&(td->mtx), &ma);
418 if (ret != 0) {
419 UNRESOLVED(ret, "[parent] Mutex init failed");
420 }
421
422 /* Initialize the other datas from the test structure */
423 td->ctrl = 0;
424 td->gotit = 0;
425 td->status = 0;
426
427 /**********
428 * Proceed to the actual testing
429 */
430 /* Lock the mutex before creating children */
431 ret = pthread_mutex_lock(&(td->mtx));
432 if (ret != 0) {
433 UNRESOLVED(ret,
434 "[parent] Unable to lock the mutex");
435 }
436
437 /* Create the children */
438 if (do_fork != 0) {
439 /* We are testing across processes */
440 for (k = 0; k < NCHILDREN; k++) {
441 child_pr[k] = fork();
442 if (child_pr[k] == -1) {
443 UNRESOLVED(errno,
444 "[parent] Fork failed");
445 }
446
447 if (child_pr[k] == 0) {
448 #if VERBOSE > 3
449 output
450 ("[child] Child process %i starting...\n",
451 k);
452 #endif
453
454 if (tf((void *)td) != NULL) {
455 UNRESOLVED(-1,
456 "[child] Got an unexpected return value from test function");
457 } else {
458 /* We cannot use the PASSED macro here since it would terminate the output */
459 exit(0);
460 }
461 }
462 }
463 /* Only the parent process goes further */
464 } else { /* do_fork == 0 */
465
466 /* We are testing across two threads */
467 for (k = 0; k < NCHILDREN; k++) {
468 ret =
469 pthread_create(&child_th[k], NULL,
470 tf, td);
471 if (ret != 0) {
472 UNRESOLVED(ret,
473 "[parent] Unable to create the child thread.");
474 }
475 }
476 }
477
478 /* Children are now running and trying to lock the mutex. */
479
480 ret = clock_gettime(cid, &ts);
481 if (ret != 0) {
482 UNRESOLVED(ret,
483 "[parent] Unable to read clock");
484 }
485
486 /* Do the junk timedwaits */
487 ts_junk.tv_sec =
488 junks_ts[j].sec_val +
489 (junks_ts[j].sec_is_offset ? ts.tv_sec : 0);
490 ts_junk.tv_nsec =
491 junks_ts[j].nsec_val +
492 (junks_ts[j].nsec_is_offset ? ts.tv_nsec : 0);
493
494 #if VERBOSE > 2
495 output("TS: s = %s%li ; ns = %s%li\n",
496 junks_ts[j].sec_is_offset ? "n + " : " ",
497 junks_ts[j].sec_val,
498 junks_ts[j].nsec_is_offset ? "n + " : " ",
499 junks_ts[j].nsec_val);
500 output("Now is: %i.%09li\n", ts.tv_sec, ts.tv_nsec);
501 output("Junk is: %i.%09li\n", ts_junk.tv_sec,
502 ts_junk.tv_nsec);
503 #endif
504
505 do {
506 ret =
507 pthread_cond_timedwait(&cnd, &(td->mtx),
508 &ts_junk);
509 } while (ret == 0);
510 #if VERBOSE > 2
511 output("timedwait returns %d (%s) - gotit = %d\n", ret,
512 strerror(ret), td->gotit);
513 #endif
514
515 /* check that when EINVAL is returned, the mutex has not been released */
516 if (ret == EINVAL) {
517 if (td->gotit != 0) {
518 FAILED
519 ("The mutex was released when an invalid timestamp was detected in the function");
520 }
521 #if VERBOSE > 0
522 } else {
523 output
524 ("Warning, struct timespec with tv_sec = %i and tv_nsec = %li was not invalid\n",
525 ts_junk.tv_sec, ts_junk.tv_nsec);
526 }
527 #endif
528
529 /* Finally unlock the mutex */
530 td->ctrl = 1;
531 ret = pthread_mutex_unlock(&(td->mtx));
532 if (ret != 0) {
533 UNRESOLVED(ret,
534 "[parent] Unable to unlock the mutex");
535 }
536
537 /* Wait for the child to terminate */
538 if (do_fork != 0) {
539 /* We were testing across processes */
540 ret = 0;
541 for (k = 0; k < NCHILDREN; k++) {
542 chkpid =
543 waitpid(child_pr[k], &status, 0);
544 if (chkpid != child_pr[k]) {
545 output
546 ("Expected pid: %i. Got %i\n",
547 (int)child_pr[k],
548 (int)chkpid);
549 UNRESOLVED(errno,
550 "Waitpid failed");
551 }
552 if (WIFSIGNALED(status)) {
553 output
554 ("Child process killed with signal %d\n",
555 WTERMSIG(status));
556 UNRESOLVED(-1,
557 "Child process was killed");
558 }
559
560 if (WIFEXITED(status)) {
561 ret |= WEXITSTATUS(status);
562 } else {
563 UNRESOLVED(-1,
564 "Child process was neither killed nor exited");
565 }
566 }
567 if (ret != 0) {
568 exit(ret); /* Output has already been closed in child */
569 }
570
571 } else { /* child was a thread */
572
573 for (k = 0; k < NCHILDREN; k++) {
574 ret = pthread_join(child_th[k], NULL);
575 if (ret != 0) {
576 UNRESOLVED(ret,
577 "[parent] Unable to join the thread");
578 }
579 }
580 }
581
582 /**********
583 * Destroy the data
584 */
585 ret = pthread_cond_destroy(&cnd);
586 if (ret != 0) {
587 UNRESOLVED(ret,
588 "Failed to destroy the cond var");
589 }
590
591 ret = pthread_mutex_destroy(&(td->mtx));
592 if (ret != 0) {
593 UNRESOLVED(ret, "Failed to destroy the mutex");
594 }
595
596 ret = pthread_condattr_destroy(&ca);
597 if (ret != 0) {
598 UNRESOLVED(ret,
599 "Failed to destroy the cond var attribute object");
600 }
601
602 ret = pthread_mutexattr_destroy(&ma);
603 if (ret != 0) {
604 UNRESOLVED(ret,
605 "Failed to destroy the mutex attribute object");
606 }
607
608 } /* Proceed to the next junk timedwait value */
609 } /* Proceed to the next scenario */
610
611 #if VERBOSE > 0
612 output("Test passed\n");
613 #endif
614
615 PASSED;
616 }
617
618 #else /* WITHOUT_XOPEN */
main(void)619 int main(void)
620 {
621 output_init();
622 UNTESTED("This test requires XSI features");
623 }
624 #endif
625