VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/GuestSessionImplTasks.cpp@ 49994

Last change on this file since 49994 was 49630, checked in by vboxsync, 11 years ago

Main/GuestCtrl: Todo, logging.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 61.7 KB
Line 
1/* $Id: GuestSessionImplTasks.cpp 49630 2013-11-22 16:19:16Z vboxsync $ */
2/** @file
3 * VirtualBox Main - Guest session tasks.
4 */
5
6/*
7 * Copyright (C) 2012-2013 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#include "GuestImpl.h"
23#include "GuestSessionImpl.h"
24#include "GuestCtrlImplPrivate.h"
25
26#include "Global.h"
27#include "AutoCaller.h"
28#include "ConsoleImpl.h"
29#include "MachineImpl.h"
30#include "ProgressImpl.h"
31
32#include <memory> /* For auto_ptr. */
33
34#include <iprt/env.h>
35#include <iprt/file.h> /* For CopyTo/From. */
36
37#ifdef LOG_GROUP
38 #undef LOG_GROUP
39#endif
40#define LOG_GROUP LOG_GROUP_GUEST_CONTROL
41#include <VBox/log.h>
42
43
44/*******************************************************************************
45* Defines *
46*******************************************************************************/
47
48/**
49 * Update file flags.
50 */
51#define UPDATEFILE_FLAG_NONE 0
52/** Copy over the file from host to the
53 * guest. */
54#define UPDATEFILE_FLAG_COPY_FROM_ISO RT_BIT(0)
55/** Execute file on the guest after it has
56 * been successfully transfered. */
57#define UPDATEFILE_FLAG_EXECUTE RT_BIT(7)
58/** File is optional, does not have to be
59 * existent on the .ISO. */
60#define UPDATEFILE_FLAG_OPTIONAL RT_BIT(8)
61
62
63// session task classes
64/////////////////////////////////////////////////////////////////////////////
65
66GuestSessionTask::GuestSessionTask(GuestSession *pSession)
67{
68 mSession = pSession;
69}
70
71GuestSessionTask::~GuestSessionTask(void)
72{
73}
74
75int GuestSessionTask::getGuestProperty(const ComObjPtr<Guest> &pGuest,
76 const Utf8Str &strPath, Utf8Str &strValue)
77{
78 ComObjPtr<Console> pConsole = pGuest->getConsole();
79 const ComPtr<IMachine> pMachine = pConsole->machine();
80
81 Assert(!pMachine.isNull());
82 Bstr strTemp, strFlags;
83 LONG64 i64Timestamp;
84 HRESULT hr = pMachine->GetGuestProperty(Bstr(strPath).raw(),
85 strTemp.asOutParam(),
86 &i64Timestamp, strFlags.asOutParam());
87 if (SUCCEEDED(hr))
88 {
89 strValue = strTemp;
90 return VINF_SUCCESS;
91 }
92 return VERR_NOT_FOUND;
93}
94
95int GuestSessionTask::setProgress(ULONG uPercent)
96{
97 if (mProgress.isNull()) /* Progress is optional. */
98 return VINF_SUCCESS;
99
100 BOOL fCanceled;
101 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
102 && fCanceled)
103 return VERR_CANCELLED;
104 BOOL fCompleted;
105 if ( SUCCEEDED(mProgress->COMGETTER(Completed(&fCompleted)))
106 && fCompleted)
107 {
108 AssertMsgFailed(("Setting value of an already completed progress\n"));
109 return VINF_SUCCESS;
110 }
111 HRESULT hr = mProgress->SetCurrentOperationProgress(uPercent);
112 if (FAILED(hr))
113 return VERR_COM_UNEXPECTED;
114
115 return VINF_SUCCESS;
116}
117
118int GuestSessionTask::setProgressSuccess(void)
119{
120 if (mProgress.isNull()) /* Progress is optional. */
121 return VINF_SUCCESS;
122
123 BOOL fCanceled;
124 BOOL fCompleted;
125 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
126 && !fCanceled
127 && SUCCEEDED(mProgress->COMGETTER(Completed(&fCompleted)))
128 && !fCompleted)
129 {
130 HRESULT hr = mProgress->notifyComplete(S_OK);
131 if (FAILED(hr))
132 return VERR_COM_UNEXPECTED; /** @todo Find a better rc. */
133 }
134
135 return VINF_SUCCESS;
136}
137
138HRESULT GuestSessionTask::setProgressErrorMsg(HRESULT hr, const Utf8Str &strMsg)
139{
140 LogFlowFunc(("hr=%Rhrc, strMsg=%s\n",
141 hr, strMsg.c_str()));
142
143 if (mProgress.isNull()) /* Progress is optional. */
144 return hr; /* Return original rc. */
145
146 BOOL fCanceled;
147 BOOL fCompleted;
148 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
149 && !fCanceled
150 && SUCCEEDED(mProgress->COMGETTER(Completed(&fCompleted)))
151 && !fCompleted)
152 {
153 HRESULT hr2 = mProgress->notifyComplete(hr,
154 COM_IIDOF(IGuestSession),
155 GuestSession::getStaticComponentName(),
156 strMsg.c_str());
157 if (FAILED(hr2))
158 return hr2;
159 }
160 return hr; /* Return original rc. */
161}
162
163SessionTaskOpen::SessionTaskOpen(GuestSession *pSession,
164 uint32_t uFlags,
165 uint32_t uTimeoutMS)
166 : GuestSessionTask(pSession),
167 mFlags(uFlags),
168 mTimeoutMS(uTimeoutMS)
169{
170
171}
172
173SessionTaskOpen::~SessionTaskOpen(void)
174{
175
176}
177
178int SessionTaskOpen::Run(int *pGuestRc)
179{
180 LogFlowThisFuncEnter();
181
182 ComObjPtr<GuestSession> pSession = mSession;
183 Assert(!pSession.isNull());
184
185 AutoCaller autoCaller(pSession);
186 if (FAILED(autoCaller.rc())) return autoCaller.rc();
187
188 int vrc = pSession->startSessionInternal(pGuestRc);
189 /* Nothing to do here anymore. */
190
191 LogFlowFuncLeaveRC(vrc);
192 return vrc;
193}
194
195int SessionTaskOpen::RunAsync(const Utf8Str &strDesc, ComObjPtr<Progress> &pProgress)
196{
197 LogFlowThisFunc(("strDesc=%s\n", strDesc.c_str()));
198
199 mDesc = strDesc;
200 mProgress = pProgress;
201
202 int rc = RTThreadCreate(NULL, SessionTaskOpen::taskThread, this,
203 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
204 "gctlSesOpen");
205 LogFlowFuncLeaveRC(rc);
206 return rc;
207}
208
209/* static */
210int SessionTaskOpen::taskThread(RTTHREAD Thread, void *pvUser)
211{
212 std::auto_ptr<SessionTaskOpen> task(static_cast<SessionTaskOpen*>(pvUser));
213 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
214
215 LogFlowFunc(("pTask=%p\n", task.get()));
216 return task->Run(NULL /* guestRc */);
217}
218
219SessionTaskCopyTo::SessionTaskCopyTo(GuestSession *pSession,
220 const Utf8Str &strSource, const Utf8Str &strDest, uint32_t uFlags)
221 : GuestSessionTask(pSession),
222 mSource(strSource),
223 mSourceFile(NULL),
224 mSourceOffset(0),
225 mSourceSize(0),
226 mDest(strDest)
227{
228 mCopyFileFlags = uFlags;
229}
230
231/** @todo Merge this and the above call and let the above call do the open/close file handling so that the
232 * inner code only has to deal with file handles. No time now ... */
233SessionTaskCopyTo::SessionTaskCopyTo(GuestSession *pSession,
234 PRTFILE pSourceFile, size_t cbSourceOffset, uint64_t cbSourceSize,
235 const Utf8Str &strDest, uint32_t uFlags)
236 : GuestSessionTask(pSession)
237{
238 mSourceFile = pSourceFile;
239 mSourceOffset = cbSourceOffset;
240 mSourceSize = cbSourceSize;
241 mDest = strDest;
242 mCopyFileFlags = uFlags;
243}
244
245SessionTaskCopyTo::~SessionTaskCopyTo(void)
246{
247
248}
249
250int SessionTaskCopyTo::Run(void)
251{
252 LogFlowThisFuncEnter();
253
254 ComObjPtr<GuestSession> pSession = mSession;
255 Assert(!pSession.isNull());
256
257 AutoCaller autoCaller(pSession);
258 if (FAILED(autoCaller.rc())) return autoCaller.rc();
259
260 if (mCopyFileFlags)
261 {
262 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
263 Utf8StrFmt(GuestSession::tr("Copy flags (%#x) not implemented yet"),
264 mCopyFileFlags));
265 return VERR_INVALID_PARAMETER;
266 }
267
268 int rc;
269
270 RTFILE fileLocal;
271 PRTFILE pFile = &fileLocal;
272
273 if (!mSourceFile)
274 {
275 /* Does our source file exist? */
276 if (!RTFileExists(mSource.c_str()))
277 {
278 rc = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
279 Utf8StrFmt(GuestSession::tr("Source file \"%s\" does not exist or is not a file"),
280 mSource.c_str()));
281 }
282 else
283 {
284 rc = RTFileOpen(pFile, mSource.c_str(),
285 RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
286 if (RT_FAILURE(rc))
287 {
288 rc = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
289 Utf8StrFmt(GuestSession::tr("Could not open source file \"%s\" for reading: %Rrc"),
290 mSource.c_str(), rc));
291 }
292 else
293 {
294 rc = RTFileGetSize(*pFile, &mSourceSize);
295 if (RT_FAILURE(rc))
296 {
297 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
298 Utf8StrFmt(GuestSession::tr("Could not query file size of \"%s\": %Rrc"),
299 mSource.c_str(), rc));
300 }
301 }
302 }
303 }
304 else
305 {
306 rc = VINF_SUCCESS;
307 pFile = mSourceFile;
308 /* Size + offset are optional. */
309 }
310
311 GuestProcessStartupInfo procInfo;
312 procInfo.mCommand = Utf8Str(VBOXSERVICE_TOOL_CAT);
313 procInfo.mFlags = ProcessCreateFlag_Hidden;
314
315 /* Set arguments.*/
316 procInfo.mArguments.push_back(Utf8StrFmt("--output=%s", mDest.c_str())); /** @todo Do we need path conversion? */
317
318 /* Startup process. */
319 ComObjPtr<GuestProcess> pProcess; int guestRc;
320 if (RT_SUCCESS(rc))
321 rc = pSession->processCreateExInteral(procInfo, pProcess);
322 if (RT_SUCCESS(rc))
323 {
324 Assert(!pProcess.isNull());
325 rc = pProcess->startProcess(30 * 1000 /* 30s timeout */,
326 &guestRc);
327 }
328
329 if (RT_FAILURE(rc))
330 {
331 switch (rc)
332 {
333 case VERR_GSTCTL_GUEST_ERROR:
334 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
335 GuestProcess::guestErrorToString(guestRc));
336 break;
337
338 default:
339 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
340 Utf8StrFmt(GuestSession::tr("Error while creating guest process for copying file \"%s\" from guest to host: %Rrc"),
341 mSource.c_str(), rc));
342 break;
343 }
344 }
345
346 if (RT_SUCCESS(rc))
347 {
348 ProcessWaitResult_T waitRes;
349 BYTE byBuf[_64K];
350
351 BOOL fCanceled = FALSE;
352 uint64_t cbWrittenTotal = 0;
353 uint64_t cbToRead = mSourceSize;
354
355 for (;;)
356 {
357 rc = pProcess->waitFor(ProcessWaitForFlag_StdIn,
358 30 * 1000 /* Timeout */, waitRes, &guestRc);
359 if ( RT_FAILURE(rc)
360 || ( waitRes != ProcessWaitResult_StdIn
361 && waitRes != ProcessWaitResult_WaitFlagNotSupported))
362 {
363 break;
364 }
365
366 /* If the guest does not support waiting for stdin, we now yield in
367 * order to reduce the CPU load due to busy waiting. */
368 if (waitRes == ProcessWaitResult_WaitFlagNotSupported)
369 RTThreadYield(); /* Optional, don't check rc. */
370
371 size_t cbRead = 0;
372 if (mSourceSize) /* If we have nothing to write, take a shortcut. */
373 {
374 /** @todo Not very efficient, but works for now. */
375 rc = RTFileSeek(*pFile, mSourceOffset + cbWrittenTotal,
376 RTFILE_SEEK_BEGIN, NULL /* poffActual */);
377 if (RT_SUCCESS(rc))
378 {
379 rc = RTFileRead(*pFile, (uint8_t*)byBuf,
380 RT_MIN((size_t)cbToRead, sizeof(byBuf)), &cbRead);
381 /*
382 * Some other error occured? There might be a chance that RTFileRead
383 * could not resolve/map the native error code to an IPRT code, so just
384 * print a generic error.
385 */
386 if (RT_FAILURE(rc))
387 {
388 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
389 Utf8StrFmt(GuestSession::tr("Could not read from file \"%s\" (%Rrc)"),
390 mSource.c_str(), rc));
391 break;
392 }
393 }
394 else
395 {
396 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
397 Utf8StrFmt(GuestSession::tr("Seeking file \"%s\" to offset %RU64 failed: %Rrc"),
398 mSource.c_str(), cbWrittenTotal, rc));
399 break;
400 }
401 }
402
403 uint32_t fFlags = ProcessInputFlag_None;
404
405 /* Did we reach the end of the content we want to transfer (last chunk)? */
406 if ( (cbRead < sizeof(byBuf))
407 /* Did we reach the last block which is exactly _64K? */
408 || (cbToRead - cbRead == 0)
409 /* ... or does the user want to cancel? */
410 || ( !mProgress.isNull()
411 && SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
412 && fCanceled)
413 )
414 {
415 LogFlowThisFunc(("Writing last chunk cbRead=%RU64\n", cbRead));
416 fFlags |= ProcessInputFlag_EndOfFile;
417 }
418
419 uint32_t cbWritten;
420 Assert(sizeof(byBuf) >= cbRead);
421 rc = pProcess->writeData(0 /* StdIn */, fFlags,
422 byBuf, cbRead,
423 30 * 1000 /* Timeout */, &cbWritten, &guestRc);
424 if (RT_FAILURE(rc))
425 {
426 switch (rc)
427 {
428 case VERR_GSTCTL_GUEST_ERROR:
429 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
430 GuestProcess::guestErrorToString(guestRc));
431 break;
432
433 default:
434 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
435 Utf8StrFmt(GuestSession::tr("Writing to file \"%s\" (offset %RU64) failed: %Rrc"),
436 mDest.c_str(), cbWrittenTotal, rc));
437 break;
438 }
439
440 break;
441 }
442
443 /* Only subtract bytes reported written by the guest. */
444 Assert(cbToRead >= cbWritten);
445 cbToRead -= cbWritten;
446
447 /* Update total bytes written to the guest. */
448 cbWrittenTotal += cbWritten;
449 Assert(cbWrittenTotal <= mSourceSize);
450
451 LogFlowThisFunc(("rc=%Rrc, cbWritten=%RU32, cbToRead=%RU64, cbWrittenTotal=%RU64, cbFileSize=%RU64\n",
452 rc, cbWritten, cbToRead, cbWrittenTotal, mSourceSize));
453
454 /* Did the user cancel the operation above? */
455 if (fCanceled)
456 break;
457
458 /* Update the progress.
459 * Watch out for division by zero. */
460 mSourceSize > 0
461 ? rc = setProgress((ULONG)(cbWrittenTotal * 100 / mSourceSize))
462 : rc = setProgress(100);
463 if (RT_FAILURE(rc))
464 break;
465
466 /* End of file reached? */
467 if (!cbToRead)
468 break;
469 } /* for */
470
471 LogFlowThisFunc(("Copy loop ended with rc=%Rrc, cbToRead=%RU64, cbWrittenTotal=%RU64, cbFileSize=%RU64\n",
472 rc, cbToRead, cbWrittenTotal, mSourceSize));
473
474 if ( !fCanceled
475 || RT_SUCCESS(rc))
476 {
477 /*
478 * Even if we succeeded until here make sure to check whether we really transfered
479 * everything.
480 */
481 if ( mSourceSize > 0
482 && cbWrittenTotal == 0)
483 {
484 /* If nothing was transfered but the file size was > 0 then "vbox_cat" wasn't able to write
485 * to the destination -> access denied. */
486 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
487 Utf8StrFmt(GuestSession::tr("Access denied when copying file \"%s\" to \"%s\""),
488 mSource.c_str(), mDest.c_str()));
489 rc = VERR_GENERAL_FAILURE; /* Fudge. */
490 }
491 else if (cbWrittenTotal < mSourceSize)
492 {
493 /* If we did not copy all let the user know. */
494 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
495 Utf8StrFmt(GuestSession::tr("Copying file \"%s\" failed (%RU64/%RU64 bytes transfered)"),
496 mSource.c_str(), cbWrittenTotal, mSourceSize));
497 rc = VERR_GENERAL_FAILURE; /* Fudge. */
498 }
499 else
500 {
501 rc = pProcess->waitFor(ProcessWaitForFlag_Terminate,
502 30 * 1000 /* Timeout */, waitRes, &guestRc);
503 if ( RT_FAILURE(rc)
504 || waitRes != ProcessWaitResult_Terminate)
505 {
506 if (RT_FAILURE(rc))
507 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
508 Utf8StrFmt(GuestSession::tr("Waiting on termination for copying file \"%s\" failed: %Rrc"),
509 mSource.c_str(), rc));
510 else
511 {
512 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
513 Utf8StrFmt(GuestSession::tr("Waiting on termination for copying file \"%s\" failed with wait result %ld"),
514 mSource.c_str(), waitRes));
515 rc = VERR_GENERAL_FAILURE; /* Fudge. */
516 }
517 }
518
519 if (RT_SUCCESS(rc))
520 {
521 ProcessStatus_T procStatus;
522 LONG exitCode;
523 if ( ( SUCCEEDED(pProcess->COMGETTER(Status(&procStatus)))
524 && procStatus != ProcessStatus_TerminatedNormally)
525 || ( SUCCEEDED(pProcess->COMGETTER(ExitCode(&exitCode)))
526 && exitCode != 0)
527 )
528 {
529 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
530 Utf8StrFmt(GuestSession::tr("Copying file \"%s\" failed with status %ld, exit code %ld"),
531 mSource.c_str(), procStatus, exitCode)); /**@todo Add stringify methods! */
532 rc = VERR_GENERAL_FAILURE; /* Fudge. */
533 }
534 }
535
536 if (RT_SUCCESS(rc))
537 rc = setProgressSuccess();
538 }
539 }
540 } /* processCreateExInteral */
541
542 if (!mSourceFile) /* Only close locally opened files. */
543 RTFileClose(*pFile);
544
545 LogFlowFuncLeaveRC(rc);
546 return rc;
547}
548
549int SessionTaskCopyTo::RunAsync(const Utf8Str &strDesc, ComObjPtr<Progress> &pProgress)
550{
551 LogFlowThisFunc(("strDesc=%s, strSource=%s, strDest=%s, mCopyFileFlags=%x\n",
552 strDesc.c_str(), mSource.c_str(), mDest.c_str(), mCopyFileFlags));
553
554 mDesc = strDesc;
555 mProgress = pProgress;
556
557 int rc = RTThreadCreate(NULL, SessionTaskCopyTo::taskThread, this,
558 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
559 "gctlCpyTo");
560 LogFlowFuncLeaveRC(rc);
561 return rc;
562}
563
564/* static */
565int SessionTaskCopyTo::taskThread(RTTHREAD Thread, void *pvUser)
566{
567 std::auto_ptr<SessionTaskCopyTo> task(static_cast<SessionTaskCopyTo*>(pvUser));
568 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
569
570 LogFlowFunc(("pTask=%p\n", task.get()));
571 return task->Run();
572}
573
574SessionTaskCopyFrom::SessionTaskCopyFrom(GuestSession *pSession,
575 const Utf8Str &strSource, const Utf8Str &strDest, uint32_t uFlags)
576 : GuestSessionTask(pSession)
577{
578 mSource = strSource;
579 mDest = strDest;
580 mFlags = uFlags;
581}
582
583SessionTaskCopyFrom::~SessionTaskCopyFrom(void)
584{
585
586}
587
588int SessionTaskCopyFrom::Run(void)
589{
590 LogFlowThisFuncEnter();
591
592 ComObjPtr<GuestSession> pSession = mSession;
593 Assert(!pSession.isNull());
594
595 AutoCaller autoCaller(pSession);
596 if (FAILED(autoCaller.rc())) return autoCaller.rc();
597
598 /*
599 * Note: There will be races between querying file size + reading the guest file's
600 * content because we currently *do not* lock down the guest file when doing the
601 * actual operations.
602 ** @todo Use the IGuestFile API for locking down the file on the guest!
603 */
604 GuestFsObjData objData; int guestRc;
605 int rc = pSession->fileQueryInfoInternal(Utf8Str(mSource), objData, &guestRc);
606 if (RT_FAILURE(rc))
607 {
608 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
609 Utf8StrFmt(GuestSession::tr("Querying guest file information for \"%s\" failed: %Rrc"),
610 mSource.c_str(), rc));
611 }
612 else if (objData.mType != FsObjType_File) /* Only single files are supported at the moment. */
613 {
614 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
615 Utf8StrFmt(GuestSession::tr("Object \"%s\" on the guest is not a file"), mSource.c_str()));
616 rc = VERR_GENERAL_FAILURE; /* Fudge. */
617 }
618
619 if (RT_SUCCESS(rc))
620 {
621 RTFILE fileDest;
622 rc = RTFileOpen(&fileDest, mDest.c_str(),
623 RTFILE_O_WRITE | RTFILE_O_OPEN_CREATE | RTFILE_O_DENY_WRITE); /** @todo Use the correct open modes! */
624 if (RT_FAILURE(rc))
625 {
626 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
627 Utf8StrFmt(GuestSession::tr("Error opening destination file \"%s\": %Rrc"),
628 mDest.c_str(), rc));
629 }
630 else
631 {
632 GuestProcessStartupInfo procInfo;
633 procInfo.mName = Utf8StrFmt(GuestSession::tr("Copying file \"%s\" from guest to the host to \"%s\" (%RI64 bytes)"),
634 mSource.c_str(), mDest.c_str(), objData.mObjectSize);
635 procInfo.mCommand = Utf8Str(VBOXSERVICE_TOOL_CAT);
636 procInfo.mFlags = ProcessCreateFlag_Hidden | ProcessCreateFlag_WaitForStdOut;
637
638 /* Set arguments.*/
639 procInfo.mArguments.push_back(mSource); /* Which file to output? */
640
641 /* Startup process. */
642 ComObjPtr<GuestProcess> pProcess;
643 rc = pSession->processCreateExInteral(procInfo, pProcess);
644 if (RT_SUCCESS(rc))
645 rc = pProcess->startProcess(30 * 1000 /* 30s timeout */,
646 &guestRc);
647 if (RT_FAILURE(rc))
648 {
649 switch (rc)
650 {
651 case VERR_GSTCTL_GUEST_ERROR:
652 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
653 GuestProcess::guestErrorToString(guestRc));
654 break;
655
656 default:
657 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
658 Utf8StrFmt(GuestSession::tr("Error while creating guest process for copying file \"%s\" from guest to host: %Rrc"),
659 mSource.c_str(), rc));
660 break;
661 }
662 }
663 else
664 {
665 ProcessWaitResult_T waitRes;
666 BYTE byBuf[_64K];
667
668 BOOL fCanceled = FALSE;
669 uint64_t cbWrittenTotal = 0;
670 uint64_t cbToRead = objData.mObjectSize;
671
672 for (;;)
673 {
674 rc = pProcess->waitFor(ProcessWaitForFlag_StdOut,
675 30 * 1000 /* Timeout */, waitRes, &guestRc);
676 if (RT_FAILURE(rc))
677 {
678 switch (rc)
679 {
680 case VERR_GSTCTL_GUEST_ERROR:
681 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
682 GuestProcess::guestErrorToString(guestRc));
683 break;
684
685 default:
686 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
687 Utf8StrFmt(GuestSession::tr("Error while creating guest process for copying file \"%s\" from guest to host: %Rrc"),
688 mSource.c_str(), rc));
689 break;
690 }
691
692 break;
693 }
694
695 if ( waitRes == ProcessWaitResult_StdOut
696 || waitRes == ProcessWaitResult_WaitFlagNotSupported)
697 {
698 /* If the guest does not support waiting for stdin, we now yield in
699 * order to reduce the CPU load due to busy waiting. */
700 if (waitRes == ProcessWaitResult_WaitFlagNotSupported)
701 RTThreadYield(); /* Optional, don't check rc. */
702
703 uint32_t cbRead = 0; /* readData can return with VWRN_GSTCTL_OBJECTSTATE_CHANGED. */
704 rc = pProcess->readData(OUTPUT_HANDLE_ID_STDOUT, sizeof(byBuf),
705 30 * 1000 /* Timeout */, byBuf, sizeof(byBuf),
706 &cbRead, &guestRc);
707 if (RT_FAILURE(rc))
708 {
709 switch (rc)
710 {
711 case VERR_GSTCTL_GUEST_ERROR:
712 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
713 GuestProcess::guestErrorToString(guestRc));
714 break;
715
716 default:
717 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
718 Utf8StrFmt(GuestSession::tr("Reading from file \"%s\" (offset %RU64) failed: %Rrc"),
719 mSource.c_str(), cbWrittenTotal, rc));
720 break;
721 }
722
723 break;
724 }
725
726 if (cbRead)
727 {
728 rc = RTFileWrite(fileDest, byBuf, cbRead, NULL /* No partial writes */);
729 if (RT_FAILURE(rc))
730 {
731 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
732 Utf8StrFmt(GuestSession::tr("Error writing to file \"%s\" (%RU64 bytes left): %Rrc"),
733 mDest.c_str(), cbToRead, rc));
734 break;
735 }
736
737 /* Only subtract bytes reported written by the guest. */
738 Assert(cbToRead >= cbRead);
739 cbToRead -= cbRead;
740
741 /* Update total bytes written to the guest. */
742 cbWrittenTotal += cbRead;
743 Assert(cbWrittenTotal <= (uint64_t)objData.mObjectSize);
744
745 /* Did the user cancel the operation above? */
746 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
747 && fCanceled)
748 break;
749
750 rc = setProgress((ULONG)(cbWrittenTotal / ((uint64_t)objData.mObjectSize / 100.0)));
751 if (RT_FAILURE(rc))
752 break;
753 }
754 }
755 else
756 {
757 break;
758 }
759
760 } /* for */
761
762 LogFlowThisFunc(("rc=%Rrc, guestrc=%Rrc, waitRes=%ld, cbWrittenTotal=%RU64, cbSize=%RI64, cbToRead=%RU64\n",
763 rc, guestRc, waitRes, cbWrittenTotal, objData.mObjectSize, cbToRead));
764
765 if ( !fCanceled
766 || RT_SUCCESS(rc))
767 {
768 /*
769 * Even if we succeeded until here make sure to check whether we really transfered
770 * everything.
771 */
772 if ( objData.mObjectSize > 0
773 && cbWrittenTotal == 0)
774 {
775 /* If nothing was transfered but the file size was > 0 then "vbox_cat" wasn't able to write
776 * to the destination -> access denied. */
777 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
778 Utf8StrFmt(GuestSession::tr("Unable to write \"%s\" to \"%s\": Access denied"),
779 mSource.c_str(), mDest.c_str()));
780 rc = VERR_GENERAL_FAILURE; /* Fudge. */
781 }
782 else if (cbWrittenTotal < (uint64_t)objData.mObjectSize)
783 {
784 /* If we did not copy all let the user know. */
785 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
786 Utf8StrFmt(GuestSession::tr("Copying file \"%s\" failed (%RU64/%RI64 bytes transfered)"),
787 mSource.c_str(), cbWrittenTotal, objData.mObjectSize));
788 rc = VERR_GENERAL_FAILURE; /* Fudge. */
789 }
790 else
791 {
792 ProcessStatus_T procStatus;
793 LONG exitCode;
794 if ( ( SUCCEEDED(pProcess->COMGETTER(Status(&procStatus)))
795 && procStatus != ProcessStatus_TerminatedNormally)
796 || ( SUCCEEDED(pProcess->COMGETTER(ExitCode(&exitCode)))
797 && exitCode != 0)
798 )
799 {
800 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
801 Utf8StrFmt(GuestSession::tr("Copying file \"%s\" failed with status %ld, exit code %d"),
802 mSource.c_str(), procStatus, exitCode)); /**@todo Add stringify methods! */
803 rc = VERR_GENERAL_FAILURE; /* Fudge. */
804 }
805 else /* Yay, success! */
806 rc = setProgressSuccess();
807 }
808 }
809 }
810
811 RTFileClose(fileDest);
812 }
813 }
814
815 LogFlowFuncLeaveRC(rc);
816 return rc;
817}
818
819int SessionTaskCopyFrom::RunAsync(const Utf8Str &strDesc, ComObjPtr<Progress> &pProgress)
820{
821 LogFlowThisFunc(("strDesc=%s, strSource=%s, strDest=%s, uFlags=%x\n",
822 strDesc.c_str(), mSource.c_str(), mDest.c_str(), mFlags));
823
824 mDesc = strDesc;
825 mProgress = pProgress;
826
827 int rc = RTThreadCreate(NULL, SessionTaskCopyFrom::taskThread, this,
828 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
829 "gctlCpyFrom");
830 LogFlowFuncLeaveRC(rc);
831 return rc;
832}
833
834/* static */
835int SessionTaskCopyFrom::taskThread(RTTHREAD Thread, void *pvUser)
836{
837 std::auto_ptr<SessionTaskCopyFrom> task(static_cast<SessionTaskCopyFrom*>(pvUser));
838 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
839
840 LogFlowFunc(("pTask=%p\n", task.get()));
841 return task->Run();
842}
843
844SessionTaskUpdateAdditions::SessionTaskUpdateAdditions(GuestSession *pSession,
845 const Utf8Str &strSource,
846 const ProcessArguments &aArguments,
847 uint32_t uFlags)
848 : GuestSessionTask(pSession)
849{
850 mSource = strSource;
851 mArguments = aArguments;
852 mFlags = uFlags;
853}
854
855SessionTaskUpdateAdditions::~SessionTaskUpdateAdditions(void)
856{
857
858}
859
860int SessionTaskUpdateAdditions::addProcessArguments(ProcessArguments &aArgumentsDest,
861 const ProcessArguments &aArgumentsSource)
862{
863 int rc = VINF_SUCCESS;
864
865 try
866 {
867 /* Filter out arguments which already are in the destination to
868 * not end up having them specified twice. Not the fastest method on the
869 * planet but does the job. */
870 ProcessArguments::const_iterator itSource = aArgumentsSource.begin();
871 while (itSource != aArgumentsSource.end())
872 {
873 bool fFound = false;
874 ProcessArguments::iterator itDest = aArgumentsDest.begin();
875 while (itDest != aArgumentsDest.end())
876 {
877 if ((*itDest).equalsIgnoreCase((*itSource)))
878 {
879 fFound = true;
880 break;
881 }
882 itDest++;
883 }
884
885 if (!fFound)
886 aArgumentsDest.push_back((*itSource));
887
888 itSource++;
889 }
890 }
891 catch(std::bad_alloc &)
892 {
893 return VERR_NO_MEMORY;
894 }
895
896 return rc;
897}
898
899int SessionTaskUpdateAdditions::copyFileToGuest(GuestSession *pSession, PRTISOFSFILE pISO,
900 Utf8Str const &strFileSource, const Utf8Str &strFileDest,
901 bool fOptional, uint32_t *pcbSize)
902{
903 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
904 AssertPtrReturn(pISO, VERR_INVALID_POINTER);
905 /* pcbSize is optional. */
906
907 uint32_t cbOffset;
908 size_t cbSize;
909
910 int rc = RTIsoFsGetFileInfo(pISO, strFileSource.c_str(), &cbOffset, &cbSize);
911 if (RT_FAILURE(rc))
912 {
913 if (fOptional)
914 return VINF_SUCCESS;
915
916 return rc;
917 }
918
919 Assert(cbOffset);
920 Assert(cbSize);
921 rc = RTFileSeek(pISO->file, cbOffset, RTFILE_SEEK_BEGIN, NULL);
922
923 /* Copy over the Guest Additions file to the guest. */
924 if (RT_SUCCESS(rc))
925 {
926 LogRel(("Copying Guest Additions installer file \"%s\" to \"%s\" on guest ...\n",
927 strFileSource.c_str(), strFileDest.c_str()));
928
929 if (RT_SUCCESS(rc))
930 {
931 SessionTaskCopyTo *pTask = new SessionTaskCopyTo(pSession /* GuestSession */,
932 &pISO->file, cbOffset, cbSize,
933 strFileDest, CopyFileFlag_None);
934 AssertPtrReturn(pTask, VERR_NO_MEMORY);
935
936 ComObjPtr<Progress> pProgressCopyTo;
937 rc = pSession->startTaskAsync(Utf8StrFmt(GuestSession::tr("Copying Guest Additions installer file \"%s\" to \"%s\" on guest"),
938 mSource.c_str(), strFileDest.c_str()),
939 pTask, pProgressCopyTo);
940 if (RT_SUCCESS(rc))
941 {
942 BOOL fCanceled = FALSE;
943 HRESULT hr = pProgressCopyTo->WaitForCompletion(-1);
944 if ( SUCCEEDED(pProgressCopyTo->COMGETTER(Canceled)(&fCanceled))
945 && fCanceled)
946 {
947 rc = VERR_GENERAL_FAILURE; /* Fudge. */
948 }
949 else if (FAILED(hr))
950 {
951 Assert(FAILED(hr));
952 rc = VERR_GENERAL_FAILURE; /* Fudge. */
953 }
954 }
955 }
956 }
957
958 /** @todo Note: Since there is no file locking involved at the moment, there can be modifications
959 * between finished copying, the verification and the actual execution. */
960
961 /* Determine where the installer image ended up and if it has the correct size. */
962 if (RT_SUCCESS(rc))
963 {
964 LogRel(("Verifying Guest Additions installer file \"%s\" ...\n",
965 strFileDest.c_str()));
966
967 GuestFsObjData objData;
968 int64_t cbSizeOnGuest; int guestRc;
969 rc = pSession->fileQuerySizeInternal(strFileDest, &cbSizeOnGuest, &guestRc);
970 if ( RT_SUCCESS(rc)
971 && cbSize == (uint64_t)cbSizeOnGuest)
972 {
973 LogFlowThisFunc(("Guest Additions installer file \"%s\" successfully verified\n",
974 strFileDest.c_str()));
975 }
976 else
977 {
978 if (RT_SUCCESS(rc)) /* Size does not match. */
979 {
980 LogRel(("Size of Guest Additions installer file \"%s\" does not match: %RI64 bytes copied, %RU64 bytes expected\n",
981 strFileDest.c_str(), cbSizeOnGuest, cbSize));
982 rc = VERR_BROKEN_PIPE; /** @todo Find a better error. */
983 }
984 else
985 {
986 switch (rc)
987 {
988 case VERR_GSTCTL_GUEST_ERROR:
989 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
990 GuestProcess::guestErrorToString(guestRc));
991 break;
992
993 default:
994 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
995 Utf8StrFmt(GuestSession::tr("Error while querying size for file \"%s\": %Rrc"),
996 strFileDest.c_str(), rc));
997 break;
998 }
999 }
1000 }
1001
1002 if (RT_SUCCESS(rc))
1003 {
1004 if (pcbSize)
1005 *pcbSize = (uint32_t)cbSizeOnGuest;
1006 }
1007 }
1008
1009 return rc;
1010}
1011
1012int SessionTaskUpdateAdditions::runFileOnGuest(GuestSession *pSession, GuestProcessStartupInfo &procInfo)
1013{
1014 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
1015
1016 LogRel(("Running %s ...\n", procInfo.mName.c_str()));
1017
1018 LONG exitCode;
1019 GuestProcessTool procTool; int guestRc;
1020 int vrc = procTool.Init(pSession, procInfo, false /* Async */, &guestRc);
1021 if (RT_SUCCESS(vrc))
1022 {
1023 if (RT_SUCCESS(guestRc))
1024 vrc = procTool.Wait(GUESTPROCESSTOOL_FLAG_NONE, &guestRc);
1025 if (RT_SUCCESS(vrc))
1026 vrc = procTool.TerminatedOk(&exitCode);
1027 }
1028
1029 if (RT_FAILURE(vrc))
1030 {
1031 switch (vrc)
1032 {
1033 case VERR_NOT_EQUAL: /** @todo Special guest control rc needed! */
1034 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1035 Utf8StrFmt(GuestSession::tr("Running update file \"%s\" on guest terminated with exit code %ld"),
1036 procInfo.mCommand.c_str(), exitCode));
1037 break;
1038
1039 case VERR_GSTCTL_GUEST_ERROR:
1040 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1041 GuestProcess::guestErrorToString(guestRc));
1042 break;
1043
1044 case VERR_INVALID_STATE: /** @todo Special guest control rc needed! */
1045 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1046 Utf8StrFmt(GuestSession::tr("Update file \"%s\" reported invalid running state"),
1047 procInfo.mCommand.c_str()));
1048 break;
1049
1050 default:
1051 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1052 Utf8StrFmt(GuestSession::tr("Error while running update file \"%s\" on guest: %Rrc"),
1053 procInfo.mCommand.c_str(), vrc));
1054 break;
1055 }
1056 }
1057
1058 return vrc;
1059}
1060
1061int SessionTaskUpdateAdditions::Run(void)
1062{
1063 LogFlowThisFuncEnter();
1064
1065 ComObjPtr<GuestSession> pSession = mSession;
1066 Assert(!pSession.isNull());
1067
1068 AutoCaller autoCaller(pSession);
1069 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1070
1071 int rc = setProgress(10);
1072 if (RT_FAILURE(rc))
1073 return rc;
1074
1075 HRESULT hr = S_OK;
1076
1077 LogRel(("Automatic update of Guest Additions started, using \"%s\"\n", mSource.c_str()));
1078
1079 ComObjPtr<Guest> pGuest(mSession->getParent());
1080#if 0
1081 /*
1082 * Wait for the guest being ready within 30 seconds.
1083 */
1084 AdditionsRunLevelType_T addsRunLevel;
1085 uint64_t tsStart = RTTimeSystemMilliTS();
1086 while ( SUCCEEDED(hr = pGuest->COMGETTER(AdditionsRunLevel)(&addsRunLevel))
1087 && ( addsRunLevel != AdditionsRunLevelType_Userland
1088 && addsRunLevel != AdditionsRunLevelType_Desktop))
1089 {
1090 if ((RTTimeSystemMilliTS() - tsStart) > 30 * 1000)
1091 {
1092 rc = VERR_TIMEOUT;
1093 break;
1094 }
1095
1096 RTThreadSleep(100); /* Wait a bit. */
1097 }
1098
1099 if (FAILED(hr)) rc = VERR_TIMEOUT;
1100 if (rc == VERR_TIMEOUT)
1101 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1102 Utf8StrFmt(GuestSession::tr("Guest Additions were not ready within time, giving up")));
1103#else
1104 /*
1105 * For use with the GUI we don't want to wait, just return so that the manual .ISO mounting
1106 * can continue.
1107 */
1108 AdditionsRunLevelType_T addsRunLevel;
1109 if ( FAILED(hr = pGuest->COMGETTER(AdditionsRunLevel)(&addsRunLevel))
1110 || ( addsRunLevel != AdditionsRunLevelType_Userland
1111 && addsRunLevel != AdditionsRunLevelType_Desktop))
1112 {
1113 if (addsRunLevel == AdditionsRunLevelType_System)
1114 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1115 Utf8StrFmt(GuestSession::tr("Guest Additions are installed but not fully loaded yet, aborting automatic update")));
1116 else
1117 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1118 Utf8StrFmt(GuestSession::tr("Guest Additions not installed or ready, aborting automatic update")));
1119 rc = VERR_NOT_SUPPORTED;
1120 }
1121#endif
1122
1123 if (RT_SUCCESS(rc))
1124 {
1125 /*
1126 * Determine if we are able to update automatically. This only works
1127 * if there are recent Guest Additions installed already.
1128 */
1129 Utf8Str strAddsVer;
1130 rc = getGuestProperty(pGuest, "/VirtualBox/GuestAdd/Version", strAddsVer);
1131 if ( RT_SUCCESS(rc)
1132 && RTStrVersionCompare(strAddsVer.c_str(), "4.1") < 0)
1133 {
1134 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1135 Utf8StrFmt(GuestSession::tr("Guest has too old Guest Additions (%s) installed for automatic updating, please update manually"),
1136 strAddsVer.c_str()));
1137 rc = VERR_NOT_SUPPORTED;
1138 }
1139 }
1140
1141 Utf8Str strOSVer;
1142 eOSType osType = eOSType_Unknown;
1143 if (RT_SUCCESS(rc))
1144 {
1145 /*
1146 * Determine guest OS type and the required installer image.
1147 */
1148 Utf8Str strOSType;
1149 rc = getGuestProperty(pGuest, "/VirtualBox/GuestInfo/OS/Product", strOSType);
1150 if (RT_SUCCESS(rc))
1151 {
1152 if ( strOSType.contains("Microsoft", Utf8Str::CaseInsensitive)
1153 || strOSType.contains("Windows", Utf8Str::CaseInsensitive))
1154 {
1155 osType = eOSType_Windows;
1156
1157 /*
1158 * Determine guest OS version.
1159 */
1160 rc = getGuestProperty(pGuest, "/VirtualBox/GuestInfo/OS/Release", strOSVer);
1161 if (RT_FAILURE(rc))
1162 {
1163 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1164 Utf8StrFmt(GuestSession::tr("Unable to detected guest OS version, please update manually")));
1165 rc = VERR_NOT_SUPPORTED;
1166 }
1167
1168 /* Because Windows 2000 + XP and is bitching with WHQL popups even if we have signed drivers we
1169 * can't do automated updates here. */
1170 /* Windows XP 64-bit (5.2) is a Windows 2003 Server actually, so skip this here. */
1171 if ( RT_SUCCESS(rc)
1172 && RTStrVersionCompare(strOSVer.c_str(), "5.0") >= 0)
1173 {
1174 if ( strOSVer.startsWith("5.0") /* Exclude the build number. */
1175 || strOSVer.startsWith("5.1") /* Exclude the build number. */)
1176 {
1177 /* If we don't have AdditionsUpdateFlag_WaitForUpdateStartOnly set we can't continue
1178 * because the Windows Guest Additions installer will fail because of WHQL popups. If the
1179 * flag is set this update routine ends successfully as soon as the installer was started
1180 * (and the user has to deal with it in the guest). */
1181 if (!(mFlags & AdditionsUpdateFlag_WaitForUpdateStartOnly))
1182 {
1183 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1184 Utf8StrFmt(GuestSession::tr("Windows 2000 and XP are not supported for automatic updating due to WHQL interaction, please update manually")));
1185 rc = VERR_NOT_SUPPORTED;
1186 }
1187 }
1188 }
1189 else
1190 {
1191 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1192 Utf8StrFmt(GuestSession::tr("%s (%s) not supported for automatic updating, please update manually"),
1193 strOSType.c_str(), strOSVer.c_str()));
1194 rc = VERR_NOT_SUPPORTED;
1195 }
1196 }
1197 else if (strOSType.contains("Solaris", Utf8Str::CaseInsensitive))
1198 {
1199 osType = eOSType_Solaris;
1200 }
1201 else /* Everything else hopefully means Linux :-). */
1202 osType = eOSType_Linux;
1203
1204#if 1 /* Only Windows is supported (and tested) at the moment. */
1205 if ( RT_SUCCESS(rc)
1206 && osType != eOSType_Windows)
1207 {
1208 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
1209 Utf8StrFmt(GuestSession::tr("Detected guest OS (%s) does not support automatic Guest Additions updating, please update manually"),
1210 strOSType.c_str()));
1211 rc = VERR_NOT_SUPPORTED;
1212 }
1213#endif
1214 }
1215 }
1216
1217 RTISOFSFILE iso;
1218 if (RT_SUCCESS(rc))
1219 {
1220 /*
1221 * Try to open the .ISO file to extract all needed files.
1222 */
1223 rc = RTIsoFsOpen(&iso, mSource.c_str());
1224 if (RT_FAILURE(rc))
1225 {
1226 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1227 Utf8StrFmt(GuestSession::tr("Unable to open Guest Additions .ISO file \"%s\": %Rrc"),
1228 mSource.c_str(), rc));
1229 }
1230 else
1231 {
1232 /* Set default installation directories. */
1233 Utf8Str strUpdateDir = "/tmp/";
1234 if (osType == eOSType_Windows)
1235 strUpdateDir = "C:\\Temp\\";
1236
1237 rc = setProgress(5);
1238
1239 /* Try looking up the Guest Additions installation directory. */
1240 if (RT_SUCCESS(rc))
1241 {
1242 /* Try getting the installed Guest Additions version to know whether we
1243 * can install our temporary Guest Addition data into the original installation
1244 * directory.
1245 *
1246 * Because versions prior to 4.2 had bugs wrt spaces in paths we have to choose
1247 * a different location then.
1248 */
1249 bool fUseInstallDir = false;
1250
1251 Utf8Str strAddsVer;
1252 rc = getGuestProperty(pGuest, "/VirtualBox/GuestAdd/Version", strAddsVer);
1253 if ( RT_SUCCESS(rc)
1254 && RTStrVersionCompare(strAddsVer.c_str(), "4.2r80329") > 0)
1255 {
1256 fUseInstallDir = true;
1257 }
1258
1259 if (fUseInstallDir)
1260 {
1261 if (RT_SUCCESS(rc))
1262 rc = getGuestProperty(pGuest, "/VirtualBox/GuestAdd/InstallDir", strUpdateDir);
1263 if (RT_SUCCESS(rc))
1264 {
1265 if (osType == eOSType_Windows)
1266 {
1267 strUpdateDir.findReplace('/', '\\');
1268 strUpdateDir.append("\\Update\\");
1269 }
1270 else
1271 strUpdateDir.append("/update/");
1272 }
1273 }
1274 }
1275
1276 if (RT_SUCCESS(rc))
1277 LogRel(("Guest Additions update directory is: %s\n",
1278 strUpdateDir.c_str()));
1279
1280 /* Create the installation directory. */
1281 int guestRc;
1282 rc = pSession->directoryCreateInternal(strUpdateDir,
1283 755 /* Mode */, DirectoryCreateFlag_Parents, &guestRc);
1284 if (RT_FAILURE(rc))
1285 {
1286 switch (rc)
1287 {
1288 case VERR_GSTCTL_GUEST_ERROR:
1289 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1290 GuestProcess::guestErrorToString(guestRc));
1291 break;
1292
1293 default:
1294 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1295 Utf8StrFmt(GuestSession::tr("Error creating installation directory \"%s\" on the guest: %Rrc"),
1296 strUpdateDir.c_str(), rc));
1297 break;
1298 }
1299 }
1300 if (RT_SUCCESS(rc))
1301 rc = setProgress(10);
1302
1303 if (RT_SUCCESS(rc))
1304 {
1305 /* Prepare the file(s) we want to copy over to the guest and
1306 * (maybe) want to run. */
1307 switch (osType)
1308 {
1309 case eOSType_Windows:
1310 {
1311 /* Do we need to install our certificates? We do this for W2K and up. */
1312 bool fInstallCert = false;
1313
1314 /* Only Windows 2000 and up need certificates to be installed. */
1315 if (RTStrVersionCompare(strOSVer.c_str(), "5.0") >= 0)
1316 {
1317 fInstallCert = true;
1318 LogRel(("Certificates for auto updating WHQL drivers will be installed\n"));
1319 }
1320 else
1321 LogRel(("Skipping installation of certificates for WHQL drivers\n"));
1322
1323 if (fInstallCert)
1324 {
1325 /* Our certificate. */
1326 mFiles.push_back(InstallerFile("CERT/ORACLE_VBOX.CER",
1327 strUpdateDir + "oracle-vbox.cer",
1328 UPDATEFILE_FLAG_COPY_FROM_ISO | UPDATEFILE_FLAG_OPTIONAL));
1329 /* Our certificate installation utility. */
1330 /* First pass: Copy over the file + execute it to remove any existing
1331 * VBox certificates. */
1332 GuestProcessStartupInfo siCertUtilRem;
1333 siCertUtilRem.mName = "VirtualBox Certificate Utility, removing old VirtualBox certificates";
1334 siCertUtilRem.mArguments.push_back(Utf8Str("remove-trusted-publisher"));
1335 siCertUtilRem.mArguments.push_back(Utf8Str("--root")); /* Add root certificate as well. */
1336 siCertUtilRem.mArguments.push_back(Utf8Str(strUpdateDir + "oracle-vbox.cer"));
1337 siCertUtilRem.mArguments.push_back(Utf8Str(strUpdateDir + "oracle-vbox.cer"));
1338 mFiles.push_back(InstallerFile("CERT/VBOXCERTUTIL.EXE",
1339 strUpdateDir + "VBoxCertUtil.exe",
1340 UPDATEFILE_FLAG_COPY_FROM_ISO | UPDATEFILE_FLAG_EXECUTE | UPDATEFILE_FLAG_OPTIONAL,
1341 siCertUtilRem));
1342 /* Second pass: Only execute (but don't copy) again, this time installng the
1343 * recent certificates just copied over. */
1344 GuestProcessStartupInfo siCertUtilAdd;
1345 siCertUtilAdd.mName = "VirtualBox Certificate Utility, installing VirtualBox certificates";
1346 siCertUtilAdd.mArguments.push_back(Utf8Str("add-trusted-publisher"));
1347 siCertUtilAdd.mArguments.push_back(Utf8Str("--root")); /* Add root certificate as well. */
1348 siCertUtilAdd.mArguments.push_back(Utf8Str(strUpdateDir + "oracle-vbox.cer"));
1349 siCertUtilAdd.mArguments.push_back(Utf8Str(strUpdateDir + "oracle-vbox.cer"));
1350 mFiles.push_back(InstallerFile("CERT/VBOXCERTUTIL.EXE",
1351 strUpdateDir + "VBoxCertUtil.exe",
1352 UPDATEFILE_FLAG_EXECUTE | UPDATEFILE_FLAG_OPTIONAL,
1353 siCertUtilAdd));
1354 }
1355 /* The installers in different flavors, as we don't know (and can't assume)
1356 * the guest's bitness. */
1357 mFiles.push_back(InstallerFile("VBOXWINDOWSADDITIONS_X86.EXE",
1358 strUpdateDir + "VBoxWindowsAdditions-x86.exe",
1359 UPDATEFILE_FLAG_COPY_FROM_ISO));
1360 mFiles.push_back(InstallerFile("VBOXWINDOWSADDITIONS_AMD64.EXE",
1361 strUpdateDir + "VBoxWindowsAdditions-amd64.exe",
1362 UPDATEFILE_FLAG_COPY_FROM_ISO));
1363 /* The stub loader which decides which flavor to run. */
1364 GuestProcessStartupInfo siInstaller;
1365 siInstaller.mName = "VirtualBox Windows Guest Additions Installer";
1366 /* Set a running timeout of 5 minutes -- the Windows Guest Additions
1367 * setup can take quite a while, so be on the safe side. */
1368 siInstaller.mTimeoutMS = 5 * 60 * 1000;
1369 siInstaller.mArguments.push_back(Utf8Str("/S")); /* We want to install in silent mode. */
1370 siInstaller.mArguments.push_back(Utf8Str("/l")); /* ... and logging enabled. */
1371 /* Don't quit VBoxService during upgrade because it still is used for this
1372 * piece of code we're in right now (that is, here!) ... */
1373 siInstaller.mArguments.push_back(Utf8Str("/no_vboxservice_exit"));
1374 /* Tell the installer to report its current installation status
1375 * using a running VBoxTray instance via balloon messages in the
1376 * Windows taskbar. */
1377 siInstaller.mArguments.push_back(Utf8Str("/post_installstatus"));
1378 /* Add optional installer command line arguments from the API to the
1379 * installer's startup info. */
1380 rc = addProcessArguments(siInstaller.mArguments, mArguments);
1381 AssertRC(rc);
1382 /* If the caller does not want to wait for out guest update process to end,
1383 * complete the progress object now so that the caller can do other work. */
1384 if (mFlags & AdditionsUpdateFlag_WaitForUpdateStartOnly)
1385 siInstaller.mFlags |= ProcessCreateFlag_WaitForProcessStartOnly;
1386 mFiles.push_back(InstallerFile("VBOXWINDOWSADDITIONS.EXE",
1387 strUpdateDir + "VBoxWindowsAdditions.exe",
1388 UPDATEFILE_FLAG_COPY_FROM_ISO | UPDATEFILE_FLAG_EXECUTE, siInstaller));
1389 break;
1390 }
1391 case eOSType_Linux:
1392 /** @todo Add Linux support. */
1393 break;
1394 case eOSType_Solaris:
1395 /** @todo Add Solaris support. */
1396 break;
1397 default:
1398 AssertReleaseMsgFailed(("Unsupported guest type: %d\n", osType));
1399 break;
1400 }
1401 }
1402
1403 if (RT_SUCCESS(rc))
1404 {
1405 /* We want to spend 40% total for all copying operations. So roughly
1406 * calculate the specific percentage step of each copied file. */
1407 uint8_t uOffset = 20; /* Start at 20%. */
1408 uint8_t uStep = 40 / mFiles.size();
1409
1410 LogRel(("Copying over Guest Additions update files to the guest ...\n"));
1411
1412 std::vector<InstallerFile>::const_iterator itFiles = mFiles.begin();
1413 while (itFiles != mFiles.end())
1414 {
1415 if (itFiles->fFlags & UPDATEFILE_FLAG_COPY_FROM_ISO)
1416 {
1417 bool fOptional = false;
1418 if (itFiles->fFlags & UPDATEFILE_FLAG_OPTIONAL)
1419 fOptional = true;
1420 rc = copyFileToGuest(pSession, &iso, itFiles->strSource, itFiles->strDest,
1421 fOptional, NULL /* cbSize */);
1422 if (RT_FAILURE(rc))
1423 {
1424 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1425 Utf8StrFmt(GuestSession::tr("Error while copying file \"%s\" to \"%s\" on the guest: %Rrc"),
1426 itFiles->strSource.c_str(), itFiles->strDest.c_str(), rc));
1427 break;
1428 }
1429 }
1430
1431 rc = setProgress(uOffset);
1432 if (RT_FAILURE(rc))
1433 break;
1434 uOffset += uStep;
1435
1436 itFiles++;
1437 }
1438 }
1439
1440 /* Done copying, close .ISO file. */
1441 RTIsoFsClose(&iso);
1442
1443 if (RT_SUCCESS(rc))
1444 {
1445 /* We want to spend 35% total for all copying operations. So roughly
1446 * calculate the specific percentage step of each copied file. */
1447 uint8_t uOffset = 60; /* Start at 60%. */
1448 uint8_t uStep = 35 / mFiles.size();
1449
1450 LogRel(("Executing Guest Additions update files ...\n"));
1451
1452 std::vector<InstallerFile>::iterator itFiles = mFiles.begin();
1453 while (itFiles != mFiles.end())
1454 {
1455 if (itFiles->fFlags & UPDATEFILE_FLAG_EXECUTE)
1456 {
1457 rc = runFileOnGuest(pSession, itFiles->mProcInfo);
1458 if (RT_FAILURE(rc))
1459 break;
1460 }
1461
1462 rc = setProgress(uOffset);
1463 if (RT_FAILURE(rc))
1464 break;
1465 uOffset += uStep;
1466
1467 itFiles++;
1468 }
1469 }
1470
1471 if (RT_SUCCESS(rc))
1472 {
1473 LogRel(("Automatic update of Guest Additions succeeded\n"));
1474 rc = setProgressSuccess();
1475 }
1476 }
1477 }
1478
1479 if (RT_FAILURE(rc))
1480 {
1481 if (rc == VERR_CANCELLED)
1482 {
1483 LogRel(("Automatic update of Guest Additions was canceled\n"));
1484
1485 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1486 Utf8StrFmt(GuestSession::tr("Installation was canceled")));
1487 }
1488 else
1489 {
1490 Utf8Str strError = Utf8StrFmt("No further error information available (%Rrc)", rc);
1491 if (!mProgress.isNull()) /* Progress object is optional. */
1492 {
1493 com::ProgressErrorInfo errorInfo(mProgress);
1494 if ( errorInfo.isFullAvailable()
1495 || errorInfo.isBasicAvailable())
1496 {
1497 strError = errorInfo.getText();
1498 }
1499 }
1500
1501 LogRel(("Automatic update of Guest Additions failed: %s (%Rhrc)\n",
1502 strError.c_str(), hr));
1503 }
1504
1505 LogRel(("Please install Guest Additions manually\n"));
1506 }
1507
1508 /** @todo Clean up copied / left over installation files. */
1509
1510 LogFlowFuncLeaveRC(rc);
1511 return rc;
1512}
1513
1514int SessionTaskUpdateAdditions::RunAsync(const Utf8Str &strDesc, ComObjPtr<Progress> &pProgress)
1515{
1516 LogFlowThisFunc(("strDesc=%s, strSource=%s, uFlags=%x\n",
1517 strDesc.c_str(), mSource.c_str(), mFlags));
1518
1519 mDesc = strDesc;
1520 mProgress = pProgress;
1521
1522 int rc = RTThreadCreate(NULL, SessionTaskUpdateAdditions::taskThread, this,
1523 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
1524 "gctlUpGA");
1525 LogFlowFuncLeaveRC(rc);
1526 return rc;
1527}
1528
1529/* static */
1530int SessionTaskUpdateAdditions::taskThread(RTTHREAD Thread, void *pvUser)
1531{
1532 std::auto_ptr<SessionTaskUpdateAdditions> task(static_cast<SessionTaskUpdateAdditions*>(pvUser));
1533 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
1534
1535 LogFlowFunc(("pTask=%p\n", task.get()));
1536 return task->Run();
1537}
1538
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