1
/* Copyright (C) 2003, 2004, 2007 Free Software Foundation, Inc.
2
This file is part of the GNU C Library.
4
The GNU C Library is free software; you can redistribute it and/or
5
modify it under the terms of the GNU Lesser General Public
6
License as published by the Free Software Foundation; either
7
version 2.1 of the License, or (at your option) any later version.
9
The GNU C Library is distributed in the hope that it will be useful,
10
but WITHOUT ANY WARRANTY; without even the implied warranty of
11
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12
Lesser General Public License for more details.
14
You should have received a copy of the GNU Lesser General Public
15
License along with the GNU C Library; if not, write to the Free
16
Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
20
#include <lowlevellock.h>
23
unsigned long int __fork_generation attribute_hidden;
26
clear_once_control (void *arg)
28
pthread_once_t *once_control = (pthread_once_t *) arg;
31
lll_futex_wake (once_control, INT_MAX, LLL_PRIVATE);
36
__pthread_once (pthread_once_t *once_control, void (*init_routine) (void))
45
newval = __fork_generation | 1;
46
oldval = *once_control;
47
if ((oldval & 2) == 0)
48
*once_control = newval;
51
newval = __fork_generation | 1;
52
__asm__ __volatile__ (
60
: "=&r" (oldval), "=&r" (tmp), "=m" (*once_control)
61
: "r" (newval), "m" (*once_control));
63
/* Check if the initializer has already been done. */
64
if ((oldval & 2) != 0)
67
/* Check if another thread already runs the initializer. */
68
if ((oldval & 1) == 0)
71
/* Check whether the initializer execution was interrupted by a fork. */
75
/* Same generation, some other thread was faster. Wait. */
76
lll_futex_wait (once_control, oldval, LLL_PRIVATE);
79
/* This thread is the first here. Do the initialization.
80
Register a cleanup handler so that in case the thread gets
81
interrupted the initialization can be restarted. */
82
pthread_cleanup_push (clear_once_control, once_control);
86
pthread_cleanup_pop (0);
88
/* Add one to *once_control to take the bottom 2 bits from 01 to 10. */
89
atomic_increment (once_control);
91
/* Wake up all other threads. */
92
lll_futex_wake (once_control, INT_MAX, LLL_PRIVATE);
96
weak_alias (__pthread_once, pthread_once)
97
strong_alias (__pthread_once, __pthread_once_internal)