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