VirtualBox

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

Last change on this file since 62497 was 62379, checked in by vboxsync, 9 years ago

Main: 2nd try: fixes for a few -Wunused -Wconversion

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 111.8 KB
Line 
1/* $Id: GuestSessionImpl.cpp 62379 2016-07-20 20:11:50Z 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 int vrc = VINF_SUCCESS;
830
831 GuestProcessStartupInfo procInfo;
832 procInfo.mFlags = ProcessCreateFlag_WaitForStdOut;
833 procInfo.mExecutable = Utf8Str(VBOXSERVICE_TOOL_MKTEMP);
834
835 try
836 {
837 procInfo.mArguments.push_back(procInfo.mExecutable); /* Set argv0. */
838 procInfo.mArguments.push_back(Utf8Str("--machinereadable"));
839 if (fDirectory)
840 procInfo.mArguments.push_back(Utf8Str("-d"));
841 if (strPath.length()) /* Otherwise use /tmp or equivalent. */
842 {
843 procInfo.mArguments.push_back(Utf8Str("-t"));
844 procInfo.mArguments.push_back(strPath);
845 }
846 procInfo.mArguments.push_back("--"); /* strTemplate could be '--help'. */
847 procInfo.mArguments.push_back(strTemplate);
848 }
849 catch (std::bad_alloc)
850 {
851 vrc = VERR_NO_MEMORY;
852 }
853
854 /** @todo Use an internal HGCM command for this operation, since
855 * we now can run in a user-dedicated session. */
856 int guestRc;
857 GuestCtrlStreamObjects stdOut;
858 if (RT_SUCCESS(vrc))
859 vrc = GuestProcessTool::i_runEx(this, procInfo,
860 &stdOut, 1 /* cStrmOutObjects */,
861 &guestRc);
862
863 if (!GuestProcess::i_isGuestError(vrc))
864 {
865 GuestFsObjData objData;
866 if (!stdOut.empty())
867 {
868 vrc = objData.FromMkTemp(stdOut.at(0));
869 if (RT_FAILURE(vrc))
870 {
871 guestRc = vrc;
872 vrc = VERR_GSTCTL_GUEST_ERROR;
873 }
874 }
875 else
876 vrc = VERR_BROKEN_PIPE;
877
878 if (RT_SUCCESS(vrc))
879 strName = objData.mName;
880 }
881 else if (pGuestRc)
882 {
883 *pGuestRc = guestRc;
884 }
885
886 LogFlowThisFunc(("Returning rc=%Rrc, guestRc=%Rrc\n", vrc, guestRc));
887 return vrc;
888}
889
890int GuestSession::i_directoryOpenInternal(const GuestDirectoryOpenInfo &openInfo,
891 ComObjPtr<GuestDirectory> &pDirectory, int *pGuestRc)
892{
893 AssertPtrReturn(pGuestRc, VERR_INVALID_POINTER);
894
895 LogFlowThisFunc(("strPath=%s, strPath=%s, uFlags=%x\n",
896 openInfo.mPath.c_str(), openInfo.mFilter.c_str(), openInfo.mFlags));
897
898 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
899
900 int rc = VERR_MAX_PROCS_REACHED;
901 if (mData.mNumObjects >= VBOX_GUESTCTRL_MAX_OBJECTS)
902 return rc;
903
904 /* Create a new (host-based) directory ID and assign it. */
905 uint32_t uNewDirID = 0;
906 ULONG uTries = 0;
907
908 for (;;)
909 {
910 /* Is the directory ID already used? */
911 if (!i_directoryExists(uNewDirID, NULL /* pDirectory */))
912 {
913 /* Callback with context ID was not found. This means
914 * we can use this context ID for our new callback we want
915 * to add below. */
916 rc = VINF_SUCCESS;
917 break;
918 }
919 uNewDirID++;
920 if (uNewDirID == VBOX_GUESTCTRL_MAX_OBJECTS)
921 uNewDirID = 0;
922
923 if (++uTries == UINT32_MAX)
924 break; /* Don't try too hard. */
925 }
926
927 if (RT_FAILURE(rc))
928 return rc;
929
930 /* Create the directory object. */
931 HRESULT hr = pDirectory.createObject();
932 if (FAILED(hr))
933 return VERR_COM_UNEXPECTED;
934
935 Console *pConsole = mParent->i_getConsole();
936 AssertPtr(pConsole);
937
938 int vrc = pDirectory->init(pConsole, this /* Parent */,
939 uNewDirID, openInfo);
940 if (RT_FAILURE(vrc))
941 return vrc;
942
943 /*
944 * Since this is a synchronous guest call we have to
945 * register the file object first, releasing the session's
946 * lock and then proceed with the actual opening command
947 * -- otherwise the file's opening callback would hang
948 * because the session's lock still is in place.
949 */
950 try
951 {
952 /* Add the created directory to our map. */
953 mData.mDirectories[uNewDirID] = pDirectory;
954 mData.mNumObjects++;
955 Assert(mData.mNumObjects <= VBOX_GUESTCTRL_MAX_OBJECTS);
956
957 LogFlowFunc(("Added new guest directory \"%s\" (Session: %RU32) (now total %zu dirs, %RU32 objects)\n",
958 openInfo.mPath.c_str(), mData.mSession.mID, mData.mFiles.size(), mData.mNumObjects));
959
960 alock.release(); /* Release lock before firing off event. */
961
962 /** @todo Fire off a VBoxEventType_OnGuestDirectoryRegistered event? */
963 }
964 catch (std::bad_alloc &)
965 {
966 rc = VERR_NO_MEMORY;
967 }
968
969 if (RT_SUCCESS(rc))
970 {
971 /* Nothing further to do here yet. */
972 if (pGuestRc)
973 *pGuestRc = VINF_SUCCESS;
974 }
975
976 LogFlowFuncLeaveRC(vrc);
977 return vrc;
978}
979
980int GuestSession::i_dispatchToDirectory(PVBOXGUESTCTRLHOSTCBCTX pCtxCb, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
981{
982 LogFlowFunc(("pCtxCb=%p, pSvcCb=%p\n", pCtxCb, pSvcCb));
983
984 AssertPtrReturn(pCtxCb, VERR_INVALID_POINTER);
985 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
986
987 if (pSvcCb->mParms < 3)
988 return VERR_INVALID_PARAMETER;
989
990 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
991
992 uint32_t uDirID = VBOX_GUESTCTRL_CONTEXTID_GET_OBJECT(pCtxCb->uContextID);
993#ifdef DEBUG
994 LogFlowFunc(("uDirID=%RU32 (%zu total)\n",
995 uDirID, mData.mFiles.size()));
996#endif
997 int rc;
998 SessionDirectories::const_iterator itDir
999 = mData.mDirectories.find(uDirID);
1000 if (itDir != mData.mDirectories.end())
1001 {
1002 ComObjPtr<GuestDirectory> pDirectory(itDir->second);
1003 Assert(!pDirectory.isNull());
1004
1005 alock.release();
1006
1007 rc = pDirectory->i_callbackDispatcher(pCtxCb, pSvcCb);
1008 }
1009 else
1010 rc = VERR_NOT_FOUND;
1011
1012 LogFlowFuncLeaveRC(rc);
1013 return rc;
1014}
1015
1016int GuestSession::i_dispatchToFile(PVBOXGUESTCTRLHOSTCBCTX pCtxCb, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
1017{
1018 LogFlowFunc(("pCtxCb=%p, pSvcCb=%p\n", pCtxCb, pSvcCb));
1019
1020 AssertPtrReturn(pCtxCb, VERR_INVALID_POINTER);
1021 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
1022
1023 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1024
1025 uint32_t uFileID = VBOX_GUESTCTRL_CONTEXTID_GET_OBJECT(pCtxCb->uContextID);
1026#ifdef DEBUG
1027 LogFlowFunc(("uFileID=%RU32 (%zu total)\n",
1028 uFileID, mData.mFiles.size()));
1029#endif
1030 int rc;
1031 SessionFiles::const_iterator itFile
1032 = mData.mFiles.find(uFileID);
1033 if (itFile != mData.mFiles.end())
1034 {
1035 ComObjPtr<GuestFile> pFile(itFile->second);
1036 Assert(!pFile.isNull());
1037
1038 alock.release();
1039
1040 rc = pFile->i_callbackDispatcher(pCtxCb, pSvcCb);
1041 }
1042 else
1043 rc = VERR_NOT_FOUND;
1044
1045 LogFlowFuncLeaveRC(rc);
1046 return rc;
1047}
1048
1049int GuestSession::i_dispatchToObject(PVBOXGUESTCTRLHOSTCBCTX pCtxCb, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
1050{
1051 LogFlowFunc(("pCtxCb=%p, pSvcCb=%p\n", pCtxCb, pSvcCb));
1052
1053 AssertPtrReturn(pCtxCb, VERR_INVALID_POINTER);
1054 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
1055
1056 int rc;
1057 uint32_t uObjectID = VBOX_GUESTCTRL_CONTEXTID_GET_OBJECT(pCtxCb->uContextID);
1058
1059 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1060
1061 /* Since we don't know which type the object is, we need to through all
1062 * all objects. */
1063 /** @todo Speed this up by adding an object type to the callback context! */
1064 SessionProcesses::const_iterator itProc = mData.mProcesses.find(uObjectID);
1065 if (itProc == mData.mProcesses.end())
1066 {
1067 SessionFiles::const_iterator itFile = mData.mFiles.find(uObjectID);
1068 if (itFile != mData.mFiles.end())
1069 {
1070 alock.release();
1071
1072 rc = i_dispatchToFile(pCtxCb, pSvcCb);
1073 }
1074 else
1075 {
1076 SessionDirectories::const_iterator itDir = mData.mDirectories.find(uObjectID);
1077 if (itDir != mData.mDirectories.end())
1078 {
1079 alock.release();
1080
1081 rc = i_dispatchToDirectory(pCtxCb, pSvcCb);
1082 }
1083 else
1084 rc = VERR_NOT_FOUND;
1085 }
1086 }
1087 else
1088 {
1089 alock.release();
1090
1091 rc = i_dispatchToProcess(pCtxCb, pSvcCb);
1092 }
1093
1094 LogFlowFuncLeaveRC(rc);
1095 return rc;
1096}
1097
1098int GuestSession::i_dispatchToProcess(PVBOXGUESTCTRLHOSTCBCTX pCtxCb, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
1099{
1100 LogFlowFunc(("pCtxCb=%p, pSvcCb=%p\n", pCtxCb, pSvcCb));
1101
1102 AssertPtrReturn(pCtxCb, VERR_INVALID_POINTER);
1103 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
1104
1105 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1106
1107 uint32_t uProcessID = VBOX_GUESTCTRL_CONTEXTID_GET_OBJECT(pCtxCb->uContextID);
1108#ifdef DEBUG
1109 LogFlowFunc(("uProcessID=%RU32 (%zu total)\n",
1110 uProcessID, mData.mProcesses.size()));
1111#endif
1112 int rc;
1113 SessionProcesses::const_iterator itProc
1114 = mData.mProcesses.find(uProcessID);
1115 if (itProc != mData.mProcesses.end())
1116 {
1117#ifdef DEBUG_andy
1118 ULONG cRefs = itProc->second->AddRef();
1119 Assert(cRefs >= 2);
1120 LogFlowFunc(("pProcess=%p, cRefs=%RU32\n", &itProc->second, cRefs - 1));
1121 itProc->second->Release();
1122#endif
1123 ComObjPtr<GuestProcess> pProcess(itProc->second);
1124 Assert(!pProcess.isNull());
1125
1126 /* Set protocol version so that pSvcCb can
1127 * be interpreted right. */
1128 pCtxCb->uProtocol = mData.mProtocolVersion;
1129
1130 alock.release();
1131 rc = pProcess->i_callbackDispatcher(pCtxCb, pSvcCb);
1132 }
1133 else
1134 rc = VERR_NOT_FOUND;
1135
1136 LogFlowFuncLeaveRC(rc);
1137 return rc;
1138}
1139
1140int GuestSession::i_dispatchToThis(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb)
1141{
1142 AssertPtrReturn(pCbCtx, VERR_INVALID_POINTER);
1143 AssertPtrReturn(pSvcCb, VERR_INVALID_POINTER);
1144
1145 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1146
1147#ifdef DEBUG
1148 LogFlowThisFunc(("sessionID=%RU32, CID=%RU32, uFunction=%RU32, pSvcCb=%p\n",
1149 mData.mSession.mID, pCbCtx->uContextID, pCbCtx->uFunction, pSvcCb));
1150#endif
1151
1152 int rc;
1153 switch (pCbCtx->uFunction)
1154 {
1155 case GUEST_DISCONNECTED:
1156 /** @todo Handle closing all guest objects. */
1157 rc = VERR_INTERNAL_ERROR;
1158 break;
1159
1160 case GUEST_SESSION_NOTIFY: /* Guest Additions >= 4.3.0. */
1161 {
1162 rc = i_onSessionStatusChange(pCbCtx, pSvcCb);
1163 break;
1164 }
1165
1166 default:
1167 /* Silently skip unknown callbacks. */
1168 rc = VERR_NOT_SUPPORTED;
1169 break;
1170 }
1171
1172 LogFlowFuncLeaveRC(rc);
1173 return rc;
1174}
1175
1176inline bool GuestSession::i_fileExists(uint32_t uFileID, ComObjPtr<GuestFile> *pFile)
1177{
1178 SessionFiles::const_iterator it = mData.mFiles.find(uFileID);
1179 if (it != mData.mFiles.end())
1180 {
1181 if (pFile)
1182 *pFile = it->second;
1183 return true;
1184 }
1185 return false;
1186}
1187
1188int GuestSession::i_fileRemoveFromList(GuestFile *pFile)
1189{
1190 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1191
1192 int rc = VERR_NOT_FOUND;
1193
1194 SessionFiles::iterator itFiles = mData.mFiles.begin();
1195 while (itFiles != mData.mFiles.end())
1196 {
1197 if (pFile == itFiles->second)
1198 {
1199 /* Make sure to consume the pointer before the one of thfe
1200 * iterator gets released. */
1201 ComObjPtr<GuestFile> pCurFile = pFile;
1202
1203 Bstr strName;
1204 HRESULT hr = pCurFile->COMGETTER(FileName)(strName.asOutParam());
1205 ComAssertComRC(hr);
1206
1207 Assert(mData.mNumObjects);
1208 LogFlowThisFunc(("Removing guest file \"%s\" (Session: %RU32) (now total %zu files, %RU32 objects)\n",
1209 Utf8Str(strName).c_str(), mData.mSession.mID, mData.mFiles.size() - 1, mData.mNumObjects - 1));
1210
1211 rc = pFile->i_onRemove();
1212 mData.mFiles.erase(itFiles);
1213 mData.mNumObjects--;
1214
1215 alock.release(); /* Release lock before firing off event. */
1216
1217 fireGuestFileRegisteredEvent(mEventSource, this, pCurFile,
1218 false /* Unregistered */);
1219 pCurFile.setNull();
1220 break;
1221 }
1222
1223 ++itFiles;
1224 }
1225
1226 LogFlowFuncLeaveRC(rc);
1227 return rc;
1228}
1229
1230int GuestSession::i_fileRemoveInternal(const Utf8Str &strPath, int *pGuestRc)
1231{
1232 LogFlowThisFunc(("strPath=%s\n", strPath.c_str()));
1233
1234 int vrc = VINF_SUCCESS;
1235
1236 GuestProcessStartupInfo procInfo;
1237 GuestProcessStream streamOut;
1238
1239 procInfo.mFlags = ProcessCreateFlag_WaitForStdOut;
1240 procInfo.mExecutable = Utf8Str(VBOXSERVICE_TOOL_RM);
1241
1242 try
1243 {
1244 procInfo.mArguments.push_back(procInfo.mExecutable); /* Set argv0. */
1245 procInfo.mArguments.push_back(Utf8Str("--machinereadable"));
1246 procInfo.mArguments.push_back("--"); /* strPath could be '--help', which is a valid filename. */
1247 procInfo.mArguments.push_back(strPath); /* The file we want to remove. */
1248 }
1249 catch (std::bad_alloc)
1250 {
1251 vrc = VERR_NO_MEMORY;
1252 }
1253
1254 if (RT_SUCCESS(vrc))
1255 vrc = GuestProcessTool::i_run(this, procInfo, pGuestRc);
1256
1257 LogFlowFuncLeaveRC(vrc);
1258 return vrc;
1259}
1260
1261int GuestSession::i_fileOpenInternal(const GuestFileOpenInfo &openInfo,
1262 ComObjPtr<GuestFile> &pFile, int *pGuestRc)
1263{
1264 LogFlowThisFunc(("strFile=%s, enmAccessMode=%d (%s) enmOpenAction=%d (%s) uCreationMode=%RU32 mfOpenEx=%RU32\n",
1265 openInfo.mFileName.c_str(), openInfo.mAccessMode, openInfo.mpszAccessMode,
1266 openInfo.mOpenAction, openInfo.mpszOpenAction, openInfo.mCreationMode, openInfo.mfOpenEx));
1267
1268 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1269
1270 /* Guest Additions < 4.3 don't support handling
1271 guest files, skip. */
1272 if (mData.mProtocolVersion < 2)
1273 {
1274 LogFlowThisFunc(("Installed Guest Additions don't support handling guest files, skipping\n"));
1275 return VERR_NOT_SUPPORTED;
1276 }
1277
1278 int rc = VERR_MAX_PROCS_REACHED;
1279 if (mData.mNumObjects >= VBOX_GUESTCTRL_MAX_OBJECTS)
1280 return rc;
1281
1282 /* Create a new (host-based) file ID and assign it. */
1283 uint32_t uNewFileID = 0;
1284 ULONG uTries = 0;
1285
1286 for (;;)
1287 {
1288 /* Is the file ID already used? */
1289 if (!i_fileExists(uNewFileID, NULL /* pFile */))
1290 {
1291 /* Callback with context ID was not found. This means
1292 * we can use this context ID for our new callback we want
1293 * to add below. */
1294 rc = VINF_SUCCESS;
1295 break;
1296 }
1297 uNewFileID++;
1298 if (uNewFileID == VBOX_GUESTCTRL_MAX_OBJECTS)
1299 uNewFileID = 0;
1300
1301 if (++uTries == UINT32_MAX)
1302 break; /* Don't try too hard. */
1303 }
1304
1305 if (RT_FAILURE(rc))
1306 return rc;
1307
1308 /* Create the directory object. */
1309 HRESULT hr = pFile.createObject();
1310 if (FAILED(hr))
1311 return VERR_COM_UNEXPECTED;
1312
1313 Console *pConsole = mParent->i_getConsole();
1314 AssertPtr(pConsole);
1315
1316 rc = pFile->init(pConsole, this /* GuestSession */,
1317 uNewFileID, openInfo);
1318 if (RT_FAILURE(rc))
1319 return rc;
1320
1321 /*
1322 * Since this is a synchronous guest call we have to
1323 * register the file object first, releasing the session's
1324 * lock and then proceed with the actual opening command
1325 * -- otherwise the file's opening callback would hang
1326 * because the session's lock still is in place.
1327 */
1328 try
1329 {
1330 /* Add the created file to our vector. */
1331 mData.mFiles[uNewFileID] = pFile;
1332 mData.mNumObjects++;
1333 Assert(mData.mNumObjects <= VBOX_GUESTCTRL_MAX_OBJECTS);
1334
1335 LogFlowFunc(("Added new guest file \"%s\" (Session: %RU32) (now total %zu files, %RU32 objects)\n",
1336 openInfo.mFileName.c_str(), mData.mSession.mID, mData.mFiles.size(), mData.mNumObjects));
1337
1338 alock.release(); /* Release lock before firing off event. */
1339
1340 fireGuestFileRegisteredEvent(mEventSource, this, pFile,
1341 true /* Registered */);
1342 }
1343 catch (std::bad_alloc &)
1344 {
1345 rc = VERR_NO_MEMORY;
1346 }
1347
1348 if (RT_SUCCESS(rc))
1349 {
1350 int guestRc;
1351 rc = pFile->i_openFile(30 * 1000 /* 30s timeout */, &guestRc);
1352 if ( rc == VERR_GSTCTL_GUEST_ERROR
1353 && pGuestRc)
1354 {
1355 *pGuestRc = guestRc;
1356 }
1357 }
1358
1359 LogFlowFuncLeaveRC(rc);
1360 return rc;
1361}
1362
1363int GuestSession::i_fileQueryInfoInternal(const Utf8Str &strPath, bool fFollowSymlinks, GuestFsObjData &objData, int *pGuestRc)
1364{
1365 LogFlowThisFunc(("strPath=%s fFollowSymlinks=%RTbool\n", strPath.c_str(), fFollowSymlinks));
1366
1367 int vrc = i_fsQueryInfoInternal(strPath, fFollowSymlinks, objData, pGuestRc);
1368 if (RT_SUCCESS(vrc))
1369 {
1370 vrc = objData.mType == FsObjType_File
1371 ? VINF_SUCCESS : VERR_NOT_A_FILE;
1372 }
1373
1374 LogFlowFuncLeaveRC(vrc);
1375 return vrc;
1376}
1377
1378int GuestSession::i_fileQuerySizeInternal(const Utf8Str &strPath, bool fFollowSymlinks, int64_t *pllSize, int *pGuestRc)
1379{
1380 AssertPtrReturn(pllSize, VERR_INVALID_POINTER);
1381
1382 GuestFsObjData objData;
1383 int vrc = i_fileQueryInfoInternal(strPath, fFollowSymlinks, objData, pGuestRc);
1384 if (RT_SUCCESS(vrc))
1385 *pllSize = objData.mObjectSize;
1386
1387 return vrc;
1388}
1389
1390/**
1391 * Queries information of a file system object (file, directory, ...).
1392 *
1393 * @return IPRT status code.
1394 * @param strPath Path to file system object to query information for.
1395 * @param fFollowSymlinks Whether to follow symbolic links or not.
1396 * @param objData Where to return the file system object data, if found.
1397 * @param pGuestRc Guest rc, when returning VERR_GSTCTL_GUEST_ERROR.
1398 * Any other return code indicates some host side error.
1399 */
1400int GuestSession::i_fsQueryInfoInternal(const Utf8Str &strPath, bool fFollowSymlinks, GuestFsObjData &objData, int *pGuestRc)
1401{
1402 LogFlowThisFunc(("strPath=%s\n", strPath.c_str()));
1403
1404 int vrc = VINF_SUCCESS;
1405
1406 /** @todo Merge this with IGuestFile::queryInfo(). */
1407 GuestProcessStartupInfo procInfo;
1408 procInfo.mFlags = ProcessCreateFlag_WaitForStdOut;
1409 procInfo.mExecutable = Utf8Str(VBOXSERVICE_TOOL_STAT);
1410
1411 try
1412 {
1413 procInfo.mArguments.push_back(procInfo.mExecutable); /* Set argv0. */
1414 procInfo.mArguments.push_back(Utf8Str("--machinereadable"));
1415 if (fFollowSymlinks)
1416 procInfo.mArguments.push_back(Utf8Str("-L"));
1417 procInfo.mArguments.push_back("--"); /* strPath could be '--help', which is a valid filename. */
1418 procInfo.mArguments.push_back(strPath);
1419 }
1420 catch (std::bad_alloc)
1421 {
1422 vrc = VERR_NO_MEMORY;
1423 }
1424
1425 int guestRc;
1426 GuestCtrlStreamObjects stdOut;
1427 if (RT_SUCCESS(vrc))
1428 vrc = GuestProcessTool::i_runEx(this, procInfo,
1429 &stdOut, 1 /* cStrmOutObjects */,
1430 &guestRc);
1431
1432 if (!GuestProcess::i_isGuestError(vrc))
1433 {
1434 if (!stdOut.empty())
1435 {
1436 vrc = objData.FromStat(stdOut.at(0));
1437 if (RT_FAILURE(vrc))
1438 {
1439 guestRc = vrc;
1440 vrc = VERR_GSTCTL_GUEST_ERROR;
1441 }
1442 }
1443 else
1444 vrc = VERR_BROKEN_PIPE;
1445 }
1446 else if (pGuestRc)
1447 {
1448 *pGuestRc = guestRc;
1449 }
1450
1451 LogFlowThisFunc(("Returning rc=%Rrc, guestRc=%Rrc\n", vrc, guestRc));
1452 return vrc;
1453}
1454
1455const GuestCredentials& GuestSession::i_getCredentials(void)
1456{
1457 return mData.mCredentials;
1458}
1459
1460Utf8Str GuestSession::i_getName(void)
1461{
1462 return mData.mSession.mName;
1463}
1464
1465/* static */
1466Utf8Str GuestSession::i_guestErrorToString(int guestRc)
1467{
1468 Utf8Str strError;
1469
1470 /** @todo pData->u32Flags: int vs. uint32 -- IPRT errors are *negative* !!! */
1471 switch (guestRc)
1472 {
1473 case VERR_INVALID_VM_HANDLE:
1474 strError += Utf8StrFmt(tr("VMM device is not available (is the VM running?)"));
1475 break;
1476
1477 case VERR_HGCM_SERVICE_NOT_FOUND:
1478 strError += Utf8StrFmt(tr("The guest execution service is not available"));
1479 break;
1480
1481 case VERR_ACCOUNT_RESTRICTED:
1482 strError += Utf8StrFmt(tr("The specified user account on the guest is restricted and can't be used to logon"));
1483 break;
1484
1485 case VERR_AUTHENTICATION_FAILURE:
1486 strError += Utf8StrFmt(tr("The specified user was not able to logon on guest"));
1487 break;
1488
1489 case VERR_TIMEOUT:
1490 strError += Utf8StrFmt(tr("The guest did not respond within time"));
1491 break;
1492
1493 case VERR_CANCELLED:
1494 strError += Utf8StrFmt(tr("The session operation was canceled"));
1495 break;
1496
1497 case VERR_PERMISSION_DENIED: /** @todo r=bird: This is probably completely and utterly misleading. VERR_AUTHENTICATION_FAILURE could have this message. */
1498 strError += Utf8StrFmt(tr("Invalid user/password credentials"));
1499 break;
1500
1501 case VERR_MAX_PROCS_REACHED:
1502 strError += Utf8StrFmt(tr("Maximum number of concurrent guest processes has been reached"));
1503 break;
1504
1505 case VERR_NOT_FOUND:
1506 strError += Utf8StrFmt(tr("The guest execution service is not ready (yet)"));
1507 break;
1508
1509 default:
1510 strError += Utf8StrFmt("%Rrc", guestRc);
1511 break;
1512 }
1513
1514 return strError;
1515}
1516
1517/**
1518 * Checks if this session is ready state where it can handle
1519 * all session-bound actions (like guest processes, guest files).
1520 * Only used by official API methods. Will set an external
1521 * error when not ready.
1522 */
1523HRESULT GuestSession::i_isReadyExternal(void)
1524{
1525 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1526
1527 /** @todo Be a bit more informative. */
1528 if (mData.mStatus != GuestSessionStatus_Started)
1529 return setError(E_UNEXPECTED, tr("Session is not in started state"));
1530
1531 return S_OK;
1532}
1533
1534/**
1535 * Called by IGuest right before this session gets removed from
1536 * the public session list.
1537 */
1538int GuestSession::i_onRemove(void)
1539{
1540 LogFlowThisFuncEnter();
1541
1542 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1543
1544 int vrc = VINF_SUCCESS;
1545
1546 /*
1547 * Note: The event source stuff holds references to this object,
1548 * so make sure that this is cleaned up *before* calling uninit.
1549 */
1550 if (!mEventSource.isNull())
1551 {
1552 mEventSource->UnregisterListener(mLocalListener);
1553
1554 mLocalListener.setNull();
1555 unconst(mEventSource).setNull();
1556 }
1557
1558 LogFlowFuncLeaveRC(vrc);
1559 return vrc;
1560}
1561
1562/** No locking! */
1563int GuestSession::i_onSessionStatusChange(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCbData)
1564{
1565 AssertPtrReturn(pCbCtx, VERR_INVALID_POINTER);
1566 /* pCallback is optional. */
1567 AssertPtrReturn(pSvcCbData, VERR_INVALID_POINTER);
1568
1569 if (pSvcCbData->mParms < 3)
1570 return VERR_INVALID_PARAMETER;
1571
1572 CALLBACKDATA_SESSION_NOTIFY dataCb;
1573 /* pSvcCb->mpaParms[0] always contains the context ID. */
1574 int vrc = pSvcCbData->mpaParms[1].getUInt32(&dataCb.uType);
1575 AssertRCReturn(vrc, vrc);
1576 vrc = pSvcCbData->mpaParms[2].getUInt32(&dataCb.uResult);
1577 AssertRCReturn(vrc, vrc);
1578
1579 LogFlowThisFunc(("ID=%RU32, uType=%RU32, guestRc=%Rrc\n",
1580 mData.mSession.mID, dataCb.uType, dataCb.uResult));
1581
1582 GuestSessionStatus_T sessionStatus = GuestSessionStatus_Undefined;
1583
1584 int guestRc = dataCb.uResult; /** @todo uint32_t vs. int. */
1585 switch (dataCb.uType)
1586 {
1587 case GUEST_SESSION_NOTIFYTYPE_ERROR:
1588 sessionStatus = GuestSessionStatus_Error;
1589 break;
1590
1591 case GUEST_SESSION_NOTIFYTYPE_STARTED:
1592 sessionStatus = GuestSessionStatus_Started;
1593#if 0 /** @todo If we get some environment stuff along with this kind notification: */
1594 const char *pszzEnvBlock = ...;
1595 uint32_t cbEnvBlock = ...;
1596 if (!mData.mpBaseEnvironment)
1597 {
1598 GuestEnvironment *pBaseEnv;
1599 try { pBaseEnv = new GuestEnvironment(); } catch (std::bad_alloc &) { pBaseEnv = NULL; }
1600 if (pBaseEnv)
1601 {
1602 int vrc = pBaseEnv->initNormal();
1603 if (RT_SUCCESS(vrc))
1604 vrc = pBaseEnv->copyUtf8Block(pszzEnvBlock, cbEnvBlock);
1605 if (RT_SUCCESS(vrc))
1606 mData.mpBaseEnvironment = pBaseEnv;
1607 else
1608 pBaseEnv->release();
1609 }
1610 }
1611#endif
1612 break;
1613
1614 case GUEST_SESSION_NOTIFYTYPE_TEN:
1615 case GUEST_SESSION_NOTIFYTYPE_TES:
1616 case GUEST_SESSION_NOTIFYTYPE_TEA:
1617 sessionStatus = GuestSessionStatus_Terminated;
1618 break;
1619
1620 case GUEST_SESSION_NOTIFYTYPE_TOK:
1621 sessionStatus = GuestSessionStatus_TimedOutKilled;
1622 break;
1623
1624 case GUEST_SESSION_NOTIFYTYPE_TOA:
1625 sessionStatus = GuestSessionStatus_TimedOutAbnormally;
1626 break;
1627
1628 case GUEST_SESSION_NOTIFYTYPE_DWN:
1629 sessionStatus = GuestSessionStatus_Down;
1630 break;
1631
1632 case GUEST_SESSION_NOTIFYTYPE_UNDEFINED:
1633 default:
1634 vrc = VERR_NOT_SUPPORTED;
1635 break;
1636 }
1637
1638 if (RT_SUCCESS(vrc))
1639 {
1640 if (RT_FAILURE(guestRc))
1641 sessionStatus = GuestSessionStatus_Error;
1642 }
1643
1644 /* Set the session status. */
1645 if (RT_SUCCESS(vrc))
1646 vrc = i_setSessionStatus(sessionStatus, guestRc);
1647
1648 LogFlowThisFunc(("ID=%RU32, guestRc=%Rrc\n", mData.mSession.mID, guestRc));
1649
1650 LogFlowFuncLeaveRC(vrc);
1651 return vrc;
1652}
1653
1654int GuestSession::i_startSessionInternal(int *pGuestRc)
1655{
1656 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1657
1658 LogFlowThisFunc(("mID=%RU32, mName=%s, uProtocolVersion=%RU32, openFlags=%x, openTimeoutMS=%RU32\n",
1659 mData.mSession.mID, mData.mSession.mName.c_str(), mData.mProtocolVersion,
1660 mData.mSession.mOpenFlags, mData.mSession.mOpenTimeoutMS));
1661
1662 /* Guest Additions < 4.3 don't support opening dedicated
1663 guest sessions. Simply return success here. */
1664 if (mData.mProtocolVersion < 2)
1665 {
1666 mData.mStatus = GuestSessionStatus_Started;
1667
1668 LogFlowThisFunc(("Installed Guest Additions don't support opening dedicated sessions, skipping\n"));
1669 return VINF_SUCCESS;
1670 }
1671
1672 if (mData.mStatus != GuestSessionStatus_Undefined)
1673 return VINF_SUCCESS;
1674
1675 /** @todo mData.mSession.uFlags validation. */
1676
1677 /* Set current session status. */
1678 mData.mStatus = GuestSessionStatus_Starting;
1679 mData.mRC = VINF_SUCCESS; /* Clear previous error, if any. */
1680
1681 int vrc;
1682
1683 GuestWaitEvent *pEvent = NULL;
1684 GuestEventTypes eventTypes;
1685 try
1686 {
1687 eventTypes.push_back(VBoxEventType_OnGuestSessionStateChanged);
1688
1689 vrc = registerWaitEvent(mData.mSession.mID, 0 /* Object ID */,
1690 eventTypes, &pEvent);
1691 }
1692 catch (std::bad_alloc)
1693 {
1694 vrc = VERR_NO_MEMORY;
1695 }
1696
1697 if (RT_FAILURE(vrc))
1698 return vrc;
1699
1700 VBOXHGCMSVCPARM paParms[8];
1701
1702 int i = 0;
1703 paParms[i++].setUInt32(pEvent->ContextID());
1704 paParms[i++].setUInt32(mData.mProtocolVersion);
1705 paParms[i++].setPointer((void*)mData.mCredentials.mUser.c_str(),
1706 (ULONG)mData.mCredentials.mUser.length() + 1);
1707 paParms[i++].setPointer((void*)mData.mCredentials.mPassword.c_str(),
1708 (ULONG)mData.mCredentials.mPassword.length() + 1);
1709 paParms[i++].setPointer((void*)mData.mCredentials.mDomain.c_str(),
1710 (ULONG)mData.mCredentials.mDomain.length() + 1);
1711 paParms[i++].setUInt32(mData.mSession.mOpenFlags);
1712
1713 alock.release(); /* Drop write lock before sending. */
1714
1715 vrc = i_sendCommand(HOST_SESSION_CREATE, i, paParms);
1716 if (RT_SUCCESS(vrc))
1717 {
1718 vrc = i_waitForStatusChange(pEvent, GuestSessionWaitForFlag_Start,
1719 30 * 1000 /* 30s timeout */,
1720 NULL /* Session status */, pGuestRc);
1721 }
1722 else
1723 {
1724 /*
1725 * Unable to start guest session - update its current state.
1726 * Since there is no (official API) way to recover a failed guest session
1727 * this also marks the end state. Internally just calling this
1728 * same function again will work though.
1729 */
1730 mData.mStatus = GuestSessionStatus_Error;
1731 mData.mRC = vrc;
1732 }
1733
1734 unregisterWaitEvent(pEvent);
1735
1736 LogFlowFuncLeaveRC(vrc);
1737 return vrc;
1738}
1739
1740int GuestSession::i_startSessionAsync(void)
1741{
1742 LogFlowThisFuncEnter();
1743
1744 int vrc = VINF_SUCCESS;
1745
1746 GuestSessionTaskInternalOpen* pTask = NULL;
1747 try
1748 {
1749 pTask = new GuestSessionTaskInternalOpen(this);
1750 if (!pTask->isOk())
1751 {
1752 delete pTask;
1753 LogFlow(("GuestSession: Could not create GuestSessionTaskInternalOpen object \n"));
1754 throw VERR_MEMOBJ_INIT_FAILED;
1755 }
1756
1757 /* Asynchronously open the session on the guest by kicking off a
1758 * worker thread. */
1759 //this function delete pTask in case of exceptions, so there is no need in the call of delete operator
1760 HRESULT hrc = pTask->createThread();
1761 }
1762 catch(std::bad_alloc &)
1763 {
1764 vrc = VERR_NO_MEMORY;
1765 }
1766 catch(int eVRC)
1767 {
1768 vrc = eVRC;
1769 LogFlow(("GuestSession: Could not create thread for GuestSessionTaskInternalOpen task %Rrc\n", vrc));
1770 }
1771
1772 LogFlowFuncLeaveRC(vrc);
1773 return vrc;
1774}
1775
1776/* static */
1777DECLCALLBACK(int) GuestSession::i_startSessionThread(RTTHREAD Thread, void *pvUser)
1778{
1779 LogFlowFunc(("pvUser=%p\n", pvUser));
1780
1781
1782 GuestSessionTaskInternalOpen* pTask = static_cast<GuestSessionTaskInternalOpen*>(pvUser);
1783 AssertPtr(pTask);
1784
1785 const ComObjPtr<GuestSession> pSession(pTask->Session());
1786 Assert(!pSession.isNull());
1787
1788 AutoCaller autoCaller(pSession);
1789 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1790
1791 int vrc = pSession->i_startSessionInternal(NULL /* Guest rc, ignored */);
1792 /* Nothing to do here anymore. */
1793
1794 LogFlowFuncLeaveRC(vrc);
1795 return vrc;
1796}
1797
1798int GuestSession::i_pathRenameInternal(const Utf8Str &strSource, const Utf8Str &strDest,
1799 uint32_t uFlags, int *pGuestRc)
1800{
1801 AssertReturn(!(uFlags & ~PATHRENAME_FLAG_VALID_MASK), VERR_INVALID_PARAMETER);
1802
1803 LogFlowThisFunc(("strSource=%s, strDest=%s, uFlags=0x%x\n",
1804 strSource.c_str(), strDest.c_str(), uFlags));
1805
1806 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1807
1808 GuestWaitEvent *pEvent = NULL;
1809 int vrc = registerWaitEvent(mData.mSession.mID, 0 /* Object ID */,
1810 &pEvent);
1811 if (RT_FAILURE(vrc))
1812 return vrc;
1813
1814 /* Prepare HGCM call. */
1815 VBOXHGCMSVCPARM paParms[8];
1816 int i = 0;
1817 paParms[i++].setUInt32(pEvent->ContextID());
1818 paParms[i++].setPointer((void*)strSource.c_str(),
1819 (ULONG)strSource.length() + 1);
1820 paParms[i++].setPointer((void*)strDest.c_str(),
1821 (ULONG)strDest.length() + 1);
1822 paParms[i++].setUInt32(uFlags);
1823
1824 alock.release(); /* Drop write lock before sending. */
1825
1826 vrc = i_sendCommand(HOST_PATH_RENAME, i, paParms);
1827 if (RT_SUCCESS(vrc))
1828 {
1829 vrc = pEvent->Wait(30 * 1000);
1830 if ( vrc == VERR_GSTCTL_GUEST_ERROR
1831 && pGuestRc)
1832 *pGuestRc = pEvent->GuestResult();
1833 }
1834
1835 unregisterWaitEvent(pEvent);
1836
1837 LogFlowFuncLeaveRC(vrc);
1838 return vrc;
1839}
1840
1841int GuestSession::i_processRemoveFromList(GuestProcess *pProcess)
1842{
1843 AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
1844
1845 LogFlowThisFunc(("pProcess=%p\n", pProcess));
1846
1847 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1848
1849 int rc = VERR_NOT_FOUND;
1850
1851 ULONG uPID;
1852 HRESULT hr = pProcess->COMGETTER(PID)(&uPID);
1853 ComAssertComRC(hr);
1854
1855 LogFlowFunc(("Removing process (PID=%RU32) ...\n", uPID));
1856
1857 SessionProcesses::iterator itProcs = mData.mProcesses.begin();
1858 while (itProcs != mData.mProcesses.end())
1859 {
1860 if (pProcess == itProcs->second)
1861 {
1862#ifdef DEBUG_andy
1863 ULONG cRefs = pProcess->AddRef();
1864 Assert(cRefs >= 2);
1865 LogFlowFunc(("pProcess=%p, cRefs=%RU32\n", pProcess, cRefs - 1));
1866 pProcess->Release();
1867#endif
1868 /* Make sure to consume the pointer before the one of the
1869 * iterator gets released. */
1870 ComObjPtr<GuestProcess> pProc = pProcess;
1871
1872 hr = pProc->COMGETTER(PID)(&uPID);
1873 ComAssertComRC(hr);
1874
1875 Assert(mData.mProcesses.size());
1876 Assert(mData.mNumObjects);
1877 LogFlowFunc(("Removing process ID=%RU32 (Session: %RU32), guest PID=%RU32 (now total %zu processes, %RU32 objects)\n",
1878 pProcess->getObjectID(), mData.mSession.mID, uPID, mData.mProcesses.size() - 1, mData.mNumObjects - 1));
1879
1880 rc = pProcess->i_onRemove();
1881 mData.mProcesses.erase(itProcs);
1882 mData.mNumObjects--;
1883
1884 alock.release(); /* Release lock before firing off event. */
1885
1886 fireGuestProcessRegisteredEvent(mEventSource, this /* Session */, pProc,
1887 uPID, false /* Process unregistered */);
1888 pProc.setNull();
1889 break;
1890 }
1891
1892 ++itProcs;
1893 }
1894
1895 LogFlowFuncLeaveRC(rc);
1896 return rc;
1897}
1898
1899/**
1900 * Creates but does *not* start the process yet.
1901 *
1902 * See GuestProcess::startProcess() or GuestProcess::startProcessAsync() for
1903 * starting the process.
1904 *
1905 * @return IPRT status code.
1906 * @param procInfo
1907 * @param pProcess
1908 */
1909int GuestSession::i_processCreateExInternal(GuestProcessStartupInfo &procInfo, ComObjPtr<GuestProcess> &pProcess)
1910{
1911 LogFlowFunc(("mExe=%s, mFlags=%x, mTimeoutMS=%RU32\n",
1912 procInfo.mExecutable.c_str(), procInfo.mFlags, procInfo.mTimeoutMS));
1913#ifdef DEBUG
1914 if (procInfo.mArguments.size())
1915 {
1916 LogFlowFunc(("Arguments:"));
1917 ProcessArguments::const_iterator it = procInfo.mArguments.begin();
1918 while (it != procInfo.mArguments.end())
1919 {
1920 LogFlow((" %s", (*it).c_str()));
1921 ++it;
1922 }
1923 LogFlow(("\n"));
1924 }
1925#endif
1926
1927 /* Validate flags. */
1928 if (procInfo.mFlags)
1929 {
1930 if ( !(procInfo.mFlags & ProcessCreateFlag_IgnoreOrphanedProcesses)
1931 && !(procInfo.mFlags & ProcessCreateFlag_WaitForProcessStartOnly)
1932 && !(procInfo.mFlags & ProcessCreateFlag_Hidden)
1933 && !(procInfo.mFlags & ProcessCreateFlag_Profile)
1934 && !(procInfo.mFlags & ProcessCreateFlag_WaitForStdOut)
1935 && !(procInfo.mFlags & ProcessCreateFlag_WaitForStdErr))
1936 {
1937 return VERR_INVALID_PARAMETER;
1938 }
1939 }
1940
1941 if ( (procInfo.mFlags & ProcessCreateFlag_WaitForProcessStartOnly)
1942 && ( (procInfo.mFlags & ProcessCreateFlag_WaitForStdOut)
1943 || (procInfo.mFlags & ProcessCreateFlag_WaitForStdErr)
1944 )
1945 )
1946 {
1947 return VERR_INVALID_PARAMETER;
1948 }
1949
1950 /* Adjust timeout.
1951 * If set to 0, we define an infinite timeout (unlimited process run time). */
1952 if (procInfo.mTimeoutMS == 0)
1953 procInfo.mTimeoutMS = UINT32_MAX;
1954
1955 /** @tood Implement process priority + affinity. */
1956
1957 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1958
1959 int rc = VERR_MAX_PROCS_REACHED;
1960 if (mData.mNumObjects >= VBOX_GUESTCTRL_MAX_OBJECTS)
1961 return rc;
1962
1963 /* Create a new (host-based) process ID and assign it. */
1964 uint32_t uNewProcessID = 0;
1965 ULONG uTries = 0;
1966
1967 for (;;)
1968 {
1969 /* Is the context ID already used? */
1970 if (!i_processExists(uNewProcessID, NULL /* pProcess */))
1971 {
1972 /* Callback with context ID was not found. This means
1973 * we can use this context ID for our new callback we want
1974 * to add below. */
1975 rc = VINF_SUCCESS;
1976 break;
1977 }
1978 uNewProcessID++;
1979 if (uNewProcessID == VBOX_GUESTCTRL_MAX_OBJECTS)
1980 uNewProcessID = 0;
1981
1982 if (++uTries == VBOX_GUESTCTRL_MAX_OBJECTS)
1983 break; /* Don't try too hard. */
1984 }
1985
1986 if (RT_FAILURE(rc))
1987 return rc;
1988
1989 /* Create the process object. */
1990 HRESULT hr = pProcess.createObject();
1991 if (FAILED(hr))
1992 return VERR_COM_UNEXPECTED;
1993
1994 rc = pProcess->init(mParent->i_getConsole() /* Console */, this /* Session */,
1995 uNewProcessID, procInfo, mData.mpBaseEnvironment);
1996 if (RT_FAILURE(rc))
1997 return rc;
1998
1999 /* Add the created process to our map. */
2000 try
2001 {
2002 mData.mProcesses[uNewProcessID] = pProcess;
2003 mData.mNumObjects++;
2004 Assert(mData.mNumObjects <= VBOX_GUESTCTRL_MAX_OBJECTS);
2005
2006 LogFlowFunc(("Added new process (Session: %RU32) with process ID=%RU32 (now total %zu processes, %RU32 objects)\n",
2007 mData.mSession.mID, uNewProcessID, mData.mProcesses.size(), mData.mNumObjects));
2008
2009 alock.release(); /* Release lock before firing off event. */
2010
2011 fireGuestProcessRegisteredEvent(mEventSource, this /* Session */, pProcess,
2012 0 /* PID */, true /* Process registered */);
2013 }
2014 catch (std::bad_alloc &)
2015 {
2016 rc = VERR_NO_MEMORY;
2017 }
2018
2019 return rc;
2020}
2021
2022inline bool GuestSession::i_processExists(uint32_t uProcessID, ComObjPtr<GuestProcess> *pProcess)
2023{
2024 SessionProcesses::const_iterator it = mData.mProcesses.find(uProcessID);
2025 if (it != mData.mProcesses.end())
2026 {
2027 if (pProcess)
2028 *pProcess = it->second;
2029 return true;
2030 }
2031 return false;
2032}
2033
2034inline int GuestSession::i_processGetByPID(ULONG uPID, ComObjPtr<GuestProcess> *pProcess)
2035{
2036 AssertReturn(uPID, false);
2037 /* pProcess is optional. */
2038
2039 SessionProcesses::iterator itProcs = mData.mProcesses.begin();
2040 for (; itProcs != mData.mProcesses.end(); ++itProcs)
2041 {
2042 ComObjPtr<GuestProcess> pCurProc = itProcs->second;
2043 AutoCaller procCaller(pCurProc);
2044 if (procCaller.rc())
2045 return VERR_COM_INVALID_OBJECT_STATE;
2046
2047 ULONG uCurPID;
2048 HRESULT hr = pCurProc->COMGETTER(PID)(&uCurPID);
2049 ComAssertComRC(hr);
2050
2051 if (uCurPID == uPID)
2052 {
2053 if (pProcess)
2054 *pProcess = pCurProc;
2055 return VINF_SUCCESS;
2056 }
2057 }
2058
2059 return VERR_NOT_FOUND;
2060}
2061
2062int GuestSession::i_sendCommand(uint32_t uFunction,
2063 uint32_t uParms, PVBOXHGCMSVCPARM paParms)
2064{
2065 LogFlowThisFuncEnter();
2066
2067#ifndef VBOX_GUESTCTRL_TEST_CASE
2068 ComObjPtr<Console> pConsole = mParent->i_getConsole();
2069 Assert(!pConsole.isNull());
2070
2071 /* Forward the information to the VMM device. */
2072 VMMDev *pVMMDev = pConsole->i_getVMMDev();
2073 AssertPtr(pVMMDev);
2074
2075 LogFlowThisFunc(("uFunction=%RU32, uParms=%RU32\n", uFunction, uParms));
2076 int vrc = pVMMDev->hgcmHostCall(HGCMSERVICE_NAME, uFunction, uParms, paParms);
2077 if (RT_FAILURE(vrc))
2078 {
2079 /** @todo What to do here? */
2080 }
2081#else
2082 /* Not needed within testcases. */
2083 int vrc = VINF_SUCCESS;
2084#endif
2085 LogFlowFuncLeaveRC(vrc);
2086 return vrc;
2087}
2088
2089/* static */
2090HRESULT GuestSession::i_setErrorExternal(VirtualBoxBase *pInterface, int guestRc)
2091{
2092 AssertPtr(pInterface);
2093 AssertMsg(RT_FAILURE(guestRc), ("Guest rc does not indicate a failure when setting error\n"));
2094
2095 return pInterface->setError(VBOX_E_IPRT_ERROR, GuestSession::i_guestErrorToString(guestRc).c_str());
2096}
2097
2098/* Does not do locking; caller is responsible for that! */
2099int GuestSession::i_setSessionStatus(GuestSessionStatus_T sessionStatus, int sessionRc)
2100{
2101 LogFlowThisFunc(("oldStatus=%RU32, newStatus=%RU32, sessionRc=%Rrc\n",
2102 mData.mStatus, sessionStatus, sessionRc));
2103
2104 if (sessionStatus == GuestSessionStatus_Error)
2105 {
2106 AssertMsg(RT_FAILURE(sessionRc), ("Guest rc must be an error (%Rrc)\n", sessionRc));
2107 /* Do not allow overwriting an already set error. If this happens
2108 * this means we forgot some error checking/locking somewhere. */
2109 AssertMsg(RT_SUCCESS(mData.mRC), ("Guest rc already set (to %Rrc)\n", mData.mRC));
2110 }
2111 else
2112 AssertMsg(RT_SUCCESS(sessionRc), ("Guest rc must not be an error (%Rrc)\n", sessionRc));
2113
2114 if (mData.mStatus != sessionStatus)
2115 {
2116 mData.mStatus = sessionStatus;
2117 mData.mRC = sessionRc;
2118
2119 ComObjPtr<VirtualBoxErrorInfo> errorInfo;
2120 HRESULT hr = errorInfo.createObject();
2121 ComAssertComRC(hr);
2122 int rc2 = errorInfo->initEx(VBOX_E_IPRT_ERROR, sessionRc,
2123 COM_IIDOF(IGuestSession), getComponentName(),
2124 i_guestErrorToString(sessionRc));
2125 AssertRC(rc2);
2126
2127 fireGuestSessionStateChangedEvent(mEventSource, this,
2128 mData.mSession.mID, sessionStatus, errorInfo);
2129 }
2130
2131 return VINF_SUCCESS;
2132}
2133
2134int GuestSession::i_signalWaiters(GuestSessionWaitResult_T enmWaitResult, int rc /*= VINF_SUCCESS */)
2135{
2136 /*LogFlowThisFunc(("enmWaitResult=%d, rc=%Rrc, mWaitCount=%RU32, mWaitEvent=%p\n",
2137 enmWaitResult, rc, mData.mWaitCount, mData.mWaitEvent));*/
2138
2139 /* Note: No write locking here -- already done in the caller. */
2140
2141 int vrc = VINF_SUCCESS;
2142 /*if (mData.mWaitEvent)
2143 vrc = mData.mWaitEvent->Signal(enmWaitResult, rc);*/
2144 LogFlowFuncLeaveRC(vrc);
2145 return vrc;
2146}
2147
2148int GuestSession::i_startTaskAsync(const Utf8Str &strTaskDesc,
2149 GuestSessionTask *pTask, ComObjPtr<Progress> &pProgress)
2150{
2151 LogFlowThisFunc(("strTaskDesc=%s, pTask=%p\n", strTaskDesc.c_str(), pTask));
2152
2153 AssertPtrReturn(pTask, VERR_INVALID_POINTER);
2154
2155 /* Create the progress object. */
2156 HRESULT hr = pProgress.createObject();
2157 if (FAILED(hr))
2158 return VERR_COM_UNEXPECTED;
2159
2160 hr = pProgress->init(static_cast<IGuestSession*>(this),
2161 Bstr(strTaskDesc).raw(),
2162 TRUE /* aCancelable */);
2163 if (FAILED(hr))
2164 return VERR_COM_UNEXPECTED;
2165
2166 /* Initialize our worker task. */
2167 RT_GCC_NO_WARN_DEPRECATED_BEGIN
2168 std::auto_ptr<GuestSessionTask> task(pTask);
2169 RT_GCC_NO_WARN_DEPRECATED_END
2170 int rc = task->RunAsync(strTaskDesc, pProgress);
2171 if (RT_FAILURE(rc))
2172 return rc;
2173
2174 /* Don't destruct on success. */
2175 task.release();
2176
2177 LogFlowFuncLeaveRC(rc);
2178 return rc;
2179}
2180
2181/**
2182 * Determines the protocol version (sets mData.mProtocolVersion).
2183 *
2184 * This is called from the init method prior to to establishing a guest
2185 * session.
2186 *
2187 * @return IPRT status code.
2188 */
2189int GuestSession::i_determineProtocolVersion(void)
2190{
2191 /*
2192 * We currently do this based on the reported guest additions version,
2193 * ASSUMING that VBoxService and VBoxDrv are at the same version.
2194 */
2195 ComObjPtr<Guest> pGuest = mParent;
2196 AssertReturn(!pGuest.isNull(), VERR_NOT_SUPPORTED);
2197 uint32_t uGaVersion = pGuest->i_getAdditionsVersion();
2198
2199 /* Everyone supports version one, if they support anything at all. */
2200 mData.mProtocolVersion = 1;
2201
2202 /* Guest control 2.0 was introduced with 4.3.0. */
2203 if (uGaVersion >= VBOX_FULL_VERSION_MAKE(4,3,0))
2204 mData.mProtocolVersion = 2; /* Guest control 2.0. */
2205
2206 LogFlowThisFunc(("uGaVersion=%u.%u.%u => mProtocolVersion=%u\n",
2207 VBOX_FULL_VERSION_GET_MAJOR(uGaVersion), VBOX_FULL_VERSION_GET_MINOR(uGaVersion),
2208 VBOX_FULL_VERSION_GET_BUILD(uGaVersion), mData.mProtocolVersion));
2209
2210 /*
2211 * Inform the user about outdated guest additions (VM release log).
2212 */
2213 if (mData.mProtocolVersion < 2)
2214 LogRelMax(3, (tr("Warning: Guest Additions v%u.%u.%u only supports the older guest control protocol version %u.\n"
2215 " Please upgrade GAs to the current version to get full guest control capabilities.\n"),
2216 VBOX_FULL_VERSION_GET_MAJOR(uGaVersion), VBOX_FULL_VERSION_GET_MINOR(uGaVersion),
2217 VBOX_FULL_VERSION_GET_BUILD(uGaVersion), mData.mProtocolVersion));
2218
2219 return VINF_SUCCESS;
2220}
2221
2222int GuestSession::i_waitFor(uint32_t fWaitFlags, ULONG uTimeoutMS, GuestSessionWaitResult_T &waitResult, int *pGuestRc)
2223{
2224 LogFlowThisFuncEnter();
2225
2226 AssertReturn(fWaitFlags, VERR_INVALID_PARAMETER);
2227
2228 /*LogFlowThisFunc(("fWaitFlags=0x%x, uTimeoutMS=%RU32, mStatus=%RU32, mWaitCount=%RU32, mWaitEvent=%p, pGuestRc=%p\n",
2229 fWaitFlags, uTimeoutMS, mData.mStatus, mData.mWaitCount, mData.mWaitEvent, pGuestRc));*/
2230
2231 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2232
2233 /* Did some error occur before? Then skip waiting and return. */
2234 if (mData.mStatus == GuestSessionStatus_Error)
2235 {
2236 waitResult = GuestSessionWaitResult_Error;
2237 AssertMsg(RT_FAILURE(mData.mRC), ("No error rc (%Rrc) set when guest session indicated an error\n", mData.mRC));
2238 if (pGuestRc)
2239 *pGuestRc = mData.mRC; /* Return last set error. */
2240 return VERR_GSTCTL_GUEST_ERROR;
2241 }
2242
2243 /* Guest Additions < 4.3 don't support session handling, skip. */
2244 if (mData.mProtocolVersion < 2)
2245 {
2246 waitResult = GuestSessionWaitResult_WaitFlagNotSupported;
2247
2248 LogFlowThisFunc(("Installed Guest Additions don't support waiting for dedicated sessions, skipping\n"));
2249 return VINF_SUCCESS;
2250 }
2251
2252 waitResult = GuestSessionWaitResult_None;
2253 if (fWaitFlags & GuestSessionWaitForFlag_Terminate)
2254 {
2255 switch (mData.mStatus)
2256 {
2257 case GuestSessionStatus_Terminated:
2258 case GuestSessionStatus_Down:
2259 waitResult = GuestSessionWaitResult_Terminate;
2260 break;
2261
2262 case GuestSessionStatus_TimedOutKilled:
2263 case GuestSessionStatus_TimedOutAbnormally:
2264 waitResult = GuestSessionWaitResult_Timeout;
2265 break;
2266
2267 case GuestSessionStatus_Error:
2268 /* Handled above. */
2269 break;
2270
2271 case GuestSessionStatus_Started:
2272 waitResult = GuestSessionWaitResult_Start;
2273 break;
2274
2275 case GuestSessionStatus_Undefined:
2276 case GuestSessionStatus_Starting:
2277 /* Do the waiting below. */
2278 break;
2279
2280 default:
2281 AssertMsgFailed(("Unhandled session status %RU32\n", mData.mStatus));
2282 return VERR_NOT_IMPLEMENTED;
2283 }
2284 }
2285 else if (fWaitFlags & GuestSessionWaitForFlag_Start)
2286 {
2287 switch (mData.mStatus)
2288 {
2289 case GuestSessionStatus_Started:
2290 case GuestSessionStatus_Terminating:
2291 case GuestSessionStatus_Terminated:
2292 case GuestSessionStatus_Down:
2293 waitResult = GuestSessionWaitResult_Start;
2294 break;
2295
2296 case GuestSessionStatus_Error:
2297 waitResult = GuestSessionWaitResult_Error;
2298 break;
2299
2300 case GuestSessionStatus_TimedOutKilled:
2301 case GuestSessionStatus_TimedOutAbnormally:
2302 waitResult = GuestSessionWaitResult_Timeout;
2303 break;
2304
2305 case GuestSessionStatus_Undefined:
2306 case GuestSessionStatus_Starting:
2307 /* Do the waiting below. */
2308 break;
2309
2310 default:
2311 AssertMsgFailed(("Unhandled session status %RU32\n", mData.mStatus));
2312 return VERR_NOT_IMPLEMENTED;
2313 }
2314 }
2315
2316 LogFlowThisFunc(("sessionStatus=%RU32, sessionRc=%Rrc, waitResult=%RU32\n",
2317 mData.mStatus, mData.mRC, waitResult));
2318
2319 /* No waiting needed? Return immediately using the last set error. */
2320 if (waitResult != GuestSessionWaitResult_None)
2321 {
2322 if (pGuestRc)
2323 *pGuestRc = mData.mRC; /* Return last set error (if any). */
2324 return RT_SUCCESS(mData.mRC) ? VINF_SUCCESS : VERR_GSTCTL_GUEST_ERROR;
2325 }
2326
2327 int vrc;
2328
2329 GuestWaitEvent *pEvent = NULL;
2330 GuestEventTypes eventTypes;
2331 try
2332 {
2333 eventTypes.push_back(VBoxEventType_OnGuestSessionStateChanged);
2334
2335 vrc = registerWaitEvent(mData.mSession.mID, 0 /* Object ID */,
2336 eventTypes, &pEvent);
2337 }
2338 catch (std::bad_alloc)
2339 {
2340 vrc = VERR_NO_MEMORY;
2341 }
2342
2343 if (RT_FAILURE(vrc))
2344 return vrc;
2345
2346 alock.release(); /* Release lock before waiting. */
2347
2348 GuestSessionStatus_T sessionStatus;
2349 vrc = i_waitForStatusChange(pEvent, fWaitFlags,
2350 uTimeoutMS, &sessionStatus, pGuestRc);
2351 if (RT_SUCCESS(vrc))
2352 {
2353 switch (sessionStatus)
2354 {
2355 case GuestSessionStatus_Started:
2356 waitResult = GuestSessionWaitResult_Start;
2357 break;
2358
2359 case GuestSessionStatus_Terminated:
2360 waitResult = GuestSessionWaitResult_Terminate;
2361 break;
2362
2363 case GuestSessionStatus_TimedOutKilled:
2364 case GuestSessionStatus_TimedOutAbnormally:
2365 waitResult = GuestSessionWaitResult_Timeout;
2366 break;
2367
2368 case GuestSessionStatus_Down:
2369 waitResult = GuestSessionWaitResult_Terminate;
2370 break;
2371
2372 case GuestSessionStatus_Error:
2373 waitResult = GuestSessionWaitResult_Error;
2374 break;
2375
2376 default:
2377 waitResult = GuestSessionWaitResult_Status;
2378 break;
2379 }
2380 }
2381
2382 unregisterWaitEvent(pEvent);
2383
2384 LogFlowFuncLeaveRC(vrc);
2385 return vrc;
2386}
2387
2388int GuestSession::i_waitForStatusChange(GuestWaitEvent *pEvent, uint32_t fWaitFlags, uint32_t uTimeoutMS,
2389 GuestSessionStatus_T *pSessionStatus, int *pGuestRc)
2390{
2391 AssertPtrReturn(pEvent, VERR_INVALID_POINTER);
2392
2393 VBoxEventType_T evtType;
2394 ComPtr<IEvent> pIEvent;
2395 int vrc = waitForEvent(pEvent, uTimeoutMS,
2396 &evtType, pIEvent.asOutParam());
2397 if (RT_SUCCESS(vrc))
2398 {
2399 Assert(evtType == VBoxEventType_OnGuestSessionStateChanged);
2400
2401 ComPtr<IGuestSessionStateChangedEvent> pChangedEvent = pIEvent;
2402 Assert(!pChangedEvent.isNull());
2403
2404 GuestSessionStatus_T sessionStatus;
2405 pChangedEvent->COMGETTER(Status)(&sessionStatus);
2406 if (pSessionStatus)
2407 *pSessionStatus = sessionStatus;
2408
2409 ComPtr<IVirtualBoxErrorInfo> errorInfo;
2410 HRESULT hr = pChangedEvent->COMGETTER(Error)(errorInfo.asOutParam());
2411 ComAssertComRC(hr);
2412
2413 LONG lGuestRc;
2414 hr = errorInfo->COMGETTER(ResultDetail)(&lGuestRc);
2415 ComAssertComRC(hr);
2416 if (RT_FAILURE((int)lGuestRc))
2417 vrc = VERR_GSTCTL_GUEST_ERROR;
2418 if (pGuestRc)
2419 *pGuestRc = (int)lGuestRc;
2420
2421 LogFlowThisFunc(("Status changed event for session ID=%RU32, new status is: %RU32 (%Rrc)\n",
2422 mData.mSession.mID, sessionStatus,
2423 RT_SUCCESS((int)lGuestRc) ? VINF_SUCCESS : (int)lGuestRc));
2424 }
2425
2426 LogFlowFuncLeaveRC(vrc);
2427 return vrc;
2428}
2429
2430// implementation of public methods
2431/////////////////////////////////////////////////////////////////////////////
2432
2433HRESULT GuestSession::close()
2434{
2435 LogFlowThisFuncEnter();
2436
2437 AutoCaller autoCaller(this);
2438 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2439
2440 /* Close session on guest. */
2441 int guestRc = VINF_SUCCESS;
2442 int rc = i_closeSession(0 /* Flags */, 30 * 1000 /* Timeout */,
2443 &guestRc);
2444 /* On failure don't return here, instead do all the cleanup
2445 * work first and then return an error. */
2446
2447 /* Remove ourselves from the session list. */
2448 int rc2 = mParent->i_sessionRemove(this);
2449 if (rc2 == VERR_NOT_FOUND) /* Not finding the session anymore isn't critical. */
2450 rc2 = VINF_SUCCESS;
2451
2452 if (RT_SUCCESS(rc))
2453 rc = rc2;
2454
2455 LogFlowThisFunc(("Returning rc=%Rrc, guestRc=%Rrc\n",
2456 rc, guestRc));
2457 if (RT_FAILURE(rc))
2458 {
2459 if (rc == VERR_GSTCTL_GUEST_ERROR)
2460 return GuestSession::i_setErrorExternal(this, guestRc);
2461
2462 return setError(VBOX_E_IPRT_ERROR,
2463 tr("Closing guest session failed with %Rrc"), rc);
2464 }
2465
2466 return S_OK;
2467}
2468
2469HRESULT GuestSession::fileCopy(const com::Utf8Str &aSource, const com::Utf8Str &aDestination,
2470 const std::vector<FileCopyFlag_T> &aFlags, ComPtr<IProgress> &aProgress)
2471{
2472 ReturnComNotImplemented();
2473}
2474
2475HRESULT GuestSession::fileCopyFromGuest(const com::Utf8Str &aSource, const com::Utf8Str &aDest,
2476 const std::vector<FileCopyFlag_T> &aFlags,
2477 ComPtr<IProgress> &aProgress)
2478{
2479 LogFlowThisFuncEnter();
2480
2481 if (RT_UNLIKELY((aSource.c_str()) == NULL || *(aSource.c_str()) == '\0'))
2482 return setError(E_INVALIDARG, tr("No source specified"));
2483 if (RT_UNLIKELY((aDest.c_str()) == NULL || *(aDest.c_str()) == '\0'))
2484 return setError(E_INVALIDARG, tr("No destination specified"));
2485
2486 uint32_t fFlags = FileCopyFlag_None;
2487 if (aFlags.size())
2488 {
2489 for (size_t i = 0; i < aFlags.size(); i++)
2490 fFlags |= aFlags[i];
2491 }
2492/** @todo r=bird: fend off flags we don't implement here! */
2493
2494 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2495
2496 HRESULT hr = S_OK;
2497
2498 try
2499 {
2500 SessionTaskCopyFrom *pTask = NULL;
2501 ComObjPtr<Progress> pProgress;
2502 try
2503 {
2504 pTask = new SessionTaskCopyFrom(this /* GuestSession */, aSource, aDest, fFlags);
2505 }
2506 catch(...)
2507 {
2508 hr = setError(VBOX_E_IPRT_ERROR, tr("Failed to create SessionTaskCopyFrom object "));
2509 throw;
2510 }
2511
2512
2513 hr = pTask->Init(Utf8StrFmt(tr("Copying \"%s\" from guest to \"%s\" on the host"), aSource.c_str(), aDest.c_str()));
2514 if (FAILED(hr))
2515 {
2516 delete pTask;
2517 hr = setError(VBOX_E_IPRT_ERROR,
2518 tr("Creating progress object for SessionTaskCopyFrom object failed"));
2519 throw hr;
2520 }
2521
2522 hr = pTask->createThread(NULL, RTTHREADTYPE_MAIN_HEAVY_WORKER);
2523
2524 if (SUCCEEDED(hr))
2525 {
2526 /* Return progress to the caller. */
2527 pProgress = pTask->GetProgressObject();
2528 hr = pProgress.queryInterfaceTo(aProgress.asOutParam());
2529 }
2530 else
2531 hr = setError(VBOX_E_IPRT_ERROR,
2532 tr("Starting thread for copying file \"%s\" from guest to \"%s\" on the host failed "),
2533 aSource.c_str(), aDest.c_str());
2534
2535 }
2536 catch(std::bad_alloc &)
2537 {
2538 hr = E_OUTOFMEMORY;
2539 }
2540 catch(HRESULT eHR)
2541 {
2542 hr = eHR;
2543 LogFlowThisFunc(("Exception was caught in the function \n"));
2544 }
2545
2546 return hr;
2547}
2548
2549HRESULT GuestSession::fileCopyToGuest(const com::Utf8Str &aSource, const com::Utf8Str &aDest,
2550 const std::vector<FileCopyFlag_T> &aFlags, ComPtr<IProgress> &aProgress)
2551{
2552 LogFlowThisFuncEnter();
2553
2554 if (RT_UNLIKELY((aSource.c_str()) == NULL || *(aSource.c_str()) == '\0'))
2555 return setError(E_INVALIDARG, tr("No source specified"));
2556 if (RT_UNLIKELY((aDest.c_str()) == NULL || *(aDest.c_str()) == '\0'))
2557 return setError(E_INVALIDARG, tr("No destination specified"));
2558
2559 uint32_t fFlags = FileCopyFlag_None;
2560 if (aFlags.size())
2561 {
2562 for (size_t i = 0; i < aFlags.size(); i++)
2563 fFlags |= aFlags[i];
2564 }
2565/** @todo r=bird: fend off flags we don't implement here! */
2566
2567 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2568
2569 HRESULT hr = S_OK;
2570
2571 try
2572 {
2573 SessionTaskCopyTo *pTask = NULL;
2574 ComObjPtr<Progress> pProgress;
2575 try
2576 {
2577 pTask = new SessionTaskCopyTo(this /* GuestSession */, aSource, aDest, fFlags);
2578 }
2579 catch(...)
2580 {
2581 hr = setError(VBOX_E_IPRT_ERROR, tr("Failed to create SessionTaskCopyTo object "));
2582 throw;
2583 }
2584
2585
2586 hr = pTask->Init(Utf8StrFmt(tr("Copying \"%s\" from host to \"%s\" on the guest"), aSource.c_str(), aDest.c_str()));
2587 if (FAILED(hr))
2588 {
2589 delete pTask;
2590 hr = setError(VBOX_E_IPRT_ERROR,
2591 tr("Creating progress object for SessionTaskCopyTo object failed"));
2592 throw hr;
2593 }
2594
2595 hr = pTask->createThread(NULL, RTTHREADTYPE_MAIN_HEAVY_WORKER);
2596
2597 if (SUCCEEDED(hr))
2598 {
2599 /* Return progress to the caller. */
2600 pProgress = pTask->GetProgressObject();
2601 hr = pProgress.queryInterfaceTo(aProgress.asOutParam());
2602 }
2603 else
2604 hr = setError(VBOX_E_IPRT_ERROR,
2605 tr("Starting thread for copying file \"%s\" from host to \"%s\" on the guest failed "),
2606 aSource.c_str(), aDest.c_str());
2607 }
2608 catch(std::bad_alloc &)
2609 {
2610 hr = E_OUTOFMEMORY;
2611 }
2612 catch(HRESULT eHR)
2613 {
2614 hr = eHR;
2615 LogFlowThisFunc(("Exception was caught in the function \n"));
2616 }
2617
2618 return hr;
2619}
2620
2621HRESULT GuestSession::directoryCopy(const com::Utf8Str &aSource, const com::Utf8Str &aDestination,
2622 const std::vector<DirectoryCopyFlags_T> &aFlags, ComPtr<IProgress> &aProgress)
2623{
2624 ReturnComNotImplemented();
2625}
2626
2627HRESULT GuestSession::directoryCopyFromGuest(const com::Utf8Str &aSource, const com::Utf8Str &aDestination,
2628 const std::vector<DirectoryCopyFlags_T> &aFlags, ComPtr<IProgress> &aProgress)
2629{
2630 ReturnComNotImplemented();
2631}
2632
2633HRESULT GuestSession::directoryCopyToGuest(const com::Utf8Str &aSource, const com::Utf8Str &aDestination,
2634 const std::vector<DirectoryCopyFlags_T> &aFlags, ComPtr<IProgress> &aProgress)
2635{
2636 ReturnComNotImplemented();
2637}
2638
2639HRESULT GuestSession::directoryCreate(const com::Utf8Str &aPath, ULONG aMode,
2640 const std::vector<DirectoryCreateFlag_T> &aFlags)
2641{
2642 LogFlowThisFuncEnter();
2643
2644 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
2645 return setError(E_INVALIDARG, tr("No directory to create specified"));
2646
2647 uint32_t fFlags = DirectoryCreateFlag_None;
2648 if (aFlags.size())
2649 {
2650 for (size_t i = 0; i < aFlags.size(); i++)
2651 fFlags |= aFlags[i];
2652
2653 if (fFlags)
2654 if (!(fFlags & DirectoryCreateFlag_Parents))
2655 return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), fFlags);
2656 }
2657
2658 HRESULT hr = S_OK;
2659
2660 ComObjPtr <GuestDirectory> pDirectory; int guestRc;
2661 int rc = i_directoryCreateInternal(aPath, (uint32_t)aMode, fFlags, &guestRc);
2662 if (RT_FAILURE(rc))
2663 {
2664 switch (rc)
2665 {
2666 case VERR_GSTCTL_GUEST_ERROR:
2667 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: %s",
2668 GuestDirectory::i_guestErrorToString(guestRc).c_str()));
2669 break;
2670
2671 case VERR_INVALID_PARAMETER:
2672 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: Invalid parameters given"));
2673 break;
2674
2675 case VERR_BROKEN_PIPE:
2676 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: Unexpectedly aborted"));
2677 break;
2678
2679 default:
2680 hr = setError(VBOX_E_IPRT_ERROR, tr("Directory creation failed: %Rrc"), rc);
2681 break;
2682 }
2683 }
2684
2685 return hr;
2686}
2687
2688HRESULT GuestSession::directoryCreateTemp(const com::Utf8Str &aTemplateName, ULONG aMode, const com::Utf8Str &aPath,
2689 BOOL aSecure, com::Utf8Str &aDirectory)
2690{
2691 LogFlowThisFuncEnter();
2692
2693 if (RT_UNLIKELY((aTemplateName.c_str()) == NULL || *(aTemplateName.c_str()) == '\0'))
2694 return setError(E_INVALIDARG, tr("No template specified"));
2695 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
2696 return setError(E_INVALIDARG, tr("No directory name specified"));
2697
2698 HRESULT hr = S_OK;
2699
2700 int guestRc;
2701 int rc = i_objectCreateTempInternal(aTemplateName,
2702 aPath,
2703 true /* Directory */, aDirectory, &guestRc);
2704 if (!RT_SUCCESS(rc))
2705 {
2706 switch (rc)
2707 {
2708 case VERR_GSTCTL_GUEST_ERROR:
2709 hr = GuestProcess::i_setErrorExternal(this, guestRc);
2710 break;
2711
2712 default:
2713 hr = setError(VBOX_E_IPRT_ERROR, tr("Temporary directory creation \"%s\" with template \"%s\" failed: %Rrc"),
2714 aPath.c_str(), aTemplateName.c_str(), rc);
2715 break;
2716 }
2717 }
2718
2719 return hr;
2720}
2721
2722HRESULT GuestSession::directoryExists(const com::Utf8Str &aPath, BOOL aFollowSymlinks, BOOL *aExists)
2723{
2724 LogFlowThisFuncEnter();
2725
2726 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
2727 return setError(E_INVALIDARG, tr("No directory to check existence for specified"));
2728
2729 HRESULT hr = S_OK;
2730
2731 GuestFsObjData objData; int guestRc;
2732 int rc = i_directoryQueryInfoInternal(aPath, aFollowSymlinks != FALSE, objData, &guestRc);
2733 if (RT_SUCCESS(rc))
2734 *aExists = objData.mType == FsObjType_Directory;
2735 else
2736 {
2737 /** @todo r=bird: Looks like this code raises errors if the directory doesn't
2738 * exist... That's of course not right. */
2739 switch (rc)
2740 {
2741 case VERR_GSTCTL_GUEST_ERROR:
2742 hr = GuestProcess::i_setErrorExternal(this, guestRc);
2743 break;
2744
2745 default:
2746 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying directory existence \"%s\" failed: %Rrc"),
2747 aPath.c_str(), rc);
2748 break;
2749 }
2750 }
2751
2752 return hr;
2753}
2754
2755HRESULT GuestSession::directoryOpen(const com::Utf8Str &aPath, const com::Utf8Str &aFilter,
2756 const std::vector<DirectoryOpenFlag_T> &aFlags, ComPtr<IGuestDirectory> &aDirectory)
2757{
2758 LogFlowThisFuncEnter();
2759
2760 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
2761 return setError(E_INVALIDARG, tr("No directory to open specified"));
2762 if (RT_UNLIKELY((aFilter.c_str()) != NULL && *(aFilter.c_str()) != '\0'))
2763 return setError(E_INVALIDARG, tr("Directory filters are not implemented yet"));
2764
2765 uint32_t fFlags = DirectoryOpenFlag_None;
2766 if (aFlags.size())
2767 {
2768 for (size_t i = 0; i < aFlags.size(); i++)
2769 fFlags |= aFlags[i];
2770
2771 if (fFlags)
2772 return setError(E_INVALIDARG, tr("Open flags (%#x) not implemented yet"), fFlags);
2773 }
2774
2775 HRESULT hr = S_OK;
2776
2777 GuestDirectoryOpenInfo openInfo;
2778 openInfo.mPath = aPath;
2779 openInfo.mFilter = aFilter;
2780 openInfo.mFlags = fFlags;
2781
2782 ComObjPtr <GuestDirectory> pDirectory; int guestRc;
2783 int rc = i_directoryOpenInternal(openInfo, pDirectory, &guestRc);
2784 if (RT_SUCCESS(rc))
2785 {
2786 /* Return directory object to the caller. */
2787 hr = pDirectory.queryInterfaceTo(aDirectory.asOutParam());
2788 }
2789 else
2790 {
2791 switch (rc)
2792 {
2793 case VERR_INVALID_PARAMETER:
2794 hr = setError(VBOX_E_IPRT_ERROR, tr("Opening directory \"%s\" failed; invalid parameters given"),
2795 aPath.c_str());
2796 break;
2797
2798 case VERR_GSTCTL_GUEST_ERROR:
2799 hr = GuestDirectory::i_setErrorExternal(this, guestRc);
2800 break;
2801
2802 default:
2803 hr = setError(VBOX_E_IPRT_ERROR, tr("Opening directory \"%s\" failed: %Rrc"),
2804 aPath.c_str(),rc);
2805 break;
2806 }
2807 }
2808
2809 return hr;
2810}
2811
2812HRESULT GuestSession::directoryRemove(const com::Utf8Str &aPath)
2813{
2814 LogFlowThisFuncEnter();
2815
2816 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
2817 return setError(E_INVALIDARG, tr("No directory to remove specified"));
2818
2819 HRESULT hr = i_isReadyExternal();
2820 if (FAILED(hr))
2821 return hr;
2822
2823 /* No flags; only remove the directory when empty. */
2824 uint32_t uFlags = 0;
2825
2826 int guestRc;
2827 int vrc = i_directoryRemoveInternal(aPath, uFlags, &guestRc);
2828 if (RT_FAILURE(vrc))
2829 {
2830 switch (vrc)
2831 {
2832 case VERR_NOT_SUPPORTED:
2833 hr = setError(VBOX_E_IPRT_ERROR,
2834 tr("Handling removing guest directories not supported by installed Guest Additions"));
2835 break;
2836
2837 case VERR_GSTCTL_GUEST_ERROR:
2838 hr = GuestDirectory::i_setErrorExternal(this, guestRc);
2839 break;
2840
2841 default:
2842 hr = setError(VBOX_E_IPRT_ERROR, tr("Removing guest directory \"%s\" failed: %Rrc"),
2843 aPath.c_str(), vrc);
2844 break;
2845 }
2846 }
2847
2848 return hr;
2849}
2850
2851HRESULT GuestSession::directoryRemoveRecursive(const com::Utf8Str &aPath, const std::vector<DirectoryRemoveRecFlag_T> &aFlags,
2852 ComPtr<IProgress> &aProgress)
2853{
2854 LogFlowThisFuncEnter();
2855
2856 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
2857 return setError(E_INVALIDARG, tr("No directory to remove recursively specified"));
2858
2859/** @todo r=bird: Must check that the flags matches the hardcoded behavior
2860 * further down!! */
2861
2862 HRESULT hr = i_isReadyExternal();
2863 if (FAILED(hr))
2864 return hr;
2865
2866 ComObjPtr<Progress> pProgress;
2867 hr = pProgress.createObject();
2868 if (SUCCEEDED(hr))
2869 hr = pProgress->init(static_cast<IGuestSession *>(this),
2870 Bstr(tr("Removing guest directory")).raw(),
2871 TRUE /*aCancelable*/);
2872 if (FAILED(hr))
2873 return hr;
2874
2875 /* Note: At the moment we don't supply progress information while
2876 * deleting a guest directory recursively. So just complete
2877 * the progress object right now. */
2878 /** @todo Implement progress reporting on guest directory deletion! */
2879 hr = pProgress->i_notifyComplete(S_OK);
2880 if (FAILED(hr))
2881 return hr;
2882
2883 /* Remove the directory + all its contents. */
2884 uint32_t uFlags = DIRREMOVE_FLAG_RECURSIVE
2885 | DIRREMOVE_FLAG_CONTENT_AND_DIR;
2886 int guestRc;
2887 int vrc = i_directoryRemoveInternal(aPath, uFlags, &guestRc);
2888 if (RT_FAILURE(vrc))
2889 {
2890 switch (vrc)
2891 {
2892 case VERR_NOT_SUPPORTED:
2893 hr = setError(VBOX_E_IPRT_ERROR,
2894 tr("Handling removing guest directories recursively not supported by installed Guest Additions"));
2895 break;
2896
2897 case VERR_GSTCTL_GUEST_ERROR:
2898 hr = GuestFile::i_setErrorExternal(this, guestRc);
2899 break;
2900
2901 default:
2902 hr = setError(VBOX_E_IPRT_ERROR, tr("Recursively removing guest directory \"%s\" failed: %Rrc"),
2903 aPath.c_str(), vrc);
2904 break;
2905 }
2906 }
2907 else
2908 {
2909 pProgress.queryInterfaceTo(aProgress.asOutParam());
2910 }
2911
2912 return hr;
2913}
2914
2915HRESULT GuestSession::environmentScheduleSet(const com::Utf8Str &aName, const com::Utf8Str &aValue)
2916{
2917 LogFlowThisFuncEnter();
2918
2919 HRESULT hrc;
2920 if (RT_LIKELY(aName.isNotEmpty()))
2921 {
2922 if (RT_LIKELY(strchr(aName.c_str(), '=') == NULL))
2923 {
2924 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2925 int vrc = mData.mEnvironmentChanges.setVariable(aName, aValue);
2926 if (RT_SUCCESS(vrc))
2927 hrc = S_OK;
2928 else
2929 hrc = setErrorVrc(vrc);
2930 }
2931 else
2932 hrc = setError(E_INVALIDARG, tr("The equal char is not allowed in environment variable names"));
2933 }
2934 else
2935 hrc = setError(E_INVALIDARG, tr("No variable name specified"));
2936
2937 LogFlowThisFuncLeave();
2938 return hrc;
2939}
2940
2941HRESULT GuestSession::environmentScheduleUnset(const com::Utf8Str &aName)
2942{
2943 LogFlowThisFuncEnter();
2944 HRESULT hrc;
2945 if (RT_LIKELY(aName.isNotEmpty()))
2946 {
2947 if (RT_LIKELY(strchr(aName.c_str(), '=') == NULL))
2948 {
2949 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2950 int vrc = mData.mEnvironmentChanges.unsetVariable(aName);
2951 if (RT_SUCCESS(vrc))
2952 hrc = S_OK;
2953 else
2954 hrc = setErrorVrc(vrc);
2955 }
2956 else
2957 hrc = setError(E_INVALIDARG, tr("The equal char is not allowed in environment variable names"));
2958 }
2959 else
2960 hrc = setError(E_INVALIDARG, tr("No variable name specified"));
2961
2962 LogFlowThisFuncLeave();
2963 return hrc;
2964}
2965
2966HRESULT GuestSession::environmentGetBaseVariable(const com::Utf8Str &aName, com::Utf8Str &aValue)
2967{
2968 LogFlowThisFuncEnter();
2969 HRESULT hrc;
2970 if (RT_LIKELY(aName.isNotEmpty()))
2971 {
2972 if (RT_LIKELY(strchr(aName.c_str(), '=') == NULL))
2973 {
2974 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2975 if (mData.mpBaseEnvironment)
2976 {
2977 int vrc = mData.mpBaseEnvironment->getVariable(aName, &aValue);
2978 if (RT_SUCCESS(vrc))
2979 hrc = S_OK;
2980 else
2981 hrc = setErrorVrc(vrc);
2982 }
2983 else if (mData.mProtocolVersion < 99999)
2984 hrc = setError(VBOX_E_NOT_SUPPORTED, tr("The base environment feature is not supported by the guest additions"));
2985 else
2986 hrc = setError(VBOX_E_INVALID_OBJECT_STATE, tr("The base environment has not yet been reported by the guest"));
2987 }
2988 else
2989 hrc = setError(E_INVALIDARG, tr("The equal char is not allowed in environment variable names"));
2990 }
2991 else
2992 hrc = setError(E_INVALIDARG, tr("No variable name specified"));
2993
2994 LogFlowThisFuncLeave();
2995 return hrc;
2996}
2997
2998HRESULT GuestSession::environmentDoesBaseVariableExist(const com::Utf8Str &aName, BOOL *aExists)
2999{
3000 LogFlowThisFuncEnter();
3001 *aExists = FALSE;
3002 HRESULT hrc;
3003 if (RT_LIKELY(aName.isNotEmpty()))
3004 {
3005 if (RT_LIKELY(strchr(aName.c_str(), '=') == NULL))
3006 {
3007 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3008 if (mData.mpBaseEnvironment)
3009 {
3010 hrc = S_OK;
3011 *aExists = mData.mpBaseEnvironment->doesVariableExist(aName);
3012 }
3013 else if (mData.mProtocolVersion < 99999)
3014 hrc = setError(VBOX_E_NOT_SUPPORTED, tr("The base environment feature is not supported by the guest additions"));
3015 else
3016 hrc = setError(VBOX_E_INVALID_OBJECT_STATE, tr("The base environment has not yet been reported by the guest"));
3017 }
3018 else
3019 hrc = setError(E_INVALIDARG, tr("The equal char is not allowed in environment variable names"));
3020 }
3021 else
3022 hrc = setError(E_INVALIDARG, tr("No variable name specified"));
3023
3024 LogFlowThisFuncLeave();
3025 return hrc;
3026}
3027
3028HRESULT GuestSession::fileCreateTemp(const com::Utf8Str &aTemplateName, ULONG aMode, const com::Utf8Str &aPath, BOOL aSecure,
3029 ComPtr<IGuestFile> &aFile)
3030{
3031 ReturnComNotImplemented();
3032}
3033
3034HRESULT GuestSession::fileExists(const com::Utf8Str &aPath, BOOL aFollowSymlinks, BOOL *aExists)
3035{
3036 LogFlowThisFuncEnter();
3037
3038/** @todo r=bird: Treat empty file with a FALSE return. */
3039 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
3040 return setError(E_INVALIDARG, tr("No file to check existence for specified"));
3041
3042 GuestFsObjData objData; int guestRc;
3043 int vrc = i_fileQueryInfoInternal(aPath, aFollowSymlinks != FALSE, objData, &guestRc);
3044 if (RT_SUCCESS(vrc))
3045 {
3046 *aExists = TRUE;
3047 return S_OK;
3048 }
3049
3050 HRESULT hr = S_OK;
3051
3052 switch (vrc)
3053 {
3054 case VERR_GSTCTL_GUEST_ERROR:
3055 hr = GuestProcess::i_setErrorExternal(this, guestRc);
3056 break;
3057
3058/** @todo r=bird: what about VERR_PATH_NOT_FOUND and VERR_FILE_NOT_FOUND?
3059 * Where does that get converted to *aExists = FALSE? */
3060 case VERR_NOT_A_FILE:
3061 *aExists = FALSE;
3062 break;
3063
3064 default:
3065 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying file information for \"%s\" failed: %Rrc"),
3066 aPath.c_str(), vrc);
3067 break;
3068 }
3069
3070 return hr;
3071}
3072
3073HRESULT GuestSession::fileOpen(const com::Utf8Str &aPath, FileAccessMode_T aAccessMode, FileOpenAction_T aOpenAction,
3074 ULONG aCreationMode, ComPtr<IGuestFile> &aFile)
3075{
3076 LogFlowThisFuncEnter();
3077 const std::vector<FileOpenExFlags_T> EmptyFlags;
3078 return fileOpenEx(aPath, aAccessMode, aOpenAction, FileSharingMode_All, aCreationMode, EmptyFlags, aFile);
3079}
3080
3081HRESULT GuestSession::fileOpenEx(const com::Utf8Str &aPath, FileAccessMode_T aAccessMode, FileOpenAction_T aOpenAction,
3082 FileSharingMode_T aSharingMode, ULONG aCreationMode,
3083 const std::vector<FileOpenExFlags_T> &aFlags, ComPtr<IGuestFile> &aFile)
3084{
3085 LogFlowThisFuncEnter();
3086
3087 if (RT_UNLIKELY((aPath.c_str()) == NULL || *(aPath.c_str()) == '\0'))
3088 return setError(E_INVALIDARG, tr("No file to open specified"));
3089
3090 HRESULT hr = i_isReadyExternal();
3091 if (FAILED(hr))
3092 return hr;
3093
3094 GuestFileOpenInfo openInfo;
3095 openInfo.mFileName = aPath;
3096 openInfo.mCreationMode = aCreationMode;
3097
3098 /* convert + validate aAccessMode to the old format. */
3099 openInfo.mAccessMode = aAccessMode;
3100 switch (aAccessMode)
3101 {
3102 case (FileAccessMode_T)FileAccessMode_ReadOnly: openInfo.mpszAccessMode = "r"; break;
3103 case (FileAccessMode_T)FileAccessMode_WriteOnly: openInfo.mpszAccessMode = "w"; break;
3104 case (FileAccessMode_T)FileAccessMode_ReadWrite: openInfo.mpszAccessMode = "r+"; break;
3105 case (FileAccessMode_T)FileAccessMode_AppendOnly:
3106 /* fall thru */
3107 case (FileAccessMode_T)FileAccessMode_AppendRead:
3108 return setError(E_NOTIMPL, tr("Append access modes are not yet implemented"));
3109 default:
3110 return setError(E_INVALIDARG, tr("Unknown FileAccessMode value %u (%#x)"), aAccessMode, aAccessMode);
3111 }
3112
3113 /* convert + validate aOpenAction to the old format. */
3114 openInfo.mOpenAction = aOpenAction;
3115 switch (aOpenAction)
3116 {
3117 case (FileOpenAction_T)FileOpenAction_OpenExisting: openInfo.mpszOpenAction = "oe"; break;
3118 case (FileOpenAction_T)FileOpenAction_OpenOrCreate: openInfo.mpszOpenAction = "oc"; break;
3119 case (FileOpenAction_T)FileOpenAction_CreateNew: openInfo.mpszOpenAction = "ce"; break;
3120 case (FileOpenAction_T)FileOpenAction_CreateOrReplace: openInfo.mpszOpenAction = "ca"; break;
3121 case (FileOpenAction_T)FileOpenAction_OpenExistingTruncated: openInfo.mpszOpenAction = "ot"; break;
3122 case (FileOpenAction_T)FileOpenAction_AppendOrCreate:
3123 openInfo.mpszOpenAction = "oa"; /** @todo get rid of this one and implement AppendOnly/AppendRead. */
3124 break;
3125 default:
3126 return setError(E_INVALIDARG, tr("Unknown FileOpenAction value %u (%#x)"), aAccessMode, aAccessMode);
3127 }
3128
3129 /* validate aSharingMode */
3130 openInfo.mSharingMode = aSharingMode;
3131 switch (aSharingMode)
3132 {
3133 case (FileSharingMode_T)FileSharingMode_All: /* OK */ break;
3134 case (FileSharingMode_T)FileSharingMode_Read:
3135 case (FileSharingMode_T)FileSharingMode_Write:
3136 case (FileSharingMode_T)FileSharingMode_ReadWrite:
3137 case (FileSharingMode_T)FileSharingMode_Delete:
3138 case (FileSharingMode_T)FileSharingMode_ReadDelete:
3139 case (FileSharingMode_T)FileSharingMode_WriteDelete:
3140 return setError(E_NOTIMPL, tr("Only FileSharingMode_All is currently implemented"));
3141
3142 default:
3143 return setError(E_INVALIDARG, tr("Unknown FileOpenAction value %u (%#x)"), aAccessMode, aAccessMode);
3144 }
3145
3146 /* Combine and validate flags. */
3147 uint32_t fOpenEx = 0;
3148 for (size_t i = 0; i < aFlags.size(); i++)
3149 fOpenEx = aFlags[i];
3150 if (fOpenEx)
3151 return setError(E_INVALIDARG, tr("Unsupported FileOpenExFlags values in aFlags (%#x)"), fOpenEx);
3152 openInfo.mfOpenEx = fOpenEx;
3153
3154 ComObjPtr <GuestFile> pFile;
3155 int guestRc;
3156 int vrc = i_fileOpenInternal(openInfo, pFile, &guestRc);
3157 if (RT_SUCCESS(vrc))
3158 /* Return directory object to the caller. */
3159 hr = pFile.queryInterfaceTo(aFile.asOutParam());
3160 else
3161 {
3162 switch (vrc)
3163 {
3164 case VERR_NOT_SUPPORTED:
3165 hr = setError(VBOX_E_IPRT_ERROR,
3166 tr("Handling guest files not supported by installed Guest Additions"));
3167 break;
3168
3169 case VERR_GSTCTL_GUEST_ERROR:
3170 hr = GuestFile::i_setErrorExternal(this, guestRc);
3171 break;
3172
3173 default:
3174 hr = setError(VBOX_E_IPRT_ERROR, tr("Opening guest file \"%s\" failed: %Rrc"),
3175 aPath.c_str(), vrc);
3176 break;
3177 }
3178 }
3179
3180 return hr;
3181}
3182
3183HRESULT GuestSession::fileQuerySize(const com::Utf8Str &aPath, BOOL aFollowSymlinks, LONG64 *aSize)
3184{
3185 if (aPath.isEmpty())
3186 return setError(E_INVALIDARG, tr("No path specified"));
3187
3188 HRESULT hr = S_OK;
3189
3190 int64_t llSize; int guestRc;
3191 int vrc = i_fileQuerySizeInternal(aPath, aFollowSymlinks != FALSE, &llSize, &guestRc);
3192 if (RT_SUCCESS(vrc))
3193 {
3194 *aSize = llSize;
3195 }
3196 else
3197 {
3198 if (GuestProcess::i_isGuestError(vrc))
3199 {
3200 hr = GuestProcess::i_setErrorExternal(this, guestRc);
3201 }
3202 else
3203 hr = setError(VBOX_E_IPRT_ERROR, tr("Querying file size failed: %Rrc"), vrc);
3204 }
3205
3206 return hr;
3207}
3208
3209HRESULT GuestSession::fsObjExists(const com::Utf8Str &aPath, BOOL aFollowSymlinks, BOOL *aExists)
3210{
3211 if (aPath.isEmpty())
3212 return setError(E_INVALIDARG, tr("No path specified"));
3213
3214 LogFlowThisFunc(("aPath=%s, aFollowSymlinks=%RTbool\n", aPath.c_str(), RT_BOOL(aFollowSymlinks)));
3215
3216 HRESULT hrc = S_OK;
3217
3218 *aExists = false;
3219
3220 GuestFsObjData objData;
3221 int rcGuest;
3222 int vrc = i_fsQueryInfoInternal(aPath, aFollowSymlinks != FALSE, objData, &rcGuest);
3223 if (RT_SUCCESS(vrc))
3224 {
3225 *aExists = TRUE;
3226 }
3227 else
3228 {
3229 if (GuestProcess::i_isGuestError(vrc))
3230 {
3231 if ( rcGuest == VERR_NOT_A_FILE
3232 || rcGuest == VERR_PATH_NOT_FOUND
3233 || rcGuest == VERR_FILE_NOT_FOUND
3234 || rcGuest == VERR_INVALID_NAME)
3235 {
3236 hrc = S_OK; /* Ignore these vrc values. */
3237 }
3238 else
3239 hrc = GuestProcess::i_setErrorExternal(this, rcGuest);
3240 }
3241 else
3242 hrc = setErrorVrc(vrc, tr("Querying file information for \"%s\" failed: %Rrc"), aPath.c_str(), vrc);
3243 }
3244
3245 return hrc;
3246}
3247
3248HRESULT GuestSession::fsObjQueryInfo(const com::Utf8Str &aPath, BOOL aFollowSymlinks, ComPtr<IGuestFsObjInfo> &aInfo)
3249{
3250 if (aPath.isEmpty())
3251 return setError(E_INVALIDARG, tr("No path specified"));
3252
3253 LogFlowThisFunc(("aPath=%s, aFollowSymlinks=%RTbool\n", aPath.c_str(), RT_BOOL(aFollowSymlinks)));
3254
3255 HRESULT hrc = S_OK;
3256
3257 GuestFsObjData Info; int guestRc;
3258 int vrc = i_fsQueryInfoInternal(aPath, aFollowSymlinks != FALSE, Info, &guestRc);
3259 if (RT_SUCCESS(vrc))
3260 {
3261 ComObjPtr<GuestFsObjInfo> ptrFsObjInfo;
3262 hrc = ptrFsObjInfo.createObject();
3263 if (SUCCEEDED(hrc))
3264 {
3265 vrc = ptrFsObjInfo->init(Info);
3266 if (RT_SUCCESS(vrc))
3267 hrc = ptrFsObjInfo.queryInterfaceTo(aInfo.asOutParam());
3268 else
3269 hrc = setErrorVrc(vrc);
3270 }
3271 }
3272 else
3273 {
3274 if (GuestProcess::i_isGuestError(vrc))
3275 {
3276 hrc = GuestProcess::i_setErrorExternal(this, guestRc);
3277 }
3278 else
3279 hrc = setErrorVrc(vrc, tr("Querying file information for \"%s\" failed: %Rrc"), aPath.c_str(), vrc);
3280 }
3281
3282 return hrc;
3283}
3284
3285HRESULT GuestSession::fsObjRemove(const com::Utf8Str &aPath)
3286{
3287 if (RT_UNLIKELY(aPath.isEmpty()))
3288 return setError(E_INVALIDARG, tr("No path specified"));
3289
3290 LogFlowThisFunc(("aPath=%s\n", aPath.c_str()));
3291
3292 HRESULT hrc = S_OK;
3293
3294 int guestRc;
3295 int vrc = i_fileRemoveInternal(aPath, &guestRc);
3296 if (RT_FAILURE(vrc))
3297 {
3298 if (GuestProcess::i_isGuestError(vrc))
3299 {
3300 hrc = GuestProcess::i_setErrorExternal(this, guestRc);
3301 }
3302 else
3303 hrc = setError(VBOX_E_IPRT_ERROR, tr("Removing file \"%s\" failed: %Rrc"), aPath.c_str(), vrc);
3304 }
3305
3306 return hrc;
3307}
3308
3309HRESULT GuestSession::fsObjRename(const com::Utf8Str &aSource,
3310 const com::Utf8Str &aDestination,
3311 const std::vector<FsObjRenameFlag_T> &aFlags)
3312{
3313 if (RT_UNLIKELY(aSource.isEmpty()))
3314 return setError(E_INVALIDARG, tr("No source path specified"));
3315
3316 if (RT_UNLIKELY(aDestination.isEmpty()))
3317 return setError(E_INVALIDARG, tr("No destination path specified"));
3318
3319 LogFlowThisFunc(("aSource=%s, aDestination=%s\n", aSource.c_str(), aDestination.c_str()));
3320
3321 HRESULT hr = i_isReadyExternal();
3322 if (FAILED(hr))
3323 return hr;
3324
3325 /* Combine, validate and convert flags. */
3326 uint32_t fApiFlags = 0;
3327 for (size_t i = 0; i < aFlags.size(); i++)
3328 fApiFlags |= aFlags[i];
3329 if (fApiFlags & ~((uint32_t)FsObjRenameFlag_NoReplace | (uint32_t)FsObjRenameFlag_Replace))
3330 return setError(E_INVALIDARG, tr("Unknown rename flag: %#x"), fApiFlags);
3331
3332 AssertCompile(FsObjRenameFlag_NoReplace == 0);
3333 AssertCompile(FsObjRenameFlag_Replace != 0);
3334 uint32_t fBackend;
3335 if ((fApiFlags & ((uint32_t)FsObjRenameFlag_NoReplace | (uint32_t)FsObjRenameFlag_Replace)) == FsObjRenameFlag_Replace)
3336 fBackend = PATHRENAME_FLAG_REPLACE;
3337 else
3338 fBackend = PATHRENAME_FLAG_NO_REPLACE;
3339
3340 /* Call worker to do the job. */
3341 int guestRc;
3342 int vrc = i_pathRenameInternal(aSource, aDestination, fBackend, &guestRc);
3343 if (RT_FAILURE(vrc))
3344 {
3345 switch (vrc)
3346 {
3347 case VERR_NOT_SUPPORTED:
3348 hr = setError(VBOX_E_IPRT_ERROR,
3349 tr("Handling renaming guest directories not supported by installed Guest Additions"));
3350 break;
3351
3352 case VERR_GSTCTL_GUEST_ERROR:
3353 hr = setError(VBOX_E_IPRT_ERROR,
3354 tr("Renaming guest directory failed: %Rrc"), guestRc);
3355 break;
3356
3357 default:
3358 hr = setError(VBOX_E_IPRT_ERROR, tr("Renaming guest directory \"%s\" failed: %Rrc"),
3359 aSource.c_str(), vrc);
3360 break;
3361 }
3362 }
3363
3364 return hr;
3365}
3366
3367HRESULT GuestSession::fsObjMove(const com::Utf8Str &aSource, const com::Utf8Str &aDestination,
3368 const std::vector<FsObjMoveFlags_T> &aFlags, ComPtr<IProgress> &aProgress)
3369{
3370 ReturnComNotImplemented();
3371}
3372
3373HRESULT GuestSession::fsObjSetACL(const com::Utf8Str &aPath, BOOL aFollowSymlinks, const com::Utf8Str &aAcl, ULONG aMode)
3374{
3375 ReturnComNotImplemented();
3376}
3377
3378
3379HRESULT GuestSession::processCreate(const com::Utf8Str &aExecutable, const std::vector<com::Utf8Str> &aArguments,
3380 const std::vector<com::Utf8Str> &aEnvironment,
3381 const std::vector<ProcessCreateFlag_T> &aFlags,
3382 ULONG aTimeoutMS, ComPtr<IGuestProcess> &aGuestProcess)
3383{
3384 LogFlowThisFuncEnter();
3385
3386 std::vector<LONG> affinityIgnored;
3387
3388 return processCreateEx(aExecutable, aArguments, aEnvironment, aFlags, aTimeoutMS, ProcessPriority_Default,
3389 affinityIgnored, aGuestProcess);
3390}
3391
3392HRESULT GuestSession::processCreateEx(const com::Utf8Str &aExecutable, const std::vector<com::Utf8Str> &aArguments,
3393 const std::vector<com::Utf8Str> &aEnvironment,
3394 const std::vector<ProcessCreateFlag_T> &aFlags, ULONG aTimeoutMS,
3395 ProcessPriority_T aPriority, const std::vector<LONG> &aAffinity,
3396 ComPtr<IGuestProcess> &aGuestProcess)
3397{
3398 LogFlowThisFuncEnter();
3399
3400 /** @todo r=bird: Check input better? aPriority is passed on to the guest
3401 * without any validation. Flags not existing in this vbox version are
3402 * ignored, potentially doing something entirely different than what the
3403 * caller had in mind. */
3404
3405 /*
3406 * Must have an executable to execute. If none is given, we try use the
3407 * zero'th argument.
3408 */
3409 const char *pszExecutable = aExecutable.c_str();
3410 if (RT_UNLIKELY(pszExecutable == NULL || *pszExecutable == '\0'))
3411 {
3412 if (aArguments.size() > 0)
3413 pszExecutable = aArguments[0].c_str();
3414 if (pszExecutable == NULL || *pszExecutable == '\0')
3415 return setError(E_INVALIDARG, tr("No command to execute specified"));
3416 }
3417
3418 /*
3419 * Check the session.
3420 */
3421 HRESULT hr = i_isReadyExternal();
3422 if (FAILED(hr))
3423 return hr;
3424
3425 /*
3426 * Build the process startup info.
3427 */
3428 GuestProcessStartupInfo procInfo;
3429
3430 /* Executable and arguments. */
3431 procInfo.mExecutable = pszExecutable;
3432 if (aArguments.size())
3433 for (size_t i = 0; i < aArguments.size(); i++)
3434 procInfo.mArguments.push_back(aArguments[i]);
3435
3436 /* Combine the environment changes associated with the ones passed in by
3437 the caller, giving priority to the latter. The changes are putenv style
3438 and will be applied to the standard environment for the guest user. */
3439 int vrc = procInfo.mEnvironmentChanges.copy(mData.mEnvironmentChanges);
3440 if (RT_SUCCESS(vrc))
3441 vrc = procInfo.mEnvironmentChanges.applyPutEnvArray(aEnvironment);
3442 if (RT_SUCCESS(vrc))
3443 {
3444 /* Convert the flag array into a mask. */
3445 if (aFlags.size())
3446 for (size_t i = 0; i < aFlags.size(); i++)
3447 procInfo.mFlags |= aFlags[i];
3448
3449 procInfo.mTimeoutMS = aTimeoutMS;
3450
3451 /** @todo use RTCPUSET instead of archaic 64-bit variables! */
3452 if (aAffinity.size())
3453 for (size_t i = 0; i < aAffinity.size(); i++)
3454 if (aAffinity[i])
3455 procInfo.mAffinity |= (uint64_t)1 << i;
3456
3457 procInfo.mPriority = aPriority;
3458
3459 /*
3460 * Create a guest process object.
3461 */
3462 ComObjPtr<GuestProcess> pProcess;
3463 vrc = i_processCreateExInternal(procInfo, pProcess);
3464 if (RT_SUCCESS(vrc))
3465 {
3466 ComPtr<IGuestProcess> pIProcess;
3467 hr = pProcess.queryInterfaceTo(pIProcess.asOutParam());
3468 if (SUCCEEDED(hr))
3469 {
3470 /*
3471 * Start the process.
3472 */
3473 vrc = pProcess->i_startProcessAsync();
3474 if (RT_SUCCESS(vrc))
3475 {
3476 aGuestProcess = pIProcess;
3477
3478 LogFlowFuncLeaveRC(vrc);
3479 return S_OK;
3480 }
3481
3482 hr = setErrorVrc(vrc, tr("Failed to start guest process: %Rrc"), vrc);
3483 }
3484 }
3485 else if (vrc == VERR_MAX_PROCS_REACHED)
3486 hr = setErrorVrc(vrc, tr("Maximum number of concurrent guest processes per session (%u) reached"),
3487 VBOX_GUESTCTRL_MAX_OBJECTS);
3488 else
3489 hr = setErrorVrc(vrc, tr("Failed to create guest process object: %Rrc"), vrc);
3490 }
3491 else
3492 hr = setErrorVrc(vrc, tr("Failed to set up the environment: %Rrc"), vrc);
3493
3494 LogFlowFuncLeaveRC(vrc);
3495 return hr;
3496}
3497
3498HRESULT GuestSession::processGet(ULONG aPid, ComPtr<IGuestProcess> &aGuestProcess)
3499
3500{
3501 LogFlowThisFunc(("PID=%RU32\n", aPid));
3502
3503 if (aPid == 0)
3504 return setError(E_INVALIDARG, tr("No valid process ID (PID) specified"));
3505
3506 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3507
3508 HRESULT hr = S_OK;
3509
3510 ComObjPtr<GuestProcess> pProcess;
3511 int rc = i_processGetByPID(aPid, &pProcess);
3512 if (RT_FAILURE(rc))
3513 hr = setError(E_INVALIDARG, tr("No process with PID %RU32 found"), aPid);
3514
3515 /* This will set (*aProcess) to NULL if pProgress is NULL. */
3516 HRESULT hr2 = pProcess.queryInterfaceTo(aGuestProcess.asOutParam());
3517 if (SUCCEEDED(hr))
3518 hr = hr2;
3519
3520 LogFlowThisFunc(("aProcess=%p, hr=%Rhrc\n", (IGuestProcess*)aGuestProcess, hr));
3521 return hr;
3522}
3523
3524HRESULT GuestSession::symlinkCreate(const com::Utf8Str &aSource, const com::Utf8Str &aTarget, SymlinkType_T aType)
3525{
3526 ReturnComNotImplemented();
3527}
3528
3529HRESULT GuestSession::symlinkExists(const com::Utf8Str &aSymlink, BOOL *aExists)
3530
3531{
3532 ReturnComNotImplemented();
3533}
3534
3535HRESULT GuestSession::symlinkRead(const com::Utf8Str &aSymlink, const std::vector<SymlinkReadFlag_T> &aFlags,
3536 com::Utf8Str &aTarget)
3537{
3538 ReturnComNotImplemented();
3539}
3540
3541HRESULT GuestSession::waitFor(ULONG aWaitFor, ULONG aTimeoutMS, GuestSessionWaitResult_T *aReason)
3542{
3543 LogFlowThisFuncEnter();
3544
3545 /*
3546 * Note: Do not hold any locks here while waiting!
3547 */
3548 HRESULT hr = S_OK;
3549
3550 int guestRc; GuestSessionWaitResult_T waitResult;
3551 int vrc = i_waitFor(aWaitFor, aTimeoutMS, waitResult, &guestRc);
3552 if (RT_SUCCESS(vrc))
3553 *aReason = waitResult;
3554 else
3555 {
3556 switch (vrc)
3557 {
3558 case VERR_GSTCTL_GUEST_ERROR:
3559 hr = GuestSession::i_setErrorExternal(this, guestRc);
3560 break;
3561
3562 case VERR_TIMEOUT:
3563 *aReason = GuestSessionWaitResult_Timeout;
3564 break;
3565
3566 default:
3567 {
3568 const char *pszSessionName = mData.mSession.mName.c_str();
3569 hr = setError(VBOX_E_IPRT_ERROR,
3570 tr("Waiting for guest session \"%s\" failed: %Rrc"),
3571 pszSessionName ? pszSessionName : tr("Unnamed"), vrc);
3572 break;
3573 }
3574 }
3575 }
3576
3577 LogFlowFuncLeaveRC(vrc);
3578 return hr;
3579}
3580
3581HRESULT GuestSession::waitForArray(const std::vector<GuestSessionWaitForFlag_T> &aWaitFor, ULONG aTimeoutMS,
3582 GuestSessionWaitResult_T *aReason)
3583{
3584 LogFlowThisFuncEnter();
3585
3586 /*
3587 * Note: Do not hold any locks here while waiting!
3588 */
3589 uint32_t fWaitFor = GuestSessionWaitForFlag_None;
3590 for (size_t i = 0; i < aWaitFor.size(); i++)
3591 fWaitFor |= aWaitFor[i];
3592
3593 return WaitFor(fWaitFor, aTimeoutMS, aReason);
3594}
3595
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