blob: 7443a79702740a279d49eb6c1ed497b73d871a9f [file] [log] [blame] [edit]
/*
* BRLTTY - A background process providing access to the console screen (when in
* text mode) for a blind person using a refreshable braille display.
*
* Copyright (C) 1995-2023 by The BRLTTY Developers.
*
* BRLTTY comes with ABSOLUTELY NO WARRANTY.
*
* This is free software, placed under the terms of the
* GNU Lesser General Public License, as published by the Free Software
* Foundation; either version 2.1 of the License, or (at your option) any
* later version. Please see the file LICENSE-LGPL for details.
*
* Web Page: http://brltty.app/
*
* This software is maintained by Dave Mielke <dave@mielke.cc>.
*/
#include "prologue.h"
#include <string.h>
#include <errno.h>
#include "lock.h"
#include "log.h"
#include "get_thread.h"
#undef CAN_LOCK
#if defined(PTHREAD_RWLOCK_INITIALIZER)
#define CAN_LOCK
struct LockDescriptorStruct {
pthread_rwlock_t lock;
};
static int
constructLockDescriptor (LockDescriptor *lock) {
int error;
if (!(error = pthread_rwlock_init(&lock->lock, NULL))) {
return 1;
} else {
logActionError(error, "pthread_rwlock_init");
}
return 0;
}
static void
destructLockDescriptor (LockDescriptor *lock) {
pthread_rwlock_destroy(&lock->lock);
}
int
obtainLock (LockDescriptor *lock, LockOptions options) {
if (options & LOCK_Exclusive) {
if (options & LOCK_NoWait) return !pthread_rwlock_trywrlock(&lock->lock);
pthread_rwlock_wrlock(&lock->lock);
} else {
if (options & LOCK_NoWait) return !pthread_rwlock_tryrdlock(&lock->lock);
pthread_rwlock_rdlock(&lock->lock);
}
return 1;
}
void
releaseLock (LockDescriptor *lock) {
pthread_rwlock_unlock(&lock->lock);
}
#elif defined(PTHREAD_MUTEX_INITIALIZER)
#define CAN_LOCK
struct LockDescriptorStruct {
pthread_mutex_t mutex;
pthread_cond_t read;
pthread_cond_t write;
int count;
unsigned int writers;
};
static int
constructLockDescriptor (LockDescriptor *lock) {
int error;
if (!(error = pthread_cond_init(&lock->read, NULL))) {
if (!(error = pthread_cond_init(&lock->write, NULL))) {
if (!(error = pthread_mutex_init(&lock->mutex, NULL))) {
lock->count = 0;
lock->writers = 0;
return 1;
} else {
logActionError(error, "pthread_mutex_init");
}
pthread_cond_destroy(&lock->write);
} else {
logActionError(error, "pthread_cond_init");
}
pthread_cond_destroy(&lock->read);
} else {
logActionError(error, "pthread_cond_init");
}
return 0;
}
static void
destructLockDescriptor (LockDescriptor *lock) {
pthread_mutex_destroy(&lock->mutex);
pthread_cond_destroy(&lock->read);
pthread_cond_destroy(&lock->write);
}
int
obtainLock (LockDescriptor *lock, LockOptions options) {
int locked = 0;
pthread_mutex_lock(&lock->mutex);
if (options & LOCK_Exclusive) {
while (lock->count) {
if (options & LOCK_NoWait) goto done;
lock->writers += 1;
pthread_cond_wait(&lock->write, &lock->mutex);
lock->writers -= 1;
}
lock->count = -1;
} else {
while (lock->count < 0) {
if (options & LOCK_NoWait) goto done;
pthread_cond_wait(&lock->read, &lock->mutex);
}
lock->count += 1;
}
locked = 1;
done:
pthread_mutex_unlock(&lock->mutex);
return locked;
}
void
releaseLock (LockDescriptor *lock) {
pthread_mutex_lock(&lock->mutex);
if (lock->count < 0) {
lock->count = 0;
} else if (--lock->count) {
goto done;
}
if (lock->writers) {
pthread_cond_signal(&lock->write);
} else {
pthread_cond_broadcast(&lock->read);
}
done:
pthread_mutex_unlock(&lock->mutex);
}
#endif /* lock paradigm */
#ifdef CAN_LOCK
LockDescriptor *
newLockDescriptor (void) {
LockDescriptor *lock;
if ((lock = malloc(sizeof(*lock)))) {
memset(lock, 0, sizeof(*lock));
if (constructLockDescriptor(lock)) return lock;
free(lock);
lock = NULL;
} else {
logMallocError();
}
return lock;
}
void
freeLockDescriptor (LockDescriptor *lock) {
destructLockDescriptor(lock);
free(lock);
}
LockDescriptor *
getLockDescriptor (LockDescriptor **lock, const char *name) {
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
int isNew = 0;
pthread_mutex_lock(&mutex);
if (!*lock) {
if ((*lock = newLockDescriptor())) {
isNew = 1;
}
}
pthread_mutex_unlock(&mutex);
if (isNew) {
logMessage(LOG_DEBUG, "lock descriptor allocated: %s", name);
}
return *lock;
}
#else /* CAN_LOCK */
#warning thread lock support not available on this platform
int
obtainLock (LockDescriptor *lock, LockOptions options) {
return 1;
}
void
releaseLock (LockDescriptor *lock) {
}
LockDescriptor *
newLockDescriptor (void) {
errno = ENOSYS;
return NULL;
}
void
freeLockDescriptor (LockDescriptor *lock) {
}
LockDescriptor *
getLockDescriptor (LockDescriptor **lock, const char *name) {
return *lock;
}
#endif /* CAN_LOCK */