VirtualBox

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

Last change on this file since 63185 was 63182, checked in by vboxsync, 8 years ago

ThreadTask: split createThread up into three methods to avoid having to pass NULL all the time when the type needs specifying. Also, explictly marked the racy variant.

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