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 <pthread.h> 2553a5a1b3Sopenharmony_ci#include <errno.h> 2653a5a1b3Sopenharmony_ci 2753a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h> 2853a5a1b3Sopenharmony_ci 2953a5a1b3Sopenharmony_ci#include <pulsecore/core-error.h> 3053a5a1b3Sopenharmony_ci#include <pulsecore/macro.h> 3153a5a1b3Sopenharmony_ci 3253a5a1b3Sopenharmony_ci#include "mutex.h" 3353a5a1b3Sopenharmony_ci 3453a5a1b3Sopenharmony_cistruct pa_mutex { 3553a5a1b3Sopenharmony_ci pthread_mutex_t mutex; 3653a5a1b3Sopenharmony_ci}; 3753a5a1b3Sopenharmony_ci 3853a5a1b3Sopenharmony_cistruct pa_cond { 3953a5a1b3Sopenharmony_ci pthread_cond_t cond; 4053a5a1b3Sopenharmony_ci}; 4153a5a1b3Sopenharmony_ci 4253a5a1b3Sopenharmony_cipa_mutex* pa_mutex_new(bool recursive, bool inherit_priority) { 4353a5a1b3Sopenharmony_ci pa_mutex *m; 4453a5a1b3Sopenharmony_ci pthread_mutexattr_t attr; 4553a5a1b3Sopenharmony_ci#ifdef HAVE_PTHREAD_PRIO_INHERIT 4653a5a1b3Sopenharmony_ci int r; 4753a5a1b3Sopenharmony_ci#endif 4853a5a1b3Sopenharmony_ci 4953a5a1b3Sopenharmony_ci pa_assert_se(pthread_mutexattr_init(&attr) == 0); 5053a5a1b3Sopenharmony_ci 5153a5a1b3Sopenharmony_ci if (recursive) 5253a5a1b3Sopenharmony_ci pa_assert_se(pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE) == 0); 5353a5a1b3Sopenharmony_ci 5453a5a1b3Sopenharmony_ci#ifdef HAVE_PTHREAD_PRIO_INHERIT 5553a5a1b3Sopenharmony_ci if (inherit_priority) { 5653a5a1b3Sopenharmony_ci r = pthread_mutexattr_setprotocol(&attr, PTHREAD_PRIO_INHERIT); 5753a5a1b3Sopenharmony_ci pa_assert(r == 0 || r == ENOTSUP); 5853a5a1b3Sopenharmony_ci } 5953a5a1b3Sopenharmony_ci#endif 6053a5a1b3Sopenharmony_ci 6153a5a1b3Sopenharmony_ci m = pa_xnew(pa_mutex, 1); 6253a5a1b3Sopenharmony_ci 6353a5a1b3Sopenharmony_ci#ifndef HAVE_PTHREAD_PRIO_INHERIT 6453a5a1b3Sopenharmony_ci pa_assert_se(pthread_mutex_init(&m->mutex, &attr) == 0); 6553a5a1b3Sopenharmony_ci 6653a5a1b3Sopenharmony_ci#else 6753a5a1b3Sopenharmony_ci if ((r = pthread_mutex_init(&m->mutex, &attr))) { 6853a5a1b3Sopenharmony_ci 6953a5a1b3Sopenharmony_ci /* If this failed, then this was probably due to non-available 7053a5a1b3Sopenharmony_ci * priority inheritance. In which case we fall back to normal 7153a5a1b3Sopenharmony_ci * mutexes. */ 7253a5a1b3Sopenharmony_ci pa_assert(r == ENOTSUP && inherit_priority); 7353a5a1b3Sopenharmony_ci 7453a5a1b3Sopenharmony_ci pa_assert_se(pthread_mutexattr_setprotocol(&attr, PTHREAD_PRIO_NONE) == 0); 7553a5a1b3Sopenharmony_ci pa_assert_se(pthread_mutex_init(&m->mutex, &attr) == 0); 7653a5a1b3Sopenharmony_ci } 7753a5a1b3Sopenharmony_ci#endif 7853a5a1b3Sopenharmony_ci 7953a5a1b3Sopenharmony_ci return m; 8053a5a1b3Sopenharmony_ci} 8153a5a1b3Sopenharmony_ci 8253a5a1b3Sopenharmony_civoid pa_mutex_free(pa_mutex *m) { 8353a5a1b3Sopenharmony_ci pa_assert(m); 8453a5a1b3Sopenharmony_ci 8553a5a1b3Sopenharmony_ci pa_assert_se(pthread_mutex_destroy(&m->mutex) == 0); 8653a5a1b3Sopenharmony_ci pa_xfree(m); 8753a5a1b3Sopenharmony_ci} 8853a5a1b3Sopenharmony_ci 8953a5a1b3Sopenharmony_civoid pa_mutex_lock(pa_mutex *m) { 9053a5a1b3Sopenharmony_ci pa_assert(m); 9153a5a1b3Sopenharmony_ci 9253a5a1b3Sopenharmony_ci pa_assert_se(pthread_mutex_lock(&m->mutex) == 0); 9353a5a1b3Sopenharmony_ci} 9453a5a1b3Sopenharmony_ci 9553a5a1b3Sopenharmony_cibool pa_mutex_try_lock(pa_mutex *m) { 9653a5a1b3Sopenharmony_ci int r; 9753a5a1b3Sopenharmony_ci pa_assert(m); 9853a5a1b3Sopenharmony_ci 9953a5a1b3Sopenharmony_ci if ((r = pthread_mutex_trylock(&m->mutex)) != 0) { 10053a5a1b3Sopenharmony_ci pa_assert(r == EBUSY); 10153a5a1b3Sopenharmony_ci return false; 10253a5a1b3Sopenharmony_ci } 10353a5a1b3Sopenharmony_ci 10453a5a1b3Sopenharmony_ci return true; 10553a5a1b3Sopenharmony_ci} 10653a5a1b3Sopenharmony_ci 10753a5a1b3Sopenharmony_civoid pa_mutex_unlock(pa_mutex *m) { 10853a5a1b3Sopenharmony_ci int err; 10953a5a1b3Sopenharmony_ci 11053a5a1b3Sopenharmony_ci pa_assert(m); 11153a5a1b3Sopenharmony_ci 11253a5a1b3Sopenharmony_ci if ((err = pthread_mutex_unlock(&m->mutex)) != 0) { 11353a5a1b3Sopenharmony_ci pa_log_error("pthread_mutex_unlock() failed: %s", pa_cstrerror(err)); 11453a5a1b3Sopenharmony_ci pa_assert_not_reached(); 11553a5a1b3Sopenharmony_ci } 11653a5a1b3Sopenharmony_ci} 11753a5a1b3Sopenharmony_ci 11853a5a1b3Sopenharmony_cipa_cond *pa_cond_new(void) { 11953a5a1b3Sopenharmony_ci pa_cond *c; 12053a5a1b3Sopenharmony_ci 12153a5a1b3Sopenharmony_ci c = pa_xnew(pa_cond, 1); 12253a5a1b3Sopenharmony_ci pa_assert_se(pthread_cond_init(&c->cond, NULL) == 0); 12353a5a1b3Sopenharmony_ci return c; 12453a5a1b3Sopenharmony_ci} 12553a5a1b3Sopenharmony_ci 12653a5a1b3Sopenharmony_civoid pa_cond_free(pa_cond *c) { 12753a5a1b3Sopenharmony_ci pa_assert(c); 12853a5a1b3Sopenharmony_ci 12953a5a1b3Sopenharmony_ci pa_assert_se(pthread_cond_destroy(&c->cond) == 0); 13053a5a1b3Sopenharmony_ci pa_xfree(c); 13153a5a1b3Sopenharmony_ci} 13253a5a1b3Sopenharmony_ci 13353a5a1b3Sopenharmony_civoid pa_cond_signal(pa_cond *c, int broadcast) { 13453a5a1b3Sopenharmony_ci pa_assert(c); 13553a5a1b3Sopenharmony_ci 13653a5a1b3Sopenharmony_ci if (broadcast) 13753a5a1b3Sopenharmony_ci pa_assert_se(pthread_cond_broadcast(&c->cond) == 0); 13853a5a1b3Sopenharmony_ci else 13953a5a1b3Sopenharmony_ci pa_assert_se(pthread_cond_signal(&c->cond) == 0); 14053a5a1b3Sopenharmony_ci} 14153a5a1b3Sopenharmony_ci 14253a5a1b3Sopenharmony_ciint pa_cond_wait(pa_cond *c, pa_mutex *m) { 14353a5a1b3Sopenharmony_ci pa_assert(c); 14453a5a1b3Sopenharmony_ci pa_assert(m); 14553a5a1b3Sopenharmony_ci 14653a5a1b3Sopenharmony_ci return pthread_cond_wait(&c->cond, &m->mutex); 14753a5a1b3Sopenharmony_ci} 14853a5a1b3Sopenharmony_ci 14953a5a1b3Sopenharmony_cipa_mutex* pa_static_mutex_get(pa_static_mutex *s, bool recursive, bool inherit_priority) { 15053a5a1b3Sopenharmony_ci pa_mutex *m; 15153a5a1b3Sopenharmony_ci 15253a5a1b3Sopenharmony_ci pa_assert(s); 15353a5a1b3Sopenharmony_ci 15453a5a1b3Sopenharmony_ci /* First, check if already initialized and short cut */ 15553a5a1b3Sopenharmony_ci if ((m = pa_atomic_ptr_load(&s->ptr))) 15653a5a1b3Sopenharmony_ci return m; 15753a5a1b3Sopenharmony_ci 15853a5a1b3Sopenharmony_ci /* OK, not initialized, so let's allocate, and fill in */ 15953a5a1b3Sopenharmony_ci m = pa_mutex_new(recursive, inherit_priority); 16053a5a1b3Sopenharmony_ci if ((pa_atomic_ptr_cmpxchg(&s->ptr, NULL, m))) 16153a5a1b3Sopenharmony_ci return m; 16253a5a1b3Sopenharmony_ci 16353a5a1b3Sopenharmony_ci pa_mutex_free(m); 16453a5a1b3Sopenharmony_ci 16553a5a1b3Sopenharmony_ci /* Him, filling in failed, so someone else must have filled in 16653a5a1b3Sopenharmony_ci * already */ 16753a5a1b3Sopenharmony_ci pa_assert_se(m = pa_atomic_ptr_load(&s->ptr)); 16853a5a1b3Sopenharmony_ci return m; 16953a5a1b3Sopenharmony_ci} 170