Changeset 3289 in vbox for trunk/src/VBox/Frontends/VBoxSDL
- Timestamp:
- Jun 26, 2007 1:56:26 PM (18 years ago)
- svn:sync-xref-src-repo-rev:
- 22390
- Location:
- trunk/src/VBox/Frontends/VBoxSDL
- Files:
-
- 5 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Frontends/VBoxSDL/Framebuffer.cpp
r3153 r3289 387 387 #ifdef __LINUX__ 388 388 /* 389 * SDL does not allow us to make this call from any other 390 * th read. So we have to send an event to the main SDL391 * t hread and process it there. For sake of simplicity, we encode392 * all information in the event parameters.389 * SDL does not allow us to make this call from any other thread than 390 * the main SDL thread (which initialized the video mode). So we have 391 * to send an event to the main SDL thread and process it there. For 392 * sake of simplicity, we encode all information in the event parameters. 393 393 */ 394 394 SDL_Event event; … … 398 398 event.user.data1 = (void*)(x << 16 | y); 399 399 event.user.data2 = (void*)(w << 16 | h); 400 int rc = SDL_PushEvent(&event); 401 NOREF(rc); 402 AssertMsg(!rc, ("SDL_PushEvent returned SDL error '%s'\n", SDL_GetError())); 403 /* in order to not flood the SDL event queue, yield the CPU */ 404 RTThreadYield(); 400 PushNotifyUpdateEvent(&event); 405 401 #else /* !__LINUX__ */ 406 402 update(x, y, w, h, true /* fGuestRelative */); … … 438 434 439 435 /* 440 * SDL does not allow us to make this call from any other 441 * th read. So we have to send an event to the main SDL442 * thread and tell VBox to wait.436 * SDL does not allow us to make this call from any other thread than 437 * the main thread (the one which initialized the video mode). So we 438 * have to send an event to the main SDL thread and tell VBox to wait. 443 439 */ 444 440 if (!finished) … … 447 443 return E_FAIL; 448 444 } 449 mGuestXRes = w;450 mGuestYRes = h;445 mGuestXRes = w; 446 mGuestYRes = h; 451 447 mPixelFormat = pixelFormat; 452 mPtrVRAM = vram;453 mLineSize = lineSize;448 mPtrVRAM = vram; 449 mLineSize = lineSize; 454 450 455 451 SDL_Event event; 456 452 event.type = SDL_USEREVENT; 457 453 event.user.type = SDL_USER_EVENT_RESIZE; 458 int rc = SDL_PushEvent(&event); 459 NOREF(rc);460 AssertMsg(!rc, ("SDL_PushEvent returned SDL error '%s'\n", SDL_GetError()));454 455 /* Try multiple times if necessary */ 456 PushSDLEventForSure(&event); 461 457 462 458 /* we want this request to be processed quickly, so yield the CPU */ -
trunk/src/VBox/Frontends/VBoxSDL/Helper.cpp
r2981 r3289 24 24 #include <VBox/err.h> 25 25 #include <VBox/log.h> 26 #include <iprt/asm.h> 26 27 #include <iprt/assert.h> 27 28 #include <iprt/thread.h> … … 39 40 40 41 /** global flag indicating that the event queue thread should terminate */ 41 bool volatile g_fTerminateXPCOMQueueThread = false; 42 static bool volatile g_fTerminateXPCOMQueueThread = false; 43 44 /** How many XPCOM user events are on air. Only allow one pending event to prevent 45 * an overflow of the SDL event queue. */ 46 static volatile int32_t g_s32XPCOMEventsPending; 42 47 43 48 /** Semaphore the XPCOM event thread will sleep on while it waits for the main thread to process pending requests. */ … … 72 77 if ((n > 0) && !g_fTerminateXPCOMQueueThread) 73 78 { 74 /* 75 * Post the event and wait for it to be processed. If we don't wait, 76 * we'll flood the queue on SMP systems and when the main thread is busy. 77 * In the event of a push error, we'll yield the timeslice and retry. 79 /* 80 * Wait until all XPCOM events are processed. 1s just for sanity. 78 81 */ 79 SDL_Event event = {0}; 80 event.type = SDL_USEREVENT; 81 event.user.type = SDL_USER_EVENT_XPCOM_EVENTQUEUE; 82 rc = SDL_PushEvent(&event); 83 if (!rc) 84 { 85 RTSemEventWait(g_EventSemXPCOMQueueThread, 100); 86 cErrors = 0; 82 int iWait = 1000; 83 /* 84 * Don't post an event if there is a pending XPCOM event to prevent an 85 * overflow of the SDL event queue. 86 */ 87 if (g_s32XPCOMEventsPending < 1) 88 { 89 /* 90 * Post the event and wait for it to be processed. If we don't wait, 91 * we'll flood the queue on SMP systems and when the main thread is busy. 92 * In the event of a push error, we'll yield the timeslice and retry. 93 */ 94 SDL_Event event = {0}; 95 event.type = SDL_USEREVENT; 96 event.user.type = SDL_USER_EVENT_XPCOM_EVENTQUEUE; 97 rc = SDL_PushEvent(&event); 98 if (!rc) 99 { 100 /* success */ 101 ASMAtomicIncS32(&g_s32XPCOMEventsPending); 102 cErrors = 0; 103 } 104 else 105 { 106 /* failure */ 107 cErrors++; 108 if (!RTThreadYield()) 109 RTThreadSleep(2); 110 iWait = (cErrors >= 10) ? RT_MIN(cErrors - 8, 50) : 0; 111 } 87 112 } 88 113 else 89 { 90 cErrors++; 91 if (!RTThreadYield()) 92 RTThreadSleep(2); 93 if (cErrors >= 10) 94 RTSemEventWait(g_EventSemXPCOMQueueThread, RT_MIN(cErrors - 8, 50)); 95 } 114 Log2(("not enqueueing SDL XPCOM event (%d)\n", g_s32XPCOMEventsPending)); 115 116 if (iWait) 117 RTSemEventWait(g_EventSemXPCOMQueueThread, iWait); 96 118 } 97 119 } while (!g_fTerminateXPCOMQueueThread); … … 115 137 AssertRC(rc); 116 138 return rc; 139 } 140 141 /* 142 * Notify the XPCOM tread that we consumed an XPCOM event. 143 */ 144 void consumedXPCOMUserEvent(void) 145 { 146 ASMAtomicDecS32(&g_s32XPCOMEventsPending); 117 147 } 118 148 … … 139 169 } 140 170 141 142 143 171 #endif /* USE_XPCOM_QUEUE_THREAD */ 144 -
trunk/src/VBox/Frontends/VBoxSDL/Helper.h
r2981 r3289 37 37 int startXPCOMEventQueueThread(int eqFD); 38 38 39 /* 40 * Notify the XPCOM thread that we consumed an XPCOM event 41 */ 42 void consumedXPCOMUserEvent(void); 43 39 44 /** 40 45 * Signal to the XPCOM even queue thread that it should select for more events. -
trunk/src/VBox/Frontends/VBoxSDL/VBoxSDL.cpp
r3176 r3289 140 140 static Uint32 StartupTimer(Uint32 interval, void *param); 141 141 static Uint32 ResizeTimer(Uint32 interval, void *param); 142 static int WaitSDLEvent(SDL_Event *event); 142 143 143 144 … … 203 204 #endif 204 205 206 static RTSEMEVENT g_EventSemSDLEvents; 207 static volatile int32_t g_cNotifyUpdateEventsPending; 208 205 209 /** 206 210 * Callback handler for VirtualBox events … … 297 301 event.type = SDL_USEREVENT; 298 302 event.user.type = SDL_USER_EVENT_SECURELABEL_UPDATE; 299 int rc = SDL_PushEvent(&event); 300 NOREF(rc); 301 AssertMsg(!rc, ("SDL_PushEvent returned with SDL error '%s'\n", SDL_GetError())); 303 PushSDLEventForSure(&event); 302 304 } 303 305 } … … 414 416 event.user.data1 = data; 415 417 416 int rc = SDL_PushEvent (&event); 417 AssertMsg(!rc, ("SDL_PushEvent returned with SDL error '%s'\n", SDL_GetError())); 418 int rc = PushSDLEventForSure (&event); 418 419 if (rc) 419 420 delete data; … … 432 433 event.user.type = SDL_USER_EVENT_GUEST_CAP_CHANGED; 433 434 434 int rc = SDL_PushEvent (&event); 435 NOREF(rc); 436 AssertMsg(!rc, ("SDL_PushEvent returned with SDL error '%s'\n", SDL_GetError())); 435 PushSDLEventForSure (&event); 437 436 return S_OK; 438 437 } … … 465 464 } 466 465 467 int rc = SDL_PushEvent(&event); 468 NOREF(rc); 469 AssertMsg(!rc, ("SDL_PushEvent returned with SDL error '%s'\n", SDL_GetError())); 466 PushSDLEventForSure(&event); 470 467 return S_OK; 471 468 } … … 1290 1287 } 1291 1288 1289 /* create SDL event semaphore */ 1290 rc = RTSemEventCreate(&g_EventSemSDLEvents); 1291 AssertReleaseRC(rc); 1292 1292 1293 rc = virtualBox->OpenSession(session, uuid); 1293 1294 if (FAILED(rc)) … … 1880 1881 * Wait for SDL events. 1881 1882 */ 1882 if ( SDL_WaitEvent(&event))1883 if (WaitSDLEvent(&event)) 1883 1884 { 1884 1885 switch (event.type) … … 1915 1916 { 1916 1917 LogFlow(("SDL_USER_EVENT_XPCOM_EVENTQUEUE: processing XPCOM event queue...\n")); 1918 consumedXPCOMUserEvent(); 1917 1919 eventQ->ProcessPendingEvents(); 1918 1920 signalXPCOMEventQueueThread(); … … 2004 2006 #endif 2005 2007 LogFlow(("VBoxSDL: Entering big event loop\n")); 2006 while ( SDL_WaitEvent(&event))2008 while (WaitSDLEvent(&event)) 2007 2009 { 2008 2010 switch (event.type) … … 2254 2256 * Decode event parameters. 2255 2257 */ 2258 ASMAtomicDecS32(&g_cNotifyUpdateEventsPending); 2256 2259 #define DECODEX(event) ((intptr_t)(event).user.data1 >> 16) 2257 2260 #define DECODEY(event) ((intptr_t)(event).user.data1 & 0xFFFF) … … 2678 2681 case SDLK_CLEAR: return 0x; 2679 2682 case SDLK_KP_EQUALS: return 0x; 2680 case SDLK_COMPOSE: 2683 case SDLK_COMPOSE: return 0x; 2681 2684 case SDLK_HELP: return 0x; 2682 2685 case SDLK_BREAK: return 0x; 2683 case SDLK_POWER: 2684 case SDLK_EURO: 2685 case SDLK_UNDO: 2686 case SDLK_POWER: return 0x; 2687 case SDLK_EURO: return 0x; 2688 case SDLK_UNDO: return 0x; 2686 2689 #endif 2687 2690 default: … … 3537 3540 */ 3538 3541 SDL_Event event; 3539 if ( SDL_WaitEvent(&event))3542 if (WaitSDLEvent(&event)) 3540 3543 { 3541 3544 switch (event.type) … … 4272 4275 event.user.type = SDL_USER_EVENT_TIMER; 4273 4276 SDL_PushEvent(&event); 4277 RTSemEventSignal(g_EventSemSDLEvents); 4274 4278 return interval; 4275 4279 } … … 4284 4288 event.type = SDL_USEREVENT; 4285 4289 event.user.type = SDL_USER_EVENT_WINDOW_RESIZE_DONE; 4286 SDL_PushEvent(&event);4290 PushSDLEventForSure(&event); 4287 4291 /* one-shot */ 4288 4292 return 0; 4289 4293 } 4294 4295 /** 4296 * Wait for the next SDL event. Don't use SDL_WaitEvent since this function 4297 * calls SDL_Delay(10) if the event queue is empty. 4298 */ 4299 static int WaitSDLEvent(SDL_Event *event) 4300 { 4301 for (;;) 4302 { 4303 int rc = SDL_PollEvent (event); 4304 if (rc == 1) 4305 return 1; 4306 /* Immediately wake up if new SDL events are available. This does not 4307 * work for internal SDL events. Don't wait more than 10ms. */ 4308 RTSemEventWait(g_EventSemSDLEvents, 10); 4309 } 4310 } 4311 4312 /** 4313 * Ensure that an SDL event is really enqueued. Try multiple times if necessary. 4314 */ 4315 int PushSDLEventForSure(SDL_Event *event) 4316 { 4317 int ntries = 10; 4318 for (; ntries > 0; ntries--) 4319 { 4320 int rc = SDL_PushEvent(event); 4321 RTSemEventSignal(g_EventSemSDLEvents); 4322 if (rc == 0) 4323 return 0; 4324 Log(("PushSDLEventForSure: waiting for 2ms\n")); 4325 RTThreadSleep(2); 4326 } 4327 LogRel(("WARNING: Failed to enqueue SDL event %d.%d!\n", 4328 event->type, event->type == SDL_USEREVENT ? event->user.type : 0)); 4329 return -1; 4330 } 4331 4332 #ifdef __LINUX__ 4333 /** 4334 * Special SDL_PushEvent function for NotifyUpdate events. These events may occur in bursts 4335 * so make sure they don't flood the SDL event queue. 4336 */ 4337 void PushNotifyUpdateEvent(SDL_Event *event) 4338 { 4339 int rc = SDL_PushEvent(event); 4340 RTSemEventSignal(g_EventSemSDLEvents); 4341 AssertMsg(!rc, ("SDL_PushEvent returned SDL error\n")); 4342 /* A global counter is faster than SDL_PeepEvents() */ 4343 if (!rc) 4344 ASMAtomicIncS32(&g_cNotifyUpdateEventsPending); 4345 /* In order to not flood the SDL event queue, yield the CPU or (if there are already many 4346 * events queued) even sleep */ 4347 if (g_cNotifyUpdateEventsPending > 96) 4348 { 4349 /* Too many NotifyUpdate events, sleep for a small amount to give the main thread time 4350 * to handle these events. The SDL queue can hold up to 128 events. */ 4351 Log(("PushNotifyUpdateEvent: Sleep 1ms\n")); 4352 RTThreadSleep(1); 4353 } 4354 else 4355 RTThreadYield(); 4356 } 4357 #endif /* __LINUX__ */ -
trunk/src/VBox/Frontends/VBoxSDL/VBoxSDL.h
r2981 r3289 72 72 #endif /* VBOX_WIN32_UI */ 73 73 74 #ifdef __LINUX__ 75 void PushNotifyUpdateEvent(SDL_Event *event); 76 #endif 77 int PushSDLEventForSure(SDL_Event *event); 78 74 79 #endif // __H_VBOXSDL
Note:
See TracChangeset
for help on using the changeset viewer.