VirtualBox

source: vbox/trunk/src/VBox/Main/glue/EventQueue.cpp@ 37989

Last change on this file since 37989 was 33720, checked in by vboxsync, 14 years ago

EventQueue::uninit: Don't run the native mac os x runloop in default mode at this time as the GUI toolkit might not be ready for this. Instead just do ProcessPendingEvents() as this will clear any signalled native events.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.7 KB
Line 
1/* $Id: EventQueue.cpp 33720 2010-11-03 10:54:42Z vboxsync $ */
2/** @file
3 * MS COM / XPCOM Abstraction Layer:
4 * Event and EventQueue class declaration
5 */
6
7/*
8 * Copyright (C) 2006-2010 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include "VBox/com/EventQueue.h"
20
21#ifdef RT_OS_DARWIN
22# include <CoreFoundation/CFRunLoop.h>
23#endif
24
25#if defined(VBOX_WITH_XPCOM) && !defined(RT_OS_DARWIN) && !defined(RT_OS_OS2)
26# define USE_XPCOM_QUEUE
27#endif
28
29#include <iprt/err.h>
30#include <iprt/time.h>
31#include <iprt/thread.h>
32#ifdef USE_XPCOM_QUEUE
33# include <errno.h>
34#endif
35
36namespace com
37{
38
39// EventQueue class
40////////////////////////////////////////////////////////////////////////////////
41
42#ifndef VBOX_WITH_XPCOM
43
44# define CHECK_THREAD_RET(ret) \
45 do { \
46 AssertMsg(GetCurrentThreadId() == mThreadId, ("Must be on event queue thread!")); \
47 if (GetCurrentThreadId() != mThreadId) \
48 return ret; \
49 } while (0)
50
51/** Magic LPARAM value for the WM_USER messages that we're posting.
52 * @remarks This magic value is duplicated in
53 * vboxapi/PlatformMSCOM::interruptWaitEvents(). */
54#define EVENTQUEUE_WIN_LPARAM_MAGIC UINT32_C(0xf241b819)
55
56
57#else // VBOX_WITH_XPCOM
58
59# define CHECK_THREAD_RET(ret) \
60 do { \
61 if (!mEventQ) \
62 return ret; \
63 BOOL isOnCurrentThread = FALSE; \
64 mEventQ->IsOnCurrentThread(&isOnCurrentThread); \
65 AssertMsg(isOnCurrentThread, ("Must be on event queue thread!")); \
66 if (!isOnCurrentThread) \
67 return ret; \
68 } while (0)
69
70#endif // VBOX_WITH_XPCOM
71
72/** Pointer to the main event queue. */
73EventQueue *EventQueue::sMainQueue = NULL;
74
75
76#ifdef VBOX_WITH_XPCOM
77
78struct MyPLEvent : public PLEvent
79{
80 MyPLEvent(Event *e) : event(e) {}
81 Event *event;
82};
83
84/* static */
85void *PR_CALLBACK com::EventQueue::plEventHandler(PLEvent *self)
86{
87 Event *ev = ((MyPLEvent *)self)->event;
88 if (ev)
89 ev->handler();
90 else
91 {
92 EventQueue *eq = (EventQueue *)self->owner;
93 Assert(eq);
94 eq->mInterrupted = true;
95 }
96 return NULL;
97}
98
99/* static */
100void PR_CALLBACK com::EventQueue::plEventDestructor(PLEvent *self)
101{
102 Event *ev = ((MyPLEvent *)self)->event;
103 if (ev)
104 delete ev;
105 delete self;
106}
107
108#endif // VBOX_WITH_XPCOM
109
110/**
111 * Constructs an event queue for the current thread.
112 *
113 * Currently, there can be only one event queue per thread, so if an event
114 * queue for the current thread already exists, this object is simply attached
115 * to the existing event queue.
116 */
117EventQueue::EventQueue()
118{
119#ifndef VBOX_WITH_XPCOM
120
121 mThreadId = GetCurrentThreadId();
122 // force the system to create the message queue for the current thread
123 MSG msg;
124 PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE);
125
126 if (!DuplicateHandle(GetCurrentProcess(),
127 GetCurrentThread(),
128 GetCurrentProcess(),
129 &mhThread,
130 0 /*dwDesiredAccess*/,
131 FALSE /*bInheritHandle*/,
132 DUPLICATE_SAME_ACCESS))
133 mhThread = INVALID_HANDLE_VALUE;
134
135#else // VBOX_WITH_XPCOM
136
137 mEQCreated = false;
138 mInterrupted = false;
139
140 // Here we reference the global nsIEventQueueService instance and hold it
141 // until we're destroyed. This is necessary to keep NS_ShutdownXPCOM() away
142 // from calling StopAcceptingEvents() on all event queues upon destruction of
143 // nsIEventQueueService, and makes sense when, for some reason, this happens
144 // *before* we're able to send a NULL event to stop our event handler thread
145 // when doing unexpected cleanup caused indirectly by NS_ShutdownXPCOM()
146 // that is performing a global cleanup of everything. A good example of such
147 // situation is when NS_ShutdownXPCOM() is called while the VirtualBox component
148 // is still alive (because it is still referenced): eventually, it results in
149 // a VirtualBox::uninit() call from where it is already not possible to post
150 // NULL to the event thread (because it stopped accepting events).
151
152 nsresult rc = NS_GetEventQueueService(getter_AddRefs(mEventQService));
153
154 if (NS_SUCCEEDED(rc))
155 {
156 rc = mEventQService->GetThreadEventQueue(NS_CURRENT_THREAD,
157 getter_AddRefs(mEventQ));
158 if (rc == NS_ERROR_NOT_AVAILABLE)
159 {
160 rc = mEventQService->CreateThreadEventQueue();
161 if (NS_SUCCEEDED(rc))
162 {
163 mEQCreated = true;
164 rc = mEventQService->GetThreadEventQueue(NS_CURRENT_THREAD,
165 getter_AddRefs(mEventQ));
166 }
167 }
168 }
169 AssertComRC(rc);
170
171#endif // VBOX_WITH_XPCOM
172}
173
174EventQueue::~EventQueue()
175{
176#ifndef VBOX_WITH_XPCOM
177 if (mhThread != INVALID_HANDLE_VALUE)
178 {
179 CloseHandle(mhThread);
180 mhThread = INVALID_HANDLE_VALUE;
181 }
182#else // VBOX_WITH_XPCOM
183 // process all pending events before destruction
184 if (mEventQ)
185 {
186 if (mEQCreated)
187 {
188 mEventQ->StopAcceptingEvents();
189 mEventQ->ProcessPendingEvents();
190 mEventQService->DestroyThreadEventQueue();
191 }
192 mEventQ = nsnull;
193 mEventQService = nsnull;
194 }
195#endif // VBOX_WITH_XPCOM
196}
197
198/**
199 * Initializes the main event queue instance.
200 * @returns VBox status code.
201 *
202 * @remarks If you're using the rest of the COM/XPCOM glue library,
203 * com::Initialize() will take care of initializing and uninitializing
204 * the EventQueue class. If you don't call com::Initialize, you must
205 * make sure to call this method on the same thread that did the
206 * XPCOM initialization or we'll end up using the wrong main queue.
207 */
208/* static */
209int EventQueue::init()
210{
211 Assert(sMainQueue == NULL);
212 Assert(RTThreadIsMain(RTThreadSelf()));
213 sMainQueue = new EventQueue();
214
215#ifdef VBOX_WITH_XPCOM
216 /* Check that it actually is the main event queue, i.e. that
217 we're called on the right thread. */
218 nsCOMPtr<nsIEventQueue> q;
219 nsresult rv = NS_GetMainEventQ(getter_AddRefs(q));
220 Assert(NS_SUCCEEDED(rv));
221 Assert(q == sMainQueue->mEventQ);
222
223 /* Check that it's a native queue. */
224 PRBool fIsNative = PR_FALSE;
225 rv = sMainQueue->mEventQ->IsQueueNative(&fIsNative);
226 Assert(NS_SUCCEEDED(rv) && fIsNative);
227#endif // VBOX_WITH_XPCOM
228
229 return VINF_SUCCESS;
230}
231
232/**
233 * Uninitialize the global resources (i.e. the main event queue instance).
234 * @returns VINF_SUCCESS
235 */
236/* static */
237int EventQueue::uninit()
238{
239 Assert(sMainQueue);
240 /* Must process all events to make sure that no NULL event is left
241 * after this point. It would need to modify the state of sMainQueue. */
242#ifdef RT_OS_DARWIN /* Do not process the native runloop, the toolkit may not be ready for it. */
243 sMainQueue->mEventQ->ProcessPendingEvents();
244#else
245 sMainQueue->processEventQueue(0);
246#endif
247 delete sMainQueue;
248 sMainQueue = NULL;
249 return VINF_SUCCESS;
250}
251
252/**
253 * Get main event queue instance.
254 *
255 * Depends on init() being called first.
256 */
257/* static */
258EventQueue* EventQueue::getMainEventQueue()
259{
260 return sMainQueue;
261}
262
263#ifdef VBOX_WITH_XPCOM
264# ifdef RT_OS_DARWIN
265/**
266 * Wait for events and process them (Darwin).
267 *
268 * @retval VINF_SUCCESS
269 * @retval VERR_TIMEOUT
270 * @retval VERR_INTERRUPTED
271 *
272 * @param cMsTimeout How long to wait, or RT_INDEFINITE_WAIT.
273 */
274static int waitForEventsOnDarwin(RTMSINTERVAL cMsTimeout)
275{
276 /*
277 * Wait for the requested time, if we get a hit we do a poll to process
278 * any other pending messages.
279 *
280 * Note! About 1.0e10: According to the sources anything above 3.1556952e+9
281 * means indefinite wait and 1.0e10 is what CFRunLoopRun() uses.
282 */
283 CFTimeInterval rdTimeout = cMsTimeout == RT_INDEFINITE_WAIT ? 1e10 : (double)cMsTimeout / 1000;
284 OSStatus orc = CFRunLoopRunInMode(kCFRunLoopDefaultMode, rdTimeout, true /*returnAfterSourceHandled*/);
285 if (orc == kCFRunLoopRunHandledSource)
286 {
287 OSStatus orc2 = CFRunLoopRunInMode(kCFRunLoopDefaultMode, 0.0, false /*returnAfterSourceHandled*/);
288 if ( orc2 == kCFRunLoopRunStopped
289 || orc2 == kCFRunLoopRunFinished)
290 orc = orc2;
291 }
292 if ( orc == 0 /*???*/
293 || orc == kCFRunLoopRunHandledSource)
294 return VINF_SUCCESS;
295 if ( orc == kCFRunLoopRunStopped
296 || orc == kCFRunLoopRunFinished)
297 return VERR_INTERRUPTED;
298 AssertMsg(orc == kCFRunLoopRunTimedOut, ("Unexpected status code from CFRunLoopRunInMode: %#x", orc));
299 return VERR_TIMEOUT;
300}
301# else // !RT_OS_DARWIN
302
303/**
304 * Wait for events (generic XPCOM).
305 *
306 * @retval VINF_SUCCESS
307 * @retval VERR_TIMEOUT
308 * @retval VINF_INTERRUPTED
309 * @retval VERR_INTERNAL_ERROR_4
310 *
311 * @param pQueue The queue to wait on.
312 * @param cMsTimeout How long to wait, or RT_INDEFINITE_WAIT.
313 */
314static int waitForEventsOnXPCOM(nsIEventQueue *pQueue, RTMSINTERVAL cMsTimeout)
315{
316 int fd = pQueue->GetEventQueueSelectFD();
317 fd_set fdsetR;
318 FD_ZERO(&fdsetR);
319 FD_SET(fd, &fdsetR);
320
321 fd_set fdsetE = fdsetR;
322
323 struct timeval tv = {0,0};
324 struct timeval *ptv;
325 if (cMsTimeout == RT_INDEFINITE_WAIT)
326 ptv = NULL;
327 else
328 {
329 tv.tv_sec = cMsTimeout / 1000;
330 tv.tv_usec = (cMsTimeout % 1000) * 1000;
331 ptv = &tv;
332 }
333
334 int rc = select(fd + 1, &fdsetR, NULL, &fdsetE, ptv);
335 if (rc > 0)
336 rc = VINF_SUCCESS;
337 else if (rc == 0)
338 rc = VERR_TIMEOUT;
339 else if (errno == EINTR)
340 rc = VINF_INTERRUPTED;
341 else
342 {
343 AssertMsgFailed(("rc=%d errno=%d\n", rc, errno));
344 rc = VERR_INTERNAL_ERROR_4;
345 }
346 return rc;
347}
348
349# endif // !RT_OS_DARWIN
350#endif // VBOX_WITH_XPCOM
351
352#ifndef VBOX_WITH_XPCOM
353
354/**
355 * Dispatch a message on Windows.
356 *
357 * This will pick out our events and handle them specially.
358 *
359 * @returns @a rc or VERR_INTERRUPTED (WM_QUIT or NULL msg).
360 * @param pMsg The message to dispatch.
361 * @param rc The current status code.
362 */
363/*static*/
364int EventQueue::dispatchMessageOnWindows(MSG const *pMsg, int rc)
365{
366 /*
367 * Check for and dispatch our events.
368 */
369 if ( pMsg->hwnd == NULL
370 && pMsg->message == WM_USER)
371 {
372 if (pMsg->lParam == EVENTQUEUE_WIN_LPARAM_MAGIC)
373 {
374 Event *pEvent = (Event *)pMsg->wParam;
375 if (pEvent)
376 {
377 pEvent->handler();
378 delete pEvent;
379 }
380 else
381 rc = VERR_INTERRUPTED;
382 return rc;
383 }
384 AssertMsgFailed(("lParam=%p wParam=%p\n", pMsg->lParam, pMsg->wParam));
385 }
386
387 /*
388 * Check for the quit message and dispatch the message the normal way.
389 */
390 if (pMsg->message == WM_QUIT)
391 rc = VERR_INTERRUPTED;
392 TranslateMessage(pMsg);
393 DispatchMessage(pMsg);
394
395 return rc;
396}
397
398
399/**
400 * Process pending events (Windows).
401 *
402 * @retval VINF_SUCCESS
403 * @retval VERR_TIMEOUT
404 * @retval VERR_INTERRUPTED.
405 */
406static int processPendingEvents(void)
407{
408 int rc = VERR_TIMEOUT;
409 MSG Msg;
410 if (PeekMessage(&Msg, NULL /*hWnd*/, 0 /*wMsgFilterMin*/, 0 /*wMsgFilterMax*/, PM_REMOVE))
411 {
412 rc = VINF_SUCCESS;
413 do
414 rc = EventQueue::dispatchMessageOnWindows(&Msg, rc);
415 while (PeekMessage(&Msg, NULL /*hWnd*/, 0 /*wMsgFilterMin*/, 0 /*wMsgFilterMax*/, PM_REMOVE));
416 }
417 return rc;
418}
419
420#else // VBOX_WITH_XPCOM
421
422/**
423 * Process pending XPCOM events.
424 * @param pQueue The queue to process events on.
425 * @retval VINF_SUCCESS
426 * @retval VERR_TIMEOUT
427 * @retval VERR_INTERRUPTED (darwin only)
428 * @retval VERR_INTERNAL_ERROR_2
429 */
430static int processPendingEvents(nsIEventQueue *pQueue)
431{
432 /* ProcessPendingEvents doesn't report back what it did, so check here. */
433 PRBool fHasEvents = PR_FALSE;
434 nsresult hr = pQueue->PendingEvents(&fHasEvents);
435 if (NS_FAILED(hr))
436 return VERR_INTERNAL_ERROR_2;
437
438 /* Process pending events. */
439 int rc = VINF_SUCCESS;
440 if (fHasEvents)
441 pQueue->ProcessPendingEvents();
442 else
443 rc = VERR_TIMEOUT;
444
445# ifdef RT_OS_DARWIN
446 /* Process pending native events. */
447 int rc2 = waitForEventsOnDarwin(0);
448 if (rc == VERR_TIMEOUT || rc2 == VERR_INTERRUPTED)
449 rc = rc2;
450# endif
451
452 return rc;
453}
454
455#endif // VBOX_WITH_XPCOM
456
457/**
458 * Process events pending on this event queue, and wait up to given timeout, if
459 * nothing is available.
460 *
461 * Must be called on same thread this event queue was created on.
462 *
463 * @param cMsTimeout The timeout specified as milliseconds. Use
464 * RT_INDEFINITE_WAIT to wait till an event is posted on the
465 * queue.
466 *
467 * @returns VBox status code
468 * @retval VINF_SUCCESS if one or more messages was processed.
469 * @retval VERR_TIMEOUT if cMsTimeout expired.
470 * @retval VERR_INVALID_CONTEXT if called on the wrong thread.
471 * @retval VERR_INTERRUPTED if interruptEventQueueProcessing was called.
472 * On Windows will also be returned when WM_QUIT is encountered.
473 * On Darwin this may also be returned when the native queue is
474 * stopped or destroyed/finished.
475 * @retval VINF_INTERRUPTED if the native system call was interrupted by a
476 * an asynchronous event delivery (signal) or just felt like returning
477 * out of bounds. On darwin it will also be returned if the queue is
478 * stopped.
479 */
480int EventQueue::processEventQueue(RTMSINTERVAL cMsTimeout)
481{
482 int rc;
483 CHECK_THREAD_RET(VERR_INVALID_CONTEXT);
484
485#ifdef VBOX_WITH_XPCOM
486 /*
487 * Process pending events, if none are available and we're not in a
488 * poll call, wait for some to appear. (We have to be a little bit
489 * careful after waiting for the events since Darwin will process
490 * them as part of the wait, while the XPCOM case will not.)
491 *
492 * Note! Unfortunately, WaitForEvent isn't interruptible with Ctrl-C,
493 * while select() is. So we cannot use it for indefinite waits.
494 */
495 rc = processPendingEvents(mEventQ);
496 if ( rc == VERR_TIMEOUT
497 && cMsTimeout > 0)
498 {
499# ifdef RT_OS_DARWIN
500 /** @todo check how Ctrl-C works on Darwin. */
501 rc = waitForEventsOnDarwin(cMsTimeout);
502 if (rc == VERR_TIMEOUT)
503 rc = processPendingEvents(mEventQ);
504# else // !RT_OS_DARWIN
505 rc = waitForEventsOnXPCOM(mEventQ, cMsTimeout);
506 if ( RT_SUCCESS(rc)
507 || rc == VERR_TIMEOUT)
508 rc = processPendingEvents(mEventQ);
509# endif // !RT_OS_DARWIN
510 }
511
512 if ( ( RT_SUCCESS(rc)
513 || rc == VERR_INTERRUPTED)
514 && mInterrupted)
515 {
516 mInterrupted = false;
517 rc = VERR_INTERRUPTED;
518 }
519
520#else // !VBOX_WITH_XPCOM
521 if (cMsTimeout == RT_INDEFINITE_WAIT)
522 {
523 BOOL fRet;
524 MSG Msg;
525 rc = VINF_SUCCESS;
526 while ( rc != VERR_INTERRUPTED
527 && (fRet = GetMessage(&Msg, NULL /*hWnd*/, WM_USER, WM_USER))
528 && fRet != -1)
529 rc = EventQueue::dispatchMessageOnWindows(&Msg, rc);
530 if (fRet == 0)
531 rc = VERR_INTERRUPTED;
532 else if (fRet == -1)
533 rc = RTErrConvertFromWin32(GetLastError());
534 }
535 else
536 {
537 rc = processPendingEvents();
538 if ( rc == VERR_TIMEOUT
539 && cMsTimeout != 0)
540 {
541 DWORD rcW = MsgWaitForMultipleObjects(1,
542 &mhThread,
543 TRUE /*fWaitAll*/,
544 cMsTimeout,
545 QS_ALLINPUT);
546 AssertMsgReturn(rcW == WAIT_TIMEOUT || rcW == WAIT_OBJECT_0,
547 ("%d\n", rcW),
548 VERR_INTERNAL_ERROR_4);
549 rc = processPendingEvents();
550 }
551 }
552#endif // !VBOX_WITH_XPCOM
553
554 Assert(rc != VERR_TIMEOUT || cMsTimeout != RT_INDEFINITE_WAIT);
555 return rc;
556}
557
558/**
559 * Interrupt thread waiting on event queue processing.
560 *
561 * Can be called on any thread.
562 *
563 * @returns VBox status code.
564 */
565int EventQueue::interruptEventQueueProcessing()
566{
567 /* Send a NULL event. This event will be picked up and handled specially
568 * both for XPCOM and Windows. It is the responsibility of the caller to
569 * take care of not running the loop again in a way which will hang. */
570 postEvent(NULL);
571 return VINF_SUCCESS;
572}
573
574/**
575 * Posts an event to this event loop asynchronously.
576 *
577 * @param event the event to post, must be allocated using |new|
578 * @return TRUE if successful and false otherwise
579 */
580BOOL EventQueue::postEvent(Event *event)
581{
582#ifndef VBOX_WITH_XPCOM
583 /* Note! The event == NULL case is duplicated in vboxapi/PlatformMSCOM::interruptWaitEvents(). */
584 return PostThreadMessage(mThreadId, WM_USER, (WPARAM)event, EVENTQUEUE_WIN_LPARAM_MAGIC);
585
586#else // VBOX_WITH_XPCOM
587
588 if (!mEventQ)
589 return FALSE;
590
591 MyPLEvent *ev = new MyPLEvent(event);
592 mEventQ->InitEvent(ev, this, com::EventQueue::plEventHandler,
593 com::EventQueue::plEventDestructor);
594 HRESULT rc = mEventQ->PostEvent(ev);
595 return NS_SUCCEEDED(rc);
596
597#endif // VBOX_WITH_XPCOM
598}
599
600
601/**
602 * Get select()'able selector for this event queue.
603 * This will return -1 on platforms and queue variants not supporting such
604 * functionality.
605 */
606int EventQueue::getSelectFD()
607{
608#ifdef VBOX_WITH_XPCOM
609 return mEventQ->GetEventQueueSelectFD();
610#else
611 return -1;
612#endif
613}
614
615}
616/* namespace com */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette