author | simon <simon> | 2002-12-02 10:25:36 (UTC) |
---|---|---|
committer | simon <simon> | 2002-12-02 10:25:36 (UTC) |
commit | 4447c70039ba1a3104bf229628699f33b8608973 (patch) (unidiff) | |
tree | 9810b07492a20f1ec9332da8c353469c8c120e2e | |
parent | 2f11390d99fba4eefa330322e11280802fb6dc26 (diff) | |
download | opie-4447c70039ba1a3104bf229628699f33b8608973.zip opie-4447c70039ba1a3104bf229628699f33b8608973.tar.gz opie-4447c70039ba1a3104bf229628699f33b8608973.tar.bz2 |
- put a thread pipe between the xine callback handler and the receiver
on qt side, to avoid problems like repaints from within a non-gui thread
-rw-r--r-- | noncore/multimedia/opieplayer2/lib.cpp | 15 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/lib.h | 8 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/threadutil.cpp | 284 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/threadutil.h | 176 |
4 files changed, 481 insertions, 2 deletions
diff --git a/noncore/multimedia/opieplayer2/lib.cpp b/noncore/multimedia/opieplayer2/lib.cpp index 19b64e8..1ac9809 100644 --- a/noncore/multimedia/opieplayer2/lib.cpp +++ b/noncore/multimedia/opieplayer2/lib.cpp | |||
@@ -29,32 +29,34 @@ | |||
29 | Inc., 59 Temple Place - Suite 330, | 29 | Inc., 59 Temple Place - Suite 330, |
30 | Boston, MA 02111-1307, USA. | 30 | Boston, MA 02111-1307, USA. |
31 | 31 | ||
32 | */ | 32 | */ |
33 | 33 | ||
34 | #include <stdio.h> | 34 | #include <stdio.h> |
35 | #include <stdlib.h> | 35 | #include <stdlib.h> |
36 | #include <qimage.h> | 36 | #include <qimage.h> |
37 | #include <qtextstream.h> | 37 | #include <qtextstream.h> |
38 | #include <qpe/resource.h> | 38 | #include <qpe/resource.h> |
39 | 39 | ||
40 | #include <qfile.h> | 40 | #include <qfile.h> |
41 | 41 | ||
42 | #include <qgfx_qws.h> | 42 | #include <qgfx_qws.h> |
43 | #include <qdirectpainter_qws.h> | 43 | #include <qdirectpainter_qws.h> |
44 | 44 | ||
45 | #include <assert.h> | ||
46 | |||
45 | #include "xinevideowidget.h" | 47 | #include "xinevideowidget.h" |
46 | #include "frame.h" | 48 | #include "frame.h" |
47 | #include "lib.h" | 49 | #include "lib.h" |
48 | 50 | ||
49 | typedef void (*display_xine_frame_t) (void *user_data, uint8_t* frame, | 51 | typedef void (*display_xine_frame_t) (void *user_data, uint8_t* frame, |
50 | int width, int height,int bytes ); | 52 | int width, int height,int bytes ); |
51 | 53 | ||
52 | extern "C" { | 54 | extern "C" { |
53 | xine_vo_driver_t* init_video_out_plugin( xine_cfg_entry_t* conf, void* video); | 55 | xine_vo_driver_t* init_video_out_plugin( xine_cfg_entry_t* conf, void* video); |
54 | int null_is_showing_video( const xine_vo_driver_t* self ); | 56 | int null_is_showing_video( const xine_vo_driver_t* self ); |
55 | void null_set_show_video( const xine_vo_driver_t* self, int show ); | 57 | void null_set_show_video( const xine_vo_driver_t* self, int show ); |
56 | int null_is_fullscreen( const xine_vo_driver_t* self ); | 58 | int null_is_fullscreen( const xine_vo_driver_t* self ); |
57 | void null_set_fullscreen( const xine_vo_driver_t* self, int screen ); | 59 | void null_set_fullscreen( const xine_vo_driver_t* self, int screen ); |
58 | int null_is_scaling( const xine_vo_driver_t* self ); | 60 | int null_is_scaling( const xine_vo_driver_t* self ); |
59 | void null_set_scaling( const xine_vo_driver_t* self, int scale ); | 61 | void null_set_scaling( const xine_vo_driver_t* self, int scale ); |
60 | void null_set_gui_width( const xine_vo_driver_t* self, int width ); | 62 | void null_set_gui_width( const xine_vo_driver_t* self, int width ); |
@@ -206,34 +208,45 @@ void Lib::seekTo( int time ) { | |||
206 | } | 208 | } |
207 | 209 | ||
208 | 210 | ||
209 | Frame Lib::currentFrame() { | 211 | Frame Lib::currentFrame() { |
210 | Frame frame; | 212 | Frame frame; |
211 | return frame; | 213 | return frame; |
212 | }; | 214 | }; |
213 | 215 | ||
214 | QString Lib::metaInfo( int number) { | 216 | QString Lib::metaInfo( int number) { |
215 | return xine_get_meta_info( m_stream, number ); | 217 | return xine_get_meta_info( m_stream, number ); |
216 | } | 218 | } |
217 | 219 | ||
218 | int Lib::error() { | 220 | int Lib::error() { |
219 | return xine_get_error( m_stream ); | 221 | return xine_get_error( m_stream ); |
220 | }; | 222 | }; |
221 | 223 | ||
224 | void Lib::receiveMessage( ThreadUtil::ChannelMessage *msg, SendType sendType ) | ||
225 | { | ||
226 | assert( sendType == ThreadUtil::Channel::OneWay ); | ||
227 | handleXineEvent( msg->type() ); | ||
228 | delete msg; | ||
229 | } | ||
230 | |||
222 | void Lib::handleXineEvent( const xine_event_t* t ) { | 231 | void Lib::handleXineEvent( const xine_event_t* t ) { |
223 | if ( t->type == XINE_EVENT_UI_PLAYBACK_FINISHED ) { | 232 | send( new ThreadUtil::ChannelMessage( t->type ), OneWay ); |
233 | } | ||
234 | |||
235 | void Lib::handleXineEvent( int type ) { | ||
236 | if ( type == XINE_EVENT_UI_PLAYBACK_FINISHED ) { | ||
224 | emit stopped(); | 237 | emit stopped(); |
225 | } | 238 | } |
226 | } | 239 | } |
227 | 240 | ||
228 | 241 | ||
229 | void Lib::setShowVideo( bool video ) { | 242 | void Lib::setShowVideo( bool video ) { |
230 | m_video = video; | 243 | m_video = video; |
231 | ::null_set_show_video( m_videoOutput, video ); | 244 | ::null_set_show_video( m_videoOutput, video ); |
232 | } | 245 | } |
233 | 246 | ||
234 | bool Lib::isShowingVideo() { | 247 | bool Lib::isShowingVideo() { |
235 | return ::null_is_showing_video( m_videoOutput ); | 248 | return ::null_is_showing_video( m_videoOutput ); |
236 | } | 249 | } |
237 | 250 | ||
238 | bool Lib::hasVideo() { | 251 | bool Lib::hasVideo() { |
239 | return xine_get_stream_info( m_stream, 18 ); | 252 | return xine_get_stream_info( m_stream, 18 ); |
diff --git a/noncore/multimedia/opieplayer2/lib.h b/noncore/multimedia/opieplayer2/lib.h index 38938a1..75b9f12 100644 --- a/noncore/multimedia/opieplayer2/lib.h +++ b/noncore/multimedia/opieplayer2/lib.h | |||
@@ -28,45 +28,47 @@ | |||
28 | If not, write to the Free Software Foundation, | 28 | If not, write to the Free Software Foundation, |
29 | Inc., 59 Temple Place - Suite 330, | 29 | Inc., 59 Temple Place - Suite 330, |
30 | Boston, MA 02111-1307, USA. | 30 | Boston, MA 02111-1307, USA. |
31 | 31 | ||
32 | */ | 32 | */ |
33 | 33 | ||
34 | #ifndef ZECKEXINELIB_H | 34 | #ifndef ZECKEXINELIB_H |
35 | #define ZECKEXINELIB_H | 35 | #define ZECKEXINELIB_H |
36 | 36 | ||
37 | #include <qcstring.h> | 37 | #include <qcstring.h> |
38 | #include <qstring.h> | 38 | #include <qstring.h> |
39 | #include <qobject.h> | 39 | #include <qobject.h> |
40 | 40 | ||
41 | #include <xine.h> | 41 | #include <xine.h> |
42 | //#include "xine.h" | 42 | //#include "xine.h" |
43 | 43 | ||
44 | #include "threadutil.h" | ||
45 | |||
44 | class XineVideoWidget; | 46 | class XineVideoWidget; |
45 | 47 | ||
46 | namespace XINE { | 48 | namespace XINE { |
47 | 49 | ||
48 | /** | 50 | /** |
49 | * Lib wrapps the simple interface | 51 | * Lib wrapps the simple interface |
50 | * of libxine for easy every day use | 52 | * of libxine for easy every day use |
51 | * This will become a full C++ Wrapper | 53 | * This will become a full C++ Wrapper |
52 | * It supports playing, pausing, info, | 54 | * It supports playing, pausing, info, |
53 | * stooping, seeking. | 55 | * stooping, seeking. |
54 | */ | 56 | */ |
55 | class Frame; | 57 | class Frame; |
56 | class Lib : public QObject { | 58 | class Lib : public ThreadUtil::Channel { |
57 | Q_OBJECT | 59 | Q_OBJECT |
58 | public: | 60 | public: |
59 | Lib(XineVideoWidget* = 0); | 61 | Lib(XineVideoWidget* = 0); |
60 | ~Lib(); | 62 | ~Lib(); |
61 | QCString version(); | 63 | QCString version(); |
62 | int majorVersion()/*const*/; | 64 | int majorVersion()/*const*/; |
63 | int minorVersion()/*const*/; | 65 | int minorVersion()/*const*/; |
64 | int subVersion()/*const*/; | 66 | int subVersion()/*const*/; |
65 | 67 | ||
66 | 68 | ||
67 | void resize ( const QSize &s ); | 69 | void resize ( const QSize &s ); |
68 | 70 | ||
69 | int play( const QString& fileName, | 71 | int play( const QString& fileName, |
70 | int startPos = 0, | 72 | int startPos = 0, |
71 | int start_time = 0 ); | 73 | int start_time = 0 ); |
72 | void stop() /*const*/; | 74 | void stop() /*const*/; |
@@ -163,40 +165,44 @@ namespace XINE { | |||
163 | */ | 165 | */ |
164 | Frame currentFrame()/*const*/; | 166 | Frame currentFrame()/*const*/; |
165 | 167 | ||
166 | /** | 168 | /** |
167 | * Returns the error code | 169 | * Returns the error code |
168 | * XINE_ERROR_NONE 0 | 170 | * XINE_ERROR_NONE 0 |
169 | * XINE_ERROR_NO_INPUT_PLUGIN 1 | 171 | * XINE_ERROR_NO_INPUT_PLUGIN 1 |
170 | * XINE_ERROR_NO_DEMUXER_PLUGIN 2 | 172 | * XINE_ERROR_NO_DEMUXER_PLUGIN 2 |
171 | * XINE_ERROR_DEMUXER_FAILED 3 | 173 | * XINE_ERROR_DEMUXER_FAILED 3 |
172 | */ | 174 | */ |
173 | int error() /*const*/; | 175 | int error() /*const*/; |
174 | 176 | ||
175 | signals: | 177 | signals: |
176 | 178 | ||
177 | void stopped(); | 179 | void stopped(); |
178 | 180 | ||
181 | protected: | ||
182 | virtual void receiveMessage( ThreadUtil::ChannelMessage *msg, SendType sendType ); | ||
183 | |||
179 | private: | 184 | private: |
180 | int m_bytes_per_pixel; | 185 | int m_bytes_per_pixel; |
181 | int m_length, m_pos, m_time; | 186 | int m_length, m_pos, m_time; |
182 | int m_major_version, m_minor_version, m_sub_version; | 187 | int m_major_version, m_minor_version, m_sub_version; |
183 | bool m_video:1; | 188 | bool m_video:1; |
184 | XineVideoWidget *m_wid; | 189 | XineVideoWidget *m_wid; |
185 | xine_t *m_xine; | 190 | xine_t *m_xine; |
186 | xine_stream_t *m_stream; | 191 | xine_stream_t *m_stream; |
187 | xine_cfg_entry_t *m_config; | 192 | xine_cfg_entry_t *m_config; |
188 | xine_vo_driver_t *m_videoOutput; | 193 | xine_vo_driver_t *m_videoOutput; |
189 | xine_ao_driver_t* m_audioOutput; | 194 | xine_ao_driver_t* m_audioOutput; |
190 | xine_event_queue_t *m_queue; | 195 | xine_event_queue_t *m_queue; |
191 | 196 | ||
192 | void handleXineEvent( const xine_event_t* t ); | 197 | void handleXineEvent( const xine_event_t* t ); |
198 | void handleXineEvent( int type ); | ||
193 | void drawFrame( uint8_t* frame, int width, int height, int bytes ); | 199 | void drawFrame( uint8_t* frame, int width, int height, int bytes ); |
194 | // C -> C++ bridge for the event system | 200 | // C -> C++ bridge for the event system |
195 | static void xine_event_handler( void* user_data, const xine_event_t* t); | 201 | static void xine_event_handler( void* user_data, const xine_event_t* t); |
196 | static void xine_display_frame( void* user_data, uint8_t* frame , | 202 | static void xine_display_frame( void* user_data, uint8_t* frame , |
197 | int width, int height, int bytes ); | 203 | int width, int height, int bytes ); |
198 | }; | 204 | }; |
199 | }; | 205 | }; |
200 | 206 | ||
201 | 207 | ||
202 | #endif | 208 | #endif |
diff --git a/noncore/multimedia/opieplayer2/threadutil.cpp b/noncore/multimedia/opieplayer2/threadutil.cpp new file mode 100644 index 0000000..63863b4 --- a/dev/null +++ b/noncore/multimedia/opieplayer2/threadutil.cpp | |||
@@ -0,0 +1,284 @@ | |||
1 | /* This file is part of the KDE project | ||
2 | Copyright (C) 1999 Simon Hausmann <hausmann@kde.org> | ||
3 | |||
4 | This library is free software; you can redistribute it and/or | ||
5 | modify it under the terms of the GNU Library General Public | ||
6 | License as published by the Free Software Foundation; either | ||
7 | version 2 of the License, or (at your option) any later version. | ||
8 | |||
9 | This library 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 GNU | ||
12 | Library General Public License for more details. | ||
13 | |||
14 | You should have received a copy of the GNU Library General Public License | ||
15 | along with this library; see the file COPYING.LIB. If not, write to | ||
16 | the Free Software Foundation, Inc., 59 Temple Place - Suite 330, | ||
17 | Boston, MA 02111-1307, USA. | ||
18 | */ | ||
19 | |||
20 | #include "threadutil.h" | ||
21 | |||
22 | #include <qsocketnotifier.h> | ||
23 | |||
24 | #include <pthread.h> | ||
25 | #include <assert.h> | ||
26 | #include <unistd.h> | ||
27 | |||
28 | using namespace ThreadUtil; | ||
29 | |||
30 | struct Mutex::Data | ||
31 | { | ||
32 | Data() | ||
33 | { | ||
34 | pthread_mutex_init( &mutex, 0 ); | ||
35 | } | ||
36 | ~Data() | ||
37 | { | ||
38 | pthread_mutex_destroy( &mutex ); | ||
39 | } | ||
40 | |||
41 | pthread_mutex_t mutex; | ||
42 | }; | ||
43 | |||
44 | Mutex::Mutex() | ||
45 | : d( new Data ) | ||
46 | { | ||
47 | } | ||
48 | |||
49 | Mutex::~Mutex() | ||
50 | { | ||
51 | delete d; | ||
52 | } | ||
53 | |||
54 | void Mutex::lock() | ||
55 | { | ||
56 | pthread_mutex_lock( &d->mutex ); | ||
57 | } | ||
58 | |||
59 | void Mutex::unlock() | ||
60 | { | ||
61 | pthread_mutex_unlock( &d->mutex ); | ||
62 | } | ||
63 | |||
64 | bool Mutex::tryLock() | ||
65 | { | ||
66 | return pthread_mutex_trylock( &d->mutex ) == 0; | ||
67 | } | ||
68 | |||
69 | bool Mutex::isLocked() | ||
70 | { | ||
71 | if ( !tryLock() ) | ||
72 | return true; | ||
73 | |||
74 | unlock(); | ||
75 | return false; | ||
76 | } | ||
77 | |||
78 | struct WaitCondition::Data | ||
79 | { | ||
80 | Data() | ||
81 | { | ||
82 | int result = pthread_cond_init( &waitCondition, 0 ); | ||
83 | assert( result == 0 ); | ||
84 | } | ||
85 | ~Data() | ||
86 | { | ||
87 | pthread_cond_destroy( &waitCondition ); | ||
88 | } | ||
89 | |||
90 | pthread_cond_t waitCondition; | ||
91 | }; | ||
92 | |||
93 | WaitCondition::WaitCondition() | ||
94 | : d( new Data ) | ||
95 | { | ||
96 | } | ||
97 | |||
98 | WaitCondition::~WaitCondition() | ||
99 | { | ||
100 | delete d; | ||
101 | } | ||
102 | |||
103 | bool WaitCondition::wait() | ||
104 | { | ||
105 | Mutex m; | ||
106 | m.lock(); | ||
107 | return wait( m ); | ||
108 | } | ||
109 | |||
110 | bool WaitCondition::wait( Mutex &mutex ) | ||
111 | { | ||
112 | return pthread_cond_wait( &d->waitCondition, &mutex.d->mutex ); | ||
113 | } | ||
114 | |||
115 | void WaitCondition::wakeOne() | ||
116 | { | ||
117 | pthread_cond_signal( &d->waitCondition ); | ||
118 | } | ||
119 | |||
120 | void WaitCondition::wakeAll() | ||
121 | { | ||
122 | pthread_cond_broadcast( &d->waitCondition ); | ||
123 | } | ||
124 | |||
125 | OnewayNotifier::OnewayNotifier() | ||
126 | { | ||
127 | int fds[ 2 ]; | ||
128 | pipe( fds ); | ||
129 | m_readFd = fds[ 0 ]; | ||
130 | m_writeFd = fds[ 1 ]; | ||
131 | |||
132 | m_notifier = new QSocketNotifier( m_readFd, QSocketNotifier::Read ); | ||
133 | connect( m_notifier, SIGNAL( activated( int ) ), | ||
134 | this, SLOT( wakeUp() ) ); | ||
135 | } | ||
136 | |||
137 | OnewayNotifier::~OnewayNotifier() | ||
138 | { | ||
139 | delete m_notifier; | ||
140 | |||
141 | ::close( m_readFd ); | ||
142 | ::close( m_writeFd ); | ||
143 | } | ||
144 | |||
145 | void OnewayNotifier::notify() | ||
146 | { | ||
147 | const char c = 42; | ||
148 | ::write( m_writeFd, &c, 1 ); | ||
149 | } | ||
150 | |||
151 | void OnewayNotifier::wakeUp() | ||
152 | { | ||
153 | char c = 0; | ||
154 | |||
155 | if ( ::read( m_readFd, &c, 1 ) != 1 ) | ||
156 | return; | ||
157 | |||
158 | emit awake(); | ||
159 | } | ||
160 | |||
161 | ChannelMessage::ChannelMessage( int type ) | ||
162 | : m_type( type ), m_isCall( false ), m_replied( false ), | ||
163 | m_inEventHandler( false ) | ||
164 | { | ||
165 | } | ||
166 | |||
167 | ChannelMessage::~ChannelMessage() | ||
168 | { | ||
169 | if ( m_guard.isLocked() ) | ||
170 | m_guard.unlock(); | ||
171 | } | ||
172 | |||
173 | void ChannelMessage::reply() | ||
174 | { | ||
175 | if ( !m_isCall ) | ||
176 | { | ||
177 | qDebug( "ChannelMessage::reply() - can't reply oneway message!" ); | ||
178 | return; | ||
179 | } | ||
180 | |||
181 | if ( m_inEventHandler ) | ||
182 | { | ||
183 | m_replied = true; | ||
184 | return; | ||
185 | } | ||
186 | |||
187 | m_condition.wakeOne(); | ||
188 | m_guard.unlock(); | ||
189 | } | ||
190 | |||
191 | struct Channel::Private | ||
192 | { | ||
193 | Private() | ||
194 | { | ||
195 | ownerThread = pthread_self(); | ||
196 | } | ||
197 | |||
198 | pthread_t ownerThread; | ||
199 | }; | ||
200 | |||
201 | Channel::Channel( QObject *parent, const char *name ) | ||
202 | : QObject( parent, name ), d( new Private ) | ||
203 | { | ||
204 | connect( &m_notifier, SIGNAL( awake() ), | ||
205 | this, SLOT( deliver() ) ); | ||
206 | } | ||
207 | |||
208 | Channel::~Channel() | ||
209 | { | ||
210 | delete d; | ||
211 | } | ||
212 | |||
213 | void Channel::send( ChannelMessage *message, SendType type ) | ||
214 | { | ||
215 | if ( type == WaitForReply ) | ||
216 | { | ||
217 | message->m_guard.lock(); | ||
218 | message->m_isCall = true; | ||
219 | } | ||
220 | |||
221 | m_pendingMessagesGuard.lock(); | ||
222 | m_pendingMessages << MsgEnvelope( type, message ); | ||
223 | m_pendingMessagesGuard.unlock(); | ||
224 | |||
225 | if ( d->ownerThread == pthread_self() ) { | ||
226 | assert( type != WaitForReply ); | ||
227 | |||
228 | deliver(); | ||
229 | } | ||
230 | else | ||
231 | m_notifier.notify(); | ||
232 | //QThread::postEvent( this, new QCustomEvent( QEvent::User, envelope ) ); | ||
233 | |||
234 | if ( type == WaitForReply ) | ||
235 | { | ||
236 | message->m_condition.wait( message->m_guard ); | ||
237 | message->m_guard.unlock(); | ||
238 | } | ||
239 | } | ||
240 | |||
241 | void Channel::deliver() | ||
242 | { | ||
243 | AutoLock lock( m_pendingMessagesGuard ); | ||
244 | |||
245 | while ( !m_pendingMessages.isEmpty() ) { | ||
246 | MsgEnvelope envelope = m_pendingMessages.first(); | ||
247 | |||
248 | m_pendingMessages.remove( m_pendingMessages.begin() ); | ||
249 | |||
250 | m_pendingMessagesGuard.unlock(); | ||
251 | deliverOne( envelope ); | ||
252 | m_pendingMessagesGuard.lock(); | ||
253 | } | ||
254 | } | ||
255 | |||
256 | void Channel::deliverOne( const MsgEnvelope &envelope ) | ||
257 | { | ||
258 | ChannelMessage *msg = envelope.msg; | ||
259 | |||
260 | assert( msg ); | ||
261 | |||
262 | if ( envelope.type == WaitForReply ) | ||
263 | { | ||
264 | msg->m_guard.lock(); | ||
265 | msg->m_inEventHandler = true; | ||
266 | } | ||
267 | |||
268 | receiveMessage( msg, envelope.type ); | ||
269 | |||
270 | if ( envelope.type == WaitForReply ) | ||
271 | { | ||
272 | msg->m_inEventHandler = false; | ||
273 | if ( msg->m_replied ) | ||
274 | { | ||
275 | msg->m_condition.wakeOne(); | ||
276 | // this is a bit tricky. we unlock only when we reply. | ||
277 | // reply() does an unlock as well. | ||
278 | msg->m_guard.unlock(); | ||
279 | } | ||
280 | } | ||
281 | } | ||
282 | |||
283 | /* vim: et sw=4 ts=4 | ||
284 | */ | ||
diff --git a/noncore/multimedia/opieplayer2/threadutil.h b/noncore/multimedia/opieplayer2/threadutil.h new file mode 100644 index 0000000..bcb9db9 --- a/dev/null +++ b/noncore/multimedia/opieplayer2/threadutil.h | |||
@@ -0,0 +1,176 @@ | |||
1 | /* This file is part of the KDE project | ||
2 | Copyright (C) 1999 Simon Hausmann <hausmann@kde.org> | ||
3 | |||
4 | This library is free software; you can redistribute it and/or | ||
5 | modify it under the terms of the GNU Library General Public | ||
6 | License as published by the Free Software Foundation; either | ||
7 | version 2 of the License, or (at your option) any later version. | ||
8 | |||
9 | This library 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 GNU | ||
12 | Library General Public License for more details. | ||
13 | |||
14 | You should have received a copy of the GNU Library General Public License | ||
15 | along with this library; see the file COPYING.LIB. If not, write to | ||
16 | the Free Software Foundation, Inc., 59 Temple Place - Suite 330, | ||
17 | Boston, MA 02111-1307, USA. | ||
18 | */ | ||
19 | |||
20 | #ifndef THREADUTIL_H | ||
21 | #define THREADUTIL_H | ||
22 | |||
23 | #include <qvaluelist.h> | ||
24 | #include <qobject.h> | ||
25 | #include <qguardedptr.h> | ||
26 | |||
27 | class QSocketNotifier; | ||
28 | |||
29 | namespace ThreadUtil | ||
30 | { | ||
31 | |||
32 | class Mutex | ||
33 | { | ||
34 | friend class WaitCondition; | ||
35 | public: | ||
36 | Mutex(); | ||
37 | ~Mutex(); | ||
38 | |||
39 | void lock(); | ||
40 | void unlock(); | ||
41 | bool tryLock(); | ||
42 | bool isLocked(); | ||
43 | |||
44 | private: | ||
45 | struct Data; | ||
46 | Data *d; | ||
47 | |||
48 | Mutex( const Mutex & ); | ||
49 | Mutex &operator=( const Mutex & ); | ||
50 | }; | ||
51 | |||
52 | class AutoLock | ||
53 | { | ||
54 | public: | ||
55 | AutoLock( Mutex &mutex ) : m_mutex( mutex ) { m_mutex.lock(); } | ||
56 | ~AutoLock() { m_mutex.unlock(); } | ||
57 | |||
58 | Mutex *operator &() const { return &m_mutex; } | ||
59 | |||
60 | private: | ||
61 | Mutex &m_mutex; | ||
62 | }; | ||
63 | |||
64 | class WaitCondition | ||
65 | { | ||
66 | public: | ||
67 | WaitCondition(); | ||
68 | ~WaitCondition(); | ||
69 | |||
70 | bool wait(); | ||
71 | bool wait( Mutex &mutex ); | ||
72 | |||
73 | void wakeOne(); | ||
74 | void wakeAll(); | ||
75 | |||
76 | private: | ||
77 | struct Data; | ||
78 | Data *d; | ||
79 | |||
80 | WaitCondition( const WaitCondition & ); | ||
81 | WaitCondition &operator=( const WaitCondition & ); | ||
82 | }; | ||
83 | |||
84 | class OnewayNotifier : public QObject | ||
85 | { | ||
86 | Q_OBJECT | ||
87 | public: | ||
88 | OnewayNotifier(); | ||
89 | ~OnewayNotifier(); | ||
90 | |||
91 | void notify(); | ||
92 | |||
93 | signals: | ||
94 | void awake(); | ||
95 | |||
96 | private slots: | ||
97 | void wakeUp(); | ||
98 | |||
99 | private: | ||
100 | int m_readFd; | ||
101 | int m_writeFd; | ||
102 | QSocketNotifier *m_notifier; | ||
103 | }; | ||
104 | |||
105 | |||
106 | class Channel; | ||
107 | |||
108 | class ChannelMessage | ||
109 | { | ||
110 | friend class Channel; | ||
111 | public: | ||
112 | ChannelMessage( int type = -1 ); | ||
113 | virtual ~ChannelMessage(); | ||
114 | |||
115 | int type() const { return m_type; } | ||
116 | |||
117 | void reply(); | ||
118 | |||
119 | private: | ||
120 | ChannelMessage( const ChannelMessage & ); | ||
121 | ChannelMessage &operator=( const ChannelMessage ); | ||
122 | |||
123 | int m_type; | ||
124 | bool m_isCall : 1; | ||
125 | bool m_replied : 1; | ||
126 | bool m_inEventHandler : 1; | ||
127 | Mutex m_guard; | ||
128 | WaitCondition m_condition; | ||
129 | QGuardedPtr<Channel> m_channel; | ||
130 | }; | ||
131 | |||
132 | class Channel : public QObject | ||
133 | { | ||
134 | Q_OBJECT | ||
135 | public: | ||
136 | enum SendType { OneWay, WaitForReply }; | ||
137 | Channel( QObject *parent = 0, const char *name = 0 ); | ||
138 | virtual ~Channel(); | ||
139 | |||
140 | void send( ChannelMessage *message, SendType type ); | ||
141 | |||
142 | protected: | ||
143 | virtual void receiveMessage( ChannelMessage *message, SendType type ) = 0; | ||
144 | |||
145 | private slots: | ||
146 | void deliver(); | ||
147 | |||
148 | private: | ||
149 | OnewayNotifier m_notifier; | ||
150 | |||
151 | struct MsgEnvelope | ||
152 | { | ||
153 | MsgEnvelope() : type( OneWay ), msg( 0 ) {} | ||
154 | MsgEnvelope( SendType _type , ChannelMessage *_msg ) | ||
155 | : type( _type ), msg( _msg ) {} | ||
156 | |||
157 | SendType type; | ||
158 | ChannelMessage *msg; | ||
159 | }; | ||
160 | |||
161 | void deliverOne( const MsgEnvelope &envelope ); | ||
162 | |||
163 | typedef QValueList<MsgEnvelope> MsgEnvelopeList; | ||
164 | |||
165 | MsgEnvelopeList m_pendingMessages; | ||
166 | Mutex m_pendingMessagesGuard; | ||
167 | |||
168 | struct Private; | ||
169 | Private *d; | ||
170 | }; | ||
171 | |||
172 | } | ||
173 | |||
174 | #endif // THREADUTIL_H | ||
175 | /* vim: et sw=4 ts=4 | ||
176 | */ | ||