diff options
Diffstat (limited to 'ACE/ace/SV_Semaphore_Complex.h')
-rw-r--r-- | ACE/ace/SV_Semaphore_Complex.h | 159 |
1 files changed, 159 insertions, 0 deletions
diff --git a/ACE/ace/SV_Semaphore_Complex.h b/ACE/ace/SV_Semaphore_Complex.h new file mode 100644 index 00000000000..de48fc09359 --- /dev/null +++ b/ACE/ace/SV_Semaphore_Complex.h @@ -0,0 +1,159 @@ +// -*- C++ -*- + +//============================================================================= +/** + * @file SV_Semaphore_Complex.h + * + * $Id$ + * + * @author Douglas C. Schmidt <schmidt@cs.wustl.edu> + */ +//============================================================================= + +#ifndef ACE_SV_SEMAPHORE_COMPLEX_H +#define ACE_SV_SEMAPHORE_COMPLEX_H +#include /**/ "ace/pre.h" + +#include "ace/SV_Semaphore_Simple.h" + +#if !defined (ACE_LACKS_PRAGMA_ONCE) +# pragma once +#endif /* ACE_LACKS_PRAGMA_ONCE */ + +ACE_BEGIN_VERSIONED_NAMESPACE_DECL + +/** + * @class ACE_SV_Semaphore_Complex + * + * @brief This is a more complex semaphore wrapper that handles race + * conditions for initialization correctly... + * + * This code is a port to C++, inspired by: W. Richard Stevens + * from his book: UNIX Network Programming (Prentice Hall, ISBN + * 0-13-949876-1 - 1990). We provide a simpler and easier to + * understand interface to the System V Semaphore system calls. + * We create and use a 2 + n-member set for the requested + * <ACE_SV_Semaphore_Complex>. The first member, [0], is a + * counter used to know when all processes have finished with + * the <ACE_SV_Semaphore_Complex>. The counter is initialized + * to a large number, decremented on every create or open and + * incremented on every close. This way we can use the "adjust" + * feature provided by System V so that any process that exit's + * without calling <close> is accounted for. It doesn't help us + * if the last process does this (as we have no way of getting + * control to remove the <ACE_SV_Semaphore_Complex>) but it + * will work if any process other than the last does an exit + * (intentional or unintentional). + * The second member, [1], of the semaphore is used as a lock + * variable to avoid any race conditions in the <create> and + * <close> functions. + * The members beyond [1] are actual semaphore values in the + * array of semaphores, which may be sized by the user in the + * constructor. + */ +class ACE_Export ACE_SV_Semaphore_Complex : private ACE_SV_Semaphore_Simple +{ +public: + enum + { + ACE_CREATE = IPC_CREAT, + ACE_OPEN = 0 + }; + + // = Initialization and termination methods. + ACE_SV_Semaphore_Complex (void); + ACE_SV_Semaphore_Complex (key_t key, + short create = ACE_SV_Semaphore_Complex::ACE_CREATE, + int initial_value = 1, + u_short nsems = 1, + mode_t perms = ACE_DEFAULT_FILE_PERMS); + ACE_SV_Semaphore_Complex (const char *name, + short create = ACE_SV_Semaphore_Complex::ACE_CREATE, + int initial_value = 1, + u_short nsems = 1, + mode_t perms = ACE_DEFAULT_FILE_PERMS); + ~ACE_SV_Semaphore_Complex (void); + + /// Open or create an array of SV_Semaphores. We return 0 if all is + /// OK, else -1. + int open (const char *name, + short flags = ACE_SV_Semaphore_Simple::ACE_CREATE, + int initial_value = 1, + u_short nsems = 1, + mode_t perms = ACE_DEFAULT_FILE_PERMS); + + /// Open or create an array of SV_Semaphores. We return 0 if all is + /// OK, else -1. + int open (key_t key, + short flags = ACE_SV_Semaphore_Simple::ACE_CREATE, + int initial_value = 1, + u_short nsems = 1, + mode_t perms = ACE_DEFAULT_FILE_PERMS); + + /** + * Close an ACE_SV_Semaphore. Unlike the <remove> method, this + * method is for a process to call before it exits, when it is done + * with the ACE_SV_Semaphore. We "decrement" the counter of + * processes using the ACE_SV_Semaphore, and if this was the last + * one, we can remove the ACE_SV_Semaphore. + */ + int close (void); + + // = Semaphore acquire and release methods. + + /// Acquire the semaphore. + int acquire (u_short n = 0, short flags = 0) const; + + /// Acquire a semaphore for reading. + int acquire_read (u_short n = 0, short flags = 0) const; + + /// Acquire a semaphore for writing + int acquire_write (u_short n = 0, short flags = 0) const; + + /// Try to acquire the semaphore. + int tryacquire (u_short n = 0, short flags = 0) const; + + /// Try to acquire the semaphore for reading. + int tryacquire_read (u_short n = 0, short flags = 0) const; + + /// Try to acquire the semaphore for writing. + int tryacquire_write (u_short n = 0, short flags = 0) const; + + /// Release the semaphore. + int release (u_short n = 0, short flags = 0) const; + + // = Semaphore operation methods. + int op (short val, u_short n = 0, short flags = 0) const; + int op (sembuf op_vec[], u_short n) const; + + // = Semaphore control methods. + int control (int cmd, semun arg, u_short n = 0) const; + int control (int cmd, int value = 0, u_short n = 0) const; + + // = Upgrade access control... + using ACE_SV_Semaphore_Simple::get_id; + using ACE_SV_Semaphore_Simple::remove; + + /// Dump the state of an object. + void dump (void) const; + + /// Declare the dynamic allocation hooks. + ACE_ALLOC_HOOK_DECLARE; + +private: + static const int BIGCOUNT_; + static sembuf op_lock_[2]; + static sembuf op_endcreate_[2]; + static sembuf op_open_[1]; + static sembuf op_close_[3]; + static sembuf op_unlock_[1]; +}; + +ACE_END_VERSIONED_NAMESPACE_DECL + +#if defined (__ACE_INLINE__) +#include "ace/SV_Semaphore_Complex.inl" +#endif /* __ACE_INLINE__ */ + +#include /**/ "ace/post.h" +#endif /* ACE_SV_SEMAPHORE_COMPLEX_H */ |