VirtualBox

Changeset 3289 in vbox for trunk/src/VBox/Frontends/VBoxSDL


Ignore:
Timestamp:
Jun 26, 2007 1:56:26 PM (18 years ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
22390
Message:

FE/SDL: Improved SDL event handling. Don't enqueue further XPCOM events if there is already an XPCOM event on air. Immediately signal the main thread when a new SDL event was enqueued.

Location:
trunk/src/VBox/Frontends/VBoxSDL
Files:
5 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/Frontends/VBoxSDL/Framebuffer.cpp

    r3153 r3289  
    387387#ifdef __LINUX__
    388388    /*
    389      * SDL does not allow us to make this call from any other
    390      * thread. So we have to send an event to the main SDL
    391      * thread and process it there. For sake of simplicity, we encode
    392      * 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.
    393393     */
    394394    SDL_Event event;
     
    398398    event.user.data1 = (void*)(x << 16 | y);
    399399    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);
    405401#else /* !__LINUX__ */
    406402    update(x, y, w, h, true /* fGuestRelative */);
     
    438434
    439435    /*
    440      * SDL does not allow us to make this call from any other
    441      * thread. So we have to send an event to the main SDL
    442      * 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.
    443439     */
    444440    if (!finished)
     
    447443        return E_FAIL;
    448444    }
    449     mGuestXRes = w;
    450     mGuestYRes = h;
     445    mGuestXRes   = w;
     446    mGuestYRes   = h;
    451447    mPixelFormat = pixelFormat;
    452     mPtrVRAM = vram;
    453     mLineSize = lineSize;
     448    mPtrVRAM     = vram;
     449    mLineSize    = lineSize;
    454450
    455451    SDL_Event event;
    456452    event.type       = SDL_USEREVENT;
    457453    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);
    461457
    462458    /* we want this request to be processed quickly, so yield the CPU */
  • trunk/src/VBox/Frontends/VBoxSDL/Helper.cpp

    r2981 r3289  
    2424#include <VBox/err.h>
    2525#include <VBox/log.h>
     26#include <iprt/asm.h>
    2627#include <iprt/assert.h>
    2728#include <iprt/thread.h>
     
    3940
    4041/** global flag indicating that the event queue thread should terminate */
    41 bool volatile   g_fTerminateXPCOMQueueThread = false;
     42static 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. */
     46static volatile int32_t g_s32XPCOMEventsPending;
    4247
    4348/** Semaphore the XPCOM event thread will sleep on while it waits for the main thread to process pending requests. */
     
    7277        if ((n > 0) && !g_fTerminateXPCOMQueueThread)
    7378        {
    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.
    7881             */
    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                }
    87112            }
    88113            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);
    96118        }
    97119    } while (!g_fTerminateXPCOMQueueThread);
     
    115137    AssertRC(rc);
    116138    return rc;
     139}
     140
     141/*
     142 * Notify the XPCOM tread that we consumed an XPCOM event.
     143 */
     144void consumedXPCOMUserEvent(void)
     145{
     146    ASMAtomicDecS32(&g_s32XPCOMEventsPending);
    117147}
    118148
     
    139169}
    140170
    141 
    142 
    143171#endif /* USE_XPCOM_QUEUE_THREAD */
    144 
  • trunk/src/VBox/Frontends/VBoxSDL/Helper.h

    r2981 r3289  
    3737int startXPCOMEventQueueThread(int eqFD);
    3838
     39/*
     40 * Notify the XPCOM thread that we consumed an XPCOM event
     41 */
     42void consumedXPCOMUserEvent(void);
     43
    3944/**
    4045 * Signal to the XPCOM even queue thread that it should select for more events.
  • trunk/src/VBox/Frontends/VBoxSDL/VBoxSDL.cpp

    r3176 r3289  
    140140static Uint32  StartupTimer(Uint32 interval, void *param);
    141141static Uint32  ResizeTimer(Uint32 interval, void *param);
     142static int     WaitSDLEvent(SDL_Event *event);
    142143
    143144
     
    203204#endif
    204205
     206static RTSEMEVENT g_EventSemSDLEvents;
     207static volatile int32_t g_cNotifyUpdateEventsPending;
     208
    205209/**
    206210 * Callback handler for VirtualBox events
     
    297301                    event.type       = SDL_USEREVENT;
    298302                    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);
    302304                }
    303305            }
     
    414416        event.user.data1 = data;
    415417
    416         int rc = SDL_PushEvent (&event);
    417         AssertMsg(!rc, ("SDL_PushEvent returned with SDL error '%s'\n", SDL_GetError()));
     418        int rc = PushSDLEventForSure (&event);
    418419        if (rc)
    419420            delete data;
     
    432433        event.user.type = SDL_USER_EVENT_GUEST_CAP_CHANGED;
    433434
    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);
    437436        return S_OK;
    438437    }
     
    465464        }
    466465
    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);
    470467        return S_OK;
    471468    }
     
    12901287    }
    12911288
     1289    /* create SDL event semaphore */
     1290    rc = RTSemEventCreate(&g_EventSemSDLEvents);
     1291    AssertReleaseRC(rc);
     1292
    12921293    rc = virtualBox->OpenSession(session, uuid);
    12931294    if (FAILED(rc))
     
    18801881             * Wait for SDL events.
    18811882             */
    1882             if (SDL_WaitEvent(&event))
     1883            if (WaitSDLEvent(&event))
    18831884            {
    18841885                switch (event.type)
     
    19151916                    {
    19161917                        LogFlow(("SDL_USER_EVENT_XPCOM_EVENTQUEUE: processing XPCOM event queue...\n"));
     1918                        consumedXPCOMUserEvent();
    19171919                        eventQ->ProcessPendingEvents();
    19181920                        signalXPCOMEventQueueThread();
     
    20042006#endif
    20052007    LogFlow(("VBoxSDL: Entering big event loop\n"));
    2006     while (SDL_WaitEvent(&event))
     2008    while (WaitSDLEvent(&event))
    20072009    {
    20082010        switch (event.type)
     
    22542256                 * Decode event parameters.
    22552257                 */
     2258                ASMAtomicDecS32(&g_cNotifyUpdateEventsPending);
    22562259                #define DECODEX(event) ((intptr_t)(event).user.data1 >> 16)
    22572260                #define DECODEY(event) ((intptr_t)(event).user.data1 & 0xFFFF)
     
    26782681        case SDLK_CLEAR:            return 0x;
    26792682        case SDLK_KP_EQUALS:        return 0x;
    2680         case SDLK_COMPOSE:          return 0x;
     2683        case SDLK_COMPOSE:          return 0x;
    26812684        case SDLK_HELP:             return 0x;
    26822685        case SDLK_BREAK:            return 0x;
    2683         case SDLK_POWER:            return 0x;
    2684         case SDLK_EURO:             return 0x;
    2685         case SDLK_UNDO:             return 0x;
     2686        case SDLK_POWER:            return 0x;
     2687        case SDLK_EURO:             return 0x;
     2688        case SDLK_UNDO:             return 0x;
    26862689#endif
    26872690        default:
     
    35373540         */
    35383541        SDL_Event event;
    3539         if (SDL_WaitEvent(&event))
     3542        if (WaitSDLEvent(&event))
    35403543        {
    35413544            switch (event.type)
     
    42724275    event.user.type = SDL_USER_EVENT_TIMER;
    42734276    SDL_PushEvent(&event);
     4277    RTSemEventSignal(g_EventSemSDLEvents);
    42744278    return interval;
    42754279}
     
    42844288    event.type      = SDL_USEREVENT;
    42854289    event.user.type = SDL_USER_EVENT_WINDOW_RESIZE_DONE;
    4286     SDL_PushEvent(&event);
     4290    PushSDLEventForSure(&event);
    42874291    /* one-shot */
    42884292    return 0;
    42894293}
     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 */
     4299static 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 */
     4315int 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 */
     4337void 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  
    7272#endif /* VBOX_WIN32_UI */
    7373
     74#ifdef __LINUX__
     75void PushNotifyUpdateEvent(SDL_Event *event);
     76#endif
     77int  PushSDLEventForSure(SDL_Event *event);
     78
    7479#endif // __H_VBOXSDL
Note: See TracChangeset for help on using the changeset viewer.

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