summaryrefslogtreecommitdiff
path: root/nptl/DESIGN-barrier.txt
blob: 782377f0c550ac0720ff0fd92587c4208c82bc6c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
Barriers pseudocode
===================

    int pthread_barrier_wait(barrier_t * barrier);

struct barrier_t {

   unsigned int lock:
         - internal mutex

   unsigned int left;
         - current barrier count, # of threads still needed.

   unsigned int init_count;
         - number of threads needed for the barrier to continue.

   unsigned int curr_event;
         - generation count
}

pthread_barrier_wait(barrier_t *barrier)
{
  unsigned int event;

  lll_lock(barrier->lock);
  if (!--barrier->left) {
    barrier->left = barrier->init_count;   
    barrier->curr_event++;
    futex_wake(&barrier->curr_event, INT_MAX)
    lll_unlock(barrier->lock);

    return BARRIER_SERIAL_THREAD;
  }

  event = barrier->curr_event;
  for (;;) {
    lll_unlock(barrier->lock);

    futex_wait(&barrier->curr_event, event)

    lll_lock(barrier->lock);
    if (event != barrier->curr_event)
      break;
  }
  lll_unlock(barrier->lock);

  return 0;
}