/* $Id: VBoxServiceInternal.h 39906 2012-01-30 12:50:03Z vboxsync $ */ /** @file * VBoxService - Guest Additions Services. */ /* * Copyright (C) 2007-2011 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ #ifndef ___VBoxServiceInternal_h #define ___VBoxServiceInternal_h #include #ifdef RT_OS_WINDOWS # include # include /* Needed for file version information. */ #endif #include #include #include #include /** * A service descriptor. */ typedef struct { /** The short service name. */ const char *pszName; /** The longer service name. */ const char *pszDescription; /** The usage options stuff for the --help screen. */ const char *pszUsage; /** The option descriptions for the --help screen. */ const char *pszOptions; /** * Called before parsing arguments. * @returns VBox status code. */ DECLCALLBACKMEMBER(int, pfnPreInit)(void); /** * Tries to parse the given command line option. * * @returns 0 if we parsed, -1 if it didn't and anything else means exit. * @param ppszShort If not NULL it points to the short option iterator. a short argument. * If NULL examine argv[*pi]. * @param argc The argument count. * @param argv The argument vector. * @param pi The argument vector index. Update if any value(s) are eaten. */ DECLCALLBACKMEMBER(int, pfnOption)(const char **ppszShort, int argc, char **argv, int *pi); /** * Called before parsing arguments. * @returns VBox status code. */ DECLCALLBACKMEMBER(int, pfnInit)(void); /** Called from the worker thread. * * @returns VBox status code. * @retval VINF_SUCCESS if exitting because *pfTerminate was set. * @param pfTerminate Pointer to a per service termination flag to check * before and after blocking. */ DECLCALLBACKMEMBER(int, pfnWorker)(bool volatile *pfTerminate); /** * Stop an service. */ DECLCALLBACKMEMBER(void, pfnStop)(void); /** * Does termination cleanups. * * @remarks This may be called even if pfnInit hasn't been called! */ DECLCALLBACKMEMBER(void, pfnTerm)(void); } VBOXSERVICE; /** Pointer to a VBOXSERVICE. */ typedef VBOXSERVICE *PVBOXSERVICE; /** Pointer to a const VBOXSERVICE. */ typedef VBOXSERVICE const *PCVBOXSERVICE; /** The service name. * @note Used on windows to name the service as well as the global mutex. */ #define VBOXSERVICE_NAME "VBoxService" #ifdef RT_OS_WINDOWS /** The friendly service name. */ # define VBOXSERVICE_FRIENDLY_NAME "VirtualBox Guest Additions Service" /** The service description (only W2K+ atm) */ # define VBOXSERVICE_DESCRIPTION "Manages VM runtime information, time synchronization, guest control execution and miscellaneous utilities for guest operating systems." /** The following constant may be defined by including NtStatus.h. */ # define STATUS_SUCCESS ((NTSTATUS)0x00000000L) #endif /* RT_OS_WINDOWS */ #ifdef VBOX_WITH_GUEST_CONTROL /** * Pipe IDs for handling the guest process poll set. */ typedef enum VBOXSERVICECTRLPIPEID { VBOXSERVICECTRLPIPEID_UNKNOWN = 0, VBOXSERVICECTRLPIPEID_STDIN = 10, VBOXSERVICECTRLPIPEID_STDIN_WRITABLE = 11, /** Pipe for reading from guest process' stdout. */ VBOXSERVICECTRLPIPEID_STDOUT = 40, /** Pipe for reading from guest process' stderr. */ VBOXSERVICECTRLPIPEID_STDERR = 50, /** Notification pipe for waking up the guest process * control thread. */ VBOXSERVICECTRLPIPEID_IPC_NOTIFY = 100 } VBOXSERVICECTRLPIPEID; /** * Request types to perform on a started guest process. */ typedef enum VBOXSERVICECTRLREQUESTTYPE { /** Unknown request. */ VBOXSERVICECTRLREQUEST_UNKNOWN = 0, /** Main control thread asked used to quit. */ VBOXSERVICECTRLREQUEST_QUIT = 1, /** Performs reading from stdout. */ VBOXSERVICECTRLREQUEST_STDOUT_READ = 50, /** Performs reading from stderr. */ VBOXSERVICECTRLREQUEST_STDERR_READ = 60, /** Performs writing to stdin. */ VBOXSERVICECTRLREQUEST_STDIN_WRITE = 70, /** Same as VBOXSERVICECTRLREQUEST_STDIN_WRITE, but * marks the end of input. */ VBOXSERVICECTRLREQUEST_STDIN_WRITE_EOF = 71, /** Kill/terminate process. * @todo Implement this! */ VBOXSERVICECTRLREQUEST_KILL = 90, /** Gently ask process to terminate. * @todo Implement this! */ VBOXSERVICECTRLREQUEST_HANGUP = 91, /** Ask the process in which status it * currently is. * @todo Implement this! */ VBOXSERVICECTRLREQUEST_STATUS = 100 } VBOXSERVICECTRLREQUESTTYPE; /** * Thread list types. */ typedef enum VBOXSERVICECTRLTHREADLISTTYPE { /** Unknown list -- uncool to use. */ VBOXSERVICECTRLTHREADLIST_UNKNOWN = 0, /** Stopped list: Here all guest threads end up * when they reached the stopped state and can * be shut down / free'd safely. */ VBOXSERVICECTRLTHREADLIST_STOPPED = 1, /** * Started list: Here all threads are registered * when they're up and running (that is, accepting * commands). */ VBOXSERVICECTRLTHREADLIST_RUNNING = 2 } VBOXSERVICECTRLTHREADLISTTYPE; /** * Structure to perform a request on a started guest * process. Needed for letting the main guest control thread * to communicate (and wait) for a certain operation which * will be done in context of the started guest process thread. */ typedef struct VBOXSERVICECTRLREQUEST { /** Event semaphore to serialize access. */ RTSEMEVENTMULTI Event; /** The request type to handle. */ VBOXSERVICECTRLREQUESTTYPE enmType; /** Payload size; on input, this contains the (maximum) amount * of data the caller wants to write or to read. On output, * this show the actual amount of data read/written. */ size_t cbData; /** Payload data; a pre-allocated data buffer for input/output. */ void *pvData; /** The context ID which is required to complete the * request. Not used at the moment. */ uint32_t uCID; /** The overall result of the operation. */ int rc; } VBOXSERVICECTRLREQUEST; /** Pointer to request. */ typedef VBOXSERVICECTRLREQUEST *PVBOXSERVICECTRLREQUEST; /** * Structure holding information for starting a guest * process. */ typedef struct VBOXSERVICECTRLPROCESS { /** Full qualified path of process to start (without arguments). */ char szCmd[GUESTPROCESS_MAX_CMD_LEN]; /** Process execution flags. @sa */ uint32_t uFlags; /** Command line arguments. */ char szArgs[GUESTPROCESS_MAX_ARGS_LEN]; /** Number of arguments specified in pszArgs. */ uint32_t uNumArgs; /** String of environment variables ("FOO=BAR") to pass to the process * to start. */ char szEnv[GUESTPROCESS_MAX_ENV_LEN]; /** Size (in bytes) of environment variables block. */ uint32_t cbEnv; /** Number of environment variables specified in pszEnv. */ uint32_t uNumEnvVars; /** User name (account) to start the process under. */ char szUser[GUESTPROCESS_MAX_USER_LEN]; /** Password of specified user name (account). */ char szPassword[GUESTPROCESS_MAX_PASSWORD_LEN]; /** Time limit (in ms) of the process' life time. */ uint32_t uTimeLimitMS; } VBOXSERVICECTRLPROCESS; /** Pointer to a guest process block. */ typedef VBOXSERVICECTRLPROCESS *PVBOXSERVICECTRLPROCESS; /** * Structure for holding data for one (started) guest process. */ typedef struct VBOXSERVICECTRLTHREAD { /** Pointer to list archor of following * list node. * @todo Would be nice to have a RTListGetAnchor(). */ PRTLISTANCHOR pAnchor; /** Node. */ RTLISTNODE Node; /** The worker thread. */ RTTHREAD Thread; /** Shutdown indicator; will be set when the thread * needs (or is asked) to shutdown. */ bool volatile fShutdown; /** Indicator set by the service thread exiting. */ bool volatile fStopped; /** Whether the service was started or not. */ bool fStarted; /** Client ID. */ uint32_t uClientID; /** Context ID. */ uint32_t uContextID; /** Critical section for thread-safe use. */ RTCRITSECT CritSect; /** @todo Document me! */ uint32_t uPID; char *pszCmd; uint32_t uFlags; char **papszArgs; uint32_t uNumArgs; char **papszEnv; uint32_t uNumEnvVars; /** Name of specified user account to run the * guest process under. */ char *pszUser; /** Password of specified user account. */ char *pszPassword; /** Overall time limit (in ms) that the guest process * is allowed to run. 0 for indefinite time. */ uint32_t uTimeLimitMS; /** Pointer to the current IPC request being * processed. */ PVBOXSERVICECTRLREQUEST pRequest; /** StdIn pipe for addressing writes to the * guest process' stdin.*/ RTPIPE pipeStdInW; /** The notification pipe associated with this guest process. * This is NIL_RTPIPE for output pipes. */ RTPIPE hNotificationPipeW; /** The other end of hNotificationPipeW. */ RTPIPE hNotificationPipeR; } VBOXSERVICECTRLTHREAD; /** Pointer to thread data. */ typedef VBOXSERVICECTRLTHREAD *PVBOXSERVICECTRLTHREAD; #endif /* VBOX_WITH_GUEST_CONTROL */ #ifdef VBOX_WITH_GUEST_PROPS /** * A guest property cache. */ typedef struct VBOXSERVICEVEPROPCACHE { /** The client ID for HGCM communication. */ uint32_t uClientID; /** Head in a list of VBOXSERVICEVEPROPCACHEENTRY nodes. */ RTLISTANCHOR NodeHead; /** Critical section for thread-safe use. */ RTCRITSECT CritSect; } VBOXSERVICEVEPROPCACHE; /** Pointer to a guest property cache. */ typedef VBOXSERVICEVEPROPCACHE *PVBOXSERVICEVEPROPCACHE; /** * An entry in the property cache (VBOXSERVICEVEPROPCACHE). */ typedef struct VBOXSERVICEVEPROPCACHEENTRY { /** Node to successor. * @todo r=bird: This is not really the node to the successor, but * rather the OUR node in the list. If it helps, remember that * its a doubly linked list. */ RTLISTNODE NodeSucc; /** Name (and full path) of guest property. */ char *pszName; /** The last value stored (for reference). */ char *pszValue; /** Reset value to write if property is temporary. If NULL, it will be * deleted. */ char *pszValueReset; /** Flags. */ uint32_t fFlags; } VBOXSERVICEVEPROPCACHEENTRY; /** Pointer to a cached guest property. */ typedef VBOXSERVICEVEPROPCACHEENTRY *PVBOXSERVICEVEPROPCACHEENTRY; #endif /* VBOX_WITH_GUEST_PROPS */ RT_C_DECLS_BEGIN extern char *g_pszProgName; extern int g_cVerbosity; extern uint32_t g_DefaultInterval; extern VBOXSERVICE g_TimeSync; extern VBOXSERVICE g_Clipboard; extern VBOXSERVICE g_Control; extern VBOXSERVICE g_VMInfo; extern VBOXSERVICE g_CpuHotPlug; #ifdef VBOXSERVICE_MANAGEMENT extern VBOXSERVICE g_MemBalloon; extern VBOXSERVICE g_VMStatistics; #endif #ifdef VBOX_WITH_PAGE_SHARING extern VBOXSERVICE g_PageSharing; #endif #ifdef VBOX_WITH_SHARED_FOLDERS extern VBOXSERVICE g_AutoMount; #endif #ifdef DEBUG extern RTCRITSECT g_csLog; /* For guest process stdout dumping. */ #endif extern RTEXITCODE VBoxServiceSyntax(const char *pszFormat, ...); extern RTEXITCODE VBoxServiceError(const char *pszFormat, ...); extern void VBoxServiceVerbose(int iLevel, const char *pszFormat, ...); extern int VBoxServiceArgUInt32(int argc, char **argv, const char *psz, int *pi, uint32_t *pu32, uint32_t u32Min, uint32_t u32Max); extern int VBoxServiceStartServices(void); extern int VBoxServiceStopServices(void); extern void VBoxServiceMainWait(void); extern int VBoxServiceReportStatus(VBoxGuestFacilityStatus enmStatus); #ifdef RT_OS_WINDOWS extern RTEXITCODE VBoxServiceWinInstall(void); extern RTEXITCODE VBoxServiceWinUninstall(void); extern RTEXITCODE VBoxServiceWinEnterCtrlDispatcher(void); extern void VBoxServiceWinSetStopPendingStatus(uint32_t uCheckPoint); #endif #ifdef VBOXSERVICE_TOOLBOX extern bool VBoxServiceToolboxMain(int argc, char **argv, RTEXITCODE *prcExit); #endif #ifdef RT_OS_WINDOWS # ifdef VBOX_WITH_GUEST_PROPS extern int VBoxServiceVMInfoWinWriteUsers(char **ppszUserList, uint32_t *pcUsersInList); extern int VBoxServiceWinGetComponentVersions(uint32_t uiClientID); # endif /* VBOX_WITH_GUEST_PROPS */ #endif /* RT_OS_WINDOWS */ #ifdef VBOX_WITH_GUEST_CONTROL /* Guest control main thread functions. */ extern int VBoxServiceControlAssignPID(PVBOXSERVICECTRLTHREAD pThread, uint32_t uPID); extern int VBoxServiceControlListSet(VBOXSERVICECTRLTHREADLISTTYPE enmList, PVBOXSERVICECTRLTHREAD pThread); extern PVBOXSERVICECTRLTHREAD VBoxServiceControlLockThread(uint32_t uPID); extern void VBoxServiceControlUnlockThread(const PVBOXSERVICECTRLTHREAD pThread); extern int VBoxServiceControlSetInactive(PVBOXSERVICECTRLTHREAD pThread); /* Per-thread guest process functions. */ extern int VBoxServiceControlThreadStart(uint32_t uContext, PVBOXSERVICECTRLPROCESS pProcess); extern int VBoxServiceControlThreadPerform(uint32_t uPID, PVBOXSERVICECTRLREQUEST pRequest); extern int VBoxServiceControlThreadStop(const PVBOXSERVICECTRLTHREAD pThread); extern int VBoxServiceControlThreadWait(const PVBOXSERVICECTRLTHREAD pThread, RTMSINTERVAL msTimeout); extern int VBoxServiceControlThreadFree(PVBOXSERVICECTRLTHREAD pThread); /* Request handling. */ extern int VBoxServiceControlThreadRequestAlloc(PVBOXSERVICECTRLREQUEST *ppReq, VBOXSERVICECTRLREQUESTTYPE enmType); extern int VBoxServiceControlThreadRequestAllocEx(PVBOXSERVICECTRLREQUEST *ppReq, VBOXSERVICECTRLREQUESTTYPE enmType, void* pbData, size_t cbData, uint32_t uCID); extern void VBoxServiceControlThreadRequestFree(PVBOXSERVICECTRLREQUEST pReq); #endif /* VBOX_WITH_GUEST_CONTROL */ #ifdef VBOXSERVICE_MANAGEMENT extern uint32_t VBoxServiceBalloonQueryPages(uint32_t cbPage); #endif #if defined(VBOX_WITH_PAGE_SHARING) && defined(RT_OS_WINDOWS) extern RTEXITCODE VBoxServicePageSharingInitFork(void); #endif RT_C_DECLS_END #endif