VirtualBox

Changeset 29220 in vbox for trunk/src/VBox/HostServices


Ignore:
Timestamp:
May 7, 2010 3:09:53 PM (15 years ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
61300
Message:

Guest Control/Main/HostService: Deal with stale clients, avoid deadlocks when doing massive output traffic.

File:
1 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/HostServices/GuestControl/service.cpp

    r29003 r29220  
    2828
    2929#include <VBox/log.h>
    30 #include <iprt/asm.h>
     30#include <iprt/asm.h> /* For ASMBreakpoint(). */
    3131#include <iprt/assert.h>
    3232#include <iprt/cpp/autores.h>
     
    6565struct GuestCall
    6666{
     67    /** Client ID; a client can have multiple handles! */
     68    uint32_t mClientID;
    6769    /** The call handle */
    6870    VBOXHGCMCALLHANDLE mHandle;
     
    7375
    7476    /** The standard constructor */
    75     GuestCall() : mHandle(0), mParms(NULL), mNumParms(0) {}
     77    GuestCall() : mClientID(0), mHandle(0), mParms(NULL), mNumParms(0) {}
    7678    /** The normal contructor */
    77     GuestCall(VBOXHGCMCALLHANDLE aHandle, VBOXHGCMSVCPARM aParms[], int cParms)
    78               : mHandle(aHandle), mParms(aParms), mNumParms(cParms) {}
     79    GuestCall(uint32_t aClientID, VBOXHGCMCALLHANDLE aHandle,
     80              VBOXHGCMSVCPARM aParms[], uint32_t cParms)
     81              : mClientID(aClientID), mHandle(aHandle), mParms(aParms), mNumParms(cParms) {}
    7982};
    8083/** The guest call list type */
    81 typedef std::list <GuestCall> CallList;
     84typedef std::list< GuestCall > CallList;
     85typedef std::list< GuestCall >::iterator CallListIter;
     86typedef std::list< GuestCall >::const_iterator CallListIterConst;
    8287
    8388/**
     
    9196    /** HGCM helper functions. */
    9297    PVBOXHGCMSVCHELPERS mpHelpers;
    93     /** @todo we should have classes for thread and request handler thread */
    94     /** Queue of outstanding property change notifications */
    95     RTREQQUEUE *mReqQueue;
    96     /** Request that we've left pending in a call to flushNotifications. */
    97     PRTREQ mPendingDummyReq;
    98     /** Thread for processing the request queue */
    99     RTTHREAD mReqThread;
    100     /** Tell the thread that it should exit */
    101     bool volatile mfExitThread;
    10298    /** Callback function supplied by the host for notification of updates
    10399     * to properties */
     
    113109    explicit Service(PVBOXHGCMSVCHELPERS pHelpers)
    114110        : mpHelpers(pHelpers)
    115         , mPendingDummyReq(NULL)
    116         , mfExitThread(false)
    117111        , mpfnHostCallback(NULL)
    118112        , mpvHostData(NULL)
    119113    {
    120         int rc = RTReqCreateQueue(&mReqQueue);
    121 #ifndef VBOX_GUEST_CTRL_TEST_NOTHREAD
    122         if (RT_SUCCESS(rc))
    123             rc = RTThreadCreate(&mReqThread, reqThreadFn, this, 0,
    124                                 RTTHREADTYPE_MSG_PUMP, RTTHREADFLAGS_WAITABLE,
    125                                 "GuestCtrlReq");
    126 #endif
    127         if (RT_FAILURE(rc))
    128             throw rc;
    129114    }
    130115
     
    231216    int paramBufferAssign(PVBOXGUESTCTRPARAMBUFFER pBuf, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
    232217    int prepareExecute(uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
    233     static DECLCALLBACK(int) reqThreadFn(RTTHREAD ThreadSelf, void *pvUser);
    234218    int clientConnect(uint32_t u32ClientID, void *pvClient);
    235219    int clientDisconnect(uint32_t u32ClientID, void *pvClient);
    236220    int sendHostCmdToGuest(HostCmd *pCmd, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
    237     int retrieveNextHostCmd(VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
     221    int retrieveNextHostCmd(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
    238222    int notifyHost(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
    239223    int processHostCmd(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
     
    246230
    247231
    248 /**
    249  * Thread function for processing the request queue
    250  * @copydoc FNRTTHREAD
    251  */
    252 /* static */
    253 DECLCALLBACK(int) Service::reqThreadFn(RTTHREAD ThreadSelf, void *pvUser)
    254 {
    255     SELF *pSelf = reinterpret_cast<SELF *>(pvUser);
    256     while (!pSelf->mfExitThread)
    257         RTReqProcess(pSelf->mReqQueue, RT_INDEFINITE_WAIT);
    258     return VINF_SUCCESS;
    259 }
    260 
    261232/** @todo Write some nice doc headers! */
    262233/* Stores a HGCM request in an internal buffer (pEx). Needs to be freed later using execBufferFree(). */
     
    265236    AssertPtr(pBuf);
    266237    int rc = VINF_SUCCESS;
     238
     239    /* Paranoia. */
     240    if (cParms > 256)
     241        cParms = 256;
    267242
    268243    /*
     
    392367{
    393368    LogFlowFunc(("Client (%ld) disconnected\n", u32ClientID));
     369    /*
     370     * Throw out all stale clients.
     371     */
     372    CallListIter it = mClientList.begin();
     373    while (it != mClientList.end())
     374    {
     375        if (it->mClientID == u32ClientID)
     376            it = mClientList.erase(it);
     377        else
     378            it++;
     379    }
    394380    return VINF_SUCCESS;
    395381}
     
    424410 * defer the guest call until we have something from the host.
    425411 */
    426 int Service::retrieveNextHostCmd(VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
     412int Service::retrieveNextHostCmd(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle,
     413                                 uint32_t cParms, VBOXHGCMSVCPARM paParms[])
    427414{
    428415    int rc = VINF_SUCCESS;
     
    435422    if (mHostCmds.empty()) /* If command list is empty, defer ... */
    436423    {
    437         mClientList.push_back(GuestCall(callHandle, paParms, cParms));
     424        mClientList.push_back(GuestCall(u32ClientID, callHandle, paParms, cParms));
    438425        rc = VINF_HGCM_ASYNC_EXECUTE;
    439426    }
     
    535522        if (!fProcessed)
    536523        {
    537             mHostCmds.push_back(newCmd);
    538    
     524            mHostCmds.push_back(newCmd);   
     525#if 0
    539526            /* Limit list size by deleting oldest element. */
    540527            if (mHostCmds.size() > 256) /** @todo Use a define! */
    541528                mHostCmds.pop_front();
     529#endif
    542530        }
    543531    }
     
    568556            case GUEST_GET_HOST_MSG:
    569557                LogFlowFunc(("GUEST_GET_HOST_MSG\n"));
    570                 rc = retrieveNextHostCmd(callHandle, cParms, paParms);
     558                rc = retrieveNextHostCmd(u32ClientID, callHandle, cParms, paParms);
    571559                break;
    572560
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