VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxTray/VBoxIPC.cpp@ 64326

Last change on this file since 64326 was 64326, checked in by vboxsync, 8 years ago

VBoxTrayMsg.h: Duh. VBOXTRAY_IPC_PIPE_PREFIX need to have the full native path, not just the filename part, now that we use RTLOCALIPC_FLAGS_NATIVE_NAME. Assertions!

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.3 KB
Line 
1/* $Id: VBoxIPC.cpp 64326 2016-10-19 17:37:25Z vboxsync $ */
2/** @file
3 * VBoxIPC - IPC thread, acts as a (purely) local IPC server.
4 * Multiple sessions are supported, whereas every session
5 * has its own thread for processing requests.
6 */
7
8/*
9 * Copyright (C) 2010-2016 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20
21/*********************************************************************************************************************************
22* Header Files *
23*********************************************************************************************************************************/
24#include <iprt/win/windows.h>
25#include "VBoxTray.h"
26#include "VBoxTrayMsg.h"
27#include "VBoxHelpers.h"
28#include "VBoxIPC.h"
29
30#include <iprt/asm.h>
31#include <iprt/assert.h>
32#include <iprt/critsect.h>
33#include <iprt/err.h>
34#include <iprt/ldr.h>
35#include <iprt/list.h>
36#include <iprt/localipc.h>
37#include <iprt/mem.h>
38#include <iprt/process.h>
39
40#include <VBox/VMMDev.h>
41#ifdef DEBUG
42# define LOG_ENABLED
43# define LOG_GROUP LOG_GROUP_DEFAULT
44#endif
45#include <VBox/log.h>
46
47
48/*********************************************************************************************************************************
49* Structures and Typedefs *
50*********************************************************************************************************************************/
51/**
52 * IPC context data.
53 */
54typedef struct VBOXIPCCONTEXT
55{
56 /** Pointer to the service environment. */
57 const VBOXSERVICEENV *pEnv;
58 /** Handle for the local IPC server. */
59 RTLOCALIPCSERVER hServer;
60 /** Critical section serializing access to the session list, the state,
61 * the response event, the session event, and the thread event. */
62 RTCRITSECT CritSect;
63 /** List of all active IPC sessions. */
64 RTLISTANCHOR SessionList;
65
66} VBOXIPCCONTEXT, *PVBOXIPCCONTEXT;
67
68/** Function pointer for GetLastInputInfo(). */
69typedef BOOL (WINAPI *PFNGETLASTINPUTINFO)(PLASTINPUTINFO);
70
71/**
72 * IPC per-session thread data.
73 */
74typedef struct VBOXIPCSESSION
75{
76 /** The list node required to be part of the
77 * IPC session list. */
78 RTLISTNODE Node;
79 /** Pointer to the IPC context data. */
80 PVBOXIPCCONTEXT volatile pCtx;
81 /** The local ipc client handle. */
82 RTLOCALIPCSESSION volatile hSession;
83 /** Indicate that the thread should terminate ASAP. */
84 bool volatile fTerminate;
85 /** The thread handle. */
86 RTTHREAD hThread;
87
88} VBOXIPCSESSION, *PVBOXIPCSESSION;
89
90
91/*********************************************************************************************************************************
92* Global Variables *
93*********************************************************************************************************************************/
94static VBOXIPCCONTEXT g_Ctx = { NULL, NIL_RTLOCALIPCSERVER };
95static PFNGETLASTINPUTINFO g_pfnGetLastInputInfo = NULL;
96
97
98/*********************************************************************************************************************************
99* Internal Functions *
100*********************************************************************************************************************************/
101static int vboxIPCSessionStop(PVBOXIPCSESSION pSession);
102
103
104
105static int vboxIPCHandleVBoxTrayRestart(PVBOXIPCSESSION pSession, PVBOXTRAYIPCHEADER pHdr)
106{
107 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
108 AssertPtrReturn(pHdr, VERR_INVALID_POINTER);
109
110 /** @todo Not implemented yet; don't return an error here. */
111 return VINF_SUCCESS;
112}
113
114static int vboxIPCHandleShowBalloonMsg(PVBOXIPCSESSION pSession, PVBOXTRAYIPCHEADER pHdr)
115{
116 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
117 AssertPtrReturn(pHdr, VERR_INVALID_POINTER);
118 AssertReturn(pHdr->uMsgLen > 0, VERR_INVALID_PARAMETER);
119
120 VBOXTRAYIPCMSG_SHOWBALLOONMSG ipcMsg;
121 int rc = RTLocalIpcSessionRead(pSession->hSession, &ipcMsg, pHdr->uMsgLen,
122 NULL /* Exact read, blocking */);
123 if (RT_SUCCESS(rc))
124 {
125 /* Showing the balloon tooltip is not critical. */
126 int rc2 = hlpShowBalloonTip(g_hInstance, g_hwndToolWindow, ID_TRAYICON,
127 ipcMsg.szMsgContent, ipcMsg.szMsgTitle,
128 ipcMsg.uShowMS, ipcMsg.uType);
129 LogFlowFunc(("Showing \"%s\" - \"%s\" (type %RU32, %RU32ms), rc=%Rrc\n",
130 ipcMsg.szMsgTitle, ipcMsg.szMsgContent,
131 ipcMsg.uType, ipcMsg.uShowMS, rc2));
132 NOREF(rc2);
133 }
134
135 return rc;
136}
137
138static int vboxIPCHandleUserLastInput(PVBOXIPCSESSION pSession, PVBOXTRAYIPCHEADER pHdr)
139{
140 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
141 AssertPtrReturn(pHdr, VERR_INVALID_POINTER);
142 /* No actual message from client. */
143
144 int rc = VINF_SUCCESS;
145
146 bool fLastInputAvailable = false;
147 VBOXTRAYIPCRES_USERLASTINPUT ipcRes;
148 if (g_pfnGetLastInputInfo)
149 {
150 /* Note: This only works up to 49.7 days (= 2^32, 32-bit counter)
151 since Windows was started. */
152 LASTINPUTINFO lastInput;
153 lastInput.cbSize = sizeof(LASTINPUTINFO);
154 BOOL fRc = g_pfnGetLastInputInfo(&lastInput);
155 if (fRc)
156 {
157 ipcRes.uLastInput = (GetTickCount() - lastInput.dwTime) / 1000;
158 fLastInputAvailable = true;
159 }
160 else
161 rc = RTErrConvertFromWin32(GetLastError());
162 }
163
164 if (!fLastInputAvailable)
165 {
166 /* No last input available. */
167 ipcRes.uLastInput = UINT32_MAX;
168 }
169
170 int rc2 = RTLocalIpcSessionWrite(pSession->hSession, &ipcRes, sizeof(ipcRes));
171 if (RT_SUCCESS(rc))
172 rc = rc2;
173
174 return rc;
175}
176
177/**
178 * Initializes the IPC communication.
179 *
180 * @return IPRT status code.
181 * @param pEnv The IPC service's environment.
182 * @param ppInstance The instance pointer which refers to this object.
183 */
184DECLCALLBACK(int) VBoxIPCInit(const PVBOXSERVICEENV pEnv, void **ppInstance)
185{
186 AssertPtrReturn(pEnv, VERR_INVALID_POINTER);
187 AssertPtrReturn(ppInstance, VERR_INVALID_POINTER);
188
189 LogFlowFuncEnter();
190
191 PVBOXIPCCONTEXT pCtx = &g_Ctx; /* Only one instance at the moment. */
192 AssertPtr(pCtx);
193
194 int rc = RTCritSectInit(&pCtx->CritSect);
195 if (RT_SUCCESS(rc))
196 {
197 char szPipeName[512 + sizeof(VBOXTRAY_IPC_PIPE_PREFIX)];
198 memcpy(szPipeName, VBOXTRAY_IPC_PIPE_PREFIX, sizeof(VBOXTRAY_IPC_PIPE_PREFIX));
199 rc = RTProcQueryUsername(NIL_RTPROCESS,
200 &szPipeName[sizeof(VBOXTRAY_IPC_PIPE_PREFIX) - 1],
201 sizeof(szPipeName) - sizeof(VBOXTRAY_IPC_PIPE_PREFIX) + 1,
202 NULL /*pcbUser*/);
203 AssertRC(rc);
204 if (RT_SUCCESS(rc))
205 {
206 rc = RTLocalIpcServerCreate(&pCtx->hServer, szPipeName, RTLOCALIPC_FLAGS_NATIVE_NAME);
207 AssertRC(rc);
208 if (RT_SUCCESS(rc))
209 {
210 pCtx->pEnv = pEnv;
211 RTListInit(&pCtx->SessionList);
212
213 *ppInstance = pCtx;
214
215 /* GetLastInputInfo only is available starting at Windows 2000 -- might fail. */
216 g_pfnGetLastInputInfo = (PFNGETLASTINPUTINFO)
217 RTLdrGetSystemSymbol("User32.dll", "GetLastInputInfo");
218
219 LogRelFunc(("Local IPC server now running at \"%s\"\n", szPipeName));
220 return VINF_SUCCESS;
221 }
222
223 }
224
225 RTCritSectDelete(&pCtx->CritSect);
226 }
227
228 LogRelFunc(("Creating local IPC server failed with rc=%Rrc\n", rc));
229 return rc;
230}
231
232DECLCALLBACK(void) VBoxIPCStop(void *pInstance)
233{
234 /* Can be NULL if VBoxIPCInit failed. */
235 if (!pInstance)
236 return;
237 AssertPtrReturnVoid(pInstance);
238
239 LogFlowFunc(("Stopping pInstance=%p\n", pInstance));
240
241 /* Shut down local IPC server. */
242 PVBOXIPCCONTEXT pCtx = (PVBOXIPCCONTEXT)pInstance;
243 AssertPtr(pCtx);
244
245 if (pCtx->hServer != NIL_RTLOCALIPCSERVER)
246 {
247 int rc2 = RTLocalIpcServerCancel(pCtx->hServer);
248 if (RT_FAILURE(rc2))
249 LogFlowFunc(("Cancelling current listening call failed with rc=%Rrc\n", rc2));
250 }
251
252 /* Stop all remaining session threads. */
253 int rc = RTCritSectEnter(&pCtx->CritSect);
254 if (RT_SUCCESS(rc))
255 {
256 PVBOXIPCSESSION pSession;
257 RTListForEach(&pCtx->SessionList, pSession, VBOXIPCSESSION, Node)
258 {
259 int rc2 = vboxIPCSessionStop(pSession);
260 if (RT_FAILURE(rc2))
261 {
262 LogFlowFunc(("Stopping IPC session %p failed with rc=%Rrc\n",
263 pSession, rc2));
264 /* Keep going. */
265 }
266 }
267 }
268}
269
270DECLCALLBACK(void) VBoxIPCDestroy(void *pInstance)
271{
272 AssertPtrReturnVoid(pInstance);
273
274 LogFlowFunc(("Destroying pInstance=%p\n", pInstance));
275
276 PVBOXIPCCONTEXT pCtx = (PVBOXIPCCONTEXT)pInstance;
277 AssertPtr(pCtx);
278
279 /* Shut down local IPC server. */
280 int rc = RTCritSectEnter(&pCtx->CritSect);
281 if (RT_SUCCESS(rc))
282 {
283 rc = RTLocalIpcServerDestroy(pCtx->hServer);
284 if (RT_FAILURE(rc))
285 LogFlowFunc(("Unable to destroy IPC server, rc=%Rrc\n", rc));
286
287 int rc2 = RTCritSectLeave(&pCtx->CritSect);
288 if (RT_SUCCESS(rc))
289 rc = rc2;
290 }
291
292 LogFlowFunc(("Waiting for remaining IPC sessions to shut down ...\n"));
293
294 /* Wait for all IPC session threads to shut down. */
295 bool fListIsEmpty = true;
296 do
297 {
298 int rc2 = RTCritSectEnter(&pCtx->CritSect);
299 if (RT_SUCCESS(rc2))
300 {
301 fListIsEmpty = RTListIsEmpty(&pCtx->SessionList);
302 rc2 = RTCritSectLeave(&pCtx->CritSect);
303
304 if (!fListIsEmpty) /* Don't hog CPU while waiting. */
305 RTThreadSleep(100);
306 }
307
308 if (RT_FAILURE(rc2))
309 break;
310
311 } while (!fListIsEmpty);
312
313 AssertMsg(fListIsEmpty,
314 ("Session thread list is not empty when it should\n"));
315
316 LogFlowFunc(("All remaining IPC sessions shut down\n"));
317
318 int rc2 = RTCritSectDelete(&pCtx->CritSect);
319 if (RT_SUCCESS(rc))
320 rc = rc2;
321
322 LogFlowFunc(("Destroyed pInstance=%p, rc=%Rrc\n",
323 pInstance, rc));
324}
325
326/**
327 * Services a client session.
328 *
329 * @returns VINF_SUCCESS.
330 * @param hThreadSelf The thread handle.
331 * @param pvSession Pointer to the session instance data.
332 */
333static DECLCALLBACK(int) vboxIPCSessionThread(RTTHREAD hThreadSelf, void *pvSession)
334{
335 RT_NOREF(hThreadSelf);
336 PVBOXIPCSESSION pThis = (PVBOXIPCSESSION)pvSession;
337 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
338 RTLOCALIPCSESSION hSession = pThis->hSession;
339 AssertReturn(hSession != NIL_RTLOCALIPCSESSION, VERR_INVALID_PARAMETER);
340
341 LogFlowFunc(("pThis=%p\n", pThis));
342
343 int rc = VINF_SUCCESS;
344
345 /*
346 * Process client requests until it quits or we're cancelled on termination.
347 */
348 while ( !ASMAtomicUoReadBool(&pThis->fTerminate)
349 && RT_SUCCESS(rc))
350 {
351 /* The next call will be cancelled via VBoxIPCStop if needed. */
352 rc = RTLocalIpcSessionWaitForData(hSession, RT_INDEFINITE_WAIT);
353 if (RT_FAILURE(rc))
354 {
355 if (rc == VERR_CANCELLED)
356 {
357 LogFlowFunc(("Session %p: Waiting for data cancelled\n", pThis));
358 rc = VINF_SUCCESS;
359 break;
360 }
361 else
362 LogFlowFunc(("Session %p: Waiting for session data failed with rc=%Rrc\n",
363 pThis, rc));
364 }
365 else
366 {
367 VBOXTRAYIPCHEADER ipcHdr;
368 rc = RTLocalIpcSessionRead(hSession, &ipcHdr, sizeof(ipcHdr),
369 NULL /* Exact read, blocking */);
370 bool fRejected = false; /* Reject current command? */
371 if (RT_SUCCESS(rc))
372 fRejected = ipcHdr.uMagic != VBOXTRAY_IPC_HDR_MAGIC
373 || ipcHdr.uHdrVersion != 0; /* We only know version 0 commands for now. */
374
375 if ( !fRejected
376 && RT_SUCCESS(rc))
377 {
378 switch (ipcHdr.uMsgType)
379 {
380 case VBOXTRAYIPCMSGTYPE_RESTART:
381 rc = vboxIPCHandleVBoxTrayRestart(pThis, &ipcHdr);
382 break;
383
384 case VBOXTRAYIPCMSGTYPE_SHOWBALLOONMSG:
385 rc = vboxIPCHandleShowBalloonMsg(pThis, &ipcHdr);
386 break;
387
388 case VBOXTRAYIPCMSGTYPE_USERLASTINPUT:
389 rc = vboxIPCHandleUserLastInput(pThis, &ipcHdr);
390 break;
391
392 default:
393 {
394 /* Unknown command, reject. */
395 fRejected = true;
396 break;
397 }
398 }
399
400 if (RT_FAILURE(rc))
401 LogFlowFunc(("Session %p: Handling command %RU32 failed with rc=%Rrc\n",
402 pThis, ipcHdr.uMsgType, rc));
403 }
404
405 if (fRejected)
406 {
407 static int s_cRejectedCmds = 0;
408 if (++s_cRejectedCmds <= 3)
409 {
410 LogRelFunc(("Session %p: Received invalid/unknown command %RU32 (%RU32 bytes), rejecting (%RU32/3)\n",
411 pThis, ipcHdr.uMsgType, ipcHdr.uMsgLen, s_cRejectedCmds + 1));
412 if (ipcHdr.uMsgLen)
413 {
414 /* Get and discard payload data. */
415 size_t cbRead;
416 uint8_t devNull[_1K];
417 while (ipcHdr.uMsgLen)
418 {
419 rc = RTLocalIpcSessionRead(hSession, &devNull, sizeof(devNull), &cbRead);
420 if (RT_FAILURE(rc))
421 break;
422 AssertRelease(cbRead <= ipcHdr.uMsgLen);
423 ipcHdr.uMsgLen -= (uint32_t)cbRead;
424 }
425 }
426 }
427 else
428 rc = VERR_INVALID_PARAMETER; /* Enough fun, bail out. */
429 }
430 }
431 }
432
433 LogFlowFunc(("Session %p: Handler ended with rc=%Rrc\n",
434 pThis, rc));
435
436 /*
437 * Close the session.
438 */
439 int rc2 = RTLocalIpcSessionClose(hSession);
440 if (RT_FAILURE(rc2))
441 LogFlowFunc(("Session %p: Failed closing session %p, rc=%Rrc\n", pThis, rc2));
442
443 /*
444 * Clean up the session.
445 */
446 PVBOXIPCCONTEXT pCtx = ASMAtomicReadPtrT(&pThis->pCtx, PVBOXIPCCONTEXT);
447 AssertMsg(pCtx, ("Session %p: No context found\n", pThis));
448 rc2 = RTCritSectEnter(&pCtx->CritSect);
449 if (RT_SUCCESS(rc2))
450 {
451 /* Remove this session from the session list. */
452 RTListNodeRemove(&pThis->Node);
453
454 rc2 = RTCritSectLeave(&pCtx->CritSect);
455 if (RT_SUCCESS(rc))
456 rc = rc2;
457 }
458
459 LogFlowFunc(("Session %p: Terminated with rc=%Rrc, freeing ...\n",
460 pThis, rc));
461
462 RTMemFree(pThis);
463 pThis = NULL;
464
465 return rc;
466}
467
468static int vboxIPCSessionCreate(PVBOXIPCCONTEXT pCtx, RTLOCALIPCSESSION hSession)
469{
470 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
471 AssertReturn(hSession != NIL_RTLOCALIPCSESSION, VERR_INVALID_PARAMETER);
472
473 int rc = RTCritSectEnter(&pCtx->CritSect);
474 if (RT_SUCCESS(rc))
475 {
476 PVBOXIPCSESSION pSession = (PVBOXIPCSESSION)RTMemAllocZ(sizeof(VBOXIPCSESSION));
477 if (pSession)
478 {
479 pSession->pCtx = pCtx;
480 pSession->hSession = hSession;
481 pSession->fTerminate = false;
482 pSession->hThread = NIL_RTTHREAD;
483
484 /* Start IPC session thread. */
485 LogFlowFunc(("Creating thread for session %p ...\n", pSession));
486 rc = RTThreadCreate(&pSession->hThread, vboxIPCSessionThread,
487 pSession /* pvUser */, 0 /* Default stack size */,
488 RTTHREADTYPE_DEFAULT, 0 /* Flags */, "IPCSESSION");
489 if (RT_SUCCESS(rc))
490 {
491 /* Add session thread to session IPC list. */
492 RTListAppend(&pCtx->SessionList, &pSession->Node);
493 }
494 else
495 {
496 int rc2 = RTLocalIpcSessionClose(hSession);
497 if (RT_FAILURE(rc2))
498 LogFlowFunc(("Failed closing session %p, rc=%Rrc\n", pSession, rc2));
499
500 LogFlowFunc(("Failed to create thread for session %p, rc=%Rrc\n", pSession, rc));
501 RTMemFree(pSession);
502 }
503 }
504 else
505 rc = VERR_NO_MEMORY;
506
507 int rc2 = RTCritSectLeave(&pCtx->CritSect);
508 AssertRC(rc2);
509 }
510
511 return rc;
512}
513
514static int vboxIPCSessionStop(PVBOXIPCSESSION pSession)
515{
516 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
517
518 ASMAtomicWriteBool(&pSession->fTerminate, true);
519
520 RTLOCALIPCSESSION hSession;
521 ASMAtomicXchgHandle(&pSession->hSession, NIL_RTLOCALIPCSESSION, &hSession);
522 if (hSession)
523 return RTLocalIpcSessionClose(hSession);
524
525 return VINF_SUCCESS;
526}
527
528/**
529 * Thread function to wait for and process seamless mode change
530 * requests
531 */
532DECLCALLBACK(int) VBoxIPCWorker(void *pInstance, bool volatile *pfShutdown)
533{
534 AssertPtr(pInstance);
535 LogFlowFunc(("pInstance=%p\n", pInstance));
536
537 LogFlowFuncEnter();
538
539 /*
540 * Tell the control thread that it can continue
541 * spawning services.
542 */
543 RTThreadUserSignal(RTThreadSelf());
544
545 PVBOXIPCCONTEXT pCtx = (PVBOXIPCCONTEXT)pInstance;
546 AssertPtr(pCtx);
547
548 int rc;
549
550 bool fShutdown = false;
551 for (;;)
552 {
553 RTLOCALIPCSESSION hClientSession = NIL_RTLOCALIPCSESSION;
554 rc = RTLocalIpcServerListen(pCtx->hServer, &hClientSession);
555 if (RT_FAILURE(rc))
556 {
557 if (rc == VERR_CANCELLED)
558 {
559 LogFlow(("Cancelled\n"));
560 fShutdown = true;
561 }
562 else
563 LogRelFunc(("Listening failed with rc=%Rrc\n", rc));
564 }
565
566 if (fShutdown)
567 break;
568 rc = vboxIPCSessionCreate(pCtx, hClientSession);
569 if (RT_FAILURE(rc))
570 {
571 LogRelFunc(("Creating new IPC server session failed with rc=%Rrc\n", rc));
572 /* Keep going. */
573 }
574
575 if (*pfShutdown)
576 break;
577 }
578
579 LogFlowFuncLeaveRC(rc);
580 return rc;
581}
582
583/**
584 * The service description.
585 */
586VBOXSERVICEDESC g_SvcDescIPC =
587{
588 /* pszName. */
589 "IPC",
590 /* pszDescription. */
591 "Inter-Process Communication",
592 /* methods */
593 VBoxIPCInit,
594 VBoxIPCWorker,
595 NULL /* pfnStop */,
596 VBoxIPCDestroy
597};
598
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