| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508 |
- // -*- C++ -*-
- // Copyright (C) 2009-2015, Vaclav Haisman. All rights reserved.
- //
- // Redistribution and use in source and binary forms, with or without modifica-
- // tion, are permitted provided that the following conditions are met:
- //
- // 1. Redistributions of source code must retain the above copyright notice,
- // this list of conditions and the following disclaimer.
- //
- // 2. Redistributions in binary form must reproduce the above copyright notice,
- // this list of conditions and the following disclaimer in the documentation
- // and/or other materials provided with the distribution.
- //
- // THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
- // INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
- // FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
- // APACHE SOFTWARE FOUNDATION OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
- // INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLU-
- // DING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
- // OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
- // ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
- // THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- //! @file
- //! This file contains implementations of synchronization
- //! primitives using the POSIX threads. It does not contain any
- //! include guards because it is only a fragment to be included by
- //! syncprims.h.
- #include "log4cplus/thread/threads.h"
- #include <limits>
- #include <algorithm>
- namespace log4cplus { namespace thread { namespace impl {
- struct PthreadMutexAttr
- {
- PthreadMutexAttr ()
- {
- int ret = pthread_mutexattr_init (&attr);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("PthreadMutexAttr::PthreadMutexAttr");
- }
- ~PthreadMutexAttr ()
- {
- try
- {
- int ret = pthread_mutexattr_destroy (&attr);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("PthreadMutexAttr::~PthreadMutexAttr");
- }
- catch (...)
- { }
- }
- void
- set_type (log4cplus::thread::Mutex::Type t)
- {
- int mutex_type;
- switch (t)
- {
- case log4cplus::thread::Mutex::RECURSIVE:
- mutex_type = PTHREAD_MUTEX_RECURSIVE;
- break;
- default:
- mutex_type = PTHREAD_MUTEX_DEFAULT;
- }
- int ret = pthread_mutexattr_settype (&attr, mutex_type);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("PthreadMutexAttr::set_type");
- }
- pthread_mutexattr_t attr;
- };
- //
- //
- //
- inline
- Mutex::Mutex (log4cplus::thread::Mutex::Type t)
- {
- PthreadMutexAttr attr;
- attr.set_type (t);
- int ret = pthread_mutex_init (&mtx, &attr.attr);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Mutex::Mutex");
- }
- inline
- Mutex::~Mutex ()
- {
- try
- {
- int ret = pthread_mutex_destroy (&mtx);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Mutex::~Mutex");
- }
- catch (...)
- { }
- }
- inline
- void
- Mutex::lock () const
- {
- int ret = pthread_mutex_lock (&mtx);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Mutex::lock");
- }
- inline
- void
- Mutex::unlock () const
- {
- int ret = pthread_mutex_unlock (&mtx);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Mutex::unlock");
- }
- //
- //
- //
- inline
- Semaphore::Semaphore (unsigned max, unsigned initial)
- {
- unsigned const sem_value_max =
- #if defined (SEM_VALUE_MAX)
- SEM_VALUE_MAX
- #else
- (std::numeric_limits<int>::max) ()
- #endif
- ;
- unsigned const limited_max = (std::min) (max, sem_value_max);
- unsigned const limited_initial = (std::min) (initial, limited_max);
- int ret = 0;
- #if defined (LOG4CPLUS_USE_NAMED_POSIX_SEMAPHORE)
- std::ostringstream oss;
- char size_check[2 * (static_cast<int>(sizeof (std::ptrdiff_t))
- - static_cast<int>(sizeof (this))) + 1];
- (void)size_check;
- oss << getpid () << "-" << reinterpret_cast<std::ptrdiff_t>(this);
- std::string name (oss.str ());
- sem = sem_open (name.c_str (), O_CREAT, S_IRWXU | S_IRWXG, limited_max);
- ret = sem == SEM_FAILED;
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::Semaphore");
- try
- {
- // Unlink the semaphore early to simulate anonymous semaphore.
- ret = sem_unlink (name.c_str ());
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::Semaphore");
- }
- catch (std::runtime_error const &)
- {
- ret = sem_close (sem);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::~Semaphore");
- throw;
- }
- #else
- ret = sem_init (&sem, 0, limited_max);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::Semaphore");
- #endif
- try
- {
- for (unsigned i = limited_initial; i < limited_max; ++i)
- lock ();
- }
- catch (std::runtime_error const &)
- {
- #if defined (LOG4CPLUS_USE_NAMED_POSIX_SEMAPHORE)
- ret = sem_close (sem);
- #else
- ret = sem_destroy (&sem);
- #endif
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::~Semaphore");
- throw;
- }
- }
- inline
- Semaphore::~Semaphore ()
- {
- try
- {
- int ret = 0;
- #if defined (LOG4CPLUS_USE_NAMED_POSIX_SEMAPHORE)
- ret = sem_close (sem);
- #else
- ret = sem_destroy (&sem);
- #endif
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::~Semaphore");
- }
- catch (...)
- { }
- }
- inline
- void
- Semaphore::unlock () const
- {
- #if defined (LOG4CPLUS_USE_NAMED_POSIX_SEMAPHORE)
- int ret = sem_post (sem);
- #else
- int ret = sem_post (&sem);
- #endif
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::unlock");
- }
- inline
- void
- Semaphore::lock () const
- {
- #if defined (LOG4CPLUS_USE_NAMED_POSIX_SEMAPHORE)
- int ret = sem_wait (sem);
- #else
- int ret = sem_wait (&sem);
- #endif
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("Semaphore::lock");
- }
- //
- //
- //
- inline
- FairMutex::FairMutex ()
- : sem (1, 1)
- { }
- inline
- FairMutex::~FairMutex ()
- { }
- inline
- void
- FairMutex::lock () const
- {
- sem.lock ();
- }
- inline
- void
- FairMutex::unlock () const
- {
- sem.unlock ();
- }
- //
- //
- //
- inline
- ManualResetEvent::ManualResetEvent (bool sig)
- : mtx (log4cplus::thread::Mutex::DEFAULT)
- , sigcount (0)
- , signaled (sig)
- {
- int ret = pthread_cond_init (&cv, 0);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("ManualResetEvent::ManualResetEvent");
- }
- inline
- ManualResetEvent::~ManualResetEvent ()
- {
- try
- {
- int ret = pthread_cond_destroy (&cv);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("ManualResetEvent::~ManualResetEvent");
- }
- catch (...)
- { }
- }
- inline
- void
- ManualResetEvent::signal () const
- {
- MutexGuard mguard (mtx);
- signaled = true;
- sigcount += 1;
- int ret = pthread_cond_broadcast (&cv);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("ManualResetEvent::signal");
- }
- inline
- void
- ManualResetEvent::wait () const
- {
- MutexGuard mguard (mtx);
- if (! signaled)
- {
- unsigned prev_count = sigcount;
- do
- {
- int ret = pthread_cond_wait (&cv, &mtx.mtx);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- {
- mguard.unlock ();
- mguard.detach ();
- LOG4CPLUS_THROW_RTE ("ManualResetEvent::wait");
- }
- }
- while (prev_count == sigcount);
- }
- }
- inline
- bool
- ManualResetEvent::timed_wait (unsigned long msec) const
- {
- MutexGuard mguard (mtx);
- if (! signaled)
- {
- helpers::Time const wakeup_time (helpers::Time::gettimeofday ()
- + helpers::Time (msec / 1000, (msec % 1000) * 1000));
- struct timespec const ts = {wakeup_time.sec (),
- wakeup_time.usec () * 1000};
- unsigned prev_count = sigcount;
- do
- {
- int ret = pthread_cond_timedwait (&cv, &mtx.mtx, &ts);
- switch (ret)
- {
- case 0:
- break;
- case ETIMEDOUT:
- return false;
- default:
- mguard.unlock ();
- mguard.detach ();
- LOG4CPLUS_THROW_RTE ("ManualResetEvent::timed_wait");
- }
- }
- while (prev_count == sigcount);
- }
- return true;
- }
- inline
- void
- ManualResetEvent::reset () const
- {
- MutexGuard mguard (mtx);
- signaled = false;
- }
- //
- //
- //
- #if defined (LOG4CPLUS_POOR_MANS_SHAREDMUTEX)
- #include "log4cplus/thread/impl/syncprims-pmsm.h"
- #else
- inline
- SharedMutex::SharedMutex ()
- {
- int ret = pthread_rwlock_init (&rwl, 0);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("SharedMutex::SharedMutex");
- }
- inline
- SharedMutex::~SharedMutex ()
- {
- try
- {
- int ret = pthread_rwlock_destroy (&rwl);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("SharedMutex::~SharedMutex");
- }
- catch (...)
- { }
- }
- inline
- void
- SharedMutex::rdlock () const
- {
- int ret;
- do
- {
- ret = pthread_rwlock_rdlock (&rwl);
- switch (ret)
- {
- case EAGAIN:
- // The read lock could not be acquired because the maximum
- // number of read locks for rwlock has been exceeded.
- log4cplus::thread::yield ();
- // Fall through.
- case 0:
- break;
- default:
- LOG4CPLUS_THROW_RTE ("SharedMutex::rdlock");
- }
- }
- while (LOG4CPLUS_UNLIKELY (ret != 0));
- }
- inline
- void
- SharedMutex::rdunlock () const
- {
- unlock ();
- }
- inline
- void
- SharedMutex::wrlock () const
- {
- int ret = pthread_rwlock_wrlock (&rwl);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("SharedMutex::wrlock");
- }
- inline
- void
- SharedMutex::wrunlock () const
- {
- unlock ();
- }
- inline
- void
- SharedMutex::unlock () const
- {
- int ret = pthread_rwlock_unlock (&rwl);
- if (LOG4CPLUS_UNLIKELY (ret != 0))
- LOG4CPLUS_THROW_RTE ("SharedMutex::unlock");
- }
- #endif
- } } } // namespace log4cplus { namespace thread { namespace impl {
|