VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/GuestSessionImpl.cpp@ 48282

Last change on this file since 48282 was 47817, checked in by vboxsync, 11 years ago

GuestCtrl: Update for IGuestFile; some renaming.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 95.0 KB
Line 
1
2/* $Id: GuestSessionImpl.cpp 47817 2013-08-16 15:30:15Z vboxsync $ */
3/** @file
4 * VirtualBox Main - Guest session handling.
5 */
6
7/*
8 * Copyright (C) 2012-2013 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
20/*******************************************************************************
21* Header Files *
22*******************************************************************************/
23#include "GuestImpl.h"
24#include "GuestSessionImpl.h"
25#include "GuestCtrlImplPrivate.h"
26#include "VirtualBoxErrorInfoImpl.h"
27
28#include "Global.h"
29#include "AutoCaller.h"
30#include "ProgressImpl.h"
31#include "VBoxEvents.h"
32#include "VMMDev.h"
33
34#include <memory> /* For auto_ptr. */
35
36#include <iprt/cpp/utils.h> /* For unconst(). */
37#include <iprt/env.h>
38#include <iprt/file.h> /* For CopyTo/From. */
39
40#include <VBox/com/array.h>
41#include <VBox/com/listeners.h>
42#include <VBox/version.h>
43
44#ifdef LOG_GROUP
45 #undef LOG_GROUP
46#endif
47#define LOG_GROUP LOG_GROUP_GUEST_CONTROL
48#include <VBox/log.h>
49
50
51/**
52 * Base class representing an internal
53 * asynchronous session task.
54 */
55class GuestSessionTaskInternal
56{
57public:
58
59 GuestSessionTaskInternal(GuestSession *pSession)
60 : mSession(pSession),
61 mRC(VINF_SUCCESS) { }
62
63 virtual ~GuestSessionTaskInternal(void) { }
64
65 int rc(void) const { return mRC; }
66 bool isOk(void) const { return RT_SUCCESS(mRC); }
67 const ComObjPtr<GuestSession> &Session(void) const { return mSession; }
68
69protected:
70
71 const ComObjPtr<GuestSession> mSession;
72 int mRC;
73};
74
75/**
76 * Class for asynchronously opening a guest session.
77 */
78class GuestSessionTaskInternalOpen : public GuestSessionTaskInternal
79{
80public:
81
82 GuestSessionTaskInternalOpen(GuestSession *pSession)
83 : GuestSessionTaskInternal(pSession) { }
84};
85
86/**
87 * Internal listener class to serve events in an
88 * active manner, e.g. without polling delays.
89 */
90class GuestSessionListener
91{
92public:
93
94 GuestSessionListener(void)
95 {
96 }
97
98 HRESULT init(GuestSession *pSession)
99 {
100 mSession = pSession;
101 return S_OK;
102 }
103
104 void uninit(void)
105 {
106 mSession.setNull();
107 }
108
109 STDMETHOD(HandleEvent)(VBoxEventType_T aType, IEvent *aEvent)
110 {
111 switch (aType)
112 {
113 case VBoxEventType_OnGuestSessionStateChanged:
114 {
115 Assert(!mSession.isNull());
116 int rc2 = mSession->signalWaitEvents(aType, aEvent);
117#ifdef DEBUG_andy
118 LogFlowFunc(("Signalling events of type=%ld, session=%p resulted in rc=%Rrc\n",
119 aType, mSession, rc2));
120#endif
121 break;
122 }
123
124 default:
125 AssertMsgFailed(("Unhandled event %ld\n", aType));
126 break;
127 }
128
129 return S_OK;
130 }
131
132private:
133
134 ComObjPtr<GuestSession> mSession;
135};
136typedef ListenerImpl<GuestSessionListener, GuestSession*> GuestSessionListenerImpl;
137
138VBOX_LISTENER_DECLARE(GuestSessionListenerImpl)
139
140// constructor / destructor
141/////////////////////////////////////////////////////////////////////////////
142
143DEFINE_EMPTY_CTOR_DTOR(GuestSession)
144
145HRESULT GuestSession::FinalConstruct(void)
146{
147 LogFlowThisFunc(("\n"));
148 return BaseFinalConstruct();
149}
150
151void GuestSession::FinalRelease(void)
152{
153 LogFlowThisFuncEnter();
154 uninit();
155 BaseFinalRelease();
156 LogFlowThisFuncLeave();
157}
158
159// public initializer/uninitializer for internal purposes only
160/////////////////////////////////////////////////////////////////////////////
161
162/**
163 * Initializes a guest session but does *not* open in on the guest side
164 * yet. This needs to be done via the openSession() / openSessionAsync calls.
165 *
166 * @return IPRT status code.
167 ** @todo Docs!
168 */
169int GuestSession::init(Guest *pGuest, const GuestSessionStartupInfo &ssInfo,
170 const GuestCredentials &guestCreds)
171{
172 LogFlowThisFunc(("pGuest=%p, ssInfo=%p, guestCreds=%p\n",
173 pGuest, &ssInfo, &guestCreds));
174
175 /* Enclose the state transition NotReady->InInit->Ready. */
176 AutoInitSpan autoInitSpan(this);
177 AssertReturn(autoInitSpan.isOk(), VERR_OBJECT_DESTROYED);
178
179#ifndef VBOX_WITH_GUEST_CONTROL
180 autoInitSpan.setSucceeded();
181 return VINF_SUCCESS;
182#else
183 AssertPtrReturn(pGuest, VERR_INVALID_POINTER);
184
185 mParent = pGuest;
186
187 /* Copy over startup info. */
188 /** @todo Use an overloaded copy operator. Later. */
189 mData.mSession.mID = ssInfo.mID;
190 mData.mSession.mIsInternal = ssInfo.mIsInternal;
191 mData.mSession.mName = ssInfo.mName;
192 mData.mSession.mOpenFlags = ssInfo.mOpenFlags;
193 mData.mSession.mOpenTimeoutMS = ssInfo.mOpenTimeoutMS;
194
195 /** @todo Use an overloaded copy operator. Later. */
196 mData.mCredentials.mUser = guestCreds.mUser;
197 mData.mCredentials.mPassword = guestCreds.mPassword;
198 mData.mCredentials.mDomain = guestCreds.mDomain;
199
200 mData.mRC = VINF_SUCCESS;
201 mData.mStatus = GuestSessionStatus_Undefined;
202 mData.mNumObjects = 0;
203
204 HRESULT hr;
205
206 int rc = queryInfo();
207 if (RT_SUCCESS(rc))
208 {
209 hr = unconst(mEventSource).createObject();
210 if (FAILED(hr))
211 rc = VERR_NO_MEMORY;
212 else
213 {
214 hr = mEventSource->init(static_cast<IGuestSession*>(this));
215 if (FAILED(hr))
216 rc = VERR_COM_UNEXPECTED;
217 }
218 }
219
220 if (RT_SUCCESS(rc))
221 {
222 try
223 {
224 GuestSessionListener *pListener = new GuestSessionListener();
225 ComObjPtr<GuestSessionListenerImpl> thisListener;
226 hr = thisListener.createObject();
227 if (SUCCEEDED(hr))
228 hr = thisListener->init(pListener, this);
229
230 if (SUCCEEDED(hr))
231 {
232 com::SafeArray <VBoxEventType_T> eventTypes;
233 eventTypes.push_back(VBoxEventType_OnGuestSessionStateChanged);
234 hr = mEventSource->RegisterListener(thisListener,
235 ComSafeArrayAsInParam(eventTypes),
236 TRUE /* Active listener */);
237 if (SUCCEEDED(hr))
238 {
239 mLocalListener = thisListener;
240
241 rc = RTCritSectInit(&mWaitEventCritSect);
242 AssertRC(rc);
243 }
244 else
245 rc = VERR_COM_UNEXPECTED;
246 }
247 else
248 rc = VERR_COM_UNEXPECTED;
249 }
250 catch(std::bad_alloc &)
251 {
252 rc = VERR_NO_MEMORY;
253 }
254 }
255
256 if (RT_SUCCESS(rc))
257 {
258 /* Confirm a successful initialization when it's the case. */
259 autoInitSpan.setSucceeded();
260 }
261 else
262 autoInitSpan.setFailed();
263
264 LogFlowThisFunc(("mName=%s, mID=%RU32, mIsInternal=%RTbool, rc=%Rrc\n",
265 mData.mSession.mName.c_str(), mData.mSession.mID, mData.mSession.mIsInternal, rc));
266 return rc;
267#endif /* VBOX_WITH_GUEST_CONTROL */
268}
269
270/**
271 * Uninitializes the instance.
272 * Called from FinalRelease().
273 */
274void GuestSession::uninit(void)
275{
276 LogFlowThisFuncEnter();
277
278 /* Enclose the state transition Ready->InUninit->NotReady. */
279 AutoUninitSpan autoUninitSpan(this);
280 if (autoUninitSpan.uninitDone())
281 return;
282
283 int rc = VINF_SUCCESS;
284
285#ifdef VBOX_WITH_GUEST_CONTROL
286 LogFlowThisFunc(("Closing directories (%RU64 total)\n",
287 mData.mDirectories.size()));
288 for (SessionDirectories::iterator itDirs = mData.mDirectories.begin();
289 itDirs != mData.mDirectories.end(); ++itDirs)
290 {
291 (*itDirs)->Release();
292 }
293 mData.mDirectories.clear();
294
295 LogFlowThisFunc(("Closing files (%RU64 total)\n",
296 mData.mFiles.size()));
297 for (SessionFiles::iterator itFiles = mData.mFiles.begin();
298 itFiles != mData.mFiles.end(); ++itFiles)
299 {
300 itFiles->second->Release();
301 }
302 mData.mFiles.clear();
303
304 LogFlowThisFunc(("Closing processes (%RU64 total)\n",
305 mData.mProcesses.size()));
306 for (SessionProcesses::iterator itProcs = mData.mProcesses.begin();
307 itProcs != mData.mProcesses.end(); ++itProcs)
308 {
309 itProcs->second->Release();
310 }
311 mData.mProcesses.clear();
312
313 LogFlowThisFunc(("mNumObjects=%RU32\n", mData.mNumObjects));
314
315 baseUninit();
316
317 mEventSource->UnregisterListener(mLocalListener);
318 unconst(mEventSource).setNull();
319
320#endif /* VBOX_WITH_GUEST_CONTROL */
321 LogFlowFuncLeaveRC(rc);
322}
323
324// implementation of public getters/setters for attributes
325/////////////////////////////////////////////////////////////////////////////
326
327STDMETHODIMP GuestSession::COMGETTER(User)(BSTR *aUser)
328{
329#ifndef VBOX_WITH_GUEST_CONTROL
330 ReturnComNotImplemented();
331#else
332 LogFlowThisFuncEnter();
333
334 CheckComArgOutPointerValid(aUser);
335
336 AutoCaller autoCaller(this);
337 if (FAILED(autoCaller.rc())) return autoCaller.rc();
338
339 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
340
341 mData.mCredentials.mUser.cloneTo(aUser);
342
343 LogFlowThisFuncLeave();
344 return S_OK;
345#endif /* VBOX_WITH_GUEST_CONTROL */
346}
347
348STDMETHODIMP GuestSession::COMGETTER(Domain)(BSTR *aDomain)
349{
350#ifndef VBOX_WITH_GUEST_CONTROL
351 ReturnComNotImplemented();
352#else
353 LogFlowThisFuncEnter();
354
355 CheckComArgOutPointerValid(aDomain);
356
357 AutoCaller autoCaller(this);
358 if (FAILED(autoCaller.rc())) return autoCaller.rc();
359
360 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
361
362 mData.mCredentials.mDomain.cloneTo(aDomain);
363
364 LogFlowThisFuncLeave();
365 return S_OK;
366#endif /* VBOX_WITH_GUEST_CONTROL */
367}
368
369STDMETHODIMP GuestSession::COMGETTER(Name)(BSTR *aName)
370{
371#ifndef VBOX_WITH_GUEST_CONTROL
372 ReturnComNotImplemented();
373#else
374 LogFlowThisFuncEnter();
375
376 CheckComArgOutPointerValid(aName);
377
378 AutoCaller autoCaller(this);
379 if (FAILED(autoCaller.rc())) return autoCaller.rc();
380
381 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
382
383 mData.mSession.mName.cloneTo(aName);
384
385 LogFlowThisFuncLeave();
386 return S_OK;
387#endif /* VBOX_WITH_GUEST_CONTROL */
388}
389
390STDMETHODIMP GuestSession::COMGETTER(Id)(ULONG *aId)
391{
392#ifndef VBOX_WITH_GUEST_CONTROL
393 ReturnComNotImplemented();
394#else
395 LogFlowThisFuncEnter();
396
397 CheckComArgOutPointerValid(aId);
398
399 AutoCaller autoCaller(this);
400 if (FAILED(autoCaller.rc())) return autoCaller.rc();
401
402 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
403
404 *aId = mData.mSession.mID;
405
406 LogFlowThisFuncLeave();
407 return S_OK;
408#endif /* VBOX_WITH_GUEST_CONTROL */
409}
410
411STDMETHODIMP GuestSession::COMGETTER(Status)(GuestSessionStatus_T *aStatus)
412{
413#ifndef VBOX_WITH_GUEST_CONTROL
414 ReturnComNotImplemented();
415#else
416 LogFlowThisFuncEnter();
417
418 CheckComArgOutPointerValid(aStatus);
419
420 AutoCaller autoCaller(this);
421 if (FAILED(autoCaller.rc())) return autoCaller.rc();
422
423 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
424
425 *aStatus = mData.mStatus;
426
427 LogFlowThisFuncLeave();
428 return S_OK;
429#endif /* VBOX_WITH_GUEST_CONTROL */
430}
431
432STDMETHODIMP GuestSession::COMGETTER(Timeout)(ULONG *aTimeout)
433{
434#ifndef VBOX_WITH_GUEST_CONTROL
435 ReturnComNotImplemented();
436#else
437 LogFlowThisFuncEnter();
438
439 CheckComArgOutPointerValid(aTimeout);
440
441 AutoCaller autoCaller(this);
442 if (FAILED(autoCaller.rc())) return autoCaller.rc();
443
444 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
445
446 *aTimeout = mData.mTimeout;
447
448 LogFlowThisFuncLeave();
449 return S_OK;
450#endif /* VBOX_WITH_GUEST_CONTROL */
451}
452
453STDMETHODIMP GuestSession::COMSETTER(Timeout)(ULONG aTimeout)
454{
455#ifndef VBOX_WITH_GUEST_CONTROL
456 ReturnComNotImplemented();
457#else
458 LogFlowThisFuncEnter();
459
460 AutoCaller autoCaller(this);
461 if (FAILED(autoCaller.rc())) return autoCaller.rc();
462
463 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
464
465 mData.mTimeout = aTimeout;
466
467 LogFlowThisFuncLeave();
468 return S_OK;
469#endif /* VBOX_WITH_GUEST_CONTROL */
470}
471
472STDMETHODIMP GuestSession::COMGETTER(ProtocolVersion)(ULONG *aVersion)
473{
474#ifndef VBOX_WITH_GUEST_CONTROL
475 ReturnComNotImplemented();
476#else
477 LogFlowThisFuncEnter();
478
479 CheckComArgOutPointerValid(aVersion);
480
481 AutoCaller autoCaller(this);
482 if (FAILED(autoCaller.rc())) return autoCaller.rc();
483
484 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
485
486 *aVersion = mData.mProtocolVersion;
487
488 LogFlowThisFuncLeave();
489 return S_OK;
490#endif /* VBOX_WITH_GUEST_CONTROL */
491}
492
493STDMETHODIMP GuestSession::COMGETTER(Environment)(ComSafeArrayOut(BSTR, aEnvironment))
494{
495#ifndef VBOX_WITH_GUEST_CONTROL
496 ReturnComNotImplemented();
497#else
498 LogFlowThisFuncEnter();
499
500 CheckComArgOutSafeArrayPointerValid(aEnvironment);
501
502 AutoCaller autoCaller(this);
503 if (FAILED(autoCaller.rc())) return autoCaller.rc();
504
505 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
506
507 size_t cEnvVars = mData.mEnvironment.Size();
508 LogFlowThisFunc(("[%s]: cEnvVars=%RU32\n",
509 mData.mSession.mName.c_str(), cEnvVars));
510 com::SafeArray<BSTR> environment(cEnvVars);
511
512 for (size_t i = 0; i < cEnvVars; i++)
513 {
514 Bstr strEnv(mData.mEnvironment.Get(i));
515 strEnv.cloneTo(&environment[i]);
516 }
517 environment.detachTo(ComSafeArrayOutArg(aEnvironment));
518
519 LogFlowThisFuncLeave();
520 return S_OK;
521#endif /* VBOX_WITH_GUEST_CONTROL */
522}
523
524STDMETHODIMP GuestSession::COMSETTER(Environment)(ComSafeArrayIn(IN_BSTR, aValues))
525{
526#ifndef VBOX_WITH_GUEST_CONTROL
527 ReturnComNotImplemented();
528#else
529 LogFlowThisFuncEnter();
530
531 AutoCaller autoCaller(this);
532 if (FAILED(autoCaller.rc())) return autoCaller.rc();
533
534 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
535
536 com::SafeArray<IN_BSTR> environment(ComSafeArrayInArg(aValues));
537
538 int rc = VINF_SUCCESS;
539 for (size_t i = 0; i < environment.size() && RT_SUCCESS(rc); i++)
540 {
541 Utf8Str strEnv(environment[i]);
542 if (!strEnv.isEmpty()) /* Silently skip empty entries. */
543 rc = mData.mEnvironment.Set(strEnv);
544 }
545
546 HRESULT hr = RT_SUCCESS(rc) ? S_OK : VBOX_E_IPRT_ERROR;
547 LogFlowFuncLeaveRC(hr);
548 return hr;
549#endif /* VBOX_WITH_GUEST_CONTROL */
550}
551
552STDMETHODIMP GuestSession::COMGETTER(Processes)(ComSafeArrayOut(IGuestProcess *, aProcesses))
553{
554#ifndef VBOX_WITH_GUEST_CONTROL
555 ReturnComNotImplemented();
556#else
557 LogFlowThisFuncEnter();
558
559 CheckComArgOutSafeArrayPointerValid(aProcesses);
560
561 AutoCaller autoCaller(this);
562 if (FAILED(autoCaller.rc())) return autoCaller.rc();
563
564 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
565
566 SafeIfaceArray<IGuestProcess> collection(mData.mProcesses);
567 collection.detachTo(ComSafeArrayOutArg(aProcesses));
568
569 LogFlowFunc(("mProcesses=%zu\n", collection.size()));
570 return S_OK;
571#endif /* VBOX_WITH_GUEST_CONTROL */
572}
573
574STDMETHODIMP GuestSession::COMGETTER(Directories)(ComSafeArrayOut(IGuestDirectory *, aDirectories))
575{
576#ifndef VBOX_WITH_GUEST_CONTROL
577 ReturnComNotImplemented();
578#else
579 LogFlowThisFuncEnter();
580
581 CheckComArgOutSafeArrayPointerValid(aDirectories);
582
583 AutoCaller autoCaller(this);
584 if (FAILED(autoCaller.rc())) return autoCaller.rc();
585
586 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
587
588 SafeIfaceArray<IGuestDirectory> collection(mData.mDirectories);
589 collection.detachTo(ComSafeArrayOutArg(aDirectories));
590
591 LogFlowFunc(("mDirectories=%zu\n", collection.size()));
592 return S_OK;
593#endif /* VBOX_WITH_GUEST_CONTROL */
594}
595
596STDMETHODIMP GuestSession::COMGETTER(Files)(ComSafeArrayOut(IGuestFile *, aFiles))
597{
598#ifndef VBOX_WITH_GUEST_CONTROL
599 ReturnComNotImplemented();
600#else
601 LogFlowThisFuncEnter();
602
603 CheckComArgOutSafeArrayPointerValid(aFiles);
604
605 AutoCaller autoCaller(this);
606 if (FAILED(autoCaller.rc())) return autoCaller.rc();
607
608 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
609
610 SafeIfaceArray<IGuestFile> collection(mData.mFiles);
611 collection.detachTo(ComSafeArrayOutArg(aFiles));
612
613 LogFlowFunc(("mFiles=%zu\n", collection.size()));
614 return S_OK;
615#endif /* VBOX_WITH_GUEST_CONTROL */
616}
617
618STDMETHODIMP GuestSession::COMGETTER(EventSource)(IEventSource ** aEventSource)
619{
620#ifndef VBOX_WITH_GUEST_CONTROL
621 ReturnComNotImplemented();
622#else
623 LogFlowThisFuncEnter();
624
625 CheckComArgOutPointerValid(aEventSource);
626
627 AutoCaller autoCaller(this);
628 if (FAILED(autoCaller.rc())) return autoCaller.rc();
629
630 // no need to lock - lifetime constant
631 mEventSource.queryInterfaceTo(aEventSource);
632
633 LogFlowThisFuncLeave();
634 return S_OK;
635#endif /* VBOX_WITH_GUEST_CONTROL */
636}
637
638// private methods
639///////////////////////////////////////////////////////////////////////////////
640
641int GuestSession::closeSession(uint32_t uFlags, uint32_t uTimeoutMS, int *pGuestRc)
642{
643 LogFlowThisFunc(("uFlags=%x, uTimeoutMS=%RU32\n", uFlags, uTimeoutMS));
644
645 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
646
647 /* Guest Additions < 4.3 don't support closing dedicated
648 guest sessions, skip. */
649 if (mData.mProtocolVersion < 2)
650 {
651 LogFlowThisFunc(("Installed Guest Additions don't support closing dedicated sessions, skipping\n"));
652 return VINF_SUCCESS;
653 }
654
655 /** @todo uFlags validation. */
656
657 if (mData.mStatus != GuestSessionStatus_Started)
658 {
659 LogFlowThisFunc(("Session ID=%RU32 not started (anymore), status now is: %ld\n",
660 mData.mSession.mID, mData.mStatus));
661 return VINF_SUCCESS;
662 }
663
664 int vrc;
665
666 GuestWaitEvent *pEvent = NULL;
667 std::list < VBoxEventType_T > eventTypes;
668 try
669 {
670 eventTypes.push_back(VBoxEventType_OnGuestSessionStateChanged);
671
672 vrc = registerWaitEvent(mData.mSession.mID, 0 /* Object ID */,
673 eventTypes, &pEvent);
674 }
675 catch (std::bad_alloc)
676 {
677 vrc = VERR_NO_MEMORY;
678 }
679
680 if (RT_FAILURE(vrc))
681 return vrc;
682
683 LogFlowThisFunc(("Sending closing request to guest session ID=%RU32, uFlags=%x\n",
684 mData.mSession.mID, uFlags));
685
686 VBOXHGCMSVCPARM paParms[4];
687 int i = 0;
688 paParms[i++].setUInt32(pEvent->ContextID());
689 paParms[i++].setUInt32(uFlags);
690
691 alock.release(); /* Drop the write lock before waiting. */
692
693 vrc = sendCommand(HOST_SESSION_CLOSE, i, paParms);
694 if (RT_SUCCESS(vrc))
695 vrc = waitForStatusChange(pEvent, GuestSessionWaitForFlag_Terminate, uTimeoutMS,
696 NULL /* Session status */, pGuestRc);
697
698 unregisterWaitEvent(pEvent);
699
700 LogFlowFuncLeaveRC(vrc);
701 return vrc;
702}
703
704int GuestSession::directoryRemoveFromList(GuestDirectory *pDirectory)
705{
706 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
707
708 for (SessionDirectories::iterator itDirs = mData.mDirectories.begin();
709 itDirs != mData.mDirectories.end(); ++itDirs)
710 {
711 if (pDirectory == (*itDirs))
712 {
713 Bstr strName;
714 HRESULT hr = (*itDirs)->COMGETTER(DirectoryName)(strName.asOutParam());
715 ComAssertComRC(hr);
716
717 Assert(mData.mDirectories.size());
718 LogFlowFunc(("Removing directory \"%s\" (Session: %RU32) (now total %ld directories)\n",
719 Utf8Str(strName).c_str(), mData.mSession.mID, mData.mDirectories.size() - 1));
720
721 mData.mDirectories.erase(itDirs);
722 return VINF_SUCCESS;
723 }
724 }
725
726 return VERR_NOT_FOUND;
727}
728
729int GuestSession::directoryCreateInternal(const Utf8Str &strPath, uint32_t uMode, uint32_t uFlags, int *pGuestRc)
730{
731 /* pGuestRc is optional. */
732
733 LogFlowThisFunc(("strPath=%s, uMode=%x, uFlags=%x\n",
734 strPath.c_str(), uMode, uFlags));
735
736 GuestProcessStartupInfo procInfo;
737 procInfo.mCommand = Utf8Str(VBOXSERVICE_TOOL_MKDIR);
738 procInfo.mFlags = ProcessCreateFlag_Hidden;
739
740 int vrc = VINF_SUCCESS;
741
742 /* Construct arguments. */
743 if (uFlags & DirectoryCreateFlag_Parents)
744 procInfo.mArguments.push_back(Utf8Str("--parents")); /* We also want to create the parent directories. */
745 if (uMode)
746 {
747 procInfo.mArguments.push_back(Utf8Str("--mode")); /* Set the creation mode. */
748
749 char szMode[16];
750 if (RTStrPrintf(szMode, sizeof(szMode), "%o", uMode))
751 {
752 procInfo.mArguments.push_back(Utf8Str(szMode));
753 }
754 else
755 vrc = VERR_BUFFER_OVERFLOW;
756 }
757 procInfo.mArguments.push_back(strPath); /* The directory we want to create. */
758
759 if (RT_SUCCESS(vrc))
760 {
761 int guestRc;
762 GuestProcessTool procTool;
763 vrc = procTool.Init(this, procInfo, false /* Async */, &guestRc);
764 if (RT_SUCCESS(vrc))
765 {
766 if (RT_SUCCESS(guestRc))
767 vrc = procTool.Wait(GUESTPROCESSTOOL_FLAG_NONE, &guestRc);
768 }
769
770 if (RT_SUCCESS(vrc))
771 {
772 if (RT_SUCCESS(guestRc))
773 guestRc = procTool.TerminatedOk(NULL /* Exit code */);
774 }
775
776 if ( vrc == VERR_GSTCTL_GUEST_ERROR
777 && pGuestRc)
778 {
779 *pGuestRc = guestRc;
780 }
781 }
782
783 LogFlowFuncLeaveRC(vrc);
784 return vrc;
785}
786
787int GuestSession::directoryQueryInfoInternal(const Utf8Str &strPath, GuestFsObjData &objData, int *pGuestRc)
788{
789 LogFlowThisFunc(("strPath=%s\n", strPath.c_str()));
790
791 int vrc = fsQueryInfoInternal(strPath, objData, pGuestRc);
792 if (RT_SUCCESS(vrc))
793 {
794 vrc = objData.mType == FsObjType_Directory
795 ? VINF_SUCCESS : VERR_NOT_A_DIRECTORY;
796 }
797
798 LogFlowFuncLeaveRC(vrc);
799 return vrc;
800}
801
802int GuestSession::objectCreateTempInternal(const Utf8Str &strTemplate, const Utf8Str &strPath,
803 bool fDirectory, const Utf8Str &strName, int *pGuestRc)
804{
805 GuestProcessStartupInfo procInfo;
806 procInfo.mCommand = Utf8Str(VBOXSERVICE_TOOL_MKTEMP);
807 procInfo.mFlags = ProcessCreateFlag_WaitForStdOut;
808 procInfo.mArguments.push_back(Utf8Str("--machinereadable"));
809 if (fDirectory)
810 procInfo.mArguments.push_back(Utf8Str("-d"));
811 if (strPath.length()) /* Otherwise use /tmp or equivalent. */
812 {
813 procInfo.mArguments.push_back(Utf8Str("-t"));
814 procInfo.mArguments.push_back(strPath);
815 }
816 procInfo.mArguments.push_back(strTemplate);
817
818 GuestProcessTool procTool; int guestRc;
819 int vrc = procTool.Init(this, procInfo, false /* Async */, &guestRc);
820 if (RT_SUCCESS(vrc))
821 {
822 if (RT_SUCCESS(guestRc))
823 vrc = procTool.Wait(GUESTPROCESSTOOL_FLAG_NONE, &guestRc);
824 }
825
826 if (RT_SUCCESS(vrc))
827 {
828 if (RT_SUCCESS(guestRc))
829 guestRc = procTool.TerminatedOk(NULL /* Exit code */);
830 }
831
832 if ( vrc == VERR_GSTCTL_GUEST_ERROR
833 && pGuestRc)
834 {
835 *pGuestRc = guestRc;
836 }
837
838 LogFlowFuncLeaveRC(vrc);
839 return vrc;
840}
841
842int GuestSession::directoryOpenInternal(const Utf8Str &strPath, const Utf8Str &strFilter,
843 uint32_t uFlags, ComObjPtr<GuestDirectory> &pDirectory)
844{
845 LogFlowThisFunc(("strPath=%s, strPath=%s, uFlags=%x\n",
846 strPath.c_str(), strFilter.c_str(), uFlags));
847
848 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
849
850 /* Create the directory object. */
851 HRESULT hr = pDirectory.createObject();
852 if (FAILED(hr))
853 return VERR_COM_UNEXPECTED;
854
855 int vrc = pDirectory->init(this /* Parent */,
856 strPath, strFilter, uFlags);
857 if (RT_FAILURE(vrc))
858 return vrc;
859
860 /* Add the created directory to our vector. */
861 mData.mDirectories.push_back(pDirectory);
862
863 LogFlowFunc(("Added new directory \"%s\" (Session: %RU32)\n",
864 strPath.c_str(), mData.mSession.mID));
865
866 LogFlowFuncLeaveRC(vrc);
867 return vrc;
868}
869
870int GuestSession::dispatchToFile(PVBOXGUESTCTRLHOSTCBCTX pCtxCb, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
871{
872 LogFlowFunc(("pCtxCb=%p, pSvcCb=%p\n", pCtxCb, pSvcCb));
873
874 AssertPtrReturn(pCtxCb, VERR_INVALID_POINTER);
875 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
876
877 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
878
879 uint32_t uFileID = VBOX_GUESTCTRL_CONTEXTID_GET_OBJECT(pCtxCb->uContextID);
880#ifdef DEBUG
881 LogFlowFunc(("uFileID=%RU32 (%RU32 total)\n",
882 uFileID, mData.mFiles.size()));
883#endif
884 int rc;
885 SessionFiles::const_iterator itFile
886 = mData.mFiles.find(uFileID);
887 if (itFile != mData.mFiles.end())
888 {
889 ComObjPtr<GuestFile> pFile(itFile->second);
890 Assert(!pFile.isNull());
891
892 alock.release();
893
894 rc = pFile->callbackDispatcher(pCtxCb, pSvcCb);
895 }
896 else
897 rc = VERR_NOT_FOUND;
898
899 LogFlowFuncLeaveRC(rc);
900 return rc;
901}
902
903int GuestSession::dispatchToProcess(PVBOXGUESTCTRLHOSTCBCTX pCtxCb, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
904{
905 LogFlowFunc(("pCtxCb=%p, pSvcCb=%p\n", pCtxCb, pSvcCb));
906
907 AssertPtrReturn(pCtxCb, VERR_INVALID_POINTER);
908 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
909
910 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
911
912 uint32_t uProcessID = VBOX_GUESTCTRL_CONTEXTID_GET_OBJECT(pCtxCb->uContextID);
913#ifdef DEBUG
914 LogFlowFunc(("uProcessID=%RU32 (%RU32 total)\n",
915 uProcessID, mData.mProcesses.size()));
916#endif
917 int rc;
918 SessionProcesses::const_iterator itProc
919 = mData.mProcesses.find(uProcessID);
920 if (itProc != mData.mProcesses.end())
921 {
922 ComObjPtr<GuestProcess> pProcess(itProc->second);
923 Assert(!pProcess.isNull());
924
925 /* Set protocol version so that pSvcCb can
926 * be interpreted right. */
927 pCtxCb->uProtocol = mData.mProtocolVersion;
928
929 alock.release();
930 rc = pProcess->callbackDispatcher(pCtxCb, pSvcCb);
931 }
932 else
933 rc = VERR_NOT_FOUND;
934
935 LogFlowFuncLeaveRC(rc);
936 return rc;
937}
938
939int GuestSession::dispatchToThis(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
940{
941 AssertPtrReturn(pCbCtx, VERR_INVALID_POINTER);
942 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
943
944 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
945
946#ifdef DEBUG
947 LogFlowThisFunc(("sessionID=%RU32, CID=%RU32, uFunction=%RU32, pSvcCb=%p\n",
948 mData.mSession.mID, pCbCtx->uContextID, pCbCtx->uFunction, pSvcCb));
949#endif
950
951 int rc = VINF_SUCCESS;
952 switch (pCbCtx->uFunction)
953 {
954 case GUEST_DISCONNECTED:
955 /** @todo Handle closing all guest objects. */
956 break;
957
958 case GUEST_SESSION_NOTIFY:
959 {
960 rc = onSessionStatusChange(pCbCtx, pSvcCb);
961 break;
962 }
963
964 default:
965 /* Silently skip unknown callbacks. */
966 rc = VERR_NOT_SUPPORTED;
967 break;
968 }
969
970 LogFlowFuncLeaveRC(rc);
971 return rc;
972}
973
974inline bool GuestSession::fileExists(uint32_t uFileID, ComObjPtr<GuestFile> *pFile)
975{
976 SessionFiles::const_iterator it = mData.mFiles.find(uFileID);
977 if (it != mData.mFiles.end())
978 {
979 if (pFile)
980 *pFile = it->second;
981 return true;
982 }
983 return false;
984}
985
986int GuestSession::fileRemoveFromList(GuestFile *pFile)
987{
988 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
989
990 for (SessionFiles::iterator itFiles = mData.mFiles.begin();
991 itFiles != mData.mFiles.end(); ++itFiles)
992 {
993 if (pFile == itFiles->second)
994 {
995 /* Make sure to consume the pointer before the one of thfe
996 * iterator gets released. */
997 ComObjPtr<GuestFile> pCurFile = pFile;
998
999 Bstr strName;
1000 HRESULT hr = pCurFile->COMGETTER(FileName)(strName.asOutParam());
1001 ComAssertComRC(hr);
1002
1003 Assert(mData.mNumObjects);
1004 LogFlowThisFunc(("Removing guest file \"%s\" (Session: %RU32) (now total %ld files, %ld objects)\n",
1005 Utf8Str(strName).c_str(), mData.mSession.mID, mData.mFiles.size() - 1, mData.mNumObjects - 1));
1006
1007 pFile->cancelWaitEvents();
1008 pFile->Release();
1009
1010 mData.mFiles.erase(itFiles);
1011 mData.mNumObjects--;
1012
1013 alock.release(); /* Release lock before firing off event. */
1014
1015 fireGuestFileRegisteredEvent(mEventSource, this, pCurFile,
1016 false /* Unregistered */);
1017 return VINF_SUCCESS;
1018 }
1019 }
1020
1021 return VERR_NOT_FOUND;
1022}
1023
1024int GuestSession::fileRemoveInternal(const Utf8Str &strPath, int *pGuestRc)
1025{
1026 GuestProcessStartupInfo procInfo;
1027 GuestProcessStream streamOut;
1028
1029 procInfo.mCommand = Utf8Str(VBOXSERVICE_TOOL_RM);
1030 procInfo.mFlags = ProcessCreateFlag_WaitForStdOut;
1031 procInfo.mArguments.push_back(Utf8Str("--machinereadable"));
1032 procInfo.mArguments.push_back(strPath); /* The file we want to remove. */
1033
1034 GuestProcessTool procTool; int guestRc;
1035 int vrc = procTool.Init(this, procInfo, false /* Async */, &guestRc);
1036 if (RT_SUCCESS(vrc))
1037 vrc = procTool.Wait(GUESTPROCESSTOOL_FLAG_NONE, &guestRc);
1038
1039 if (RT_SUCCESS(vrc))
1040 {
1041 if (RT_SUCCESS(guestRc))
1042 guestRc = procTool.TerminatedOk(NULL /* Exit code */);
1043 }
1044
1045 if ( vrc == VERR_GSTCTL_GUEST_ERROR
1046 && pGuestRc)
1047 {
1048 *pGuestRc = guestRc;
1049 }
1050
1051 LogFlowFuncLeaveRC(vrc);
1052 return vrc;
1053}
1054
1055int GuestSession::fileOpenInternal(const GuestFileOpenInfo &openInfo, ComObjPtr<GuestFile> &pFile, int *pGuestRc)
1056{
1057 LogFlowThisFunc(("strPath=%s, strOpenMode=%s, strDisposition=%s, uCreationMode=%x, uOffset=%RU64\n",
1058 openInfo.mFileName.c_str(), openInfo.mOpenMode.c_str(), openInfo.mDisposition.c_str(),
1059 openInfo.mCreationMode, openInfo.mInitialOffset));
1060
1061 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1062
1063 /* Guest Additions < 4.3 don't support handling
1064 guest files, skip. */
1065 if (mData.mProtocolVersion < 2)
1066 {
1067 LogFlowThisFunc(("Installed Guest Additions don't support handling guest files, skipping\n"));
1068 return VERR_NOT_SUPPORTED;
1069 }
1070
1071 int rc = VERR_MAX_PROCS_REACHED;
1072 if (mData.mNumObjects >= VBOX_GUESTCTRL_MAX_OBJECTS)
1073 return rc;
1074
1075 /* Create a new (host-based) file ID and assign it. */
1076 uint32_t uNewFileID = 0;
1077 ULONG uTries = 0;
1078
1079 for (;;)
1080 {
1081 /* Is the file ID already used? */
1082 if (!fileExists(uNewFileID, NULL /* pProgress */))
1083 {
1084 /* Callback with context ID was not found. This means
1085 * we can use this context ID for our new callback we want
1086 * to add below. */
1087 rc = VINF_SUCCESS;
1088 break;
1089 }
1090 uNewFileID++;
1091 if (uNewFileID == VBOX_GUESTCTRL_MAX_OBJECTS)
1092 uNewFileID = 0;
1093
1094 if (++uTries == UINT32_MAX)
1095 break; /* Don't try too hard. */
1096 }
1097
1098 if (RT_FAILURE(rc))
1099 return rc;
1100
1101 /* Create the directory object. */
1102 HRESULT hr = pFile.createObject();
1103 if (FAILED(hr))
1104 return VERR_COM_UNEXPECTED;
1105
1106 Console *pConsole = mParent->getConsole();
1107 AssertPtr(pConsole);
1108
1109 rc = pFile->init(pConsole, this /* GuestSession */,
1110 uNewFileID, openInfo);
1111 if (RT_FAILURE(rc))
1112 return rc;
1113
1114 /*
1115 * Since this is a synchronous guest call we have to
1116 * register the file object first, releasing the session's
1117 * lock and then proceed with the actual opening command
1118 * -- otherwise the file's opening callback would hang
1119 * because the session's lock still is in place.
1120 */
1121 try
1122 {
1123 /* Add the created file to our vector. */
1124 mData.mFiles[uNewFileID] = pFile;
1125 mData.mNumObjects++;
1126 Assert(mData.mNumObjects <= VBOX_GUESTCTRL_MAX_OBJECTS);
1127
1128 LogFlowFunc(("Added new guest file \"%s\" (Session: %RU32) (now total %ld files, %ld objects)\n",
1129 openInfo.mFileName.c_str(), mData.mSession.mID, mData.mFiles.size(), mData.mNumObjects));
1130
1131 alock.release(); /* Release lock before firing off event. */
1132
1133 fireGuestFileRegisteredEvent(mEventSource, this, pFile,
1134 true /* Registered */);
1135 }
1136 catch (std::bad_alloc &)
1137 {
1138 rc = VERR_NO_MEMORY;
1139 }
1140
1141 if (RT_SUCCESS(rc))
1142 {
1143 int guestRc;
1144 rc = pFile->openFile(30 * 1000 /* 30s timeout */, &guestRc);
1145 if ( rc == VERR_GSTCTL_GUEST_ERROR
1146 && pGuestRc)
1147 {
1148 *pGuestRc = guestRc;
1149 }
1150 }
1151
1152 LogFlowFuncLeaveRC(rc);
1153 return rc;
1154}
1155
1156int GuestSession::fileQueryInfoInternal(const Utf8Str &strPath, GuestFsObjData &objData, int *pGuestRc)
1157{
1158 LogFlowThisFunc(("strPath=%s\n", strPath.c_str()));
1159
1160 int vrc = fsQueryInfoInternal(strPath, objData, pGuestRc);
1161 if (RT_SUCCESS(vrc))
1162 {
1163 vrc = objData.mType == FsObjType_File
1164 ? VINF_SUCCESS : VERR_NOT_A_FILE;
1165 }
1166
1167 LogFlowFuncLeaveRC(vrc);
1168 return vrc;
1169}
1170
1171int GuestSession::fileQuerySizeInternal(const Utf8Str &strPath, int64_t *pllSize, int *pGuestRc)
1172{
1173 AssertPtrReturn(pllSize, VERR_INVALID_POINTER);
1174
1175 GuestFsObjData objData;
1176 int vrc = fileQueryInfoInternal(strPath, objData, pGuestRc);
1177 if (RT_SUCCESS(vrc))
1178 *pllSize = objData.mObjectSize;
1179
1180 return vrc;
1181}
1182
1183int GuestSession::fsQueryInfoInternal(const Utf8Str &strPath, GuestFsObjData &objData, int *pGuestRc)
1184{
1185 LogFlowThisFunc(("strPath=%s\n", strPath.c_str()));
1186
1187 /** @todo Merge this with IGuestFile::queryInfo(). */
1188 GuestProcessStartupInfo procInfo;
1189 procInfo.mCommand = Utf8Str(VBOXSERVICE_TOOL_STAT);
1190 procInfo.mFlags = ProcessCreateFlag_WaitForStdOut;
1191
1192 /* Construct arguments. */
1193 procInfo.mArguments.push_back(Utf8Str("--machinereadable"));
1194 procInfo.mArguments.push_back(strPath);
1195
1196 GuestProcessTool procTool; int guestRc;
1197 int vrc = procTool.Init(this, procInfo, false /* Async */, &guestRc);
1198 if (RT_SUCCESS(vrc))
1199 vrc = procTool.Wait(GUESTPROCESSTOOL_FLAG_NONE, &guestRc);
1200 if (RT_SUCCESS(vrc))
1201 {
1202 guestRc = procTool.TerminatedOk(NULL /* Exit code */);
1203 if (RT_SUCCESS(guestRc))
1204 {
1205 GuestProcessStreamBlock curBlock;
1206 vrc = procTool.GetCurrentBlock(OUTPUT_HANDLE_ID_STDOUT, curBlock);
1207 /** @todo Check for more / validate blocks! */
1208 if (RT_SUCCESS(vrc))
1209 vrc = objData.FromStat(curBlock);
1210 }
1211 }
1212
1213 if ( vrc == VERR_GSTCTL_GUEST_ERROR
1214 && pGuestRc)
1215 {
1216 *pGuestRc = guestRc;
1217 }
1218
1219 LogFlowFuncLeaveRC(vrc);
1220 return vrc;
1221}
1222
1223const GuestCredentials& GuestSession::getCredentials(void)
1224{
1225 return mData.mCredentials;
1226}
1227
1228const GuestEnvironment& GuestSession::getEnvironment(void)
1229{
1230 return mData.mEnvironment;
1231}
1232
1233Utf8Str GuestSession::getName(void)
1234{
1235 return mData.mSession.mName;
1236}
1237
1238/* static */
1239Utf8Str GuestSession::guestErrorToString(int guestRc)
1240{
1241 Utf8Str strError;
1242
1243 /** @todo pData->u32Flags: int vs. uint32 -- IPRT errors are *negative* !!! */
1244 switch (guestRc)
1245 {
1246 case VERR_INVALID_VM_HANDLE:
1247 strError += Utf8StrFmt(tr("VMM device is not available (is the VM running?)"));
1248 break;
1249
1250 case VERR_HGCM_SERVICE_NOT_FOUND:
1251 strError += Utf8StrFmt(tr("The guest execution service is not available"));
1252 break;
1253
1254 case VERR_AUTHENTICATION_FAILURE:
1255 strError += Utf8StrFmt(tr("The specified user was not able to logon on guest"));
1256 break;
1257
1258 case VERR_TIMEOUT:
1259 strError += Utf8StrFmt(tr("The guest did not respond within time"));
1260 break;
1261
1262 case VERR_CANCELLED:
1263 strError += Utf8StrFmt(tr("The session operation was canceled"));
1264 break;
1265
1266 case VERR_PERMISSION_DENIED:
1267 strError += Utf8StrFmt(tr("Invalid user/password credentials"));
1268 break;
1269
1270 case VERR_MAX_PROCS_REACHED:
1271 strError += Utf8StrFmt(tr("Maximum number of concurrent guest processes has been reached"));
1272 break;
1273
1274 case VERR_NOT_EQUAL: /** @todo Imprecise to the user; can mean anything and all. */
1275 strError += Utf8StrFmt(tr("Unable to retrieve requested information"));
1276 break;
1277
1278 case VERR_NOT_FOUND:
1279 strError += Utf8StrFmt(tr("The guest execution service is not ready (yet)"));
1280 break;
1281
1282 default:
1283 strError += Utf8StrFmt("%Rrc", guestRc);
1284 break;
1285 }
1286
1287 return strError;
1288}
1289
1290/**
1291 * Checks if this session is ready state where it can handle
1292 * all session-bound actions (like guest processes, guest files).
1293 * Only used by official API methods. Will set an external
1294 * error when not ready.
1295 */
1296HRESULT GuestSession::isReadyExternal(void)
1297{
1298 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1299
1300 /** @todo Be a bit more informative. */
1301 if (mData.mStatus != GuestSessionStatus_Started)
1302 return setError(E_UNEXPECTED, tr("Session is not in started state"));
1303
1304 return S_OK;
1305}
1306
1307/** No locking! */
1308int GuestSession::onSessionStatusChange(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCbData)
1309{
1310 AssertPtrReturn(pCbCtx, VERR_INVALID_POINTER);
1311 /* pCallback is optional. */
1312 AssertPtrReturn(pSvcCbData, VERR_INVALID_POINTER);
1313
1314 if (pSvcCbData->mParms < 3)
1315 return VERR_INVALID_PARAMETER;
1316
1317 CALLBACKDATA_SESSION_NOTIFY dataCb;
1318 /* pSvcCb->mpaParms[0] always contains the context ID. */
1319 pSvcCbData->mpaParms[1].getUInt32(&dataCb.uType);
1320 pSvcCbData->mpaParms[2].getUInt32(&dataCb.uResult);
1321
1322 LogFlowThisFunc(("ID=%RU32, uType=%RU32, guestRc=%Rrc\n",
1323 mData.mSession.mID, dataCb.uType, dataCb.uResult));
1324
1325 int vrc = VINF_SUCCESS;
1326
1327 GuestSessionStatus_T sessionStatus = GuestSessionStatus_Undefined;
1328
1329 int guestRc = dataCb.uResult; /** @todo uint32_t vs. int. */
1330 switch (dataCb.uType)
1331 {
1332 case GUEST_SESSION_NOTIFYTYPE_ERROR:
1333 sessionStatus = GuestSessionStatus_Error;
1334 break;
1335
1336 case GUEST_SESSION_NOTIFYTYPE_STARTED:
1337 sessionStatus = GuestSessionStatus_Started;
1338 break;
1339
1340 case GUEST_SESSION_NOTIFYTYPE_TEN:
1341 case GUEST_SESSION_NOTIFYTYPE_TES:
1342 case GUEST_SESSION_NOTIFYTYPE_TEA:
1343 sessionStatus = GuestSessionStatus_Terminated;
1344 break;
1345
1346 case GUEST_SESSION_NOTIFYTYPE_TOK:
1347 sessionStatus = GuestSessionStatus_TimedOutKilled;
1348 break;
1349
1350 case GUEST_SESSION_NOTIFYTYPE_TOA:
1351 sessionStatus = GuestSessionStatus_TimedOutAbnormally;
1352 break;
1353
1354 case GUEST_SESSION_NOTIFYTYPE_DWN:
1355 sessionStatus = GuestSessionStatus_Down;
1356 break;
1357
1358 case GUEST_SESSION_NOTIFYTYPE_UNDEFINED:
1359 default:
1360 vrc = VERR_NOT_SUPPORTED;
1361 break;
1362 }
1363
1364 if (RT_SUCCESS(vrc))
1365 {
1366 if (RT_FAILURE(guestRc))
1367 sessionStatus = GuestSessionStatus_Error;
1368 }
1369
1370 /* Set the session status. */
1371 if (RT_SUCCESS(vrc))
1372 vrc = setSessionStatus(sessionStatus, guestRc);
1373
1374 LogFlowThisFunc(("ID=%RU32, guestRc=%Rrc\n", mData.mSession.mID, guestRc));
1375
1376 LogFlowFuncLeaveRC(vrc);
1377 return vrc;
1378}
1379
1380int GuestSession::startSessionInternal(int *pGuestRc)
1381{
1382 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1383
1384 LogFlowThisFunc(("mID=%RU32, mName=%s, uProtocolVersion=%RU32, openFlags=%x, openTimeoutMS=%RU32\n",
1385 mData.mSession.mID, mData.mSession.mName.c_str(), mData.mProtocolVersion,
1386 mData.mSession.mOpenFlags, mData.mSession.mOpenTimeoutMS));
1387
1388 /* Guest Additions < 4.3 don't support opening dedicated
1389 guest sessions. Simply return success here. */
1390 if (mData.mProtocolVersion < 2)
1391 {
1392 mData.mStatus = GuestSessionStatus_Started;
1393
1394 LogFlowThisFunc(("Installed Guest Additions don't support opening dedicated sessions, skipping\n"));
1395 return VINF_SUCCESS;
1396 }
1397
1398 if (mData.mStatus != GuestSessionStatus_Undefined)
1399 return VINF_SUCCESS;
1400
1401 /** @todo mData.mSession.uFlags validation. */
1402
1403 /* Set current session status. */
1404 mData.mStatus = GuestSessionStatus_Starting;
1405
1406 int vrc;
1407
1408 GuestWaitEvent *pEvent = NULL;
1409 std::list < VBoxEventType_T > eventTypes;
1410 try
1411 {
1412 eventTypes.push_back(VBoxEventType_OnGuestSessionStateChanged);
1413
1414 vrc = registerWaitEvent(mData.mSession.mID, 0 /* Object ID */,
1415 eventTypes, &pEvent);
1416 }
1417 catch (std::bad_alloc)
1418 {
1419 vrc = VERR_NO_MEMORY;
1420 }
1421
1422 if (RT_FAILURE(vrc))
1423 return vrc;
1424
1425 VBOXHGCMSVCPARM paParms[8];
1426
1427 int i = 0;
1428 paParms[i++].setUInt32(pEvent->ContextID());
1429 paParms[i++].setUInt32(mData.mProtocolVersion);
1430 paParms[i++].setPointer((void*)mData.mCredentials.mUser.c_str(),
1431 (ULONG)mData.mCredentials.mUser.length() + 1);
1432 paParms[i++].setPointer((void*)mData.mCredentials.mPassword.c_str(),
1433 (ULONG)mData.mCredentials.mPassword.length() + 1);
1434 paParms[i++].setPointer((void*)mData.mCredentials.mDomain.c_str(),
1435 (ULONG)mData.mCredentials.mDomain.length() + 1);
1436 paParms[i++].setUInt32(mData.mSession.mOpenFlags);
1437
1438 alock.release(); /* Drop write lock before sending. */
1439
1440 vrc = sendCommand(HOST_SESSION_CREATE, i, paParms);
1441 if (RT_SUCCESS(vrc))
1442 vrc = waitForStatusChange(pEvent, GuestSessionWaitForFlag_Start,
1443 30 * 1000 /* 30s timeout */,
1444 NULL /* Session status */, pGuestRc);
1445
1446 unregisterWaitEvent(pEvent);
1447
1448 LogFlowFuncLeaveRC(vrc);
1449 return vrc;
1450}
1451
1452int GuestSession::startSessionAsync(void)
1453{
1454 LogFlowThisFuncEnter();
1455
1456 int vrc;
1457
1458 try
1459 {
1460 /* Asynchronously open the session on the guest by kicking off a
1461 * worker thread. */
1462 std::auto_ptr<GuestSessionTaskInternalOpen> pTask(new GuestSessionTaskInternalOpen(this));
1463 AssertReturn(pTask->isOk(), pTask->rc());
1464
1465 vrc = RTThreadCreate(NULL, GuestSession::startSessionThread,
1466 (void *)pTask.get(), 0,
1467 RTTHREADTYPE_MAIN_WORKER, 0,
1468 "gctlSesStart");
1469 if (RT_SUCCESS(vrc))
1470 {
1471 /* pTask is now owned by openSessionThread(), so release it. */
1472 pTask.release();
1473 }
1474 }
1475 catch(std::bad_alloc &)
1476 {
1477 vrc = VERR_NO_MEMORY;
1478 }
1479
1480 LogFlowFuncLeaveRC(vrc);
1481 return vrc;
1482}
1483
1484/* static */
1485DECLCALLBACK(int) GuestSession::startSessionThread(RTTHREAD Thread, void *pvUser)
1486{
1487 LogFlowFunc(("pvUser=%p\n", pvUser));
1488
1489 std::auto_ptr<GuestSessionTaskInternalOpen> pTask(static_cast<GuestSessionTaskInternalOpen*>(pvUser));
1490 AssertPtr(pTask.get());
1491
1492 const ComObjPtr<GuestSession> pSession(pTask->Session());
1493 Assert(!pSession.isNull());
1494
1495 AutoCaller autoCaller(pSession);
1496 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1497
1498 int vrc = pSession->startSessionInternal(NULL /* Guest rc, ignored */);
1499 /* Nothing to do here anymore. */
1500
1501 LogFlowFuncLeaveRC(vrc);
1502 return vrc;
1503}
1504
1505int GuestSession::processRemoveFromList(GuestProcess *pProcess)
1506{
1507 AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
1508
1509 LogFlowThisFunc(("pProcess=%p\n", pProcess));
1510
1511 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1512
1513 int rc = VERR_NOT_FOUND;
1514
1515 ULONG uPID;
1516 HRESULT hr = pProcess->COMGETTER(PID)(&uPID);
1517 ComAssertComRC(hr);
1518
1519 LogFlowFunc(("Closing process (PID=%RU32) ...\n", uPID));
1520
1521 SessionProcesses::iterator itProcs = mData.mProcesses.begin();
1522 while (itProcs != mData.mProcesses.end())
1523 {
1524 if (pProcess == itProcs->second)
1525 {
1526 /* Make sure to consume the pointer before the one of thfe
1527 * iterator gets released. */
1528 ComObjPtr<GuestProcess> pCurProcess = pProcess;
1529
1530 hr = pCurProcess->COMGETTER(PID)(&uPID);
1531 ComAssertComRC(hr);
1532
1533 Assert(mData.mNumObjects);
1534 LogFlowFunc(("Removing process ID=%RU32 (Session: %RU32), guest PID=%RU32 (now total %ld processes, %ld objects)\n",
1535 pProcess->getObjectID(), mData.mSession.mID, uPID, mData.mProcesses.size() - 1, mData.mNumObjects - 1));
1536
1537 pProcess->cancelWaitEvents();
1538 pProcess->Release();
1539
1540 mData.mProcesses.erase(itProcs);
1541 mData.mNumObjects--;
1542
1543 alock.release(); /* Release lock before firing off event. */
1544
1545 fireGuestProcessRegisteredEvent(mEventSource, this /* Session */, pCurProcess,
1546 uPID, false /* Process unregistered */);
1547 rc = VINF_SUCCESS;
1548 break;
1549 }
1550
1551 itProcs++;
1552 }
1553
1554 LogFlowFuncLeaveRC(rc);
1555 return rc;
1556}
1557
1558/**
1559 * Creates but does *not* start the process yet. See GuestProcess::startProcess() or
1560 * GuestProcess::startProcessAsync() for that.
1561 *
1562 * @return IPRT status code.
1563 * @param procInfo
1564 * @param pProcess
1565 */
1566int GuestSession::processCreateExInteral(GuestProcessStartupInfo &procInfo, ComObjPtr<GuestProcess> &pProcess)
1567{
1568 LogFlowFunc(("mCmd=%s, mFlags=%x, mTimeoutMS=%RU32\n",
1569 procInfo.mCommand.c_str(), procInfo.mFlags, procInfo.mTimeoutMS));
1570#ifdef DEBUG
1571 if (procInfo.mArguments.size())
1572 {
1573 LogFlowFunc(("Arguments:"));
1574 ProcessArguments::const_iterator it = procInfo.mArguments.begin();
1575 while (it != procInfo.mArguments.end())
1576 {
1577 LogFlow((" %s", (*it).c_str()));
1578 it++;
1579 }
1580 LogFlow(("\n"));
1581 }
1582#endif
1583
1584 /* Validate flags. */
1585 if (procInfo.mFlags)
1586 {
1587 if ( !(procInfo.mFlags & ProcessCreateFlag_IgnoreOrphanedProcesses)
1588 && !(procInfo.mFlags & ProcessCreateFlag_WaitForProcessStartOnly)
1589 && !(procInfo.mFlags & ProcessCreateFlag_Hidden)
1590 && !(procInfo.mFlags & ProcessCreateFlag_NoProfile)
1591 && !(procInfo.mFlags & ProcessCreateFlag_WaitForStdOut)
1592 && !(procInfo.mFlags & ProcessCreateFlag_WaitForStdErr))
1593 {
1594 return VERR_INVALID_PARAMETER;
1595 }
1596 }
1597
1598 if ( (procInfo.mFlags & ProcessCreateFlag_WaitForProcessStartOnly)
1599 && ( (procInfo.mFlags & ProcessCreateFlag_WaitForStdOut)
1600 || (procInfo.mFlags & ProcessCreateFlag_WaitForStdErr)
1601 )
1602 )
1603 {
1604 return VERR_INVALID_PARAMETER;
1605 }
1606
1607 /* Adjust timeout. If set to 0, we define
1608 * an infinite timeout. */
1609 if (procInfo.mTimeoutMS == 0)
1610 procInfo.mTimeoutMS = UINT32_MAX;
1611
1612 /** @tood Implement process priority + affinity. */
1613
1614 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1615
1616 int rc = VERR_MAX_PROCS_REACHED;
1617 if (mData.mNumObjects >= VBOX_GUESTCTRL_MAX_OBJECTS)
1618 return rc;
1619
1620 /* Create a new (host-based) process ID and assign it. */
1621 uint32_t uNewProcessID = 0;
1622 ULONG uTries = 0;
1623
1624 for (;;)
1625 {
1626 /* Is the context ID already used? */
1627 if (!processExists(uNewProcessID, NULL /* pProgress */))
1628 {
1629 /* Callback with context ID was not found. This means
1630 * we can use this context ID for our new callback we want
1631 * to add below. */
1632 rc = VINF_SUCCESS;
1633 break;
1634 }
1635 uNewProcessID++;
1636 if (uNewProcessID == VBOX_GUESTCTRL_MAX_OBJECTS)
1637 uNewProcessID = 0;
1638
1639 if (++uTries == VBOX_GUESTCTRL_MAX_OBJECTS)
1640 break; /* Don't try too hard. */
1641 }
1642
1643 if (RT_FAILURE(rc))
1644 return rc;
1645
1646 /* Create the process object. */
1647 HRESULT hr = pProcess.createObject();
1648 if (FAILED(hr))
1649 return VERR_COM_UNEXPECTED;
1650
1651 rc = pProcess->init(mParent->getConsole() /* Console */, this /* Session */,
1652 uNewProcessID, procInfo);
1653 if (RT_FAILURE(rc))
1654 return rc;
1655
1656 /* Add the created process to our map. */
1657 try
1658 {
1659 mData.mProcesses[uNewProcessID] = pProcess;
1660 mData.mNumObjects++;
1661 Assert(mData.mNumObjects <= VBOX_GUESTCTRL_MAX_OBJECTS);
1662
1663 LogFlowFunc(("Added new process (Session: %RU32) with process ID=%RU32 (now total %ld processes, %ld objects)\n",
1664 mData.mSession.mID, uNewProcessID, mData.mProcesses.size(), mData.mNumObjects));
1665
1666 fireGuestProcessRegisteredEvent(mEventSource, this /* Session */, pProcess,
1667 0 /* PID */, true /* Process registered */);
1668 }
1669 catch (std::bad_alloc &)
1670 {
1671 rc = VERR_NO_MEMORY;
1672 }
1673
1674 return rc;
1675}
1676
1677inline bool GuestSession::processExists(uint32_t uProcessID, ComObjPtr<GuestProcess> *pProcess)
1678{
1679 SessionProcesses::const_iterator it = mData.mProcesses.find(uProcessID);
1680 if (it != mData.mProcesses.end())
1681 {
1682 if (pProcess)
1683 *pProcess = it->second;
1684 return true;
1685 }
1686 return false;
1687}
1688
1689inline int GuestSession::processGetByPID(ULONG uPID, ComObjPtr<GuestProcess> *pProcess)
1690{
1691 AssertReturn(uPID, false);
1692 /* pProcess is optional. */
1693
1694 SessionProcesses::iterator itProcs = mData.mProcesses.begin();
1695 for (; itProcs != mData.mProcesses.end(); itProcs++)
1696 {
1697 ComObjPtr<GuestProcess> pCurProc = itProcs->second;
1698 AutoCaller procCaller(pCurProc);
1699 if (procCaller.rc())
1700 return VERR_COM_INVALID_OBJECT_STATE;
1701
1702 ULONG uCurPID;
1703 HRESULT hr = pCurProc->COMGETTER(PID)(&uCurPID);
1704 ComAssertComRC(hr);
1705
1706 if (uCurPID == uPID)
1707 {
1708 if (pProcess)
1709 *pProcess = pCurProc;
1710 return VINF_SUCCESS;
1711 }
1712 }
1713
1714 return VERR_NOT_FOUND;
1715}
1716
1717int GuestSession::sendCommand(uint32_t uFunction,
1718 uint32_t uParms, PVBOXHGCMSVCPARM paParms)
1719{
1720 LogFlowThisFuncEnter();
1721
1722#ifndef VBOX_GUESTCTRL_TEST_CASE
1723 ComObjPtr<Console> pConsole = mParent->getConsole();
1724 Assert(!pConsole.isNull());
1725
1726 /* Forward the information to the VMM device. */
1727 VMMDev *pVMMDev = pConsole->getVMMDev();
1728 AssertPtr(pVMMDev);
1729
1730 LogFlowThisFunc(("uFunction=%RU32, uParms=%RU32\n", uFunction, uParms));
1731 int vrc = pVMMDev->hgcmHostCall(HGCMSERVICE_NAME, uFunction, uParms, paParms);
1732 if (RT_FAILURE(vrc))
1733 {
1734 /** @todo What to do here? */
1735 }
1736#else
1737 /* Not needed within testcases. */
1738 int vrc = VINF_SUCCESS;
1739#endif
1740 LogFlowFuncLeaveRC(vrc);
1741 return vrc;
1742}
1743
1744/* static */
1745HRESULT GuestSession::setErrorExternal(VirtualBoxBase *pInterface, int guestRc)
1746{
1747 AssertPtr(pInterface);
1748 AssertMsg(RT_FAILURE(guestRc), ("Guest rc does not indicate a failure when setting error\n"));
1749
1750 return pInterface->setError(VBOX_E_IPRT_ERROR, GuestSession::guestErrorToString(guestRc).c_str());
1751}
1752
1753/* Does not do locking; caller is responsible for that! */
1754int GuestSession::setSessionStatus(GuestSessionStatus_T sessionStatus, int sessionRc)
1755{
1756 LogFlowThisFunc(("oldStatus=%ld, newStatus=%ld, sessionRc=%Rrc\n",
1757 mData.mStatus, sessionStatus, sessionRc));
1758
1759 if (sessionStatus == GuestSessionStatus_Error)
1760 {
1761 AssertMsg(RT_FAILURE(sessionRc), ("Guest rc must be an error (%Rrc)\n", sessionRc));
1762 /* Do not allow overwriting an already set error. If this happens
1763 * this means we forgot some error checking/locking somewhere. */
1764 AssertMsg(RT_SUCCESS(mData.mRC), ("Guest rc already set (to %Rrc)\n", mData.mRC));
1765 }
1766 else
1767 AssertMsg(RT_SUCCESS(sessionRc), ("Guest rc must not be an error (%Rrc)\n", sessionRc));
1768
1769 if (mData.mStatus != sessionStatus)
1770 {
1771 mData.mStatus = sessionStatus;
1772 mData.mRC = sessionRc;
1773
1774 ComObjPtr<VirtualBoxErrorInfo> errorInfo;
1775 HRESULT hr = errorInfo.createObject();
1776 ComAssertComRC(hr);
1777 int rc2 = errorInfo->initEx(VBOX_E_IPRT_ERROR, sessionRc,
1778 COM_IIDOF(IGuestSession), getComponentName(),
1779 guestErrorToString(sessionRc));
1780 AssertRC(rc2);
1781
1782 fireGuestSessionStateChangedEvent(mEventSource, this,
1783 mData.mSession.mID, sessionStatus, errorInfo);
1784 }
1785
1786 return VINF_SUCCESS;
1787}
1788
1789int GuestSession::signalWaiters(GuestSessionWaitResult_T enmWaitResult, int rc /*= VINF_SUCCESS */)
1790{
1791 /*LogFlowThisFunc(("enmWaitResult=%d, rc=%Rrc, mWaitCount=%RU32, mWaitEvent=%p\n",
1792 enmWaitResult, rc, mData.mWaitCount, mData.mWaitEvent));*/
1793
1794 /* Note: No write locking here -- already done in the caller. */
1795
1796 int vrc = VINF_SUCCESS;
1797 /*if (mData.mWaitEvent)
1798 vrc = mData.mWaitEvent->Signal(enmWaitResult, rc);*/
1799 LogFlowFuncLeaveRC(vrc);
1800 return vrc;
1801}
1802
1803int GuestSession::startTaskAsync(const Utf8Str &strTaskDesc,
1804 GuestSessionTask *pTask, ComObjPtr<Progress> &pProgress)
1805{
1806 LogFlowThisFunc(("strTaskDesc=%s, pTask=%p\n", strTaskDesc.c_str(), pTask));
1807
1808 AssertPtrReturn(pTask, VERR_INVALID_POINTER);
1809
1810 /* Create the progress object. */
1811 HRESULT hr = pProgress.createObject();
1812 if (FAILED(hr))
1813 return VERR_COM_UNEXPECTED;
1814
1815 hr = pProgress->init(static_cast<IGuestSession*>(this),
1816 Bstr(strTaskDesc).raw(),
1817 TRUE /* aCancelable */);
1818 if (FAILED(hr))
1819 return VERR_COM_UNEXPECTED;
1820
1821 /* Initialize our worker task. */
1822 std::auto_ptr<GuestSessionTask> task(pTask);
1823
1824 int rc = task->RunAsync(strTaskDesc, pProgress);
1825 if (RT_FAILURE(rc))
1826 return rc;
1827
1828 /* Don't destruct on success. */
1829 task.release();
1830
1831 LogFlowFuncLeaveRC(rc);
1832 return rc;
1833}
1834
1835/**
1836 * Queries/collects information prior to establishing a guest session.
1837 * This is necessary to know which guest control protocol version to use,
1838 * among other things (later).
1839 *
1840 * @return IPRT status code.
1841 */
1842int GuestSession::queryInfo(void)
1843{
1844 /*
1845 * Try querying the guest control protocol version running on the guest.
1846 * This is done using the Guest Additions version
1847 */
1848 ComObjPtr<Guest> pGuest = mParent;
1849 Assert(!pGuest.isNull());
1850
1851 uint32_t uVerAdditions = pGuest->getAdditionsVersion();
1852 uint32_t uVBoxMajor = VBOX_FULL_VERSION_GET_MAJOR(uVerAdditions);
1853 uint32_t uVBoxMinor = VBOX_FULL_VERSION_GET_MINOR(uVerAdditions);
1854
1855#ifdef DEBUG_andy
1856 /* Hardcode the to-used protocol version; nice for testing side effects. */
1857 mData.mProtocolVersion = 2;
1858#else
1859 mData.mProtocolVersion = (
1860 /* VBox 5.0 and up. */
1861 uVBoxMajor >= 5
1862 /* VBox 4.3 and up. */
1863 || (uVBoxMajor == 4 && uVBoxMinor >= 3))
1864 ? 2 /* Guest control 2.0. */
1865 : 1; /* Legacy guest control (VBox < 4.3). */
1866 /* Build revision is ignored. */
1867#endif
1868
1869 LogFlowThisFunc(("uVerAdditions=%RU32 (%RU32.%RU32), mProtocolVersion=%RU32\n",
1870 uVerAdditions, uVBoxMajor, uVBoxMinor, mData.mProtocolVersion));
1871
1872 /* Tell the user but don't bitch too often. */
1873 static short s_gctrlLegacyWarning = 0;
1874 if ( mData.mProtocolVersion < 2
1875 && s_gctrlLegacyWarning++ < 3) /** @todo Find a bit nicer text. */
1876 LogRel((tr("Warning: Guest Additions are older (%ld.%ld) than host capabilities for guest control, please upgrade them. Using protocol version %ld now\n"),
1877 uVBoxMajor, uVBoxMinor, mData.mProtocolVersion));
1878
1879 return VINF_SUCCESS;
1880}
1881
1882int GuestSession::waitFor(uint32_t fWaitFlags, ULONG uTimeoutMS, GuestSessionWaitResult_T &waitResult, int *pGuestRc)
1883{
1884 LogFlowThisFuncEnter();
1885
1886 AssertReturn(fWaitFlags, VERR_INVALID_PARAMETER);
1887
1888 /*LogFlowThisFunc(("fWaitFlags=0x%x, uTimeoutMS=%RU32, mStatus=%RU32, mWaitCount=%RU32, mWaitEvent=%p, pGuestRc=%p\n",
1889 fWaitFlags, uTimeoutMS, mData.mStatus, mData.mWaitCount, mData.mWaitEvent, pGuestRc));*/
1890
1891 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1892
1893 /* Did some error occur before? Then skip waiting and return. */
1894 if (mData.mStatus == GuestSessionStatus_Error)
1895 {
1896 waitResult = GuestSessionWaitResult_Error;
1897 AssertMsg(RT_FAILURE(mData.mRC), ("No error rc (%Rrc) set when guest session indicated an error\n", mData.mRC));
1898 if (pGuestRc)
1899 *pGuestRc = mData.mRC; /* Return last set error. */
1900 return VERR_GSTCTL_GUEST_ERROR;
1901 }
1902
1903 /* Guest Additions < 4.3 don't support session handling, skip. */
1904 if (mData.mProtocolVersion < 2)
1905 {
1906 waitResult = GuestSessionWaitResult_WaitFlagNotSupported;
1907
1908 LogFlowThisFunc(("Installed Guest Additions don't support waiting for dedicated sessions, skipping\n"));
1909 return VINF_SUCCESS;
1910 }
1911
1912 waitResult = GuestSessionWaitResult_None;
1913 if (fWaitFlags & GuestSessionWaitForFlag_Terminate)
1914 {
1915 switch (mData.mStatus)
1916 {
1917 case GuestSessionStatus_Terminated:
1918 case GuestSessionStatus_Down:
1919 waitResult = GuestSessionWaitResult_Terminate;
1920 break;
1921
1922 case GuestSessionStatus_TimedOutKilled:
1923 case GuestSessionStatus_TimedOutAbnormally:
1924 waitResult = GuestSessionWaitResult_Timeout;
1925 break;
1926
1927 case GuestSessionStatus_Error:
1928 /* Handled above. */
1929 break;
1930
1931 case GuestSessionStatus_Started:
1932 waitResult = GuestSessionWaitResult_Start;
1933 break;
1934
1935 case GuestSessionStatus_Undefined:
1936 case GuestSessionStatus_Starting:
1937 /* Do the waiting below. */
1938 break;
1939
1940 default:
1941 AssertMsgFailed(("Unhandled session status %ld\n", mData.mStatus));
1942 return VERR_NOT_IMPLEMENTED;
1943 }
1944 }
1945 else if (fWaitFlags & GuestSessionWaitForFlag_Start)
1946 {
1947 switch (mData.mStatus)
1948 {
1949 case GuestSessionStatus_Started:
1950 case GuestSessionStatus_Terminating:
1951 case GuestSessionStatus_Terminated:
1952 case GuestSessionStatus_Down:
1953 waitResult = GuestSessionWaitResult_Start;
1954 break;
1955
1956 case GuestSessionStatus_Error:
1957 waitResult = GuestSessionWaitResult_Error;
1958 break;
1959
1960 case GuestSessionStatus_TimedOutKilled:
1961 case GuestSessionStatus_TimedOutAbnormally:
1962 waitResult = GuestSessionWaitResult_Timeout;
1963 break;
1964
1965 case GuestSessionStatus_Undefined:
1966 case GuestSessionStatus_Starting:
1967 /* Do the waiting below. */
1968 break;
1969
1970 default:
1971 AssertMsgFailed(("Unhandled session status %ld\n", mData.mStatus));
1972 return VERR_NOT_IMPLEMENTED;
1973 }
1974 }
1975
1976 LogFlowThisFunc(("sessionStatus=%ld, sessionRc=%Rrc, waitResult=%ld\n",
1977 mData.mStatus, mData.mRC, waitResult));
1978
1979 /* No waiting needed? Return immediately using the last set error. */
1980 if (waitResult != GuestSessionWaitResult_None)
1981 {
1982 if (pGuestRc)
1983 *pGuestRc = mData.mRC; /* Return last set error (if any). */
1984 return RT_SUCCESS(mData.mRC) ? VINF_SUCCESS : VERR_GSTCTL_GUEST_ERROR;
1985 }
1986
1987 int vrc;
1988
1989 GuestWaitEvent *pEvent = NULL;
1990 std::list < VBoxEventType_T > eventTypes;
1991 try
1992 {
1993 eventTypes.push_back(VBoxEventType_OnGuestSessionStateChanged);
1994
1995 vrc = registerWaitEvent(mData.mSession.mID, 0 /* Object ID */,
1996 eventTypes, &pEvent);
1997 }
1998 catch (std::bad_alloc)
1999 {
2000 vrc = VERR_NO_MEMORY;
2001 }
2002
2003 if (RT_FAILURE(vrc))
2004 return vrc;
2005
2006 alock.release(); /* Release lock before waiting. */
2007
2008 GuestSessionStatus_T sessionStatus;
2009 vrc = waitForStatusChange(pEvent, fWaitFlags,
2010 uTimeoutMS, &sessionStatus, pGuestRc);
2011 if (RT_SUCCESS(vrc))
2012 {
2013 switch (sessionStatus)
2014 {
2015 case GuestSessionStatus_Started:
2016 waitResult = GuestSessionWaitResult_Start;
2017 break;
2018
2019 case GuestSessionStatus_Terminated:
2020 waitResult = GuestSessionWaitResult_Terminate;
2021 break;
2022
2023 case GuestSessionStatus_TimedOutKilled:
2024 case GuestSessionStatus_TimedOutAbnormally:
2025 waitResult = GuestSessionWaitResult_Timeout;
2026 break;
2027
2028 case GuestSessionStatus_Down:
2029 waitResult = GuestSessionWaitResult_Terminate;
2030 break;
2031
2032 case GuestSessionStatus_Error:
2033 waitResult = GuestSessionWaitResult_Error;
2034 break;
2035
2036 default:
2037 waitResult = GuestSessionWaitResult_Status;
2038 break;
2039 }
2040 }
2041
2042 unregisterWaitEvent(pEvent);
2043
2044 LogFlowFuncLeaveRC(vrc);
2045 return vrc;
2046}
2047
2048int GuestSession::waitForStatusChange(GuestWaitEvent *pEvent, uint32_t fWaitFlags, uint32_t uTimeoutMS,
2049 GuestSessionStatus_T *pSessionStatus, int *pGuestRc)
2050{
2051 AssertPtrReturn(pEvent, VERR_INVALID_POINTER);
2052
2053 VBoxEventType_T evtType;
2054 ComPtr<IEvent> pIEvent;
2055 int vrc = waitForEvent(pEvent, uTimeoutMS,
2056 &evtType, pIEvent.asOutParam());
2057 if (RT_SUCCESS(vrc))
2058 {
2059 Assert(evtType == VBoxEventType_OnGuestSessionStateChanged);
2060
2061 ComPtr<IGuestSessionStateChangedEvent> pChangedEvent = pIEvent;
2062 Assert(!pChangedEvent.isNull());
2063
2064 GuestSessionStatus_T sessionStatus;
2065 pChangedEvent->COMGETTER(Status)(&sessionStatus);
2066 if (pSessionStatus)
2067 *pSessionStatus = sessionStatus;
2068
2069 ComPtr<IVirtualBoxErrorInfo> errorInfo;
2070 HRESULT hr = pChangedEvent->COMGETTER(Error)(errorInfo.asOutParam());
2071 ComAssertComRC(hr);
2072
2073 LONG lGuestRc;
2074 hr = errorInfo->COMGETTER(ResultDetail)(&lGuestRc);
2075 ComAssertComRC(hr);
2076 if (RT_FAILURE((int)lGuestRc))
2077 vrc = VERR_GSTCTL_GUEST_ERROR;
2078 if (pGuestRc)
2079 *pGuestRc = (int)lGuestRc;
2080
2081 LogFlowThisFunc(("Status changed event for session ID=%RU32, new status is: %ld (%Rrc)\n",
2082 mData.mSession.mID, sessionStatus,
2083 RT_SUCCESS((int)lGuestRc) ? VINF_SUCCESS : (int)lGuestRc));
2084 }
2085
2086 LogFlowFuncLeaveRC(vrc);
2087 return vrc;
2088}
2089
2090// implementation of public methods
2091/////////////////////////////////////////////////////////////////////////////
2092
2093STDMETHODIMP GuestSession::Close(void)
2094{
2095#ifndef VBOX_WITH_GUEST_CONTROL
2096 ReturnComNotImplemented();
2097#else
2098 LogFlowThisFuncEnter();
2099
2100 AutoCaller autoCaller(this);
2101 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2102
2103 /* Close session on guest. */
2104 int guestRc = VINF_SUCCESS;
2105 int rc = closeSession(0 /* Flags */, 30 * 1000 /* Timeout */,
2106 &guestRc);
2107 /* On failure don't return here, instead do all the cleanup
2108 * work first and then return an error. */
2109
2110 /* Remove ourselves from the session list. */
2111 int rc2 = mParent->sessionRemove(this);
2112 if (RT_SUCCESS(rc))
2113 rc = rc2;
2114
2115 LogFlowThisFunc(("Returning rc=%Rrc, guestRc=%Rrc\n",
2116 rc, guestRc));
2117 if (RT_FAILURE(rc))
2118 {
2119 if (rc == VERR_GSTCTL_GUEST_ERROR)
2120 return GuestSession::setErrorExternal(this, guestRc);
2121
2122 return setError(VBOX_E_IPRT_ERROR,
2123 tr("Closing guest session failed with %Rrc"), rc);
2124 }
2125
2126 return S_OK;
2127#endif /* VBOX_WITH_GUEST_CONTROL */
2128}
2129
2130STDMETHODIMP GuestSession::CopyFrom(IN_BSTR aSource, IN_BSTR aDest, ComSafeArrayIn(CopyFileFlag_T, aFlags), IProgress **aProgress)
2131{
2132#ifndef VBOX_WITH_GUEST_CONTROL
2133 ReturnComNotImplemented();
2134#else
2135 CheckComArgStrNotEmptyOrNull(aSource);
2136 CheckComArgStrNotEmptyOrNull(aDest);
2137 CheckComArgOutPointerValid(aProgress);
2138
2139 LogFlowThisFuncEnter();
2140
2141 if (RT_UNLIKELY((aSource) == NULL || *(aSource) == '\0'))
2142 return setError(E_INVALIDARG, tr("No source specified"));
2143 if (RT_UNLIKELY((aDest) == NULL || *(aDest) == '\0'))
2144 return setError(E_INVALIDARG, tr("No destination specified"));
2145
2146 AutoCaller autoCaller(this);
2147 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2148
2149 uint32_t fFlags = CopyFileFlag_None;
2150 if (aFlags)
2151 {
2152 com::SafeArray<CopyFileFlag_T> flags(ComSafeArrayInArg(aFlags));
2153 for (size_t i = 0; i < flags.size(); i++)
2154 fFlags |= flags[i];
2155 }
2156
2157 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2158
2159 HRESULT hr = S_OK;
2160
2161 try
2162 {
2163 ComObjPtr<Progress> pProgress;
2164 SessionTaskCopyFrom *pTask = new SessionTaskCopyFrom(this /* GuestSession */,
2165 Utf8Str(aSource), Utf8Str(aDest), fFlags);
2166 int rc = startTaskAsync(Utf8StrFmt(tr("Copying \"%ls\" from guest to \"%ls\" on the host"), aSource, aDest),
2167 pTask, pProgress);
2168 if (RT_SUCCESS(rc))
2169 {
2170 /* Return progress to the caller. */
2171 hr = pProgress.queryInterfaceTo(aProgress);
2172 }
2173 else
2174 hr = setError(VBOX_E_IPRT_ERROR,
2175 tr("Starting task for copying file \"%ls\" from guest to \"%ls\" on the host failed: %Rrc"), rc);
2176 }
2177 catch(std::bad_alloc &)
2178 {
2179 hr = E_OUTOFMEMORY;
2180 }
2181
2182 return hr;
2183#endif /* VBOX_WITH_GUEST_CONTROL */
2184}
2185
2186STDMETHODIMP GuestSession::CopyTo(IN_BSTR aSource, IN_BSTR aDest, ComSafeArrayIn(CopyFileFlag_T, aFlags), IProgress **aProgress)
2187{
2188#ifndef VBOX_WITH_GUEST_CONTROL
2189 ReturnComNotImplemented();
2190#else
2191 CheckComArgStrNotEmptyOrNull(aSource);
2192 CheckComArgStrNotEmptyOrNull(aDest);
2193 CheckComArgOutPointerValid(aProgress);
2194
2195 LogFlowThisFuncEnter();
2196
2197 if (RT_UNLIKELY((aSource) == NULL || *(aSource) == '\0'))
2198 return setError(E_INVALIDARG, tr("No source specified"));
2199 if (RT_UNLIKELY((aDest) == NULL || *(aDest) == '\0'))
2200 return setError(E_INVALIDARG, tr("No destination specified"));
2201
2202 AutoCaller autoCaller(this);
2203 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2204
2205 uint32_t fFlags = CopyFileFlag_None;
2206 if (aFlags)
2207 {
2208 com::SafeArray<CopyFileFlag_T> flags(ComSafeArrayInArg(aFlags));
2209 for (size_t i = 0; i < flags.size(); i++)
2210 fFlags |= flags[i];
2211 }
2212
2213 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2214
2215 HRESULT hr = S_OK;
2216
2217 try
2218 {
2219 ComObjPtr<Progress> pProgress;
2220 SessionTaskCopyTo *pTask = new SessionTaskCopyTo(this /* GuestSession */,
2221 Utf8Str(aSource), Utf8Str(aDest), fFlags);
2222 AssertPtrReturn(pTask, E_OUTOFMEMORY);
2223 int rc = startTaskAsync(Utf8StrFmt(tr("Copying \"%ls\" from host to \"%ls\" on the guest"), aSource, aDest),
2224 pTask, pProgress);
2225 if (RT_SUCCESS(rc))
2226 {
2227 /* Return progress to the caller. */
2228 hr = pProgress.queryInterfaceTo(aProgress);
2229 }
2230 else
2231 hr = setError(VBOX_E_IPRT_ERROR,
2232 tr("Starting task for copying file \"%ls\" from host to \"%ls\" on the guest failed: %Rrc"), rc);
2233 }
2234 catch(std::bad_alloc &)
2235 {
2236 hr = E_OUTOFMEMORY;
2237 }
2238
2239 return hr;
2240#endif /* VBOX_WITH_GUEST_CONTROL */
2241}
2242
2243STDMETHODIMP GuestSession::DirectoryCreate(IN_BSTR aPath, ULONG aMode,
2244 ComSafeArrayIn(DirectoryCreateFlag_T, aFlags))
2245{
2246#ifndef VBOX_WITH_GUEST_CONTROL
2247 ReturnComNotImplemented();
2248#else
2249 LogFlowThisFuncEnter();
2250
2251 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2252 return setError(E_INVALIDARG, tr("No directory to create specified"));
2253
2254 AutoCaller autoCaller(this);
2255 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2256
2257 uint32_t fFlags = DirectoryCreateFlag_None;
2258 if (aFlags)
2259 {
2260 com::SafeArray<DirectoryCreateFlag_T> flags(ComSafeArrayInArg(aFlags));
2261 for (size_t i = 0; i < flags.size(); i++)
2262 fFlags |= flags[i];
2263
2264 if (fFlags)
2265 {
2266 if (!(fFlags & DirectoryCreateFlag_Parents))
2267 return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), fFlags);
2268 }
2269 }
2270
2271 HRESULT hr = S_OK;
2272
2273 ComObjPtr <GuestDirectory> pDirectory; int guestRc;
2274 int rc = directoryCreateInternal(Utf8Str(aPath), (uint32_t)aMode, fFlags, &guestRc);
2275 if (RT_FAILURE(rc))
2276 {
2277 switch (rc)
2278 {
2279 case VERR_GSTCTL_GUEST_ERROR:
2280 hr = GuestProcess::setErrorExternal(this, guestRc);
2281 break;
2282
2283 case VERR_INVALID_PARAMETER:
2284 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: Invalid parameters given"));
2285 break;
2286
2287 case VERR_BROKEN_PIPE:
2288 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: Unexpectedly aborted"));
2289 break;
2290
2291 case VERR_CANT_CREATE:
2292 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: Could not create directory"));
2293 break;
2294
2295 default:
2296 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: %Rrc"), rc);
2297 break;
2298 }
2299 }
2300
2301 return hr;
2302#endif /* VBOX_WITH_GUEST_CONTROL */
2303}
2304
2305STDMETHODIMP GuestSession::DirectoryCreateTemp(IN_BSTR aTemplate, ULONG aMode, IN_BSTR aPath, BOOL aSecure, BSTR *aDirectory)
2306{
2307#ifndef VBOX_WITH_GUEST_CONTROL
2308 ReturnComNotImplemented();
2309#else
2310 LogFlowThisFuncEnter();
2311
2312 if (RT_UNLIKELY((aTemplate) == NULL || *(aTemplate) == '\0'))
2313 return setError(E_INVALIDARG, tr("No template specified"));
2314 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2315 return setError(E_INVALIDARG, tr("No directory name specified"));
2316 CheckComArgOutPointerValid(aDirectory);
2317
2318 AutoCaller autoCaller(this);
2319 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2320
2321 HRESULT hr = S_OK;
2322
2323 Utf8Str strName; int guestRc;
2324 int rc = objectCreateTempInternal(Utf8Str(aTemplate),
2325 Utf8Str(aPath),
2326 true /* Directory */, strName, &guestRc);
2327 if (RT_SUCCESS(rc))
2328 {
2329 strName.cloneTo(aDirectory);
2330 }
2331 else
2332 {
2333 switch (rc)
2334 {
2335 case VERR_GSTCTL_GUEST_ERROR:
2336 hr = GuestProcess::setErrorExternal(this, guestRc);
2337 break;
2338
2339 default:
2340 hr = setError(VBOX_E_IPRT_ERROR, tr("Temporary directory creation \"%s\" with template \"%s\" failed: %Rrc"),
2341 Utf8Str(aPath).c_str(), Utf8Str(aTemplate).c_str(), rc);
2342 break;
2343 }
2344 }
2345
2346 return hr;
2347#endif /* VBOX_WITH_GUEST_CONTROL */
2348}
2349
2350STDMETHODIMP GuestSession::DirectoryExists(IN_BSTR aPath, BOOL *aExists)
2351{
2352#ifndef VBOX_WITH_GUEST_CONTROL
2353 ReturnComNotImplemented();
2354#else
2355 LogFlowThisFuncEnter();
2356
2357 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2358 return setError(E_INVALIDARG, tr("No directory to check existence for specified"));
2359 CheckComArgOutPointerValid(aExists);
2360
2361 AutoCaller autoCaller(this);
2362 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2363
2364 HRESULT hr = S_OK;
2365
2366 GuestFsObjData objData; int guestRc;
2367 int rc = directoryQueryInfoInternal(Utf8Str(aPath), objData, &guestRc);
2368 if (RT_SUCCESS(rc))
2369 {
2370 *aExists = objData.mType == FsObjType_Directory;
2371 }
2372 else
2373 {
2374 switch (rc)
2375 {
2376 case VERR_GSTCTL_GUEST_ERROR:
2377 hr = GuestProcess::setErrorExternal(this, guestRc);
2378 break;
2379
2380 default:
2381 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying directory existence \"%s\" failed: %Rrc"),
2382 Utf8Str(aPath).c_str(), rc);
2383 break;
2384 }
2385 }
2386
2387 return hr;
2388#endif /* VBOX_WITH_GUEST_CONTROL */
2389}
2390
2391STDMETHODIMP GuestSession::DirectoryOpen(IN_BSTR aPath, IN_BSTR aFilter, ComSafeArrayIn(DirectoryOpenFlag_T, aFlags), IGuestDirectory **aDirectory)
2392{
2393#ifndef VBOX_WITH_GUEST_CONTROL
2394 ReturnComNotImplemented();
2395#else
2396 LogFlowThisFuncEnter();
2397
2398 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2399 return setError(E_INVALIDARG, tr("No directory to open specified"));
2400 if (RT_UNLIKELY((aFilter) != NULL && *(aFilter) != '\0'))
2401 return setError(E_INVALIDARG, tr("Directory filters are not implemented yet"));
2402 CheckComArgOutPointerValid(aDirectory);
2403
2404 AutoCaller autoCaller(this);
2405 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2406
2407 uint32_t fFlags = DirectoryOpenFlag_None;
2408 if (aFlags)
2409 {
2410 com::SafeArray<DirectoryOpenFlag_T> flags(ComSafeArrayInArg(aFlags));
2411 for (size_t i = 0; i < flags.size(); i++)
2412 fFlags |= flags[i];
2413
2414 if (fFlags)
2415 return setError(E_INVALIDARG, tr("Open flags (%#x) not implemented yet"), fFlags);
2416 }
2417
2418 HRESULT hr = S_OK;
2419
2420 ComObjPtr <GuestDirectory> pDirectory;
2421 int rc = directoryOpenInternal(Utf8Str(aPath), Utf8Str(aFilter), fFlags, pDirectory);
2422 if (RT_SUCCESS(rc))
2423 {
2424 /* Return directory object to the caller. */
2425 hr = pDirectory.queryInterfaceTo(aDirectory);
2426 }
2427 else
2428 {
2429 switch (rc)
2430 {
2431 case VERR_INVALID_PARAMETER:
2432 hr = setError(VBOX_E_IPRT_ERROR, tr("Opening directory \"%s\" failed; invalid parameters given",
2433 Utf8Str(aPath).c_str()));
2434 break;
2435
2436 default:
2437 hr = setError(VBOX_E_IPRT_ERROR, tr("Opening directory \"%s\" failed: %Rrc"),
2438 Utf8Str(aPath).c_str(),rc);
2439 break;
2440 }
2441 }
2442
2443 return hr;
2444#endif /* VBOX_WITH_GUEST_CONTROL */
2445}
2446
2447STDMETHODIMP GuestSession::DirectoryQueryInfo(IN_BSTR aPath, IGuestFsObjInfo **aInfo)
2448{
2449#ifndef VBOX_WITH_GUEST_CONTROL
2450 ReturnComNotImplemented();
2451#else
2452 LogFlowThisFuncEnter();
2453
2454 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2455 return setError(E_INVALIDARG, tr("No directory to query information for specified"));
2456 CheckComArgOutPointerValid(aInfo);
2457
2458 AutoCaller autoCaller(this);
2459 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2460
2461 HRESULT hr = S_OK;
2462
2463 GuestFsObjData objData; int guestRc;
2464 int vrc = directoryQueryInfoInternal(Utf8Str(aPath), objData, &guestRc);
2465 if (RT_SUCCESS(vrc))
2466 {
2467 if (objData.mType == FsObjType_Directory)
2468 {
2469 ComObjPtr<GuestFsObjInfo> pFsObjInfo;
2470 hr = pFsObjInfo.createObject();
2471 if (FAILED(hr)) return hr;
2472
2473 vrc = pFsObjInfo->init(objData);
2474 if (RT_SUCCESS(vrc))
2475 {
2476 hr = pFsObjInfo.queryInterfaceTo(aInfo);
2477 if (FAILED(hr)) return hr;
2478 }
2479 }
2480 }
2481
2482 if (RT_FAILURE(vrc))
2483 {
2484 switch (vrc)
2485 {
2486 case VERR_GSTCTL_GUEST_ERROR:
2487 hr = GuestProcess::setErrorExternal(this, guestRc);
2488 break;
2489
2490 case VERR_NOT_A_DIRECTORY:
2491 hr = setError(VBOX_E_IPRT_ERROR, tr("Element \"%s\" exists but is not a directory",
2492 Utf8Str(aPath).c_str()));
2493 break;
2494
2495 default:
2496 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying directory information for \"%s\" failed: %Rrc"),
2497 Utf8Str(aPath).c_str(), vrc);
2498 break;
2499 }
2500 }
2501
2502 return hr;
2503#endif /* VBOX_WITH_GUEST_CONTROL */
2504}
2505
2506STDMETHODIMP GuestSession::DirectoryRemove(IN_BSTR aPath)
2507{
2508#ifndef VBOX_WITH_GUEST_CONTROL
2509 ReturnComNotImplemented();
2510#else
2511 LogFlowThisFuncEnter();
2512
2513 AutoCaller autoCaller(this);
2514 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2515
2516 ReturnComNotImplemented();
2517#endif /* VBOX_WITH_GUEST_CONTROL */
2518}
2519
2520STDMETHODIMP GuestSession::DirectoryRemoveRecursive(IN_BSTR aPath, ComSafeArrayIn(DirectoryRemoveRecFlag_T, aFlags), IProgress **aProgress)
2521{
2522#ifndef VBOX_WITH_GUEST_CONTROL
2523 ReturnComNotImplemented();
2524#else
2525 LogFlowThisFuncEnter();
2526
2527 AutoCaller autoCaller(this);
2528 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2529
2530 ReturnComNotImplemented();
2531#endif /* VBOX_WITH_GUEST_CONTROL */
2532}
2533
2534STDMETHODIMP GuestSession::DirectoryRename(IN_BSTR aSource, IN_BSTR aDest, ComSafeArrayIn(PathRenameFlag_T, aFlags))
2535{
2536#ifndef VBOX_WITH_GUEST_CONTROL
2537 ReturnComNotImplemented();
2538#else
2539 LogFlowThisFuncEnter();
2540
2541 AutoCaller autoCaller(this);
2542 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2543
2544 ReturnComNotImplemented();
2545#endif /* VBOX_WITH_GUEST_CONTROL */
2546}
2547
2548STDMETHODIMP GuestSession::DirectorySetACL(IN_BSTR aPath, IN_BSTR aACL)
2549{
2550#ifndef VBOX_WITH_GUEST_CONTROL
2551 ReturnComNotImplemented();
2552#else
2553 LogFlowThisFuncEnter();
2554
2555 AutoCaller autoCaller(this);
2556 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2557
2558 ReturnComNotImplemented();
2559#endif /* VBOX_WITH_GUEST_CONTROL */
2560}
2561
2562STDMETHODIMP GuestSession::EnvironmentClear(void)
2563{
2564#ifndef VBOX_WITH_GUEST_CONTROL
2565 ReturnComNotImplemented();
2566#else
2567 LogFlowThisFuncEnter();
2568
2569 AutoCaller autoCaller(this);
2570 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2571
2572 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2573
2574 mData.mEnvironment.Clear();
2575
2576 LogFlowThisFuncLeave();
2577 return S_OK;
2578#endif /* VBOX_WITH_GUEST_CONTROL */
2579}
2580
2581STDMETHODIMP GuestSession::EnvironmentGet(IN_BSTR aName, BSTR *aValue)
2582{
2583#ifndef VBOX_WITH_GUEST_CONTROL
2584 ReturnComNotImplemented();
2585#else
2586 LogFlowThisFuncEnter();
2587
2588 if (RT_UNLIKELY((aName) == NULL || *(aName) == '\0'))
2589 return setError(E_INVALIDARG, tr("No value name specified"));
2590
2591 CheckComArgOutPointerValid(aValue);
2592
2593 AutoCaller autoCaller(this);
2594 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2595
2596 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2597
2598 Bstr strValue(mData.mEnvironment.Get(Utf8Str(aName)));
2599 strValue.cloneTo(aValue);
2600
2601 LogFlowThisFuncLeave();
2602 return S_OK;
2603#endif /* VBOX_WITH_GUEST_CONTROL */
2604}
2605
2606STDMETHODIMP GuestSession::EnvironmentSet(IN_BSTR aName, IN_BSTR aValue)
2607{
2608#ifndef VBOX_WITH_GUEST_CONTROL
2609 ReturnComNotImplemented();
2610#else
2611 LogFlowThisFuncEnter();
2612
2613 if (RT_UNLIKELY((aName) == NULL || *(aName) == '\0'))
2614 return setError(E_INVALIDARG, tr("No value name specified"));
2615
2616 AutoCaller autoCaller(this);
2617 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2618
2619 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2620
2621 int rc = mData.mEnvironment.Set(Utf8Str(aName), Utf8Str(aValue));
2622
2623 HRESULT hr = RT_SUCCESS(rc) ? S_OK : VBOX_E_IPRT_ERROR;
2624 LogFlowFuncLeaveRC(hr);
2625 return hr;
2626#endif /* VBOX_WITH_GUEST_CONTROL */
2627}
2628
2629STDMETHODIMP GuestSession::EnvironmentUnset(IN_BSTR aName)
2630{
2631#ifndef VBOX_WITH_GUEST_CONTROL
2632 ReturnComNotImplemented();
2633#else
2634 LogFlowThisFuncEnter();
2635
2636 AutoCaller autoCaller(this);
2637 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2638
2639 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2640
2641 mData.mEnvironment.Unset(Utf8Str(aName));
2642
2643 LogFlowThisFuncLeave();
2644 return S_OK;
2645#endif /* VBOX_WITH_GUEST_CONTROL */
2646}
2647
2648STDMETHODIMP GuestSession::FileCreateTemp(IN_BSTR aTemplate, ULONG aMode, IN_BSTR aPath, BOOL aSecure, IGuestFile **aFile)
2649{
2650#ifndef VBOX_WITH_GUEST_CONTROL
2651 ReturnComNotImplemented();
2652#else
2653 LogFlowThisFuncEnter();
2654
2655 AutoCaller autoCaller(this);
2656 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2657
2658 ReturnComNotImplemented();
2659#endif /* VBOX_WITH_GUEST_CONTROL */
2660}
2661
2662STDMETHODIMP GuestSession::FileExists(IN_BSTR aPath, BOOL *aExists)
2663{
2664#ifndef VBOX_WITH_GUEST_CONTROL
2665 ReturnComNotImplemented();
2666#else
2667 LogFlowThisFuncEnter();
2668
2669 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2670 return setError(E_INVALIDARG, tr("No file to check existence for specified"));
2671 CheckComArgOutPointerValid(aExists);
2672
2673 AutoCaller autoCaller(this);
2674 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2675
2676 GuestFsObjData objData; int guestRc;
2677 int vrc = fileQueryInfoInternal(Utf8Str(aPath), objData, &guestRc);
2678 if (RT_SUCCESS(vrc))
2679 {
2680 *aExists = TRUE;
2681 return S_OK;
2682 }
2683
2684 HRESULT hr = S_OK;
2685
2686 switch (vrc)
2687 {
2688 case VERR_GSTCTL_GUEST_ERROR:
2689 hr = GuestProcess::setErrorExternal(this, guestRc);
2690 break;
2691
2692 case VERR_NOT_A_FILE:
2693 *aExists = FALSE;
2694 break;
2695
2696 default:
2697 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying file information for \"%s\" failed: %Rrc"),
2698 Utf8Str(aPath).c_str(), vrc);
2699 break;
2700 }
2701
2702 return hr;
2703#endif /* VBOX_WITH_GUEST_CONTROL */
2704}
2705
2706STDMETHODIMP GuestSession::FileRemove(IN_BSTR aPath)
2707{
2708#ifndef VBOX_WITH_GUEST_CONTROL
2709 ReturnComNotImplemented();
2710#else
2711 LogFlowThisFuncEnter();
2712
2713 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2714 return setError(E_INVALIDARG, tr("No file to remove specified"));
2715
2716 AutoCaller autoCaller(this);
2717 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2718
2719 HRESULT hr = S_OK;
2720
2721 int guestRc;
2722 int vrc = fileRemoveInternal(Utf8Str(aPath), &guestRc);
2723 if (RT_FAILURE(vrc))
2724 {
2725 switch (vrc)
2726 {
2727 case VERR_GSTCTL_GUEST_ERROR:
2728 hr = GuestProcess::setErrorExternal(this, guestRc);
2729 break;
2730
2731 default:
2732 hr = setError(VBOX_E_IPRT_ERROR, tr("Removing file \"%s\" failed: %Rrc"),
2733 Utf8Str(aPath).c_str(), vrc);
2734 break;
2735 }
2736 }
2737
2738 return hr;
2739#endif /* VBOX_WITH_GUEST_CONTROL */
2740}
2741
2742STDMETHODIMP GuestSession::FileOpen(IN_BSTR aPath, IN_BSTR aOpenMode, IN_BSTR aDisposition, ULONG aCreationMode, IGuestFile **aFile)
2743{
2744#ifndef VBOX_WITH_GUEST_CONTROL
2745 ReturnComNotImplemented();
2746#else
2747 LogFlowThisFuncEnter();
2748
2749 Bstr strSharingMode = ""; /* Sharing mode is ignored. */
2750
2751 return FileOpenEx(aPath, aOpenMode, aDisposition, strSharingMode.raw(), aCreationMode,
2752 0 /* aOffset */, aFile);
2753#endif /* VBOX_WITH_GUEST_CONTROL */
2754}
2755
2756STDMETHODIMP GuestSession::FileOpenEx(IN_BSTR aPath, IN_BSTR aOpenMode, IN_BSTR aDisposition, IN_BSTR aSharingMode,
2757 ULONG aCreationMode, LONG64 aOffset, IGuestFile **aFile)
2758{
2759#ifndef VBOX_WITH_GUEST_CONTROL
2760 ReturnComNotImplemented();
2761#else
2762 LogFlowThisFuncEnter();
2763
2764 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2765 return setError(E_INVALIDARG, tr("No file to open specified"));
2766 if (RT_UNLIKELY((aOpenMode) == NULL || *(aOpenMode) == '\0'))
2767 return setError(E_INVALIDARG, tr("No open mode specified"));
2768 if (RT_UNLIKELY((aDisposition) == NULL || *(aDisposition) == '\0'))
2769 return setError(E_INVALIDARG, tr("No disposition mode specified"));
2770 /* aSharingMode is optional. */
2771
2772 CheckComArgOutPointerValid(aFile);
2773
2774 AutoCaller autoCaller(this);
2775 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2776
2777 HRESULT hr = isReadyExternal();
2778 if (FAILED(hr))
2779 return hr;
2780
2781 /** @todo Validate creation mode. */
2782 uint32_t uCreationMode = 0;
2783
2784 GuestFileOpenInfo openInfo;
2785 openInfo.mFileName = Utf8Str(aPath);
2786 openInfo.mOpenMode = Utf8Str(aOpenMode);
2787 openInfo.mDisposition = Utf8Str(aDisposition);
2788 openInfo.mSharingMode = Utf8Str(aSharingMode);
2789 openInfo.mCreationMode = aCreationMode;
2790 openInfo.mInitialOffset = aOffset;
2791
2792 uint64_t uFlagsIgnored;
2793 int vrc = RTFileModeToFlagsEx(openInfo.mOpenMode.c_str(),
2794 openInfo.mDisposition.c_str(),
2795 openInfo.mSharingMode.c_str(),
2796 &uFlagsIgnored);
2797 if (RT_FAILURE(vrc))
2798 return setError(E_INVALIDARG, tr("Invalid open mode / disposition / sharing mode specified"));
2799
2800 ComObjPtr <GuestFile> pFile; int guestRc;
2801 vrc = fileOpenInternal(openInfo, pFile, &guestRc);
2802 if (RT_SUCCESS(vrc))
2803 {
2804 /* Return directory object to the caller. */
2805 hr = pFile.queryInterfaceTo(aFile);
2806 }
2807 else
2808 {
2809 switch (vrc)
2810 {
2811 case VERR_NOT_SUPPORTED:
2812 hr = setError(VBOX_E_IPRT_ERROR,
2813 tr("Handling guest files not supported by installed Guest Additions"));
2814 break;
2815
2816 case VERR_GSTCTL_GUEST_ERROR:
2817 hr = GuestFile::setErrorExternal(this, guestRc);
2818 break;
2819
2820 default:
2821 hr = setError(VBOX_E_IPRT_ERROR, tr("Opening guest file \"%s\" failed: %Rrc"),
2822 Utf8Str(aPath).c_str(), vrc);
2823 break;
2824 }
2825 }
2826
2827 return hr;
2828#endif /* VBOX_WITH_GUEST_CONTROL */
2829}
2830
2831STDMETHODIMP GuestSession::FileQueryInfo(IN_BSTR aPath, IGuestFsObjInfo **aInfo)
2832{
2833#ifndef VBOX_WITH_GUEST_CONTROL
2834 ReturnComNotImplemented();
2835#else
2836 LogFlowThisFuncEnter();
2837
2838 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2839 return setError(E_INVALIDARG, tr("No file to query information for specified"));
2840 CheckComArgOutPointerValid(aInfo);
2841
2842 AutoCaller autoCaller(this);
2843 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2844
2845 HRESULT hr = S_OK;
2846
2847 GuestFsObjData objData; int guestRc;
2848 int vrc = fileQueryInfoInternal(Utf8Str(aPath), objData, &guestRc);
2849 if (RT_SUCCESS(vrc))
2850 {
2851 ComObjPtr<GuestFsObjInfo> pFsObjInfo;
2852 hr = pFsObjInfo.createObject();
2853 if (FAILED(hr)) return hr;
2854
2855 vrc = pFsObjInfo->init(objData);
2856 if (RT_SUCCESS(vrc))
2857 {
2858 hr = pFsObjInfo.queryInterfaceTo(aInfo);
2859 if (FAILED(hr)) return hr;
2860 }
2861 }
2862
2863 if (RT_FAILURE(vrc))
2864 {
2865 switch (vrc)
2866 {
2867 case VERR_GSTCTL_GUEST_ERROR:
2868 hr = GuestProcess::setErrorExternal(this, guestRc);
2869 break;
2870
2871 case VERR_NOT_A_FILE:
2872 hr = setError(VBOX_E_IPRT_ERROR, tr("Element exists but is not a file"));
2873 break;
2874
2875 default:
2876 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying file information failed: %Rrc"), vrc);
2877 break;
2878 }
2879 }
2880
2881 return hr;
2882#endif /* VBOX_WITH_GUEST_CONTROL */
2883}
2884
2885STDMETHODIMP GuestSession::FileQuerySize(IN_BSTR aPath, LONG64 *aSize)
2886{
2887#ifndef VBOX_WITH_GUEST_CONTROL
2888 ReturnComNotImplemented();
2889#else
2890 LogFlowThisFuncEnter();
2891
2892 if (RT_UNLIKELY((aPath) == NULL || *(aPath) == '\0'))
2893 return setError(E_INVALIDARG, tr("No file to query size for specified"));
2894 CheckComArgOutPointerValid(aSize);
2895
2896 AutoCaller autoCaller(this);
2897 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2898
2899 HRESULT hr = S_OK;
2900
2901 int64_t llSize; int guestRc;
2902 int vrc = fileQuerySizeInternal(Utf8Str(aPath), &llSize, &guestRc);
2903 if (RT_SUCCESS(vrc))
2904 {
2905 *aSize = llSize;
2906 }
2907 else
2908 {
2909 switch (vrc)
2910 {
2911 case VERR_GSTCTL_GUEST_ERROR:
2912 hr = GuestProcess::setErrorExternal(this, guestRc);
2913 break;
2914
2915 default:
2916 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying file size failed: %Rrc"), vrc);
2917 break;
2918 }
2919 }
2920
2921 return hr;
2922#endif /* VBOX_WITH_GUEST_CONTROL */
2923}
2924
2925STDMETHODIMP GuestSession::FileRename(IN_BSTR aSource, IN_BSTR aDest, ComSafeArrayIn(PathRenameFlag_T, aFlags))
2926{
2927#ifndef VBOX_WITH_GUEST_CONTROL
2928 ReturnComNotImplemented();
2929#else
2930 LogFlowThisFuncEnter();
2931
2932 AutoCaller autoCaller(this);
2933 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2934
2935 ReturnComNotImplemented();
2936#endif /* VBOX_WITH_GUEST_CONTROL */
2937}
2938
2939STDMETHODIMP GuestSession::FileSetACL(IN_BSTR aPath, IN_BSTR aACL)
2940{
2941#ifndef VBOX_WITH_GUEST_CONTROL
2942 ReturnComNotImplemented();
2943#else
2944 LogFlowThisFuncEnter();
2945
2946 AutoCaller autoCaller(this);
2947 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2948
2949 ReturnComNotImplemented();
2950#endif /* VBOX_WITH_GUEST_CONTROL */
2951}
2952
2953STDMETHODIMP GuestSession::ProcessCreate(IN_BSTR aCommand, ComSafeArrayIn(IN_BSTR, aArguments), ComSafeArrayIn(IN_BSTR, aEnvironment),
2954 ComSafeArrayIn(ProcessCreateFlag_T, aFlags), ULONG aTimeoutMS, IGuestProcess **aProcess)
2955{
2956#ifndef VBOX_WITH_GUEST_CONTROL
2957 ReturnComNotImplemented();
2958#else
2959 LogFlowThisFuncEnter();
2960
2961 com::SafeArray<LONG> affinityIgnored;
2962
2963 return ProcessCreateEx(aCommand, ComSafeArrayInArg(aArguments), ComSafeArrayInArg(aEnvironment),
2964 ComSafeArrayInArg(aFlags), aTimeoutMS, ProcessPriority_Default, ComSafeArrayAsInParam(affinityIgnored), aProcess);
2965#endif /* VBOX_WITH_GUEST_CONTROL */
2966}
2967
2968STDMETHODIMP GuestSession::ProcessCreateEx(IN_BSTR aCommand, ComSafeArrayIn(IN_BSTR, aArguments), ComSafeArrayIn(IN_BSTR, aEnvironment),
2969 ComSafeArrayIn(ProcessCreateFlag_T, aFlags), ULONG aTimeoutMS,
2970 ProcessPriority_T aPriority, ComSafeArrayIn(LONG, aAffinity),
2971 IGuestProcess **aProcess)
2972{
2973#ifndef VBOX_WITH_GUEST_CONTROL
2974 ReturnComNotImplemented();
2975#else
2976 LogFlowThisFuncEnter();
2977
2978 if (RT_UNLIKELY((aCommand) == NULL || *(aCommand) == '\0'))
2979 return setError(E_INVALIDARG, tr("No command to execute specified"));
2980 CheckComArgOutPointerValid(aProcess);
2981
2982 AutoCaller autoCaller(this);
2983 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2984
2985 HRESULT hr = isReadyExternal();
2986 if (FAILED(hr))
2987 return hr;
2988
2989 GuestProcessStartupInfo procInfo;
2990 procInfo.mCommand = Utf8Str(aCommand);
2991
2992 if (aArguments)
2993 {
2994 com::SafeArray<IN_BSTR> arguments(ComSafeArrayInArg(aArguments));
2995 for (size_t i = 0; i < arguments.size(); i++)
2996 procInfo.mArguments.push_back(Utf8Str(arguments[i]));
2997 }
2998
2999 int rc = VINF_SUCCESS;
3000
3001 /*
3002 * Create the process environment:
3003 * - Apply the session environment in a first step, and
3004 * - Apply environment variables specified by this call to
3005 * have the chance of overwriting/deleting session entries.
3006 */
3007 procInfo.mEnvironment = mData.mEnvironment; /* Apply original session environment. */
3008
3009 if (aEnvironment)
3010 {
3011 com::SafeArray<IN_BSTR> environment(ComSafeArrayInArg(aEnvironment));
3012 for (size_t i = 0; i < environment.size() && RT_SUCCESS(rc); i++)
3013 rc = procInfo.mEnvironment.Set(Utf8Str(environment[i]));
3014 }
3015
3016 if (RT_SUCCESS(rc))
3017 {
3018 if (aFlags)
3019 {
3020 com::SafeArray<ProcessCreateFlag_T> flags(ComSafeArrayInArg(aFlags));
3021 for (size_t i = 0; i < flags.size(); i++)
3022 procInfo.mFlags |= flags[i];
3023 }
3024
3025 procInfo.mTimeoutMS = aTimeoutMS;
3026
3027 if (aAffinity)
3028 {
3029 com::SafeArray<LONG> affinity(ComSafeArrayInArg(aAffinity));
3030 for (size_t i = 0; i < affinity.size(); i++)
3031 {
3032 if (affinity[i])
3033 procInfo.mAffinity |= (uint64_t)1 << i;
3034 }
3035 }
3036
3037 procInfo.mPriority = aPriority;
3038
3039 ComObjPtr<GuestProcess> pProcess;
3040 rc = processCreateExInteral(procInfo, pProcess);
3041 if (RT_SUCCESS(rc))
3042 {
3043 /* Return guest session to the caller. */
3044 HRESULT hr2 = pProcess.queryInterfaceTo(aProcess);
3045 if (FAILED(hr2))
3046 rc = VERR_COM_OBJECT_NOT_FOUND;
3047
3048 if (RT_SUCCESS(rc))
3049 rc = pProcess->startProcessAsync();
3050 }
3051 }
3052
3053 if (RT_FAILURE(rc))
3054 {
3055 switch (rc)
3056 {
3057 case VERR_MAX_PROCS_REACHED:
3058 hr = setError(VBOX_E_IPRT_ERROR, tr("Maximum number of concurrent guest processes per session (%ld) reached"),
3059 VBOX_GUESTCTRL_MAX_OBJECTS);
3060 break;
3061
3062 /** @todo Add more errors here. */
3063
3064 default:
3065 hr = setError(VBOX_E_IPRT_ERROR, tr("Could not create guest process, rc=%Rrc"), rc);
3066 break;
3067 }
3068 }
3069
3070 LogFlowFuncLeaveRC(rc);
3071 return hr;
3072#endif /* VBOX_WITH_GUEST_CONTROL */
3073}
3074
3075STDMETHODIMP GuestSession::ProcessGet(ULONG aPID, IGuestProcess **aProcess)
3076{
3077#ifndef VBOX_WITH_GUEST_CONTROL
3078 ReturnComNotImplemented();
3079#else
3080 LogFlowThisFunc(("aPID=%RU32\n", aPID));
3081
3082 CheckComArgOutPointerValid(aProcess);
3083 if (aPID == 0)
3084 return setError(E_INVALIDARG, tr("No valid process ID (PID) specified"));
3085
3086 AutoCaller autoCaller(this);
3087 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3088
3089 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3090
3091 HRESULT hr = S_OK;
3092
3093 ComObjPtr<GuestProcess> pProcess;
3094 int rc = processGetByPID(aPID, &pProcess);
3095 if (RT_FAILURE(rc))
3096 hr = setError(E_INVALIDARG, tr("No process with PID %RU32 found"), aPID);
3097
3098 /* This will set (*aProcess) to NULL if pProgress is NULL. */
3099 HRESULT hr2 = pProcess.queryInterfaceTo(aProcess);
3100 if (SUCCEEDED(hr))
3101 hr = hr2;
3102
3103 LogFlowThisFunc(("aProcess=%p, hr=%Rhrc\n", *aProcess, hr));
3104 return hr;
3105#endif /* VBOX_WITH_GUEST_CONTROL */
3106}
3107
3108STDMETHODIMP GuestSession::SymlinkCreate(IN_BSTR aSource, IN_BSTR aTarget, SymlinkType_T aType)
3109{
3110#ifndef VBOX_WITH_GUEST_CONTROL
3111 ReturnComNotImplemented();
3112#else
3113 LogFlowThisFuncEnter();
3114
3115 AutoCaller autoCaller(this);
3116 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3117
3118 ReturnComNotImplemented();
3119#endif /* VBOX_WITH_GUEST_CONTROL */
3120}
3121
3122STDMETHODIMP GuestSession::SymlinkExists(IN_BSTR aSymlink, BOOL *aExists)
3123{
3124#ifndef VBOX_WITH_GUEST_CONTROL
3125 ReturnComNotImplemented();
3126#else
3127 LogFlowThisFuncEnter();
3128
3129 AutoCaller autoCaller(this);
3130 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3131
3132 ReturnComNotImplemented();
3133#endif /* VBOX_WITH_GUEST_CONTROL */
3134}
3135
3136STDMETHODIMP GuestSession::SymlinkRead(IN_BSTR aSymlink, ComSafeArrayIn(SymlinkReadFlag_T, aFlags), BSTR *aTarget)
3137{
3138#ifndef VBOX_WITH_GUEST_CONTROL
3139 ReturnComNotImplemented();
3140#else
3141 LogFlowThisFuncEnter();
3142
3143 AutoCaller autoCaller(this);
3144 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3145
3146 ReturnComNotImplemented();
3147#endif /* VBOX_WITH_GUEST_CONTROL */
3148}
3149
3150STDMETHODIMP GuestSession::SymlinkRemoveDirectory(IN_BSTR aPath)
3151{
3152#ifndef VBOX_WITH_GUEST_CONTROL
3153 ReturnComNotImplemented();
3154#else
3155 LogFlowThisFuncEnter();
3156
3157 AutoCaller autoCaller(this);
3158 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3159
3160 ReturnComNotImplemented();
3161#endif /* VBOX_WITH_GUEST_CONTROL */
3162}
3163
3164STDMETHODIMP GuestSession::SymlinkRemoveFile(IN_BSTR aFile)
3165{
3166#ifndef VBOX_WITH_GUEST_CONTROL
3167 ReturnComNotImplemented();
3168#else
3169 LogFlowThisFuncEnter();
3170
3171 AutoCaller autoCaller(this);
3172 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3173
3174 ReturnComNotImplemented();
3175#endif /* VBOX_WITH_GUEST_CONTROL */
3176}
3177
3178STDMETHODIMP GuestSession::WaitFor(ULONG aWaitFlags, ULONG aTimeoutMS, GuestSessionWaitResult_T *aReason)
3179{
3180#ifndef VBOX_WITH_GUEST_CONTROL
3181 ReturnComNotImplemented();
3182#else
3183 LogFlowThisFuncEnter();
3184
3185 CheckComArgOutPointerValid(aReason);
3186
3187 AutoCaller autoCaller(this);
3188 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3189
3190 /*
3191 * Note: Do not hold any locks here while waiting!
3192 */
3193 HRESULT hr = S_OK;
3194
3195 int guestRc; GuestSessionWaitResult_T waitResult;
3196 int vrc = waitFor(aWaitFlags, aTimeoutMS, waitResult, &guestRc);
3197 if (RT_SUCCESS(vrc))
3198 {
3199 *aReason = waitResult;
3200 }
3201 else
3202 {
3203 switch (vrc)
3204 {
3205 case VERR_GSTCTL_GUEST_ERROR:
3206 hr = GuestSession::setErrorExternal(this, guestRc);
3207 break;
3208
3209 case VERR_TIMEOUT:
3210 *aReason = GuestSessionWaitResult_Timeout;
3211 break;
3212
3213 default:
3214 {
3215 const char *pszSessionName = mData.mSession.mName.c_str();
3216 hr = setError(VBOX_E_IPRT_ERROR,
3217 tr("Waiting for guest session \"%s\" failed: %Rrc"),
3218 pszSessionName ? pszSessionName : tr("Unnamed"), vrc);
3219 break;
3220 }
3221 }
3222 }
3223
3224 LogFlowFuncLeaveRC(vrc);
3225 return hr;
3226#endif /* VBOX_WITH_GUEST_CONTROL */
3227}
3228
3229STDMETHODIMP GuestSession::WaitForArray(ComSafeArrayIn(GuestSessionWaitForFlag_T, aFlags), ULONG aTimeoutMS, GuestSessionWaitResult_T *aReason)
3230{
3231#ifndef VBOX_WITH_GUEST_CONTROL
3232 ReturnComNotImplemented();
3233#else
3234 LogFlowThisFuncEnter();
3235
3236 CheckComArgOutPointerValid(aReason);
3237
3238 AutoCaller autoCaller(this);
3239 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3240
3241 /*
3242 * Note: Do not hold any locks here while waiting!
3243 */
3244 uint32_t fWaitFor = GuestSessionWaitForFlag_None;
3245 com::SafeArray<GuestSessionWaitForFlag_T> flags(ComSafeArrayInArg(aFlags));
3246 for (size_t i = 0; i < flags.size(); i++)
3247 fWaitFor |= flags[i];
3248
3249 return WaitFor(fWaitFor, aTimeoutMS, aReason);
3250#endif /* VBOX_WITH_GUEST_CONTROL */
3251}
3252
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