00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033 #ifndef _UT_BARRIER_H
00034 #define _UT_BARRIER_H
00035
00036 #include <errno.h>
00037 #include <pthread.h>
00038
00039
00040
00041 #define UT_BARRIER_T ut_barrier_t
00042 #define UT_BARRIER_INIT ut_barrier_init
00043 #define UT_BARRIER_WAIT ut_barrier_wait
00044 #define UT_BARRIER_DESTROY ut_barrier_destroy
00045
00046 typedef struct ut_barrier_s ut_barrier_t;
00047
00048 struct ut_barrier_s {
00049 int maxcnt;
00050 struct _sb {
00051 pthread_cond_t wait_cv;
00052 pthread_mutex_t wait_lk;
00053 int runners;
00054 } sb[2];
00055 struct _sb *sbp;
00056 };
00057
00058
00059 static int
00060 ut_barrier_init(ut_barrier_t *bp, int count)
00061 {
00062 int n;
00063 int i;
00064
00065 if (count < 1) {
00066 return(EINVAL);
00067 }
00068
00069 bp->maxcnt = count;
00070 bp->sbp = &bp->sb[0];
00071
00072 for (i = 0; i < 2; ++i) {
00073 struct _sb *sbp = &( bp->sb[i] );
00074 sbp->runners = count;
00075
00076 if (n = pthread_mutex_init(&sbp->wait_lk, NULL)) {
00077 return(n);
00078 }
00079
00080 if (n = pthread_cond_init(&sbp->wait_cv, NULL)) {
00081 return(n);
00082 }
00083 }
00084 return(0);
00085 }
00086
00087
00088 static int
00089 ut_barrier_wait(register ut_barrier_t *bp)
00090 {
00091 register struct _sb *sbp = bp->sbp;
00092
00093 pthread_mutex_lock(&sbp->wait_lk);
00094
00095 if (sbp->runners == 1) {
00096 if (bp->maxcnt != 1) {
00097
00098 sbp->runners = bp->maxcnt;
00099 bp->sbp = (bp->sbp == &bp->sb[0]) ? &bp->sb[1] : &bp->sb[0];
00100
00101
00102 pthread_cond_broadcast(&sbp->wait_cv);
00103 }
00104 } else {
00105 sbp->runners--;
00106
00107 while (sbp->runners != bp->maxcnt) {
00108 pthread_cond_wait( &sbp->wait_cv, &sbp->wait_lk);
00109 }
00110 }
00111
00112 pthread_mutex_unlock(&sbp->wait_lk);
00113
00114 return(0);
00115 }
00116
00117
00118 static int
00119 ut_barrier_destroy(ut_barrier_t *bp) {
00120 int n;
00121 int i;
00122
00123 for (i=0; i < 2; ++ i) {
00124 if (n = pthread_cond_destroy(&bp->sb[i].wait_cv)) {
00125 return( n );
00126 }
00127
00128 if (n = pthread_mutex_destroy( &bp->sb[i].wait_lk)) {
00129 return(n);
00130 }
00131 }
00132 return (0);
00133 }
00134
00135 #endif