#ifndef SHARED_MUTEX_LOCKING_THREAD_HPP #define SHARED_MUTEX_LOCKING_THREAD_HPP // (C) Copyright 2008 Anthony Williams // // 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) #include #include #include #include template class locking_thread { boost::shared_mutex& rw_mutex; unsigned& unblocked_count; boost::condition_variable& unblocked_condition; unsigned& simultaneous_running_count; unsigned& max_simultaneous_running; boost::mutex& unblocked_count_mutex; boost::mutex& finish_mutex; public: locking_thread(boost::shared_mutex& rw_mutex_, unsigned& unblocked_count_, boost::mutex& unblocked_count_mutex_, boost::condition_variable& unblocked_condition_, boost::mutex& finish_mutex_, unsigned& simultaneous_running_count_, unsigned& max_simultaneous_running_): rw_mutex(rw_mutex_), unblocked_count(unblocked_count_), unblocked_condition(unblocked_condition_), simultaneous_running_count(simultaneous_running_count_), max_simultaneous_running(max_simultaneous_running_), unblocked_count_mutex(unblocked_count_mutex_), finish_mutex(finish_mutex_) {} #if !defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS) locking_thread(locking_thread const&) = default; #endif void operator()() { // acquire lock lock_type lock(rw_mutex); // increment count to show we're unblocked { boost::unique_lock ublock(unblocked_count_mutex); ++unblocked_count; unblocked_condition.notify_one(); ++simultaneous_running_count; if(simultaneous_running_count>max_simultaneous_running) { max_simultaneous_running=simultaneous_running_count; } } // wait to finish boost::unique_lock finish_lock(finish_mutex); { boost::unique_lock ublock(unblocked_count_mutex); --simultaneous_running_count; } } private: void operator=(locking_thread&); }; class simple_writing_thread { boost::shared_mutex& rwm; boost::mutex& finish_mutex; boost::mutex& unblocked_mutex; unsigned& unblocked_count; void operator=(simple_writing_thread&); public: simple_writing_thread(boost::shared_mutex& rwm_, boost::mutex& finish_mutex_, boost::mutex& unblocked_mutex_, unsigned& unblocked_count_): rwm(rwm_),finish_mutex(finish_mutex_), unblocked_mutex(unblocked_mutex_),unblocked_count(unblocked_count_) {} #if !defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS) simple_writing_thread(simple_writing_thread const&) = default; #endif void operator()() { boost::unique_lock lk(rwm); { boost::unique_lock ulk(unblocked_mutex); ++unblocked_count; } boost::unique_lock flk(finish_mutex); } }; class simple_reading_thread { boost::shared_mutex& rwm; boost::mutex& finish_mutex; boost::mutex& unblocked_mutex; unsigned& unblocked_count; void operator=(simple_reading_thread&); public: simple_reading_thread(boost::shared_mutex& rwm_, boost::mutex& finish_mutex_, boost::mutex& unblocked_mutex_, unsigned& unblocked_count_): rwm(rwm_),finish_mutex(finish_mutex_), unblocked_mutex(unblocked_mutex_),unblocked_count(unblocked_count_) {} #if !defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS) simple_reading_thread(simple_reading_thread const&) = default; #endif void operator()() { boost::shared_lock lk(rwm); { boost::unique_lock ulk(unblocked_mutex); ++unblocked_count; } boost::unique_lock flk(finish_mutex); } }; #endif