VirtualBox

source: vbox/trunk/src/VBox/Main/include/GuestCtrlImplPrivate.h@ 76815

Last change on this file since 76815 was 76562, checked in by vboxsync, 6 years ago

Main: Use MAIN_INCLUDED_ and MAIN_INCLUDED_SRC_ as header guard prefixes with scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 34.9 KB
Line 
1/* $Id: GuestCtrlImplPrivate.h 76562 2019-01-01 03:22:50Z vboxsync $ */
2/** @file
3 * Internal helpers/structures for guest control functionality.
4 */
5
6/*
7 * Copyright (C) 2011-2019 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#ifndef MAIN_INCLUDED_GuestCtrlImplPrivate_h
19#define MAIN_INCLUDED_GuestCtrlImplPrivate_h
20#ifndef RT_WITHOUT_PRAGMA_ONCE
21# pragma once
22#endif
23
24#include "ConsoleImpl.h"
25#include "Global.h"
26
27#include <iprt/asm.h>
28#include <iprt/env.h>
29#include <iprt/semaphore.h>
30#include <iprt/cpp/utils.h>
31
32#include <VBox/com/com.h>
33#include <VBox/com/ErrorInfo.h>
34#include <VBox/com/string.h>
35#include <VBox/com/VirtualBox.h>
36#include <VBox/err.h> /* VERR_GSTCTL_GUEST_ERROR */
37
38#include <map>
39#include <vector>
40
41using namespace com;
42
43#ifdef VBOX_WITH_GUEST_CONTROL
44# include <VBox/GuestHost/GuestControl.h>
45# include <VBox/HostServices/GuestControlSvc.h>
46using namespace guestControl;
47#endif
48
49/** Vector holding a process' CPU affinity. */
50typedef std::vector <LONG> ProcessAffinity;
51/** Vector holding process startup arguments. */
52typedef std::vector <Utf8Str> ProcessArguments;
53
54class GuestProcessStreamBlock;
55class GuestSession;
56
57
58/**
59 * Simple structure mantaining guest credentials.
60 */
61struct GuestCredentials
62{
63 Utf8Str mUser;
64 Utf8Str mPassword;
65 Utf8Str mDomain;
66};
67
68
69
70/**
71 * Wrapper around the RTEnv API, unusable base class.
72 *
73 * @remarks Feel free to elevate this class to iprt/cpp/env.h as RTCEnv.
74 */
75class GuestEnvironmentBase
76{
77public:
78 /**
79 * Default constructor.
80 *
81 * The user must invoke one of the init methods before using the object.
82 */
83 GuestEnvironmentBase(void)
84 : m_hEnv(NIL_RTENV)
85 , m_cRefs(1)
86 { }
87
88 /**
89 * Destructor.
90 */
91 virtual ~GuestEnvironmentBase(void)
92 {
93 Assert(m_cRefs <= 1);
94 int rc = RTEnvDestroy(m_hEnv); AssertRC(rc);
95 m_hEnv = NIL_RTENV;
96 }
97
98 /**
99 * Retains a reference to this object.
100 * @returns New reference count.
101 * @remarks Sharing an object is currently only safe if no changes are made to
102 * it because RTENV does not yet implement any locking. For the only
103 * purpose we need this, implementing IGuestProcess::environment by
104 * using IGuestSession::environmentBase, that's fine as the session
105 * base environment is immutable.
106 */
107 uint32_t retain(void)
108 {
109 uint32_t cRefs = ASMAtomicIncU32(&m_cRefs);
110 Assert(cRefs > 1); Assert(cRefs < _1M);
111 return cRefs;
112
113 }
114 /** Useful shortcut. */
115 uint32_t retainConst(void) const { return unconst(this)->retain(); }
116
117 /**
118 * Releases a reference to this object, deleting the object when reaching zero.
119 * @returns New reference count.
120 */
121 uint32_t release(void)
122 {
123 uint32_t cRefs = ASMAtomicDecU32(&m_cRefs);
124 Assert(cRefs < _1M);
125 if (cRefs == 0)
126 delete this;
127 return cRefs;
128 }
129
130 /** Useful shortcut. */
131 uint32_t releaseConst(void) const { return unconst(this)->retain(); }
132
133 /**
134 * Checks if the environment has been successfully initialized or not.
135 *
136 * @returns @c true if initialized, @c false if not.
137 */
138 bool isInitialized(void) const
139 {
140 return m_hEnv != NIL_RTENV;
141 }
142
143 /**
144 * Returns the variable count.
145 * @return Number of variables.
146 * @sa RTEnvCountEx
147 */
148 uint32_t count(void) const
149 {
150 return RTEnvCountEx(m_hEnv);
151 }
152
153 /**
154 * Deletes the environment change record entirely.
155 *
156 * The count() method will return zero after this call.
157 *
158 * @sa RTEnvReset
159 */
160 void reset(void)
161 {
162 int rc = RTEnvReset(m_hEnv);
163 AssertRC(rc);
164 }
165
166 /**
167 * Exports the environment change block as an array of putenv style strings.
168 *
169 *
170 * @returns VINF_SUCCESS or VERR_NO_MEMORY.
171 * @param pArray The output array.
172 */
173 int queryPutEnvArray(std::vector<com::Utf8Str> *pArray) const
174 {
175 uint32_t cVars = RTEnvCountEx(m_hEnv);
176 try
177 {
178 pArray->resize(cVars);
179 for (uint32_t iVar = 0; iVar < cVars; iVar++)
180 {
181 const char *psz = RTEnvGetByIndexRawEx(m_hEnv, iVar);
182 AssertReturn(psz, VERR_INTERNAL_ERROR_3); /* someone is racing us! */
183 (*pArray)[iVar] = psz;
184 }
185 return VINF_SUCCESS;
186 }
187 catch (std::bad_alloc &)
188 {
189 return VERR_NO_MEMORY;
190 }
191 }
192
193 /**
194 * Applies an array of putenv style strings.
195 *
196 * @returns IPRT status code.
197 * @param rArray The array with the putenv style strings.
198 * @sa RTEnvPutEnvEx
199 */
200 int applyPutEnvArray(const std::vector<com::Utf8Str> &rArray)
201 {
202 size_t cArray = rArray.size();
203 for (size_t i = 0; i < cArray; i++)
204 {
205 int rc = RTEnvPutEx(m_hEnv, rArray[i].c_str());
206 if (RT_FAILURE(rc))
207 return rc;
208 }
209 return VINF_SUCCESS;
210 }
211
212 /**
213 * Applies the changes from another environment to this.
214 *
215 * @returns IPRT status code.
216 * @param rChanges Reference to an environment which variables will be
217 * imported and, if it's a change record, schedule
218 * variable unsets will be applied.
219 * @sa RTEnvApplyChanges
220 */
221 int applyChanges(const GuestEnvironmentBase &rChanges)
222 {
223 return RTEnvApplyChanges(m_hEnv, rChanges.m_hEnv);
224 }
225
226
227 /**
228 * See RTEnvQueryUtf8Block for details.
229 * @returns IPRT status code.
230 * @param ppszzBlock Where to return the block pointer.
231 * @param pcbBlock Where to optionally return the block size.
232 * @sa RTEnvQueryUtf8Block
233 */
234 int queryUtf8Block(char **ppszzBlock, size_t *pcbBlock)
235 {
236 return RTEnvQueryUtf8Block(m_hEnv, true /*fSorted*/, ppszzBlock, pcbBlock);
237 }
238
239 /**
240 * Frees what queryUtf8Block returned, NULL ignored.
241 * @sa RTEnvFreeUtf8Block
242 */
243 static void freeUtf8Block(char *pszzBlock)
244 {
245 return RTEnvFreeUtf8Block(pszzBlock);
246 }
247
248 /**
249 * Applies a block on the format returned by queryUtf8Block.
250 *
251 * @returns IPRT status code.
252 * @param pszzBlock Pointer to the block.
253 * @param cbBlock The size of the block.
254 * @param fNoEqualMeansUnset Whether the lack of a '=' (equal) sign in a
255 * string means it should be unset (@c true), or if
256 * it means the variable should be defined with an
257 * empty value (@c false, the default).
258 * @todo move this to RTEnv!
259 */
260 int copyUtf8Block(const char *pszzBlock, size_t cbBlock, bool fNoEqualMeansUnset = false)
261 {
262 int rc = VINF_SUCCESS;
263 while (cbBlock > 0 && *pszzBlock != '\0')
264 {
265 const char *pszEnd = (const char *)memchr(pszzBlock, '\0', cbBlock);
266 if (!pszEnd)
267 return VERR_BUFFER_UNDERFLOW;
268 int rc2;
269 if (fNoEqualMeansUnset || strchr(pszzBlock, '='))
270 rc2 = RTEnvPutEx(m_hEnv, pszzBlock);
271 else
272 rc2 = RTEnvSetEx(m_hEnv, pszzBlock, "");
273 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
274 rc = rc2;
275
276 /* Advance. */
277 cbBlock -= pszEnd - pszzBlock;
278 if (cbBlock < 2)
279 return VERR_BUFFER_UNDERFLOW;
280 cbBlock--;
281 pszzBlock = pszEnd + 1;
282 }
283
284 /* The remainder must be zero padded. */
285 if (RT_SUCCESS(rc))
286 {
287 if (ASMMemIsZero(pszzBlock, cbBlock))
288 return VINF_SUCCESS;
289 return VERR_TOO_MUCH_DATA;
290 }
291 return rc;
292 }
293
294
295 /**
296 * Get an environment variable.
297 *
298 * @returns IPRT status code.
299 * @param rName The variable name.
300 * @param pValue Where to return the value.
301 * @sa RTEnvGetEx
302 */
303 int getVariable(const com::Utf8Str &rName, com::Utf8Str *pValue) const
304 {
305 size_t cchNeeded;
306 int rc = RTEnvGetEx(m_hEnv, rName.c_str(), NULL, 0, &cchNeeded);
307 if ( RT_SUCCESS(rc)
308 || rc == VERR_BUFFER_OVERFLOW)
309 {
310 try
311 {
312 pValue->reserve(cchNeeded + 1);
313 rc = RTEnvGetEx(m_hEnv, rName.c_str(), pValue->mutableRaw(), pValue->capacity(), NULL);
314 pValue->jolt();
315 }
316 catch (std::bad_alloc &)
317 {
318 rc = VERR_NO_STR_MEMORY;
319 }
320 }
321 return rc;
322 }
323
324 /**
325 * Checks if the given variable exists.
326 *
327 * @returns @c true if it exists, @c false if not or if it's an scheduled unset
328 * in a environment change record.
329 * @param rName The variable name.
330 * @sa RTEnvExistEx
331 */
332 bool doesVariableExist(const com::Utf8Str &rName) const
333 {
334 return RTEnvExistEx(m_hEnv, rName.c_str());
335 }
336
337 /**
338 * Set an environment variable.
339 *
340 * @returns IPRT status code.
341 * @param rName The variable name.
342 * @param rValue The value of the variable.
343 * @sa RTEnvSetEx
344 */
345 int setVariable(const com::Utf8Str &rName, const com::Utf8Str &rValue)
346 {
347 return RTEnvSetEx(m_hEnv, rName.c_str(), rValue.c_str());
348 }
349
350 /**
351 * Unset an environment variable.
352 *
353 * @returns IPRT status code.
354 * @param rName The variable name.
355 * @sa RTEnvUnsetEx
356 */
357 int unsetVariable(const com::Utf8Str &rName)
358 {
359 return RTEnvUnsetEx(m_hEnv, rName.c_str());
360 }
361
362protected:
363 /**
364 * Copy constructor.
365 * @throws HRESULT
366 */
367 GuestEnvironmentBase(const GuestEnvironmentBase &rThat, bool fChangeRecord)
368 : m_hEnv(NIL_RTENV)
369 , m_cRefs(1)
370 {
371 int rc = cloneCommon(rThat, fChangeRecord);
372 if (RT_FAILURE(rc))
373 throw (Global::vboxStatusCodeToCOM(rc));
374 }
375
376 /**
377 * Common clone/copy method with type conversion abilities.
378 *
379 * @returns IPRT status code.
380 * @param rThat The object to clone.
381 * @param fChangeRecord Whether the this instance is a change record (true)
382 * or normal (false) environment.
383 */
384 int cloneCommon(const GuestEnvironmentBase &rThat, bool fChangeRecord)
385 {
386 int rc = VINF_SUCCESS;
387 RTENV hNewEnv = NIL_RTENV;
388 if (rThat.m_hEnv != NIL_RTENV)
389 {
390 /*
391 * Clone it.
392 */
393 if (RTEnvIsChangeRecord(rThat.m_hEnv) == fChangeRecord)
394 rc = RTEnvClone(&hNewEnv, rThat.m_hEnv);
395 else
396 {
397 /* Need to type convert it. */
398 if (fChangeRecord)
399 rc = RTEnvCreateChangeRecord(&hNewEnv);
400 else
401 rc = RTEnvCreate(&hNewEnv);
402 if (RT_SUCCESS(rc))
403 {
404 rc = RTEnvApplyChanges(hNewEnv, rThat.m_hEnv);
405 if (RT_FAILURE(rc))
406 RTEnvDestroy(hNewEnv);
407 }
408 }
409 }
410 else
411 {
412 /*
413 * Create an empty one so the object works smoothly.
414 * (Relevant for GuestProcessStartupInfo and internal commands.)
415 */
416 if (fChangeRecord)
417 rc = RTEnvCreateChangeRecord(&hNewEnv);
418 else
419 rc = RTEnvCreate(&hNewEnv);
420 }
421 if (RT_SUCCESS(rc))
422 {
423 RTEnvDestroy(m_hEnv);
424 m_hEnv = hNewEnv;
425 }
426 return rc;
427 }
428
429
430 /** The environment change record. */
431 RTENV m_hEnv;
432 /** Reference counter. */
433 uint32_t volatile m_cRefs;
434};
435
436class GuestEnvironmentChanges;
437
438
439/**
440 * Wrapper around the RTEnv API for a normal environment.
441 */
442class GuestEnvironment : public GuestEnvironmentBase
443{
444public:
445 /**
446 * Default constructor.
447 *
448 * The user must invoke one of the init methods before using the object.
449 */
450 GuestEnvironment(void)
451 : GuestEnvironmentBase()
452 { }
453
454 /**
455 * Copy operator.
456 * @param rThat The object to copy.
457 * @throws HRESULT
458 */
459 GuestEnvironment(const GuestEnvironment &rThat)
460 : GuestEnvironmentBase(rThat, false /*fChangeRecord*/)
461 { }
462
463 /**
464 * Copy operator.
465 * @param rThat The object to copy.
466 * @throws HRESULT
467 */
468 GuestEnvironment(const GuestEnvironmentBase &rThat)
469 : GuestEnvironmentBase(rThat, false /*fChangeRecord*/)
470 { }
471
472 /**
473 * Initialize this as a normal environment block.
474 * @returns IPRT status code.
475 */
476 int initNormal(void)
477 {
478 AssertReturn(m_hEnv == NIL_RTENV, VERR_WRONG_ORDER);
479 return RTEnvCreate(&m_hEnv);
480 }
481
482 /**
483 * Replaces this environemnt with that in @a rThat.
484 *
485 * @returns IPRT status code
486 * @param rThat The environment to copy. If it's a different type
487 * we'll convert the data to a normal environment block.
488 */
489 int copy(const GuestEnvironmentBase &rThat)
490 {
491 return cloneCommon(rThat, false /*fChangeRecord*/);
492 }
493
494 /**
495 * @copydoc copy()
496 */
497 GuestEnvironment &operator=(const GuestEnvironmentBase &rThat)
498 {
499 int rc = cloneCommon(rThat, true /*fChangeRecord*/);
500 if (RT_FAILURE(rc))
501 throw (Global::vboxStatusCodeToCOM(rc));
502 return *this;
503 }
504
505 /** @copydoc copy() */
506 GuestEnvironment &operator=(const GuestEnvironment &rThat)
507 { return operator=((const GuestEnvironmentBase &)rThat); }
508
509 /** @copydoc copy() */
510 GuestEnvironment &operator=(const GuestEnvironmentChanges &rThat)
511 { return operator=((const GuestEnvironmentBase &)rThat); }
512
513};
514
515
516/**
517 * Wrapper around the RTEnv API for a environment change record.
518 *
519 * This class is used as a record of changes to be applied to a different
520 * environment block (in VBoxService before launching a new process).
521 */
522class GuestEnvironmentChanges : public GuestEnvironmentBase
523{
524public:
525 /**
526 * Default constructor.
527 *
528 * The user must invoke one of the init methods before using the object.
529 */
530 GuestEnvironmentChanges(void)
531 : GuestEnvironmentBase()
532 { }
533
534 /**
535 * Copy operator.
536 * @param rThat The object to copy.
537 * @throws HRESULT
538 */
539 GuestEnvironmentChanges(const GuestEnvironmentChanges &rThat)
540 : GuestEnvironmentBase(rThat, true /*fChangeRecord*/)
541 { }
542
543 /**
544 * Copy operator.
545 * @param rThat The object to copy.
546 * @throws HRESULT
547 */
548 GuestEnvironmentChanges(const GuestEnvironmentBase &rThat)
549 : GuestEnvironmentBase(rThat, true /*fChangeRecord*/)
550 { }
551
552 /**
553 * Initialize this as a environment change record.
554 * @returns IPRT status code.
555 */
556 int initChangeRecord(void)
557 {
558 AssertReturn(m_hEnv == NIL_RTENV, VERR_WRONG_ORDER);
559 return RTEnvCreateChangeRecord(&m_hEnv);
560 }
561
562 /**
563 * Replaces this environemnt with that in @a rThat.
564 *
565 * @returns IPRT status code
566 * @param rThat The environment to copy. If it's a different type
567 * we'll convert the data to a set of changes.
568 */
569 int copy(const GuestEnvironmentBase &rThat)
570 {
571 return cloneCommon(rThat, true /*fChangeRecord*/);
572 }
573
574 /**
575 * @copydoc copy()
576 */
577 GuestEnvironmentChanges &operator=(const GuestEnvironmentBase &rThat)
578 {
579 int rc = cloneCommon(rThat, true /*fChangeRecord*/);
580 if (RT_FAILURE(rc))
581 throw (Global::vboxStatusCodeToCOM(rc));
582 return *this;
583 }
584
585 /** @copydoc copy() */
586 GuestEnvironmentChanges &operator=(const GuestEnvironmentChanges &rThat)
587 { return operator=((const GuestEnvironmentBase &)rThat); }
588
589 /** @copydoc copy() */
590 GuestEnvironmentChanges &operator=(const GuestEnvironment &rThat)
591 { return operator=((const GuestEnvironmentBase &)rThat); }
592};
593
594
595/**
596 * Structure for keeping all the relevant guest directory
597 * information around.
598 */
599struct GuestDirectoryOpenInfo
600{
601 /** The directory path. */
602 Utf8Str mPath;
603 /** Then open filter. */
604 Utf8Str mFilter;
605 /** Opening flags. */
606 uint32_t mFlags;
607};
608
609
610/**
611 * Structure for keeping all the relevant guest file
612 * information around.
613 */
614struct GuestFileOpenInfo
615{
616 /** The filename. */
617 Utf8Str mFilename;
618 /** The file access mode. */
619 FileAccessMode_T mAccessMode;
620 /** The file open action. */
621 FileOpenAction_T mOpenAction;
622 /** The file sharing mode. */
623 FileSharingMode_T mSharingMode;
624 /** Octal creation mode. */
625 uint32_t mCreationMode;
626 /** Extended open flags (currently none defined). */
627 uint32_t mfOpenEx;
628 /** Initial file offset. */
629 uint64_t muOffset; /** @todo Remove this in the next protocol version. */
630};
631
632
633/**
634 * Structure representing information of a
635 * file system object.
636 */
637struct GuestFsObjData
638{
639 /** @name Helper functions to extract the data from a certin VBoxService tool's guest stream block.
640 * @{ */
641 int FromLs(const GuestProcessStreamBlock &strmBlk, bool fLong);
642 int FromStat(const GuestProcessStreamBlock &strmBlk);
643 int FromMkTemp(const GuestProcessStreamBlock &strmBlk);
644 /** @} */
645
646 /** @name Static helper functions to work with time from stream block keys.
647 * @{ */
648 static PRTTIMESPEC TimeSpecFromKey(const GuestProcessStreamBlock &strmBlk, const Utf8Str &strKey, PRTTIMESPEC pTimeSpec);
649 static int64_t UnixEpochNsFromKey(const GuestProcessStreamBlock &strmBlk, const Utf8Str &strKey);
650 /** @} */
651
652 /** @name helper functions to work with IPRT stuff.
653 * @{ */
654 RTFMODE GetFileMode(void) const;
655 /** @} */
656
657 Utf8Str mName;
658 FsObjType_T mType;
659 Utf8Str mFileAttrs;
660 int64_t mObjectSize;
661 int64_t mAllocatedSize;
662 int64_t mAccessTime;
663 int64_t mBirthTime;
664 int64_t mChangeTime;
665 int64_t mModificationTime;
666 Utf8Str mUserName;
667 int32_t mUID;
668 int32_t mGID;
669 Utf8Str mGroupName;
670 Utf8Str mACL;
671 int64_t mNodeID;
672 uint32_t mNodeIDDevice;
673 uint32_t mNumHardLinks;
674 uint32_t mDeviceNumber;
675 uint32_t mGenerationID;
676 uint32_t mUserFlags;
677};
678
679
680/**
681 * Structure for keeping all the relevant guest session
682 * startup parameters around.
683 */
684class GuestSessionStartupInfo
685{
686public:
687
688 GuestSessionStartupInfo(void)
689 : mIsInternal(false /* Non-internal session */),
690 mOpenTimeoutMS(30 * 1000 /* 30s opening timeout */),
691 mOpenFlags(0 /* No opening flags set */) { }
692
693 /** The session's friendly name. Optional. */
694 Utf8Str mName;
695 /** The session's unique ID. Used to encode a context ID. */
696 uint32_t mID;
697 /** Flag indicating if this is an internal session
698 * or not. Internal session are not accessible by
699 * public API clients. */
700 bool mIsInternal;
701 /** Timeout (in ms) used for opening the session. */
702 uint32_t mOpenTimeoutMS;
703 /** Session opening flags. */
704 uint32_t mOpenFlags;
705};
706
707
708/**
709 * Structure for keeping all the relevant guest process
710 * startup parameters around.
711 */
712class GuestProcessStartupInfo
713{
714public:
715
716 GuestProcessStartupInfo(void)
717 : mFlags(ProcessCreateFlag_None),
718 mTimeoutMS(UINT32_MAX /* No timeout by default */),
719 mPriority(ProcessPriority_Default) { }
720
721 /** The process' friendly name. */
722 Utf8Str mName;
723 /** The executable. */
724 Utf8Str mExecutable;
725 /** Arguments vector (starting with argument \#0). */
726 ProcessArguments mArguments;
727 /** The process environment change record. */
728 GuestEnvironmentChanges mEnvironmentChanges;
729 /** Process creation flags. */
730 uint32_t mFlags;
731 /** Timeout (in ms) the process is allowed to run.
732 * Specify UINT32_MAX if no timeout (unlimited run time) is given. */
733 ULONG mTimeoutMS;
734 /** Process priority. */
735 ProcessPriority_T mPriority;
736 /** Process affinity. At the moment we
737 * only support 64 VCPUs. API and
738 * guest can do more already! */
739 uint64_t mAffinity;
740};
741
742
743/**
744 * Class representing the "value" side of a "key=value" pair.
745 */
746class GuestProcessStreamValue
747{
748public:
749
750 GuestProcessStreamValue(void) { }
751 GuestProcessStreamValue(const char *pszValue)
752 : mValue(pszValue) {}
753
754 GuestProcessStreamValue(const GuestProcessStreamValue& aThat)
755 : mValue(aThat.mValue) { }
756
757 Utf8Str mValue;
758};
759
760/** Map containing "key=value" pairs of a guest process stream. */
761typedef std::pair< Utf8Str, GuestProcessStreamValue > GuestCtrlStreamPair;
762typedef std::map < Utf8Str, GuestProcessStreamValue > GuestCtrlStreamPairMap;
763typedef std::map < Utf8Str, GuestProcessStreamValue >::iterator GuestCtrlStreamPairMapIter;
764typedef std::map < Utf8Str, GuestProcessStreamValue >::const_iterator GuestCtrlStreamPairMapIterConst;
765
766/**
767 * Class representing a block of stream pairs (key=value). Each block in a raw guest
768 * output stream is separated by "\0\0", each pair is separated by "\0". The overall
769 * end of a guest stream is marked by "\0\0\0\0".
770 */
771class GuestProcessStreamBlock
772{
773public:
774
775 GuestProcessStreamBlock(void);
776
777 virtual ~GuestProcessStreamBlock(void);
778
779public:
780
781 void Clear(void);
782
783#ifdef DEBUG
784 void DumpToLog(void) const;
785#endif
786
787 const char *GetString(const char *pszKey) const;
788 size_t GetCount(void) const;
789 int GetRc(void) const;
790 int GetInt64Ex(const char *pszKey, int64_t *piVal) const;
791 int64_t GetInt64(const char *pszKey) const;
792 int GetUInt32Ex(const char *pszKey, uint32_t *puVal) const;
793 uint32_t GetUInt32(const char *pszKey, uint32_t uDefault = 0) const;
794 int32_t GetInt32(const char *pszKey, int32_t iDefault = 0) const;
795
796 bool IsEmpty(void) { return mPairs.empty(); }
797
798 int SetValue(const char *pszKey, const char *pszValue);
799
800protected:
801
802 GuestCtrlStreamPairMap mPairs;
803};
804
805/** Vector containing multiple allocated stream pair objects. */
806typedef std::vector< GuestProcessStreamBlock > GuestCtrlStreamObjects;
807typedef std::vector< GuestProcessStreamBlock >::iterator GuestCtrlStreamObjectsIter;
808typedef std::vector< GuestProcessStreamBlock >::const_iterator GuestCtrlStreamObjectsIterConst;
809
810/**
811 * Class for parsing machine-readable guest process output by VBoxService'
812 * toolbox commands ("vbox_ls", "vbox_stat" etc), aka "guest stream".
813 */
814class GuestProcessStream
815{
816
817public:
818
819 GuestProcessStream();
820
821 virtual ~GuestProcessStream();
822
823public:
824
825 int AddData(const BYTE *pbData, size_t cbData);
826
827 void Destroy();
828
829#ifdef DEBUG
830 void Dump(const char *pszFile);
831#endif
832
833 size_t GetOffset() { return m_offBuffer; }
834
835 size_t GetSize() { return m_cbUsed; }
836
837 int ParseBlock(GuestProcessStreamBlock &streamBlock);
838
839protected:
840
841 /** Currently allocated size of internal stream buffer. */
842 size_t m_cbAllocated;
843 /** Currently used size at m_offBuffer. */
844 size_t m_cbUsed;
845 /** Current byte offset within the internal stream buffer. */
846 size_t m_offBuffer;
847 /** Internal stream buffer. */
848 BYTE *m_pbBuffer;
849};
850
851class Guest;
852class Progress;
853
854class GuestTask
855{
856
857public:
858
859 enum TaskType
860 {
861 /** Copies a file from host to the guest. */
862 TaskType_CopyFileToGuest = 50,
863 /** Copies a file from guest to the host. */
864 TaskType_CopyFileFromGuest = 55,
865 /** Update Guest Additions by directly copying the required installer
866 * off the .ISO file, transfer it to the guest and execute the installer
867 * with system privileges. */
868 TaskType_UpdateGuestAdditions = 100
869 };
870
871 GuestTask(TaskType aTaskType, Guest *aThat, Progress *aProgress);
872
873 virtual ~GuestTask();
874
875 int startThread();
876
877 static int taskThread(RTTHREAD aThread, void *pvUser);
878 static int uploadProgress(unsigned uPercent, void *pvUser);
879 static HRESULT setProgressSuccess(ComObjPtr<Progress> pProgress);
880 static HRESULT setProgressErrorMsg(HRESULT hr,
881 ComObjPtr<Progress> pProgress, const char * pszText, ...);
882 static HRESULT setProgressErrorParent(HRESULT hr,
883 ComObjPtr<Progress> pProgress, ComObjPtr<Guest> pGuest);
884
885 TaskType taskType;
886 ComObjPtr<Guest> pGuest;
887 ComObjPtr<Progress> pProgress;
888 HRESULT rc;
889
890 /* Task data. */
891 Utf8Str strSource;
892 Utf8Str strDest;
893 Utf8Str strUserName;
894 Utf8Str strPassword;
895 ULONG uFlags;
896};
897
898class GuestWaitEventPayload
899{
900
901public:
902
903 GuestWaitEventPayload(void)
904 : uType(0),
905 cbData(0),
906 pvData(NULL) { }
907
908 GuestWaitEventPayload(uint32_t uTypePayload,
909 const void *pvPayload, uint32_t cbPayload)
910 : uType(0),
911 cbData(0),
912 pvData(NULL)
913 {
914 int rc = copyFrom(uTypePayload, pvPayload, cbPayload);
915 if (RT_FAILURE(rc))
916 throw rc;
917 }
918
919 virtual ~GuestWaitEventPayload(void)
920 {
921 Clear();
922 }
923
924 GuestWaitEventPayload& operator=(const GuestWaitEventPayload &that)
925 {
926 CopyFromDeep(that);
927 return *this;
928 }
929
930public:
931
932 void Clear(void)
933 {
934 if (pvData)
935 {
936 Assert(cbData);
937 RTMemFree(pvData);
938 cbData = 0;
939 pvData = NULL;
940 }
941 uType = 0;
942 }
943
944 int CopyFromDeep(const GuestWaitEventPayload &payload)
945 {
946 return copyFrom(payload.uType, payload.pvData, payload.cbData);
947 }
948
949 const void* Raw(void) const { return pvData; }
950
951 size_t Size(void) const { return cbData; }
952
953 uint32_t Type(void) const { return uType; }
954
955 void* MutableRaw(void) { return pvData; }
956
957 Utf8Str ToString(void)
958 {
959 const char *pszStr = (const char *)pvData;
960 size_t cbStr = cbData;
961
962 if (RT_FAILURE(RTStrValidateEncodingEx(pszStr, cbStr,
963 RTSTR_VALIDATE_ENCODING_ZERO_TERMINATED | RTSTR_VALIDATE_ENCODING_EXACT_LENGTH)))
964 {
965 AssertFailed();
966 return "";
967 }
968
969 return Utf8Str(pszStr, cbStr);
970 }
971
972protected:
973
974 int copyFrom(uint32_t uTypePayload, const void *pvPayload, uint32_t cbPayload)
975 {
976 if (cbPayload > _64K) /* Paranoia. */
977 return VERR_TOO_MUCH_DATA;
978
979 Clear();
980
981 int rc = VINF_SUCCESS;
982
983 if (cbPayload)
984 {
985 pvData = RTMemAlloc(cbPayload);
986 if (pvData)
987 {
988 uType = uTypePayload;
989
990 memcpy(pvData, pvPayload, cbPayload);
991 cbData = cbPayload;
992 }
993 else
994 rc = VERR_NO_MEMORY;
995 }
996 else
997 {
998 uType = uTypePayload;
999
1000 pvData = NULL;
1001 cbData = 0;
1002 }
1003
1004 return rc;
1005 }
1006
1007protected:
1008
1009 /** Type of payload. */
1010 uint32_t uType;
1011 /** Size (in bytes) of payload. */
1012 uint32_t cbData;
1013 /** Pointer to actual payload data. */
1014 void *pvData;
1015};
1016
1017class GuestWaitEventBase
1018{
1019
1020protected:
1021
1022 GuestWaitEventBase(void);
1023 virtual ~GuestWaitEventBase(void);
1024
1025public:
1026
1027 uint32_t ContextID(void) { return mCID; };
1028 int GuestResult(void) { return mGuestRc; }
1029 int Result(void) { return mRc; }
1030 GuestWaitEventPayload & Payload(void) { return mPayload; }
1031 int SignalInternal(int rc, int guestRc, const GuestWaitEventPayload *pPayload);
1032 int Wait(RTMSINTERVAL uTimeoutMS);
1033
1034protected:
1035
1036 int Init(uint32_t uCID);
1037
1038protected:
1039
1040 /* Shutdown indicator. */
1041 bool mfAborted;
1042 /* Associated context ID (CID). */
1043 uint32_t mCID;
1044 /** The event semaphore for triggering
1045 * the actual event. */
1046 RTSEMEVENT mEventSem;
1047 /** The event's overall result. If
1048 * set to VERR_GSTCTL_GUEST_ERROR,
1049 * mGuestRc will contain the actual
1050 * error code from the guest side. */
1051 int mRc;
1052 /** The event'S overall result from the
1053 * guest side. If used, mRc must be
1054 * set to VERR_GSTCTL_GUEST_ERROR. */
1055 int mGuestRc;
1056 /** The event's payload data. Optional. */
1057 GuestWaitEventPayload mPayload;
1058};
1059
1060/** List of public guest event types. */
1061typedef std::list < VBoxEventType_T > GuestEventTypes;
1062
1063class GuestWaitEvent : public GuestWaitEventBase
1064{
1065
1066public:
1067
1068 GuestWaitEvent(uint32_t uCID);
1069 GuestWaitEvent(uint32_t uCID, const GuestEventTypes &lstEvents);
1070 virtual ~GuestWaitEvent(void);
1071
1072public:
1073
1074 int Cancel(void);
1075 const ComPtr<IEvent> Event(void) { return mEvent; }
1076 bool HasGuestError(void) const { return mRc == VERR_GSTCTL_GUEST_ERROR; }
1077 int GetGuestError(void) const { return mGuestRc; }
1078 int SignalExternal(IEvent *pEvent);
1079 const GuestEventTypes &Types(void) { return mEventTypes; }
1080 size_t TypeCount(void) { return mEventTypes.size(); }
1081
1082protected:
1083
1084 int Init(uint32_t uCID);
1085
1086protected:
1087
1088 /** List of public event types this event should
1089 * be signalled on. Optional. */
1090 GuestEventTypes mEventTypes;
1091 /** Pointer to the actual public event, if any. */
1092 ComPtr<IEvent> mEvent;
1093};
1094/** Map of pointers to guest events. The primary key
1095 * contains the context ID. */
1096typedef std::map < uint32_t, GuestWaitEvent* > GuestWaitEvents;
1097/** Map of wait events per public guest event. Nice for
1098 * faster lookups when signalling a whole event group. */
1099typedef std::map < VBoxEventType_T, GuestWaitEvents > GuestEventGroup;
1100
1101class GuestBase
1102{
1103
1104public:
1105
1106 GuestBase(void);
1107 virtual ~GuestBase(void);
1108
1109public:
1110
1111 /** Signals a wait event using a public guest event; also used for
1112 * for external event listeners. */
1113 int signalWaitEvent(VBoxEventType_T aType, IEvent *aEvent);
1114 /** Signals a wait event using a guest rc. */
1115 int signalWaitEventInternal(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, int guestRc, const GuestWaitEventPayload *pPayload);
1116 /** Signals a wait event without letting public guest events know,
1117 * extended director's cut version. */
1118 int signalWaitEventInternalEx(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, int rc, int guestRc, const GuestWaitEventPayload *pPayload);
1119
1120public:
1121
1122 int baseInit(void);
1123 void baseUninit(void);
1124 int cancelWaitEvents(void);
1125 int dispatchGeneric(PVBOXGUESTCTRLHOSTCBCTX pCtxCb, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb);
1126 int generateContextID(uint32_t uSessionID, uint32_t uObjectID, uint32_t *puContextID);
1127 int registerWaitEvent(uint32_t uSessionID, uint32_t uObjectID, GuestWaitEvent **ppEvent);
1128 int registerWaitEventEx(uint32_t uSessionID, uint32_t uObjectID, const GuestEventTypes &lstEvents, GuestWaitEvent **ppEvent);
1129 int unregisterWaitEvent(GuestWaitEvent *pEvent);
1130 int waitForEvent(GuestWaitEvent *pEvent, uint32_t uTimeoutMS, VBoxEventType_T *pType, IEvent **ppEvent);
1131
1132public:
1133
1134 static FsObjType_T fileModeToFsObjType(RTFMODE fMode);
1135
1136protected:
1137
1138 /** Pointer to the console object. Needed
1139 * for HGCM (VMMDev) communication. */
1140 Console *mConsole;
1141 /** The next context ID counter component for this object. */
1142 uint32_t mNextContextID;
1143 /** Local listener for handling the waiting events
1144 * internally. */
1145 ComPtr<IEventListener> mLocalListener;
1146 /** Critical section for wait events access. */
1147 RTCRITSECT mWaitEventCritSect;
1148 /** Map of registered wait events per event group. */
1149 GuestEventGroup mWaitEventGroups;
1150 /** Map of registered wait events. */
1151 GuestWaitEvents mWaitEvents;
1152};
1153
1154/**
1155 * Virtual class (interface) for guest objects (processes, files, ...) --
1156 * contains all per-object callback management.
1157 */
1158class GuestObject : public GuestBase
1159{
1160
1161public:
1162
1163 GuestObject(void);
1164 virtual ~GuestObject(void);
1165
1166public:
1167
1168 ULONG getObjectID(void) { return mObjectID; }
1169
1170protected:
1171
1172 virtual int i_onRemove(void) = 0;
1173
1174 /** Callback dispatcher -- must be implemented by the actual object. */
1175 virtual int i_callbackDispatcher(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb) = 0;
1176
1177protected:
1178
1179 int bindToSession(Console *pConsole, GuestSession *pSession, uint32_t uObjectID);
1180 int registerWaitEvent(const GuestEventTypes &lstEvents, GuestWaitEvent **ppEvent);
1181 int sendCommand(uint32_t uFunction, uint32_t cParms, PVBOXHGCMSVCPARM paParms);
1182
1183protected:
1184
1185 /** @name Common parameters for all derived objects. They have their own
1186 * mData structure to keep their specific data around.
1187 * @{ */
1188 /** Pointer to parent session. Per definition
1189 * this objects *always* lives shorter than the
1190 * parent.
1191 * @todo r=bird: When wanting to use mSession in the
1192 * IGuestProcess::getEnvironment() implementation I wanted to access
1193 * GuestSession::mData::mpBaseEnvironment. Seeing the comment in
1194 * GuestProcess::terminate() saying:
1195 * "Now only API clients still can hold references to it."
1196 * and recalling seeing similar things in VirtualBox.xidl or some such place,
1197 * I'm wondering how this "per definition" behavior is enforced. Is there any
1198 * GuestProcess:uninit() call or similar magic that invalidates objects that
1199 * GuestSession loses track of in place like GuestProcess::terminate() that I've
1200 * failed to spot?
1201 *
1202 * Please enlighten me.
1203 */
1204 GuestSession *mSession;
1205 /** The object ID -- must be unique for each guest
1206 * object and is encoded into the context ID. Must
1207 * be set manually when initializing the object.
1208 *
1209 * For guest processes this is the internal PID,
1210 * for guest files this is the internal file ID. */
1211 uint32_t mObjectID;
1212 /** @} */
1213};
1214#endif /* !MAIN_INCLUDED_GuestCtrlImplPrivate_h */
1215
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