////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Ion Gaztanaga 2005-2012. Distributed under the Boost // Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // // See http://www.boost.org/libs/interprocess for documentation. // ////////////////////////////////////////////////////////////////////////////// // // Parts of the pthread code come from Boost Threads code: // ////////////////////////////////////////////////////////////////////////////// // // Copyright (C) 2001-2003 // William E. Kempf // // Permission to use, copy, modify, distribute and sell this software // and its documentation for any purpose is hereby granted without fee, // provided that the above copyright notice appear in all copies and // that both that copyright notice and this permission notice appear // in supporting documentation. William E. Kempf makes no representations // about the suitability of this software for any purpose. // It is provided "as is" without express or implied warranty. ////////////////////////////////////////////////////////////////////////////// #ifndef BOOST_INTERPROCESS_RECURSIVE_MUTEX_HPP #define BOOST_INTERPROCESS_RECURSIVE_MUTEX_HPP /// @cond #if (defined _MSC_VER) && (_MSC_VER >= 1200) # pragma once #endif #include #include #include #include #if !defined(BOOST_INTERPROCESS_FORCE_GENERIC_EMULATION) && \ (defined(BOOST_INTERPROCESS_POSIX_PROCESS_SHARED) && defined (BOOST_INTERPROCESS_POSIX_RECURSIVE_MUTEXES)) #include #define BOOST_INTERPROCESS_USE_POSIX //Experimental... #elif !defined(BOOST_INTERPROCESS_FORCE_GENERIC_EMULATION) && defined (BOOST_INTERPROCESS_WINDOWS) #include #define BOOST_INTERPROCESS_USE_WINDOWS #elif !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED) #include #define BOOST_INTERPROCESS_USE_GENERIC_EMULATION #endif #if defined (BOOST_INTERPROCESS_USE_GENERIC_EMULATION) namespace boost { namespace interprocess { namespace ipcdetail{ namespace robust_emulation_helpers { template class mutex_traits; }}}} #endif /// @endcond //!\file //!Describes interprocess_recursive_mutex and shared_recursive_try_mutex classes namespace boost { namespace interprocess { //!Wraps a interprocess_mutex that can be placed in shared memory and can be //!shared between processes. Allows several locking calls by the same //!process. Allows timed lock tries class interprocess_recursive_mutex { /// @cond //Non-copyable interprocess_recursive_mutex(const interprocess_recursive_mutex &); interprocess_recursive_mutex &operator=(const interprocess_recursive_mutex &); /// @endcond public: //!Constructor. //!Throws interprocess_exception on error. interprocess_recursive_mutex(); //!Destructor. If any process uses the mutex after the destructor is called //!the result is undefined. Does not throw. ~interprocess_recursive_mutex(); //!Effects: The calling thread tries to obtain ownership of the mutex, and //! if another thread has ownership of the mutex, it waits until it can //! obtain the ownership. If a thread takes ownership of the mutex the //! mutex must be unlocked by the same mutex. The mutex must be unlocked //! the same number of times it is locked. //!Throws: interprocess_exception on error. void lock(); //!Tries to lock the interprocess_mutex, returns false when interprocess_mutex //!is already locked, returns true when success. The mutex must be unlocked //!the same number of times it is locked. //!Throws: interprocess_exception if a severe error is found bool try_lock(); //!Tries to lock the interprocess_mutex, if interprocess_mutex can't be locked before //!abs_time time, returns false. The mutex must be unlocked //! the same number of times it is locked. //!Throws: interprocess_exception if a severe error is found bool timed_lock(const boost::posix_time::ptime &abs_time); //!Effects: The calling thread releases the exclusive ownership of the mutex. //! If the mutex supports recursive locking, the mutex must be unlocked the //! same number of times it is locked. //!Throws: interprocess_exception on error. void unlock(); /// @cond private: #if defined (BOOST_INTERPROCESS_USE_GENERIC_EMULATION) #undef BOOST_INTERPROCESS_USE_GENERIC_EMULATION void take_ownership(){ mutex.take_ownership(); } friend class ipcdetail::robust_emulation_helpers::mutex_traits; ipcdetail::spin_recursive_mutex mutex; #elif defined(BOOST_INTERPROCESS_USE_POSIX) #undef BOOST_INTERPROCESS_USE_POSIX ipcdetail::posix_recursive_mutex mutex; #elif defined(BOOST_INTERPROCESS_USE_WINDOWS) #undef BOOST_INTERPROCESS_USE_WINDOWS ipcdetail::windows_recursive_mutex mutex; #else #error "Unknown platform for interprocess_mutex" #endif /// @endcond }; } //namespace interprocess { } //namespace boost { namespace boost { namespace interprocess { inline interprocess_recursive_mutex::interprocess_recursive_mutex(){} inline interprocess_recursive_mutex::~interprocess_recursive_mutex(){} inline void interprocess_recursive_mutex::lock() { #ifdef BOOST_INTERPROCESS_ENABLE_TIMEOUT_WHEN_LOCKING boost::posix_time::ptime wait_time = boost::posix_time::microsec_clock::universal_time() + boost::posix_time::milliseconds(BOOST_INTERPROCESS_TIMEOUT_WHEN_LOCKING_DURATION_MS); if (!mutex.timed_lock(wait_time)){ throw interprocess_exception(timeout_when_locking_error, "Interprocess mutex timeout when locking. Possible deadlock: owner died without unlocking?"); } #else mutex.lock(); #endif } inline bool interprocess_recursive_mutex::try_lock() { return mutex.try_lock(); } inline bool interprocess_recursive_mutex::timed_lock(const boost::posix_time::ptime &abs_time) { return mutex.timed_lock(abs_time); } inline void interprocess_recursive_mutex::unlock() { mutex.unlock(); } } //namespace interprocess { } //namespace boost { #include #endif //BOOST_INTERPROCESS_RECURSIVE_MUTEX_HPP