VirtualBox

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

Last change on this file since 56927 was 56711, checked in by vboxsync, 10 years ago

IGuestSession::fsObjQueryInfo: Return E_INVALIDARG if empty input path, we currently don't have a documented way of indicating that something doesn't exists it seems. sigh.

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