/*
**********************************************************************
* Copyright (C) 1997-2011, International Business Machines
* Corporation and others. All Rights Reserved.
**********************************************************************
*
* File UMUTEX.H
*
* Modification History:
*
* Date Name Description
* 04/02/97 aliu Creation.
* 04/07/99 srl rewrite - C interface, multiple mutices
* 05/13/99 stephen Changed to umutex (from cmutex)
******************************************************************************
*/
#ifndef UMUTEX_H
#define UMUTEX_H
#include "unicode/utypes.h"
#include "unicode/uclean.h"
#if defined(U_WINDOWS)
# include <intrin.h>
#endif
#if defined(U_DARWIN)
#if defined(__STRICT_ANSI__)
#define UPRV_REMAP_INLINE
#define inline
#endif
#include <libkern/OSAtomic.h>
#define USE_MAC_OS_ATOMIC_INCREMENT 1
#if defined(UPRV_REMAP_INLINE)
#undef inline
#undef UPRV_REMAP_INLINE
#endif
#endif
/*
* If we do not compile with dynamic_annotations.h then define
* empty annotation macros.
* See http://code.google.com/p/data-race-test/wiki/DynamicAnnotations
*/
#ifndef ANNOTATE_HAPPENS_BEFORE
# define ANNOTATE_HAPPENS_BEFORE(obj)
# define ANNOTATE_HAPPENS_AFTER(obj)
# define ANNOTATE_UNPROTECTED_READ(x) (x)
#endif
/* APP_NO_THREADS is an old symbol. We'll honour it if present. */
#ifdef APP_NO_THREADS
# define ICU_USE_THREADS 0
#endif
/* ICU_USE_THREADS
*
* Allows thread support (use of mutexes) to be compiled out of ICU.
* Default: use threads.
* Even with thread support compiled out, applications may override the
* (empty) mutex implementation with the u_setMutexFunctions() functions.
*/
#ifndef ICU_USE_THREADS
# define ICU_USE_THREADS 1
#endif
#ifndef UMTX_FULL_BARRIER
# if !ICU_USE_THREADS
# define UMTX_FULL_BARRIER
# elif U_HAVE_GCC_ATOMICS
# define UMTX_FULL_BARRIER __sync_synchronize();
# elif defined(U_WINDOWS)
# define UMTX_FULL_BARRIER _ReadWriteBarrier();
# elif defined(U_DARWIN)
# define UMTX_FULL_BARRIER OSMemoryBarrier();
# else
# define UMTX_FULL_BARRIER \
{ \
umtx_lock(NULL); \
umtx_unlock(NULL); \
}
# endif
#endif
#ifndef UMTX_ACQUIRE_BARRIER
# define UMTX_ACQUIRE_BARRIER UMTX_FULL_BARRIER
#endif
#ifndef UMTX_RELEASE_BARRIER
# define UMTX_RELEASE_BARRIER UMTX_FULL_BARRIER
#endif
/**
* \def UMTX_CHECK
* Encapsulates a safe check of an expression
* for use with double-checked lazy inititialization.
* Either memory barriers or mutexes are required, to prevent both the hardware
* and the compiler from reordering operations across the check.
* The expression must involve only a _single_ variable, typically
* a possibly null pointer or a boolean that indicates whether some service
* is initialized or not.
* The setting of the variable involved in the test must be the last step of
* the initialization process.
*
* @internal
*/
#define UMTX_CHECK(pMutex, expression, result) \
{ \
(result)=(expression); \
UMTX_ACQUIRE_BARRIER; \
}
/*
* TODO: Replace all uses of UMTX_CHECK and surrounding code
* with SimpleSingleton or TriStateSingleton, and remove UMTX_CHECK.
*/
/*
* Code within ICU that accesses shared static or global data should
* instantiate a Mutex object while doing so. The unnamed global mutex
* is used throughout ICU, so keep locking short and sweet.
*
* For example:
*
* void Function(int arg1, int arg2)
* {
* static Object* foo; // Shared read-write object
* umtx_lock(NULL); // Lock the ICU global mutex
* foo->Method();
* umtx_unlock(NULL);
* }
*
* an alternative C++ mutex API is defined in the file common/mutex.h
*/
/* Lock a mutex.
* @param mutex The given mutex to be locked. Pass NULL to specify
* the global ICU mutex. Recursive locks are an error
* and may cause a deadlock on some platforms.
*/
U_CAPI void U_EXPORT2 umtx_lock ( UMTX* mutex );
/* Unlock a mutex. Pass in NULL if you want the single global
mutex.
* @param mutex The given mutex to be unlocked. Pass NULL to specify
* the global ICU mutex.
*/
U_CAPI void U_EXPORT2 umtx_unlock ( UMTX* mutex );
/* Initialize a mutex. Use it this way:
umtx_init( &aMutex );
* ICU Mutexes do not need explicit initialization before use. Use of this
* function is not necessary.
* Initialization of an already initialized mutex has no effect, and is safe to do.
* Initialization of mutexes is thread safe. Two threads can concurrently
* initialize the same mutex without causing problems.
* @param mutex The given mutex to be initialized
*/
U_CAPI void U_EXPORT2 umtx_init ( UMTX* mutex );
/* Destroy a mutex. This will free the resources of a mutex.
* Use it this way:
* umtx_destroy( &aMutex );
* Destroying an already destroyed mutex has no effect, and causes no problems.
* This function is not thread safe. Two threads must not attempt to concurrently
* destroy the same mutex.
* @param mutex The given mutex to be destroyed.
*/
U_CAPI void U_EXPORT2 umtx_destroy( UMTX *mutex );
/*
* Atomic Increment and Decrement of an int32_t value.
*
* Return Values:
* If the result of the operation is zero, the return zero.
* If the result of the operation is not zero, the sign of returned value
* is the same as the sign of the result, but the returned value itself may
* be different from the result of the operation.
*/
U_CAPI int32_t U_EXPORT2 umtx_atomic_inc(int32_t *);
U_CAPI int32_t U_EXPORT2 umtx_atomic_dec(int32_t *);
#endif /*_CMUTEX*/
/*eof*/