/* * %CopyrightBegin% * * Copyright Ericsson AB 1997-2009. All Rights Reserved. * * The contents of this file are subject to the Erlang Public License, * Version 1.1, (the "License"); you may not use this file except in * compliance with the License. You should have received a copy of the * Erlang Public License along with this software. If not, it can be * retrieved online at http://www.erlang.org/. * * Software distributed under the License is distributed on an "AS IS" * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See * the License for the specific language governing rights and limitations * under the License. * * %CopyrightEnd% * */ /* * common interface to some simple synchronisation primitives for * internal use by ei. */ /* Note that these locks are NOT recursive on Win32 or Solaris, * i.e. self-deadlock will occur if a thread tries to obtain a lock it * is already holding. The primitives used on VxWorks are recursive however. */ #include "eidef.h" #ifdef __WIN32__ #include <winsock2.h> #include <windows.h> #include <winbase.h> #elif VXWORKS #include <vxWorks.h> #include <semLib.h> #else /* unix */ #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #endif /* platforms */ #include "ei_malloc.h" #include "ei_locking.h" #ifdef _REENTRANT /* * Create a new mutex object. * Returns a pointer to the mutex if successful, NULL otherwise. */ ei_mutex_t *ei_mutex_create(void) { ei_mutex_t *l; if ((l = ei_malloc(sizeof(*l))) == NULL) return NULL; #ifdef __WIN32__ l->lock = CreateMutex(NULL,FALSE,NULL); #elif VXWORKS if (!(l->lock = semMCreate(SEM_DELETE_SAFE))) { ei_free(l); return NULL; } #else /* unix */ l->lock = ei_m_create(); #endif return l; } /* * Free a mutex and the structure asociated with it. * * This function attempts to obtain the mutex before releasing it; * If nblock == 1 and the mutex was unavailable, the function will * return failure and the mutex will not have been removed. * * If nblock == 0 the function will block until the mutex becomes * available, at which time it will be removed and the function will * succeed. * * returns 0 if the mutex is removed, -1 on failure (busy) */ int ei_mutex_free(ei_mutex_t *l, int nblock) { /* attempt to lock it first, to make sure it's really free */ if (ei_mutex_lock(l,nblock)) return -1; /* attempt failed */ /* we are now holding the lock */ #ifdef __WIN32__ CloseHandle(l->lock); #elif VXWORKS if (semDelete(l->lock) == ERROR) return -1; #else /* unix */ ei_m_destroy(l->lock); #endif ei_free(l); return 0; } /* Grab a mutex. If the mutex is not held by any other process the * function returns so that the caller may enter a critical section. * Processes subsequently wishing to obtain the lock will block * until this process releases it. * * If the mutex is busy (held by some other process) and nblock == 0, * the function will block until the mutex is freed by the process * holding it, returning only when the mutex has been grabbed. * * If the mutex is busy and nblock != 0, the function will not block. * Instead it will return -1 immediately, indicating that the * operation failed. * Returns 0 on success, -1 on failure. */ int ei_mutex_lock(ei_mutex_t *l, int nblock) { #ifdef __WIN32__ /* check valid values for timeout: is 0 ok? */ if (WaitForSingleObject(l->lock,(nblock? 0 : INFINITE)) != WAIT_OBJECT_0) return -1; #elif VXWORKS if (semTake(l->lock,(nblock? NO_WAIT : WAIT_FOREVER)) == ERROR) return -1; #else /* unix */ if (nblock) { if (ei_m_trylock(l->lock) < 0) return -1; } else ei_m_lock(l->lock); #endif return 0; } /* Release a mutex */ int ei_mutex_unlock(ei_mutex_t *l) { #ifdef __WIN32__ ReleaseMutex(l->lock); #elif VXWORKS semGive(l->lock); #else /* unix */ ei_m_unlock(l->lock); #endif return 0; } #endif /* _REENTRANT */