etlegacy-libs/openal/common/rwlock.c

60 lines
1.6 KiB
C
Raw Normal View History

2015-12-12 21:07:33 +00:00
#include "config.h"
#include "rwlock.h"
#include "bool.h"
#include "atomic.h"
#include "threads.h"
/* A simple spinlock. Yield the thread while the given integer is set by
* another. Could probably be improved... */
#define LOCK(l) do { \
2017-07-01 14:18:03 +00:00
while(ATOMIC_FLAG_TEST_AND_SET(&(l), almemory_order_acq_rel) == true) \
2015-12-12 21:07:33 +00:00
althrd_yield(); \
} while(0)
2017-07-01 14:18:03 +00:00
#define UNLOCK(l) ATOMIC_FLAG_CLEAR(&(l), almemory_order_release)
2015-12-12 21:07:33 +00:00
void RWLockInit(RWLock *lock)
{
InitRef(&lock->read_count, 0);
InitRef(&lock->write_count, 0);
2017-07-01 14:18:03 +00:00
ATOMIC_FLAG_CLEAR(&lock->read_lock, almemory_order_relaxed);
ATOMIC_FLAG_CLEAR(&lock->read_entry_lock, almemory_order_relaxed);
ATOMIC_FLAG_CLEAR(&lock->write_lock, almemory_order_relaxed);
2015-12-12 21:07:33 +00:00
}
void ReadLock(RWLock *lock)
{
LOCK(lock->read_entry_lock);
LOCK(lock->read_lock);
2017-07-01 14:18:03 +00:00
/* NOTE: ATOMIC_ADD returns the *old* value! */
if(ATOMIC_ADD(&lock->read_count, 1, almemory_order_acq_rel) == 0)
2015-12-12 21:07:33 +00:00
LOCK(lock->write_lock);
UNLOCK(lock->read_lock);
UNLOCK(lock->read_entry_lock);
}
void ReadUnlock(RWLock *lock)
{
2017-07-01 14:18:03 +00:00
/* NOTE: ATOMIC_SUB returns the *old* value! */
if(ATOMIC_SUB(&lock->read_count, 1, almemory_order_acq_rel) == 1)
2015-12-12 21:07:33 +00:00
UNLOCK(lock->write_lock);
}
void WriteLock(RWLock *lock)
{
2017-07-01 14:18:03 +00:00
if(ATOMIC_ADD(&lock->write_count, 1, almemory_order_acq_rel) == 0)
2015-12-12 21:07:33 +00:00
LOCK(lock->read_lock);
LOCK(lock->write_lock);
}
void WriteUnlock(RWLock *lock)
{
UNLOCK(lock->write_lock);
2017-07-01 14:18:03 +00:00
if(ATOMIC_SUB(&lock->write_count, 1, almemory_order_acq_rel) == 1)
2015-12-12 21:07:33 +00:00
UNLOCK(lock->read_lock);
}