VirtualBox

source: vbox/trunk/src/VBox/Main/src-all/ExtPackManagerImpl.cpp@ 73003

Last change on this file since 73003 was 73003, checked in by vboxsync, 7 years ago

Main: Use setErrorBoth when we've got a VBox status code handy. (The COM status codes aren't too specfic and this may help us decode error messages and provide an alternative to strstr for API clients. setErrorBoth isn't new, btw.)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 109.4 KB
Line 
1/* $Id: ExtPackManagerImpl.cpp 73003 2018-07-09 11:09:32Z vboxsync $ */
2/** @file
3 * VirtualBox Main - interface for Extension Packs, VBoxSVC & VBoxC.
4 */
5
6/*
7 * Copyright (C) 2010-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include "ExtPackManagerImpl.h"
23#include "ExtPackUtil.h"
24#include "ThreadTask.h"
25
26#include <iprt/buildconfig.h>
27#include <iprt/ctype.h>
28#include <iprt/dir.h>
29#include <iprt/env.h>
30#include <iprt/file.h>
31#include <iprt/ldr.h>
32#include <iprt/manifest.h>
33#include <iprt/param.h>
34#include <iprt/path.h>
35#include <iprt/pipe.h>
36#include <iprt/process.h>
37#include <iprt/string.h>
38
39#include <VBox/com/array.h>
40#include <VBox/com/ErrorInfo.h>
41#include <VBox/err.h>
42#include <VBox/log.h>
43#include <VBox/sup.h>
44#include <VBox/version.h>
45#include "AutoCaller.h"
46#include "Global.h"
47#include "ProgressImpl.h"
48#ifdef VBOX_COM_INPROC
49# include "ConsoleImpl.h"
50#else
51# include "VirtualBoxImpl.h"
52#endif
53
54
55/*********************************************************************************************************************************
56* Defined Constants And Macros *
57*********************************************************************************************************************************/
58/** @def VBOX_EXTPACK_HELPER_NAME
59 * The name of the utility application we employ to install and uninstall the
60 * extension packs. This is a set-uid-to-root binary on unixy platforms, which
61 * is why it has to be a separate application.
62 */
63#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
64# define VBOX_EXTPACK_HELPER_NAME "VBoxExtPackHelperApp.exe"
65#else
66# define VBOX_EXTPACK_HELPER_NAME "VBoxExtPackHelperApp"
67#endif
68
69
70/*********************************************************************************************************************************
71* Structures and Typedefs *
72*********************************************************************************************************************************/
73struct ExtPackBaseData
74{
75public:
76 /** The extension pack descriptor (loaded from the XML, mostly). */
77 VBOXEXTPACKDESC Desc;
78 /** The file system object info of the XML file.
79 * This is for detecting changes and save time in refresh(). */
80 RTFSOBJINFO ObjInfoDesc;
81 /** Whether it's usable or not. */
82 bool fUsable;
83 /** Why it is unusable. */
84 Utf8Str strWhyUnusable;
85};
86
87#ifndef VBOX_COM_INPROC
88/**
89 * Private extension pack data.
90 */
91struct ExtPackFile::Data : public ExtPackBaseData
92{
93public:
94 /** The path to the tarball. */
95 Utf8Str strExtPackFile;
96 /** The SHA-256 hash of the file (as string). */
97 Utf8Str strDigest;
98 /** The file handle of the extension pack file. */
99 RTFILE hExtPackFile;
100 /** Our manifest for the tarball. */
101 RTMANIFEST hOurManifest;
102 /** Pointer to the extension pack manager. */
103 ComObjPtr<ExtPackManager> ptrExtPackMgr;
104 /** Pointer to the VirtualBox object so we can create a progress object. */
105 VirtualBox *pVirtualBox;
106
107 RTMEMEF_NEW_AND_DELETE_OPERATORS();
108};
109#endif
110
111/**
112 * Private extension pack data.
113 */
114struct ExtPack::Data : public ExtPackBaseData
115{
116public:
117 /** Where the extension pack is located. */
118 Utf8Str strExtPackPath;
119 /** The file system object info of the extension pack directory.
120 * This is for detecting changes and save time in refresh(). */
121 RTFSOBJINFO ObjInfoExtPack;
122 /** The full path to the main module. */
123 Utf8Str strMainModPath;
124 /** The file system object info of the main module.
125 * This is used to determin whether to bother try reload it. */
126 RTFSOBJINFO ObjInfoMainMod;
127 /** The module handle of the main extension pack module. */
128 RTLDRMOD hMainMod;
129
130 /** The helper callbacks for the extension pack. */
131 VBOXEXTPACKHLP Hlp;
132 /** Pointer back to the extension pack object (for Hlp methods). */
133 ExtPack *pThis;
134#ifndef VBOX_COM_INPROC
135 /** The extension pack main registration structure. */
136 PCVBOXEXTPACKREG pReg;
137#else
138 /** The extension pack main VM registration structure. */
139 PCVBOXEXTPACKVMREG pReg;
140#endif
141 /** The current context. */
142 VBOXEXTPACKCTX enmContext;
143 /** Set if we've made the pfnVirtualBoxReady or pfnConsoleReady call. */
144 bool fMadeReadyCall;
145
146 RTMEMEF_NEW_AND_DELETE_OPERATORS();
147};
148
149/** List of extension packs. */
150typedef std::list< ComObjPtr<ExtPack> > ExtPackList;
151
152/**
153 * Private extension pack manager data.
154 */
155struct ExtPackManager::Data
156{
157 /** The directory where the extension packs are installed. */
158 Utf8Str strBaseDir;
159 /** The directory where the certificates this installation recognizes are
160 * stored. */
161 Utf8Str strCertificatDirPath;
162 /** The list of installed extension packs. */
163 ExtPackList llInstalledExtPacks;
164#ifndef VBOX_COM_INPROC
165 /** Pointer to the VirtualBox object, our parent. */
166 VirtualBox *pVirtualBox;
167#endif
168 /** The current context. */
169 VBOXEXTPACKCTX enmContext;
170
171 RTMEMEF_NEW_AND_DELETE_OPERATORS();
172};
173
174#ifndef VBOX_COM_INPROC
175
176/**
177 * Extension pack installation job.
178 */
179class ExtPackInstallTask : public ThreadTask
180{
181public:
182 explicit ExtPackInstallTask() : ThreadTask("ExtPackInst") { }
183 ~ExtPackInstallTask() { }
184
185 void handler()
186 {
187 HRESULT hrc = ptrExtPackMgr->i_doInstall(ptrExtPackFile, fReplace, &strDisplayInfo);
188 ptrProgress->i_notifyComplete(hrc);
189 }
190
191 HRESULT Init(const ComPtr<ExtPackFile> &a_strExtPackFile, bool a_fReplace,
192 const Utf8Str &strDispInfo, const ComPtr<ExtPackManager> &a_ptrExtPackMgr)
193 {
194 ptrExtPackFile = a_strExtPackFile;
195 fReplace = a_fReplace;
196 strDisplayInfo = strDispInfo;
197 ptrExtPackMgr = a_ptrExtPackMgr;
198
199 HRESULT hrc = ptrProgress.createObject();
200 if (SUCCEEDED(hrc))
201 {
202 Bstr bstrDescription("Installing extension pack");
203 hrc = ptrProgress->init(ptrExtPackFile->m->pVirtualBox,
204 static_cast<IExtPackFile *>(ptrExtPackFile),
205 bstrDescription.raw(),
206 FALSE /*aCancelable*/);
207 }
208
209 return hrc;
210 }
211
212 /** Smart pointer to the progress object for this job. */
213 ComObjPtr<Progress> ptrProgress;
214private:
215 /** Smart pointer to the extension pack file. */
216 ComPtr<ExtPackFile> ptrExtPackFile;
217 /** The replace argument. */
218 bool fReplace;
219 /** The display info argument. */
220 Utf8Str strDisplayInfo;
221 /** Smart pointer to the extension manager. */
222 ComPtr<ExtPackManager> ptrExtPackMgr;
223};
224
225/**
226 * Extension pack uninstallation job.
227 */
228class ExtPackUninstallTask : public ThreadTask
229{
230public:
231 explicit ExtPackUninstallTask() : ThreadTask("ExtPackUninst") { }
232 ~ExtPackUninstallTask() { }
233
234 void handler()
235 {
236 HRESULT hrc = ptrExtPackMgr->i_doUninstall(&strName, fForcedRemoval, &strDisplayInfo);
237 ptrProgress->i_notifyComplete(hrc);
238 }
239
240 HRESULT Init(const ComPtr<ExtPackManager> &a_ptrExtPackMgr, const Utf8Str &a_strName,
241 bool a_fForcedRemoval, const Utf8Str &a_strDisplayInfo)
242 {
243 ptrExtPackMgr = a_ptrExtPackMgr;
244 strName = a_strName;
245 fForcedRemoval = a_fForcedRemoval;
246 strDisplayInfo = a_strDisplayInfo;
247
248 HRESULT hrc = ptrProgress.createObject();
249 if (SUCCEEDED(hrc))
250 {
251 Bstr bstrDescription("Uninstalling extension pack");
252 hrc = ptrProgress->init(ptrExtPackMgr->m->pVirtualBox,
253 static_cast<IExtPackManager *>(ptrExtPackMgr),
254 bstrDescription.raw(),
255 FALSE /*aCancelable*/);
256 }
257
258 return hrc;
259 }
260
261 /** Smart pointer to the progress object for this job. */
262 ComObjPtr<Progress> ptrProgress;
263private:
264 /** Smart pointer to the extension manager. */
265 ComPtr<ExtPackManager> ptrExtPackMgr;
266 /** The name of the extension pack. */
267 Utf8Str strName;
268 /** The replace argument. */
269 bool fForcedRemoval;
270 /** The display info argument. */
271 Utf8Str strDisplayInfo;
272};
273
274DEFINE_EMPTY_CTOR_DTOR(ExtPackFile)
275
276/**
277 * Called by ComObjPtr::createObject when creating the object.
278 *
279 * Just initialize the basic object state, do the rest in initWithDir().
280 *
281 * @returns S_OK.
282 */
283HRESULT ExtPackFile::FinalConstruct()
284{
285 m = NULL;
286 return BaseFinalConstruct();
287}
288
289/**
290 * Initializes the extension pack by reading its file.
291 *
292 * @returns COM status code.
293 * @param a_pszFile The path to the extension pack file.
294 * @param a_pszDigest The SHA-256 digest of the file. Or an empty string.
295 * @param a_pExtPackMgr Pointer to the extension pack manager.
296 * @param a_pVirtualBox Pointer to the VirtualBox object.
297 */
298HRESULT ExtPackFile::initWithFile(const char *a_pszFile, const char *a_pszDigest, ExtPackManager *a_pExtPackMgr,
299 VirtualBox *a_pVirtualBox)
300{
301 AutoInitSpan autoInitSpan(this);
302 AssertReturn(autoInitSpan.isOk(), E_FAIL);
303
304 /*
305 * Allocate + initialize our private data.
306 */
307 m = new ExtPackFile::Data;
308 VBoxExtPackInitDesc(&m->Desc);
309 RT_ZERO(m->ObjInfoDesc);
310 m->fUsable = false;
311 m->strWhyUnusable = tr("ExtPack::init failed");
312 m->strExtPackFile = a_pszFile;
313 m->strDigest = a_pszDigest;
314 m->hExtPackFile = NIL_RTFILE;
315 m->hOurManifest = NIL_RTMANIFEST;
316 m->ptrExtPackMgr = a_pExtPackMgr;
317 m->pVirtualBox = a_pVirtualBox;
318
319 RTCString *pstrTarName = VBoxExtPackExtractNameFromTarballPath(a_pszFile);
320 if (pstrTarName)
321 {
322 m->Desc.strName = *pstrTarName;
323 delete pstrTarName;
324 pstrTarName = NULL;
325 }
326
327 autoInitSpan.setSucceeded();
328
329 /*
330 * Try open the extension pack and check that it is a regular file.
331 */
332 int vrc = RTFileOpen(&m->hExtPackFile, a_pszFile,
333 RTFILE_O_READ | RTFILE_O_DENY_WRITE | RTFILE_O_OPEN);
334 if (RT_FAILURE(vrc))
335 {
336 if (vrc == VERR_FILE_NOT_FOUND || vrc == VERR_PATH_NOT_FOUND)
337 return initFailed(tr("'%s' file not found"), a_pszFile);
338 return initFailed(tr("RTFileOpen('%s',,) failed with %Rrc"), a_pszFile, vrc);
339 }
340
341 RTFSOBJINFO ObjInfo;
342 vrc = RTFileQueryInfo(m->hExtPackFile, &ObjInfo, RTFSOBJATTRADD_UNIX);
343 if (RT_FAILURE(vrc))
344 return initFailed(tr("RTFileQueryInfo failed with %Rrc on '%s'"), vrc, a_pszFile);
345 if (!RTFS_IS_FILE(ObjInfo.Attr.fMode))
346 return initFailed(tr("Not a regular file: %s"), a_pszFile);
347
348 /*
349 * Validate the tarball and extract the XML file.
350 */
351 char szError[8192];
352 RTVFSFILE hXmlFile;
353 vrc = VBoxExtPackValidateTarball(m->hExtPackFile, NULL /*pszExtPackName*/, a_pszFile, a_pszDigest,
354 szError, sizeof(szError), &m->hOurManifest, &hXmlFile, &m->strDigest);
355 if (RT_FAILURE(vrc))
356 return initFailed(tr("%s"), szError);
357
358 /*
359 * Parse the XML.
360 */
361 RTCString strSavedName(m->Desc.strName);
362 RTCString *pStrLoadErr = VBoxExtPackLoadDescFromVfsFile(hXmlFile, &m->Desc, &m->ObjInfoDesc);
363 RTVfsFileRelease(hXmlFile);
364 if (pStrLoadErr != NULL)
365 {
366 m->strWhyUnusable.printf(tr("Failed to the xml file: %s"), pStrLoadErr->c_str());
367 m->Desc.strName = strSavedName;
368 delete pStrLoadErr;
369 return S_OK;
370 }
371
372 /*
373 * Match the tarball name with the name from the XML.
374 */
375 /** @todo drop this restriction after the old install interface is
376 * dropped. */
377 if (!strSavedName.equalsIgnoreCase(m->Desc.strName))
378 return initFailed(tr("Extension pack name mismatch between the downloaded file and the XML inside it (xml='%s' file='%s')"),
379 m->Desc.strName.c_str(), strSavedName.c_str());
380
381
382 m->fUsable = true;
383 m->strWhyUnusable.setNull();
384 return S_OK;
385}
386
387/**
388 * Protected helper that formats the strWhyUnusable value.
389 *
390 * @returns S_OK
391 * @param a_pszWhyFmt Why it failed, format string.
392 * @param ... The format arguments.
393 */
394HRESULT ExtPackFile::initFailed(const char *a_pszWhyFmt, ...)
395{
396 va_list va;
397 va_start(va, a_pszWhyFmt);
398 m->strWhyUnusable.printfV(a_pszWhyFmt, va);
399 va_end(va);
400 return S_OK;
401}
402
403/**
404 * COM cruft.
405 */
406void ExtPackFile::FinalRelease()
407{
408 uninit();
409 BaseFinalRelease();
410}
411
412/**
413 * Do the actual cleanup.
414 */
415void ExtPackFile::uninit()
416{
417 /* Enclose the state transition Ready->InUninit->NotReady */
418 AutoUninitSpan autoUninitSpan(this);
419 if (!autoUninitSpan.uninitDone() && m != NULL)
420 {
421 VBoxExtPackFreeDesc(&m->Desc);
422 RTFileClose(m->hExtPackFile);
423 m->hExtPackFile = NIL_RTFILE;
424 RTManifestRelease(m->hOurManifest);
425 m->hOurManifest = NIL_RTMANIFEST;
426
427 delete m;
428 m = NULL;
429 }
430}
431
432HRESULT ExtPackFile::getName(com::Utf8Str &aName)
433{
434 aName = m->Desc.strName;
435 return S_OK;
436}
437
438HRESULT ExtPackFile::getDescription(com::Utf8Str &aDescription)
439{
440 aDescription = m->Desc.strDescription;
441 return S_OK;
442}
443
444HRESULT ExtPackFile::getVersion(com::Utf8Str &aVersion)
445{
446 aVersion = m->Desc.strVersion;
447 return S_OK;
448}
449
450HRESULT ExtPackFile::getEdition(com::Utf8Str &aEdition)
451{
452 aEdition = m->Desc.strEdition;
453 return S_OK;
454}
455
456HRESULT ExtPackFile::getRevision(ULONG *aRevision)
457{
458 *aRevision = m->Desc.uRevision;
459 return S_OK;
460}
461
462HRESULT ExtPackFile::getVRDEModule(com::Utf8Str &aVRDEModule)
463{
464 aVRDEModule = m->Desc.strVrdeModule;
465 return S_OK;
466}
467
468HRESULT ExtPackFile::getPlugIns(std::vector<ComPtr<IExtPackPlugIn> > &aPlugIns)
469{
470 /** @todo implement plug-ins. */
471#ifdef VBOX_WITH_XPCOM
472 NOREF(aPlugIns);
473#endif
474 NOREF(aPlugIns);
475 ReturnComNotImplemented();
476}
477
478HRESULT ExtPackFile::getUsable(BOOL *aUsable)
479{
480 *aUsable = m->fUsable;
481 return S_OK;
482}
483
484HRESULT ExtPackFile::getWhyUnusable(com::Utf8Str &aWhyUnusable)
485{
486 aWhyUnusable = m->strWhyUnusable;
487 return S_OK;
488}
489
490HRESULT ExtPackFile::getShowLicense(BOOL *aShowLicense)
491{
492 *aShowLicense = m->Desc.fShowLicense;
493 return S_OK;
494}
495
496HRESULT ExtPackFile::getLicense(com::Utf8Str &aLicense)
497{
498 Utf8Str strHtml("html");
499 Utf8Str str("");
500 return queryLicense(str, str, strHtml, aLicense);
501}
502
503/* Same as ExtPack::QueryLicense, should really explore the subject of base classes here... */
504HRESULT ExtPackFile::queryLicense(const com::Utf8Str &aPreferredLocale, const com::Utf8Str &aPreferredLanguage,
505 const com::Utf8Str &aFormat, com::Utf8Str &aLicenseText)
506{
507 HRESULT hrc = S_OK;
508
509 /*
510 * Validate input.
511 */
512
513 if (aPreferredLocale.length() != 2 && aPreferredLocale.length() != 0)
514 return setError(E_FAIL, tr("The preferred locale is a two character string or empty."));
515
516 if (aPreferredLanguage.length() != 2 && aPreferredLanguage.length() != 0)
517 return setError(E_FAIL, tr("The preferred lanuage is a two character string or empty."));
518
519 if ( !aFormat.equals("html")
520 && !aFormat.equals("rtf")
521 && !aFormat.equals("txt"))
522 return setError(E_FAIL, tr("The license format can only have the values 'html', 'rtf' and 'txt'."));
523
524 /*
525 * Combine the options to form a file name before locking down anything.
526 */
527 char szName[sizeof(VBOX_EXTPACK_LICENSE_NAME_PREFIX "-de_DE.html") + 2];
528 if (aPreferredLocale.isNotEmpty() && aPreferredLanguage.isNotEmpty())
529 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX "-%s_%s.%s",
530 aPreferredLocale.c_str(), aPreferredLanguage.c_str(), aFormat.c_str());
531 else if (aPreferredLocale.isNotEmpty())
532 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX "-%s.%s",
533 aPreferredLocale.c_str(), aFormat.c_str());
534 else if (aPreferredLanguage.isNotEmpty())
535 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX "-_%s.%s",
536 aPreferredLocale.c_str(), aFormat.c_str());
537 else
538 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX ".%s",
539 aFormat.c_str());
540 /*
541 * Lock the extension pack. We need a write lock here as there must not be
542 * concurrent accesses to the tar file handle.
543 */
544 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
545
546 /*
547 * Do not permit this query on a pack that isn't considered usable (could
548 * be marked so because of bad license files).
549 */
550 if (!m->fUsable)
551 hrc = setError(E_FAIL, tr("%s"), m->strWhyUnusable.c_str());
552 else
553 {
554 /*
555 * Look it up in the manifest before scanning the tarball for it
556 */
557 if (RTManifestEntryExists(m->hOurManifest, szName))
558 {
559 RTVFSFSSTREAM hTarFss;
560 char szError[8192];
561 int vrc = VBoxExtPackOpenTarFss(m->hExtPackFile, szError, sizeof(szError), &hTarFss, NULL);
562 if (RT_SUCCESS(vrc))
563 {
564 for (;;)
565 {
566 /* Get the first/next. */
567 char *pszName;
568 RTVFSOBJ hVfsObj;
569 RTVFSOBJTYPE enmType;
570 vrc = RTVfsFsStrmNext(hTarFss, &pszName, &enmType, &hVfsObj);
571 if (RT_FAILURE(vrc))
572 {
573 if (vrc != VERR_EOF)
574 hrc = setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("RTVfsFsStrmNext failed: %Rrc"), vrc);
575 else
576 hrc = setErrorBoth(E_UNEXPECTED, vrc, tr("'%s' was found in the manifest but not in the tarball"), szName);
577 break;
578 }
579
580 /* Is this it? */
581 const char *pszAdjName = pszName[0] == '.' && pszName[1] == '/' ? &pszName[2] : pszName;
582 if ( !strcmp(pszAdjName, szName)
583 && ( enmType == RTVFSOBJTYPE_IO_STREAM
584 || enmType == RTVFSOBJTYPE_FILE))
585 {
586 RTVFSIOSTREAM hVfsIos = RTVfsObjToIoStream(hVfsObj);
587 RTVfsObjRelease(hVfsObj);
588 RTStrFree(pszName);
589
590 /* Load the file into memory. */
591 RTFSOBJINFO ObjInfo;
592 vrc = RTVfsIoStrmQueryInfo(hVfsIos, &ObjInfo, RTFSOBJATTRADD_NOTHING);
593 if (RT_SUCCESS(vrc))
594 {
595 size_t cbFile = (size_t)ObjInfo.cbObject;
596 void *pvFile = RTMemAllocZ(cbFile + 1);
597 if (pvFile)
598 {
599 vrc = RTVfsIoStrmRead(hVfsIos, pvFile, cbFile, true /*fBlocking*/, NULL);
600 if (RT_SUCCESS(vrc))
601 {
602 /* try translate it into a string we can return. */
603 Bstr bstrLicense((const char *)pvFile, cbFile);
604 if (bstrLicense.isNotEmpty())
605 {
606 aLicenseText = Utf8Str(bstrLicense);
607 hrc = S_OK;
608 }
609 else
610 hrc = setErrorBoth(VBOX_E_IPRT_ERROR, vrc,
611 tr("The license file '%s' is empty or contains invalid UTF-8 encoding"),
612 szName);
613 }
614 else
615 hrc = setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("Failed to read '%s': %Rrc"), szName, vrc);
616 RTMemFree(pvFile);
617 }
618 else
619 hrc = setError(E_OUTOFMEMORY, tr("Failed to allocate %zu bytes for '%s'"), cbFile, szName);
620 }
621 else
622 hrc = setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("RTVfsIoStrmQueryInfo on '%s': %Rrc"), szName, vrc);
623 RTVfsIoStrmRelease(hVfsIos);
624 break;
625 }
626
627 /* Release current. */
628 RTVfsObjRelease(hVfsObj);
629 RTStrFree(pszName);
630 }
631 RTVfsFsStrmRelease(hTarFss);
632 }
633 else
634 hrc = setError(VBOX_E_OBJECT_NOT_FOUND, tr("%s"), szError);
635 }
636 else
637 hrc = setError(VBOX_E_OBJECT_NOT_FOUND, tr("The license file '%s' was not found in '%s'"),
638 szName, m->strExtPackFile.c_str());
639 }
640 return hrc;
641}
642
643HRESULT ExtPackFile::getFilePath(com::Utf8Str &aFilePath)
644{
645
646 aFilePath = m->strExtPackFile;
647 return S_OK;
648}
649
650HRESULT ExtPackFile::install(BOOL aReplace, const com::Utf8Str &aDisplayInfo, ComPtr<IProgress> &aProgress)
651{
652 HRESULT hrc;
653 if (m->fUsable)
654 {
655 ExtPackInstallTask *pTask = NULL;
656 try
657 {
658 pTask = new ExtPackInstallTask();
659 hrc = pTask->Init(this, aReplace != FALSE, aDisplayInfo, m->ptrExtPackMgr);
660 if (SUCCEEDED(hrc))
661 {
662 ComPtr<Progress> ptrProgress = pTask->ptrProgress;
663 hrc = pTask->createThreadWithType(RTTHREADTYPE_DEFAULT);
664 pTask = NULL; /* The _completely_ _undocumented_ createThread method always consumes pTask. */
665 if (SUCCEEDED(hrc))
666 hrc = ptrProgress.queryInterfaceTo(aProgress.asOutParam());
667 else
668 hrc = setError(VBOX_E_IPRT_ERROR,
669 tr("Starting thread for an extension pack installation failed with %Rrc"), hrc);
670 }
671 else
672 hrc = setError(VBOX_E_IPRT_ERROR,
673 tr("Looks like creating a progress object for ExtraPackInstallTask object failed"));
674 }
675 catch (std::bad_alloc &)
676 {
677 hrc = E_OUTOFMEMORY;
678 }
679 catch (HRESULT hrcXcpt)
680 {
681 LogFlowThisFunc(("Exception was caught in the function ExtPackFile::install() \n"));
682 hrc = hrcXcpt;
683 }
684 if (pTask)
685 delete pTask;
686 }
687 else
688 hrc = setError(E_FAIL, "%s", m->strWhyUnusable.c_str());
689 return hrc;
690}
691
692#endif /* !VBOX_COM_INPROC */
693
694
695
696
697DEFINE_EMPTY_CTOR_DTOR(ExtPack)
698
699/**
700 * Called by ComObjPtr::createObject when creating the object.
701 *
702 * Just initialize the basic object state, do the rest in initWithDir().
703 *
704 * @returns S_OK.
705 */
706HRESULT ExtPack::FinalConstruct()
707{
708 m = NULL;
709 return BaseFinalConstruct();
710}
711
712/**
713 * Initializes the extension pack by reading its file.
714 *
715 * @returns COM status code.
716 * @param a_enmContext The context we're in.
717 * @param a_pszName The name of the extension pack. This is also the
718 * name of the subdirector under @a a_pszParentDir
719 * where the extension pack is installed.
720 * @param a_pszDir The extension pack directory name.
721 */
722HRESULT ExtPack::initWithDir(VBOXEXTPACKCTX a_enmContext, const char *a_pszName, const char *a_pszDir)
723{
724 AutoInitSpan autoInitSpan(this);
725 AssertReturn(autoInitSpan.isOk(), E_FAIL);
726
727 static const VBOXEXTPACKHLP s_HlpTmpl =
728 {
729 /* u32Version = */ VBOXEXTPACKHLP_VERSION,
730 /* uVBoxFullVersion = */ VBOX_FULL_VERSION,
731 /* uVBoxVersionRevision = */ 0,
732 /* u32Padding = */ 0,
733 /* pszVBoxVersion = */ "",
734 /* pfnFindModule = */ ExtPack::i_hlpFindModule,
735 /* pfnGetFilePath = */ ExtPack::i_hlpGetFilePath,
736 /* pfnGetContext = */ ExtPack::i_hlpGetContext,
737 /* pfnLoadHGCMService = */ ExtPack::i_hlpLoadHGCMService,
738 /* pfnLoadVDPlugin = */ ExtPack::i_hlpLoadVDPlugin,
739 /* pfnUnloadVDPlugin = */ ExtPack::i_hlpUnloadVDPlugin,
740 /* pfnReserved1 = */ ExtPack::i_hlpReservedN,
741 /* pfnReserved2 = */ ExtPack::i_hlpReservedN,
742 /* pfnReserved3 = */ ExtPack::i_hlpReservedN,
743 /* pfnReserved4 = */ ExtPack::i_hlpReservedN,
744 /* pfnReserved5 = */ ExtPack::i_hlpReservedN,
745 /* pfnReserved6 = */ ExtPack::i_hlpReservedN,
746 /* u32EndMarker = */ VBOXEXTPACKHLP_VERSION
747 };
748
749 /*
750 * Allocate + initialize our private data.
751 */
752 m = new Data;
753 VBoxExtPackInitDesc(&m->Desc);
754 m->Desc.strName = a_pszName;
755 RT_ZERO(m->ObjInfoDesc);
756 m->fUsable = false;
757 m->strWhyUnusable = tr("ExtPack::init failed");
758 m->strExtPackPath = a_pszDir;
759 RT_ZERO(m->ObjInfoExtPack);
760 m->strMainModPath.setNull();
761 RT_ZERO(m->ObjInfoMainMod);
762 m->hMainMod = NIL_RTLDRMOD;
763 m->Hlp = s_HlpTmpl;
764 m->Hlp.pszVBoxVersion = RTBldCfgVersion();
765 m->Hlp.uVBoxInternalRevision = RTBldCfgRevision();
766 m->pThis = this;
767 m->pReg = NULL;
768 m->enmContext = a_enmContext;
769 m->fMadeReadyCall = false;
770
771 /*
772 * Make sure the SUPR3Hardened API works (ignoring errors for now).
773 */
774 int rc = SUPR3HardenedVerifyInit();
775 if (RT_FAILURE(rc))
776 LogRel(("SUPR3HardenedVerifyInit failed: %Rrc\n", rc));
777
778 /*
779 * Probe the extension pack (this code is shared with refresh()).
780 */
781 i_probeAndLoad();
782
783 autoInitSpan.setSucceeded();
784 return S_OK;
785}
786
787/**
788 * COM cruft.
789 */
790void ExtPack::FinalRelease()
791{
792 uninit();
793 BaseFinalRelease();
794}
795
796/**
797 * Do the actual cleanup.
798 */
799void ExtPack::uninit()
800{
801 /* Enclose the state transition Ready->InUninit->NotReady */
802 AutoUninitSpan autoUninitSpan(this);
803 if (!autoUninitSpan.uninitDone() && m != NULL)
804 {
805 if (m->hMainMod != NIL_RTLDRMOD)
806 {
807 AssertPtr(m->pReg);
808 if (m->pReg->pfnUnload != NULL)
809 m->pReg->pfnUnload(m->pReg);
810
811 RTLdrClose(m->hMainMod);
812 m->hMainMod = NIL_RTLDRMOD;
813 m->pReg = NULL;
814 }
815
816 VBoxExtPackFreeDesc(&m->Desc);
817
818 delete m;
819 m = NULL;
820 }
821}
822
823
824#ifndef VBOX_COM_INPROC
825/**
826 * Calls the installed hook.
827 *
828 * @returns true if we left the lock, false if we didn't.
829 * @param a_pVirtualBox The VirtualBox interface.
830 * @param a_pLock The write lock held by the caller.
831 * @param pErrInfo Where to return error information.
832 */
833bool ExtPack::i_callInstalledHook(IVirtualBox *a_pVirtualBox, AutoWriteLock *a_pLock, PRTERRINFO pErrInfo)
834{
835 if ( m != NULL
836 && m->hMainMod != NIL_RTLDRMOD)
837 {
838 if (m->pReg->pfnInstalled)
839 {
840 ComPtr<ExtPack> ptrSelfRef = this;
841 a_pLock->release();
842 pErrInfo->rc = m->pReg->pfnInstalled(m->pReg, a_pVirtualBox, pErrInfo);
843 a_pLock->acquire();
844 return true;
845 }
846 }
847 pErrInfo->rc = VINF_SUCCESS;
848 return false;
849}
850
851/**
852 * Calls the uninstall hook and closes the module.
853 *
854 * @returns S_OK or COM error status with error information.
855 * @param a_pVirtualBox The VirtualBox interface.
856 * @param a_fForcedRemoval When set, we'll ignore complaints from the
857 * uninstall hook.
858 * @remarks The caller holds the manager's write lock, not released.
859 */
860HRESULT ExtPack::i_callUninstallHookAndClose(IVirtualBox *a_pVirtualBox, bool a_fForcedRemoval)
861{
862 HRESULT hrc = S_OK;
863
864 if ( m != NULL
865 && m->hMainMod != NIL_RTLDRMOD)
866 {
867 if (m->pReg->pfnUninstall && !a_fForcedRemoval)
868 {
869 int vrc = m->pReg->pfnUninstall(m->pReg, a_pVirtualBox);
870 if (RT_FAILURE(vrc))
871 {
872 LogRel(("ExtPack pfnUninstall returned %Rrc for %s\n", vrc, m->Desc.strName.c_str()));
873 if (!a_fForcedRemoval)
874 hrc = setErrorBoth(E_FAIL, vrc, tr("pfnUninstall returned %Rrc"), vrc);
875 }
876 }
877 if (SUCCEEDED(hrc))
878 {
879 RTLdrClose(m->hMainMod);
880 m->hMainMod = NIL_RTLDRMOD;
881 m->pReg = NULL;
882 }
883 }
884
885 return hrc;
886}
887
888/**
889 * Calls the pfnVirtualBoxReady hook.
890 *
891 * @returns true if we left the lock, false if we didn't.
892 * @param a_pVirtualBox The VirtualBox interface.
893 * @param a_pLock The write lock held by the caller.
894 */
895bool ExtPack::i_callVirtualBoxReadyHook(IVirtualBox *a_pVirtualBox, AutoWriteLock *a_pLock)
896{
897 if ( m != NULL
898 && m->fUsable
899 && m->hMainMod != NIL_RTLDRMOD
900 && !m->fMadeReadyCall)
901 {
902 m->fMadeReadyCall = true;
903 if (m->pReg->pfnVirtualBoxReady)
904 {
905 ComPtr<ExtPack> ptrSelfRef = this;
906 a_pLock->release();
907 m->pReg->pfnVirtualBoxReady(m->pReg, a_pVirtualBox);
908 a_pLock->acquire();
909 return true;
910 }
911 }
912 return false;
913}
914#endif /* !VBOX_COM_INPROC */
915
916#ifdef VBOX_COM_INPROC
917/**
918 * Calls the pfnConsoleReady hook.
919 *
920 * @returns true if we left the lock, false if we didn't.
921 * @param a_pConsole The Console interface.
922 * @param a_pLock The write lock held by the caller.
923 */
924bool ExtPack::i_callConsoleReadyHook(IConsole *a_pConsole, AutoWriteLock *a_pLock)
925{
926 if ( m != NULL
927 && m->fUsable
928 && m->hMainMod != NIL_RTLDRMOD
929 && !m->fMadeReadyCall)
930 {
931 m->fMadeReadyCall = true;
932 if (m->pReg->pfnConsoleReady)
933 {
934 ComPtr<ExtPack> ptrSelfRef = this;
935 a_pLock->release();
936 m->pReg->pfnConsoleReady(m->pReg, a_pConsole);
937 a_pLock->acquire();
938 return true;
939 }
940 }
941 return false;
942}
943#endif /* VBOX_COM_INPROC */
944
945#ifndef VBOX_COM_INPROC
946/**
947 * Calls the pfnVMCreate hook.
948 *
949 * @returns true if we left the lock, false if we didn't.
950 * @param a_pVirtualBox The VirtualBox interface.
951 * @param a_pMachine The machine interface of the new VM.
952 * @param a_pLock The write lock held by the caller.
953 */
954bool ExtPack::i_callVmCreatedHook(IVirtualBox *a_pVirtualBox, IMachine *a_pMachine, AutoWriteLock *a_pLock)
955{
956 if ( m != NULL
957 && m->hMainMod != NIL_RTLDRMOD
958 && m->fUsable)
959 {
960 if (m->pReg->pfnVMCreated)
961 {
962 ComPtr<ExtPack> ptrSelfRef = this;
963 a_pLock->release();
964 m->pReg->pfnVMCreated(m->pReg, a_pVirtualBox, a_pMachine);
965 a_pLock->acquire();
966 return true;
967 }
968 }
969 return false;
970}
971#endif /* !VBOX_COM_INPROC */
972
973#ifdef VBOX_COM_INPROC
974/**
975 * Calls the pfnVMConfigureVMM hook.
976 *
977 * @returns true if we left the lock, false if we didn't.
978 * @param a_pConsole The console interface.
979 * @param a_pVM The VM handle.
980 * @param a_pLock The write lock held by the caller.
981 * @param a_pvrc Where to return the status code of the
982 * callback. This is always set. LogRel is
983 * called on if a failure status is returned.
984 */
985bool ExtPack::i_callVmConfigureVmmHook(IConsole *a_pConsole, PVM a_pVM, AutoWriteLock *a_pLock, int *a_pvrc)
986{
987 *a_pvrc = VINF_SUCCESS;
988 if ( m != NULL
989 && m->hMainMod != NIL_RTLDRMOD
990 && m->fUsable)
991 {
992 if (m->pReg->pfnVMConfigureVMM)
993 {
994 ComPtr<ExtPack> ptrSelfRef = this;
995 a_pLock->release();
996 int vrc = m->pReg->pfnVMConfigureVMM(m->pReg, a_pConsole, a_pVM);
997 *a_pvrc = vrc;
998 a_pLock->acquire();
999 if (RT_FAILURE(vrc))
1000 LogRel(("ExtPack pfnVMConfigureVMM returned %Rrc for %s\n", vrc, m->Desc.strName.c_str()));
1001 return true;
1002 }
1003 }
1004 return false;
1005}
1006
1007/**
1008 * Calls the pfnVMPowerOn hook.
1009 *
1010 * @returns true if we left the lock, false if we didn't.
1011 * @param a_pConsole The console interface.
1012 * @param a_pVM The VM handle.
1013 * @param a_pLock The write lock held by the caller.
1014 * @param a_pvrc Where to return the status code of the
1015 * callback. This is always set. LogRel is
1016 * called on if a failure status is returned.
1017 */
1018bool ExtPack::i_callVmPowerOnHook(IConsole *a_pConsole, PVM a_pVM, AutoWriteLock *a_pLock, int *a_pvrc)
1019{
1020 *a_pvrc = VINF_SUCCESS;
1021 if ( m != NULL
1022 && m->hMainMod != NIL_RTLDRMOD
1023 && m->fUsable)
1024 {
1025 if (m->pReg->pfnVMPowerOn)
1026 {
1027 ComPtr<ExtPack> ptrSelfRef = this;
1028 a_pLock->release();
1029 int vrc = m->pReg->pfnVMPowerOn(m->pReg, a_pConsole, a_pVM);
1030 *a_pvrc = vrc;
1031 a_pLock->acquire();
1032 if (RT_FAILURE(vrc))
1033 LogRel(("ExtPack pfnVMPowerOn returned %Rrc for %s\n", vrc, m->Desc.strName.c_str()));
1034 return true;
1035 }
1036 }
1037 return false;
1038}
1039
1040/**
1041 * Calls the pfnVMPowerOff hook.
1042 *
1043 * @returns true if we left the lock, false if we didn't.
1044 * @param a_pConsole The console interface.
1045 * @param a_pVM The VM handle.
1046 * @param a_pLock The write lock held by the caller.
1047 */
1048bool ExtPack::i_callVmPowerOffHook(IConsole *a_pConsole, PVM a_pVM, AutoWriteLock *a_pLock)
1049{
1050 if ( m != NULL
1051 && m->hMainMod != NIL_RTLDRMOD
1052 && m->fUsable)
1053 {
1054 if (m->pReg->pfnVMPowerOff)
1055 {
1056 ComPtr<ExtPack> ptrSelfRef = this;
1057 a_pLock->release();
1058 m->pReg->pfnVMPowerOff(m->pReg, a_pConsole, a_pVM);
1059 a_pLock->acquire();
1060 return true;
1061 }
1062 }
1063 return false;
1064}
1065#endif /* !VBOX_COM_INPROC */
1066
1067/**
1068 * Check if the extension pack is usable and has an VRDE module.
1069 *
1070 * @returns S_OK or COM error status with error information.
1071 *
1072 * @remarks Caller holds the extension manager lock for reading, no locking
1073 * necessary.
1074 */
1075HRESULT ExtPack::i_checkVrde(void)
1076{
1077 HRESULT hrc;
1078 if ( m != NULL
1079 && m->fUsable)
1080 {
1081 if (m->Desc.strVrdeModule.isNotEmpty())
1082 hrc = S_OK;
1083 else
1084 hrc = setError(E_FAIL, tr("The extension pack '%s' does not include a VRDE module"), m->Desc.strName.c_str());
1085 }
1086 else
1087 hrc = setError(E_FAIL, tr("%s"), m->strWhyUnusable.c_str());
1088 return hrc;
1089}
1090
1091/**
1092 * Same as checkVrde(), except that it also resolves the path to the module.
1093 *
1094 * @returns S_OK or COM error status with error information.
1095 * @param a_pstrVrdeLibrary Where to return the path on success.
1096 *
1097 * @remarks Caller holds the extension manager lock for reading, no locking
1098 * necessary.
1099 */
1100HRESULT ExtPack::i_getVrdpLibraryName(Utf8Str *a_pstrVrdeLibrary)
1101{
1102 HRESULT hrc = i_checkVrde();
1103 if (SUCCEEDED(hrc))
1104 {
1105 if (i_findModule(m->Desc.strVrdeModule.c_str(), NULL, VBOXEXTPACKMODKIND_R3,
1106 a_pstrVrdeLibrary, NULL /*a_pfNative*/, NULL /*a_pObjInfo*/))
1107 hrc = S_OK;
1108 else
1109 hrc = setError(E_FAIL, tr("Failed to locate the VRDE module '%s' in extension pack '%s'"),
1110 m->Desc.strVrdeModule.c_str(), m->Desc.strName.c_str());
1111 }
1112 return hrc;
1113}
1114
1115/**
1116 * Resolves the path to the module.
1117 *
1118 * @returns S_OK or COM error status with error information.
1119 * @param a_pszModuleName The library.
1120 * @param a_pstrLibrary Where to return the path on success.
1121 *
1122 * @remarks Caller holds the extension manager lock for reading, no locking
1123 * necessary.
1124 */
1125HRESULT ExtPack::i_getLibraryName(const char *a_pszModuleName, Utf8Str *a_pstrLibrary)
1126{
1127 HRESULT hrc;
1128 if (i_findModule(a_pszModuleName, NULL, VBOXEXTPACKMODKIND_R3,
1129 a_pstrLibrary, NULL /*a_pfNative*/, NULL /*a_pObjInfo*/))
1130 hrc = S_OK;
1131 else
1132 hrc = setError(E_FAIL, tr("Failed to locate the module '%s' in extension pack '%s'"),
1133 a_pszModuleName, m->Desc.strName.c_str());
1134 return hrc;
1135}
1136
1137/**
1138 * Check if this extension pack wishes to be the default VRDE provider.
1139 *
1140 * @returns @c true if it wants to and it is in a usable state, otherwise
1141 * @c false.
1142 *
1143 * @remarks Caller holds the extension manager lock for reading, no locking
1144 * necessary.
1145 */
1146bool ExtPack::i_wantsToBeDefaultVrde(void) const
1147{
1148 return m->fUsable
1149 && m->Desc.strVrdeModule.isNotEmpty();
1150}
1151
1152/**
1153 * Refreshes the extension pack state.
1154 *
1155 * This is called by the manager so that the on disk changes are picked up.
1156 *
1157 * @returns S_OK or COM error status with error information.
1158 *
1159 * @param a_pfCanDelete Optional can-delete-this-object output indicator.
1160 *
1161 * @remarks Caller holds the extension manager lock for writing.
1162 * @remarks Only called in VBoxSVC.
1163 */
1164HRESULT ExtPack::i_refresh(bool *a_pfCanDelete)
1165{
1166 if (a_pfCanDelete)
1167 *a_pfCanDelete = false;
1168
1169 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS); /* for the COMGETTERs */
1170
1171 /*
1172 * Has the module been deleted?
1173 */
1174 RTFSOBJINFO ObjInfoExtPack;
1175 int vrc = RTPathQueryInfoEx(m->strExtPackPath.c_str(), &ObjInfoExtPack, RTFSOBJATTRADD_UNIX, RTPATH_F_ON_LINK);
1176 if ( RT_FAILURE(vrc)
1177 || !RTFS_IS_DIRECTORY(ObjInfoExtPack.Attr.fMode))
1178 {
1179 if (a_pfCanDelete)
1180 *a_pfCanDelete = true;
1181 return S_OK;
1182 }
1183
1184 /*
1185 * We've got a directory, so try query file system object info for the
1186 * files we are interested in as well.
1187 */
1188 RTFSOBJINFO ObjInfoDesc;
1189 char szDescFilePath[RTPATH_MAX];
1190 vrc = RTPathJoin(szDescFilePath, sizeof(szDescFilePath), m->strExtPackPath.c_str(), VBOX_EXTPACK_DESCRIPTION_NAME);
1191 if (RT_SUCCESS(vrc))
1192 vrc = RTPathQueryInfoEx(szDescFilePath, &ObjInfoDesc, RTFSOBJATTRADD_UNIX, RTPATH_F_ON_LINK);
1193 if (RT_FAILURE(vrc))
1194 RT_ZERO(ObjInfoDesc);
1195
1196 RTFSOBJINFO ObjInfoMainMod;
1197 if (m->strMainModPath.isNotEmpty())
1198 vrc = RTPathQueryInfoEx(m->strMainModPath.c_str(), &ObjInfoMainMod, RTFSOBJATTRADD_UNIX, RTPATH_F_ON_LINK);
1199 if (m->strMainModPath.isEmpty() || RT_FAILURE(vrc))
1200 RT_ZERO(ObjInfoMainMod);
1201
1202 /*
1203 * If we have a usable module already, just verify that things haven't
1204 * changed since we loaded it.
1205 */
1206 if (m->fUsable)
1207 {
1208 if (m->hMainMod == NIL_RTLDRMOD)
1209 i_probeAndLoad();
1210 else if ( !i_objinfoIsEqual(&ObjInfoDesc, &m->ObjInfoDesc)
1211 || !i_objinfoIsEqual(&ObjInfoMainMod, &m->ObjInfoMainMod)
1212 || !i_objinfoIsEqual(&ObjInfoExtPack, &m->ObjInfoExtPack) )
1213 {
1214 /** @todo not important, so it can wait. */
1215 }
1216 }
1217 /*
1218 * Ok, it is currently not usable. If anything has changed since last time
1219 * reprobe the extension pack.
1220 */
1221 else if ( !i_objinfoIsEqual(&ObjInfoDesc, &m->ObjInfoDesc)
1222 || !i_objinfoIsEqual(&ObjInfoMainMod, &m->ObjInfoMainMod)
1223 || !i_objinfoIsEqual(&ObjInfoExtPack, &m->ObjInfoExtPack) )
1224 i_probeAndLoad();
1225
1226 return S_OK;
1227}
1228
1229/**
1230 * Probes the extension pack, loading the main dll and calling its registration
1231 * entry point.
1232 *
1233 * This updates the state accordingly, the strWhyUnusable and fUnusable members
1234 * being the most important ones.
1235 */
1236void ExtPack::i_probeAndLoad(void)
1237{
1238 m->fUsable = false;
1239 m->fMadeReadyCall = false;
1240
1241 /*
1242 * Query the file system info for the extension pack directory. This and
1243 * all other file system info we save is for the benefit of refresh().
1244 */
1245 int vrc = RTPathQueryInfoEx(m->strExtPackPath.c_str(), &m->ObjInfoExtPack, RTFSOBJATTRADD_UNIX, RTPATH_F_ON_LINK);
1246 if (RT_FAILURE(vrc))
1247 {
1248 m->strWhyUnusable.printf(tr("RTPathQueryInfoEx on '%s' failed: %Rrc"), m->strExtPackPath.c_str(), vrc);
1249 return;
1250 }
1251 if (!RTFS_IS_DIRECTORY(m->ObjInfoExtPack.Attr.fMode))
1252 {
1253 if (RTFS_IS_SYMLINK(m->ObjInfoExtPack.Attr.fMode))
1254 m->strWhyUnusable.printf(tr("'%s' is a symbolic link, this is not allowed"),
1255 m->strExtPackPath.c_str(), vrc);
1256 else if (RTFS_IS_FILE(m->ObjInfoExtPack.Attr.fMode))
1257 m->strWhyUnusable.printf(tr("'%s' is a symbolic file, not a directory"),
1258 m->strExtPackPath.c_str(), vrc);
1259 else
1260 m->strWhyUnusable.printf(tr("'%s' is not a directory (fMode=%#x)"),
1261 m->strExtPackPath.c_str(), m->ObjInfoExtPack.Attr.fMode);
1262 return;
1263 }
1264
1265 RTERRINFOSTATIC ErrInfo;
1266 RTErrInfoInitStatic(&ErrInfo);
1267 vrc = SUPR3HardenedVerifyDir(m->strExtPackPath.c_str(), true /*fRecursive*/, true /*fCheckFiles*/, &ErrInfo.Core);
1268 if (RT_FAILURE(vrc))
1269 {
1270 m->strWhyUnusable.printf(tr("%s (rc=%Rrc)"), ErrInfo.Core.pszMsg, vrc);
1271 return;
1272 }
1273
1274 /*
1275 * Read the description file.
1276 */
1277 RTCString strSavedName(m->Desc.strName);
1278 RTCString *pStrLoadErr = VBoxExtPackLoadDesc(m->strExtPackPath.c_str(), &m->Desc, &m->ObjInfoDesc);
1279 if (pStrLoadErr != NULL)
1280 {
1281 m->strWhyUnusable.printf(tr("Failed to load '%s/%s': %s"),
1282 m->strExtPackPath.c_str(), VBOX_EXTPACK_DESCRIPTION_NAME, pStrLoadErr->c_str());
1283 m->Desc.strName = strSavedName;
1284 delete pStrLoadErr;
1285 return;
1286 }
1287
1288 /*
1289 * Make sure the XML name and directory matches.
1290 */
1291 if (!m->Desc.strName.equalsIgnoreCase(strSavedName))
1292 {
1293 m->strWhyUnusable.printf(tr("The description name ('%s') and directory name ('%s') does not match"),
1294 m->Desc.strName.c_str(), strSavedName.c_str());
1295 m->Desc.strName = strSavedName;
1296 return;
1297 }
1298
1299 /*
1300 * Load the main DLL and call the predefined entry point.
1301 */
1302#ifndef VBOX_COM_INPROC
1303 const char *pszMainModule = m->Desc.strMainModule.c_str();
1304#else
1305 const char *pszMainModule = m->Desc.strMainVMModule.c_str();
1306 if (m->Desc.strMainVMModule.isEmpty())
1307 {
1308 /*
1309 * We're good! The main module for VM processes is optional.
1310 */
1311 m->fUsable = true;
1312 m->strWhyUnusable.setNull();
1313 return;
1314 }
1315#endif
1316 bool fIsNative;
1317 if (!i_findModule(pszMainModule, NULL /* default extension */, VBOXEXTPACKMODKIND_R3,
1318 &m->strMainModPath, &fIsNative, &m->ObjInfoMainMod))
1319 {
1320 m->strWhyUnusable.printf(tr("Failed to locate the main module ('%s')"), pszMainModule);
1321 return;
1322 }
1323
1324 vrc = SUPR3HardenedVerifyPlugIn(m->strMainModPath.c_str(), &ErrInfo.Core);
1325 if (RT_FAILURE(vrc))
1326 {
1327 m->strWhyUnusable.printf(tr("%s"), ErrInfo.Core.pszMsg);
1328 return;
1329 }
1330
1331 if (fIsNative)
1332 {
1333 vrc = SUPR3HardenedLdrLoadPlugIn(m->strMainModPath.c_str(), &m->hMainMod, &ErrInfo.Core);
1334 if (RT_FAILURE(vrc))
1335 {
1336 m->hMainMod = NIL_RTLDRMOD;
1337 m->strWhyUnusable.printf(tr("Failed to load the main module ('%s'): %Rrc - %s"),
1338 m->strMainModPath.c_str(), vrc, ErrInfo.Core.pszMsg);
1339 return;
1340 }
1341 }
1342 else
1343 {
1344 m->strWhyUnusable.printf(tr("Only native main modules are currently supported"));
1345 return;
1346 }
1347
1348 /*
1349 * Resolve the predefined entry point.
1350 */
1351#ifndef VBOX_COM_INPROC
1352 const char *pszMainEntryPoint = VBOX_EXTPACK_MAIN_MOD_ENTRY_POINT;
1353 PFNVBOXEXTPACKREGISTER pfnRegistration;
1354 uint32_t uVersion = VBOXEXTPACKREG_VERSION;
1355#else
1356 const char *pszMainEntryPoint = VBOX_EXTPACK_MAIN_VM_MOD_ENTRY_POINT;
1357 PFNVBOXEXTPACKVMREGISTER pfnRegistration;
1358 uint32_t uVersion = VBOXEXTPACKVMREG_VERSION;
1359#endif
1360 vrc = RTLdrGetSymbol(m->hMainMod, pszMainEntryPoint, (void **)&pfnRegistration);
1361 if (RT_SUCCESS(vrc))
1362 {
1363 RTErrInfoClear(&ErrInfo.Core);
1364 vrc = pfnRegistration(&m->Hlp, &m->pReg, &ErrInfo.Core);
1365 if ( RT_SUCCESS(vrc)
1366 && !RTErrInfoIsSet(&ErrInfo.Core)
1367 && VALID_PTR(m->pReg))
1368 {
1369 if ( VBOXEXTPACK_IS_MAJOR_VER_EQUAL(m->pReg->u32Version, uVersion)
1370 && m->pReg->u32EndMarker == m->pReg->u32Version)
1371 {
1372#ifndef VBOX_COM_INPROC
1373 if ( (!m->pReg->pfnInstalled || RT_VALID_PTR(m->pReg->pfnInstalled))
1374 && (!m->pReg->pfnUninstall || RT_VALID_PTR(m->pReg->pfnUninstall))
1375 && (!m->pReg->pfnVirtualBoxReady || RT_VALID_PTR(m->pReg->pfnVirtualBoxReady))
1376 && (!m->pReg->pfnUnload || RT_VALID_PTR(m->pReg->pfnUnload))
1377 && (!m->pReg->pfnVMCreated || RT_VALID_PTR(m->pReg->pfnVMCreated))
1378 && (!m->pReg->pfnQueryObject || RT_VALID_PTR(m->pReg->pfnQueryObject))
1379 )
1380 {
1381 /*
1382 * We're good!
1383 */
1384 m->fUsable = true;
1385 m->strWhyUnusable.setNull();
1386 return;
1387 }
1388#else
1389 if ( (!m->pReg->pfnConsoleReady || RT_VALID_PTR(m->pReg->pfnConsoleReady))
1390 && (!m->pReg->pfnUnload || RT_VALID_PTR(m->pReg->pfnUnload))
1391 && (!m->pReg->pfnVMConfigureVMM || RT_VALID_PTR(m->pReg->pfnVMConfigureVMM))
1392 && (!m->pReg->pfnVMPowerOn || RT_VALID_PTR(m->pReg->pfnVMPowerOn))
1393 && (!m->pReg->pfnVMPowerOff || RT_VALID_PTR(m->pReg->pfnVMPowerOff))
1394 && (!m->pReg->pfnQueryObject || RT_VALID_PTR(m->pReg->pfnQueryObject))
1395 )
1396 {
1397 /*
1398 * We're good!
1399 */
1400 m->fUsable = true;
1401 m->strWhyUnusable.setNull();
1402 return;
1403 }
1404#endif
1405
1406 m->strWhyUnusable = tr("The registration structure contains one or more invalid function pointers");
1407 }
1408 else
1409 m->strWhyUnusable.printf(tr("Unsupported registration structure version %u.%u"),
1410 RT_HIWORD(m->pReg->u32Version), RT_LOWORD(m->pReg->u32Version));
1411 }
1412 else
1413 m->strWhyUnusable.printf(tr("%s returned %Rrc, pReg=%p ErrInfo='%s'"),
1414 pszMainEntryPoint, vrc, m->pReg, ErrInfo.Core.pszMsg);
1415 m->pReg = NULL;
1416 }
1417 else
1418 m->strWhyUnusable.printf(tr("Failed to resolve exported symbol '%s' in the main module: %Rrc"),
1419 pszMainEntryPoint, vrc);
1420
1421 RTLdrClose(m->hMainMod);
1422 m->hMainMod = NIL_RTLDRMOD;
1423}
1424
1425/**
1426 * Finds a module.
1427 *
1428 * @returns true if found, false if not.
1429 * @param a_pszName The module base name (no extension).
1430 * @param a_pszExt The extension. If NULL we use default
1431 * extensions.
1432 * @param a_enmKind The kind of module to locate.
1433 * @param a_pStrFound Where to return the path to the module we've
1434 * found.
1435 * @param a_pfNative Where to return whether this is a native module
1436 * or an agnostic one. Optional.
1437 * @param a_pObjInfo Where to return the file system object info for
1438 * the module. Optional.
1439 */
1440bool ExtPack::i_findModule(const char *a_pszName, const char *a_pszExt, VBOXEXTPACKMODKIND a_enmKind,
1441 Utf8Str *a_pStrFound, bool *a_pfNative, PRTFSOBJINFO a_pObjInfo) const
1442{
1443 /*
1444 * Try the native path first.
1445 */
1446 char szPath[RTPATH_MAX];
1447 int vrc = RTPathJoin(szPath, sizeof(szPath), m->strExtPackPath.c_str(), RTBldCfgTargetDotArch());
1448 AssertLogRelRCReturn(vrc, false);
1449 vrc = RTPathAppend(szPath, sizeof(szPath), a_pszName);
1450 AssertLogRelRCReturn(vrc, false);
1451 if (!a_pszExt)
1452 {
1453 const char *pszDefExt;
1454 switch (a_enmKind)
1455 {
1456 case VBOXEXTPACKMODKIND_RC: pszDefExt = ".rc"; break;
1457 case VBOXEXTPACKMODKIND_R0: pszDefExt = ".r0"; break;
1458 case VBOXEXTPACKMODKIND_R3: pszDefExt = RTLdrGetSuff(); break;
1459 default:
1460 AssertFailedReturn(false);
1461 }
1462 vrc = RTStrCat(szPath, sizeof(szPath), pszDefExt);
1463 AssertLogRelRCReturn(vrc, false);
1464 }
1465
1466 RTFSOBJINFO ObjInfo;
1467 if (!a_pObjInfo)
1468 a_pObjInfo = &ObjInfo;
1469 vrc = RTPathQueryInfo(szPath, a_pObjInfo, RTFSOBJATTRADD_UNIX);
1470 if (RT_SUCCESS(vrc) && RTFS_IS_FILE(a_pObjInfo->Attr.fMode))
1471 {
1472 if (a_pfNative)
1473 *a_pfNative = true;
1474 *a_pStrFound = szPath;
1475 return true;
1476 }
1477
1478 /*
1479 * Try the platform agnostic modules.
1480 */
1481 /* gcc.x86/module.rel */
1482 char szSubDir[32];
1483 RTStrPrintf(szSubDir, sizeof(szSubDir), "%s.%s", RTBldCfgCompiler(), RTBldCfgTargetArch());
1484 vrc = RTPathJoin(szPath, sizeof(szPath), m->strExtPackPath.c_str(), szSubDir);
1485 AssertLogRelRCReturn(vrc, false);
1486 vrc = RTPathAppend(szPath, sizeof(szPath), a_pszName);
1487 AssertLogRelRCReturn(vrc, false);
1488 if (!a_pszExt)
1489 {
1490 vrc = RTStrCat(szPath, sizeof(szPath), ".rel");
1491 AssertLogRelRCReturn(vrc, false);
1492 }
1493 vrc = RTPathQueryInfo(szPath, a_pObjInfo, RTFSOBJATTRADD_UNIX);
1494 if (RT_SUCCESS(vrc) && RTFS_IS_FILE(a_pObjInfo->Attr.fMode))
1495 {
1496 if (a_pfNative)
1497 *a_pfNative = false;
1498 *a_pStrFound = szPath;
1499 return true;
1500 }
1501
1502 /* x86/module.rel */
1503 vrc = RTPathJoin(szPath, sizeof(szPath), m->strExtPackPath.c_str(), RTBldCfgTargetArch());
1504 AssertLogRelRCReturn(vrc, false);
1505 vrc = RTPathAppend(szPath, sizeof(szPath), a_pszName);
1506 AssertLogRelRCReturn(vrc, false);
1507 if (!a_pszExt)
1508 {
1509 vrc = RTStrCat(szPath, sizeof(szPath), ".rel");
1510 AssertLogRelRCReturn(vrc, false);
1511 }
1512 vrc = RTPathQueryInfo(szPath, a_pObjInfo, RTFSOBJATTRADD_UNIX);
1513 if (RT_SUCCESS(vrc) && RTFS_IS_FILE(a_pObjInfo->Attr.fMode))
1514 {
1515 if (a_pfNative)
1516 *a_pfNative = false;
1517 *a_pStrFound = szPath;
1518 return true;
1519 }
1520
1521 return false;
1522}
1523
1524/**
1525 * Compares two file system object info structures.
1526 *
1527 * @returns true if equal, false if not.
1528 * @param pObjInfo1 The first.
1529 * @param pObjInfo2 The second.
1530 * @todo IPRT should do this, really.
1531 */
1532/* static */ bool ExtPack::i_objinfoIsEqual(PCRTFSOBJINFO pObjInfo1, PCRTFSOBJINFO pObjInfo2)
1533{
1534 if (!RTTimeSpecIsEqual(&pObjInfo1->ModificationTime, &pObjInfo2->ModificationTime))
1535 return false;
1536 if (!RTTimeSpecIsEqual(&pObjInfo1->ChangeTime, &pObjInfo2->ChangeTime))
1537 return false;
1538 if (!RTTimeSpecIsEqual(&pObjInfo1->BirthTime, &pObjInfo2->BirthTime))
1539 return false;
1540 if (pObjInfo1->cbObject != pObjInfo2->cbObject)
1541 return false;
1542 if (pObjInfo1->Attr.fMode != pObjInfo2->Attr.fMode)
1543 return false;
1544 if (pObjInfo1->Attr.enmAdditional == pObjInfo2->Attr.enmAdditional)
1545 {
1546 switch (pObjInfo1->Attr.enmAdditional)
1547 {
1548 case RTFSOBJATTRADD_UNIX:
1549 if (pObjInfo1->Attr.u.Unix.uid != pObjInfo2->Attr.u.Unix.uid)
1550 return false;
1551 if (pObjInfo1->Attr.u.Unix.gid != pObjInfo2->Attr.u.Unix.gid)
1552 return false;
1553 if (pObjInfo1->Attr.u.Unix.INodeIdDevice != pObjInfo2->Attr.u.Unix.INodeIdDevice)
1554 return false;
1555 if (pObjInfo1->Attr.u.Unix.INodeId != pObjInfo2->Attr.u.Unix.INodeId)
1556 return false;
1557 if (pObjInfo1->Attr.u.Unix.GenerationId != pObjInfo2->Attr.u.Unix.GenerationId)
1558 return false;
1559 break;
1560 default:
1561 break;
1562 }
1563 }
1564 return true;
1565}
1566
1567
1568/**
1569 * @interface_method_impl{VBOXEXTPACKHLP,pfnFindModule}
1570 */
1571/*static*/ DECLCALLBACK(int)
1572ExtPack::i_hlpFindModule(PCVBOXEXTPACKHLP pHlp, const char *pszName, const char *pszExt, VBOXEXTPACKMODKIND enmKind,
1573 char *pszFound, size_t cbFound, bool *pfNative)
1574{
1575 /*
1576 * Validate the input and get our bearings.
1577 */
1578 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
1579 AssertPtrNullReturn(pszExt, VERR_INVALID_POINTER);
1580 AssertPtrReturn(pszFound, VERR_INVALID_POINTER);
1581 AssertPtrNullReturn(pfNative, VERR_INVALID_POINTER);
1582 AssertReturn(enmKind > VBOXEXTPACKMODKIND_INVALID && enmKind < VBOXEXTPACKMODKIND_END, VERR_INVALID_PARAMETER);
1583
1584 AssertPtrReturn(pHlp, VERR_INVALID_POINTER);
1585 AssertReturn(pHlp->u32Version == VBOXEXTPACKHLP_VERSION, VERR_INVALID_POINTER);
1586 ExtPack::Data *m = RT_FROM_CPP_MEMBER(pHlp, Data, Hlp);
1587 AssertPtrReturn(m, VERR_INVALID_POINTER);
1588 ExtPack *pThis = m->pThis;
1589 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1590
1591 /*
1592 * This is just a wrapper around findModule.
1593 */
1594 Utf8Str strFound;
1595 if (pThis->i_findModule(pszName, pszExt, enmKind, &strFound, pfNative, NULL))
1596 return RTStrCopy(pszFound, cbFound, strFound.c_str());
1597 return VERR_FILE_NOT_FOUND;
1598}
1599
1600/*static*/ DECLCALLBACK(int)
1601ExtPack::i_hlpGetFilePath(PCVBOXEXTPACKHLP pHlp, const char *pszFilename, char *pszPath, size_t cbPath)
1602{
1603 /*
1604 * Validate the input and get our bearings.
1605 */
1606 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
1607 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
1608 AssertReturn(cbPath > 0, VERR_BUFFER_OVERFLOW);
1609
1610 AssertPtrReturn(pHlp, VERR_INVALID_POINTER);
1611 AssertReturn(pHlp->u32Version == VBOXEXTPACKHLP_VERSION, VERR_INVALID_POINTER);
1612 ExtPack::Data *m = RT_FROM_CPP_MEMBER(pHlp, Data, Hlp);
1613 AssertPtrReturn(m, VERR_INVALID_POINTER);
1614 ExtPack *pThis = m->pThis;
1615 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1616
1617 /*
1618 * This is a simple RTPathJoin, no checking if things exists or anything.
1619 */
1620 int vrc = RTPathJoin(pszPath, cbPath, pThis->m->strExtPackPath.c_str(), pszFilename);
1621 if (RT_FAILURE(vrc))
1622 RT_BZERO(pszPath, cbPath);
1623 return vrc;
1624}
1625
1626/*static*/ DECLCALLBACK(VBOXEXTPACKCTX)
1627ExtPack::i_hlpGetContext(PCVBOXEXTPACKHLP pHlp)
1628{
1629 /*
1630 * Validate the input and get our bearings.
1631 */
1632 AssertPtrReturn(pHlp, VBOXEXTPACKCTX_INVALID);
1633 AssertReturn(pHlp->u32Version == VBOXEXTPACKHLP_VERSION, VBOXEXTPACKCTX_INVALID);
1634 ExtPack::Data *m = RT_FROM_CPP_MEMBER(pHlp, Data, Hlp);
1635 AssertPtrReturn(m, VBOXEXTPACKCTX_INVALID);
1636 ExtPack *pThis = m->pThis;
1637 AssertPtrReturn(pThis, VBOXEXTPACKCTX_INVALID);
1638
1639 return pThis->m->enmContext;
1640}
1641
1642/*static*/ DECLCALLBACK(int)
1643ExtPack::i_hlpLoadHGCMService(PCVBOXEXTPACKHLP pHlp, VBOXEXTPACK_IF_CS(IConsole) *pConsole,
1644 const char *pszServiceLibrary, const char *pszServiceName)
1645{
1646#ifdef VBOX_COM_INPROC
1647 /*
1648 * Validate the input and get our bearings.
1649 */
1650 AssertPtrReturn(pszServiceLibrary, VERR_INVALID_POINTER);
1651 AssertPtrReturn(pszServiceName, VERR_INVALID_POINTER);
1652
1653 AssertPtrReturn(pHlp, VERR_INVALID_POINTER);
1654 AssertReturn(pHlp->u32Version == VBOXEXTPACKHLP_VERSION, VERR_INVALID_POINTER);
1655 ExtPack::Data *m = RT_FROM_CPP_MEMBER(pHlp, Data, Hlp);
1656 AssertPtrReturn(m, VERR_INVALID_POINTER);
1657 ExtPack *pThis = m->pThis;
1658 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1659 AssertPtrReturn(pConsole, VERR_INVALID_POINTER);
1660
1661 Console *pCon = (Console *)pConsole;
1662 return pCon->i_hgcmLoadService(pszServiceLibrary, pszServiceName);
1663#else
1664 NOREF(pHlp); NOREF(pConsole); NOREF(pszServiceLibrary); NOREF(pszServiceName);
1665 return VERR_INVALID_STATE;
1666#endif
1667}
1668
1669/*static*/ DECLCALLBACK(int)
1670ExtPack::i_hlpLoadVDPlugin(PCVBOXEXTPACKHLP pHlp, VBOXEXTPACK_IF_CS(IVirtualBox) *pVirtualBox, const char *pszPluginLibrary)
1671{
1672#ifndef VBOX_COM_INPROC
1673 /*
1674 * Validate the input and get our bearings.
1675 */
1676 AssertPtrReturn(pszPluginLibrary, VERR_INVALID_POINTER);
1677
1678 AssertPtrReturn(pHlp, VERR_INVALID_POINTER);
1679 AssertReturn(pHlp->u32Version == VBOXEXTPACKHLP_VERSION, VERR_INVALID_POINTER);
1680 ExtPack::Data *m = RT_FROM_CPP_MEMBER(pHlp, Data, Hlp);
1681 AssertPtrReturn(m, VERR_INVALID_POINTER);
1682 ExtPack *pThis = m->pThis;
1683 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1684 AssertPtrReturn(pVirtualBox, VERR_INVALID_POINTER);
1685
1686 VirtualBox *pVBox = (VirtualBox *)pVirtualBox;
1687 return pVBox->i_loadVDPlugin(pszPluginLibrary);
1688#else
1689 NOREF(pHlp); NOREF(pVirtualBox); NOREF(pszPluginLibrary);
1690 return VERR_INVALID_STATE;
1691#endif
1692}
1693
1694/*static*/ DECLCALLBACK(int)
1695ExtPack::i_hlpUnloadVDPlugin(PCVBOXEXTPACKHLP pHlp, VBOXEXTPACK_IF_CS(IVirtualBox) *pVirtualBox, const char *pszPluginLibrary)
1696{
1697#ifndef VBOX_COM_INPROC
1698 /*
1699 * Validate the input and get our bearings.
1700 */
1701 AssertPtrReturn(pszPluginLibrary, VERR_INVALID_POINTER);
1702
1703 AssertPtrReturn(pHlp, VERR_INVALID_POINTER);
1704 AssertReturn(pHlp->u32Version == VBOXEXTPACKHLP_VERSION, VERR_INVALID_POINTER);
1705 ExtPack::Data *m = RT_FROM_CPP_MEMBER(pHlp, Data, Hlp);
1706 AssertPtrReturn(m, VERR_INVALID_POINTER);
1707 ExtPack *pThis = m->pThis;
1708 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1709 AssertPtrReturn(pVirtualBox, VERR_INVALID_POINTER);
1710
1711 VirtualBox *pVBox = (VirtualBox *)pVirtualBox;
1712 return pVBox->i_unloadVDPlugin(pszPluginLibrary);
1713#else
1714 NOREF(pHlp); NOREF(pVirtualBox); NOREF(pszPluginLibrary);
1715 return VERR_INVALID_STATE;
1716#endif
1717}
1718
1719/*static*/ DECLCALLBACK(int)
1720ExtPack::i_hlpReservedN(PCVBOXEXTPACKHLP pHlp)
1721{
1722 /*
1723 * Validate the input and get our bearings.
1724 */
1725 AssertPtrReturn(pHlp, VERR_INVALID_POINTER);
1726 AssertReturn(pHlp->u32Version == VBOXEXTPACKHLP_VERSION, VERR_INVALID_POINTER);
1727 ExtPack::Data *m = RT_FROM_CPP_MEMBER(pHlp, Data, Hlp);
1728 AssertPtrReturn(m, VERR_INVALID_POINTER);
1729 ExtPack *pThis = m->pThis;
1730 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1731
1732 return VERR_NOT_IMPLEMENTED;
1733}
1734
1735
1736
1737
1738HRESULT ExtPack::getName(com::Utf8Str &aName)
1739{
1740 aName = m->Desc.strName;
1741 return S_OK;
1742}
1743
1744HRESULT ExtPack::getDescription(com::Utf8Str &aDescription)
1745{
1746 aDescription = m->Desc.strDescription;
1747 return S_OK;
1748}
1749
1750HRESULT ExtPack::getVersion(com::Utf8Str &aVersion)
1751{
1752 aVersion = m->Desc.strVersion;
1753 return S_OK;
1754}
1755
1756HRESULT ExtPack::getRevision(ULONG *aRevision)
1757{
1758 *aRevision = m->Desc.uRevision;
1759 return S_OK;
1760}
1761
1762HRESULT ExtPack::getEdition(com::Utf8Str &aEdition)
1763{
1764 aEdition = m->Desc.strEdition;
1765 return S_OK;
1766}
1767
1768HRESULT ExtPack::getVRDEModule(com::Utf8Str &aVRDEModule)
1769{
1770 aVRDEModule = m->Desc.strVrdeModule;
1771 return S_OK;
1772}
1773
1774HRESULT ExtPack::getPlugIns(std::vector<ComPtr<IExtPackPlugIn> > &aPlugIns)
1775{
1776 /** @todo implement plug-ins. */
1777 NOREF(aPlugIns);
1778 ReturnComNotImplemented();
1779}
1780
1781HRESULT ExtPack::getUsable(BOOL *aUsable)
1782{
1783 *aUsable = m->fUsable;
1784 return S_OK;
1785}
1786
1787HRESULT ExtPack::getWhyUnusable(com::Utf8Str &aWhyUnusable)
1788{
1789 aWhyUnusable = m->strWhyUnusable;
1790 return S_OK;
1791}
1792
1793HRESULT ExtPack::getShowLicense(BOOL *aShowLicense)
1794{
1795 *aShowLicense = m->Desc.fShowLicense;
1796 return S_OK;
1797}
1798
1799HRESULT ExtPack::getLicense(com::Utf8Str &aLicense)
1800{
1801 Utf8Str strHtml("html");
1802 Utf8Str str("");
1803 return queryLicense(str, str, strHtml, aLicense);
1804}
1805
1806HRESULT ExtPack::queryLicense(const com::Utf8Str &aPreferredLocale, const com::Utf8Str &aPreferredLanguage,
1807 const com::Utf8Str &aFormat, com::Utf8Str &aLicenseText)
1808{
1809 HRESULT hrc = S_OK;
1810
1811 /*
1812 * Validate input.
1813 */
1814 if (aPreferredLocale.length() != 2 && aPreferredLocale.length() != 0)
1815 return setError(E_FAIL, tr("The preferred locale is a two character string or empty."));
1816
1817 if (aPreferredLanguage.length() != 2 && aPreferredLanguage.length() != 0)
1818 return setError(E_FAIL, tr("The preferred lanuage is a two character string or empty."));
1819
1820 if ( !aFormat.equals("html")
1821 && !aFormat.equals("rtf")
1822 && !aFormat.equals("txt"))
1823 return setError(E_FAIL, tr("The license format can only have the values 'html', 'rtf' and 'txt'."));
1824
1825 /*
1826 * Combine the options to form a file name before locking down anything.
1827 */
1828 char szName[sizeof(VBOX_EXTPACK_LICENSE_NAME_PREFIX "-de_DE.html") + 2];
1829 if (aPreferredLocale.isNotEmpty() && aPreferredLanguage.isNotEmpty())
1830 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX "-%s_%s.%s",
1831 aPreferredLocale.c_str(), aPreferredLanguage.c_str(), aFormat.c_str());
1832 else if (aPreferredLocale.isNotEmpty())
1833 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX "-%s.%s",
1834 aPreferredLocale.c_str(), aFormat.c_str());
1835 else if (aPreferredLanguage.isNotEmpty())
1836 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX "-_%s.%s",
1837 aPreferredLocale.c_str(), aFormat.c_str());
1838 else
1839 RTStrPrintf(szName, sizeof(szName), VBOX_EXTPACK_LICENSE_NAME_PREFIX ".%s",
1840 aFormat.c_str());
1841
1842 /*
1843 * Effectuate the query.
1844 */
1845 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS); /* paranoia */
1846
1847 if (!m->fUsable)
1848 hrc = setError(E_FAIL, tr("%s"), m->strWhyUnusable.c_str());
1849 else
1850 {
1851 char szPath[RTPATH_MAX];
1852 int vrc = RTPathJoin(szPath, sizeof(szPath), m->strExtPackPath.c_str(), szName);
1853 if (RT_SUCCESS(vrc))
1854 {
1855 void *pvFile;
1856 size_t cbFile;
1857 vrc = RTFileReadAllEx(szPath, 0, RTFOFF_MAX, RTFILE_RDALL_O_DENY_READ, &pvFile, &cbFile);
1858 if (RT_SUCCESS(vrc))
1859 {
1860 Bstr bstrLicense((const char *)pvFile, cbFile);
1861 if (bstrLicense.isNotEmpty())
1862 {
1863 aLicenseText = Utf8Str(bstrLicense);
1864 hrc = S_OK;
1865 }
1866 else
1867 hrc = setError(VBOX_E_IPRT_ERROR, tr("The license file '%s' is empty or contains invalid UTF-8 encoding"),
1868 szPath);
1869 RTFileReadAllFree(pvFile, cbFile);
1870 }
1871 else if (vrc == VERR_FILE_NOT_FOUND || vrc == VERR_PATH_NOT_FOUND)
1872 hrc = setErrorBoth(VBOX_E_OBJECT_NOT_FOUND, vrc, tr("The license file '%s' was not found in extension pack '%s'"),
1873 szName, m->Desc.strName.c_str());
1874 else
1875 hrc = setErrorBoth(VBOX_E_FILE_ERROR, vrc, tr("Failed to open the license file '%s': %Rrc"), szPath, vrc);
1876 }
1877 else
1878 hrc = setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("RTPathJoin failed: %Rrc"), vrc);
1879 }
1880 return hrc;
1881}
1882
1883HRESULT ExtPack::queryObject(const com::Utf8Str &aObjUuid, ComPtr<IUnknown> &aReturnInterface)
1884{
1885 com::Guid ObjectId;
1886 CheckComArgGuid(aObjUuid, ObjectId);
1887
1888 HRESULT hrc S_OK;
1889
1890 if ( m->pReg
1891 && m->pReg->pfnQueryObject)
1892 {
1893 void *pvUnknown = m->pReg->pfnQueryObject(m->pReg, ObjectId.raw());
1894 if (pvUnknown)
1895 aReturnInterface = (IUnknown *)pvUnknown;
1896 else
1897 hrc = E_NOINTERFACE;
1898 }
1899 else
1900 hrc = E_NOINTERFACE;
1901 return hrc;
1902}
1903
1904DEFINE_EMPTY_CTOR_DTOR(ExtPackManager)
1905
1906/**
1907 * Called by ComObjPtr::createObject when creating the object.
1908 *
1909 * Just initialize the basic object state, do the rest in init().
1910 *
1911 * @returns S_OK.
1912 */
1913HRESULT ExtPackManager::FinalConstruct()
1914{
1915 m = NULL;
1916 return BaseFinalConstruct();
1917}
1918
1919/**
1920 * Initializes the extension pack manager.
1921 *
1922 * @returns COM status code.
1923 * @param a_pVirtualBox Pointer to the VirtualBox object.
1924 * @param a_enmContext The context we're in.
1925 */
1926HRESULT ExtPackManager::initExtPackManager(VirtualBox *a_pVirtualBox, VBOXEXTPACKCTX a_enmContext)
1927{
1928 AutoInitSpan autoInitSpan(this);
1929 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1930
1931 /*
1932 * Figure some stuff out before creating the instance data.
1933 */
1934 char szBaseDir[RTPATH_MAX];
1935 int rc = RTPathAppPrivateArchTop(szBaseDir, sizeof(szBaseDir));
1936 AssertLogRelRCReturn(rc, E_FAIL);
1937 rc = RTPathAppend(szBaseDir, sizeof(szBaseDir), VBOX_EXTPACK_INSTALL_DIR);
1938 AssertLogRelRCReturn(rc, E_FAIL);
1939
1940 char szCertificatDir[RTPATH_MAX];
1941 rc = RTPathAppPrivateNoArch(szCertificatDir, sizeof(szCertificatDir));
1942 AssertLogRelRCReturn(rc, E_FAIL);
1943 rc = RTPathAppend(szCertificatDir, sizeof(szCertificatDir), VBOX_EXTPACK_CERT_DIR);
1944 AssertLogRelRCReturn(rc, E_FAIL);
1945
1946 /*
1947 * Allocate and initialize the instance data.
1948 */
1949 m = new Data;
1950 m->strBaseDir = szBaseDir;
1951 m->strCertificatDirPath = szCertificatDir;
1952 m->enmContext = a_enmContext;
1953#ifndef VBOX_COM_INPROC
1954 m->pVirtualBox = a_pVirtualBox;
1955#else
1956 RT_NOREF_PV(a_pVirtualBox);
1957#endif
1958
1959 /*
1960 * Go looking for extensions. The RTDirOpen may fail if nothing has been
1961 * installed yet, or if root is paranoid and has revoked our access to them.
1962 *
1963 * We ASSUME that there are no files, directories or stuff in the directory
1964 * that exceed the max name length in RTDIRENTRYEX.
1965 */
1966 HRESULT hrc = S_OK;
1967 RTDIR hDir;
1968 int vrc = RTDirOpen(&hDir, szBaseDir);
1969 if (RT_SUCCESS(vrc))
1970 {
1971 for (;;)
1972 {
1973 RTDIRENTRYEX Entry;
1974 vrc = RTDirReadEx(hDir, &Entry, NULL /*pcbDirEntry*/, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
1975 if (RT_FAILURE(vrc))
1976 {
1977 AssertLogRelMsg(vrc == VERR_NO_MORE_FILES, ("%Rrc\n", vrc));
1978 break;
1979 }
1980 if ( RTFS_IS_DIRECTORY(Entry.Info.Attr.fMode)
1981 && strcmp(Entry.szName, ".") != 0
1982 && strcmp(Entry.szName, "..") != 0
1983 && VBoxExtPackIsValidMangledName(Entry.szName) )
1984 {
1985 /*
1986 * All directories are extensions, the shall be nothing but
1987 * extensions in this subdirectory.
1988 */
1989 char szExtPackDir[RTPATH_MAX];
1990 vrc = RTPathJoin(szExtPackDir, sizeof(szExtPackDir), m->strBaseDir.c_str(), Entry.szName);
1991 AssertLogRelRC(vrc);
1992 if (RT_SUCCESS(vrc))
1993 {
1994 RTCString *pstrName = VBoxExtPackUnmangleName(Entry.szName, RTSTR_MAX);
1995 AssertLogRel(pstrName);
1996 if (pstrName)
1997 {
1998 ComObjPtr<ExtPack> NewExtPack;
1999 HRESULT hrc2 = NewExtPack.createObject();
2000 if (SUCCEEDED(hrc2))
2001 hrc2 = NewExtPack->initWithDir(a_enmContext, pstrName->c_str(), szExtPackDir);
2002 delete pstrName;
2003 if (SUCCEEDED(hrc2))
2004 m->llInstalledExtPacks.push_back(NewExtPack);
2005 else if (SUCCEEDED(rc))
2006 hrc = hrc2;
2007 }
2008 else
2009 hrc = E_UNEXPECTED;
2010 }
2011 else
2012 hrc = E_UNEXPECTED;
2013 }
2014 }
2015 RTDirClose(hDir);
2016 }
2017 /* else: ignore, the directory probably does not exist or something. */
2018
2019 if (SUCCEEDED(hrc))
2020 autoInitSpan.setSucceeded();
2021 return hrc;
2022}
2023
2024/**
2025 * COM cruft.
2026 */
2027void ExtPackManager::FinalRelease()
2028{
2029 uninit();
2030 BaseFinalRelease();
2031}
2032
2033/**
2034 * Do the actual cleanup.
2035 */
2036void ExtPackManager::uninit()
2037{
2038 /* Enclose the state transition Ready->InUninit->NotReady */
2039 AutoUninitSpan autoUninitSpan(this);
2040 if (!autoUninitSpan.uninitDone() && m != NULL)
2041 {
2042 delete m;
2043 m = NULL;
2044 }
2045}
2046
2047HRESULT ExtPackManager::getInstalledExtPacks(std::vector<ComPtr<IExtPack> > &aInstalledExtPacks)
2048{
2049 Assert(m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON);
2050
2051 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2052
2053
2054 SafeIfaceArray<IExtPack> SaExtPacks(m->llInstalledExtPacks);
2055 aInstalledExtPacks.resize(SaExtPacks.size());
2056 for(size_t i = 0; i < SaExtPacks.size(); ++i)
2057 aInstalledExtPacks[i] = SaExtPacks[i];
2058
2059 return S_OK;
2060}
2061
2062HRESULT ExtPackManager::find(const com::Utf8Str &aName, ComPtr<IExtPack> &aReturnData)
2063{
2064 HRESULT hrc = S_OK;
2065
2066 Assert(m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON);
2067
2068 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2069
2070 ComPtr<ExtPack> ptrExtPack = i_findExtPack(aName.c_str());
2071 if (!ptrExtPack.isNull())
2072 ptrExtPack.queryInterfaceTo(aReturnData.asOutParam());
2073 else
2074 hrc = VBOX_E_OBJECT_NOT_FOUND;
2075
2076 return hrc;
2077}
2078
2079HRESULT ExtPackManager::openExtPackFile(const com::Utf8Str &aPath, ComPtr<IExtPackFile> &aFile)
2080{
2081 AssertReturn(m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON, E_UNEXPECTED);
2082
2083#ifndef VBOX_COM_INPROC
2084 /* The API can optionally take a ::SHA-256=<hex-digest> attribute at the
2085 end of the file name. This is just a temporary measure for
2086 backporting, in 4.2 we'll add another parameter to the method. */
2087 Utf8Str strTarball;
2088 Utf8Str strDigest;
2089 size_t offSha256 = aPath.find("::SHA-256=");
2090 if (offSha256 == Utf8Str::npos)
2091 strTarball = aPath;
2092 else
2093 {
2094 strTarball = aPath.substr(0, offSha256);
2095 strDigest = aPath.substr(offSha256 + sizeof("::SHA-256=") - 1);
2096 }
2097
2098 ComObjPtr<ExtPackFile> NewExtPackFile;
2099 HRESULT hrc = NewExtPackFile.createObject();
2100 if (SUCCEEDED(hrc))
2101 hrc = NewExtPackFile->initWithFile(strTarball.c_str(), strDigest.c_str(), this, m->pVirtualBox);
2102 if (SUCCEEDED(hrc))
2103 NewExtPackFile.queryInterfaceTo(aFile.asOutParam());
2104
2105 return hrc;
2106#else
2107 RT_NOREF(aPath, aFile);
2108 return E_NOTIMPL;
2109#endif
2110}
2111
2112HRESULT ExtPackManager::uninstall(const com::Utf8Str &aName, BOOL aForcedRemoval,
2113 const com::Utf8Str &aDisplayInfo, ComPtr<IProgress> &aProgress)
2114{
2115 Assert(m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON);
2116
2117#ifndef VBOX_COM_INPROC
2118
2119 HRESULT hrc;
2120 ExtPackUninstallTask *pTask = NULL;
2121 try
2122 {
2123 pTask = new ExtPackUninstallTask();
2124 hrc = pTask->Init(this, aName, aForcedRemoval != FALSE, aDisplayInfo);
2125 if (SUCCEEDED(hrc))
2126 {
2127 ComPtr<Progress> ptrProgress = pTask->ptrProgress;
2128 hrc = pTask->createThreadWithType(RTTHREADTYPE_DEFAULT);
2129 pTask = NULL; /* always consumed by createThread */
2130 if (SUCCEEDED(hrc))
2131 hrc = ptrProgress.queryInterfaceTo(aProgress.asOutParam());
2132 else
2133 hrc = setError(VBOX_E_IPRT_ERROR,
2134 tr("Starting thread for an extension pack uninstallation failed with %Rrc"), hrc);
2135 }
2136 else
2137 hrc = setError(hrc, tr("Looks like creating a progress object for ExtraPackUninstallTask object failed"));
2138 }
2139 catch (std::bad_alloc &)
2140 {
2141 hrc = E_OUTOFMEMORY;
2142 }
2143 catch (HRESULT hrcXcpt)
2144 {
2145 LogFlowThisFunc(("Exception was caught in the function ExtPackManager::uninstall()\n"));
2146 hrc = hrcXcpt;
2147 }
2148 if (pTask)
2149 delete pTask;
2150 return hrc;
2151#else
2152 RT_NOREF(aName, aForcedRemoval, aDisplayInfo, aProgress);
2153 return E_NOTIMPL;
2154#endif
2155}
2156
2157HRESULT ExtPackManager::cleanup(void)
2158{
2159 Assert(m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON);
2160
2161 AutoCaller autoCaller(this);
2162 HRESULT hrc = autoCaller.rc();
2163 if (SUCCEEDED(hrc))
2164 {
2165 /*
2166 * Run the set-uid-to-root binary that performs the cleanup.
2167 *
2168 * Take the write lock to prevent conflicts with other calls to this
2169 * VBoxSVC instance.
2170 */
2171 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2172 hrc = i_runSetUidToRootHelper(NULL,
2173 "cleanup",
2174 "--base-dir", m->strBaseDir.c_str(),
2175 (const char *)NULL);
2176 }
2177
2178 return hrc;
2179}
2180
2181HRESULT ExtPackManager::queryAllPlugInsForFrontend(const com::Utf8Str &aFrontendName, std::vector<com::Utf8Str> &aPlugInModules)
2182{
2183 NOREF(aFrontendName);
2184 aPlugInModules.resize(0);
2185 return S_OK;
2186}
2187
2188HRESULT ExtPackManager::isExtPackUsable(const com::Utf8Str &aName, BOOL *aUsable)
2189{
2190 *aUsable = i_isExtPackUsable(aName.c_str());
2191 return S_OK;
2192}
2193
2194/**
2195 * Finds the success indicator string in the stderr output ofr hte helper app.
2196 *
2197 * @returns Pointer to the indicator.
2198 * @param psz The stderr output string. Can be NULL.
2199 * @param cch The size of the string.
2200 */
2201static char *findSuccessIndicator(char *psz, size_t cch)
2202{
2203 static const char s_szSuccessInd[] = "rcExit=RTEXITCODE_SUCCESS";
2204 Assert(!cch || strlen(psz) == cch);
2205 if (cch < sizeof(s_szSuccessInd) - 1)
2206 return NULL;
2207 char *pszInd = &psz[cch - sizeof(s_szSuccessInd) + 1];
2208 if (strcmp(s_szSuccessInd, pszInd))
2209 return NULL;
2210 return pszInd;
2211}
2212
2213/**
2214 * Runs the helper application that does the privileged operations.
2215 *
2216 * @returns S_OK or a failure status with error information set.
2217 * @param a_pstrDisplayInfo Platform specific display info hacks.
2218 * @param a_pszCommand The command to execute.
2219 * @param ... The argument strings that goes along with the
2220 * command. Maximum is about 16. Terminated by a
2221 * NULL.
2222 */
2223HRESULT ExtPackManager::i_runSetUidToRootHelper(Utf8Str const *a_pstrDisplayInfo, const char *a_pszCommand, ...)
2224{
2225 /*
2226 * Calculate the path to the helper application.
2227 */
2228 char szExecName[RTPATH_MAX];
2229 int vrc = RTPathAppPrivateArch(szExecName, sizeof(szExecName));
2230 AssertLogRelRCReturn(vrc, E_UNEXPECTED);
2231
2232 vrc = RTPathAppend(szExecName, sizeof(szExecName), VBOX_EXTPACK_HELPER_NAME);
2233 AssertLogRelRCReturn(vrc, E_UNEXPECTED);
2234
2235 /*
2236 * Convert the variable argument list to a RTProcCreate argument vector.
2237 */
2238 const char *apszArgs[20];
2239 unsigned cArgs = 0;
2240
2241 LogRel(("ExtPack: Executing '%s'", szExecName));
2242 apszArgs[cArgs++] = &szExecName[0];
2243
2244 if ( a_pstrDisplayInfo
2245 && a_pstrDisplayInfo->isNotEmpty())
2246 {
2247 LogRel((" '--display-info-hack' '%s'", a_pstrDisplayInfo->c_str()));
2248 apszArgs[cArgs++] = "--display-info-hack";
2249 apszArgs[cArgs++] = a_pstrDisplayInfo->c_str();
2250 }
2251
2252 LogRel(("'%s'", a_pszCommand));
2253 apszArgs[cArgs++] = a_pszCommand;
2254
2255 va_list va;
2256 va_start(va, a_pszCommand);
2257 const char *pszLastArg;
2258 for (;;)
2259 {
2260 AssertReturn(cArgs < RT_ELEMENTS(apszArgs) - 1, E_UNEXPECTED);
2261 pszLastArg = va_arg(va, const char *);
2262 if (!pszLastArg)
2263 break;
2264 LogRel((" '%s'", pszLastArg));
2265 apszArgs[cArgs++] = pszLastArg;
2266 };
2267 va_end(va);
2268
2269 LogRel(("\n"));
2270 apszArgs[cArgs] = NULL;
2271
2272 /*
2273 * Create a PIPE which we attach to stderr so that we can read the error
2274 * message on failure and report it back to the caller.
2275 */
2276 RTPIPE hPipeR;
2277 RTHANDLE hStdErrPipe;
2278 hStdErrPipe.enmType = RTHANDLETYPE_PIPE;
2279 vrc = RTPipeCreate(&hPipeR, &hStdErrPipe.u.hPipe, RTPIPE_C_INHERIT_WRITE);
2280 AssertLogRelRCReturn(vrc, E_UNEXPECTED);
2281
2282 /*
2283 * Spawn the process.
2284 */
2285 HRESULT hrc;
2286 RTPROCESS hProcess;
2287 vrc = RTProcCreateEx(szExecName,
2288 apszArgs,
2289 RTENV_DEFAULT,
2290 0 /*fFlags*/,
2291 NULL /*phStdIn*/,
2292 NULL /*phStdOut*/,
2293 &hStdErrPipe,
2294 NULL /*pszAsUser*/,
2295 NULL /*pszPassword*/,
2296 &hProcess);
2297 if (RT_SUCCESS(vrc))
2298 {
2299 vrc = RTPipeClose(hStdErrPipe.u.hPipe);
2300 hStdErrPipe.u.hPipe = NIL_RTPIPE;
2301
2302 /*
2303 * Read the pipe output until the process completes.
2304 */
2305 RTPROCSTATUS ProcStatus = { -42, RTPROCEXITREASON_ABEND };
2306 size_t cbStdErrBuf = 0;
2307 size_t offStdErrBuf = 0;
2308 char *pszStdErrBuf = NULL;
2309 do
2310 {
2311 /*
2312 * Service the pipe. Block waiting for output or the pipe breaking
2313 * when the process terminates.
2314 */
2315 if (hPipeR != NIL_RTPIPE)
2316 {
2317 char achBuf[1024];
2318 size_t cbRead;
2319 vrc = RTPipeReadBlocking(hPipeR, achBuf, sizeof(achBuf), &cbRead);
2320 if (RT_SUCCESS(vrc))
2321 {
2322 /* grow the buffer? */
2323 size_t cbBufReq = offStdErrBuf + cbRead + 1;
2324 if ( cbBufReq > cbStdErrBuf
2325 && cbBufReq < _256K)
2326 {
2327 size_t cbNew = RT_ALIGN_Z(cbBufReq, 16); // 1024
2328 void *pvNew = RTMemRealloc(pszStdErrBuf, cbNew);
2329 if (pvNew)
2330 {
2331 pszStdErrBuf = (char *)pvNew;
2332 cbStdErrBuf = cbNew;
2333 }
2334 }
2335
2336 /* append if we've got room. */
2337 if (cbBufReq <= cbStdErrBuf)
2338 {
2339 memcpy(&pszStdErrBuf[offStdErrBuf], achBuf, cbRead);
2340 offStdErrBuf = offStdErrBuf + cbRead;
2341 pszStdErrBuf[offStdErrBuf] = '\0';
2342 }
2343 }
2344 else
2345 {
2346 AssertLogRelMsg(vrc == VERR_BROKEN_PIPE, ("%Rrc\n", vrc));
2347 RTPipeClose(hPipeR);
2348 hPipeR = NIL_RTPIPE;
2349 }
2350 }
2351
2352 /*
2353 * Service the process. Block if we have no pipe.
2354 */
2355 if (hProcess != NIL_RTPROCESS)
2356 {
2357 vrc = RTProcWait(hProcess,
2358 hPipeR == NIL_RTPIPE ? RTPROCWAIT_FLAGS_BLOCK : RTPROCWAIT_FLAGS_NOBLOCK,
2359 &ProcStatus);
2360 if (RT_SUCCESS(vrc))
2361 hProcess = NIL_RTPROCESS;
2362 else
2363 AssertLogRelMsgStmt(vrc == VERR_PROCESS_RUNNING, ("%Rrc\n", vrc), hProcess = NIL_RTPROCESS);
2364 }
2365 } while ( hPipeR != NIL_RTPIPE
2366 || hProcess != NIL_RTPROCESS);
2367
2368 LogRel(("ExtPack: enmReason=%d iStatus=%d stderr='%s'\n",
2369 ProcStatus.enmReason, ProcStatus.iStatus, offStdErrBuf ? pszStdErrBuf : ""));
2370
2371 /*
2372 * Look for rcExit=RTEXITCODE_SUCCESS at the end of the error output,
2373 * cut it as it is only there to attest the success.
2374 */
2375 if (offStdErrBuf > 0)
2376 {
2377 RTStrStripR(pszStdErrBuf);
2378 offStdErrBuf = strlen(pszStdErrBuf);
2379 }
2380
2381 char *pszSuccessInd = findSuccessIndicator(pszStdErrBuf, offStdErrBuf);
2382 if (pszSuccessInd)
2383 {
2384 *pszSuccessInd = '\0';
2385 offStdErrBuf = pszSuccessInd - pszStdErrBuf;
2386 }
2387 else if ( ProcStatus.enmReason == RTPROCEXITREASON_NORMAL
2388 && ProcStatus.iStatus == 0)
2389 ProcStatus.iStatus = offStdErrBuf ? 667 : 666;
2390
2391 /*
2392 * Compose the status code and, on failure, error message.
2393 */
2394 if ( ProcStatus.enmReason == RTPROCEXITREASON_NORMAL
2395 && ProcStatus.iStatus == 0)
2396 hrc = S_OK;
2397 else if (ProcStatus.enmReason == RTPROCEXITREASON_NORMAL)
2398 {
2399 AssertMsg(ProcStatus.iStatus != 0, ("%s\n", pszStdErrBuf));
2400 hrc = setError(E_FAIL, tr("The installer failed with exit code %d: %s"),
2401 ProcStatus.iStatus, offStdErrBuf ? pszStdErrBuf : "");
2402 }
2403 else if (ProcStatus.enmReason == RTPROCEXITREASON_SIGNAL)
2404 hrc = setError(E_UNEXPECTED, tr("The installer was killed by signal #d (stderr: %s)"),
2405 ProcStatus.iStatus, offStdErrBuf ? pszStdErrBuf : "");
2406 else if (ProcStatus.enmReason == RTPROCEXITREASON_ABEND)
2407 hrc = setError(E_UNEXPECTED, tr("The installer aborted abnormally (stderr: %s)"),
2408 offStdErrBuf ? pszStdErrBuf : "");
2409 else
2410 hrc = setError(E_UNEXPECTED, tr("internal error: enmReason=%d iStatus=%d stderr='%s'"),
2411 ProcStatus.enmReason, ProcStatus.iStatus, offStdErrBuf ? pszStdErrBuf : "");
2412
2413 RTMemFree(pszStdErrBuf);
2414 }
2415 else
2416 hrc = setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("Failed to launch the helper application '%s' (%Rrc)"), szExecName, vrc);
2417
2418 RTPipeClose(hPipeR);
2419 RTPipeClose(hStdErrPipe.u.hPipe);
2420
2421 return hrc;
2422}
2423
2424/**
2425 * Finds an installed extension pack.
2426 *
2427 * @returns Pointer to the extension pack if found, NULL if not. (No reference
2428 * counting problem here since the caller must be holding the lock.)
2429 * @param a_pszName The name of the extension pack.
2430 */
2431ExtPack *ExtPackManager::i_findExtPack(const char *a_pszName)
2432{
2433 size_t cchName = strlen(a_pszName);
2434
2435 for (ExtPackList::iterator it = m->llInstalledExtPacks.begin();
2436 it != m->llInstalledExtPacks.end();
2437 ++it)
2438 {
2439 ExtPack::Data *pExtPackData = (*it)->m;
2440 if ( pExtPackData
2441 && pExtPackData->Desc.strName.length() == cchName
2442 && pExtPackData->Desc.strName.equalsIgnoreCase(a_pszName))
2443 return (*it);
2444 }
2445 return NULL;
2446}
2447
2448/**
2449 * Removes an installed extension pack from the internal list.
2450 *
2451 * The package is expected to exist!
2452 *
2453 * @param a_pszName The name of the extension pack.
2454 */
2455void ExtPackManager::i_removeExtPack(const char *a_pszName)
2456{
2457 size_t cchName = strlen(a_pszName);
2458
2459 for (ExtPackList::iterator it = m->llInstalledExtPacks.begin();
2460 it != m->llInstalledExtPacks.end();
2461 ++it)
2462 {
2463 ExtPack::Data *pExtPackData = (*it)->m;
2464 if ( pExtPackData
2465 && pExtPackData->Desc.strName.length() == cchName
2466 && pExtPackData->Desc.strName.equalsIgnoreCase(a_pszName))
2467 {
2468 m->llInstalledExtPacks.erase(it);
2469 return;
2470 }
2471 }
2472 AssertMsgFailed(("%s\n", a_pszName));
2473}
2474
2475#ifndef VBOX_COM_INPROC
2476
2477/**
2478 * Refreshes the specified extension pack.
2479 *
2480 * This may remove the extension pack from the list, so any non-smart pointers
2481 * to the extension pack object may become invalid.
2482 *
2483 * @returns S_OK and *a_ppExtPack on success, COM status code and error
2484 * message on failure. Note that *a_ppExtPack can be NULL.
2485 *
2486 * @param a_pszName The extension to update..
2487 * @param a_fUnusableIsError If @c true, report an unusable extension pack
2488 * as an error.
2489 * @param a_ppExtPack Where to store the pointer to the extension
2490 * pack of it is still around after the refresh.
2491 * This is optional.
2492 *
2493 * @remarks Caller holds the extension manager lock.
2494 * @remarks Only called in VBoxSVC.
2495 */
2496HRESULT ExtPackManager::i_refreshExtPack(const char *a_pszName, bool a_fUnusableIsError, ExtPack **a_ppExtPack)
2497{
2498 Assert(m->pVirtualBox != NULL); /* Only called from VBoxSVC. */
2499
2500 HRESULT hrc;
2501 ExtPack *pExtPack = i_findExtPack(a_pszName);
2502 if (pExtPack)
2503 {
2504 /*
2505 * Refresh existing object.
2506 */
2507 bool fCanDelete;
2508 hrc = pExtPack->i_refresh(&fCanDelete);
2509 if (SUCCEEDED(hrc))
2510 {
2511 if (fCanDelete)
2512 {
2513 i_removeExtPack(a_pszName);
2514 pExtPack = NULL;
2515 }
2516 }
2517 }
2518 else
2519 {
2520 /*
2521 * Do this check here, otherwise VBoxExtPackCalcDir() will fail with a strange
2522 * error.
2523 */
2524 bool fValid = VBoxExtPackIsValidName(a_pszName);
2525 if (!fValid)
2526 return setError(E_FAIL, "Invalid extension pack name specified");
2527
2528 /*
2529 * Does the dir exist? Make some special effort to deal with case
2530 * sensitivie file systems (a_pszName is case insensitive and mangled).
2531 */
2532 char szDir[RTPATH_MAX];
2533 int vrc = VBoxExtPackCalcDir(szDir, sizeof(szDir), m->strBaseDir.c_str(), a_pszName);
2534 AssertLogRelRCReturn(vrc, E_FAIL);
2535
2536 RTDIRENTRYEX Entry;
2537 RTFSOBJINFO ObjInfo;
2538 vrc = RTPathQueryInfoEx(szDir, &ObjInfo, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
2539 bool fExists = RT_SUCCESS(vrc) && RTFS_IS_DIRECTORY(ObjInfo.Attr.fMode);
2540 if (!fExists)
2541 {
2542 RTDIR hDir;
2543 vrc = RTDirOpen(&hDir, m->strBaseDir.c_str());
2544 if (RT_SUCCESS(vrc))
2545 {
2546 const char *pszMangledName = RTPathFilename(szDir);
2547 for (;;)
2548 {
2549 vrc = RTDirReadEx(hDir, &Entry, NULL /*pcbDirEntry*/, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
2550 if (RT_FAILURE(vrc))
2551 {
2552 AssertLogRelMsg(vrc == VERR_NO_MORE_FILES, ("%Rrc\n", vrc));
2553 break;
2554 }
2555 if ( RTFS_IS_DIRECTORY(Entry.Info.Attr.fMode)
2556 && !RTStrICmp(Entry.szName, pszMangledName))
2557 {
2558 /*
2559 * The installed extension pack has a uses different case.
2560 * Update the name and directory variables.
2561 */
2562 vrc = RTPathJoin(szDir, sizeof(szDir), m->strBaseDir.c_str(), Entry.szName); /* not really necessary */
2563 AssertLogRelRCReturnStmt(vrc, RTDirClose(hDir), E_UNEXPECTED);
2564 a_pszName = Entry.szName;
2565 fExists = true;
2566 break;
2567 }
2568 }
2569 RTDirClose(hDir);
2570 }
2571 }
2572 if (fExists)
2573 {
2574 /*
2575 * We've got something, create a new extension pack object for it.
2576 */
2577 ComObjPtr<ExtPack> ptrNewExtPack;
2578 hrc = ptrNewExtPack.createObject();
2579 if (SUCCEEDED(hrc))
2580 hrc = ptrNewExtPack->initWithDir(m->enmContext, a_pszName, szDir);
2581 if (SUCCEEDED(hrc))
2582 {
2583 m->llInstalledExtPacks.push_back(ptrNewExtPack);
2584 if (ptrNewExtPack->m->fUsable)
2585 LogRel(("ExtPackManager: Found extension pack '%s'.\n", a_pszName));
2586 else
2587 LogRel(("ExtPackManager: Found bad extension pack '%s': %s\n",
2588 a_pszName, ptrNewExtPack->m->strWhyUnusable.c_str() ));
2589 pExtPack = ptrNewExtPack;
2590 }
2591 }
2592 else
2593 hrc = S_OK;
2594 }
2595
2596 /*
2597 * Report error if not usable, if that is desired.
2598 */
2599 if ( SUCCEEDED(hrc)
2600 && pExtPack
2601 && a_fUnusableIsError
2602 && !pExtPack->m->fUsable)
2603 hrc = setError(E_FAIL, "%s", pExtPack->m->strWhyUnusable.c_str());
2604
2605 if (a_ppExtPack)
2606 *a_ppExtPack = pExtPack;
2607 return hrc;
2608}
2609
2610/**
2611 * Checks if there are any running VMs.
2612 *
2613 * This is called when uninstalling or replacing an extension pack.
2614 *
2615 * @returns true / false
2616 */
2617bool ExtPackManager::i_areThereAnyRunningVMs(void) const
2618{
2619 Assert(m->pVirtualBox != NULL); /* Only called from VBoxSVC. */
2620
2621 /*
2622 * Get list of machines and their states.
2623 */
2624 com::SafeIfaceArray<IMachine> SaMachines;
2625 HRESULT hrc = m->pVirtualBox->COMGETTER(Machines)(ComSafeArrayAsOutParam(SaMachines));
2626 if (SUCCEEDED(hrc))
2627 {
2628 com::SafeArray<MachineState_T> SaStates;
2629 hrc = m->pVirtualBox->GetMachineStates(ComSafeArrayAsInParam(SaMachines), ComSafeArrayAsOutParam(SaStates));
2630 if (SUCCEEDED(hrc))
2631 {
2632 /*
2633 * Scan the two parallel arrays for machines in the running state.
2634 */
2635 Assert(SaStates.size() == SaMachines.size());
2636 for (size_t i = 0; i < SaMachines.size(); ++i)
2637 if (SaMachines[i] && Global::IsOnline(SaStates[i]))
2638 return true;
2639 }
2640 }
2641 return false;
2642}
2643
2644/**
2645 * Worker for IExtPackFile::Install.
2646 *
2647 * Called on a worker thread via doInstallThreadProc.
2648 *
2649 * @returns COM status code.
2650 * @param a_pExtPackFile The extension pack file, caller checks that
2651 * it's usable.
2652 * @param a_fReplace Whether to replace any existing extpack or just
2653 * fail.
2654 * @param a_pstrDisplayInfo Host specific display information hacks.
2655 * be NULL.
2656 */
2657HRESULT ExtPackManager::i_doInstall(ExtPackFile *a_pExtPackFile, bool a_fReplace, Utf8Str const *a_pstrDisplayInfo)
2658{
2659 AssertReturn(m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON, E_UNEXPECTED);
2660 RTCString const * const pStrName = &a_pExtPackFile->m->Desc.strName;
2661 RTCString const * const pStrTarball = &a_pExtPackFile->m->strExtPackFile;
2662 RTCString const * const pStrTarballDigest = &a_pExtPackFile->m->strDigest;
2663
2664 AutoCaller autoCaller(this);
2665 HRESULT hrc = autoCaller.rc();
2666 if (SUCCEEDED(hrc))
2667 {
2668 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2669
2670 /*
2671 * Refresh the data we have on the extension pack as it
2672 * may be made stale by direct meddling or some other user.
2673 */
2674 ExtPack *pExtPack;
2675 hrc = i_refreshExtPack(pStrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack);
2676 if (SUCCEEDED(hrc))
2677 {
2678 if (pExtPack && a_fReplace)
2679 {
2680 /* We must leave the lock when calling i_areThereAnyRunningVMs,
2681 which means we have to redo the refresh call afterwards. */
2682 autoLock.release();
2683 bool fRunningVMs = i_areThereAnyRunningVMs();
2684 autoLock.acquire();
2685 hrc = i_refreshExtPack(pStrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack);
2686 if (fRunningVMs)
2687 {
2688 LogRel(("Install extension pack '%s' failed because at least one VM is still running.", pStrName->c_str()));
2689 hrc = setError(E_FAIL, tr("Install extension pack '%s' failed because at least one VM is still running"),
2690 pStrName->c_str());
2691 }
2692 else if (SUCCEEDED(hrc) && pExtPack)
2693 hrc = pExtPack->i_callUninstallHookAndClose(m->pVirtualBox, false /*a_ForcedRemoval*/);
2694 }
2695 else if (pExtPack)
2696 hrc = setError(E_FAIL,
2697 tr("Extension pack '%s' is already installed."
2698 " In case of a reinstallation, please uninstall it first"),
2699 pStrName->c_str());
2700 }
2701 if (SUCCEEDED(hrc))
2702 {
2703 /*
2704 * Run the privileged helper binary that performs the actual
2705 * installation. Then create an object for the packet (we do this
2706 * even on failure, to be on the safe side).
2707 */
2708 hrc = i_runSetUidToRootHelper(a_pstrDisplayInfo,
2709 "install",
2710 "--base-dir", m->strBaseDir.c_str(),
2711 "--cert-dir", m->strCertificatDirPath.c_str(),
2712 "--name", pStrName->c_str(),
2713 "--tarball", pStrTarball->c_str(),
2714 "--sha-256", pStrTarballDigest->c_str(),
2715 pExtPack ? "--replace" : (const char *)NULL,
2716 (const char *)NULL);
2717 if (SUCCEEDED(hrc))
2718 {
2719 hrc = i_refreshExtPack(pStrName->c_str(), true /*a_fUnusableIsError*/, &pExtPack);
2720 if (SUCCEEDED(hrc) && pExtPack)
2721 {
2722 RTERRINFOSTATIC ErrInfo;
2723 RTErrInfoInitStatic(&ErrInfo);
2724 pExtPack->i_callInstalledHook(m->pVirtualBox, &autoLock, &ErrInfo.Core);
2725 if (RT_SUCCESS(ErrInfo.Core.rc))
2726 LogRel(("ExtPackManager: Successfully installed extension pack '%s'.\n", pStrName->c_str()));
2727 else
2728 {
2729 LogRel(("ExtPackManager: Installed hook for '%s' failed: %Rrc - %s\n",
2730 pStrName->c_str(), ErrInfo.Core.rc, ErrInfo.Core.pszMsg));
2731
2732 /*
2733 * Uninstall the extpack if the error indicates that.
2734 */
2735 if (ErrInfo.Core.rc == VERR_EXTPACK_UNSUPPORTED_HOST_UNINSTALL)
2736 i_runSetUidToRootHelper(a_pstrDisplayInfo,
2737 "uninstall",
2738 "--base-dir", m->strBaseDir.c_str(),
2739 "--name", pStrName->c_str(),
2740 "--forced",
2741 (const char *)NULL);
2742 hrc = setErrorBoth(E_FAIL, ErrInfo.Core.rc, tr("The installation hook failed: %Rrc - %s"),
2743 ErrInfo.Core.rc, ErrInfo.Core.pszMsg);
2744 }
2745 }
2746 else if (SUCCEEDED(hrc))
2747 hrc = setError(E_FAIL, tr("Installing extension pack '%s' failed under mysterious circumstances"),
2748 pStrName->c_str());
2749 }
2750 else
2751 {
2752 ErrorInfoKeeper Eik;
2753 i_refreshExtPack(pStrName->c_str(), false /*a_fUnusableIsError*/, NULL);
2754 }
2755 }
2756
2757 /*
2758 * Do VirtualBoxReady callbacks now for any freshly installed
2759 * extension pack (old ones will not be called).
2760 */
2761 if (m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON)
2762 {
2763 autoLock.release();
2764 i_callAllVirtualBoxReadyHooks();
2765 }
2766 }
2767
2768 return hrc;
2769}
2770
2771/**
2772 * Worker for IExtPackManager::Uninstall.
2773 *
2774 * Called on a worker thread via doUninstallThreadProc.
2775 *
2776 * @returns COM status code.
2777 * @param a_pstrName The name of the extension pack to uninstall.
2778 * @param a_fForcedRemoval Whether to be skip and ignore certain bits of
2779 * the extpack feedback. To deal with misbehaving
2780 * extension pack hooks.
2781 * @param a_pstrDisplayInfo Host specific display information hacks.
2782 */
2783HRESULT ExtPackManager::i_doUninstall(Utf8Str const *a_pstrName, bool a_fForcedRemoval, Utf8Str const *a_pstrDisplayInfo)
2784{
2785 Assert(m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON);
2786
2787 AutoCaller autoCaller(this);
2788 HRESULT hrc = autoCaller.rc();
2789 if (SUCCEEDED(hrc))
2790 {
2791 if (a_fForcedRemoval || !i_areThereAnyRunningVMs())
2792 {
2793 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2794
2795 /*
2796 * Refresh the data we have on the extension pack as it may be made
2797 * stale by direct meddling or some other user.
2798 */
2799 ExtPack *pExtPack;
2800 hrc = i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack);
2801 if (SUCCEEDED(hrc))
2802 {
2803 if (!pExtPack)
2804 {
2805 LogRel(("ExtPackManager: Extension pack '%s' is not installed, so nothing to uninstall.\n", a_pstrName->c_str()));
2806 hrc = S_OK; /* nothing to uninstall */
2807 }
2808 else
2809 {
2810 /*
2811 * Call the uninstall hook and unload the main dll.
2812 */
2813 hrc = pExtPack->i_callUninstallHookAndClose(m->pVirtualBox, a_fForcedRemoval);
2814 if (SUCCEEDED(hrc))
2815 {
2816 /*
2817 * Run the set-uid-to-root binary that performs the
2818 * uninstallation. Then refresh the object.
2819 *
2820 * This refresh is theorically subject to races, but it's of
2821 * the don't-do-that variety.
2822 */
2823 const char *pszForcedOpt = a_fForcedRemoval ? "--forced" : NULL;
2824 hrc = i_runSetUidToRootHelper(a_pstrDisplayInfo,
2825 "uninstall",
2826 "--base-dir", m->strBaseDir.c_str(),
2827 "--name", a_pstrName->c_str(),
2828 pszForcedOpt, /* Last as it may be NULL. */
2829 (const char *)NULL);
2830 if (SUCCEEDED(hrc))
2831 {
2832 hrc = i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack);
2833 if (SUCCEEDED(hrc))
2834 {
2835 if (!pExtPack)
2836 LogRel(("ExtPackManager: Successfully uninstalled extension pack '%s'.\n", a_pstrName->c_str()));
2837 else
2838 hrc = setError(E_FAIL,
2839 tr("Uninstall extension pack '%s' failed under mysterious circumstances"),
2840 a_pstrName->c_str());
2841 }
2842 }
2843 else
2844 {
2845 ErrorInfoKeeper Eik;
2846 i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, NULL);
2847 }
2848 }
2849 }
2850 }
2851 }
2852 else
2853 {
2854 LogRel(("Uninstall extension pack '%s' failed because at least one VM is still running.", a_pstrName->c_str()));
2855 hrc = setError(E_FAIL, tr("Uninstall extension pack '%s' failed because at least one VM is still running"),
2856 a_pstrName->c_str());
2857 }
2858
2859 /*
2860 * Do VirtualBoxReady callbacks now for any freshly installed
2861 * extension pack (old ones will not be called).
2862 */
2863 if (m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON)
2864 i_callAllVirtualBoxReadyHooks();
2865 }
2866
2867 return hrc;
2868}
2869
2870
2871/**
2872 * Calls the pfnVirtualBoxReady hook for all working extension packs.
2873 *
2874 * @remarks The caller must not hold any locks.
2875 */
2876void ExtPackManager::i_callAllVirtualBoxReadyHooks(void)
2877{
2878 AutoCaller autoCaller(this);
2879 HRESULT hrc = autoCaller.rc();
2880 if (FAILED(hrc))
2881 return;
2882 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2883 ComPtr<ExtPackManager> ptrSelfRef = this;
2884
2885 for (ExtPackList::iterator it = m->llInstalledExtPacks.begin();
2886 it != m->llInstalledExtPacks.end();
2887 /* advancing below */)
2888 {
2889 if ((*it)->i_callVirtualBoxReadyHook(m->pVirtualBox, &autoLock))
2890 it = m->llInstalledExtPacks.begin();
2891 else
2892 ++it;
2893 }
2894}
2895
2896#endif /* !VBOX_COM_INPROC */
2897
2898#ifdef VBOX_COM_INPROC
2899/**
2900 * Calls the pfnConsoleReady hook for all working extension packs.
2901 *
2902 * @param a_pConsole The console interface.
2903 * @remarks The caller must not hold any locks.
2904 */
2905void ExtPackManager::i_callAllConsoleReadyHooks(IConsole *a_pConsole)
2906{
2907 AutoCaller autoCaller(this);
2908 HRESULT hrc = autoCaller.rc();
2909 if (FAILED(hrc))
2910 return;
2911 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2912 ComPtr<ExtPackManager> ptrSelfRef = this;
2913
2914 for (ExtPackList::iterator it = m->llInstalledExtPacks.begin();
2915 it != m->llInstalledExtPacks.end();
2916 /* advancing below */)
2917 {
2918 if ((*it)->i_callConsoleReadyHook(a_pConsole, &autoLock))
2919 it = m->llInstalledExtPacks.begin();
2920 else
2921 ++it;
2922 }
2923}
2924#endif
2925
2926#ifndef VBOX_COM_INPROC
2927/**
2928 * Calls the pfnVMCreated hook for all working extension packs.
2929 *
2930 * @param a_pMachine The machine interface of the new VM.
2931 */
2932void ExtPackManager::i_callAllVmCreatedHooks(IMachine *a_pMachine)
2933{
2934 AutoCaller autoCaller(this);
2935 HRESULT hrc = autoCaller.rc();
2936 if (FAILED(hrc))
2937 return;
2938 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2939 ComPtr<ExtPackManager> ptrSelfRef = this; /* paranoia */
2940 ExtPackList llExtPacks = m->llInstalledExtPacks;
2941
2942 for (ExtPackList::iterator it = llExtPacks.begin(); it != llExtPacks.end(); ++it)
2943 (*it)->i_callVmCreatedHook(m->pVirtualBox, a_pMachine, &autoLock);
2944}
2945#endif
2946
2947#ifdef VBOX_COM_INPROC
2948/**
2949 * Calls the pfnVMConfigureVMM hook for all working extension packs.
2950 *
2951 * @returns VBox status code. Stops on the first failure, expecting the caller
2952 * to signal this to the caller of the CFGM constructor.
2953 * @param a_pConsole The console interface for the VM.
2954 * @param a_pVM The VM handle.
2955 */
2956int ExtPackManager::i_callAllVmConfigureVmmHooks(IConsole *a_pConsole, PVM a_pVM)
2957{
2958 AutoCaller autoCaller(this);
2959 HRESULT hrc = autoCaller.rc();
2960 if (FAILED(hrc))
2961 return Global::vboxStatusCodeFromCOM(hrc);
2962 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2963 ComPtr<ExtPackManager> ptrSelfRef = this; /* paranoia */
2964 ExtPackList llExtPacks = m->llInstalledExtPacks;
2965
2966 for (ExtPackList::iterator it = llExtPacks.begin(); it != llExtPacks.end(); ++it)
2967 {
2968 int vrc;
2969 (*it)->i_callVmConfigureVmmHook(a_pConsole, a_pVM, &autoLock, &vrc);
2970 if (RT_FAILURE(vrc))
2971 return vrc;
2972 }
2973
2974 return VINF_SUCCESS;
2975}
2976
2977/**
2978 * Calls the pfnVMPowerOn hook for all working extension packs.
2979 *
2980 * @returns VBox status code. Stops on the first failure, expecting the caller
2981 * to not power on the VM.
2982 * @param a_pConsole The console interface for the VM.
2983 * @param a_pVM The VM handle.
2984 */
2985int ExtPackManager::i_callAllVmPowerOnHooks(IConsole *a_pConsole, PVM a_pVM)
2986{
2987 AutoCaller autoCaller(this);
2988 HRESULT hrc = autoCaller.rc();
2989 if (FAILED(hrc))
2990 return Global::vboxStatusCodeFromCOM(hrc);
2991 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
2992 ComPtr<ExtPackManager> ptrSelfRef = this; /* paranoia */
2993 ExtPackList llExtPacks = m->llInstalledExtPacks;
2994
2995 for (ExtPackList::iterator it = llExtPacks.begin(); it != llExtPacks.end(); ++it)
2996 {
2997 int vrc;
2998 (*it)->i_callVmPowerOnHook(a_pConsole, a_pVM, &autoLock, &vrc);
2999 if (RT_FAILURE(vrc))
3000 return vrc;
3001 }
3002
3003 return VINF_SUCCESS;
3004}
3005
3006/**
3007 * Calls the pfnVMPowerOff hook for all working extension packs.
3008 *
3009 * @param a_pConsole The console interface for the VM.
3010 * @param a_pVM The VM handle. Can be NULL.
3011 */
3012void ExtPackManager::i_callAllVmPowerOffHooks(IConsole *a_pConsole, PVM a_pVM)
3013{
3014 AutoCaller autoCaller(this);
3015 HRESULT hrc = autoCaller.rc();
3016 if (FAILED(hrc))
3017 return;
3018 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS);
3019 ComPtr<ExtPackManager> ptrSelfRef = this; /* paranoia */
3020 ExtPackList llExtPacks = m->llInstalledExtPacks;
3021
3022 for (ExtPackList::iterator it = llExtPacks.begin(); it != llExtPacks.end(); ++it)
3023 (*it)->i_callVmPowerOffHook(a_pConsole, a_pVM, &autoLock);
3024}
3025#endif
3026
3027
3028/**
3029 * Checks that the specified extension pack contains a VRDE module and that it
3030 * is shipshape.
3031 *
3032 * @returns S_OK if ok, appropriate failure status code with details.
3033 * @param a_pstrExtPack The name of the extension pack.
3034 */
3035HRESULT ExtPackManager::i_checkVrdeExtPack(Utf8Str const *a_pstrExtPack)
3036{
3037 AutoCaller autoCaller(this);
3038 HRESULT hrc = autoCaller.rc();
3039 if (SUCCEEDED(hrc))
3040 {
3041 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
3042
3043 ExtPack *pExtPack = i_findExtPack(a_pstrExtPack->c_str());
3044 if (pExtPack)
3045 hrc = pExtPack->i_checkVrde();
3046 else
3047 hrc = setError(VBOX_E_OBJECT_NOT_FOUND, tr("No extension pack by the name '%s' was found"), a_pstrExtPack->c_str());
3048 }
3049
3050 return hrc;
3051}
3052
3053/**
3054 * Gets the full path to the VRDE library of the specified extension pack.
3055 *
3056 * This will do extacly the same as checkVrdeExtPack and then resolve the
3057 * library path.
3058 *
3059 * @returns S_OK if a path is returned, COM error status and message return if
3060 * not.
3061 * @param a_pstrExtPack The extension pack.
3062 * @param a_pstrVrdeLibrary Where to return the path.
3063 */
3064int ExtPackManager::i_getVrdeLibraryPathForExtPack(Utf8Str const *a_pstrExtPack, Utf8Str *a_pstrVrdeLibrary)
3065{
3066 AutoCaller autoCaller(this);
3067 HRESULT hrc = autoCaller.rc();
3068 if (SUCCEEDED(hrc))
3069 {
3070 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
3071
3072 ExtPack *pExtPack = i_findExtPack(a_pstrExtPack->c_str());
3073 if (pExtPack)
3074 hrc = pExtPack->i_getVrdpLibraryName(a_pstrVrdeLibrary);
3075 else
3076 hrc = setError(VBOX_E_OBJECT_NOT_FOUND, tr("No extension pack by the name '%s' was found"),
3077 a_pstrExtPack->c_str());
3078 }
3079
3080 return hrc;
3081}
3082
3083/**
3084 * Gets the full path to the specified library of the specified extension pack.
3085 *
3086 * @returns S_OK if a path is returned, COM error status and message return if
3087 * not.
3088 * @param a_pszModuleName The library.
3089 * @param a_pszExtPack The extension pack.
3090 * @param a_pstrLibrary Where to return the path.
3091 */
3092HRESULT ExtPackManager::i_getLibraryPathForExtPack(const char *a_pszModuleName, const char *a_pszExtPack, Utf8Str *a_pstrLibrary)
3093{
3094 AutoCaller autoCaller(this);
3095 HRESULT hrc = autoCaller.rc();
3096 if (SUCCEEDED(hrc))
3097 {
3098 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
3099
3100 ExtPack *pExtPack = i_findExtPack(a_pszExtPack);
3101 if (pExtPack)
3102 hrc = pExtPack->i_getLibraryName(a_pszModuleName, a_pstrLibrary);
3103 else
3104 hrc = setError(VBOX_E_OBJECT_NOT_FOUND, tr("No extension pack by the name '%s' was found"), a_pszExtPack);
3105 }
3106
3107 return hrc;
3108}
3109
3110/**
3111 * Gets the name of the default VRDE extension pack.
3112 *
3113 * @returns S_OK or some COM error status on red tape failure.
3114 * @param a_pstrExtPack Where to return the extension pack name. Returns
3115 * empty if no extension pack wishes to be the default
3116 * VRDP provider.
3117 */
3118HRESULT ExtPackManager::i_getDefaultVrdeExtPack(Utf8Str *a_pstrExtPack)
3119{
3120 a_pstrExtPack->setNull();
3121
3122 AutoCaller autoCaller(this);
3123 HRESULT hrc = autoCaller.rc();
3124 if (SUCCEEDED(hrc))
3125 {
3126 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
3127
3128 for (ExtPackList::iterator it = m->llInstalledExtPacks.begin();
3129 it != m->llInstalledExtPacks.end();
3130 ++it)
3131 {
3132 if ((*it)->i_wantsToBeDefaultVrde())
3133 {
3134 *a_pstrExtPack = (*it)->m->Desc.strName;
3135 break;
3136 }
3137 }
3138 }
3139 return hrc;
3140}
3141
3142/**
3143 * Checks if an extension pack is (present and) usable.
3144 *
3145 * @returns @c true if it is, otherwise @c false.
3146 * @param a_pszExtPack The name of the extension pack.
3147 */
3148bool ExtPackManager::i_isExtPackUsable(const char *a_pszExtPack)
3149{
3150 AutoCaller autoCaller(this);
3151 HRESULT hrc = autoCaller.rc();
3152 if (FAILED(hrc))
3153 return false;
3154 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
3155
3156 ExtPack *pExtPack = i_findExtPack(a_pszExtPack);
3157 return pExtPack != NULL
3158 && pExtPack->m->fUsable;
3159}
3160
3161/**
3162 * Dumps all extension packs to the release log.
3163 */
3164void ExtPackManager::i_dumpAllToReleaseLog(void)
3165{
3166 AutoCaller autoCaller(this);
3167 HRESULT hrc = autoCaller.rc();
3168 if (FAILED(hrc))
3169 return;
3170 AutoReadLock autoLock(this COMMA_LOCKVAL_SRC_POS);
3171
3172 LogRel(("Installed Extension Packs:\n"));
3173 for (ExtPackList::iterator it = m->llInstalledExtPacks.begin();
3174 it != m->llInstalledExtPacks.end();
3175 ++it)
3176 {
3177 ExtPack::Data *pExtPackData = (*it)->m;
3178 if (pExtPackData)
3179 {
3180 if (pExtPackData->fUsable)
3181 LogRel((" %s (Version: %s r%u%s%s; VRDE Module: %s)\n",
3182 pExtPackData->Desc.strName.c_str(),
3183 pExtPackData->Desc.strVersion.c_str(),
3184 pExtPackData->Desc.uRevision,
3185 pExtPackData->Desc.strEdition.isEmpty() ? "" : " ",
3186 pExtPackData->Desc.strEdition.c_str(),
3187 pExtPackData->Desc.strVrdeModule.c_str() ));
3188 else
3189 LogRel((" %s (Version: %s r%u%s%s; VRDE Module: %s unusable because of '%s')\n",
3190 pExtPackData->Desc.strName.c_str(),
3191 pExtPackData->Desc.strVersion.c_str(),
3192 pExtPackData->Desc.uRevision,
3193 pExtPackData->Desc.strEdition.isEmpty() ? "" : " ",
3194 pExtPackData->Desc.strEdition.c_str(),
3195 pExtPackData->Desc.strVrdeModule.c_str(),
3196 pExtPackData->strWhyUnusable.c_str() ));
3197 }
3198 else
3199 LogRel((" pExtPackData is NULL\n"));
3200 }
3201
3202 if (!m->llInstalledExtPacks.size())
3203 LogRel((" None installed!\n"));
3204}
3205
3206/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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