VirtualBox

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

Last change on this file since 56720 was 56681, checked in by vboxsync, 10 years ago

GuestCtrlImplPrivate.h: Deal with internal stuff (like mkdir) having m_hEnv = NIL.

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