/*
* Copyright (c) 1999 Apple Computer, Inc. All rights reserved.
*
* @APPLE_LICENSE_HEADER_START@
*
* "Portions Copyright (c) 1999 Apple Computer, Inc. All Rights
* Reserved. This file contains Original Code and/or Modifications of
* Original Code as defined in and that are subject to the Apple Public
* Source License Version 1.0 (the 'License'). You may not use this file
* except in compliance with the License. Please obtain a copy of the
* License at http://www.apple.com/publicsource and read it before using
* this file.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
* INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT. Please see the
* License for the specific language governing rights and limitations
* under the License."
*
* @APPLE_LICENSE_HEADER_END@
*/
#include <NetInfo/syslock.h>
#include <string.h>
#ifdef _THREAD_TYPE_PTHREAD_
#include <stdlib.h>
#endif
typedef struct
{
char *name;
syslock *lock;
}
named_syslock_t;
static named_syslock_t *named_lock = NULL;
static unsigned int named_lock_count = 0;
static thread_id_t
_thread_id(void)
{
#ifdef _THREAD_TYPE_PTHREAD_
return (unsigned int)pthread_self();
#else
return (unsigned int)cthread_self();
#endif
}
static void
_internal_lock(syslock *s)
{
#ifdef _THREAD_TYPE_PTHREAD_
pthread_mutex_lock(s->internal);
#else
spin_lock(&(s->internal));
#endif
}
static void
_internal_unlock(syslock *s)
{
#ifdef _THREAD_TYPE_PTHREAD_
pthread_mutex_unlock(s->internal);
#else
spin_unlock(&(s->internal));
#endif
}
static void
_main_lock(syslock *s)
{
#ifdef _THREAD_TYPE_PTHREAD_
pthread_mutex_lock(s->mutex);
#else
mutex_lock(s->mutex);
#endif
}
static void
_main_unlock(syslock *s)
{
#ifdef _THREAD_TYPE_PTHREAD_
pthread_mutex_unlock(s->mutex);
#else
mutex_unlock(s->mutex);
#endif
}
syslock *
syslock_new(bool_t recursive)
{
syslock *s;
s = (syslock *)malloc(sizeof(syslock));
memset(s, 0, sizeof(syslock));
#ifdef _THREAD_TYPE_PTHREAD_
s->mutex = (pthread_mutex_t *)malloc(sizeof(pthread_mutex_t));
pthread_mutex_init(s->mutex, NULL);
s->internal = (pthread_mutex_t *)malloc(sizeof(pthread_mutex_t));
pthread_mutex_init(s->internal, NULL);
s->condition = (pthread_cond_t *)malloc(sizeof(pthread_cond_t));
pthread_cond_init(s->condition, NULL);
#else
s->mutex = mutex_alloc();
mutex_init(s->mutex);
s->internal = 0;
s->condition = condition_alloc();
condition_init(s->condition);
#endif
s->locked = 0;
s->thread = NO_THREAD;
s->recursive = recursive;
return s;
}
void
syslock_free(syslock *s)
{
if (s == NULL) return;
#ifdef _THREAD_TYPE_PTHREAD_
pthread_mutex_destroy(s->mutex);
free(s->mutex);
pthread_mutex_destroy(s->internal);
free(s->internal);
pthread_cond_destroy(s->condition);
free(s->condition);
#else
mutex_free(s->mutex);
condition_free(s->condition);
#endif
free(s);
}
void
syslock_lock(syslock *s)
{
thread_id_t t;
if (s == NULL) return;
t = _thread_id();
_internal_lock(s);
if ((s->locked > 0) && s->recursive && (s->thread == t))
{
/* Recursive locks just increment the locked counter */
s->locked++;
_internal_unlock(s);
return;
}
_internal_unlock(s);
_main_lock(s);
_internal_lock(s);
s->locked = 1;
s->thread = _thread_id();
_internal_unlock(s);
}
void
syslock_unlock(syslock *s)
{
int unlock_me;
if (s == NULL) return;
unlock_me = 0;
_internal_lock(s);
if (s->locked > 0) s->locked--;
if (s->locked == 0)
{
s->thread = NO_THREAD;
unlock_me = 1;
}
_internal_unlock(s);
if (unlock_me == 1) _main_unlock(s);
}
bool_t
syslock_trylock(syslock *s)
{
int t;
if (s == NULL) return FALSE;
_internal_lock(s);
if ((s->locked > 0) && s->recursive && (s->thread == _thread_id()))
{
/* Recursive locks just increment the locked counter */
s->locked++;
_internal_unlock(s);
return TRUE;
}
_internal_unlock(s);
#ifdef _THREAD_TYPE_PTHREAD_
t = pthread_mutex_trylock(s->mutex);
#else
t = mutex_try_lock(s->mutex);
#endif
if (t != 0) return FALSE;
_internal_lock(s);
s->locked = 1;
s->thread = _thread_id();
_internal_unlock(s);
return TRUE;
}
bool_t
syslock_is_locked(syslock *s)
{
bool_t ret;
if (s == NULL) return FALSE;
ret = FALSE;
_internal_lock(s);
if (s->locked > 0) ret = TRUE;
_internal_unlock(s);
return ret;
}
void
syslock_signal_wait(syslock *s)
{
if (s == NULL) return;
#ifdef _THREAD_TYPE_PTHREAD_
pthread_cond_wait(s->condition, s->mutex);
#else
condition_wait(s->condition, s->mutex);
#endif
}
void
syslock_signal_send(syslock *s)
{
if (s == NULL) return;
#ifdef _THREAD_TYPE_PTHREAD_
pthread_cond_signal(s->condition);
#else
condition_signal(s->condition);
#endif
}
void
syslock_signal_broadcast(syslock *s)
{
if (s == NULL) return;
#ifdef _THREAD_TYPE_PTHREAD_
pthread_cond_broadcast(s->condition);
#else
condition_broadcast(s->condition);
#endif
}
bool_t
syslock_set_name(syslock *s, char *n)
{
unsigned int i, j, unset;
if (s == NULL) return FALSE;
unset = 0;
if (n == NULL) unset = 1;
for (i = 0; i < named_lock_count; i++)
{
if ((unset == 1) && (s == named_lock[i].lock))
{
free(named_lock[i].name);
for (j = i + 1; j < named_lock_count; j++)
{
named_lock[j - 1] = named_lock[j];
}
if (named_lock_count == 1)
{
free(named_lock);
named_lock = NULL;
return TRUE;
}
named_lock_count--;
named_lock = (named_syslock_t *)realloc(named_lock, named_lock_count * sizeof(named_syslock_t));
return TRUE;
}
if ((unset == 0) && (strcmp(named_lock[i].name, n) == 0))
{
if (s == named_lock[i].lock) return TRUE;
return FALSE;
}
}
if (unset == 1) return FALSE;
if (named_lock_count == 0)
{
named_lock = (named_syslock_t *)malloc(sizeof(named_syslock_t));
}
else
{
named_lock = (named_syslock_t *)realloc(named_lock, (named_lock_count + 1) * sizeof(named_syslock_t));
}
named_lock[named_lock_count].name = strdup(n);
named_lock[named_lock_count].lock = s;
named_lock_count++;
return TRUE;
}
syslock *syslock_get(char *name)
{
unsigned int i;
if (name == NULL) return NULL;
for (i = 0; i < named_lock_count; i++)
{
if (strcmp(name, named_lock[i].name) == 0) return named_lock[i].lock;
}
return NULL;
}
syntax highlighted by Code2HTML, v. 0.9.1