xref: /trunk/main/fpicker/source/win32/filepicker/asynceventnotifier.cxx (revision 1ecadb572e7010ff3b3382ad9bf179dbc6efadbb)
1 /*************************************************************************
2  *
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * Copyright 2000, 2010 Oracle and/or its affiliates.
6  *
7  * OpenOffice.org - a multi-platform office productivity suite
8  *
9  * This file is part of OpenOffice.org.
10  *
11  * OpenOffice.org is free software: you can redistribute it and/or modify
12  * it under the terms of the GNU Lesser General Public License version 3
13  * only, as published by the Free Software Foundation.
14  *
15  * OpenOffice.org is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU Lesser General Public License version 3 for more details
19  * (a copy is included in the LICENSE file that accompanied this code).
20  *
21  * You should have received a copy of the GNU Lesser General Public License
22  * version 3 along with OpenOffice.org.  If not, see
23  * <http://www.openoffice.org/license.html>
24  * for a copy of the LGPLv3 License.
25  *
26  ************************************************************************/
27 
28 // MARKER(update_precomp.py): autogen include statement, do not remove
29 #include "precompiled_fpicker.hxx"
30 
31 //------------------------------------------------------------------------
32 // includes
33 //------------------------------------------------------------------------
34 #include <osl/diagnose.h>
35 #include "asynceventnotifier.hxx"
36 #include <com/sun/star/uno/RuntimeException.hpp>
37 #include <com/sun/star/ui/dialogs/XFilePickerListener.hpp>
38 
39 #include <process.h>
40 #include <memory>
41 #include "SolarMutex.hxx"
42 
43 //------------------------------------------------
44 //
45 //------------------------------------------------
46 
47 using namespace com::sun::star;
48 using ::com::sun::star::ui::dialogs::XFilePickerListener;
49 
50 //------------------------------------------------
51 //
52 //------------------------------------------------
53 
54 CAsyncEventNotifier::CAsyncEventNotifier(cppu::OBroadcastHelper& rBroadcastHelper) :
55     m_hThread(0),
56     m_bRun(false),
57     m_ThreadId(0),
58     m_rBroadcastHelper(rBroadcastHelper),
59     m_NotifyEvent(m_hEvents[0]),
60     m_ResumeNotifying(m_hEvents[1])
61 {
62     // m_NotifyEvent
63     m_hEvents[0] = CreateEvent(0,       /* no security */
64                                true,    /* manual reset */
65                                false,   /* initial state not signaled */
66                                0);      /* automatic name */
67 
68     // m_ResumeNotifying
69     m_hEvents[1] = CreateEvent(0,       /* no security */
70                                true,    /* manual reset */
71                                false,   /* initial state not signaled */
72                                0);      /* automatic name */
73 }
74 
75 //------------------------------------------------
76 //
77 //------------------------------------------------
78 
79 CAsyncEventNotifier::~CAsyncEventNotifier()
80 {
81     OSL_ENSURE(0 == m_hThread,"Thread not stopped, destroying this instance leads to desaster");
82 
83     CloseHandle(m_hEvents[0]);
84     CloseHandle(m_hEvents[1]);
85 }
86 
87 //------------------------------------------------
88 //
89 //------------------------------------------------
90 
91 void SAL_CALL CAsyncEventNotifier::addListener(const uno::Type&                         aType    ,
92                                                const uno::Reference< uno::XInterface >& xListener)
93 {
94     if ( m_rBroadcastHelper.bDisposed )
95         throw lang::DisposedException(
96             ::rtl::OUString::createFromAscii( "FilePicker is already disposed" ),
97             uno::Reference< uno::XInterface >() );
98 
99     if ( m_rBroadcastHelper.bInDispose )
100         throw lang::DisposedException(
101             ::rtl::OUString::createFromAscii( "FilePicker will be disposed now." ),
102             uno::Reference< uno::XInterface >() );
103 
104     m_rBroadcastHelper.aLC.addInterface( aType, xListener );
105 }
106 
107 //------------------------------------------------
108 //
109 //------------------------------------------------
110 
111 void SAL_CALL CAsyncEventNotifier::removeListener(const uno::Type&                         aType    ,
112                                                   const uno::Reference< uno::XInterface >& xListener)
113 {
114     if ( m_rBroadcastHelper.bDisposed )
115         throw lang::DisposedException(
116             ::rtl::OUString::createFromAscii( "FilePicker is already disposed." ),
117             uno::Reference< uno::XInterface >() );
118 
119     m_rBroadcastHelper.aLC.removeInterface( aType, xListener );
120 }
121 
122 //------------------------------------------------
123 //
124 //------------------------------------------------
125 
126 bool SAL_CALL CAsyncEventNotifier::startup(bool bCreateSuspended)
127 {
128     osl::MutexGuard aGuard(m_Mutex);
129 
130     // m_bRun may already be false because of a
131     // call to stop but the thread did not yet
132     // terminate so m_hEventNotifierThread is
133     // yet a valid thread handle that should
134     // not be overwritten
135     if (!m_bRun)
136     {
137         if (!bCreateSuspended)
138             SetEvent(m_ResumeNotifying);
139 
140         m_hThread = (HANDLE)_beginthreadex(
141             NULL, 0, CAsyncEventNotifier::ThreadProc, this, 0, &m_ThreadId);
142 
143         OSL_ASSERT(0 != m_hThread);
144 
145         if (m_hThread)
146             m_bRun = true;
147     }
148 
149     OSL_POSTCOND(m_bRun,"Could not start event notifier!");
150 
151     return m_bRun;
152 }
153 
154 //------------------------------------------------
155 //
156 //------------------------------------------------
157 
158 void SAL_CALL CAsyncEventNotifier::shutdown()
159 {
160     unsigned nThreadId = GetCurrentThreadId();
161 
162     OSL_PRECOND(nThreadId != m_ThreadId, "Method called in wrong thread context!");
163 
164     osl::ResettableMutexGuard aGuard(m_Mutex);
165 
166     OSL_PRECOND(m_bRun,"Event notifier does not run!");
167 
168     m_bRun = false;
169     m_EventList.clear();
170 
171     // awake the the notifier thread
172     SetEvent(m_ResumeNotifying);
173     SetEvent(m_NotifyEvent);
174 
175     // releas the mutex here because the event
176     // notifier thread may need it to finish
177     aGuard.clear();
178 
179     // we are waiting infinite, so error will
180     // be better detected in form of deadlocks
181     if (WaitForSingleObject(m_hThread, INFINITE) == WAIT_FAILED) {
182         OSL_ENSURE(false, "Waiting for thread termination failed!");
183     }
184 
185     // lock mutex again to reset m_hThread
186     // and prevent a race with start()
187     aGuard.reset();
188 
189     CloseHandle(m_hThread);
190     m_hThread = 0;
191 }
192 
193 //------------------------------------------------
194 //
195 //------------------------------------------------
196 
197 void CAsyncEventNotifier::suspend()
198 {
199     ResetEvent(m_ResumeNotifying);
200 }
201 
202 //------------------------------------------------
203 //
204 //------------------------------------------------
205 
206 void CAsyncEventNotifier::resume()
207 {
208     SetEvent(m_ResumeNotifying);
209 }
210 
211 //------------------------------------------------
212 //
213 //------------------------------------------------
214 
215 void SAL_CALL CAsyncEventNotifier::notifyEvent(CEventNotification* EventNotification)
216 {
217     osl::MutexGuard aGuard(m_Mutex);
218 
219     OSL_ENSURE(m_bRun,"Event notifier is not running!");
220 
221     if (m_bRun)
222     {
223         m_EventList.push_back(EventNotification);
224         SetEvent(m_NotifyEvent);
225     }
226 }
227 
228 //------------------------------------------------
229 //
230 //------------------------------------------------
231 
232 size_t SAL_CALL CAsyncEventNotifier::getEventListSize()
233 {
234     osl::MutexGuard aGuard(m_Mutex);
235     return m_EventList.size();
236 }
237 
238 //------------------------------------------------
239 //
240 //------------------------------------------------
241 
242 void SAL_CALL CAsyncEventNotifier::resetNotifyEvent()
243 {
244     osl::MutexGuard aGuard(m_Mutex);
245     if (0 == m_EventList.size())
246         ResetEvent(m_NotifyEvent);
247 }
248 
249 //------------------------------------------------
250 //
251 //------------------------------------------------
252 
253 CEventNotification* SAL_CALL CAsyncEventNotifier::getNextEventRecord()
254 {
255     osl::MutexGuard aGuard(m_Mutex);
256     return m_EventList.front();
257 }
258 
259 //------------------------------------------------
260 //
261 //------------------------------------------------
262 
263 void SAL_CALL CAsyncEventNotifier::removeNextEventRecord()
264 {
265     osl::MutexGuard aGuard(m_Mutex);
266     m_EventList.pop_front();
267 }
268 
269 //------------------------------------------------
270 //
271 //------------------------------------------------
272 
273 void SAL_CALL CAsyncEventNotifier::run()
274 {
275     while (m_bRun)
276     {
277         WaitForMultipleObjects(2, m_hEvents, true, INFINITE);
278 
279         if (m_bRun)
280         {
281             while (getEventListSize() > 0)
282             {
283                 std::auto_ptr<CEventNotification> EventNotification(getNextEventRecord());
284                 removeNextEventRecord();
285 
286                 ::cppu::OInterfaceContainerHelper* pICHelper =
287                     m_rBroadcastHelper.getContainer(getCppuType((uno::Reference<XFilePickerListener>*)0));
288 
289                 if (pICHelper)
290                 {
291                     ::cppu::OInterfaceIteratorHelper iter(*pICHelper);
292 
293                     while(iter.hasMoreElements())
294                     {
295                         try
296                         {
297                             EventNotification->notifyEventListener(iter.next());
298                         }
299                         catch(uno::RuntimeException&)
300                         {
301                             OSL_ENSURE(sal_False,"RuntimeException during event dispatching");
302                         }
303                     }
304                 }
305 
306             } // while(getEventListSize() > 0)
307 
308             resetNotifyEvent();
309 
310         } // if (m_bRun)
311 
312     } // while(m_bRun)
313 }
314 
315 //------------------------------------------------
316 //
317 //------------------------------------------------
318 
319 unsigned int WINAPI CAsyncEventNotifier::ThreadProc(LPVOID pParam)
320 {
321     CAsyncEventNotifier* pInst = reinterpret_cast< CAsyncEventNotifier* >(pParam);
322     OSL_ASSERT(pInst);
323 
324     pInst->run();
325 
326     return 0;
327 }
328