VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/SessionImpl.cpp@ 107402

Last change on this file since 107402 was 106976, checked in by vboxsync, 2 months ago

Added tracking for medium, machine, session, progress objects.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 41.5 KB
Line 
1/* $Id: SessionImpl.cpp 106976 2024-11-13 08:52:53Z vboxsync $ */
2/** @file
3 * VBox Client Session COM Class implementation in VBoxC.
4 */
5
6/*
7 * Copyright (C) 2006-2024 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * SPDX-License-Identifier: GPL-3.0-only
26 */
27
28#define LOG_GROUP LOG_GROUP_MAIN_SESSION
29#include "LoggingNew.h"
30
31#include "SessionImpl.h"
32#include "ConsoleImpl.h"
33#include "ClientTokenHolder.h"
34#include "Global.h"
35#include "StringifyEnums.h"
36
37#include "AutoCaller.h"
38
39#include <iprt/errcore.h>
40#include <iprt/process.h>
41
42
43/**
44 * Local macro to check whether the session is open and return an error if not.
45 * @note Don't forget to do |Auto[Reader]Lock alock (this);| before using this
46 * macro.
47 */
48#define CHECK_OPEN() \
49 do { \
50 if (mState != SessionState_Locked) \
51 return setError(E_UNEXPECTED, Session::tr("The session is not locked (session state: %s)"), \
52 Global::stringifySessionState(mState)); \
53 } while (0)
54
55// constructor / destructor
56/////////////////////////////////////////////////////////////////////////////
57
58Session::Session()
59{
60}
61
62Session::~Session()
63{
64}
65
66HRESULT Session::FinalConstruct()
67{
68 LogFlowThisFunc(("\n"));
69
70 HRESULT hrc = init();
71
72 BaseFinalConstruct();
73
74 return hrc;
75}
76
77void Session::FinalRelease()
78{
79 LogFlowThisFunc(("\n"));
80
81 uninit();
82
83 BaseFinalRelease();
84}
85
86// public initializer/uninitializer for internal purposes only
87/////////////////////////////////////////////////////////////////////////////
88
89/**
90 * Initializes the Session object.
91 */
92HRESULT Session::init()
93{
94 /* Enclose the state transition NotReady->InInit->Ready */
95 AutoInitSpan autoInitSpan(this);
96 AssertReturn(autoInitSpan.isOk(), E_FAIL);
97
98 LogFlowThisFuncEnter();
99
100 mState = SessionState_Unlocked;
101 mType = SessionType_Null;
102
103 mClientTokenHolder = NULL;
104
105 setTracked(18000, 60);//5 hours, 1 hour
106
107 /* Confirm a successful initialization when it's the case */
108 autoInitSpan.setSucceeded();
109
110 LogFlowThisFuncLeave();
111
112 return S_OK;
113}
114
115/**
116 * Uninitializes the Session object.
117 *
118 * @note Locks this object for writing.
119 */
120void Session::uninit()
121{
122 LogFlowThisFuncEnter();
123
124 /* Enclose the state transition Ready->InUninit->NotReady */
125 AutoUninitSpan autoUninitSpan(this);
126 if (autoUninitSpan.uninitDone())
127 {
128 LogFlowThisFunc(("Already uninitialized.\n"));
129 LogFlowThisFuncLeave();
130 return;
131 }
132
133 /* close() needs write lock */
134 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
135
136 if (mState != SessionState_Unlocked)
137 {
138 Assert(mState == SessionState_Locked ||
139 mState == SessionState_Spawning);
140
141 HRESULT hrc = i_unlockMachine(true /* aFinalRelease */, false /* aFromServer */, alock);
142 AssertComRC(hrc);
143 }
144
145 LogFlowThisFuncLeave();
146}
147
148// ISession properties
149/////////////////////////////////////////////////////////////////////////////
150
151HRESULT Session::getState(SessionState_T *aState)
152{
153 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
154
155 *aState = mState;
156
157 return S_OK;
158}
159
160HRESULT Session::getType(SessionType_T *aType)
161{
162 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
163
164 CHECK_OPEN();
165
166 *aType = mType;
167 return S_OK;
168}
169
170HRESULT Session::getName(com::Utf8Str &aName)
171{
172 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
173
174 aName = mName;
175 return S_OK;
176}
177
178HRESULT Session::setName(const com::Utf8Str &aName)
179{
180 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
181
182 if (mState != SessionState_Unlocked)
183 return setError(VBOX_E_INVALID_OBJECT_STATE, tr("Trying to set name for a session which is not in state \"unlocked\""));
184
185 mName = aName;
186 return S_OK;
187}
188
189HRESULT Session::getMachine(ComPtr<IMachine> &aMachine)
190{
191 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
192
193 CHECK_OPEN();
194
195 HRESULT hrc;
196#ifndef VBOX_COM_INPROC_API_CLIENT
197 if (mConsole)
198 hrc = mConsole->i_machine().queryInterfaceTo(aMachine.asOutParam());
199 else
200#endif
201 hrc = mRemoteMachine.queryInterfaceTo(aMachine.asOutParam());
202 if (FAILED(hrc))
203 {
204#ifndef VBOX_COM_INPROC_API_CLIENT
205 if (mConsole)
206 setError(hrc, tr("Failed to query the session machine"));
207 else
208#endif
209 if (FAILED_DEAD_INTERFACE(hrc))
210 setError(hrc, tr("Peer process crashed"));
211 else
212 setError(hrc, tr("Failed to query the remote session machine"));
213 }
214
215 return hrc;
216}
217
218HRESULT Session::getConsole(ComPtr<IConsole> &aConsole)
219{
220 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
221
222 CHECK_OPEN();
223
224 HRESULT hrc = S_OK;
225#ifndef VBOX_COM_INPROC_API_CLIENT
226 if (mConsole)
227 hrc = mConsole.queryInterfaceTo(aConsole.asOutParam());
228 else
229#endif
230 hrc = mRemoteConsole.queryInterfaceTo(aConsole.asOutParam());
231
232 if (FAILED(hrc))
233 {
234#ifndef VBOX_COM_INPROC_API_CLIENT
235 if (mConsole)
236 setError(hrc, tr("Failed to query the console"));
237 else
238#endif
239 if (FAILED_DEAD_INTERFACE(hrc))
240 setError(hrc, tr("Peer process crashed"));
241 else
242 setError(hrc, tr("Failed to query the remote console"));
243 }
244
245 return hrc;
246}
247
248// ISession methods
249/////////////////////////////////////////////////////////////////////////////
250HRESULT Session::unlockMachine()
251{
252 LogFlowThisFunc(("mState=%d, mType=%d\n", mState, mType));
253
254 /* close() needs write lock */
255 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
256
257 CHECK_OPEN();
258 return i_unlockMachine(false /* aFinalRelease */, false /* aFromServer */, alock);
259}
260
261// IInternalSessionControl methods
262/////////////////////////////////////////////////////////////////////////////
263HRESULT Session::getPID(ULONG *aPid)
264{
265 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
266
267 *aPid = (ULONG)RTProcSelf();
268 AssertCompile(sizeof(*aPid) == sizeof(RTPROCESS));
269
270 return S_OK;
271}
272
273HRESULT Session::getRemoteConsole(ComPtr<IConsole> &aConsole)
274{
275 LogFlowThisFuncEnter();
276#ifndef VBOX_COM_INPROC_API_CLIENT
277 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
278
279 if (mType == SessionType_WriteLock && !!mConsole)
280 {
281 /* return a failure if the session already transitioned to Closing
282 * but the server hasn't processed Machine::OnSessionEnd() yet. */
283 if (mState == SessionState_Locked)
284 {
285 mConsole.queryInterfaceTo(aConsole.asOutParam());
286
287 LogFlowThisFuncLeave();
288 return S_OK;
289 }
290 return VBOX_E_INVALID_VM_STATE;
291 }
292 return setError(VBOX_E_INVALID_OBJECT_STATE, "This is not a direct session");
293
294#else /* VBOX_COM_INPROC_API_CLIENT */
295 RT_NOREF(aConsole);
296 AssertFailed();
297 return VBOX_E_INVALID_OBJECT_STATE;
298#endif /* VBOX_COM_INPROC_API_CLIENT */
299}
300
301HRESULT Session::getNominalState(MachineState_T *aNominalState)
302{
303 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
304 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
305 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
306#ifndef VBOX_COM_INPROC_API_CLIENT
307 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
308
309 return mConsole->i_getNominalState(*aNominalState);
310#else
311 RT_NOREF(aNominalState);
312 AssertFailed();
313 return E_NOTIMPL;
314#endif
315}
316
317#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
318HRESULT Session::assignMachine(const ComPtr<IMachine> &aMachine,
319 LockType_T aLockType,
320 const com::Utf8Str &aTokenId)
321#else
322HRESULT Session::assignMachine(const ComPtr<IMachine> &aMachine,
323 LockType_T aLockType,
324 const ComPtr<IToken> &aToken)
325#endif /* !VBOX_WITH_GENERIC_SESSION_WATCHER */
326{
327 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
328
329 AssertReturn(mState == SessionState_Unlocked, VBOX_E_INVALID_VM_STATE);
330
331 if (!aMachine)
332 {
333 /*
334 * A special case: the server informs us that this session has been
335 * passed to IMachine::launchVMProcess() so this session will become
336 * remote (but not existing) when AssignRemoteMachine() is called.
337 */
338
339 AssertReturn(mType == SessionType_Null, VBOX_E_INVALID_OBJECT_STATE);
340 mType = SessionType_Remote;
341 mState = SessionState_Spawning;
342
343 return S_OK;
344 }
345
346 /* query IInternalMachineControl interface */
347 mControl = aMachine;
348 AssertReturn(!!mControl, E_FAIL);
349
350 HRESULT hrc = S_OK;
351#ifndef VBOX_COM_INPROC_API_CLIENT
352 if (aLockType == LockType_VM)
353 {
354 /* This is what is special about VM processes: they have a Console
355 * object which is the root of all VM related activity. */
356 hrc = mConsole.createObject();
357 AssertComRCReturn(hrc, hrc);
358
359 hrc = mConsole->initWithMachine(aMachine, mControl, aLockType);
360 AssertComRCReturn(hrc, hrc);
361 }
362 else
363 mRemoteMachine = aMachine;
364#else
365 RT_NOREF(aLockType);
366 mRemoteMachine = aMachine;
367#endif
368
369#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
370 Utf8Str strTokenId(aTokenId);
371 Assert(!strTokenId.isEmpty());
372#else /* VBOX_WITH_GENERIC_SESSION_WATCHER */
373 Assert(!aToken.isNull());
374#endif /* VBOX_WITH_GENERIC_SESSION_WATCHER */
375 /* create the machine client token */
376 try
377 {
378#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
379 mClientTokenHolder = new ClientTokenHolder(strTokenId);
380#else /* VBOX_WITH_GENERIC_SESSION_WATCHER */
381 mClientTokenHolder = new ClientTokenHolder(aToken);
382#endif /* VBOX_WITH_GENERIC_SESSION_WATCHER */
383 if (!mClientTokenHolder->isReady())
384 {
385 delete mClientTokenHolder;
386 mClientTokenHolder = NULL;
387 hrc = E_FAIL;
388 }
389 }
390 catch (std::bad_alloc &)
391 {
392 hrc = E_OUTOFMEMORY;
393 }
394
395 /*
396 * Reference the VirtualBox object to ensure the server is up
397 * until the session is closed
398 */
399 if (SUCCEEDED(hrc))
400 hrc = aMachine->COMGETTER(Parent)(mVirtualBox.asOutParam());
401
402 if (SUCCEEDED(hrc))
403 {
404 mType = SessionType_WriteLock;
405 mState = SessionState_Locked;
406 }
407 else
408 {
409 /* some cleanup */
410 mControl.setNull();
411#ifndef VBOX_COM_INPROC_API_CLIENT
412 if (!mConsole.isNull())
413 {
414 mConsole->uninit();
415 mConsole.setNull();
416 }
417#endif
418 }
419
420 return hrc;
421}
422
423HRESULT Session::assignRemoteMachine(const ComPtr<IMachine> &aMachine,
424 const ComPtr<IConsole> &aConsole)
425
426{
427 AssertReturn(aMachine, E_INVALIDARG);
428
429 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
430
431 AssertReturn(mState == SessionState_Unlocked ||
432 mState == SessionState_Spawning, VBOX_E_INVALID_VM_STATE);
433
434 HRESULT hrc = E_FAIL;
435
436 /* query IInternalMachineControl interface */
437 mControl = aMachine;
438 AssertReturn(!!mControl, E_FAIL);
439
440 /// @todo (dmik)
441 // currently, the remote session returns the same machine and
442 // console objects as the direct session, thus giving the
443 // (remote) client full control over the direct session. For the
444 // console, it is the desired behavior (the ability to control
445 // VM execution is a must for the remote session). What about
446 // the machine object, we may want to prevent the remote client
447 // from modifying machine data. In this case, we must:
448 // 1) assign the Machine object (instead of the SessionMachine
449 // object that is passed to this method) to mRemoteMachine;
450 // 2) remove GetMachine() property from the IConsole interface
451 // because it always returns the SessionMachine object
452 // (alternatively, we can supply a separate IConsole
453 // implementation that will return the Machine object in
454 // response to GetMachine()).
455
456 mRemoteMachine = aMachine;
457 mRemoteConsole = aConsole;
458
459 /*
460 * Reference the VirtualBox object to ensure the server is up
461 * until the session is closed
462 */
463 hrc = aMachine->COMGETTER(Parent)(mVirtualBox.asOutParam());
464
465 if (SUCCEEDED(hrc))
466 {
467 /*
468 * RemoteSession type can be already set by AssignMachine() when its
469 * argument is NULL (a special case)
470 */
471 if (mType != SessionType_Remote)
472 mType = SessionType_Shared;
473 else
474 Assert(mState == SessionState_Spawning);
475
476 mState = SessionState_Locked;
477 }
478 else
479 {
480 /* some cleanup */
481 mControl.setNull();
482 mRemoteMachine.setNull();
483 mRemoteConsole.setNull();
484 }
485
486 LogFlowThisFunc(("hrc=%08X\n", hrc));
487 LogFlowThisFuncLeave();
488
489 return hrc;
490}
491
492HRESULT Session::updateMachineState(MachineState_T aMachineState)
493{
494
495 if (getObjectState().getState() != ObjectState::Ready)
496 {
497 /*
498 * We might have already entered Session::uninit() at this point, so
499 * return silently (not interested in the state change during uninit)
500 */
501 LogFlowThisFunc(("Already uninitialized.\n"));
502 return S_OK;
503 }
504
505 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
506
507 if (mState == SessionState_Unlocking)
508 {
509 LogFlowThisFunc(("Already being unlocked.\n"));
510 return S_OK;
511 }
512
513 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
514 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
515
516 AssertReturn(!mControl.isNull(), E_FAIL);
517#ifndef VBOX_COM_INPROC_API_CLIENT
518 AssertReturn(!mConsole.isNull(), E_FAIL);
519
520 return mConsole->i_updateMachineState(aMachineState);
521#else
522 RT_NOREF(aMachineState);
523 return S_OK;
524#endif
525}
526
527HRESULT Session::uninitialize()
528{
529 LogFlowThisFuncEnter();
530
531 AutoCaller autoCaller(this);
532
533 HRESULT hrc = S_OK;
534
535 if (getObjectState().getState() == ObjectState::Ready)
536 {
537 /* close() needs write lock */
538 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
539
540 LogFlowThisFunc(("mState=%s, mType=%d\n", ::stringifySessionState(mState), mType));
541
542 if (mState == SessionState_Unlocking)
543 {
544 LogFlowThisFunc(("Already being unlocked.\n"));
545 return S_OK;
546 }
547
548 if ( mState == SessionState_Locked
549 || mState == SessionState_Spawning)
550 { /* likely */ }
551 else
552 {
553#ifndef DEBUG_bird /* bird: hitting this all the time running tdAddBaseic1.py. */
554 AssertMsgFailed(("Session is in wrong state (%d), expected locked (%d) or spawning (%d)\n",
555 mState, SessionState_Locked, SessionState_Spawning));
556#endif
557 return VBOX_E_INVALID_VM_STATE;
558 }
559
560 /* close ourselves */
561 hrc = i_unlockMachine(false /* aFinalRelease */, true /* aFromServer */, alock);
562 }
563 else if (getObjectState().getState() == ObjectState::InUninit)
564 {
565 /*
566 * We might have already entered Session::uninit() at this point,
567 * return silently
568 */
569 LogFlowThisFunc(("Already uninitialized.\n"));
570 }
571 else
572 {
573 Log1WarningThisFunc(("UNEXPECTED uninitialization!\n"));
574 hrc = autoCaller.hrc();
575 }
576
577 LogFlowThisFunc(("hrc=%08X\n", hrc));
578 LogFlowThisFuncLeave();
579
580 return hrc;
581}
582
583HRESULT Session::onNetworkAdapterChange(const ComPtr<INetworkAdapter> &aNetworkAdapter,
584 BOOL aChangeAdapter)
585
586{
587 LogFlowThisFunc(("\n"));
588
589 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
590 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
591 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
592#ifndef VBOX_COM_INPROC_API_CLIENT
593 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
594
595 return mConsole->i_onNetworkAdapterChange(aNetworkAdapter, aChangeAdapter);
596#else
597 RT_NOREF(aNetworkAdapter, aChangeAdapter);
598 return S_OK;
599#endif
600}
601
602HRESULT Session::onAudioAdapterChange(const ComPtr<IAudioAdapter> &aAudioAdapter)
603{
604 LogFlowThisFunc(("\n"));
605
606 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
607 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
608 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
609#ifndef VBOX_COM_INPROC_API_CLIENT
610 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
611
612 return mConsole->i_onAudioAdapterChange(aAudioAdapter);
613#else
614 RT_NOREF(aAudioAdapter);
615 return S_OK;
616#endif
617
618}
619
620HRESULT Session::onHostAudioDeviceChange(const ComPtr<IHostAudioDevice> &aDevice,
621 BOOL aNew, AudioDeviceState_T aState,
622 const ComPtr<IVirtualBoxErrorInfo> &aErrInfo)
623{
624 LogFlowThisFunc(("\n"));
625
626 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
627 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
628 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
629#ifndef VBOX_COM_INPROC_API_CLIENT
630 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
631
632 return mConsole->i_onHostAudioDeviceChange(aDevice, aNew, aState, aErrInfo);
633#else
634 RT_NOREF(aDevice, aNew, aState, aErrInfo);
635 return S_OK;
636#endif
637}
638
639HRESULT Session::onSerialPortChange(const ComPtr<ISerialPort> &aSerialPort)
640{
641 LogFlowThisFunc(("\n"));
642
643 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
644 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
645 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
646#ifndef VBOX_COM_INPROC_API_CLIENT
647 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
648
649 return mConsole->i_onSerialPortChange(aSerialPort);
650#else
651 RT_NOREF(aSerialPort);
652 return S_OK;
653#endif
654}
655
656HRESULT Session::onParallelPortChange(const ComPtr<IParallelPort> &aParallelPort)
657{
658 LogFlowThisFunc(("\n"));
659
660 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
661 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
662 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
663#ifndef VBOX_COM_INPROC_API_CLIENT
664 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
665
666 return mConsole->i_onParallelPortChange(aParallelPort);
667#else
668 RT_NOREF(aParallelPort);
669 return S_OK;
670#endif
671}
672
673HRESULT Session::onStorageControllerChange(const Guid &aMachineId, const Utf8Str &aControllerName)
674{
675 LogFlowThisFunc(("\n"));
676
677 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
678 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
679 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
680#ifndef VBOX_COM_INPROC_API_CLIENT
681 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
682
683 return mConsole->i_onStorageControllerChange(aMachineId, aControllerName);
684#else
685 NOREF(aMachineId);
686 NOREF(aControllerName);
687 return S_OK;
688#endif
689}
690
691HRESULT Session::onMediumChange(const ComPtr<IMediumAttachment> &aMediumAttachment,
692 BOOL aForce)
693{
694 LogFlowThisFunc(("\n"));
695
696 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
697 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
698 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
699#ifndef VBOX_COM_INPROC_API_CLIENT
700 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
701
702 return mConsole->i_onMediumChange(aMediumAttachment, aForce);
703#else
704 RT_NOREF(aMediumAttachment, aForce);
705 return S_OK;
706#endif
707}
708
709HRESULT Session::onVMProcessPriorityChange(VMProcPriority_T priority)
710{
711 LogFlowThisFunc(("\n"));
712
713 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
714 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
715 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
716#ifndef VBOX_COM_INPROC_API_CLIENT
717 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
718
719 return mConsole->i_onVMProcessPriorityChange(priority);
720#else
721 RT_NOREF(priority);
722 return S_OK;
723#endif
724}
725
726HRESULT Session::onCPUChange(ULONG aCpu, BOOL aAdd)
727{
728 LogFlowThisFunc(("\n"));
729
730 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
731 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
732 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
733#ifndef VBOX_COM_INPROC_API_CLIENT
734 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
735
736 return mConsole->i_onCPUChange(aCpu, aAdd);
737#else
738 RT_NOREF(aCpu, aAdd);
739 return S_OK;
740#endif
741}
742
743HRESULT Session::onCPUExecutionCapChange(ULONG aExecutionCap)
744{
745 LogFlowThisFunc(("\n"));
746
747 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
748 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
749 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
750#ifndef VBOX_COM_INPROC_API_CLIENT
751 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
752
753 return mConsole->i_onCPUExecutionCapChange(aExecutionCap);
754#else
755 RT_NOREF(aExecutionCap);
756 return S_OK;
757#endif
758}
759
760HRESULT Session::onVRDEServerChange(BOOL aRestart)
761{
762 LogFlowThisFunc(("\n"));
763
764 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
765 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
766 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
767#ifndef VBOX_COM_INPROC_API_CLIENT
768 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
769
770 return mConsole->i_onVRDEServerChange(aRestart);
771#else
772 RT_NOREF(aRestart);
773 return S_OK;
774#endif
775}
776
777HRESULT Session::onRecordingStateChange(BOOL aEnable, ComPtr<IProgress> &aProgress)
778{
779 LogFlowThisFunc(("\n"));
780
781 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
782 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
783 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
784#ifndef VBOX_COM_INPROC_API_CLIENT
785 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
786
787 return mConsole->i_onRecordingStateChange(aEnable, aProgress);
788#else
789 RT_NOREF(aEnable, aProgress);
790 return S_OK;
791#endif
792}
793
794HRESULT Session::onRecordingScreenStateChange(BOOL aEnable, ULONG aScreen)
795{
796 LogFlowThisFunc(("\n"));
797
798 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
799 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
800 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
801#ifndef VBOX_COM_INPROC_API_CLIENT
802 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
803
804 return mConsole->i_onRecordingScreenStateChange(aEnable, aScreen);
805#else
806 RT_NOREF(aEnable, aScreen);
807 return S_OK;
808#endif
809}
810
811HRESULT Session::onUSBControllerChange()
812{
813 LogFlowThisFunc(("\n"));
814
815 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
816 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
817 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
818#ifndef VBOX_COM_INPROC_API_CLIENT
819 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
820
821 return mConsole->i_onUSBControllerChange();
822#else
823 return S_OK;
824#endif
825}
826
827HRESULT Session::onSharedFolderChange(BOOL aGlobal)
828{
829 LogFlowThisFunc(("\n"));
830
831 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
832 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
833 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
834#ifndef VBOX_COM_INPROC_API_CLIENT
835 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
836
837 return mConsole->i_onSharedFolderChange(aGlobal);
838#else
839 RT_NOREF(aGlobal);
840 return S_OK;
841#endif
842}
843
844HRESULT Session::onClipboardModeChange(ClipboardMode_T aClipboardMode)
845{
846 LogFlowThisFunc(("\n"));
847
848 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
849 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
850 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
851#ifndef VBOX_COM_INPROC_API_CLIENT
852 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
853
854 return mConsole->i_onClipboardModeChange(aClipboardMode);
855#else
856 RT_NOREF(aClipboardMode);
857 return S_OK;
858#endif
859}
860
861HRESULT Session::onClipboardFileTransferModeChange(BOOL aEnabled)
862{
863 LogFlowThisFunc(("\n"));
864
865 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
866 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
867 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
868#ifndef VBOX_COM_INPROC_API_CLIENT
869 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
870
871 return mConsole->i_onClipboardFileTransferModeChange(RT_BOOL(aEnabled));
872#else
873 RT_NOREF(aEnabled);
874 return S_OK;
875#endif
876}
877
878HRESULT Session::onDnDModeChange(DnDMode_T aDndMode)
879{
880 LogFlowThisFunc(("\n"));
881
882 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
883 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
884#ifndef VBOX_COM_INPROC_API_CLIENT
885 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
886 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
887
888 return mConsole->i_onDnDModeChange(aDndMode);
889#else
890 RT_NOREF(aDndMode);
891 return S_OK;
892#endif
893}
894
895HRESULT Session::onGuestDebugControlChange(const ComPtr<IGuestDebugControl> &aGuestDebugControl)
896{
897 LogFlowThisFunc(("\n"));
898
899 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
900 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
901 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
902#ifndef VBOX_COM_INPROC_API_CLIENT
903 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
904
905 return mConsole->i_onGuestDebugControlChange(aGuestDebugControl);
906#else
907 RT_NOREF(aGuestDebugControl);
908 return S_OK;
909#endif
910}
911
912HRESULT Session::onUSBDeviceAttach(const ComPtr<IUSBDevice> &aDevice,
913 const ComPtr<IVirtualBoxErrorInfo> &aError,
914 ULONG aMaskedInterfaces,
915 const com::Utf8Str &aCaptureFilename)
916{
917 LogFlowThisFunc(("\n"));
918
919 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
920 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
921 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
922#ifndef VBOX_COM_INPROC_API_CLIENT
923 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
924
925 return mConsole->i_onUSBDeviceAttach(aDevice, aError, aMaskedInterfaces, aCaptureFilename);
926#else
927 RT_NOREF(aDevice, aError, aMaskedInterfaces, aCaptureFilename);
928 return S_OK;
929#endif
930}
931
932HRESULT Session::onUSBDeviceDetach(const com::Guid &aId,
933 const ComPtr<IVirtualBoxErrorInfo> &aError)
934{
935 LogFlowThisFunc(("\n"));
936
937 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
938 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
939 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
940#ifndef VBOX_COM_INPROC_API_CLIENT
941 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
942
943 return mConsole->i_onUSBDeviceDetach(aId.toUtf16().raw(), aError);
944#else
945 RT_NOREF(aId, aError);
946 return S_OK;
947#endif
948}
949
950HRESULT Session::onShowWindow(BOOL aCheck, BOOL *aCanShow, LONG64 *aWinId)
951{
952 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
953
954 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
955#ifndef VBOX_COM_INPROC_API_CLIENT
956 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
957#endif
958
959 if (mState != SessionState_Locked)
960 {
961 /* the call from Machine issued when the session is open can arrive
962 * after the session starts closing or gets closed. Note that when
963 * aCheck is false, we return E_FAIL to indicate that aWinId we return
964 * is not valid */
965 *aCanShow = FALSE;
966 *aWinId = 0;
967 return aCheck ? S_OK : E_FAIL;
968 }
969
970#ifndef VBOX_COM_INPROC_API_CLIENT
971 return mConsole->i_onShowWindow(aCheck, aCanShow, aWinId);
972#else
973 return S_OK;
974#endif
975}
976
977HRESULT Session::onBandwidthGroupChange(const ComPtr<IBandwidthGroup> &aBandwidthGroup)
978{
979 LogFlowThisFunc(("\n"));
980
981 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
982 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
983 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
984#ifndef VBOX_COM_INPROC_API_CLIENT
985 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
986
987 return mConsole->i_onBandwidthGroupChange(aBandwidthGroup);
988#else
989 RT_NOREF(aBandwidthGroup);
990 return S_OK;
991#endif
992}
993
994HRESULT Session::onStorageDeviceChange(const ComPtr<IMediumAttachment> &aMediumAttachment, BOOL aRemove, BOOL aSilent)
995{
996 LogFlowThisFunc(("\n"));
997
998 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
999 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
1000 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1001#ifndef VBOX_COM_INPROC_API_CLIENT
1002 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1003
1004 return mConsole->i_onStorageDeviceChange(aMediumAttachment, aRemove, aSilent);
1005#else
1006 RT_NOREF(aMediumAttachment, aRemove, aSilent);
1007 return S_OK;
1008#endif
1009}
1010
1011HRESULT Session::accessGuestProperty(const com::Utf8Str &aName, const com::Utf8Str &aValue, const com::Utf8Str &aFlags,
1012 ULONG aAccessMode, com::Utf8Str &aRetValue, LONG64 *aRetTimestamp, com::Utf8Str &aRetFlags)
1013{
1014#ifdef VBOX_WITH_GUEST_PROPS
1015# ifndef VBOX_COM_INPROC_API_CLIENT
1016 if (mState != SessionState_Locked)
1017 return setError(VBOX_E_INVALID_VM_STATE,
1018 tr("Machine is not locked by session (session state: %s)."),
1019 Global::stringifySessionState(mState));
1020 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1021 if (aName.isEmpty())
1022 return E_INVALIDARG;
1023 if (aAccessMode == 0 && !RT_VALID_PTR(aRetTimestamp))
1024 return E_POINTER;
1025
1026 /* If this session is not in a VM process fend off the call. The caller
1027 * handles this correctly, by doing the operation in VBoxSVC. */
1028 if (!mConsole)
1029 return E_ACCESSDENIED;
1030
1031 HRESULT hr;
1032 if (aAccessMode == 2)
1033 hr = mConsole->i_deleteGuestProperty(aName);
1034 else if (aAccessMode == 1)
1035 hr = mConsole->i_setGuestProperty(aName, aValue, aFlags);
1036 else if (aAccessMode == 0)
1037 hr = mConsole->i_getGuestProperty(aName, &aRetValue, aRetTimestamp, &aRetFlags);
1038 else
1039 hr = E_INVALIDARG;
1040
1041 return hr;
1042# else /* VBOX_COM_INPROC_API_CLIENT */
1043 /** @todo This is nonsense, non-VM API users shouldn't need to deal with this
1044 * method call, VBoxSVC should be clever enough to see that the
1045 * session doesn't have a console! */
1046 RT_NOREF(aName, aValue, aFlags, aAccessMode, aRetValue, aRetTimestamp, aRetFlags);
1047 return E_ACCESSDENIED;
1048# endif /* VBOX_COM_INPROC_API_CLIENT */
1049
1050#else /* VBOX_WITH_GUEST_PROPS */
1051 ReturnComNotImplemented();
1052#endif /* VBOX_WITH_GUEST_PROPS */
1053}
1054
1055HRESULT Session::enumerateGuestProperties(const com::Utf8Str &aPatterns,
1056 std::vector<com::Utf8Str> &aKeys,
1057 std::vector<com::Utf8Str> &aValues,
1058 std::vector<LONG64> &aTimestamps,
1059 std::vector<com::Utf8Str> &aFlags)
1060{
1061#if defined(VBOX_WITH_GUEST_PROPS) && !defined(VBOX_COM_INPROC_API_CLIENT)
1062 if (mState != SessionState_Locked)
1063 return setError(VBOX_E_INVALID_VM_STATE,
1064 tr("Machine is not locked by session (session state: %s)."),
1065 Global::stringifySessionState(mState));
1066 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1067
1068 /* If this session is not in a VM process fend off the call. The caller
1069 * handles this correctly, by doing the operation in VBoxSVC. */
1070 if (!mConsole)
1071 return E_ACCESSDENIED;
1072
1073 return mConsole->i_enumerateGuestProperties(aPatterns, aKeys, aValues, aTimestamps, aFlags);
1074
1075#else /* VBOX_WITH_GUEST_PROPS not defined */
1076 RT_NOREF(aPatterns, aKeys, aValues, aTimestamps, aFlags);
1077 ReturnComNotImplemented();
1078#endif /* VBOX_WITH_GUEST_PROPS not defined */
1079}
1080
1081HRESULT Session::onlineMergeMedium(const ComPtr<IMediumAttachment> &aMediumAttachment, ULONG aSourceIdx,
1082 ULONG aTargetIdx, const ComPtr<IProgress> &aProgress)
1083{
1084 if (mState != SessionState_Locked)
1085 return setError(VBOX_E_INVALID_VM_STATE,
1086 tr("Machine is not locked by session (session state: %s)."),
1087 Global::stringifySessionState(mState));
1088#ifndef VBOX_COM_INPROC_API_CLIENT
1089 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1090 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1091
1092 return mConsole->i_onlineMergeMedium(aMediumAttachment,
1093 aSourceIdx, aTargetIdx,
1094 aProgress);
1095#else
1096 RT_NOREF(aMediumAttachment, aSourceIdx, aTargetIdx, aProgress);
1097 AssertFailed();
1098 return E_NOTIMPL;
1099#endif
1100}
1101
1102HRESULT Session::reconfigureMediumAttachments(const std::vector<ComPtr<IMediumAttachment> > &aAttachments)
1103{
1104 if (mState != SessionState_Locked)
1105 return setError(VBOX_E_INVALID_VM_STATE,
1106 tr("Machine is not locked by session (session state: %s)."),
1107 Global::stringifySessionState(mState));
1108#ifndef VBOX_COM_INPROC_API_CLIENT
1109 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1110 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1111
1112 return mConsole->i_reconfigureMediumAttachments(aAttachments);
1113#else
1114 RT_NOREF(aAttachments);
1115 AssertFailed();
1116 return E_NOTIMPL;
1117#endif
1118}
1119
1120HRESULT Session::enableVMMStatistics(BOOL aEnable)
1121{
1122 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1123 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
1124 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1125#ifndef VBOX_COM_INPROC_API_CLIENT
1126 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1127
1128 mConsole->i_enableVMMStatistics(aEnable);
1129
1130 return S_OK;
1131#else
1132 RT_NOREF(aEnable);
1133 AssertFailed();
1134 return E_NOTIMPL;
1135#endif
1136}
1137
1138HRESULT Session::pauseWithReason(Reason_T aReason)
1139{
1140 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1141 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
1142 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1143#ifndef VBOX_COM_INPROC_API_CLIENT
1144 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1145
1146 return mConsole->i_pause(aReason);
1147#else
1148 RT_NOREF(aReason);
1149 AssertFailed();
1150 return E_NOTIMPL;
1151#endif
1152}
1153
1154HRESULT Session::resumeWithReason(Reason_T aReason)
1155{
1156 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1157 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
1158 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1159#ifndef VBOX_COM_INPROC_API_CLIENT
1160 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1161
1162 AutoWriteLock dummyLock(mConsole COMMA_LOCKVAL_SRC_POS);
1163 return mConsole->i_resume(aReason, dummyLock);
1164#else
1165 RT_NOREF(aReason);
1166 AssertFailed();
1167 return E_NOTIMPL;
1168#endif
1169}
1170
1171HRESULT Session::saveStateWithReason(Reason_T aReason,
1172 const ComPtr<IProgress> &aProgress,
1173 const ComPtr<ISnapshot> &aSnapshot,
1174 const Utf8Str &aStateFilePath,
1175 BOOL aPauseVM, BOOL *aLeftPaused)
1176{
1177 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1178 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
1179 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1180#ifndef VBOX_COM_INPROC_API_CLIENT
1181 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1182
1183 bool fLeftPaused = false;
1184 HRESULT hrc = mConsole->i_saveState(aReason, aProgress, aSnapshot, aStateFilePath, !!aPauseVM, fLeftPaused);
1185 if (aLeftPaused)
1186 *aLeftPaused = fLeftPaused;
1187 return hrc;
1188#else
1189 RT_NOREF(aReason, aProgress, aSnapshot, aStateFilePath, aPauseVM, aLeftPaused);
1190 AssertFailed();
1191 return E_NOTIMPL;
1192#endif
1193}
1194
1195HRESULT Session::cancelSaveStateWithReason()
1196{
1197 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1198 AssertReturn(mState == SessionState_Locked, VBOX_E_INVALID_VM_STATE);
1199 AssertReturn(mType == SessionType_WriteLock, VBOX_E_INVALID_OBJECT_STATE);
1200#ifndef VBOX_COM_INPROC_API_CLIENT
1201 AssertReturn(mConsole, VBOX_E_INVALID_OBJECT_STATE);
1202
1203 return mConsole->i_cancelSaveState();
1204#else
1205 AssertFailed();
1206 return E_NOTIMPL;
1207#endif
1208}
1209
1210// private methods
1211///////////////////////////////////////////////////////////////////////////////
1212
1213/**
1214 * Unlocks a machine associated with the current session.
1215 *
1216 * @param aFinalRelease called as a result of FinalRelease()
1217 * @param aFromServer called as a result of Uninitialize()
1218 * @param aLockW The write lock this object is protected with.
1219 * Must be acquired already and will be released
1220 * and later reacquired during the unlocking.
1221 *
1222 * @note To be called only from #uninit(), ISession::UnlockMachine() or
1223 * ISession::Uninitialize().
1224 */
1225HRESULT Session::i_unlockMachine(bool aFinalRelease, bool aFromServer, AutoWriteLock &aLockW)
1226{
1227 LogFlowThisFuncEnter();
1228 LogFlowThisFunc(("aFinalRelease=%d, isFromServer=%d\n",
1229 aFinalRelease, aFromServer));
1230
1231 LogFlowThisFunc(("mState=%s, mType=%d\n", ::stringifySessionState(mState), mType));
1232
1233 Assert(aLockW.isWriteLockOnCurrentThread());
1234
1235 if (mState != SessionState_Locked)
1236 {
1237 Assert(mState == SessionState_Spawning);
1238
1239 /* The session object is going to be uninitialized before it has been
1240 * assigned a direct console of the machine the client requested to open
1241 * a remote session to using IVirtualBox:: openRemoteSession(). It is OK
1242 * only if this close request comes from the server (for example, it
1243 * detected that the VM process it started terminated before opening a
1244 * direct session). Otherwise, it means that the client is too fast and
1245 * trying to close the session before waiting for the progress object it
1246 * got from IVirtualBox:: openRemoteSession() to complete, so assert. */
1247 Assert(aFromServer);
1248
1249 mState = SessionState_Unlocked;
1250 mType = SessionType_Null;
1251
1252 Assert(!mClientTokenHolder);
1253
1254 LogFlowThisFuncLeave();
1255 return S_OK;
1256 }
1257
1258 /* go to the closing state */
1259 mState = SessionState_Unlocking;
1260
1261 if (mType == SessionType_WriteLock)
1262 {
1263#ifndef VBOX_COM_INPROC_API_CLIENT
1264 if (!mConsole.isNull())
1265 {
1266 mConsole->uninit();
1267 mConsole.setNull();
1268 }
1269#else
1270 mRemoteMachine.setNull();
1271#endif
1272 }
1273 else
1274 {
1275 mRemoteMachine.setNull();
1276 mRemoteConsole.setNull();
1277 }
1278
1279 ComPtr<IProgress> progress;
1280
1281 if (!aFinalRelease && !aFromServer)
1282 {
1283 /*
1284 * We trigger OnSessionEnd() only when the session closes itself using
1285 * Close(). Note that if isFinalRelease = TRUE here, this means that
1286 * the client process has already initialized the termination procedure
1287 * without issuing Close() and the IPC channel is no more operational --
1288 * so we cannot call the server's method (it will definitely fail). The
1289 * server will instead simply detect the abnormal client death (since
1290 * OnSessionEnd() is not called) and reset the machine state to Aborted.
1291 */
1292
1293 /*
1294 * while waiting for OnSessionEnd() to complete one of our methods
1295 * can be called by the server (for example, Uninitialize(), if the
1296 * direct session has initiated a closure just a bit before us) so
1297 * we need to release the lock to avoid deadlocks. The state is already
1298 * SessionState_Closing here, so it's safe.
1299 */
1300 aLockW.release();
1301
1302 Assert(!aLockW.isWriteLockOnCurrentThread());
1303
1304 LogFlowThisFunc(("Calling mControl->OnSessionEnd()...\n"));
1305 HRESULT hrc = mControl->OnSessionEnd(this, progress.asOutParam());
1306 LogFlowThisFunc(("mControl->OnSessionEnd()=%08X\n", hrc));
1307
1308 aLockW.acquire();
1309
1310 /*
1311 * If we get E_UNEXPECTED this means that the direct session has already
1312 * been closed, we're just too late with our notification and nothing more
1313 *
1314 * bird: Seems E_ACCESSDENIED is what gets returned these days; see
1315 * ObjectState::addCaller.
1316 */
1317 if (mType != SessionType_WriteLock && (hrc == E_UNEXPECTED || hrc == E_ACCESSDENIED))
1318 hrc = S_OK;
1319
1320#if !defined(DEBUG_bird) && !defined(DEBUG_andy) /* I don't want clients crashing on me just because VBoxSVC went belly up. */
1321 AssertComRC(hrc);
1322#endif
1323 }
1324
1325 mControl.setNull();
1326
1327 if (mType == SessionType_WriteLock)
1328 {
1329 if (mClientTokenHolder)
1330 {
1331 delete mClientTokenHolder;
1332 mClientTokenHolder = NULL;
1333 }
1334
1335 if (!aFinalRelease && !aFromServer)
1336 {
1337 /*
1338 * Wait for the server to grab the semaphore and destroy the session
1339 * machine (allowing us to open a new session with the same machine
1340 * once this method returns)
1341 */
1342 Assert(!!progress);
1343 if (progress)
1344 progress->WaitForCompletion(-1);
1345 }
1346 }
1347
1348 mState = SessionState_Unlocked;
1349 mType = SessionType_Null;
1350
1351 /* release the VirtualBox instance as the very last step */
1352 mVirtualBox.setNull();
1353
1354 LogFlowThisFuncLeave();
1355 return S_OK;
1356}
1357
1358/* 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