syslock.c   [plain text]


/*
 * 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;
}