/* * 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 #include #ifdef _THREAD_TYPE_PTHREAD_ #include #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; }