#ifndef _NISDB_RW_H
#define _NISDB_RW_H
#include <pthread.h>
#include <thread.h>
#include <synch.h>
#include <stdlib.h>
#include <malloc.h>
#include <sys/errno.h>
#ifdef __cplusplus
extern "C" {
#endif
#define INV_PTHREAD_ID 0
#define DEFAULTNISDBRWLOCK_RW {DEFAULTMUTEX, DEFAULTCV, 0, 0, \
0, {INV_PTHREAD_ID, 0, 0, 0}, \
0, 0, {INV_PTHREAD_ID, 0, 0, 0}}
#define DEFAULTNISDBRWLOCK_W {DEFAULTMUTEX, DEFAULTCV, 0, 1, \
0, {INV_PTHREAD_ID, 0, 0, 0}, \
0, 0, {INV_PTHREAD_ID, 0, 0, 0}}
#define DEFAULTNISDBRWLOCK DEFAULTNISDBRWLOCK_W
#define NISDB_FORCE_WRITE 1
#ifdef NISDB_MT_DEBUG
#define DECLMUTEXLOCK(var) pthread_mutex_t var ## _pmutex = \
PTHREAD_MUTEX_INITIALIZER; \
pthread_t var ## _owner = INV_PTHREAD_ID
#define USEMUTEXLOCK(var) extern pthread_mutex_t var ## _pmutex; \
extern pthread_t var ## _owner
#define STRUCTMUTEXLOCK(var) pthread_mutex_t var ## _pmutex; \
pthread_t var ## _owner
#define INITMUTEX(var) (void) pthread_mutex_init(&var ## _pmutex, 0)
#define MUTEXLOCK(var, msg) if (var ## _owner != pthread_self()) { \
pthread_mutex_lock(&var ## _pmutex); \
var ## _owner = pthread_self(); \
} else \
abort();
#define MUTEXUNLOCK(var, msg) if (var ## _owner == pthread_self()) { \
var ## _owner = INV_PTHREAD_ID; \
pthread_mutex_unlock(&var ## _pmutex);\
} else \
abort();
#define ASSERTMUTEXHELD(var) if (var ## _owner != pthread_self()) \
abort();
#define DECLRWLOCK(var) __nisdb_rwlock_t var ## _rwlock = \
DEFAULTNISDBRWLOCK
#define USERWLOCK(var) extern __nisdb_rwlock_t var ## _rwlock
#define STRUCTRWLOCK(var) __nisdb_rwlock_t var ## _rwlock
#define INITRW(var) (void) __nisdb_rwinit(&var ## _rwlock)
#define READLOCKOK(var) (void) __nisdb_rw_readlock_ok(&var ## _rwlock)
#define RLOCK(var) __nisdb_rlock(&var ## _rwlock)
#define WLOCK(var) __nisdb_wlock(&var ## _rwlock)
#define TRYWLOCK(var) __nisdb_wlock_trylock(&var ## _rwlock, 1)
#define RULOCK(var) __nisdb_rulock(&var ## _rwlock)
#define WULOCK(var) __nisdb_wulock(&var ## _rwlock)
#define DESTROYRW(var) __nisdb_destroy_lock(&var ## _rwlock)
#define ASSERTWHELD(var) if (__nisdb_assert_wheld(&var ## _rwlock) \
!= 0) \
abort();
#define ASSERTRHELD(var) if (__nisdb_assert_rheld(&var ## _rwlock) \
!= 0) \
abort();
#else
#define DECLMUTEXLOCK(var) pthread_mutex_t var ## _pmutex = \
PTHREAD_MUTEX_INITIALIZER
#define USEMUTEXLOCK(var) extern pthread_mutex_t var ## _pmutex
#define STRUCTMUTEXLOCK(var) pthread_mutex_t var ## _pmutex
#define INITMUTEX(var) (void) pthread_mutex_init(&var ## _pmutex, 0)
#define MUTEXLOCK(var, msg) pthread_mutex_lock(&var ## _pmutex)
#define MUTEXUNLOCK(var, msg) pthread_mutex_unlock(&var ## _pmutex)
#define DECLRWLOCK(var) __nisdb_rwlock_t var ## _rwlock = \
DEFAULTNISDBRWLOCK
#define USERWLOCK(var) extern __nisdb_rwlock_t var ## _rwlock
#define STRUCTRWLOCK(var) __nisdb_rwlock_t var ## _rwlock
#define INITRW(var) (void) __nisdb_rwinit(&var ## _rwlock)
#define READLOCKOK(var) (void) __nisdb_rw_readlock_ok(&var ## _rwlock)
#define RLOCK(var) __nisdb_rlock(&var ## _rwlock)
#define WLOCK(var) __nisdb_wlock(&var ## _rwlock)
#define TRYWLOCK(var) __nisdb_wlock_trylock(&var ## _rwlock, 1)
#define RULOCK(var) __nisdb_rulock(&var ## _rwlock)
#define WULOCK(var) __nisdb_wulock(&var ## _rwlock)
#define DESTROYRW(var) __nisdb_destroy_lock(&var ## _rwlock)
#define ASSERTMUTEXHELD(var)
#define ASSERTWHELD(var)
#define ASSERTRHELD(var)
#endif
typedef struct __nisdb_rwlock {
pthread_t id;
uint32_t count;
uint32_t wait;
struct __nisdb_rwlock *next;
} __nisdb_rl_t;
typedef struct {
mutex_t mutex;
cond_t cv;
uint32_t destroyed;
uint32_t force_write;
uint32_t writer_count;
__nisdb_rl_t writer;
uint32_t reader_count;
uint32_t reader_blocked;
__nisdb_rl_t reader;
} __nisdb_rwlock_t;
extern int __nisdb_rwinit(__nisdb_rwlock_t *);
extern int __nisdb_rw_readlock_ok(__nisdb_rwlock_t *rw);
extern int __nisdb_rw_force_writelock(__nisdb_rwlock_t *rw);
extern int __nisdb_wlock(__nisdb_rwlock_t *);
extern int __nisdb_wlock_trylock(__nisdb_rwlock_t *, int);
extern int __nisdb_rlock(__nisdb_rwlock_t *);
extern int __nisdb_wulock(__nisdb_rwlock_t *);
extern int __nisdb_rulock(__nisdb_rwlock_t *);
extern int __nisdb_assert_wheld(__nisdb_rwlock_t *);
extern int __nisdb_assert_rheld(__nisdb_rwlock_t *);
extern int __nisdb_destroy_lock(__nisdb_rwlock_t *);
extern void __nisdb_lock_report(__nisdb_rwlock_t *rw);
#ifdef __cplusplus
}
#endif
#endif