/* Simple reader/writer locks. Copyright (C) 1994, 1995, 1996 Free Software Foundation, Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #ifndef _RWLOCK_H #define _RWLOCK_H 1 #include <cthreads.h> #include <assert.h> struct rwlock { struct mutex master; struct condition wakeup; int readers; int writers_waiting; int readers_waiting; }; #ifdef _RWLOCK_DEFINE_FUNCTIONS #undef _EXTERN_INLINE #define _EXTERN_INLINE #else /* ! _RWLOCK_DEFINE_FUNCTIONS */ #ifndef _EXTERN_INLINE #define _EXTERN_INLINE extern __inline #endif #endif /* _RWLOCK_DEFINE_FUNCTIONS */ /* Get a reader lock on reader-writer lock LOCK for disknode DN */ _EXTERN_INLINE void rwlock_reader_lock (struct rwlock *lock) { mutex_lock (&lock->master); if (lock->readers == -1 || lock->writers_waiting) { lock->readers_waiting++; do condition_wait (&lock->wakeup, &lock->master); while (lock->readers == -1 || lock->writers_waiting); lock->readers_waiting--; } lock->readers++; mutex_unlock (&lock->master); } /* Get a writer lock on reader-writer lock LOCK for disknode DN */ _EXTERN_INLINE void rwlock_writer_lock (struct rwlock *lock) { mutex_lock (&lock->master); if (lock->readers) { lock->writers_waiting++; do condition_wait (&lock->wakeup, &lock->master); while (lock->readers); lock->writers_waiting--; } lock->readers = -1; mutex_unlock (&lock->master); } /* Release a reader lock on reader-writer lock LOCK for disknode DN */ _EXTERN_INLINE void rwlock_reader_unlock (struct rwlock *lock) { mutex_lock (&lock->master); assert (lock->readers); lock->readers--; if (lock->readers_waiting || lock->writers_waiting) condition_broadcast (&lock->wakeup); mutex_unlock (&lock->master); } /* Release a writer lock on reader-writer lock LOCK for disknode DN */ _EXTERN_INLINE void rwlock_writer_unlock (struct rwlock *lock) { mutex_lock (&lock->master); assert (lock->readers == -1); lock->readers = 0; if (lock->readers_waiting || lock->writers_waiting) condition_broadcast (&lock->wakeup); mutex_unlock (&lock->master); } /* Initialize reader-writer lock LOCK */ _EXTERN_INLINE void rwlock_init (struct rwlock *lock) { mutex_init (&lock->master); condition_init (&lock->wakeup); lock->readers = 0; lock->readers_waiting = 0; lock->writers_waiting = 0; } #define RWLOCK_INITIALIZER \ { MUTEX_INITIALIZER, CONDITION_INITIALIZER, 0, 0, 0 } #endif /* rwlock.h */