1/************************************************************************** 2 * 3 * Copyright 2009-2010 VMware, Inc. 4 * All Rights Reserved. 5 * 6 * Permission is hereby granted, free of charge, to any person obtaining a 7 * copy of this software and associated documentation files (the 8 * "Software"), to deal in the Software without restriction, including 9 * without limitation the rights to use, copy, modify, merge, publish, 10 * distribute, sub license, and/or sell copies of the Software, and to 11 * permit persons to whom the Software is furnished to do so, subject to 12 * the following conditions: 13 * 14 * The above copyright notice and this permission notice (including the 15 * next paragraph) shall be included in all copies or substantial portions 16 * of the Software. 17 * 18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS 19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 20 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. 21 * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR 22 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, 23 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE 24 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 25 * 26 **************************************************************************/ 27 28 29/* 30 * Test case for util_barrier. 31 * 32 * The test succeeds if no thread exits before all the other threads reach 33 * the barrier. 34 */ 35 36 37#include <stdio.h> 38#include <stdlib.h> 39#include <string.h> 40 41#include "util/os_time.h" 42#include "util/u_atomic.h" 43#include "util/u_thread.h" 44 45 46#define NUM_THREADS 10 47 48static int verbosity = 0; 49 50static thrd_t threads[NUM_THREADS]; 51static util_barrier barrier; 52static int thread_ids[NUM_THREADS]; 53 54static volatile int waiting = 0; 55static volatile int proceeded = 0; 56 57 58#define LOG(fmt, ...) \ 59 if (verbosity > 0) { \ 60 fprintf(stdout, fmt, ##__VA_ARGS__); \ 61 } 62 63#define CHECK(_cond) \ 64 if (!(_cond)) { \ 65 fprintf(stderr, "%s:%u: `%s` failed\n", __FILE__, __LINE__, #_cond); \ 66 _exit(EXIT_FAILURE); \ 67 } 68 69 70static int 71thread_function(void *thread_data) 72{ 73 int thread_id = *((int *) thread_data); 74 75 LOG("thread %d starting\n", thread_id); 76 os_time_sleep(thread_id * 100 * 1000); 77 LOG("thread %d before barrier\n", thread_id); 78 79 CHECK(p_atomic_read(&proceeded) == 0); 80 p_atomic_inc(&waiting); 81 82 util_barrier_wait(&barrier); 83 84 CHECK(p_atomic_read(&waiting) == NUM_THREADS); 85 86 p_atomic_inc(&proceeded); 87 88 LOG("thread %d exiting\n", thread_id); 89 90 return 0; 91} 92 93 94int main(int argc, char *argv[]) 95{ 96 int i; 97 98 for (i = 1; i < argc; ++i) { 99 const char *arg = argv[i]; 100 if (strcmp(arg, "-v") == 0) { 101 ++verbosity; 102 } else { 103 fprintf(stderr, "error: unrecognized option `%s`\n", arg); 104 exit(EXIT_FAILURE); 105 } 106 } 107 108 // Disable buffering 109 setbuf(stdout, NULL); 110 111 LOG("pipe_barrier_test starting\n"); 112 113 util_barrier_init(&barrier, NUM_THREADS); 114 115 for (i = 0; i < NUM_THREADS; i++) { 116 thread_ids[i] = i; 117 threads[i] = u_thread_create(thread_function, (void *) &thread_ids[i]); 118 } 119 120 for (i = 0; i < NUM_THREADS; i++ ) { 121 thrd_join(threads[i], NULL); 122 } 123 124 CHECK(p_atomic_read(&proceeded) == NUM_THREADS); 125 126 util_barrier_destroy(&barrier); 127 128 LOG("pipe_barrier_test exiting\n"); 129 130 return 0; 131} 132