VirtualBox

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

Last change on this file since 48406 was 48299, checked in by vboxsync, 11 years ago

SessionImpl.cpp: VBoxClient-x86.so fixes.

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