VirtualBox

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

Last change on this file since 59989 was 59747, checked in by vboxsync, 9 years ago

iprt/asm.h: Cleaned up the ASMMemIsAll8/U32 mess and implmeneted the former in assembly. (Found inverted usage due to bad naming in copyUtf8Block, but it is fortunately an unused method.) Replaces the complicated ASMBitFirstSet based scanning in RTSgBufIsZero with a simple call to the new ASMMemIsZero function.

  • 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 59747 2016-02-19 23:18:18Z 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 (ASMMemIsZero(pszzBlock, cbBlock))
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