153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2006 Lennart Poettering 553a5a1b3Sopenharmony_ci 653a5a1b3Sopenharmony_ci PulseAudio is free software; you can redistribute it and/or modify 753a5a1b3Sopenharmony_ci it under the terms of the GNU Lesser General Public License as published 853a5a1b3Sopenharmony_ci by the Free Software Foundation; either version 2.1 of the License, 953a5a1b3Sopenharmony_ci or (at your option) any later version. 1053a5a1b3Sopenharmony_ci 1153a5a1b3Sopenharmony_ci PulseAudio is distributed in the hope that it will be useful, but 1253a5a1b3Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 1353a5a1b3Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 1453a5a1b3Sopenharmony_ci General Public License for more details. 1553a5a1b3Sopenharmony_ci 1653a5a1b3Sopenharmony_ci You should have received a copy of the GNU Lesser General Public License 1753a5a1b3Sopenharmony_ci along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 1853a5a1b3Sopenharmony_ci***/ 1953a5a1b3Sopenharmony_ci 2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H 2153a5a1b3Sopenharmony_ci#include <config.h> 2253a5a1b3Sopenharmony_ci#endif 2353a5a1b3Sopenharmony_ci 2453a5a1b3Sopenharmony_ci#include <pulsecore/macro.h> 2553a5a1b3Sopenharmony_ci 2653a5a1b3Sopenharmony_ci#include "once.h" 2753a5a1b3Sopenharmony_ci 2853a5a1b3Sopenharmony_ci/* See http://www.hpl.hp.com/research/linux/atomic_ops/example.php4 for the 2953a5a1b3Sopenharmony_ci * reference algorithm used here. */ 3053a5a1b3Sopenharmony_ci 3153a5a1b3Sopenharmony_cibool pa_once_begin(pa_once *control) { 3253a5a1b3Sopenharmony_ci pa_mutex *m; 3353a5a1b3Sopenharmony_ci 3453a5a1b3Sopenharmony_ci pa_assert(control); 3553a5a1b3Sopenharmony_ci 3653a5a1b3Sopenharmony_ci if (pa_atomic_load(&control->done)) 3753a5a1b3Sopenharmony_ci return false; 3853a5a1b3Sopenharmony_ci 3953a5a1b3Sopenharmony_ci /* Caveat: We have to make sure that the once func has completed 4053a5a1b3Sopenharmony_ci * before returning, even if the once func is not actually 4153a5a1b3Sopenharmony_ci * executed by us. Hence the awkward locking. */ 4253a5a1b3Sopenharmony_ci 4353a5a1b3Sopenharmony_ci m = pa_static_mutex_get(&control->mutex, false, false); 4453a5a1b3Sopenharmony_ci pa_mutex_lock(m); 4553a5a1b3Sopenharmony_ci 4653a5a1b3Sopenharmony_ci if (pa_atomic_load(&control->done)) { 4753a5a1b3Sopenharmony_ci pa_mutex_unlock(m); 4853a5a1b3Sopenharmony_ci return false; 4953a5a1b3Sopenharmony_ci } 5053a5a1b3Sopenharmony_ci 5153a5a1b3Sopenharmony_ci return true; 5253a5a1b3Sopenharmony_ci} 5353a5a1b3Sopenharmony_ci 5453a5a1b3Sopenharmony_civoid pa_once_end(pa_once *control) { 5553a5a1b3Sopenharmony_ci pa_mutex *m; 5653a5a1b3Sopenharmony_ci 5753a5a1b3Sopenharmony_ci pa_assert(control); 5853a5a1b3Sopenharmony_ci 5953a5a1b3Sopenharmony_ci pa_assert(!pa_atomic_load(&control->done)); 6053a5a1b3Sopenharmony_ci pa_atomic_store(&control->done, 1); 6153a5a1b3Sopenharmony_ci 6253a5a1b3Sopenharmony_ci m = pa_static_mutex_get(&control->mutex, false, false); 6353a5a1b3Sopenharmony_ci pa_mutex_unlock(m); 6453a5a1b3Sopenharmony_ci} 6553a5a1b3Sopenharmony_ci 6653a5a1b3Sopenharmony_ci/* Not reentrant -- how could it be? */ 6753a5a1b3Sopenharmony_civoid pa_run_once(pa_once *control, pa_once_func_t func) { 6853a5a1b3Sopenharmony_ci pa_assert(control); 6953a5a1b3Sopenharmony_ci pa_assert(func); 7053a5a1b3Sopenharmony_ci 7153a5a1b3Sopenharmony_ci if (pa_once_begin(control)) { 7253a5a1b3Sopenharmony_ci func(); 7353a5a1b3Sopenharmony_ci pa_once_end(control); 7453a5a1b3Sopenharmony_ci } 7553a5a1b3Sopenharmony_ci} 76