Commit 22a4046d authored by KO Myung-Hun's avatar KO Myung-Hun Committed by Michael Niedermayer

compat/os2threads: Improve pthread_cond_xxx() functions

1. Manipulate waiting count in pthread_cond_wait()
2. Use builtin atomic functions to manipulate waiting count
Signed-off-by: 's avatarMichael Niedermayer <michael@niedermayer.cc>
parent b65ea6ab
...@@ -32,6 +32,7 @@ ...@@ -32,6 +32,7 @@
#undef __STRICT_ANSI__ /* for _beginthread() */ #undef __STRICT_ANSI__ /* for _beginthread() */
#include <stdlib.h> #include <stdlib.h>
#include <sys/builtin.h>
#include <sys/fmutex.h> #include <sys/fmutex.h>
#include "libavutil/mem.h" #include "libavutil/mem.h"
...@@ -43,8 +44,9 @@ typedef HMTX pthread_mutex_t; ...@@ -43,8 +44,9 @@ typedef HMTX pthread_mutex_t;
typedef void pthread_mutexattr_t; typedef void pthread_mutexattr_t;
typedef struct { typedef struct {
HEV event_sem; HEV event_sem;
int wait_count; HEV ack_sem;
volatile unsigned wait_count;
} pthread_cond_t; } pthread_cond_t;
typedef void pthread_condattr_t; typedef void pthread_condattr_t;
...@@ -124,6 +126,7 @@ static av_always_inline int pthread_mutex_unlock(pthread_mutex_t *mutex) ...@@ -124,6 +126,7 @@ static av_always_inline int pthread_mutex_unlock(pthread_mutex_t *mutex)
static av_always_inline int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr) static av_always_inline int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr)
{ {
DosCreateEventSem(NULL, &cond->event_sem, DCE_POSTONE, FALSE); DosCreateEventSem(NULL, &cond->event_sem, DCE_POSTONE, FALSE);
DosCreateEventSem(NULL, &cond->ack_sem, DCE_POSTONE, FALSE);
cond->wait_count = 0; cond->wait_count = 0;
...@@ -133,16 +136,16 @@ static av_always_inline int pthread_cond_init(pthread_cond_t *cond, const pthrea ...@@ -133,16 +136,16 @@ static av_always_inline int pthread_cond_init(pthread_cond_t *cond, const pthrea
static av_always_inline int pthread_cond_destroy(pthread_cond_t *cond) static av_always_inline int pthread_cond_destroy(pthread_cond_t *cond)
{ {
DosCloseEventSem(cond->event_sem); DosCloseEventSem(cond->event_sem);
DosCloseEventSem(cond->ack_sem);
return 0; return 0;
} }
static av_always_inline int pthread_cond_signal(pthread_cond_t *cond) static av_always_inline int pthread_cond_signal(pthread_cond_t *cond)
{ {
if (cond->wait_count > 0) { if (!__atomic_cmpxchg32(&cond->wait_count, 0, 0)) {
DosPostEventSem(cond->event_sem); DosPostEventSem(cond->event_sem);
DosWaitEventSem(cond->ack_sem, SEM_INDEFINITE_WAIT);
cond->wait_count--;
} }
return 0; return 0;
...@@ -150,23 +153,24 @@ static av_always_inline int pthread_cond_signal(pthread_cond_t *cond) ...@@ -150,23 +153,24 @@ static av_always_inline int pthread_cond_signal(pthread_cond_t *cond)
static av_always_inline int pthread_cond_broadcast(pthread_cond_t *cond) static av_always_inline int pthread_cond_broadcast(pthread_cond_t *cond)
{ {
while (cond->wait_count > 0) { while (!__atomic_cmpxchg32(&cond->wait_count, 0, 0))
DosPostEventSem(cond->event_sem); pthread_cond_signal(cond);
cond->wait_count--;
}
return 0; return 0;
} }
static av_always_inline int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex) static av_always_inline int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex)
{ {
cond->wait_count++; __atomic_increment(&cond->wait_count);
pthread_mutex_unlock(mutex); pthread_mutex_unlock(mutex);
DosWaitEventSem(cond->event_sem, SEM_INDEFINITE_WAIT); DosWaitEventSem(cond->event_sem, SEM_INDEFINITE_WAIT);
__atomic_decrement(&cond->wait_count);
DosPostEventSem(cond->ack_sem);
pthread_mutex_lock(mutex); pthread_mutex_lock(mutex);
return 0; return 0;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment