VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/ClientWatcher.cpp@ 58414

Last change on this file since 58414 was 56958, checked in by vboxsync, 9 years ago

ClientWatcher.cpp: Deal with possible race between watcher getting the session PID and the session/who-ever reporting the PID.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.7 KB
Line 
1/* $Id: ClientWatcher.cpp 56958 2015-07-16 17:29:44Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox API client session crash watcher
5 */
6
7/*
8 * Copyright (C) 2006-2014 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include <iprt/asm.h>
20#include <iprt/assert.h>
21#include <iprt/log.h>
22#include <iprt/semaphore.h>
23#include <iprt/process.h>
24
25#include <VBox/com/defs.h>
26
27#include <vector>
28
29#include "VirtualBoxBase.h"
30#include "AutoCaller.h"
31#include "ClientWatcher.h"
32#include "ClientToken.h"
33#include "VirtualBoxImpl.h"
34#include "MachineImpl.h"
35
36#if defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER) || defined(VBOX_WITH_GENERIC_SESSION_WATCHER)
37/** Table for adaptive timeouts. After an update the counter starts at the
38 * maximum value and decreases to 0, i.e. first the short timeouts are used
39 * and then the longer ones. This minimizes the detection latency in the
40 * cases where a change is expected, for crashes. */
41static const RTMSINTERVAL s_aUpdateTimeoutSteps[] = { 500, 200, 100, 50, 20, 10, 5 };
42#endif
43
44
45
46VirtualBox::ClientWatcher::ClientWatcher() :
47 mLock(LOCKCLASS_OBJECTSTATE)
48{
49 AssertReleaseFailed();
50}
51
52VirtualBox::ClientWatcher::~ClientWatcher()
53{
54 if (mThread != NIL_RTTHREAD)
55 {
56 /* signal the client watcher thread, should be exiting now */
57 update();
58 /* wait for termination */
59 RTThreadWait(mThread, RT_INDEFINITE_WAIT, NULL);
60 mThread = NIL_RTTHREAD;
61 }
62 mProcesses.clear();
63#if defined(RT_OS_WINDOWS)
64 if (mUpdateReq != NULL)
65 {
66 ::CloseHandle(mUpdateReq);
67 mUpdateReq = NULL;
68 }
69#elif defined(RT_OS_OS2) || defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER) || defined(VBOX_WITH_GENERIC_SESSION_WATCHER)
70 if (mUpdateReq != NIL_RTSEMEVENT)
71 {
72 RTSemEventDestroy(mUpdateReq);
73 mUpdateReq = NIL_RTSEMEVENT;
74 }
75#else
76# error "Port me!"
77#endif
78}
79
80VirtualBox::ClientWatcher::ClientWatcher(const ComObjPtr<VirtualBox> &pVirtualBox) :
81 mVirtualBox(pVirtualBox),
82 mThread(NIL_RTTHREAD),
83 mUpdateReq(CWUPDATEREQARG),
84 mLock(LOCKCLASS_OBJECTSTATE)
85{
86#if defined(RT_OS_WINDOWS)
87 mUpdateReq = ::CreateEvent(NULL, FALSE, FALSE, NULL);
88#elif defined(RT_OS_OS2)
89 RTSemEventCreate(&mUpdateReq);
90#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER) || defined(VBOX_WITH_GENERIC_SESSION_WATCHER)
91 RTSemEventCreate(&mUpdateReq);
92 /* start with high timeouts, nothing to do */
93 ASMAtomicUoWriteU8(&mUpdateAdaptCtr, 0);
94#else
95# error "Port me!"
96#endif
97
98 int vrc = RTThreadCreate(&mThread,
99 worker,
100 (void *)this,
101 0,
102 RTTHREADTYPE_MAIN_WORKER,
103 RTTHREADFLAGS_WAITABLE,
104 "Watcher");
105 AssertRC(vrc);
106}
107
108bool VirtualBox::ClientWatcher::isReady()
109{
110 return mThread != NIL_RTTHREAD;
111}
112
113/**
114 * Sends a signal to the thread to rescan the clients/VMs having open sessions.
115 */
116void VirtualBox::ClientWatcher::update()
117{
118 AssertReturnVoid(mThread != NIL_RTTHREAD);
119
120 /* sent an update request */
121#if defined(RT_OS_WINDOWS)
122 ::SetEvent(mUpdateReq);
123#elif defined(RT_OS_OS2)
124 RTSemEventSignal(mUpdateReq);
125#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
126 /* use short timeouts, as we expect changes */
127 ASMAtomicUoWriteU8(&mUpdateAdaptCtr, RT_ELEMENTS(s_aUpdateTimeoutSteps) - 1);
128 RTSemEventSignal(mUpdateReq);
129#elif defined(VBOX_WITH_GENERIC_SESSION_WATCHER)
130 RTSemEventSignal(mUpdateReq);
131#else
132# error "Port me!"
133#endif
134}
135
136/**
137 * Adds a process to the list of processes to be reaped. This call should be
138 * followed by a call to update() to cause the necessary actions immediately,
139 * in case the process crashes straight away.
140 */
141void VirtualBox::ClientWatcher::addProcess(RTPROCESS pid)
142{
143 AssertReturnVoid(mThread != NIL_RTTHREAD);
144 /* @todo r=klaus, do the reaping on all platforms! */
145#ifndef RT_OS_WINDOWS
146 AutoWriteLock alock(mLock COMMA_LOCKVAL_SRC_POS);
147 mProcesses.push_back(pid);
148#endif
149}
150
151/**
152 * Thread worker function that watches the termination of all client processes
153 * that have open sessions using IMachine::LockMachine()
154 */
155/*static*/
156DECLCALLBACK(int) VirtualBox::ClientWatcher::worker(RTTHREAD /* thread */, void *pvUser)
157{
158 LogFlowFuncEnter();
159
160 VirtualBox::ClientWatcher *that = (VirtualBox::ClientWatcher *)pvUser;
161 Assert(that);
162
163 typedef std::vector<ComObjPtr<Machine> > MachineVector;
164 typedef std::vector<ComObjPtr<SessionMachine> > SessionMachineVector;
165
166 SessionMachineVector machines;
167 MachineVector spawnedMachines;
168
169 size_t cnt = 0;
170 size_t cntSpawned = 0;
171
172 VirtualBoxBase::initializeComForThread();
173
174#if defined(RT_OS_WINDOWS)
175
176 /// @todo (dmik) processes reaping!
177
178 HANDLE handles[MAXIMUM_WAIT_OBJECTS];
179 handles[0] = that->mUpdateReq;
180
181 do
182 {
183 AutoCaller autoCaller(that->mVirtualBox);
184 /* VirtualBox has been early uninitialized, terminate */
185 if (!autoCaller.isOk())
186 break;
187
188 bool fPidRace = false;
189 do
190 {
191 /* release the caller to let uninit() ever proceed */
192 autoCaller.release();
193
194 DWORD rc = ::WaitForMultipleObjects((DWORD)(1 + cnt + cntSpawned),
195 handles,
196 FALSE,
197 !fPidRace ? INFINITE : 500);
198
199 /* Restore the caller before using VirtualBox. If it fails, this
200 * means VirtualBox is being uninitialized and we must terminate. */
201 autoCaller.add();
202 if (!autoCaller.isOk())
203 break;
204
205 bool update = fPidRace;
206
207 if (rc == WAIT_OBJECT_0)
208 {
209 /* update event is signaled */
210 update = true;
211 }
212 else if (rc > WAIT_OBJECT_0 && rc <= (WAIT_OBJECT_0 + cnt))
213 {
214 /* machine mutex is released */
215 (machines[rc - WAIT_OBJECT_0 - 1])->i_checkForDeath();
216 update = true;
217 }
218 else if (rc > WAIT_ABANDONED_0 && rc <= (WAIT_ABANDONED_0 + cnt))
219 {
220 /* machine mutex is abandoned due to client process termination */
221 (machines[rc - WAIT_ABANDONED_0 - 1])->i_checkForDeath();
222 update = true;
223 }
224 else if (rc > WAIT_OBJECT_0 + cnt && rc <= (WAIT_OBJECT_0 + cntSpawned))
225 {
226 /* spawned VM process has terminated (normally or abnormally) */
227 (spawnedMachines[rc - WAIT_OBJECT_0 - cnt - 1])->
228 i_checkForSpawnFailure();
229 update = true;
230 }
231
232 if (update)
233 {
234 /* close old process handles */
235 for (size_t i = 1 + cnt; i < 1 + cnt + cntSpawned; ++i)
236 CloseHandle(handles[i]);
237
238 // get reference to the machines list in VirtualBox
239 VirtualBox::MachinesOList &allMachines = that->mVirtualBox->i_getMachinesList();
240
241 // lock the machines list for reading
242 AutoReadLock thatLock(allMachines.getLockHandle() COMMA_LOCKVAL_SRC_POS);
243
244 /* obtain a new set of opened machines */
245 cnt = 0;
246 machines.clear();
247
248 for (MachinesOList::iterator it = allMachines.begin();
249 it != allMachines.end();
250 ++it)
251 {
252 /// @todo handle situations with more than 64 objects
253 AssertMsgBreak((1 + cnt) <= MAXIMUM_WAIT_OBJECTS,
254 ("MAXIMUM_WAIT_OBJECTS reached"));
255
256 ComObjPtr<SessionMachine> sm;
257 if ((*it)->i_isSessionOpenOrClosing(sm))
258 {
259 AutoCaller smCaller(sm);
260 if (smCaller.isOk())
261 {
262 AutoReadLock smLock(sm COMMA_LOCKVAL_SRC_POS);
263 Machine::ClientToken *ct = sm->i_getClientToken();
264 if (ct)
265 {
266 HANDLE ipcSem = ct->getToken();
267 machines.push_back(sm);
268 handles[1 + cnt] = ipcSem;
269 ++cnt;
270 }
271 }
272 }
273 }
274
275 LogFlowFunc(("UPDATE: direct session count = %d\n", cnt));
276
277 /* obtain a new set of spawned machines */
278 fPidRace = false;
279 cntSpawned = 0;
280 spawnedMachines.clear();
281
282 for (MachinesOList::iterator it = allMachines.begin();
283 it != allMachines.end();
284 ++it)
285 {
286 /// @todo handle situations with more than 64 objects
287 AssertMsgBreak((1 + cnt + cntSpawned) <= MAXIMUM_WAIT_OBJECTS,
288 ("MAXIMUM_WAIT_OBJECTS reached"));
289
290 if ((*it)->i_isSessionSpawning())
291 {
292 ULONG pid;
293 HRESULT hrc = (*it)->COMGETTER(SessionPID)(&pid);
294 if (SUCCEEDED(hrc))
295 {
296 if (pid != NIL_RTPROCESS)
297 {
298 HANDLE hProc = OpenProcess(SYNCHRONIZE, FALSE, pid);
299 AssertMsg(hProc != NULL, ("OpenProcess (pid=%d) failed with %d\n", pid, GetLastError()));
300 if (hProc != NULL)
301 {
302 spawnedMachines.push_back(*it);
303 handles[1 + cnt + cntSpawned] = hProc;
304 ++cntSpawned;
305 }
306 }
307 else
308 fPidRace = true;
309 }
310 }
311 }
312
313 LogFlowFunc(("UPDATE: spawned session count = %d\n", cntSpawned));
314
315 // machines lock unwinds here
316 }
317 }
318 while (true);
319 }
320 while (0);
321
322 /* close old process handles */
323 for (size_t i = 1 + cnt; i < 1 + cnt + cntSpawned; ++i)
324 CloseHandle(handles[i]);
325
326 /* release sets of machines if any */
327 machines.clear();
328 spawnedMachines.clear();
329
330 ::CoUninitialize();
331
332#elif defined(RT_OS_OS2)
333
334 /// @todo (dmik) processes reaping!
335
336 /* according to PMREF, 64 is the maximum for the muxwait list */
337 SEMRECORD handles[64];
338
339 HMUX muxSem = NULLHANDLE;
340
341 do
342 {
343 AutoCaller autoCaller(that->mVirtualBox);
344 /* VirtualBox has been early uninitialized, terminate */
345 if (!autoCaller.isOk())
346 break;
347
348 do
349 {
350 /* release the caller to let uninit() ever proceed */
351 autoCaller.release();
352
353 int vrc = RTSemEventWait(that->mUpdateReq, 500);
354
355 /* Restore the caller before using VirtualBox. If it fails, this
356 * means VirtualBox is being uninitialized and we must terminate. */
357 autoCaller.add();
358 if (!autoCaller.isOk())
359 break;
360
361 bool update = false;
362 bool updateSpawned = false;
363
364 if (RT_SUCCESS(vrc))
365 {
366 /* update event is signaled */
367 update = true;
368 updateSpawned = true;
369 }
370 else
371 {
372 AssertMsg(vrc == VERR_TIMEOUT || vrc == VERR_INTERRUPTED,
373 ("RTSemEventWait returned %Rrc\n", vrc));
374
375 /* are there any mutexes? */
376 if (cnt > 0)
377 {
378 /* figure out what's going on with machines */
379
380 unsigned long semId = 0;
381 APIRET arc = ::DosWaitMuxWaitSem(muxSem,
382 SEM_IMMEDIATE_RETURN, &semId);
383
384 if (arc == NO_ERROR)
385 {
386 /* machine mutex is normally released */
387 Assert(semId >= 0 && semId < cnt);
388 if (semId >= 0 && semId < cnt)
389 {
390#if 0//def DEBUG
391 {
392 AutoReadLock machineLock(machines[semId] COMMA_LOCKVAL_SRC_POS);
393 LogFlowFunc(("released mutex: machine='%ls'\n",
394 machines[semId]->name().raw()));
395 }
396#endif
397 machines[semId]->i_checkForDeath();
398 }
399 update = true;
400 }
401 else if (arc == ERROR_SEM_OWNER_DIED)
402 {
403 /* machine mutex is abandoned due to client process
404 * termination; find which mutex is in the Owner Died
405 * state */
406 for (size_t i = 0; i < cnt; ++i)
407 {
408 PID pid; TID tid;
409 unsigned long reqCnt;
410 arc = DosQueryMutexSem((HMTX)handles[i].hsemCur, &pid, &tid, &reqCnt);
411 if (arc == ERROR_SEM_OWNER_DIED)
412 {
413 /* close the dead mutex as asked by PMREF */
414 ::DosCloseMutexSem((HMTX)handles[i].hsemCur);
415
416 Assert(i >= 0 && i < cnt);
417 if (i >= 0 && i < cnt)
418 {
419#if 0//def DEBUG
420 {
421 AutoReadLock machineLock(machines[semId] COMMA_LOCKVAL_SRC_POS);
422 LogFlowFunc(("mutex owner dead: machine='%ls'\n",
423 machines[i]->name().raw()));
424 }
425#endif
426 machines[i]->i_checkForDeath();
427 }
428 }
429 }
430 update = true;
431 }
432 else
433 AssertMsg(arc == ERROR_INTERRUPT || arc == ERROR_TIMEOUT,
434 ("DosWaitMuxWaitSem returned %d\n", arc));
435 }
436
437 /* are there any spawning sessions? */
438 if (cntSpawned > 0)
439 {
440 for (size_t i = 0; i < cntSpawned; ++i)
441 updateSpawned |= (spawnedMachines[i])->
442 i_checkForSpawnFailure();
443 }
444 }
445
446 if (update || updateSpawned)
447 {
448 // get reference to the machines list in VirtualBox
449 VirtualBox::MachinesOList &allMachines = that->mVirtualBox->i_getMachinesList();
450
451 // lock the machines list for reading
452 AutoReadLock thatLock(allMachines.getLockHandle() COMMA_LOCKVAL_SRC_POS);
453
454 if (update)
455 {
456 /* close the old muxsem */
457 if (muxSem != NULLHANDLE)
458 ::DosCloseMuxWaitSem(muxSem);
459
460 /* obtain a new set of opened machines */
461 cnt = 0;
462 machines.clear();
463
464 for (MachinesOList::iterator it = allMachines.begin();
465 it != allMachines.end(); ++it)
466 {
467 /// @todo handle situations with more than 64 objects
468 AssertMsg(cnt <= 64 /* according to PMREF */,
469 ("maximum of 64 mutex semaphores reached (%d)",
470 cnt));
471
472 ComObjPtr<SessionMachine> sm;
473 if ((*it)->i_isSessionOpenOrClosing(sm))
474 {
475 AutoCaller smCaller(sm);
476 if (smCaller.isOk())
477 {
478 AutoReadLock smLock(sm COMMA_LOCKVAL_SRC_POS);
479 ClientToken *ct = sm->i_getClientToken();
480 if (ct)
481 {
482 HMTX ipcSem = ct->getToken();
483 machines.push_back(sm);
484 handles[cnt].hsemCur = (HSEM)ipcSem;
485 handles[cnt].ulUser = cnt;
486 ++cnt;
487 }
488 }
489 }
490 }
491
492 LogFlowFunc(("UPDATE: direct session count = %d\n", cnt));
493
494 if (cnt > 0)
495 {
496 /* create a new muxsem */
497 APIRET arc = ::DosCreateMuxWaitSem(NULL, &muxSem, cnt,
498 handles,
499 DCMW_WAIT_ANY);
500 AssertMsg(arc == NO_ERROR,
501 ("DosCreateMuxWaitSem returned %d\n", arc));
502 NOREF(arc);
503 }
504 }
505
506 if (updateSpawned)
507 {
508 /* obtain a new set of spawned machines */
509 spawnedMachines.clear();
510
511 for (MachinesOList::iterator it = allMachines.begin();
512 it != allMachines.end(); ++it)
513 {
514 if ((*it)->i_isSessionSpawning())
515 spawnedMachines.push_back(*it);
516 }
517
518 cntSpawned = spawnedMachines.size();
519 LogFlowFunc(("UPDATE: spawned session count = %d\n", cntSpawned));
520 }
521 }
522 }
523 while (true);
524 }
525 while (0);
526
527 /* close the muxsem */
528 if (muxSem != NULLHANDLE)
529 ::DosCloseMuxWaitSem(muxSem);
530
531 /* release sets of machines if any */
532 machines.clear();
533 spawnedMachines.clear();
534
535#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
536
537 bool update = false;
538 bool updateSpawned = false;
539
540 do
541 {
542 AutoCaller autoCaller(that->mVirtualBox);
543 if (!autoCaller.isOk())
544 break;
545
546 do
547 {
548 /* release the caller to let uninit() ever proceed */
549 autoCaller.release();
550
551 /* determine wait timeout adaptively: after updating information
552 * relevant to the client watcher, check a few times more
553 * frequently. This ensures good reaction time when the signalling
554 * has to be done a bit before the actual change for technical
555 * reasons, and saves CPU cycles when no activities are expected. */
556 RTMSINTERVAL cMillies;
557 {
558 uint8_t uOld, uNew;
559 do
560 {
561 uOld = ASMAtomicUoReadU8(&that->mUpdateAdaptCtr);
562 uNew = uOld ? uOld - 1 : uOld;
563 } while (!ASMAtomicCmpXchgU8(&that->mUpdateAdaptCtr, uNew, uOld));
564 Assert(uOld <= RT_ELEMENTS(s_aUpdateTimeoutSteps) - 1);
565 cMillies = s_aUpdateTimeoutSteps[uOld];
566 }
567
568 int rc = RTSemEventWait(that->mUpdateReq, cMillies);
569
570 /*
571 * Restore the caller before using VirtualBox. If it fails, this
572 * means VirtualBox is being uninitialized and we must terminate.
573 */
574 autoCaller.add();
575 if (!autoCaller.isOk())
576 break;
577
578 if (RT_SUCCESS(rc) || update || updateSpawned)
579 {
580 /* RT_SUCCESS(rc) means an update event is signaled */
581
582 // get reference to the machines list in VirtualBox
583 VirtualBox::MachinesOList &allMachines = that->mVirtualBox->i_getMachinesList();
584
585 // lock the machines list for reading
586 AutoReadLock thatLock(allMachines.getLockHandle() COMMA_LOCKVAL_SRC_POS);
587
588 if (RT_SUCCESS(rc) || update)
589 {
590 /* obtain a new set of opened machines */
591 machines.clear();
592
593 for (MachinesOList::iterator it = allMachines.begin();
594 it != allMachines.end();
595 ++it)
596 {
597 ComObjPtr<SessionMachine> sm;
598 if ((*it)->i_isSessionOpenOrClosing(sm))
599 machines.push_back(sm);
600 }
601
602 cnt = machines.size();
603 LogFlowFunc(("UPDATE: direct session count = %d\n", cnt));
604 }
605
606 if (RT_SUCCESS(rc) || updateSpawned)
607 {
608 /* obtain a new set of spawned machines */
609 spawnedMachines.clear();
610
611 for (MachinesOList::iterator it = allMachines.begin();
612 it != allMachines.end();
613 ++it)
614 {
615 if ((*it)->i_isSessionSpawning())
616 spawnedMachines.push_back(*it);
617 }
618
619 cntSpawned = spawnedMachines.size();
620 LogFlowFunc(("UPDATE: spawned session count = %d\n", cntSpawned));
621 }
622
623 // machines lock unwinds here
624 }
625
626 update = false;
627 for (size_t i = 0; i < cnt; ++i)
628 update |= (machines[i])->i_checkForDeath();
629
630 updateSpawned = false;
631 for (size_t i = 0; i < cntSpawned; ++i)
632 updateSpawned |= (spawnedMachines[i])->i_checkForSpawnFailure();
633
634 /* reap child processes */
635 {
636 AutoWriteLock alock(that->mLock COMMA_LOCKVAL_SRC_POS);
637 if (that->mProcesses.size())
638 {
639 LogFlowFunc(("UPDATE: child process count = %d\n",
640 that->mProcesses.size()));
641 VirtualBox::ClientWatcher::ProcessList::iterator it = that->mProcesses.begin();
642 while (it != that->mProcesses.end())
643 {
644 RTPROCESS pid = *it;
645 RTPROCSTATUS status;
646 int vrc = ::RTProcWait(pid, RTPROCWAIT_FLAGS_NOBLOCK, &status);
647 if (vrc == VINF_SUCCESS)
648 {
649 if ( status.enmReason != RTPROCEXITREASON_NORMAL
650 || status.iStatus != RTEXITCODE_SUCCESS)
651 {
652 switch (status.enmReason)
653 {
654 default:
655 case RTPROCEXITREASON_NORMAL:
656 LogRel(("Reaper: Pid %d (%x) exited normally: %d (%#x)\n",
657 pid, pid, status.iStatus, status.iStatus));
658 break;
659 case RTPROCEXITREASON_ABEND:
660 LogRel(("Reaper: Pid %d (%x) abended: %d (%#x)\n",
661 pid, pid, status.iStatus, status.iStatus));
662 break;
663 case RTPROCEXITREASON_SIGNAL:
664 LogRel(("Reaper: Pid %d (%x) was signalled: %d (%#x)\n",
665 pid, pid, status.iStatus, status.iStatus));
666 break;
667 }
668 }
669 else
670 LogFlowFunc(("pid %d (%x) was reaped, status=%d, reason=%d\n",
671 pid, pid, status.iStatus,
672 status.enmReason));
673 it = that->mProcesses.erase(it);
674 }
675 else
676 {
677 LogFlowFunc(("pid %d (%x) was NOT reaped, vrc=%Rrc\n",
678 pid, pid, vrc));
679 if (vrc != VERR_PROCESS_RUNNING)
680 {
681 /* remove the process if it is not already running */
682 it = that->mProcesses.erase(it);
683 }
684 else
685 ++it;
686 }
687 }
688 }
689 }
690 }
691 while (true);
692 }
693 while (0);
694
695 /* release sets of machines if any */
696 machines.clear();
697 spawnedMachines.clear();
698
699#elif defined(VBOX_WITH_GENERIC_SESSION_WATCHER)
700
701 bool updateSpawned = false;
702
703 do
704 {
705 AutoCaller autoCaller(that->mVirtualBox);
706 if (!autoCaller.isOk())
707 break;
708
709 do
710 {
711 /* release the caller to let uninit() ever proceed */
712 autoCaller.release();
713
714 /* determine wait timeout adaptively: after updating information
715 * relevant to the client watcher, check a few times more
716 * frequently. This ensures good reaction time when the signalling
717 * has to be done a bit before the actual change for technical
718 * reasons, and saves CPU cycles when no activities are expected. */
719 RTMSINTERVAL cMillies;
720 {
721 uint8_t uOld, uNew;
722 do
723 {
724 uOld = ASMAtomicUoReadU8(&that->mUpdateAdaptCtr);
725 uNew = uOld ? uOld - 1 : uOld;
726 } while (!ASMAtomicCmpXchgU8(&that->mUpdateAdaptCtr, uNew, uOld));
727 Assert(uOld <= RT_ELEMENTS(s_aUpdateTimeoutSteps) - 1);
728 cMillies = s_aUpdateTimeoutSteps[uOld];
729 }
730
731 int rc = RTSemEventWait(that->mUpdateReq, cMillies);
732
733 /*
734 * Restore the caller before using VirtualBox. If it fails, this
735 * means VirtualBox is being uninitialized and we must terminate.
736 */
737 autoCaller.add();
738 if (!autoCaller.isOk())
739 break;
740
741 /** @todo this quite big effort for catching machines in spawning
742 * state which can't be caught by the token mechanism (as the token
743 * can't be in the other process yet) could be eliminated if the
744 * reaping is made smarter, having cross-reference information
745 * from the pid to the corresponding machine object. Both cases do
746 * more or less the same thing anyway. */
747 if (RT_SUCCESS(rc) || updateSpawned)
748 {
749 /* RT_SUCCESS(rc) means an update event is signaled */
750
751 // get reference to the machines list in VirtualBox
752 VirtualBox::MachinesOList &allMachines = that->mVirtualBox->i_getMachinesList();
753
754 // lock the machines list for reading
755 AutoReadLock thatLock(allMachines.getLockHandle() COMMA_LOCKVAL_SRC_POS);
756
757 if (RT_SUCCESS(rc) || updateSpawned)
758 {
759 /* obtain a new set of spawned machines */
760 spawnedMachines.clear();
761
762 for (MachinesOList::iterator it = allMachines.begin();
763 it != allMachines.end();
764 ++it)
765 {
766 if ((*it)->i_isSessionSpawning())
767 spawnedMachines.push_back(*it);
768 }
769
770 cntSpawned = spawnedMachines.size();
771 LogFlowFunc(("UPDATE: spawned session count = %d\n", cntSpawned));
772 }
773
774 NOREF(cnt);
775 // machines lock unwinds here
776 }
777
778 updateSpawned = false;
779 for (size_t i = 0; i < cntSpawned; ++i)
780 updateSpawned |= (spawnedMachines[i])->i_checkForSpawnFailure();
781
782 /* reap child processes */
783 {
784 AutoWriteLock alock(that->mLock COMMA_LOCKVAL_SRC_POS);
785 if (that->mProcesses.size())
786 {
787 LogFlowFunc(("UPDATE: child process count = %d\n",
788 that->mProcesses.size()));
789 VirtualBox::ClientWatcher::ProcessList::iterator it = that->mProcesses.begin();
790 while (it != that->mProcesses.end())
791 {
792 RTPROCESS pid = *it;
793 RTPROCSTATUS status;
794 int vrc = ::RTProcWait(pid, RTPROCWAIT_FLAGS_NOBLOCK, &status);
795 if (vrc == VINF_SUCCESS)
796 {
797 if ( status.enmReason != RTPROCEXITREASON_NORMAL
798 || status.iStatus != RTEXITCODE_SUCCESS)
799 {
800 switch (status.enmReason)
801 {
802 default:
803 case RTPROCEXITREASON_NORMAL:
804 LogRel(("Reaper: Pid %d (%x) exited normally: %d (%#x)\n",
805 pid, pid, status.iStatus, status.iStatus));
806 break;
807 case RTPROCEXITREASON_ABEND:
808 LogRel(("Reaper: Pid %d (%x) abended: %d (%#x)\n",
809 pid, pid, status.iStatus, status.iStatus));
810 break;
811 case RTPROCEXITREASON_SIGNAL:
812 LogRel(("Reaper: Pid %d (%x) was signalled: %d (%#x)\n",
813 pid, pid, status.iStatus, status.iStatus));
814 break;
815 }
816 }
817 else
818 LogFlowFunc(("pid %d (%x) was reaped, status=%d, reason=%d\n",
819 pid, pid, status.iStatus,
820 status.enmReason));
821 it = that->mProcesses.erase(it);
822 }
823 else
824 {
825 LogFlowFunc(("pid %d (%x) was NOT reaped, vrc=%Rrc\n",
826 pid, pid, vrc));
827 if (vrc != VERR_PROCESS_RUNNING)
828 {
829 /* remove the process if it is not already running */
830 it = that->mProcesses.erase(it);
831 }
832 else
833 ++it;
834 }
835 }
836 }
837 }
838 }
839 while (true);
840 }
841 while (0);
842
843 /* release sets of machines if any */
844 machines.clear();
845 spawnedMachines.clear();
846
847#else
848# error "Port me!"
849#endif
850
851 VirtualBoxBase::uninitializeComForThread();
852
853 LogFlowFuncLeave();
854 return 0;
855}
856/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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