git-svn-id: http://libopenmetaverse.googlecode.com/svn/trunk@18 52acb1d6-8a22-11de-b505-999d5b087335
378 lines
9.4 KiB
C++
378 lines
9.4 KiB
C++
//
|
|
// task_demuxer_service.hpp
|
|
// ~~~~~~~~~~~~~~~~~~~~~~~~
|
|
//
|
|
// Copyright (c) 2003-2005 Christopher M. Kohlhoff (chris at kohlhoff dot com)
|
|
//
|
|
// 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)
|
|
//
|
|
|
|
#ifndef BOOST_ASIO_DETAIL_TASK_DEMUXER_SERVICE_HPP
|
|
#define BOOST_ASIO_DETAIL_TASK_DEMUXER_SERVICE_HPP
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER >= 1200)
|
|
# pragma once
|
|
#endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
|
|
|
|
#include <boost/asio/detail/push_options.hpp>
|
|
|
|
#include <boost/asio/detail/push_options.hpp>
|
|
#include <memory>
|
|
#include <boost/asio/detail/pop_options.hpp>
|
|
|
|
#include <boost/asio/basic_demuxer.hpp>
|
|
#include <boost/asio/demuxer_service.hpp>
|
|
#include <boost/asio/service_factory.hpp>
|
|
#include <boost/asio/detail/bind_handler.hpp>
|
|
#include <boost/asio/detail/demuxer_run_call_stack.hpp>
|
|
#include <boost/asio/detail/event.hpp>
|
|
#include <boost/asio/detail/mutex.hpp>
|
|
|
|
namespace boost {
|
|
namespace asio {
|
|
namespace detail {
|
|
|
|
template <typename Task>
|
|
class task_demuxer_service
|
|
{
|
|
public:
|
|
// Constructor.
|
|
template <typename Demuxer>
|
|
task_demuxer_service(Demuxer& demuxer)
|
|
: mutex_(),
|
|
task_(demuxer.get_service(service_factory<Task>())),
|
|
task_is_running_(false),
|
|
outstanding_work_(0),
|
|
handler_queue_(0),
|
|
handler_queue_end_(0),
|
|
interrupted_(false),
|
|
first_idle_thread_(0)
|
|
{
|
|
}
|
|
|
|
// Run the demuxer's event processing loop.
|
|
void run()
|
|
{
|
|
typename demuxer_run_call_stack<task_demuxer_service>::context ctx(this);
|
|
|
|
idle_thread_info this_idle_thread;
|
|
this_idle_thread.prev = &this_idle_thread;
|
|
this_idle_thread.next = &this_idle_thread;
|
|
|
|
boost::asio::detail::mutex::scoped_lock lock(mutex_);
|
|
|
|
while (!interrupted_ && outstanding_work_ > 0)
|
|
{
|
|
if (handler_queue_)
|
|
{
|
|
// Prepare to execute first handler from queue.
|
|
handler_base* h = handler_queue_;
|
|
handler_queue_ = h->next_;
|
|
if (handler_queue_ == 0)
|
|
handler_queue_end_ = 0;
|
|
lock.unlock();
|
|
|
|
// Helper class to perform operations on block exit.
|
|
class cleanup
|
|
{
|
|
public:
|
|
cleanup(boost::asio::detail::mutex::scoped_lock& lock,
|
|
int& outstanding_work)
|
|
: lock_(lock),
|
|
outstanding_work_(outstanding_work)
|
|
{
|
|
}
|
|
|
|
~cleanup()
|
|
{
|
|
lock_.lock();
|
|
--outstanding_work_;
|
|
}
|
|
|
|
private:
|
|
boost::asio::detail::mutex::scoped_lock& lock_;
|
|
int& outstanding_work_;
|
|
} c(lock, outstanding_work_);
|
|
|
|
// Invoke the handler. May throw an exception.
|
|
h->call(); // call() deletes the handler object
|
|
}
|
|
else if (!task_is_running_)
|
|
{
|
|
// Prepare to execute the task.
|
|
task_is_running_ = true;
|
|
task_.reset();
|
|
lock.unlock();
|
|
|
|
// Helper class to perform operations on block exit.
|
|
class cleanup
|
|
{
|
|
public:
|
|
cleanup(boost::asio::detail::mutex::scoped_lock& lock,
|
|
bool& task_is_running)
|
|
: lock_(lock),
|
|
task_is_running_(task_is_running)
|
|
{
|
|
}
|
|
|
|
~cleanup()
|
|
{
|
|
lock_.lock();
|
|
task_is_running_ = false;
|
|
}
|
|
|
|
private:
|
|
boost::asio::detail::mutex::scoped_lock& lock_;
|
|
bool& task_is_running_;
|
|
} c(lock, task_is_running_);
|
|
|
|
// Run the task. May throw an exception.
|
|
task_.run();
|
|
}
|
|
else
|
|
{
|
|
// Nothing to run right now, so just wait for work to do.
|
|
if (first_idle_thread_)
|
|
{
|
|
this_idle_thread.next = first_idle_thread_;
|
|
this_idle_thread.prev = first_idle_thread_->prev;
|
|
first_idle_thread_->prev->next = &this_idle_thread;
|
|
first_idle_thread_->prev = &this_idle_thread;
|
|
}
|
|
first_idle_thread_ = &this_idle_thread;
|
|
this_idle_thread.wakeup_event.clear();
|
|
lock.unlock();
|
|
this_idle_thread.wakeup_event.wait();
|
|
lock.lock();
|
|
if (this_idle_thread.next == &this_idle_thread)
|
|
{
|
|
first_idle_thread_ = 0;
|
|
}
|
|
else
|
|
{
|
|
if (first_idle_thread_ == &this_idle_thread)
|
|
first_idle_thread_ = this_idle_thread.next;
|
|
this_idle_thread.next->prev = this_idle_thread.prev;
|
|
this_idle_thread.prev->next = this_idle_thread.next;
|
|
this_idle_thread.next = &this_idle_thread;
|
|
this_idle_thread.prev = &this_idle_thread;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!interrupted_)
|
|
{
|
|
// No more work to do!
|
|
interrupt_all_threads();
|
|
}
|
|
}
|
|
|
|
// Interrupt the demuxer's event processing loop.
|
|
void interrupt()
|
|
{
|
|
boost::asio::detail::mutex::scoped_lock lock(mutex_);
|
|
interrupt_all_threads();
|
|
}
|
|
|
|
// Reset the demuxer in preparation for a subsequent run invocation.
|
|
void reset()
|
|
{
|
|
boost::asio::detail::mutex::scoped_lock lock(mutex_);
|
|
interrupted_ = false;
|
|
}
|
|
|
|
// Notify the demuxer that some work has started.
|
|
void work_started()
|
|
{
|
|
boost::asio::detail::mutex::scoped_lock lock(mutex_);
|
|
++outstanding_work_;
|
|
}
|
|
|
|
// Notify the demuxer that some work has finished.
|
|
void work_finished()
|
|
{
|
|
boost::asio::detail::mutex::scoped_lock lock(mutex_);
|
|
if (--outstanding_work_ == 0)
|
|
interrupt_all_threads();
|
|
}
|
|
|
|
// Request the demuxer to invoke the given handler.
|
|
template <typename Handler>
|
|
void dispatch(Handler handler)
|
|
{
|
|
if (demuxer_run_call_stack<task_demuxer_service>::contains(this))
|
|
handler();
|
|
else
|
|
post(handler);
|
|
}
|
|
|
|
// Request the demuxer to invoke the given handler and return immediately.
|
|
template <typename Handler>
|
|
void post(Handler handler)
|
|
{
|
|
boost::asio::detail::mutex::scoped_lock lock(mutex_);
|
|
|
|
// Add the handler to the end of the queue.
|
|
handler_base* h = new handler_wrapper<Handler>(handler);
|
|
if (handler_queue_end_)
|
|
{
|
|
handler_queue_end_->next_ = h;
|
|
handler_queue_end_ = h;
|
|
}
|
|
else
|
|
{
|
|
handler_queue_ = handler_queue_end_ = h;
|
|
}
|
|
|
|
// An undelivered handler is treated as unfinished work.
|
|
++outstanding_work_;
|
|
|
|
// Wake up a thread to execute the handler.
|
|
if (!interrupt_one_idle_thread())
|
|
interrupt_task();
|
|
}
|
|
|
|
private:
|
|
// Interrupt the task and all idle threads.
|
|
void interrupt_all_threads()
|
|
{
|
|
interrupted_ = true;
|
|
interrupt_all_idle_threads();
|
|
interrupt_task();
|
|
}
|
|
|
|
// Interrupt a single idle thread. Returns true if a thread was interrupted,
|
|
// false if no running thread could be found to interrupt.
|
|
bool interrupt_one_idle_thread()
|
|
{
|
|
if (first_idle_thread_)
|
|
{
|
|
first_idle_thread_->wakeup_event.signal();
|
|
first_idle_thread_ = first_idle_thread_->next;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Interrupt all idle threads.
|
|
void interrupt_all_idle_threads()
|
|
{
|
|
if (first_idle_thread_)
|
|
{
|
|
first_idle_thread_->wakeup_event.signal();
|
|
idle_thread_info* current_idle_thread = first_idle_thread_->next;
|
|
while (current_idle_thread != first_idle_thread_)
|
|
{
|
|
current_idle_thread->wakeup_event.signal();
|
|
current_idle_thread = current_idle_thread->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Interrupt the task. Returns true if the task was interrupted, false if
|
|
// the task was not running and so could not be interrupted.
|
|
bool interrupt_task()
|
|
{
|
|
if (task_is_running_)
|
|
{
|
|
task_.interrupt();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// The base class for all handler wrappers. A function pointer is used
|
|
// instead of virtual functions to avoid the associated overhead.
|
|
class handler_base
|
|
{
|
|
public:
|
|
typedef void (*func_type)(handler_base*);
|
|
|
|
handler_base(func_type func)
|
|
: next_(0),
|
|
func_(func)
|
|
{
|
|
}
|
|
|
|
void call()
|
|
{
|
|
func_(this);
|
|
}
|
|
|
|
protected:
|
|
// Prevent deletion through this type.
|
|
~handler_base()
|
|
{
|
|
}
|
|
|
|
private:
|
|
friend class task_demuxer_service<Task>;
|
|
handler_base* next_;
|
|
func_type func_;
|
|
};
|
|
|
|
// Template wrapper for handlers.
|
|
template <typename Handler>
|
|
class handler_wrapper
|
|
: public handler_base
|
|
{
|
|
public:
|
|
handler_wrapper(Handler handler)
|
|
: handler_base(&handler_wrapper<Handler>::do_call),
|
|
handler_(handler)
|
|
{
|
|
}
|
|
|
|
static void do_call(handler_base* base)
|
|
{
|
|
std::auto_ptr<handler_wrapper<Handler> > h(
|
|
static_cast<handler_wrapper<Handler>*>(base));
|
|
h->handler_();
|
|
}
|
|
|
|
private:
|
|
Handler handler_;
|
|
};
|
|
|
|
// Mutex to protect access to internal data.
|
|
boost::asio::detail::mutex mutex_;
|
|
|
|
// The task to be run by this demuxer service.
|
|
Task& task_;
|
|
|
|
// Whether the task is currently running.
|
|
bool task_is_running_;
|
|
|
|
// The count of unfinished work.
|
|
int outstanding_work_;
|
|
|
|
// The start of a linked list of handlers that are ready to be delivered.
|
|
handler_base* handler_queue_;
|
|
|
|
// The end of a linked list of handlers that are ready to be delivered.
|
|
handler_base* handler_queue_end_;
|
|
|
|
// Flag to indicate that the dispatcher has been interrupted.
|
|
bool interrupted_;
|
|
|
|
// Structure containing information about an idle thread.
|
|
struct idle_thread_info
|
|
{
|
|
event wakeup_event;
|
|
idle_thread_info* prev;
|
|
idle_thread_info* next;
|
|
};
|
|
|
|
// The number of threads that are currently idle.
|
|
idle_thread_info* first_idle_thread_;
|
|
};
|
|
|
|
} // namespace detail
|
|
} // namespace asio
|
|
} // namespace boost
|
|
|
|
#include <boost/asio/detail/pop_options.hpp>
|
|
|
|
#endif // BOOST_ASIO_DETAIL_TASK_DEMUXER_SERVICE_HPP
|