1 // templated callback system
2 // Copyright (C) 2008, 2009 Tim Blechmann
4 // This program is free software; you can redistribute it and/or modify
5 // it under the terms of the GNU General Public License as published by
6 // the Free Software Foundation; either version 2 of the License, or
7 // (at your option) any later version.
9 // This program is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
14 // You should have received a copy of the GNU General Public License
15 // along with this program; see the file COPYING. If not, write to
16 // the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 // Boston, MA 02111-1307, USA.
19 #ifndef UTILITIES_CALLBACK_INTERPRETER_HPP
20 #define UTILITIES_CALLBACK_INTERPRETER_HPP
22 #include "branch_hints.hpp"
23 #include "callback_system.hpp"
25 #include "nova-tt/semaphore.hpp"
26 #include "nova-tt/thread_priority.hpp"
28 #include <boost/atomic.hpp>
29 #include <boost/bind.hpp>
30 #include <boost/checked_delete.hpp>
31 #include <boost/ref.hpp>
32 #include <boost/thread.hpp>
42 void operator()() const {}
44 } /* namespace detail */
48 template <class callback_type
,
50 class callback_deleter
= boost::checked_deleter
<callback_type
> >
51 class callback_interpreter
:
52 callback_system
<callback_type
, mpmc
, callback_deleter
>
54 typedef callback_system
<callback_type
, mpmc
, callback_deleter
> super_t
;
57 callback_interpreter(void):
58 sem(0), running(false)
61 void add_callback(callback_type
* cb
)
63 super_t::add_callback(cb
);
69 running
.store(true, boost::memory_order_relaxed
);
75 running
.store(false, boost::memory_order_relaxed
);
80 void run(semaphore
& sync_sem
)
82 running
.store(true, boost::memory_order_relaxed
);
93 super_t::run_callbacks();
94 } while(likely(running
.load(boost::memory_order_relaxed
)));
99 boost::atomic
<bool> running
;
103 template <class callback_type
,
104 class init_functor
= detail::nop_functor
,
105 class callback_deleter
= boost::checked_deleter
<callback_type
> >
106 class threaded_callback_interpreter
:
107 public callback_interpreter
<callback_type
, true, callback_deleter
>,
110 typedef callback_interpreter
<callback_type
, true, callback_deleter
> super
;
112 boost::thread thread
;
115 threaded_callback_interpreter(void)
118 ~threaded_callback_interpreter(void)
120 if (super::running
.load())
124 void start_thread(void)
127 semaphore_sync
<semaphore
> sync(sync_sem
);
128 boost::thread
thr(boost::bind(&threaded_callback_interpreter::run_thread
, this, boost::ref(sync_sem
)));
132 void run_thread(semaphore
& sync_sem
)
134 init_functor::operator()();
135 super::run(sync_sem
);
138 void join_thread(void)
145 template <class callback_type
,
146 class init_functor
= detail::nop_functor
,
147 class callback_deleter
= boost::checked_deleter
<callback_type
> >
148 class callback_interpreter_threadpool
:
149 public callback_interpreter
<callback_type
, true, callback_deleter
>,
152 typedef callback_interpreter
<callback_type
, true, callback_deleter
> super
;
155 callback_interpreter_threadpool(uint16_t worker_thread_count
, bool rt
, uint16_t priority
):
156 worker_thread_count_(worker_thread_count
), priority(priority
), rt(rt
)
159 for (uint16_t i
= 0; i
!= worker_thread_count
; ++i
)
160 threads
.create_thread(boost::bind(&callback_interpreter_threadpool::run
, this, boost::ref(sync_sem
)));
162 for (uint16_t i
= 0; i
!= worker_thread_count
; ++i
)
166 ~callback_interpreter_threadpool(void)
168 if (super::running
.load())
172 void join_threads(void)
174 super::running
.store(false);
176 for (uint16_t i
= 0; i
!= worker_thread_count_
; ++i
)
182 void run(semaphore
& sync_sem
)
184 #ifdef NOVA_TT_PRIORITY_RT
186 thread_set_priority_rt(priority
);
189 thread_set_priority(priority
);
191 init_functor::operator()();
192 super::run(sync_sem
);
195 boost::thread_group threads
;
196 uint16_t worker_thread_count_
;
202 } /* namespace nova */
204 #endif /* UTILITIES_CALLBACK_INTERPRETER_HPP */