VirtualBox

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

Last change on this file since 106411 was 106411, checked in by vboxsync, 3 months ago

Additions/VBoxTray: Implemented ability for easier user-controllable logging (also via verbose levels), support for running in foreground mode (with a console window attached to) and selective starting of sub services to easier pinpoint errors in release builds. Cleaned up initialization / termination code a little. See command line help for new options. bugref:10763

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 20.6 KB
Line 
1/* $Id: VBoxIPC.cpp 106411 2024-10-17 07:44:43Z 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-2024 Oracle and/or its affiliates.
10 *
11 * This file is part of VirtualBox base platform packages, as
12 * available from https://www.virtualbox.org.
13 *
14 * This program is free software; you can redistribute it and/or
15 * modify it under the terms of the GNU General Public License
16 * as published by the Free Software Foundation, in version 3 of the
17 * License.
18 *
19 * This program is distributed in the hope that it will be useful, but
20 * WITHOUT ANY WARRANTY; without even the implied warranty of
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
22 * General Public License for more details.
23 *
24 * You should have received a copy of the GNU General Public License
25 * along with this program; if not, see <https://www.gnu.org/licenses>.
26 *
27 * SPDX-License-Identifier: GPL-3.0-only
28 */
29
30
31/*********************************************************************************************************************************
32* Header Files *
33*********************************************************************************************************************************/
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/critsect.h>
37#include <iprt/errcore.h>
38#include <iprt/ldr.h>
39#include <iprt/list.h>
40#include <iprt/localipc.h>
41#include <iprt/log.h>
42#include <iprt/mem.h>
43#include <iprt/process.h>
44#include <iprt/win/windows.h>
45
46#include "VBoxTray.h"
47#include "VBoxTrayMsg.h"
48#include "VBoxHelpers.h"
49#include "VBoxIPC.h"
50
51
52/*********************************************************************************************************************************
53* Structures and Typedefs *
54*********************************************************************************************************************************/
55/**
56 * IPC context data.
57 */
58typedef struct VBOXIPCCONTEXT
59{
60 /** Pointer to the service environment. */
61 const VBOXTRAYSVCENV *pEnv;
62 /** Handle for the local IPC server. */
63 RTLOCALIPCSERVER hServer;
64 /** Critical section serializing access to the session list, the state,
65 * the response event, the session event, and the thread event. */
66 RTCRITSECT CritSect;
67 /** List of all active IPC sessions. */
68 RTLISTANCHOR SessionList;
69
70} VBOXIPCCONTEXT, *PVBOXIPCCONTEXT;
71
72/** Function pointer for GetLastInputInfo(). */
73typedef BOOL (WINAPI *PFNGETLASTINPUTINFO)(PLASTINPUTINFO);
74
75/**
76 * IPC per-session thread data.
77 */
78typedef struct VBOXIPCSESSION
79{
80 /** The list node required to be part of the
81 * IPC session list. */
82 RTLISTNODE Node;
83 /** Pointer to the IPC context data. */
84 PVBOXIPCCONTEXT volatile pCtx;
85 /** The local ipc client handle. */
86 RTLOCALIPCSESSION volatile hSession;
87 /** Indicate that the thread should terminate ASAP. */
88 bool volatile fTerminate;
89 /** The thread handle. */
90 RTTHREAD hThread;
91
92} VBOXIPCSESSION, *PVBOXIPCSESSION;
93
94
95/*********************************************************************************************************************************
96* Global Variables *
97*********************************************************************************************************************************/
98static VBOXIPCCONTEXT g_Ctx = { NULL, NIL_RTLOCALIPCSERVER };
99static PFNGETLASTINPUTINFO g_pfnGetLastInputInfo = NULL;
100
101
102/*********************************************************************************************************************************
103* Internal Functions *
104*********************************************************************************************************************************/
105static int vboxIPCSessionStop(PVBOXIPCSESSION pSession);
106
107
108
109/**
110 * Handles VBOXTRAYIPCMSGTYPE_RESTART.
111 */
112static int vboxIPCHandleVBoxTrayRestart(PVBOXIPCSESSION pSession, PVBOXTRAYIPCHEADER pHdr)
113{
114 RT_NOREF(pSession, pHdr);
115
116 /** @todo Not implemented yet; don't return an error here. */
117 return VINF_SUCCESS;
118}
119
120/**
121 * Handles VBOXTRAYIPCMSGTYPE_SHOW_BALLOON_MSG.
122 */
123static int vboxIPCHandleShowBalloonMsg(PVBOXIPCSESSION pSession, PVBOXTRAYIPCHEADER pHdr)
124{
125 /*
126 * Unmarshal and validate the data.
127 */
128 union
129 {
130 uint8_t abBuf[_4K];
131 VBOXTRAYIPCMSG_SHOW_BALLOON_MSG_T s;
132 } Payload;
133 AssertReturn(pHdr->cbPayload >= RT_UOFFSETOF_DYN(VBOXTRAYIPCMSG_SHOW_BALLOON_MSG_T, szzStrings[2]), VERR_INVALID_PARAMETER);
134 AssertReturn(pHdr->cbPayload < sizeof(Payload), VERR_BUFFER_OVERFLOW);
135
136 int rc = RTLocalIpcSessionRead(pSession->hSession, &Payload, pHdr->cbPayload, NULL /*pcbRead - exact, blocking*/);
137 if (RT_FAILURE(rc))
138 return rc;
139
140 /* String lengths: */
141 AssertReturn( Payload.s.cchMsg + 1 + Payload.s.cchTitle + 1 + RT_UOFFSETOF(VBOXTRAYIPCMSG_SHOW_BALLOON_MSG_T, szzStrings)
142 <= pHdr->cbPayload, VERR_INVALID_PARAMETER);
143
144 /* Message text: */
145 const char *pszMsg = Payload.s.szzStrings;
146 rc = RTStrValidateEncodingEx(pszMsg, Payload.s.cchMsg + 1,
147 RTSTR_VALIDATE_ENCODING_EXACT_LENGTH | RTSTR_VALIDATE_ENCODING_ZERO_TERMINATED);
148 AssertRCReturn(rc, rc);
149
150 /* Title text: */
151 const char *pszTitle = &Payload.s.szzStrings[Payload.s.cchMsg + 1];
152 rc = RTStrValidateEncodingEx(pszMsg, Payload.s.cchTitle + 1,
153 RTSTR_VALIDATE_ENCODING_EXACT_LENGTH | RTSTR_VALIDATE_ENCODING_ZERO_TERMINATED);
154 AssertRCReturn(rc, rc);
155
156 /* Type/dwInfoFlags: */
157 AssertReturn( Payload.s.uType == NIIF_NONE
158 || Payload.s.uType == NIIF_INFO
159 || Payload.s.uType == NIIF_WARNING
160 || Payload.s.uType == NIIF_ERROR,
161 VERR_WRONG_TYPE);
162
163 /* Timeout: */
164 if (!Payload.s.cMsTimeout)
165 Payload.s.cMsTimeout = RT_MS_5SEC;
166 AssertStmt(Payload.s.cMsTimeout >= RT_MS_1SEC, Payload.s.cMsTimeout = RT_MS_1SEC);
167 AssertStmt(Payload.s.cMsTimeout <= RT_MS_1MIN, Payload.s.cMsTimeout = RT_MS_1MIN);
168
169 /*
170 * Showing the balloon tooltip is not critical.
171 */
172 VBoxTrayHlpShowBalloonTipEx(g_hInstance, g_hwndToolWindow, ID_TRAYICON,
173 pszMsg, pszTitle, Payload.s.cMsTimeout, Payload.s.uType);
174
175 return VINF_SUCCESS;
176}
177
178/**
179 * Handles VBOXTRAYIPCMSGTYPE_USER_LAST_INPUT.
180 */
181static int vboxIPCHandleUserLastInput(PVBOXIPCSESSION pSession, PVBOXTRAYIPCHEADER pHdr)
182{
183 RT_NOREF(pHdr);
184
185 int rc = VINF_SUCCESS;
186 VBOXTRAYIPCREPLY_USER_LAST_INPUT_T Reply = { UINT32_MAX };
187 if (g_pfnGetLastInputInfo)
188 {
189 /* Note: This only works up to 49.7 days (= 2^32, 32-bit counter) since Windows was started. */
190 LASTINPUTINFO LastInput;
191 LastInput.cbSize = sizeof(LastInput);
192 if (g_pfnGetLastInputInfo(&LastInput))
193 Reply.cSecSinceLastInput = (GetTickCount() - LastInput.dwTime) / 1000;
194 else
195 rc = RTErrConvertFromWin32(GetLastError());
196 }
197
198 int rc2 = RTLocalIpcSessionWrite(pSession->hSession, &Reply, sizeof(Reply));
199 if (RT_SUCCESS(rc))
200 rc = rc2;
201
202 return rc;
203}
204
205/**
206 * @interface_method_impl{VBOXSERVICEDESC,pfnPreInit}
207 */
208static DECLCALLBACK(int) vbtrIPCPreInit(void)
209{
210 return VINF_SUCCESS;
211}
212
213
214/**
215 * @interface_method_impl{VBOXSERVICEDESC,pfnOption}
216 */
217static DECLCALLBACK(int) vbtrIPCOption(const char **ppszShort, int argc, char **argv, int *pi)
218{
219 RT_NOREF(ppszShort, argc, argv, pi);
220
221 return -1;
222}
223
224/**
225 * @interface_method_impl{VBOXSERVICEDESC,pfnInit}
226 */
227DECLCALLBACK(int) vbtrIPCInit(const PVBOXTRAYSVCENV pEnv, void **ppInstance)
228{
229 AssertPtrReturn(pEnv, VERR_INVALID_POINTER);
230 AssertPtrReturn(ppInstance, VERR_INVALID_POINTER);
231
232 LogFlowFuncEnter();
233
234 PVBOXIPCCONTEXT pCtx = &g_Ctx; /* Only one instance at the moment. */
235 AssertPtr(pCtx);
236
237 int rc = RTCritSectInit(&pCtx->CritSect);
238 if (RT_SUCCESS(rc))
239 {
240 char szPipeName[512 + sizeof(VBOXTRAY_IPC_PIPE_PREFIX)];
241 memcpy(szPipeName, VBOXTRAY_IPC_PIPE_PREFIX, sizeof(VBOXTRAY_IPC_PIPE_PREFIX));
242 rc = RTProcQueryUsername(NIL_RTPROCESS,
243 &szPipeName[sizeof(VBOXTRAY_IPC_PIPE_PREFIX) - 1],
244 sizeof(szPipeName) - sizeof(VBOXTRAY_IPC_PIPE_PREFIX) + 1,
245 NULL /*pcbUser*/);
246 AssertRC(rc);
247 if (RT_SUCCESS(rc))
248 {
249 rc = RTLocalIpcServerCreate(&pCtx->hServer, szPipeName, RTLOCALIPC_FLAGS_NATIVE_NAME);
250 AssertRC(rc);
251 if (RT_SUCCESS(rc))
252 {
253 pCtx->pEnv = pEnv;
254 RTListInit(&pCtx->SessionList);
255
256 *ppInstance = pCtx;
257
258 /* GetLastInputInfo only is available starting at Windows 2000 -- might fail. */
259 g_pfnGetLastInputInfo = (PFNGETLASTINPUTINFO)RTLdrGetSystemSymbol("User32.dll", "GetLastInputInfo");
260
261 LogRelFunc(("Local IPC server now running at \"%s\"\n", szPipeName));
262 return VINF_SUCCESS;
263 }
264
265 }
266
267 RTCritSectDelete(&pCtx->CritSect);
268 }
269
270 LogRelFunc(("Creating local IPC server failed with rc=%Rrc\n", rc));
271 return rc;
272}
273
274/**
275 * @interface_method_impl{VBOXSERVICEDESC,pfnStop}
276 */
277DECLCALLBACK(void) VBoxIPCStop(void *pInstance)
278{
279 /* Can be NULL if VBoxIPCInit failed. */
280 if (!pInstance)
281 return;
282 AssertPtrReturnVoid(pInstance);
283
284 LogFlowFunc(("Stopping pInstance=%p\n", pInstance));
285
286 /* Shut down local IPC server. */
287 PVBOXIPCCONTEXT pCtx = (PVBOXIPCCONTEXT)pInstance;
288 AssertPtr(pCtx);
289
290 if (pCtx->hServer != NIL_RTLOCALIPCSERVER)
291 {
292 int rc2 = RTLocalIpcServerCancel(pCtx->hServer);
293 if (RT_FAILURE(rc2))
294 LogFlowFunc(("Cancelling current listening call failed with rc=%Rrc\n", rc2));
295 }
296
297 /* Stop all remaining session threads. */
298 int rc = RTCritSectEnter(&pCtx->CritSect);
299 if (RT_SUCCESS(rc))
300 {
301 PVBOXIPCSESSION pSession;
302 RTListForEach(&pCtx->SessionList, pSession, VBOXIPCSESSION, Node)
303 {
304 int rc2 = vboxIPCSessionStop(pSession);
305 if (RT_FAILURE(rc2))
306 {
307 LogFlowFunc(("Stopping IPC session %p failed with rc=%Rrc\n",
308 pSession, rc2));
309 /* Keep going. */
310 }
311 }
312 }
313}
314
315/**
316 * @interface_method_impl{VBOXSERVICEDESC,pfnDestroy}
317 */
318DECLCALLBACK(void) vbtrIPCDestroy(void *pInstance)
319{
320 AssertPtrReturnVoid(pInstance);
321
322 LogFlowFunc(("Destroying pInstance=%p\n", pInstance));
323
324 PVBOXIPCCONTEXT pCtx = (PVBOXIPCCONTEXT)pInstance;
325 AssertPtr(pCtx);
326
327 /* Shut down local IPC server. */
328 int rc = RTCritSectEnter(&pCtx->CritSect);
329 if (RT_SUCCESS(rc))
330 {
331 rc = RTLocalIpcServerDestroy(pCtx->hServer);
332 if (RT_FAILURE(rc))
333 LogFlowFunc(("Unable to destroy IPC server, rc=%Rrc\n", rc));
334
335 int rc2 = RTCritSectLeave(&pCtx->CritSect);
336 if (RT_SUCCESS(rc))
337 rc = rc2;
338 }
339
340 LogFlowFunc(("Waiting for remaining IPC sessions to shut down ...\n"));
341
342 /* Wait for all IPC session threads to shut down. */
343 bool fListIsEmpty = true;
344 do
345 {
346 int rc2 = RTCritSectEnter(&pCtx->CritSect);
347 if (RT_SUCCESS(rc2))
348 {
349 fListIsEmpty = RTListIsEmpty(&pCtx->SessionList);
350 rc2 = RTCritSectLeave(&pCtx->CritSect);
351
352 if (!fListIsEmpty) /* Don't hog CPU while waiting. */
353 RTThreadSleep(100);
354 }
355
356 if (RT_FAILURE(rc2))
357 break;
358
359 } while (!fListIsEmpty);
360
361 AssertMsg(fListIsEmpty,
362 ("Session thread list is not empty when it should\n"));
363
364 LogFlowFunc(("All remaining IPC sessions shut down\n"));
365
366 int rc2 = RTCritSectDelete(&pCtx->CritSect);
367 if (RT_SUCCESS(rc))
368 rc = rc2;
369
370 LogFlowFunc(("Destroyed pInstance=%p, rc=%Rrc\n",
371 pInstance, rc));
372}
373
374/**
375 * Services a client session.
376 *
377 * @returns VINF_SUCCESS.
378 * @param hThreadSelf The thread handle.
379 * @param pvSession Pointer to the session instance data.
380 */
381static DECLCALLBACK(int) vboxIPCSessionThread(RTTHREAD hThreadSelf, void *pvSession)
382{
383 RT_NOREF(hThreadSelf);
384 PVBOXIPCSESSION pThis = (PVBOXIPCSESSION)pvSession;
385 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
386 RTLOCALIPCSESSION hSession = pThis->hSession;
387 AssertReturn(hSession != NIL_RTLOCALIPCSESSION, VERR_INVALID_PARAMETER);
388
389 LogFlowFunc(("pThis=%p\n", pThis));
390
391 int rc = VINF_SUCCESS;
392
393 /*
394 * Process client requests until it quits or we're cancelled on termination.
395 */
396 while ( !ASMAtomicUoReadBool(&pThis->fTerminate)
397 && RT_SUCCESS(rc))
398 {
399 /* The next call will be cancelled via VBoxIPCStop if needed. */
400 rc = RTLocalIpcSessionWaitForData(hSession, RT_INDEFINITE_WAIT);
401 if (RT_SUCCESS(rc))
402 {
403 /*
404 * Read the message header.
405 */
406 VBOXTRAYIPCHEADER Hdr = {0};
407 rc = RTLocalIpcSessionRead(hSession, &Hdr, sizeof(Hdr), NULL /*pcbRead - exact, blocking*/);
408 if (RT_FAILURE(rc))
409 break;
410
411 /*
412 * Validate the message header.
413 *
414 * Disconnecting the client if invalid or something we don't grok.
415 * Currently all clients are one-shots, so there is no need to get
416 * in complicated recovery code if we don't understand one another.
417 */
418 if ( Hdr.uMagic != VBOXTRAY_IPC_HDR_MAGIC
419 || Hdr.uVersion != VBOXTRAY_IPC_HDR_VERSION)
420 {
421 LogRelFunc(("Session %p: Invalid header magic/version: %#x, %#x, %#x, %#x\n",
422 pThis, Hdr.uMagic, Hdr.uVersion, Hdr.enmMsgType, Hdr.cbPayload));
423 rc = VERR_INVALID_MAGIC;
424 break;
425 }
426 if (Hdr.cbPayload > VBOXTRAY_IPC_MAX_PAYLOAD)
427 {
428 LogRelFunc(("Session %p: Payload to big: %#x, %#x, %#x, %#x - max %#x\n",
429 pThis, Hdr.uMagic, Hdr.uVersion, Hdr.enmMsgType, Hdr.cbPayload, VBOXTRAY_IPC_MAX_PAYLOAD));
430 rc = VERR_TOO_MUCH_DATA;
431 break;
432 }
433 if ( Hdr.enmMsgType <= VBOXTRAYIPCMSGTYPE_INVALID
434 || Hdr.enmMsgType >= VBOXTRAYIPCMSGTYPE_END)
435 {
436 LogRelFunc(("Session %p: Unknown message: %#x, %#x, %#x, %#x\n",
437 pThis, Hdr.uMagic, Hdr.uVersion, Hdr.enmMsgType, Hdr.cbPayload));
438 rc = VERR_INVALID_FUNCTION;
439 break;
440 }
441
442 /*
443 * Handle the message.
444 */
445 switch (Hdr.enmMsgType)
446 {
447 case VBOXTRAYIPCMSGTYPE_RESTART:
448 rc = vboxIPCHandleVBoxTrayRestart(pThis, &Hdr);
449 break;
450
451 case VBOXTRAYIPCMSGTYPE_SHOW_BALLOON_MSG:
452 rc = vboxIPCHandleShowBalloonMsg(pThis, &Hdr);
453 break;
454
455 case VBOXTRAYIPCMSGTYPE_USER_LAST_INPUT:
456 rc = vboxIPCHandleUserLastInput(pThis, &Hdr);
457 break;
458
459 default:
460 AssertFailedBreakStmt(rc = VERR_IPE_NOT_REACHED_DEFAULT_CASE);
461 }
462 if (RT_FAILURE(rc))
463 LogFlowFunc(("Session %p: Handling command %RU32 failed with rc=%Rrc\n", pThis, Hdr.enmMsgType, rc));
464 }
465 else if (rc == VERR_CANCELLED)
466 {
467 LogFlowFunc(("Session %p: Waiting for data cancelled\n", pThis));
468 rc = VINF_SUCCESS;
469 break;
470 }
471 else
472 LogFlowFunc(("Session %p: Waiting for session data failed with rc=%Rrc\n", pThis, rc));
473 }
474
475 LogFlowFunc(("Session %p: Handler ended with rc=%Rrc\n", pThis, rc));
476
477 /*
478 * Close the session.
479 */
480 int rc2 = RTLocalIpcSessionClose(hSession);
481 if (RT_FAILURE(rc2))
482 LogFlowFunc(("Session %p: Failed closing session %p, rc=%Rrc\n", pThis, rc2));
483
484 /*
485 * Clean up the session.
486 */
487 PVBOXIPCCONTEXT pCtx = ASMAtomicReadPtrT(&pThis->pCtx, PVBOXIPCCONTEXT);
488 AssertMsg(pCtx, ("Session %p: No context found\n", pThis));
489 rc2 = RTCritSectEnter(&pCtx->CritSect);
490 if (RT_SUCCESS(rc2))
491 {
492 /* Remove this session from the session list. */
493 RTListNodeRemove(&pThis->Node);
494
495 rc2 = RTCritSectLeave(&pCtx->CritSect);
496 if (RT_SUCCESS(rc))
497 rc = rc2;
498 }
499
500 LogFlowFunc(("Session %p: Terminated with rc=%Rrc, freeing ...\n", pThis, rc));
501
502 RTMemFree(pThis);
503 pThis = NULL;
504
505 return rc;
506}
507
508static int vboxIPCSessionCreate(PVBOXIPCCONTEXT pCtx, RTLOCALIPCSESSION hSession)
509{
510 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
511 AssertReturn(hSession != NIL_RTLOCALIPCSESSION, VERR_INVALID_PARAMETER);
512
513 int rc = RTCritSectEnter(&pCtx->CritSect);
514 if (RT_SUCCESS(rc))
515 {
516 PVBOXIPCSESSION pSession = (PVBOXIPCSESSION)RTMemAllocZ(sizeof(VBOXIPCSESSION));
517 if (pSession)
518 {
519 pSession->pCtx = pCtx;
520 pSession->hSession = hSession;
521 pSession->fTerminate = false;
522 pSession->hThread = NIL_RTTHREAD;
523
524 /* Start IPC session thread. */
525 LogFlowFunc(("Creating thread for session %p ...\n", pSession));
526 rc = RTThreadCreate(&pSession->hThread, vboxIPCSessionThread,
527 pSession /* pvUser */, 0 /* Default stack size */,
528 RTTHREADTYPE_DEFAULT, 0 /* Flags */, "IPCSESSION");
529 if (RT_SUCCESS(rc))
530 {
531 /* Add session thread to session IPC list. */
532 RTListAppend(&pCtx->SessionList, &pSession->Node);
533 }
534 else
535 {
536 int rc2 = RTLocalIpcSessionClose(hSession);
537 if (RT_FAILURE(rc2))
538 LogFlowFunc(("Failed closing session %p, rc=%Rrc\n", pSession, rc2));
539
540 LogFlowFunc(("Failed to create thread for session %p, rc=%Rrc\n", pSession, rc));
541 RTMemFree(pSession);
542 }
543 }
544 else
545 rc = VERR_NO_MEMORY;
546
547 int rc2 = RTCritSectLeave(&pCtx->CritSect);
548 AssertRC(rc2);
549 }
550
551 return rc;
552}
553
554static int vboxIPCSessionStop(PVBOXIPCSESSION pSession)
555{
556 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
557
558 ASMAtomicWriteBool(&pSession->fTerminate, true);
559
560 RTLOCALIPCSESSION hSession;
561 ASMAtomicXchgHandle(&pSession->hSession, NIL_RTLOCALIPCSESSION, &hSession);
562 if (hSession)
563 return RTLocalIpcSessionClose(hSession);
564
565 return VINF_SUCCESS;
566}
567
568/**
569 * Thread function to wait for and process seamless mode change
570 * requests
571 */
572DECLCALLBACK(int) vbtrIPCWorker(void *pInstance, bool volatile *pfShutdown)
573{
574 AssertPtr(pInstance);
575 LogFlowFunc(("pInstance=%p\n", pInstance));
576
577 LogFlowFuncEnter();
578
579 /*
580 * Tell the control thread that it can continue
581 * spawning services.
582 */
583 RTThreadUserSignal(RTThreadSelf());
584
585 PVBOXIPCCONTEXT pCtx = (PVBOXIPCCONTEXT)pInstance;
586 AssertPtr(pCtx);
587
588 int rc;
589
590 bool fShutdown = false;
591 for (;;)
592 {
593 RTLOCALIPCSESSION hClientSession = NIL_RTLOCALIPCSESSION;
594 rc = RTLocalIpcServerListen(pCtx->hServer, &hClientSession);
595 if (RT_FAILURE(rc))
596 {
597 if (rc == VERR_CANCELLED)
598 {
599 LogFlow(("Cancelled\n"));
600 fShutdown = true;
601 }
602 else
603 LogRelFunc(("Listening failed with rc=%Rrc\n", rc));
604 }
605
606 if (fShutdown)
607 break;
608 rc = vboxIPCSessionCreate(pCtx, hClientSession);
609 if (RT_FAILURE(rc))
610 {
611 LogRelFunc(("Creating new IPC server session failed with rc=%Rrc\n", rc));
612 /* Keep going. */
613 }
614
615 if (*pfShutdown)
616 break;
617 }
618
619 LogFlowFuncLeaveRC(rc);
620 return rc;
621}
622
623/**
624 * The service description.
625 */
626VBOXTRAYSVCDESC g_SvcDescIPC =
627{
628 /* pszName. */
629 "IPC",
630 /* pszDescription. */
631 "Inter-Process Communication",
632 /* pszUsage. */
633 NULL,
634 /* pszOptions. */
635 NULL,
636 /* methods */
637 vbtrIPCPreInit,
638 vbtrIPCOption,
639 vbtrIPCInit,
640 vbtrIPCWorker,
641 NULL /* pfnStop */,
642 vbtrIPCDestroy
643};
644
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