X-Git-Url: http://git.indexdata.com/?a=blobdiff_plain;f=src%2Fthread_pool_observer.cpp;h=c2a854f18c833e40b7004ca4392f0bd52e7de96d;hb=HEAD;hp=d630fd953a5b4cbe2c5661b2861cf04c2a36c1f8;hpb=d3786d00ef2609f57271545410349e3c3c9d10fd;p=metaproxy-moved-to-github.git diff --git a/src/thread_pool_observer.cpp b/src/thread_pool_observer.cpp index d630fd9..c2a854f 100644 --- a/src/thread_pool_observer.cpp +++ b/src/thread_pool_observer.cpp @@ -1,5 +1,5 @@ /* This file is part of Metaproxy. - Copyright (C) 2005-2012 Index Data + Copyright (C) Index Data Metaproxy is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free @@ -68,8 +68,11 @@ namespace metaproxy_1 { std::deque m_input; std::deque m_output; bool m_stop_flag; + unsigned m_stack_size; unsigned m_no_threads; - unsigned m_no_threads_waiting; + unsigned m_min_threads; + unsigned m_max_threads; + unsigned m_waiting_threads; }; const unsigned int queue_size_per_thread = 64; } @@ -94,21 +97,23 @@ IThreadPoolMsg::~IThreadPoolMsg() } ThreadPoolSocketObserver::ThreadPoolSocketObserver( - yazpp_1::ISocketObservable *obs, int no_threads) + yazpp_1::ISocketObservable *obs, + unsigned min_threads, unsigned max_threads, + unsigned stack_size) : m_p(new Rep(obs)) { obs->addObserver(m_p->m_pipe.read_fd(), this); obs->maskObserver(this, SOCKET_OBSERVE_READ); m_p->m_stop_flag = false; - m_p->m_no_threads = no_threads; - m_p->m_no_threads_waiting = 0; - int i; - for (i = 0; im_thrds.add_thread(new boost::thread(w)); - } + m_p->m_no_threads = 0; + m_p->m_min_threads = min_threads; + m_p->m_max_threads = max_threads; + m_p->m_waiting_threads = 0; + m_p->m_stack_size = stack_size; + unsigned i; + for (i = 0; i < min_threads; i++) + add_worker(); } ThreadPoolSocketObserver::~ThreadPoolSocketObserver() @@ -119,10 +124,24 @@ ThreadPoolSocketObserver::~ThreadPoolSocketObserver() m_p->m_cond_input_data.notify_all(); } m_p->m_thrds.join_all(); - m_p->m_socketObservable->deleteObserver(this); } +void ThreadPoolSocketObserver::add_worker(void) +{ + Worker w(this); +#if BOOST_VERSION >= 1050000 + boost::thread::attributes attrs; + if (m_p->m_stack_size) + attrs.set_stack_size(m_stack_size); + boost::thread *x = new boost::thread(attrs, w); +#else + boost::thread *x = new boost::thread(w); +#endif + m_p->m_no_threads++; + m_p->m_thrds.add_thread(x); +} + void ThreadPoolSocketObserver::socketNotify(int event) { if (event & SOCKET_OBSERVE_READ) @@ -148,15 +167,13 @@ void ThreadPoolSocketObserver::socketNotify(int event) out = m_p->m_output.front(); m_p->m_output.pop_front(); } - - if (out) { std::ostringstream os; { boost::mutex::scoped_lock input_lock(m_p->m_mutex_input_data); os << "tbusy/total " << - m_p->m_no_threads - m_p->m_no_threads_waiting << + m_p->m_no_threads - m_p->m_waiting_threads << "/" << m_p->m_no_threads << " queue in/out " << m_p->m_input.size() << "/" << m_p->m_output.size(); @@ -166,6 +183,12 @@ void ThreadPoolSocketObserver::socketNotify(int event) } } +void ThreadPoolSocketObserver::get_thread_info(int &tbusy, int &total) +{ + tbusy = m_p->m_no_threads - m_p->m_waiting_threads; + total = m_p->m_no_threads; +} + void ThreadPoolSocketObserver::run(void *p) { while(1) @@ -173,15 +196,15 @@ void ThreadPoolSocketObserver::run(void *p) IThreadPoolMsg *in = 0; { boost::mutex::scoped_lock input_lock(m_p->m_mutex_input_data); - m_p->m_no_threads_waiting++; + m_p->m_waiting_threads++; while (!m_p->m_stop_flag && m_p->m_input.size() == 0) m_p->m_cond_input_data.wait(input_lock); - m_p->m_no_threads_waiting--; + m_p->m_waiting_threads--; if (m_p->m_stop_flag) break; - + in = m_p->m_input.front(); - m_p->m_input.pop_front(); + m_p->m_input.pop_front(); m_p->m_cond_input_full.notify_all(); } IThreadPoolMsg *out = in->handle(); @@ -206,14 +229,37 @@ void ThreadPoolSocketObserver::run(void *p) } } +void ThreadPoolSocketObserver::cleanup(IThreadPoolMsg *m, void *info) +{ + boost::mutex::scoped_lock input_lock(m_p->m_mutex_input_data); + + std::deque::iterator it = m_p->m_input.begin(); + while (it != m_p->m_input.end()) + { + if ((*it)->cleanup(info)) + { + delete *it; + it = m_p->m_input.erase(it); + } + else + it++; + } +} + void ThreadPoolSocketObserver::put(IThreadPoolMsg *m) { boost::mutex::scoped_lock input_lock(m_p->m_mutex_input_data); + if (m_p->m_waiting_threads == 0 && + m_p->m_no_threads < m_p->m_max_threads) + { + add_worker(); + } while (m_p->m_input.size() >= m_p->m_no_threads * queue_size_per_thread) m_p->m_cond_input_full.wait(input_lock); m_p->m_input.push_back(m); m_p->m_cond_input_data.notify_one(); } + /* * Local variables: * c-basic-offset: 4