153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2006 Pierre Ossman <ossman@cendio.se> for Cendio AB 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 <windows.h> 2553a5a1b3Sopenharmony_ci 2653a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h> 2753a5a1b3Sopenharmony_ci#include <pulsecore/hashmap.h> 2853a5a1b3Sopenharmony_ci 2953a5a1b3Sopenharmony_ci#include "mutex.h" 3053a5a1b3Sopenharmony_ci 3153a5a1b3Sopenharmony_cistruct pa_mutex { 3253a5a1b3Sopenharmony_ci CRITICAL_SECTION mutex; 3353a5a1b3Sopenharmony_ci}; 3453a5a1b3Sopenharmony_ci 3553a5a1b3Sopenharmony_cistruct pa_cond { 3653a5a1b3Sopenharmony_ci pa_hashmap *wait_events; 3753a5a1b3Sopenharmony_ci}; 3853a5a1b3Sopenharmony_ci 3953a5a1b3Sopenharmony_cipa_mutex* pa_mutex_new(bool recursive, bool inherit_priority) { 4053a5a1b3Sopenharmony_ci pa_mutex *m; 4153a5a1b3Sopenharmony_ci 4253a5a1b3Sopenharmony_ci m = pa_xnew(pa_mutex, 1); 4353a5a1b3Sopenharmony_ci 4453a5a1b3Sopenharmony_ci InitializeCriticalSection(&m->mutex); 4553a5a1b3Sopenharmony_ci 4653a5a1b3Sopenharmony_ci return m; 4753a5a1b3Sopenharmony_ci} 4853a5a1b3Sopenharmony_ci 4953a5a1b3Sopenharmony_civoid pa_mutex_free(pa_mutex *m) { 5053a5a1b3Sopenharmony_ci assert(m); 5153a5a1b3Sopenharmony_ci 5253a5a1b3Sopenharmony_ci DeleteCriticalSection(&m->mutex); 5353a5a1b3Sopenharmony_ci pa_xfree(m); 5453a5a1b3Sopenharmony_ci} 5553a5a1b3Sopenharmony_ci 5653a5a1b3Sopenharmony_civoid pa_mutex_lock(pa_mutex *m) { 5753a5a1b3Sopenharmony_ci assert(m); 5853a5a1b3Sopenharmony_ci 5953a5a1b3Sopenharmony_ci EnterCriticalSection(&m->mutex); 6053a5a1b3Sopenharmony_ci} 6153a5a1b3Sopenharmony_ci 6253a5a1b3Sopenharmony_civoid pa_mutex_unlock(pa_mutex *m) { 6353a5a1b3Sopenharmony_ci assert(m); 6453a5a1b3Sopenharmony_ci 6553a5a1b3Sopenharmony_ci LeaveCriticalSection(&m->mutex); 6653a5a1b3Sopenharmony_ci} 6753a5a1b3Sopenharmony_ci 6853a5a1b3Sopenharmony_cipa_cond *pa_cond_new(void) { 6953a5a1b3Sopenharmony_ci pa_cond *c; 7053a5a1b3Sopenharmony_ci 7153a5a1b3Sopenharmony_ci c = pa_xnew(pa_cond, 1); 7253a5a1b3Sopenharmony_ci c->wait_events = pa_hashmap_new(NULL, NULL); 7353a5a1b3Sopenharmony_ci assert(c->wait_events); 7453a5a1b3Sopenharmony_ci 7553a5a1b3Sopenharmony_ci return c; 7653a5a1b3Sopenharmony_ci} 7753a5a1b3Sopenharmony_ci 7853a5a1b3Sopenharmony_civoid pa_cond_free(pa_cond *c) { 7953a5a1b3Sopenharmony_ci assert(c); 8053a5a1b3Sopenharmony_ci 8153a5a1b3Sopenharmony_ci pa_hashmap_free(c->wait_events); 8253a5a1b3Sopenharmony_ci pa_xfree(c); 8353a5a1b3Sopenharmony_ci} 8453a5a1b3Sopenharmony_ci 8553a5a1b3Sopenharmony_civoid pa_cond_signal(pa_cond *c, int broadcast) { 8653a5a1b3Sopenharmony_ci assert(c); 8753a5a1b3Sopenharmony_ci 8853a5a1b3Sopenharmony_ci if (pa_hashmap_size(c->wait_events) == 0) 8953a5a1b3Sopenharmony_ci return; 9053a5a1b3Sopenharmony_ci 9153a5a1b3Sopenharmony_ci if (broadcast) 9253a5a1b3Sopenharmony_ci SetEvent(pa_hashmap_first(c->wait_events)); 9353a5a1b3Sopenharmony_ci else { 9453a5a1b3Sopenharmony_ci void *iter; 9553a5a1b3Sopenharmony_ci const void *key; 9653a5a1b3Sopenharmony_ci HANDLE event; 9753a5a1b3Sopenharmony_ci 9853a5a1b3Sopenharmony_ci iter = NULL; 9953a5a1b3Sopenharmony_ci while (1) { 10053a5a1b3Sopenharmony_ci pa_hashmap_iterate(c->wait_events, &iter, &key); 10153a5a1b3Sopenharmony_ci if (key == NULL) 10253a5a1b3Sopenharmony_ci break; 10353a5a1b3Sopenharmony_ci event = (HANDLE)pa_hashmap_get(c->wait_events, key); 10453a5a1b3Sopenharmony_ci SetEvent(event); 10553a5a1b3Sopenharmony_ci } 10653a5a1b3Sopenharmony_ci } 10753a5a1b3Sopenharmony_ci} 10853a5a1b3Sopenharmony_ci 10953a5a1b3Sopenharmony_ciint pa_cond_wait(pa_cond *c, pa_mutex *m) { 11053a5a1b3Sopenharmony_ci HANDLE event; 11153a5a1b3Sopenharmony_ci 11253a5a1b3Sopenharmony_ci assert(c); 11353a5a1b3Sopenharmony_ci assert(m); 11453a5a1b3Sopenharmony_ci 11553a5a1b3Sopenharmony_ci event = CreateEvent(NULL, FALSE, FALSE, NULL); 11653a5a1b3Sopenharmony_ci assert(event); 11753a5a1b3Sopenharmony_ci 11853a5a1b3Sopenharmony_ci pa_hashmap_put(c->wait_events, event, event); 11953a5a1b3Sopenharmony_ci 12053a5a1b3Sopenharmony_ci pa_mutex_unlock(m); 12153a5a1b3Sopenharmony_ci 12253a5a1b3Sopenharmony_ci WaitForSingleObject(event, INFINITE); 12353a5a1b3Sopenharmony_ci 12453a5a1b3Sopenharmony_ci pa_mutex_lock(m); 12553a5a1b3Sopenharmony_ci 12653a5a1b3Sopenharmony_ci pa_hashmap_remove(c->wait_events, event); 12753a5a1b3Sopenharmony_ci 12853a5a1b3Sopenharmony_ci CloseHandle(event); 12953a5a1b3Sopenharmony_ci 13053a5a1b3Sopenharmony_ci return 0; 13153a5a1b3Sopenharmony_ci} 13253a5a1b3Sopenharmony_ci 13353a5a1b3Sopenharmony_ci/* This is a copy of the function in mutex-posix.c */ 13453a5a1b3Sopenharmony_cipa_mutex* pa_static_mutex_get(pa_static_mutex *s, bool recursive, bool inherit_priority) { 13553a5a1b3Sopenharmony_ci pa_mutex *m; 13653a5a1b3Sopenharmony_ci 13753a5a1b3Sopenharmony_ci pa_assert(s); 13853a5a1b3Sopenharmony_ci 13953a5a1b3Sopenharmony_ci /* First, check if already initialized and short cut */ 14053a5a1b3Sopenharmony_ci if ((m = pa_atomic_ptr_load(&s->ptr))) 14153a5a1b3Sopenharmony_ci return m; 14253a5a1b3Sopenharmony_ci 14353a5a1b3Sopenharmony_ci /* OK, not initialized, so let's allocate, and fill in */ 14453a5a1b3Sopenharmony_ci m = pa_mutex_new(recursive, inherit_priority); 14553a5a1b3Sopenharmony_ci if ((pa_atomic_ptr_cmpxchg(&s->ptr, NULL, m))) 14653a5a1b3Sopenharmony_ci return m; 14753a5a1b3Sopenharmony_ci 14853a5a1b3Sopenharmony_ci pa_mutex_free(m); 14953a5a1b3Sopenharmony_ci 15053a5a1b3Sopenharmony_ci /* Him, filling in failed, so someone else must have filled in 15153a5a1b3Sopenharmony_ci * already */ 15253a5a1b3Sopenharmony_ci pa_assert_se(m = pa_atomic_ptr_load(&s->ptr)); 15353a5a1b3Sopenharmony_ci return m; 15453a5a1b3Sopenharmony_ci} 155