VirtualBox

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

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

Guest Control/Main: Added stubs for IGuestSession:copy[From|To]Guest() (@bugref{9135}) and implemented internal source specification handling for sharing as much code as possible across all the guest session copy methods. Work in progress.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 93.7 KB
Line 
1/* $Id: GuestSessionImplTasks.cpp 71976 2018-04-23 10:52:51Z vboxsync $ */
2/** @file
3 * VirtualBox Main - Guest session tasks.
4 */
5
6/*
7 * Copyright (C) 2012-2018 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#define LOG_GROUP LOG_GROUP_MAIN_GUESTSESSION
23#include "LoggingNew.h"
24
25#include "GuestImpl.h"
26#ifndef VBOX_WITH_GUEST_CONTROL
27# error "VBOX_WITH_GUEST_CONTROL must defined in this file"
28#endif
29#include "GuestSessionImpl.h"
30#include "GuestSessionImplTasks.h"
31#include "GuestCtrlImplPrivate.h"
32
33#include "Global.h"
34#include "AutoCaller.h"
35#include "ConsoleImpl.h"
36#include "ProgressImpl.h"
37
38#include <memory> /* For auto_ptr. */
39
40#include <iprt/env.h>
41#include <iprt/file.h> /* For CopyTo/From. */
42#include <iprt/dir.h>
43#include <iprt/path.h>
44
45
46/*********************************************************************************************************************************
47* Defines *
48*********************************************************************************************************************************/
49
50/**
51 * (Guest Additions) ISO file flags.
52 * Needed for handling Guest Additions updates.
53 */
54#define ISOFILE_FLAG_NONE 0
55/** Copy over the file from host to the
56 * guest. */
57#define ISOFILE_FLAG_COPY_FROM_ISO RT_BIT(0)
58/** Execute file on the guest after it has
59 * been successfully transfered. */
60#define ISOFILE_FLAG_EXECUTE RT_BIT(7)
61/** File is optional, does not have to be
62 * existent on the .ISO. */
63#define ISOFILE_FLAG_OPTIONAL RT_BIT(8)
64
65
66// session task classes
67/////////////////////////////////////////////////////////////////////////////
68
69GuestSessionTask::GuestSessionTask(GuestSession *pSession)
70 : ThreadTask("GenericGuestSessionTask")
71{
72 mSession = pSession;
73
74 switch (mSession->i_getPathStyle())
75 {
76 case PathStyle_DOS:
77 mfPathStyle = RTPATH_STR_F_STYLE_DOS;
78 mPathStyle = "\\";
79 break;
80
81 default:
82 mfPathStyle = RTPATH_STR_F_STYLE_UNIX;
83 mPathStyle = "/";
84 break;
85 }
86}
87
88GuestSessionTask::~GuestSessionTask(void)
89{
90}
91
92int GuestSessionTask::createAndSetProgressObject(ULONG cOperations /* = 1 */)
93{
94 LogFlowThisFunc(("cOperations=%ld\n", cOperations));
95
96 /* Create the progress object. */
97 ComObjPtr<Progress> pProgress;
98 HRESULT hr = pProgress.createObject();
99 if (FAILED(hr))
100 return VERR_COM_UNEXPECTED;
101
102 hr = pProgress->init(static_cast<IGuestSession*>(mSession),
103 Bstr(mDesc).raw(),
104 TRUE /* aCancelable */, cOperations, Bstr(mDesc).raw());
105 if (FAILED(hr))
106 return VERR_COM_UNEXPECTED;
107
108 mProgress = pProgress;
109
110 LogFlowFuncLeave();
111 return VINF_SUCCESS;
112}
113
114int GuestSessionTask::RunAsync(const Utf8Str &strDesc, ComObjPtr<Progress> &pProgress)
115{
116 LogFlowThisFunc(("strDesc=%s\n", strDesc.c_str()));
117
118 mDesc = strDesc;
119 mProgress = pProgress;
120 HRESULT hrc = createThreadWithType(RTTHREADTYPE_MAIN_HEAVY_WORKER);
121
122 LogFlowThisFunc(("Returning hrc=%Rhrc\n", hrc));
123 return Global::vboxStatusCodeToCOM(hrc);
124}
125
126
127int GuestSessionTask::getGuestProperty(const ComObjPtr<Guest> &pGuest,
128 const Utf8Str &strPath, Utf8Str &strValue)
129{
130 ComObjPtr<Console> pConsole = pGuest->i_getConsole();
131 const ComPtr<IMachine> pMachine = pConsole->i_machine();
132
133 Assert(!pMachine.isNull());
134 Bstr strTemp, strFlags;
135 LONG64 i64Timestamp;
136 HRESULT hr = pMachine->GetGuestProperty(Bstr(strPath).raw(),
137 strTemp.asOutParam(),
138 &i64Timestamp, strFlags.asOutParam());
139 if (SUCCEEDED(hr))
140 {
141 strValue = strTemp;
142 return VINF_SUCCESS;
143 }
144 return VERR_NOT_FOUND;
145}
146
147int GuestSessionTask::setProgress(ULONG uPercent)
148{
149 if (mProgress.isNull()) /* Progress is optional. */
150 return VINF_SUCCESS;
151
152 BOOL fCanceled;
153 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
154 && fCanceled)
155 return VERR_CANCELLED;
156 BOOL fCompleted;
157 if ( SUCCEEDED(mProgress->COMGETTER(Completed(&fCompleted)))
158 && fCompleted)
159 {
160 AssertMsgFailed(("Setting value of an already completed progress\n"));
161 return VINF_SUCCESS;
162 }
163 HRESULT hr = mProgress->SetCurrentOperationProgress(uPercent);
164 if (FAILED(hr))
165 return VERR_COM_UNEXPECTED;
166
167 return VINF_SUCCESS;
168}
169
170int GuestSessionTask::setProgressSuccess(void)
171{
172 if (mProgress.isNull()) /* Progress is optional. */
173 return VINF_SUCCESS;
174
175 BOOL fCompleted;
176 if ( SUCCEEDED(mProgress->COMGETTER(Completed(&fCompleted)))
177 && !fCompleted)
178 {
179 HRESULT hr = mProgress->i_notifyComplete(S_OK);
180 if (FAILED(hr))
181 return VERR_COM_UNEXPECTED; /** @todo Find a better rc. */
182 }
183
184 return VINF_SUCCESS;
185}
186
187HRESULT GuestSessionTask::setProgressErrorMsg(HRESULT hr, const Utf8Str &strMsg)
188{
189 LogFlowFunc(("hr=%Rhrc, strMsg=%s\n",
190 hr, strMsg.c_str()));
191
192 if (mProgress.isNull()) /* Progress is optional. */
193 return hr; /* Return original rc. */
194
195 BOOL fCanceled;
196 BOOL fCompleted;
197 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
198 && !fCanceled
199 && SUCCEEDED(mProgress->COMGETTER(Completed(&fCompleted)))
200 && !fCompleted)
201 {
202 HRESULT hr2 = mProgress->i_notifyComplete(hr,
203 COM_IIDOF(IGuestSession),
204 GuestSession::getStaticComponentName(),
205 strMsg.c_str());
206 if (FAILED(hr2))
207 return hr2;
208 }
209 return hr; /* Return original rc. */
210}
211
212/**
213 * Creates a directory on the guest.
214 *
215 * @return VBox status code. VWRN_ALREADY_EXISTS if directory on the guest already exists.
216 * @param strPath Absolute path to directory on the guest (guest style path) to create.
217 * @param enmDirectoryCreateFlags Directory creation flags.
218 * @param uMode Directory mode to use for creation.
219 * @param fFollowSymlinks Whether to follow symlinks on the guest or not.
220 */
221int GuestSessionTask::directoryCreate(const com::Utf8Str &strPath,
222 DirectoryCreateFlag_T enmDirectoryCreateFlags, uint32_t uMode, bool fFollowSymlinks)
223{
224 LogFlowFunc(("strPath=%s, fFlags=0x%x, uMode=%RU32, fFollowSymlinks=%RTbool\n",
225 strPath.c_str(), enmDirectoryCreateFlags, uMode, fFollowSymlinks));
226
227 GuestFsObjData objData; int rcGuest;
228 int rc = mSession->i_directoryQueryInfo(strPath, fFollowSymlinks, objData, &rcGuest);
229 if (RT_SUCCESS(rc))
230 {
231 return VWRN_ALREADY_EXISTS;
232 }
233 else
234 {
235 switch (rc)
236 {
237 case VERR_GSTCTL_GUEST_ERROR:
238 {
239 switch (rcGuest)
240 {
241 case VERR_FILE_NOT_FOUND:
242 case VERR_PATH_NOT_FOUND:
243 rc = mSession->i_directoryCreate(strPath.c_str(), uMode, enmDirectoryCreateFlags, &rcGuest);
244 break;
245 default:
246 break;
247 }
248 break;
249 }
250
251 default:
252 break;
253 }
254 }
255
256 if (RT_FAILURE(rc))
257 {
258 if (rc == VERR_GSTCTL_GUEST_ERROR)
259 {
260 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestProcess::i_guestErrorToString(rcGuest));
261 }
262 else
263 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
264 Utf8StrFmt(GuestSession::tr("Error creating directory on the guest: %Rrc"), strPath.c_str(), rc));
265 }
266
267 LogFlowFuncLeaveRC(rc);
268 return rc;
269}
270
271/**
272 * Main function for copying a file from guest to the host.
273 *
274 * @return VBox status code.
275 * @param srcFile Guest file (source) to copy to the host. Must be in opened and ready state already.
276 * @param phDstFile Pointer to host file handle (destination) to copy to. Must be in opened and ready state already.
277 * @param fFileCopyFlags File copy flags.
278 * @param offCopy Offset (in bytes) where to start copying the source file.
279 * @param cbSize Size (in bytes) to copy from the source file.
280 */
281int GuestSessionTask::fileCopyFromGuestInner(ComObjPtr<GuestFile> &srcFile, PRTFILE phDstFile, FileCopyFlag_T fFileCopyFlags,
282 uint64_t offCopy, uint64_t cbSize)
283{
284 RT_NOREF(fFileCopyFlags);
285
286 BOOL fCanceled = FALSE;
287 uint64_t cbWrittenTotal = 0;
288 uint64_t cbToRead = cbSize;
289
290 uint32_t uTimeoutMs = 30 * 1000; /* 30s timeout. */
291
292 int rc = VINF_SUCCESS;
293
294 if (offCopy)
295 {
296 uint64_t offActual;
297 rc = srcFile->i_seekAt(offCopy, GUEST_FILE_SEEKTYPE_BEGIN, uTimeoutMs, &offActual);
298 if (RT_FAILURE(rc))
299 {
300 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
301 Utf8StrFmt(GuestSession::tr("Seeking to offset %RU64 failed: %Rrc"), offCopy, rc));
302 return rc;
303 }
304 }
305
306 BYTE byBuf[_64K];
307 while (cbToRead)
308 {
309 uint32_t cbRead;
310 const uint32_t cbChunk = RT_MIN(cbToRead, sizeof(byBuf));
311 rc = srcFile->i_readData(cbChunk, uTimeoutMs, byBuf, sizeof(byBuf), &cbRead);
312 if (RT_FAILURE(rc))
313 {
314 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
315 Utf8StrFmt(GuestSession::tr("Reading %RU32 bytes @ %RU64, failed: %Rrc"), cbChunk, cbWrittenTotal, rc));
316 break;
317 }
318
319 rc = RTFileWrite(*phDstFile, byBuf, cbRead, NULL /* No partial writes */);
320 if (RT_FAILURE(rc))
321 {
322 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
323 Utf8StrFmt(GuestSession::tr("Writing to %RU32 bytes to file failed: %Rrc"), cbRead, rc));
324 break;
325 }
326
327 Assert(cbToRead >= cbRead);
328 cbToRead -= cbRead;
329
330 /* Update total bytes written to the guest. */
331 cbWrittenTotal += cbRead;
332 Assert(cbWrittenTotal <= cbSize);
333
334 /* Did the user cancel the operation above? */
335 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
336 && fCanceled)
337 break;
338
339 rc = setProgress((ULONG)(cbWrittenTotal / ((uint64_t)cbSize / 100.0)));
340 if (RT_FAILURE(rc))
341 break;
342 }
343
344 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
345 && fCanceled)
346 return VINF_SUCCESS;
347
348 if (RT_FAILURE(rc))
349 return rc;
350
351 /*
352 * Even if we succeeded until here make sure to check whether we really transfered
353 * everything.
354 */
355 if ( cbSize > 0
356 && cbWrittenTotal == 0)
357 {
358 /* If nothing was transfered but the file size was > 0 then "vbox_cat" wasn't able to write
359 * to the destination -> access denied. */
360 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
361 Utf8StrFmt(GuestSession::tr("Writing guest file to host failed: Access denied")));
362 rc = VERR_ACCESS_DENIED;
363 }
364 else if (cbWrittenTotal < cbSize)
365 {
366 /* If we did not copy all let the user know. */
367 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
368 Utf8StrFmt(GuestSession::tr("Copying guest file to host to failed (%RU64/%RU64 bytes transfered)"),
369 cbWrittenTotal, cbSize));
370 rc = VERR_INTERRUPTED;
371 }
372
373 LogFlowFuncLeaveRC(rc);
374 return rc;
375}
376
377/**
378 * Copies a file from the guest to the host.
379 *
380 * @return VBox status code. VINF_NO_CHANGE if file was skipped.
381 * @param strSource Full path of source file on the guest to copy.
382 * @param strDest Full destination path and file name (host style) to copy file to.
383 * @param fFileCopyFlags File copy flags.
384 */
385int GuestSessionTask::fileCopyFromGuest(const Utf8Str &strSource, const Utf8Str &strDest, FileCopyFlag_T fFileCopyFlags)
386{
387 LogFlowThisFunc(("strSource=%s, strDest=%s, enmFileCopyFlags=0x%x\n", strSource.c_str(), strDest.c_str(), fFileCopyFlags));
388
389 GuestFileOpenInfo srcOpenInfo;
390 RT_ZERO(srcOpenInfo);
391 srcOpenInfo.mFileName = strSource;
392 srcOpenInfo.mOpenAction = FileOpenAction_OpenExisting;
393 srcOpenInfo.mAccessMode = FileAccessMode_ReadOnly;
394 srcOpenInfo.mSharingMode = FileSharingMode_All; /** @todo Use _Read when implemented. */
395
396 ComObjPtr<GuestFile> srcFile;
397
398 GuestFsObjData srcObjData;
399 int rcGuest = VERR_IPE_UNINITIALIZED_STATUS;
400 int rc = mSession->i_fsQueryInfo(strSource, TRUE /* fFollowSymlinks */, srcObjData, &rcGuest);
401 if (RT_FAILURE(rc))
402 {
403 switch (rc)
404 {
405 case VERR_GSTCTL_GUEST_ERROR:
406 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestFile::i_guestErrorToString(rcGuest));
407 break;
408
409 default:
410 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
411 Utf8StrFmt(GuestSession::tr("Source file lookup for \"%s\" failed: %Rrc"),
412 strSource.c_str(), rc));
413 break;
414 }
415 }
416 else
417 {
418 switch (srcObjData.mType)
419 {
420 case FsObjType_File:
421 break;
422
423 case FsObjType_Symlink:
424 if (!(fFileCopyFlags & FileCopyFlag_FollowLinks))
425 {
426 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
427 Utf8StrFmt(GuestSession::tr("Source file \"%s\" is a symbolic link"),
428 strSource.c_str(), rc));
429 rc = VERR_IS_A_SYMLINK;
430 }
431 break;
432
433 default:
434 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
435 Utf8StrFmt(GuestSession::tr("Source element \"%s\" is not a file"), strSource.c_str()));
436 rc = VERR_NOT_A_FILE;
437 break;
438 }
439 }
440
441 if (RT_FAILURE(rc))
442 return rc;
443
444 rc = mSession->i_fileOpen(srcOpenInfo, srcFile, &rcGuest);
445 if (RT_FAILURE(rc))
446 {
447 switch (rc)
448 {
449 case VERR_GSTCTL_GUEST_ERROR:
450 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestFile::i_guestErrorToString(rcGuest));
451 break;
452
453 default:
454 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
455 Utf8StrFmt(GuestSession::tr("Source file \"%s\" could not be opened: %Rrc"),
456 strSource.c_str(), rc));
457 break;
458 }
459 }
460
461 if (RT_FAILURE(rc))
462 return rc;
463
464 char *pszDstFile = NULL;
465 RTFSOBJINFO dstObjInfo;
466 RT_ZERO(dstObjInfo);
467
468 bool fSkip = false; /* Whether to skip handling the file. */
469
470 if (RT_SUCCESS(rc))
471 {
472 rc = RTPathQueryInfo(strDest.c_str(), &dstObjInfo, RTFSOBJATTRADD_NOTHING);
473 if (RT_SUCCESS(rc))
474 {
475 if (fFileCopyFlags & FileCopyFlag_NoReplace)
476 {
477 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
478 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" already exists"), strDest.c_str()));
479 rc = VERR_ALREADY_EXISTS;
480 }
481
482 if (fFileCopyFlags & FileCopyFlag_Update)
483 {
484 RTTIMESPEC srcModificationTimeTS;
485 RTTimeSpecSetSeconds(&srcModificationTimeTS, srcObjData.mModificationTime);
486 if (RTTimeSpecCompare(&srcModificationTimeTS, &dstObjInfo.ModificationTime) <= 0)
487 {
488 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
489 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" has same or newer modification date"),
490 strDest.c_str()));
491 fSkip = true;
492 }
493 }
494 }
495 else
496 {
497 if (rc != VERR_FILE_NOT_FOUND) /* Destination file does not exist (yet)? */
498 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
499 Utf8StrFmt(GuestSession::tr("Destination file lookup for \"%s\" failed: %Rrc"),
500 strDest.c_str(), rc));
501 }
502 }
503
504 if (fSkip)
505 {
506 int rc2 = srcFile->i_closeFile(&rcGuest);
507 AssertRC(rc2);
508 return VINF_SUCCESS;
509 }
510
511 if (RT_SUCCESS(rc))
512 {
513 if (RTFS_IS_FILE(dstObjInfo.Attr.fMode))
514 {
515 if (fFileCopyFlags & FileCopyFlag_NoReplace)
516 {
517 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
518 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" already exists"),
519 strDest.c_str(), rc));
520 rc = VERR_ALREADY_EXISTS;
521 }
522 else
523 pszDstFile = RTStrDup(strDest.c_str());
524 }
525 else if (RTFS_IS_DIRECTORY(dstObjInfo.Attr.fMode))
526 {
527 /* Build the final file name with destination path (on the host). */
528 char szDstPath[RTPATH_MAX];
529 RTStrPrintf2(szDstPath, sizeof(szDstPath), "%s", strDest.c_str());
530
531 if ( !strDest.endsWith("\\")
532 && !strDest.endsWith("/"))
533 RTPathAppend(szDstPath, sizeof(szDstPath), "/"); /* IPRT can handle / on all hosts. */
534
535 RTPathAppend(szDstPath, sizeof(szDstPath), RTPathFilenameEx(strSource.c_str(), mfPathStyle));
536
537 pszDstFile = RTStrDup(szDstPath);
538 }
539 else if (RTFS_IS_SYMLINK(dstObjInfo.Attr.fMode))
540 {
541 if (!(fFileCopyFlags & FileCopyFlag_FollowLinks))
542 {
543 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
544 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" is a symbolic link"),
545 strDest.c_str(), rc));
546 rc = VERR_IS_A_SYMLINK;
547 }
548 else
549 pszDstFile = RTStrDup(strDest.c_str());
550 }
551 else
552 {
553 LogFlowThisFunc(("Object type %RU32 not implemented yet\n", dstObjInfo.Attr.fMode));
554 rc = VERR_NOT_IMPLEMENTED;
555 }
556 }
557 else if (rc == VERR_FILE_NOT_FOUND)
558 pszDstFile = RTStrDup(strDest.c_str());
559
560 if ( RT_SUCCESS(rc)
561 || rc == VERR_FILE_NOT_FOUND)
562 {
563 if (!pszDstFile)
564 {
565 setProgressErrorMsg(VBOX_E_IPRT_ERROR, Utf8StrFmt(GuestSession::tr("No memory to allocate destination file path")));
566 rc = VERR_NO_MEMORY;
567 }
568 else
569 {
570 RTFILE hDstFile;
571 rc = RTFileOpen(&hDstFile, pszDstFile,
572 RTFILE_O_WRITE | RTFILE_O_OPEN_CREATE | RTFILE_O_DENY_WRITE); /** @todo Use the correct open modes! */
573 if (RT_SUCCESS(rc))
574 {
575 LogFlowThisFunc(("Copying '%s' to '%s' (%RI64 bytes) ...\n", strSource.c_str(), pszDstFile, srcObjData.mObjectSize));
576
577 rc = fileCopyFromGuestInner(srcFile, &hDstFile, fFileCopyFlags, 0 /* Offset, unused */, (uint64_t)srcObjData.mObjectSize);
578
579 int rc2 = RTFileClose(hDstFile);
580 AssertRC(rc2);
581 }
582 else
583 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
584 Utf8StrFmt(GuestSession::tr("Opening/creating destination file \"%s\" failed: %Rrc"),
585 pszDstFile, rc));
586 }
587 }
588
589 RTStrFree(pszDstFile);
590
591 int rc2 = srcFile->i_closeFile(&rcGuest);
592 AssertRC(rc2);
593
594 LogFlowFuncLeaveRC(rc);
595 return rc;
596}
597
598/**
599 * Main function for copying a file from host to the guest.
600 *
601 * @return VBox status code.
602 * @param phSrcFile Pointer to host file handle (source) to copy from. Must be in opened and ready state already.
603 * @param dstFile Guest file (destination) to copy to the guest. Must be in opened and ready state already.
604 * @param fFileCopyFlags File copy flags.
605 * @param offCopy Offset (in bytes) where to start copying the source file.
606 * @param cbSize Size (in bytes) to copy from the source file.
607 */
608int GuestSessionTask::fileCopyToGuestInner(PRTFILE phSrcFile, ComObjPtr<GuestFile> &dstFile, FileCopyFlag_T fFileCopyFlags,
609 uint64_t offCopy, uint64_t cbSize)
610{
611 RT_NOREF(fFileCopyFlags);
612
613 BOOL fCanceled = FALSE;
614 uint64_t cbWrittenTotal = 0;
615 uint64_t cbToRead = cbSize;
616
617 uint32_t uTimeoutMs = 30 * 1000; /* 30s timeout. */
618
619 int rc = VINF_SUCCESS;
620
621 if (offCopy)
622 {
623 uint64_t offActual;
624 rc = RTFileSeek(*phSrcFile, offCopy, RTFILE_SEEK_END, &offActual);
625 if (RT_FAILURE(rc))
626 {
627 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
628 Utf8StrFmt(GuestSession::tr("Seeking to offset %RU64 failed: %Rrc"), offCopy, rc));
629 return rc;
630 }
631 }
632
633 BYTE byBuf[_64K];
634 while (cbToRead)
635 {
636 size_t cbRead;
637 const uint32_t cbChunk = RT_MIN(cbToRead, sizeof(byBuf));
638 rc = RTFileRead(*phSrcFile, byBuf, cbChunk, &cbRead);
639 if (RT_FAILURE(rc))
640 {
641 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
642 Utf8StrFmt(GuestSession::tr("Reading %RU32 bytes @ %RU64, failed: %Rrc"), cbChunk, cbWrittenTotal, rc));
643 break;
644 }
645
646 rc = dstFile->i_writeData(uTimeoutMs, byBuf, (uint32_t)cbRead, NULL /* No partial writes */);
647 if (RT_FAILURE(rc))
648 {
649 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
650 Utf8StrFmt(GuestSession::tr("Writing to %zu bytes to file failed: %Rrc"), cbRead, rc));
651 break;
652 }
653
654 Assert(cbToRead >= cbRead);
655 cbToRead -= cbRead;
656
657 /* Update total bytes written to the guest. */
658 cbWrittenTotal += cbRead;
659 Assert(cbWrittenTotal <= cbSize);
660
661 /* Did the user cancel the operation above? */
662 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
663 && fCanceled)
664 break;
665
666 rc = setProgress((ULONG)(cbWrittenTotal / ((uint64_t)cbSize / 100.0)));
667 if (RT_FAILURE(rc))
668 break;
669 }
670
671 if (RT_FAILURE(rc))
672 return rc;
673
674 /*
675 * Even if we succeeded until here make sure to check whether we really transfered
676 * everything.
677 */
678 if ( cbSize > 0
679 && cbWrittenTotal == 0)
680 {
681 /* If nothing was transfered but the file size was > 0 then "vbox_cat" wasn't able to write
682 * to the destination -> access denied. */
683 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
684 Utf8StrFmt(GuestSession::tr("Writing to destination file failed: Access denied")));
685 rc = VERR_ACCESS_DENIED;
686 }
687 else if (cbWrittenTotal < cbSize)
688 {
689 /* If we did not copy all let the user know. */
690 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
691 Utf8StrFmt(GuestSession::tr("Copying to destination failed (%RU64/%RU64 bytes transfered)"),
692 cbWrittenTotal, cbSize));
693 rc = VERR_INTERRUPTED;
694 }
695
696 LogFlowFuncLeaveRC(rc);
697 return rc;
698}
699
700/**
701 * Copies a file from the guest to the host.
702 *
703 * @return VBox status code. VINF_NO_CHANGE if file was skipped.
704 * @param strSource Full path of source file on the host to copy.
705 * @param strDest Full destination path and file name (guest style) to copy file to.
706 * @param fFileCopyFlags File copy flags.
707 */
708int GuestSessionTask::fileCopyToGuest(const Utf8Str &strSource, const Utf8Str &strDest, FileCopyFlag_T fFileCopyFlags)
709{
710 LogFlowThisFunc(("strSource=%s, strDest=%s, fFileCopyFlags=0x%x\n", strSource.c_str(), strDest.c_str(), fFileCopyFlags));
711
712 Utf8Str strDestFinal = strDest;
713
714 GuestFsObjData dstObjData;
715 int rcGuest = VERR_IPE_UNINITIALIZED_STATUS;
716 int rc = mSession->i_fsQueryInfo(strDest, TRUE /* fFollowSymlinks */, dstObjData, &rcGuest);
717 if (RT_FAILURE(rc))
718 {
719 switch (rc)
720 {
721 case VERR_GSTCTL_GUEST_ERROR:
722 if (rcGuest == VERR_FILE_NOT_FOUND) /* File might not exist on the guest yet. */
723 {
724 rc = VINF_SUCCESS;
725 }
726 else
727 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestFile::i_guestErrorToString(rcGuest));
728 break;
729
730 default:
731 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
732 Utf8StrFmt(GuestSession::tr("Destination file lookup for \"%s\" failed: %Rrc"),
733 strDest.c_str(), rc));
734 break;
735 }
736 }
737 else
738 {
739 switch (dstObjData.mType)
740 {
741 case FsObjType_Directory:
742 {
743 /* Build the final file name with destination path (on the host). */
744 char szDstPath[RTPATH_MAX];
745 RTStrPrintf2(szDstPath, sizeof(szDstPath), "%s", strDest.c_str());
746
747 if ( !strDest.endsWith("\\")
748 && !strDest.endsWith("/"))
749 RTStrCat(szDstPath, sizeof(szDstPath), mPathStyle.c_str());
750
751 RTStrCat(szDstPath, sizeof(szDstPath), RTPathFilename(strSource.c_str()));
752
753 strDestFinal = szDstPath;
754 break;
755 }
756
757 case FsObjType_File:
758 if (fFileCopyFlags & FileCopyFlag_NoReplace)
759 {
760 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
761 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" already exists"),
762 strDest.c_str(), rc));
763 rc = VERR_ALREADY_EXISTS;
764 }
765 break;
766
767 case FsObjType_Symlink:
768 if (!(fFileCopyFlags & FileCopyFlag_FollowLinks))
769 {
770 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
771 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" is a symbolic link"),
772 strDest.c_str(), rc));
773 rc = VERR_IS_A_SYMLINK;
774 }
775 break;
776
777 default:
778 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
779 Utf8StrFmt(GuestSession::tr("Destination element \"%s\" not supported"), strDest.c_str()));
780 rc = VERR_NOT_SUPPORTED;
781 break;
782 }
783 }
784
785 if (RT_FAILURE(rc))
786 return rc;
787
788 GuestFileOpenInfo dstOpenInfo;
789 RT_ZERO(dstOpenInfo);
790 dstOpenInfo.mFileName = strDestFinal;
791 if (fFileCopyFlags & FileCopyFlag_NoReplace)
792 dstOpenInfo.mOpenAction = FileOpenAction_CreateNew;
793 else
794 dstOpenInfo.mOpenAction = FileOpenAction_CreateOrReplace;
795 dstOpenInfo.mAccessMode = FileAccessMode_WriteOnly;
796 dstOpenInfo.mSharingMode = FileSharingMode_All; /** @todo Use _Read when implemented. */
797
798 ComObjPtr<GuestFile> dstFile;
799 rc = mSession->i_fileOpen(dstOpenInfo, dstFile, &rcGuest);
800 if (RT_FAILURE(rc))
801 {
802 switch (rc)
803 {
804 case VERR_GSTCTL_GUEST_ERROR:
805 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestFile::i_guestErrorToString(rcGuest));
806 break;
807
808 default:
809 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
810 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" could not be opened: %Rrc"),
811 strDestFinal.c_str(), rc));
812 break;
813 }
814 }
815
816 if (RT_FAILURE(rc))
817 return rc;
818
819 char szSrcReal[RTPATH_MAX];
820
821 RTFSOBJINFO srcObjInfo;
822 RT_ZERO(srcObjInfo);
823
824 bool fSkip = false; /* Whether to skip handling the file. */
825
826 if (RT_SUCCESS(rc))
827 {
828 rc = RTPathReal(strSource.c_str(), szSrcReal, sizeof(szSrcReal));
829 if (RT_FAILURE(rc))
830 {
831 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
832 Utf8StrFmt(GuestSession::tr("Source path lookup for \"%s\" failed: %Rrc"),
833 strSource.c_str(), rc));
834 }
835 else
836 {
837 rc = RTPathQueryInfo(szSrcReal, &srcObjInfo, RTFSOBJATTRADD_NOTHING);
838 if (RT_SUCCESS(rc))
839 {
840 if (fFileCopyFlags & FileCopyFlag_Update)
841 {
842 RTTIMESPEC dstModificationTimeTS;
843 RTTimeSpecSetSeconds(&dstModificationTimeTS, dstObjData.mModificationTime);
844 if (RTTimeSpecCompare(&dstModificationTimeTS, &srcObjInfo.ModificationTime) <= 0)
845 {
846 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
847 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" has same or newer modification date"),
848 strDestFinal.c_str()));
849 fSkip = true;
850 }
851 }
852 }
853 else
854 {
855 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
856 Utf8StrFmt(GuestSession::tr("Source file lookup for \"%s\" failed: %Rrc"),
857 szSrcReal, rc));
858 }
859 }
860 }
861
862 if (fSkip)
863 {
864 int rc2 = dstFile->i_closeFile(&rcGuest);
865 AssertRC(rc2);
866 return VINF_SUCCESS;
867 }
868
869 if (RT_SUCCESS(rc))
870 {
871 RTFILE hSrcFile;
872 rc = RTFileOpen(&hSrcFile, szSrcReal,
873 RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_WRITE); /** @todo Use the correct open modes! */
874 if (RT_SUCCESS(rc))
875 {
876 LogFlowThisFunc(("Copying '%s' to '%s' (%RI64 bytes) ...\n",
877 szSrcReal, strDestFinal.c_str(), srcObjInfo.cbObject));
878
879 rc = fileCopyToGuestInner(&hSrcFile, dstFile, fFileCopyFlags, 0 /* Offset, unused */, srcObjInfo.cbObject);
880
881 int rc2 = RTFileClose(hSrcFile);
882 AssertRC(rc2);
883 }
884 else
885 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
886 Utf8StrFmt(GuestSession::tr("Opening source file \"%s\" failed: %Rrc"),
887 szSrcReal, rc));
888 }
889
890 int rc2 = dstFile->i_closeFile(&rcGuest);
891 AssertRC(rc2);
892
893 LogFlowFuncLeaveRC(rc);
894 return rc;
895}
896
897/**
898 * Adds a guest file system entry to a given list.
899 *
900 * @return VBox status code.
901 * @param strFile Path to file system entry to add.
902 * @param fsObjData Guest file system information of entry to add.
903 */
904int FsList::AddEntryFromGuest(const Utf8Str &strFile, const GuestFsObjData &fsObjData)
905{
906 LogFlowFunc(("Adding '%s'\n", strFile.c_str()));
907
908 FsEntry *pEntry = NULL;
909 try
910 {
911 pEntry = new FsEntry();
912 pEntry->fMode = fsObjData.GetFileMode();
913 pEntry->strPath = strFile;
914
915 mVecEntries.push_back(pEntry);
916 }
917 catch (...)
918 {
919 if (pEntry)
920 delete pEntry;
921 return VERR_NO_MEMORY;
922 }
923
924 return VINF_SUCCESS;
925}
926
927/**
928 * Adds a host file system entry to a given list.
929 *
930 * @return VBox status code.
931 * @param strFile Path to file system entry to add.
932 * @param pcObjInfo File system information of entry to add.
933 */
934int FsList::AddEntryFromHost(const Utf8Str &strFile, PCRTFSOBJINFO pcObjInfo)
935{
936 LogFlowFunc(("Adding '%s'\n", strFile.c_str()));
937
938 FsEntry *pEntry = NULL;
939 try
940 {
941 pEntry = new FsEntry();
942 pEntry->fMode = pcObjInfo->Attr.fMode & RTFS_TYPE_MASK;
943 pEntry->strPath = strFile;
944
945 mVecEntries.push_back(pEntry);
946 }
947 catch (...)
948 {
949 if (pEntry)
950 delete pEntry;
951 return VERR_NO_MEMORY;
952 }
953
954 return VINF_SUCCESS;
955}
956
957FsList::FsList(const GuestSessionTask &Task)
958 : mTask(Task)
959{
960}
961
962FsList::~FsList()
963{
964 Destroy();
965}
966
967/**
968 * Initializes a file list.
969 *
970 * @return VBox status code.
971 * @param strSrcRootAbs Source root path (absolute) for this file list.
972 * @param strDstRootAbs Destination root path (absolute) for this file list.
973 * @param SourceSpec Source specification to use.
974 */
975int FsList::Init(const Utf8Str &strSrcRootAbs, const Utf8Str &strDstRootAbs,
976 const GuestSessionFsSourceSpec &SourceSpec)
977{
978 mSrcRootAbs = strSrcRootAbs;
979 mDstRootAbs = strDstRootAbs;
980 mSourceSpec = SourceSpec;
981
982 /* If the source is a directory, make sure the path is properly terminated already. */
983 if (mSourceSpec.enmType == GuestSessionFsSourceType_Dir)
984 {
985 if ( !mSrcRootAbs.endsWith("/")
986 && !mSrcRootAbs.endsWith("\\"))
987 mSrcRootAbs += mSourceSpec.enmPathStyle == PathStyle_DOS ? "\\" : "/";
988
989 if ( !mDstRootAbs.endsWith("/")
990 && !mDstRootAbs.endsWith("\\"))
991 mDstRootAbs += "/";
992 }
993
994 return VINF_SUCCESS;
995}
996
997/**
998 * Destroys a file list.
999 */
1000void FsList::Destroy(void)
1001{
1002 LogFlowFuncEnter();
1003
1004 FsEntries::iterator itEntry = mVecEntries.begin();
1005 while (itEntry != mVecEntries.end())
1006 {
1007 FsEntry *pEntry = *itEntry;
1008 delete pEntry;
1009 mVecEntries.erase(itEntry);
1010 itEntry = mVecEntries.begin();
1011 }
1012
1013 Assert(mVecEntries.empty());
1014
1015 LogFlowFuncLeave();
1016}
1017
1018/**
1019 * Builds a guest file list from a given path (and optional filter).
1020 *
1021 * @return VBox status code.
1022 * @param strPath Directory on the guest to build list from.
1023 * @param strSubDir Current sub directory path; needed for recursion.
1024 * Set to an empty path.
1025 */
1026int FsList::AddDirFromGuest(const Utf8Str &strPath, const Utf8Str &strSubDir /* = "" */)
1027{
1028 Utf8Str strPathAbs = strPath;
1029 if ( !strPathAbs.endsWith("/")
1030 && !strPathAbs.endsWith("\\"))
1031 strPathAbs += mSourceSpec.enmPathStyle == PathStyle_DOS ? "\\" : "/";
1032
1033 Utf8Str strPathSub = strSubDir;
1034 if ( strPathSub.isNotEmpty()
1035 && !strPathSub.endsWith("/")
1036 && !strPathSub.endsWith("\\"))
1037 strPathSub += mSourceSpec.enmPathStyle == PathStyle_DOS ? "\\" : "/";
1038
1039 strPathAbs += strPathSub;
1040
1041 LogFlowFunc(("Entering '%s' (sub '%s')\n", strPathAbs.c_str(), strPathSub.c_str()));
1042
1043 GuestDirectoryOpenInfo dirOpenInfo;
1044 dirOpenInfo.mFilter = "";
1045 dirOpenInfo.mPath = strPathAbs;
1046 dirOpenInfo.mFlags = 0; /** @todo Handle flags? */
1047
1048 const ComObjPtr<GuestSession> &pSession = mTask.GetSession();
1049
1050 ComObjPtr <GuestDirectory> pDir; int rcGuest;
1051 int rc = pSession->i_directoryOpen(dirOpenInfo, pDir, &rcGuest);
1052 if (RT_FAILURE(rc))
1053 {
1054 switch (rc)
1055 {
1056 case VERR_INVALID_PARAMETER:
1057 break;
1058
1059 case VERR_GSTCTL_GUEST_ERROR:
1060 break;
1061
1062 default:
1063 break;
1064 }
1065
1066 return rc;
1067 }
1068
1069 if (strPathSub.isNotEmpty())
1070 {
1071 GuestFsObjData fsObjData;
1072 fsObjData.mType = FsObjType_Directory;
1073
1074 rc = AddEntryFromGuest(strPathSub, fsObjData);
1075 }
1076
1077 if (RT_SUCCESS(rc))
1078 {
1079 ComObjPtr<GuestFsObjInfo> fsObjInfo;
1080 while (RT_SUCCESS(rc = pDir->i_readInternal(fsObjInfo, &rcGuest)))
1081 {
1082 FsObjType_T enmObjType = FsObjType_Unknown; /* Shut up MSC. */
1083 HRESULT hr2 = fsObjInfo->COMGETTER(Type)(&enmObjType);
1084 AssertComRC(hr2);
1085
1086 com::Bstr bstrName;
1087 hr2 = fsObjInfo->COMGETTER(Name)(bstrName.asOutParam());
1088 AssertComRC(hr2);
1089
1090 Utf8Str strEntry = strPathSub + Utf8Str(bstrName);
1091
1092 LogFlowFunc(("Entry '%s'\n", strEntry.c_str()));
1093
1094 switch (enmObjType)
1095 {
1096 case FsObjType_Directory:
1097 {
1098 if ( bstrName.equals(".")
1099 || bstrName.equals(".."))
1100 {
1101 break;
1102 }
1103
1104 if (!(mSourceSpec.Type.Dir.fRecursive))
1105 break;
1106
1107 rc = AddDirFromGuest(strPath, strEntry);
1108 break;
1109 }
1110
1111 case FsObjType_Symlink:
1112 {
1113 if (mSourceSpec.fFollowSymlinks)
1114 {
1115 /** @todo Symlink handling from guest is not imlemented yet.
1116 * See IGuestSession::symlinkRead(). */
1117 }
1118 break;
1119 }
1120
1121 case FsObjType_File:
1122 {
1123 rc = AddEntryFromGuest(strEntry, fsObjInfo->i_getData());
1124 break;
1125 }
1126
1127 default:
1128 break;
1129 }
1130 }
1131 }
1132
1133 int rc2 = pDir->i_closeInternal(&rcGuest);
1134 if (RT_SUCCESS(rc))
1135 rc = rc2;
1136
1137 return rc;
1138}
1139
1140/**
1141 * Builds a host file list from a given path (and optional filter).
1142 *
1143 * @return VBox status code.
1144 * @param strPath Directory on the host to build list from.
1145 * @param strSubDir Current sub directory path; needed for recursion.
1146 * Set to an empty path.
1147 */
1148int FsList::AddDirFromHost(const Utf8Str &strPath, const Utf8Str &strSubDir)
1149{
1150 Utf8Str strPathAbs = strPath;
1151 if ( !strPathAbs.endsWith("/")
1152 && !strPathAbs.endsWith("\\"))
1153 strPathAbs += "/";
1154
1155 Utf8Str strPathSub = strSubDir;
1156 if ( strPathSub.isNotEmpty()
1157 && !strPathSub.endsWith("/")
1158 && !strPathSub.endsWith("\\"))
1159 strPathSub += "/";
1160
1161 strPathAbs += strPathSub;
1162
1163 LogFlowFunc(("Entering '%s' (sub '%s')\n", strPathAbs.c_str(), strPathSub.c_str()));
1164
1165 RTFSOBJINFO objInfo;
1166 int rc = RTPathQueryInfo(strPathAbs.c_str(), &objInfo, RTFSOBJATTRADD_NOTHING);
1167 if (RT_SUCCESS(rc))
1168 {
1169 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
1170 {
1171 if (strPathSub.isNotEmpty())
1172 rc = AddEntryFromHost(strPathSub, &objInfo);
1173
1174 if (RT_SUCCESS(rc))
1175 {
1176 RTDIR hDir;
1177 rc = RTDirOpen(&hDir, strPathAbs.c_str());
1178 if (RT_SUCCESS(rc))
1179 {
1180 do
1181 {
1182 /* Retrieve the next directory entry. */
1183 RTDIRENTRYEX Entry;
1184 rc = RTDirReadEx(hDir, &Entry, NULL, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
1185 if (RT_FAILURE(rc))
1186 {
1187 if (rc == VERR_NO_MORE_FILES)
1188 rc = VINF_SUCCESS;
1189 break;
1190 }
1191
1192 Utf8Str strEntry = strPathSub + Utf8Str(Entry.szName);
1193
1194 LogFlowFunc(("Entry '%s'\n", strEntry.c_str()));
1195
1196 switch (Entry.Info.Attr.fMode & RTFS_TYPE_MASK)
1197 {
1198 case RTFS_TYPE_DIRECTORY:
1199 {
1200 /* Skip "." and ".." entries. */
1201 if (RTDirEntryExIsStdDotLink(&Entry))
1202 break;
1203
1204 if (!(mSourceSpec.Type.Dir.fRecursive))
1205 break;
1206
1207 rc = AddDirFromHost(strPath, strEntry);
1208 break;
1209 }
1210
1211 case RTFS_TYPE_FILE:
1212 {
1213 rc = AddEntryFromHost(strEntry, &Entry.Info);
1214 break;
1215 }
1216
1217 case RTFS_TYPE_SYMLINK:
1218 {
1219 if (mSourceSpec.fFollowSymlinks)
1220 {
1221 Utf8Str strEntryAbs = strPathAbs + Utf8Str(Entry.szName);
1222
1223 char szPathReal[RTPATH_MAX];
1224 rc = RTPathReal(strEntryAbs.c_str(), szPathReal, sizeof(szPathReal));
1225 if (RT_SUCCESS(rc))
1226 {
1227 rc = RTPathQueryInfo(szPathReal, &objInfo, RTFSOBJATTRADD_NOTHING);
1228 if (RT_SUCCESS(rc))
1229 {
1230 LogFlowFunc(("Symlink '%s' -> '%s'\n", strEntryAbs.c_str(), szPathReal));
1231
1232 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
1233 {
1234 LogFlowFunc(("Symlink to directory\n"));
1235 rc = AddDirFromHost(strPath, strEntry);
1236 }
1237 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
1238 {
1239 LogFlowFunc(("Symlink to file\n"));
1240 rc = AddEntryFromHost(strEntry, &objInfo);
1241 }
1242 else
1243 rc = VERR_NOT_SUPPORTED;
1244 }
1245 else
1246 LogFlowFunc(("Unable to query symlink info for '%s', rc=%Rrc\n", szPathReal, rc));
1247 }
1248 else
1249 {
1250 LogFlowFunc(("Unable to resolve symlink for '%s', rc=%Rrc\n", strPathAbs.c_str(), rc));
1251 if (rc == VERR_FILE_NOT_FOUND) /* Broken symlink, skip. */
1252 rc = VINF_SUCCESS;
1253 }
1254 }
1255 break;
1256 }
1257
1258 default:
1259 break;
1260 }
1261
1262 } while (RT_SUCCESS(rc));
1263
1264 RTDirClose(hDir);
1265 }
1266 }
1267 }
1268 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
1269 {
1270 rc = VERR_IS_A_FILE;
1271 }
1272 else if (RTFS_IS_SYMLINK(objInfo.Attr.fMode))
1273 {
1274 rc = VERR_IS_A_SYMLINK;
1275 }
1276 else
1277 rc = VERR_NOT_SUPPORTED;
1278 }
1279 else
1280 LogFlowFunc(("Unable to query '%s', rc=%Rrc\n", strPathAbs.c_str(), rc));
1281
1282 LogFlowFuncLeaveRC(rc);
1283 return rc;
1284}
1285
1286GuestSessionTaskOpen::GuestSessionTaskOpen(GuestSession *pSession, uint32_t uFlags, uint32_t uTimeoutMS)
1287 : GuestSessionTask(pSession)
1288 , mFlags(uFlags)
1289 , mTimeoutMS(uTimeoutMS)
1290{
1291 m_strTaskName = "gctlSesOpen";
1292}
1293
1294GuestSessionTaskOpen::~GuestSessionTaskOpen(void)
1295{
1296
1297}
1298
1299int GuestSessionTaskOpen::Run(void)
1300{
1301 LogFlowThisFuncEnter();
1302
1303 AutoCaller autoCaller(mSession);
1304 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1305
1306 int vrc = mSession->i_startSession(NULL /*pvrcGuest*/);
1307 /* Nothing to do here anymore. */
1308
1309 LogFlowFuncLeaveRC(vrc);
1310 return vrc;
1311}
1312
1313GuestSessionCopyTask::GuestSessionCopyTask(GuestSession *pSession)
1314 : GuestSessionTask(pSession)
1315{
1316}
1317
1318GuestSessionCopyTask::~GuestSessionCopyTask()
1319{
1320 FsLists::iterator itList = mVecLists.begin();
1321 while (itList != mVecLists.end())
1322 {
1323 FsList *pFsList = (*itList);
1324 pFsList->Destroy();
1325 delete pFsList;
1326 mVecLists.erase(itList);
1327 itList = mVecLists.begin();
1328 }
1329
1330 Assert(mVecLists.empty());
1331}
1332
1333GuestSessionTaskCopyFrom::GuestSessionTaskCopyFrom(GuestSession *pSession, GuestSessionFsSourceSet vecSrc, const Utf8Str &strDest)
1334 : GuestSessionCopyTask(pSession)
1335{
1336 m_strTaskName = "gctlCpyFrm";
1337
1338 mSources = vecSrc;
1339 mDest = strDest;
1340}
1341
1342GuestSessionTaskCopyFrom::~GuestSessionTaskCopyFrom(void)
1343{
1344}
1345
1346#if 0
1347/**
1348 * Copys a directory (tree) from guest to the host.
1349 *
1350 * @return IPRT status code.
1351 * @param strSource Source directory on the guest to copy to the host.
1352 * @param strFilter DOS-style wildcard filter (?, *). Optional.
1353 * @param strDest Destination directory on the host.
1354 * @param fRecursive Whether to recursively copy the directory contents or not.
1355 * @param fFollowSymlinks Whether to follow symlinks or not.
1356 * @param strSubDir Current sub directory to handle. Needs to NULL and only
1357 * is needed for recursion.
1358 */
1359int SessionTaskCopyDirFrom::directoryCopyToHost(const Utf8Str &strSource, const Utf8Str &strFilter,
1360 const Utf8Str &strDest, bool fRecursive, bool fFollowSymlinks,
1361 const Utf8Str &strSubDir /* For recursion. */)
1362{
1363 Utf8Str strSrcDir = strSource;
1364 Utf8Str strDstDir = strDest;
1365 Utf8Str strSrcSubDir = strSubDir;
1366
1367 /* Validation and sanity. */
1368 if ( !strSrcDir.endsWith("/")
1369 && !strSrcDir.endsWith("\\"))
1370 strSrcDir += "/";
1371
1372 if ( !strDstDir.endsWith("/")
1373 && !strDstDir.endsWith("\\"))
1374 strDstDir+= "/";
1375
1376 if ( strSrcSubDir.isNotEmpty() /* Optional, for recursion. */
1377 && !strSrcSubDir.endsWith("/")
1378 && !strSrcSubDir.endsWith("\\"))
1379 strSrcSubDir += "/";
1380
1381 Utf8Str strSrcCur = strSrcDir + strSrcSubDir;
1382
1383 LogFlowFunc(("strSrcDir=%s, strDstDir=%s, strFilter=%s, strSubDir=%s\n",
1384 strSrcDir.c_str(), strDstDir.c_str(), strFilter.c_str(), strSubDir.c_str()));
1385
1386 int rc;
1387
1388 if (strSrcSubDir.isNotEmpty())
1389 {
1390 const uint32_t fMode = 0700; /** @todo */
1391
1392 rc = RTDirCreate(Utf8Str(strDstDir + strSrcSubDir).c_str(), fMode, 0);
1393 if (RT_FAILURE(rc))
1394 return rc;
1395 }
1396
1397 GuestDirectoryOpenInfo dirOpenInfo;
1398 dirOpenInfo.mFilter = strFilter;
1399 dirOpenInfo.mPath = strSrcCur;
1400 dirOpenInfo.mFlags = 0; /** @todo Handle flags? */
1401
1402 ComObjPtr <GuestDirectory> pDir; int rcGuest;
1403 rc = mSession->i_directoryOpen(dirOpenInfo, pDir, &rcGuest);
1404
1405 if (RT_FAILURE(rc))
1406 {
1407 switch (rc)
1408 {
1409 case VERR_INVALID_PARAMETER:
1410 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1411 Utf8StrFmt(GuestSession::tr("Opening directory \"%s\" failed: Invalid parameter"), dirOpenInfo.mPath.c_str()));
1412 break;
1413
1414 case VERR_GSTCTL_GUEST_ERROR:
1415 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestProcess::i_guestErrorToString(rcGuest));
1416 break;
1417
1418 default:
1419 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1420 Utf8StrFmt(GuestSession::tr("Opening directory \"%s\" failed: %Rrc"), dirOpenInfo.mPath.c_str(), rc));
1421 break;
1422 }
1423
1424 return rc;
1425 }
1426
1427 ComObjPtr<GuestFsObjInfo> fsObjInfo;
1428 while (RT_SUCCESS(rc = pDir->i_readInternal(fsObjInfo, &rcGuest)))
1429 {
1430 FsObjType_T enmObjType = FsObjType_Unknown; /* Shut up MSC. */
1431 HRESULT hr2 = fsObjInfo->COMGETTER(Type)(&enmObjType);
1432 AssertComRC(hr2);
1433
1434 com::Bstr bstrName;
1435 hr2 = fsObjInfo->COMGETTER(Name)(bstrName.asOutParam());
1436 AssertComRC(hr2);
1437
1438 Utf8Str strName(bstrName);
1439
1440 switch (enmObjType)
1441 {
1442 case FsObjType_Directory:
1443 {
1444 bool fSkip = false;
1445
1446 if ( strName.equals(".")
1447 || strName.equals(".."))
1448 {
1449 fSkip = true;
1450 }
1451
1452 if ( strFilter.isNotEmpty()
1453 && !RTStrSimplePatternMatch(strFilter.c_str(), strName.c_str()))
1454 fSkip = true;
1455
1456 if ( fRecursive
1457 && !fSkip)
1458 rc = directoryCopyToHost(strSrcDir, strFilter, strDstDir, fRecursive, fFollowSymlinks,
1459 strSrcSubDir + strName);
1460 break;
1461 }
1462
1463 case FsObjType_Symlink:
1464 {
1465 if (fFollowSymlinks)
1466 { /* Fall through to next case is intentional. */ }
1467 else
1468 break;
1469 RT_FALL_THRU();
1470 }
1471
1472 case FsObjType_File:
1473 {
1474 if ( strFilter.isNotEmpty()
1475 && !RTStrSimplePatternMatch(strFilter.c_str(), strName.c_str()))
1476 {
1477 break; /* Filter does not match. */
1478 }
1479
1480 if (RT_SUCCESS(rc))
1481 {
1482 Utf8Str strSrcFile = strSrcDir + strSrcSubDir + strName;
1483 Utf8Str strDstFile = strDstDir + strSrcSubDir + strName;
1484 rc = fileCopyFromGuest(strSrcFile, strDstFile, FileCopyFlag_None);
1485 }
1486 break;
1487 }
1488
1489 default:
1490 break;
1491 }
1492
1493 BOOL fCanceled = FALSE;
1494 if ( SUCCEEDED(mProgress->COMGETTER(Canceled(&fCanceled)))
1495 && fCanceled)
1496 break;
1497 }
1498
1499 if (rc == VERR_NO_MORE_FILES) /* Reading done? */
1500 rc = VINF_SUCCESS;
1501
1502 pDir->i_closeInternal(&rcGuest);
1503
1504 LogFlowFunc(("Leaving '%s', rc=%Rrc\n", strSrcCur.c_str(), rc));
1505 return rc;
1506}
1507#endif
1508
1509HRESULT GuestSessionTaskCopyFrom::Init(const Utf8Str &strTaskDesc)
1510{
1511 setTaskDesc(strTaskDesc);
1512
1513 /* Create the progress object. */
1514 ComObjPtr<Progress> pProgress;
1515 HRESULT hr = pProgress.createObject();
1516 if (FAILED(hr))
1517 return hr;
1518
1519 mProgress = pProgress;
1520
1521 int rc = VINF_SUCCESS;
1522
1523 ULONG cOperations = 0;
1524
1525 GuestSessionFsSourceSet::iterator itSrc = mSources.begin();
1526 while (itSrc != mSources.end())
1527 {
1528 Utf8Str strSrc = itSrc->strSource;
1529 Utf8Str strDst = mDest;
1530
1531 if (itSrc->enmType == GuestSessionFsSourceType_Dir)
1532 {
1533 /* If the source does not end with a slash, copy over the entire directory
1534 * (and not just its contents). */
1535 if ( !strSrc.endsWith("/")
1536 && !strSrc.endsWith("\\"))
1537 {
1538 if ( !strDst.endsWith("/")
1539 && !strDst.endsWith("\\"))
1540 strDst += mPathStyle;
1541
1542 strDst += Utf8StrFmt("%s", RTPathFilenameEx(strSrc.c_str(), mfPathStyle));
1543 }
1544 }
1545
1546 GuestFsObjData srcObjData; int rcGuest;
1547 rc = mSession->i_fsQueryInfo(strSrc, itSrc->fFollowSymlinks ? TRUE : FALSE,
1548 srcObjData, &rcGuest);
1549 if (RT_FAILURE(rc))
1550 {
1551 break;
1552 }
1553
1554 if (srcObjData.mType == FsObjType_Directory)
1555 {
1556 if (itSrc->enmType != GuestSessionFsSourceType_Dir)
1557 {
1558 rc = VERR_NOT_A_FILE;
1559 break;
1560 }
1561 }
1562 else
1563 {
1564 if (itSrc->enmType != GuestSessionFsSourceType_File)
1565 {
1566 rc = VERR_NOT_A_DIRECTORY;
1567 break;
1568 }
1569 }
1570
1571 LogFlowFunc(("strSrc=%s, strDst=%s\n", strSrc.c_str(), strDst.c_str()));
1572
1573#if 0
1574 if (itSrc->enmType == GuestSessionFsSourceType_Dir)
1575 {
1576 bool fDstExists = RTDirExists(strDstDir.c_str());
1577 if (fDstExists)
1578 {
1579 if (!(itSrc->Type.Dir.fCopyFlags & DirectoryCopyFlag_CopyIntoExisting))
1580 {
1581 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1582 Utf8StrFmt(GuestSession::tr("Destination directory \"%s\" exists when it must not"), mDest.c_str()));
1583 rc = VERR_ALREADY_EXISTS;
1584 break;
1585 }
1586 }
1587
1588 if ( !strSrcDir.endsWith("/")
1589 && !strSrcDir.endsWith("\\"))
1590 {
1591 if ( !strDstDir.endsWith("/")
1592 && !strDstDir.endsWith("\\"))
1593 strDstDir += "/"; /* IPRT can handle / on all hosts. */
1594
1595 strDstDir += Utf8StrFmt("%s", RTPathFilenameEx(strSrcDir.c_str(), mfPathStyle));
1596 }
1597
1598 /* Create the final target directory on the host.
1599 * The target directory might already exist on the host (which can be controlled by the copy flags). */
1600 fDstExists = RTDirExists(strDstDir.c_str());
1601 if (fDstExists)
1602 {
1603 if (!(itSrc->Type.Dir.fCopyFlags & DirectoryCopyFlag_CopyIntoExisting))
1604 {
1605 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1606 Utf8StrFmt(GuestSession::tr("Destination directory \"%s\" exists when it must not"), strDstDir.c_str()));
1607 rc = VERR_ALREADY_EXISTS;
1608 break;
1609 }
1610 }
1611 else
1612 {
1613 rc = RTDirCreate(strDstDir.c_str(), fDirMode, 0);
1614 if (RT_FAILURE(rc))
1615 {
1616 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
1617 Utf8StrFmt(GuestSession::tr("Error creating destination directory \"%s\", rc=%Rrc"),
1618 strDstDir.c_str(), rc));
1619 break;
1620 }
1621 }
1622 }
1623 else if (itSrc->enmType == GuestSessionFsSourceType_File)
1624 {
1625
1626 }
1627 else
1628 AssertFailed();
1629#endif
1630
1631 FsList *pFsList = NULL;
1632 try
1633 {
1634 pFsList = new FsList(*this);
1635 rc = pFsList->Init(strSrc, strDst, *itSrc);
1636 if (RT_SUCCESS(rc))
1637 {
1638 if (itSrc->enmType == GuestSessionFsSourceType_Dir)
1639 {
1640 rc = pFsList->AddDirFromGuest(strSrc);
1641 }
1642 else
1643 rc = pFsList->AddEntryFromGuest(RTPathFilename(strSrc.c_str()), srcObjData);
1644 }
1645
1646 if (RT_FAILURE(rc))
1647 {
1648 delete pFsList;
1649 break;
1650 }
1651
1652 mVecLists.push_back(pFsList);
1653 }
1654 catch (...)
1655 {
1656 rc = VERR_NO_MEMORY;
1657 break;
1658 }
1659
1660 AssertPtr(pFsList);
1661 cOperations += (ULONG)pFsList->mVecEntries.size();
1662
1663 itSrc++;
1664 }
1665
1666 if (RT_SUCCESS(rc))
1667 {
1668 Utf8Str strFirstOp;
1669 if (cOperations) /* Use the first element as description (if available). */
1670 {
1671 Assert(mVecLists.size());
1672 Assert(mVecLists[0]->mVecEntries.size());
1673
1674 strFirstOp = mDest + mVecLists[0]->mVecEntries[0]->strPath;
1675 }
1676
1677 hr = pProgress->init(static_cast<IGuestSession*>(mSession), Bstr(mDesc).raw(),
1678 TRUE /* aCancelable */, cOperations + 1 /* Number of operations */,
1679 Bstr(strFirstOp).raw());
1680 }
1681
1682 return rc;
1683}
1684
1685int GuestSessionTaskCopyFrom::Run(void)
1686{
1687 LogFlowThisFuncEnter();
1688
1689 AutoCaller autoCaller(mSession);
1690 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1691
1692 int rc = VINF_SUCCESS;
1693
1694 FsLists::const_iterator itList = mVecLists.begin();
1695 while (itList != mVecLists.end())
1696 {
1697 FsList *pList = *itList;
1698 AssertPtr(pList);
1699
1700 const bool fCopyIntoExisting = pList->mSourceSpec.Type.Dir.fCopyFlags & DirectoryCopyFlag_CopyIntoExisting;
1701 const uint32_t fDirMode = 0700; /** @todo Play safe by default; implement ACLs. */
1702
1703 LogFlowFunc(("List: srcRootAbs=%s, dstRootAbs=%s\n", pList->mSrcRootAbs.c_str(), pList->mDstRootAbs.c_str()));
1704
1705 /* Create the root directory. */
1706 if (pList->mSourceSpec.enmType == GuestSessionFsSourceType_Dir)
1707 {
1708 rc = RTDirCreate(pList->mDstRootAbs.c_str(), fDirMode, 0 /* fCreate */);
1709 if ( rc == VWRN_ALREADY_EXISTS
1710 && !fCopyIntoExisting)
1711 {
1712 break;
1713 }
1714 }
1715
1716 FsEntries::const_iterator itEntry = pList->mVecEntries.begin();
1717 while (itEntry != pList->mVecEntries.end())
1718 {
1719 FsEntry *pEntry = *itEntry;
1720 AssertPtr(pEntry);
1721
1722 Utf8Str strSrcAbs = pList->mSrcRootAbs;
1723 Utf8Str strDstAbs = pList->mDstRootAbs;
1724 if (pList->mSourceSpec.enmType == GuestSessionFsSourceType_Dir)
1725 {
1726 strSrcAbs += pEntry->strPath;
1727 strDstAbs += pEntry->strPath;
1728 }
1729
1730 LogFlowFunc(("Entry '%s': %s -> %s\n", pEntry->strPath.c_str(), strSrcAbs.c_str(), strDstAbs.c_str()));
1731
1732 switch (pEntry->fMode & RTFS_TYPE_MASK)
1733 {
1734 case RTFS_TYPE_DIRECTORY:
1735 if (!pList->mSourceSpec.fDryRun)
1736 {
1737 rc = RTDirCreate(strDstAbs.c_str(), fDirMode, 0 /* fCreate */);
1738 if (rc == VERR_ALREADY_EXISTS)
1739 {
1740 if (!fCopyIntoExisting)
1741 {
1742 break;
1743 }
1744
1745 rc = VINF_SUCCESS;
1746 }
1747
1748 if (RT_FAILURE(rc))
1749 break;
1750 }
1751 break;
1752
1753 case RTFS_TYPE_FILE:
1754 if (!pList->mSourceSpec.fDryRun)
1755 rc = fileCopyFromGuest(strSrcAbs, strDstAbs, FileCopyFlag_None);
1756 break;
1757
1758 default:
1759 LogFlowFunc(("Warning: Type %d for '%s' is not supported\n",
1760 pEntry->fMode & RTFS_TYPE_MASK, strSrcAbs.c_str()));
1761 break;
1762 }
1763
1764 if (RT_FAILURE(rc))
1765 break;
1766
1767 mProgress->SetNextOperation(Bstr(strSrcAbs).raw(), 1);
1768
1769 ++itEntry;
1770 }
1771
1772 if (RT_FAILURE(rc))
1773 break;
1774
1775 ++itList;
1776 }
1777
1778 if (RT_SUCCESS(rc))
1779 rc = setProgressSuccess();
1780
1781 LogFlowFuncLeaveRC(rc);
1782 return rc;
1783}
1784
1785GuestSessionTaskCopyTo::GuestSessionTaskCopyTo(GuestSession *pSession, GuestSessionFsSourceSet vecSrc, const Utf8Str &strDest)
1786 : GuestSessionCopyTask(pSession)
1787{
1788 m_strTaskName = "gctlCpyTo";
1789
1790 mSources = vecSrc;
1791 mDest = strDest;
1792}
1793
1794GuestSessionTaskCopyTo::~GuestSessionTaskCopyTo(void)
1795{
1796}
1797
1798HRESULT GuestSessionTaskCopyTo::Init(const Utf8Str &strTaskDesc)
1799{
1800 LogFlowFuncEnter();
1801
1802 setTaskDesc(strTaskDesc);
1803
1804 /* Create the progress object. */
1805 ComObjPtr<Progress> pProgress;
1806 HRESULT hr = pProgress.createObject();
1807 if (FAILED(hr))
1808 return hr;
1809
1810 mProgress = pProgress;
1811
1812 int rc = VINF_SUCCESS;
1813
1814 ULONG cOperations = 0;
1815
1816 GuestSessionFsSourceSet::iterator itSrc = mSources.begin();
1817 while (itSrc != mSources.end())
1818 {
1819 Utf8Str strSrc = itSrc->strSource;
1820 Utf8Str strDst = mDest;
1821
1822 if (itSrc->enmType == GuestSessionFsSourceType_Dir)
1823 {
1824 /* If the source does not end with a slash, copy over the entire directory
1825 * (and not just its contents). */
1826 if ( !strSrc.endsWith("/")
1827 && !strSrc.endsWith("\\"))
1828 {
1829 if ( !strDst.endsWith("/")
1830 && !strDst.endsWith("\\"))
1831 strDst += mPathStyle;
1832
1833 strDst += Utf8StrFmt("%s", RTPathFilenameEx(strSrc.c_str(), mfPathStyle));
1834 }
1835 }
1836
1837 RTFSOBJINFO srcFsObjInfo;
1838 rc = RTPathQueryInfo(strSrc.c_str(), &srcFsObjInfo, RTFSOBJATTRADD_NOTHING);
1839 if (RT_FAILURE(rc))
1840 break;
1841
1842 if (RTFS_IS_DIRECTORY(srcFsObjInfo.Attr.fMode))
1843 {
1844 if (itSrc->enmType != GuestSessionFsSourceType_Dir)
1845 {
1846 rc = VERR_NOT_A_FILE;
1847 break;
1848 }
1849 }
1850 else
1851 {
1852 if (itSrc->enmType == GuestSessionFsSourceType_Dir)
1853 {
1854 rc = VERR_NOT_A_DIRECTORY;
1855 break;
1856 }
1857 }
1858
1859 LogFlowFunc(("strSrc=%s, strDst=%s\n", strSrc.c_str(), strDst.c_str()));
1860
1861 FsList *pFsList = NULL;
1862 try
1863 {
1864 pFsList = new FsList(*this);
1865 rc = pFsList->Init(strSrc, strDst, *itSrc);
1866 if (RT_SUCCESS(rc))
1867 {
1868 if (itSrc->enmType == GuestSessionFsSourceType_Dir)
1869 {
1870 rc = pFsList->AddDirFromHost(strSrc);
1871 }
1872 else
1873 rc = pFsList->AddEntryFromHost(RTPathFilename(strSrc.c_str()), &srcFsObjInfo);
1874 }
1875
1876 if (RT_FAILURE(rc))
1877 break;
1878
1879 mVecLists.push_back(pFsList);
1880 }
1881 catch (...)
1882 {
1883 rc = VERR_NO_MEMORY;
1884 break;
1885 }
1886
1887 AssertPtr(pFsList);
1888 cOperations += (ULONG)pFsList->mVecEntries.size();
1889
1890 itSrc++;
1891 }
1892
1893 if (RT_SUCCESS(rc))
1894 {
1895 Utf8Str strFirstOp;
1896 if (cOperations) /* Use the first element as description (if available). */
1897 {
1898 Assert(mVecLists.size());
1899 Assert(mVecLists[0]->mVecEntries.size());
1900
1901 strFirstOp = mDest + mVecLists[0]->mVecEntries[0]->strPath;
1902 }
1903
1904 hr = pProgress->init(static_cast<IGuestSession*>(mSession), Bstr(mDesc).raw(),
1905 TRUE /* aCancelable */, cOperations + 1 /* Number of operations */,
1906 Bstr(strFirstOp).raw());
1907 }
1908
1909 LogFlowFunc(("rc=%Rrc\n", rc));
1910 return hr;
1911}
1912
1913int GuestSessionTaskCopyTo::Run(void)
1914{
1915 LogFlowThisFuncEnter();
1916
1917 AutoCaller autoCaller(mSession);
1918 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1919
1920 int rc = VINF_SUCCESS;
1921
1922 FsLists::const_iterator itList = mVecLists.begin();
1923 while (itList != mVecLists.end())
1924 {
1925 FsList *pList = *itList;
1926 AssertPtr(pList);
1927
1928 const bool fFollowSymlinks = pList->mSourceSpec.fFollowSymlinks;
1929 const bool fCopyIntoExisting = pList->mSourceSpec.Type.Dir.fCopyFlags & DirectoryCopyFlag_CopyIntoExisting;
1930 const uint32_t fDirMode = 0700; /** @todo Play safe by default; implement ACLs. */
1931
1932 LogFlowFunc(("List: srcRootAbs=%s, dstRootAbs=%s\n", pList->mSrcRootAbs.c_str(), pList->mDstRootAbs.c_str()));
1933
1934 /* Create the root directory. */
1935 if (pList->mSourceSpec.enmType == GuestSessionFsSourceType_Dir)
1936 {
1937 rc = directoryCreate(pList->mDstRootAbs.c_str(), DirectoryCreateFlag_None, fDirMode, fFollowSymlinks);
1938 if ( rc == VWRN_ALREADY_EXISTS
1939 && !fCopyIntoExisting)
1940 {
1941 break;
1942 }
1943 }
1944
1945 FsEntries::const_iterator itEntry = pList->mVecEntries.begin();
1946 while (itEntry != pList->mVecEntries.end())
1947 {
1948 FsEntry *pEntry = *itEntry;
1949 AssertPtr(pEntry);
1950
1951 Utf8Str strSrcAbs = pList->mSrcRootAbs;
1952 Utf8Str strDstAbs = pList->mDstRootAbs;
1953 if (pList->mSourceSpec.enmType == GuestSessionFsSourceType_Dir)
1954 {
1955 strSrcAbs += pEntry->strPath;
1956 strDstAbs += pEntry->strPath;
1957 }
1958
1959 LogFlowFunc(("Entry '%s': %s -> %s\n", pEntry->strPath.c_str(), strSrcAbs.c_str(), strDstAbs.c_str()));
1960
1961 switch (pEntry->fMode & RTFS_TYPE_MASK)
1962 {
1963 case RTFS_TYPE_DIRECTORY:
1964 if (!pList->mSourceSpec.fDryRun)
1965 {
1966 rc = directoryCreate(strDstAbs.c_str(), DirectoryCreateFlag_None, fDirMode, fFollowSymlinks);
1967 if ( rc == VWRN_ALREADY_EXISTS
1968 && !fCopyIntoExisting)
1969 {
1970 break;
1971 }
1972 }
1973 break;
1974
1975 case RTFS_TYPE_FILE:
1976 if (!pList->mSourceSpec.fDryRun)
1977 rc = fileCopyToGuest(strSrcAbs, strDstAbs, FileCopyFlag_None);
1978 break;
1979
1980 default:
1981 LogFlowFunc(("Warning: Type %d for '%s' is not supported\n",
1982 pEntry->fMode & RTFS_TYPE_MASK, strSrcAbs.c_str()));
1983 break;
1984 }
1985
1986 if (RT_FAILURE(rc))
1987 break;
1988
1989 mProgress->SetNextOperation(Bstr(strSrcAbs).raw(), 1);
1990
1991 ++itEntry;
1992 }
1993
1994 if (RT_FAILURE(rc))
1995 break;
1996
1997 ++itList;
1998 }
1999
2000 if (RT_SUCCESS(rc))
2001 rc = setProgressSuccess();
2002
2003 LogFlowFuncLeaveRC(rc);
2004 return rc;
2005}
2006
2007#if 0
2008SessionTaskCopyFileFrom::SessionTaskCopyFileFrom(GuestSession *pSession,
2009 const Utf8Str &strSource, const Utf8Str &strDest, FileCopyFlag_T fFileCopyFlags)
2010 : GuestSessionCopyTask(pSession)
2011{
2012 m_strTaskName = "gctlCpyFileFrm";
2013
2014 mSource = strSource;
2015 mDest = strDest;
2016
2017 Type.File.fCopyFlags = fFileCopyFlags;
2018}
2019
2020SessionTaskCopyFileFrom::~SessionTaskCopyFileFrom(void)
2021{
2022
2023}
2024
2025int SessionTaskCopyFileFrom::Run(void)
2026{
2027 LogFlowThisFuncEnter();
2028
2029 AutoCaller autoCaller(mSession);
2030 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2031
2032 int rc = fileCopyFromGuest(mSource, mDest, Type.File.fCopyFlags);
2033 if (RT_SUCCESS(rc))
2034 rc = setProgressSuccess();
2035
2036 LogFlowFuncLeaveRC(rc);
2037 return rc;
2038}
2039#endif
2040
2041GuestSessionTaskUpdateAdditions::GuestSessionTaskUpdateAdditions(GuestSession *pSession,
2042 const Utf8Str &strSource,
2043 const ProcessArguments &aArguments,
2044 uint32_t fFlags)
2045 : GuestSessionTask(pSession)
2046{
2047 m_strTaskName = "gctlUpGA";
2048
2049 mSource = strSource;
2050 mArguments = aArguments;
2051 mFlags = fFlags;
2052}
2053
2054GuestSessionTaskUpdateAdditions::~GuestSessionTaskUpdateAdditions(void)
2055{
2056
2057}
2058
2059int GuestSessionTaskUpdateAdditions::addProcessArguments(ProcessArguments &aArgumentsDest, const ProcessArguments &aArgumentsSource)
2060{
2061 int rc = VINF_SUCCESS;
2062
2063 try
2064 {
2065 /* Filter out arguments which already are in the destination to
2066 * not end up having them specified twice. Not the fastest method on the
2067 * planet but does the job. */
2068 ProcessArguments::const_iterator itSource = aArgumentsSource.begin();
2069 while (itSource != aArgumentsSource.end())
2070 {
2071 bool fFound = false;
2072 ProcessArguments::iterator itDest = aArgumentsDest.begin();
2073 while (itDest != aArgumentsDest.end())
2074 {
2075 if ((*itDest).equalsIgnoreCase((*itSource)))
2076 {
2077 fFound = true;
2078 break;
2079 }
2080 ++itDest;
2081 }
2082
2083 if (!fFound)
2084 aArgumentsDest.push_back((*itSource));
2085
2086 ++itSource;
2087 }
2088 }
2089 catch(std::bad_alloc &)
2090 {
2091 return VERR_NO_MEMORY;
2092 }
2093
2094 return rc;
2095}
2096
2097int GuestSessionTaskUpdateAdditions::copyFileToGuest(GuestSession *pSession, PRTISOFSFILE pISO,
2098 Utf8Str const &strFileSource, const Utf8Str &strFileDest,
2099 bool fOptional)
2100{
2101 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
2102 AssertPtrReturn(pISO, VERR_INVALID_POINTER);
2103
2104 uint32_t cbSrcOffset;
2105 size_t cbSrcSize;
2106
2107 int rc = RTIsoFsGetFileInfo(pISO, strFileSource.c_str(), &cbSrcOffset, &cbSrcSize);
2108 if (RT_FAILURE(rc))
2109 {
2110 if (fOptional)
2111 return VINF_SUCCESS;
2112
2113 return rc;
2114 }
2115
2116 Assert(cbSrcOffset);
2117 Assert(cbSrcSize);
2118 rc = RTFileSeek(pISO->file, cbSrcOffset, RTFILE_SEEK_BEGIN, NULL);
2119 if (RT_SUCCESS(rc))
2120 {
2121 LogRel(("Copying Guest Additions installer file \"%s\" to \"%s\" on guest ...\n",
2122 strFileSource.c_str(), strFileDest.c_str()));
2123
2124 GuestFileOpenInfo dstOpenInfo;
2125 RT_ZERO(dstOpenInfo);
2126 dstOpenInfo.mFileName = strFileDest;
2127 dstOpenInfo.mOpenAction = FileOpenAction_CreateOrReplace;
2128 dstOpenInfo.mAccessMode = FileAccessMode_WriteOnly;
2129 dstOpenInfo.mSharingMode = FileSharingMode_All; /** @todo Use _Read when implemented. */
2130
2131 ComObjPtr<GuestFile> dstFile; int rcGuest;
2132 rc = mSession->i_fileOpen(dstOpenInfo, dstFile, &rcGuest);
2133 if (RT_FAILURE(rc))
2134 {
2135 switch (rc)
2136 {
2137 case VERR_GSTCTL_GUEST_ERROR:
2138 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestFile::i_guestErrorToString(rcGuest));
2139 break;
2140
2141 default:
2142 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2143 Utf8StrFmt(GuestSession::tr("Destination file \"%s\" could not be opened: %Rrc"),
2144 strFileDest.c_str(), rc));
2145 break;
2146 }
2147 }
2148 else
2149 {
2150 rc = fileCopyToGuestInner(&pISO->file, dstFile, FileCopyFlag_None, cbSrcOffset, cbSrcSize);
2151
2152 int rc2 = dstFile->i_closeFile(&rcGuest);
2153 AssertRC(rc2);
2154 }
2155
2156 if (RT_FAILURE(rc))
2157 return rc;
2158 }
2159
2160 return rc;
2161}
2162
2163int GuestSessionTaskUpdateAdditions::runFileOnGuest(GuestSession *pSession, GuestProcessStartupInfo &procInfo)
2164{
2165 AssertPtrReturn(pSession, VERR_INVALID_POINTER);
2166
2167 LogRel(("Running %s ...\n", procInfo.mName.c_str()));
2168
2169 GuestProcessTool procTool;
2170 int rcGuest = VERR_IPE_UNINITIALIZED_STATUS;
2171 int vrc = procTool.init(pSession, procInfo, false /* Async */, &rcGuest);
2172 if (RT_SUCCESS(vrc))
2173 {
2174 if (RT_SUCCESS(rcGuest))
2175 vrc = procTool.wait(GUESTPROCESSTOOL_WAIT_FLAG_NONE, &rcGuest);
2176 if (RT_SUCCESS(vrc))
2177 vrc = procTool.terminatedOk();
2178 }
2179
2180 if (RT_FAILURE(vrc))
2181 {
2182 switch (vrc)
2183 {
2184 case VWRN_GSTCTL_PROCESS_EXIT_CODE:
2185 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2186 Utf8StrFmt(GuestSession::tr("Running update file \"%s\" on guest failed: %Rrc"),
2187 procInfo.mExecutable.c_str(), procTool.getRc()));
2188 break;
2189
2190 case VERR_GSTCTL_GUEST_ERROR:
2191 setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestProcess::i_guestErrorToString(rcGuest));
2192 break;
2193
2194 case VERR_INVALID_STATE: /** @todo Special guest control rc needed! */
2195 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2196 Utf8StrFmt(GuestSession::tr("Update file \"%s\" reported invalid running state"),
2197 procInfo.mExecutable.c_str()));
2198 break;
2199
2200 default:
2201 setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2202 Utf8StrFmt(GuestSession::tr("Error while running update file \"%s\" on guest: %Rrc"),
2203 procInfo.mExecutable.c_str(), vrc));
2204 break;
2205 }
2206 }
2207
2208 return vrc;
2209}
2210
2211int GuestSessionTaskUpdateAdditions::Run(void)
2212{
2213 LogFlowThisFuncEnter();
2214
2215 ComObjPtr<GuestSession> pSession = mSession;
2216 Assert(!pSession.isNull());
2217
2218 AutoCaller autoCaller(pSession);
2219 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2220
2221 int rc = setProgress(10);
2222 if (RT_FAILURE(rc))
2223 return rc;
2224
2225 HRESULT hr = S_OK;
2226
2227 LogRel(("Automatic update of Guest Additions started, using \"%s\"\n", mSource.c_str()));
2228
2229 ComObjPtr<Guest> pGuest(mSession->i_getParent());
2230#if 0
2231 /*
2232 * Wait for the guest being ready within 30 seconds.
2233 */
2234 AdditionsRunLevelType_T addsRunLevel;
2235 uint64_t tsStart = RTTimeSystemMilliTS();
2236 while ( SUCCEEDED(hr = pGuest->COMGETTER(AdditionsRunLevel)(&addsRunLevel))
2237 && ( addsRunLevel != AdditionsRunLevelType_Userland
2238 && addsRunLevel != AdditionsRunLevelType_Desktop))
2239 {
2240 if ((RTTimeSystemMilliTS() - tsStart) > 30 * 1000)
2241 {
2242 rc = VERR_TIMEOUT;
2243 break;
2244 }
2245
2246 RTThreadSleep(100); /* Wait a bit. */
2247 }
2248
2249 if (FAILED(hr)) rc = VERR_TIMEOUT;
2250 if (rc == VERR_TIMEOUT)
2251 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2252 Utf8StrFmt(GuestSession::tr("Guest Additions were not ready within time, giving up")));
2253#else
2254 /*
2255 * For use with the GUI we don't want to wait, just return so that the manual .ISO mounting
2256 * can continue.
2257 */
2258 AdditionsRunLevelType_T addsRunLevel;
2259 if ( FAILED(hr = pGuest->COMGETTER(AdditionsRunLevel)(&addsRunLevel))
2260 || ( addsRunLevel != AdditionsRunLevelType_Userland
2261 && addsRunLevel != AdditionsRunLevelType_Desktop))
2262 {
2263 if (addsRunLevel == AdditionsRunLevelType_System)
2264 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2265 Utf8StrFmt(GuestSession::tr("Guest Additions are installed but not fully loaded yet, aborting automatic update")));
2266 else
2267 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2268 Utf8StrFmt(GuestSession::tr("Guest Additions not installed or ready, aborting automatic update")));
2269 rc = VERR_NOT_SUPPORTED;
2270 }
2271#endif
2272
2273 if (RT_SUCCESS(rc))
2274 {
2275 /*
2276 * Determine if we are able to update automatically. This only works
2277 * if there are recent Guest Additions installed already.
2278 */
2279 Utf8Str strAddsVer;
2280 rc = getGuestProperty(pGuest, "/VirtualBox/GuestAdd/Version", strAddsVer);
2281 if ( RT_SUCCESS(rc)
2282 && RTStrVersionCompare(strAddsVer.c_str(), "4.1") < 0)
2283 {
2284 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2285 Utf8StrFmt(GuestSession::tr("Guest has too old Guest Additions (%s) installed for automatic updating, please update manually"),
2286 strAddsVer.c_str()));
2287 rc = VERR_NOT_SUPPORTED;
2288 }
2289 }
2290
2291 Utf8Str strOSVer;
2292 eOSType osType = eOSType_Unknown;
2293 if (RT_SUCCESS(rc))
2294 {
2295 /*
2296 * Determine guest OS type and the required installer image.
2297 */
2298 Utf8Str strOSType;
2299 rc = getGuestProperty(pGuest, "/VirtualBox/GuestInfo/OS/Product", strOSType);
2300 if (RT_SUCCESS(rc))
2301 {
2302 if ( strOSType.contains("Microsoft", Utf8Str::CaseInsensitive)
2303 || strOSType.contains("Windows", Utf8Str::CaseInsensitive))
2304 {
2305 osType = eOSType_Windows;
2306
2307 /*
2308 * Determine guest OS version.
2309 */
2310 rc = getGuestProperty(pGuest, "/VirtualBox/GuestInfo/OS/Release", strOSVer);
2311 if (RT_FAILURE(rc))
2312 {
2313 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2314 Utf8StrFmt(GuestSession::tr("Unable to detected guest OS version, please update manually")));
2315 rc = VERR_NOT_SUPPORTED;
2316 }
2317
2318 /* Because Windows 2000 + XP and is bitching with WHQL popups even if we have signed drivers we
2319 * can't do automated updates here. */
2320 /* Windows XP 64-bit (5.2) is a Windows 2003 Server actually, so skip this here. */
2321 if ( RT_SUCCESS(rc)
2322 && RTStrVersionCompare(strOSVer.c_str(), "5.0") >= 0)
2323 {
2324 if ( strOSVer.startsWith("5.0") /* Exclude the build number. */
2325 || strOSVer.startsWith("5.1") /* Exclude the build number. */)
2326 {
2327 /* If we don't have AdditionsUpdateFlag_WaitForUpdateStartOnly set we can't continue
2328 * because the Windows Guest Additions installer will fail because of WHQL popups. If the
2329 * flag is set this update routine ends successfully as soon as the installer was started
2330 * (and the user has to deal with it in the guest). */
2331 if (!(mFlags & AdditionsUpdateFlag_WaitForUpdateStartOnly))
2332 {
2333 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2334 Utf8StrFmt(GuestSession::tr("Windows 2000 and XP are not supported for automatic updating due to WHQL interaction, please update manually")));
2335 rc = VERR_NOT_SUPPORTED;
2336 }
2337 }
2338 }
2339 else
2340 {
2341 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2342 Utf8StrFmt(GuestSession::tr("%s (%s) not supported for automatic updating, please update manually"),
2343 strOSType.c_str(), strOSVer.c_str()));
2344 rc = VERR_NOT_SUPPORTED;
2345 }
2346 }
2347 else if (strOSType.contains("Solaris", Utf8Str::CaseInsensitive))
2348 {
2349 osType = eOSType_Solaris;
2350 }
2351 else /* Everything else hopefully means Linux :-). */
2352 osType = eOSType_Linux;
2353
2354#if 1 /* Only Windows is supported (and tested) at the moment. */
2355 if ( RT_SUCCESS(rc)
2356 && osType != eOSType_Windows)
2357 {
2358 hr = setProgressErrorMsg(VBOX_E_NOT_SUPPORTED,
2359 Utf8StrFmt(GuestSession::tr("Detected guest OS (%s) does not support automatic Guest Additions updating, please update manually"),
2360 strOSType.c_str()));
2361 rc = VERR_NOT_SUPPORTED;
2362 }
2363#endif
2364 }
2365 }
2366
2367 RTISOFSFILE iso;
2368 if (RT_SUCCESS(rc))
2369 {
2370 /*
2371 * Try to open the .ISO file to extract all needed files.
2372 */
2373 rc = RTIsoFsOpen(&iso, mSource.c_str());
2374 if (RT_FAILURE(rc))
2375 {
2376 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2377 Utf8StrFmt(GuestSession::tr("Unable to open Guest Additions .ISO file \"%s\": %Rrc"),
2378 mSource.c_str(), rc));
2379 }
2380 else
2381 {
2382 /* Set default installation directories. */
2383 Utf8Str strUpdateDir = "/tmp/";
2384 if (osType == eOSType_Windows)
2385 strUpdateDir = "C:\\Temp\\";
2386
2387 rc = setProgress(5);
2388
2389 /* Try looking up the Guest Additions installation directory. */
2390 if (RT_SUCCESS(rc))
2391 {
2392 /* Try getting the installed Guest Additions version to know whether we
2393 * can install our temporary Guest Addition data into the original installation
2394 * directory.
2395 *
2396 * Because versions prior to 4.2 had bugs wrt spaces in paths we have to choose
2397 * a different location then.
2398 */
2399 bool fUseInstallDir = false;
2400
2401 Utf8Str strAddsVer;
2402 rc = getGuestProperty(pGuest, "/VirtualBox/GuestAdd/Version", strAddsVer);
2403 if ( RT_SUCCESS(rc)
2404 && RTStrVersionCompare(strAddsVer.c_str(), "4.2r80329") > 0)
2405 {
2406 fUseInstallDir = true;
2407 }
2408
2409 if (fUseInstallDir)
2410 {
2411 if (RT_SUCCESS(rc))
2412 rc = getGuestProperty(pGuest, "/VirtualBox/GuestAdd/InstallDir", strUpdateDir);
2413 if (RT_SUCCESS(rc))
2414 {
2415 if (osType == eOSType_Windows)
2416 {
2417 strUpdateDir.findReplace('/', '\\');
2418 strUpdateDir.append("\\Update\\");
2419 }
2420 else
2421 strUpdateDir.append("/update/");
2422 }
2423 }
2424 }
2425
2426 if (RT_SUCCESS(rc))
2427 LogRel(("Guest Additions update directory is: %s\n",
2428 strUpdateDir.c_str()));
2429
2430 /* Create the installation directory. */
2431 int rcGuest = VERR_IPE_UNINITIALIZED_STATUS;
2432 rc = pSession->i_directoryCreate(strUpdateDir, 755 /* Mode */, DirectoryCreateFlag_Parents, &rcGuest);
2433 if (RT_FAILURE(rc))
2434 {
2435 switch (rc)
2436 {
2437 case VERR_GSTCTL_GUEST_ERROR:
2438 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR, GuestProcess::i_guestErrorToString(rcGuest));
2439 break;
2440
2441 default:
2442 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2443 Utf8StrFmt(GuestSession::tr("Error creating installation directory \"%s\" on the guest: %Rrc"),
2444 strUpdateDir.c_str(), rc));
2445 break;
2446 }
2447 }
2448 if (RT_SUCCESS(rc))
2449 rc = setProgress(10);
2450
2451 if (RT_SUCCESS(rc))
2452 {
2453 /* Prepare the file(s) we want to copy over to the guest and
2454 * (maybe) want to run. */
2455 switch (osType)
2456 {
2457 case eOSType_Windows:
2458 {
2459 /* Do we need to install our certificates? We do this for W2K and up. */
2460 bool fInstallCert = false;
2461
2462 /* Only Windows 2000 and up need certificates to be installed. */
2463 if (RTStrVersionCompare(strOSVer.c_str(), "5.0") >= 0)
2464 {
2465 fInstallCert = true;
2466 LogRel(("Certificates for auto updating WHQL drivers will be installed\n"));
2467 }
2468 else
2469 LogRel(("Skipping installation of certificates for WHQL drivers\n"));
2470
2471 if (fInstallCert)
2472 {
2473 static struct { const char *pszDst, *pszIso; } const s_aCertFiles[] =
2474 {
2475 { "vbox.cer", "CERT/VBOX.CER" },
2476 { "vbox-sha1.cer", "CERT/VBOX_SHA1.CER" },
2477 { "vbox-sha256.cer", "CERT/VBOX_SHA256.CER" },
2478 { "vbox-sha256-r3.cer", "CERT/VBOX_SHA256_R3.CER" },
2479 { "oracle-vbox.cer", "CERT/ORACLE_VBOX.CER" },
2480 };
2481 uint32_t fCopyCertUtil = ISOFILE_FLAG_COPY_FROM_ISO;
2482 for (uint32_t i = 0; i < RT_ELEMENTS(s_aCertFiles); i++)
2483 {
2484 /* Skip if not present on the ISO. */
2485 uint32_t offIgn;
2486 size_t cbIgn;
2487 rc = RTIsoFsGetFileInfo(&iso, s_aCertFiles[i].pszIso, &offIgn, &cbIgn);
2488 if (RT_FAILURE(rc))
2489 continue;
2490
2491 /* Copy the certificate certificate. */
2492 Utf8Str const strDstCert(strUpdateDir + s_aCertFiles[i].pszDst);
2493 mFiles.push_back(ISOFile(s_aCertFiles[i].pszIso,
2494 strDstCert,
2495 ISOFILE_FLAG_COPY_FROM_ISO | ISOFILE_FLAG_OPTIONAL));
2496
2497 /* Out certificate installation utility. */
2498 /* First pass: Copy over the file (first time only) + execute it to remove any
2499 * existing VBox certificates. */
2500 GuestProcessStartupInfo siCertUtilRem;
2501 siCertUtilRem.mName = "VirtualBox Certificate Utility, removing old VirtualBox certificates";
2502 siCertUtilRem.mArguments.push_back(Utf8Str("remove-trusted-publisher"));
2503 siCertUtilRem.mArguments.push_back(Utf8Str("--root")); /* Add root certificate as well. */
2504 siCertUtilRem.mArguments.push_back(strDstCert);
2505 siCertUtilRem.mArguments.push_back(strDstCert);
2506 mFiles.push_back(ISOFile("CERT/VBOXCERTUTIL.EXE",
2507 strUpdateDir + "VBoxCertUtil.exe",
2508 fCopyCertUtil | ISOFILE_FLAG_EXECUTE | ISOFILE_FLAG_OPTIONAL,
2509 siCertUtilRem));
2510 fCopyCertUtil = 0;
2511 /* Second pass: Only execute (but don't copy) again, this time installng the
2512 * recent certificates just copied over. */
2513 GuestProcessStartupInfo siCertUtilAdd;
2514 siCertUtilAdd.mName = "VirtualBox Certificate Utility, installing VirtualBox certificates";
2515 siCertUtilAdd.mArguments.push_back(Utf8Str("add-trusted-publisher"));
2516 siCertUtilAdd.mArguments.push_back(Utf8Str("--root")); /* Add root certificate as well. */
2517 siCertUtilAdd.mArguments.push_back(strDstCert);
2518 siCertUtilAdd.mArguments.push_back(strDstCert);
2519 mFiles.push_back(ISOFile("CERT/VBOXCERTUTIL.EXE",
2520 strUpdateDir + "VBoxCertUtil.exe",
2521 ISOFILE_FLAG_EXECUTE | ISOFILE_FLAG_OPTIONAL,
2522 siCertUtilAdd));
2523 }
2524 }
2525 /* The installers in different flavors, as we don't know (and can't assume)
2526 * the guest's bitness. */
2527 mFiles.push_back(ISOFile("VBOXWINDOWSADDITIONS_X86.EXE",
2528 strUpdateDir + "VBoxWindowsAdditions-x86.exe",
2529 ISOFILE_FLAG_COPY_FROM_ISO));
2530 mFiles.push_back(ISOFile("VBOXWINDOWSADDITIONS_AMD64.EXE",
2531 strUpdateDir + "VBoxWindowsAdditions-amd64.exe",
2532 ISOFILE_FLAG_COPY_FROM_ISO));
2533 /* The stub loader which decides which flavor to run. */
2534 GuestProcessStartupInfo siInstaller;
2535 siInstaller.mName = "VirtualBox Windows Guest Additions Installer";
2536 /* Set a running timeout of 5 minutes -- the Windows Guest Additions
2537 * setup can take quite a while, so be on the safe side. */
2538 siInstaller.mTimeoutMS = 5 * 60 * 1000;
2539 siInstaller.mArguments.push_back(Utf8Str("/S")); /* We want to install in silent mode. */
2540 siInstaller.mArguments.push_back(Utf8Str("/l")); /* ... and logging enabled. */
2541 /* Don't quit VBoxService during upgrade because it still is used for this
2542 * piece of code we're in right now (that is, here!) ... */
2543 siInstaller.mArguments.push_back(Utf8Str("/no_vboxservice_exit"));
2544 /* Tell the installer to report its current installation status
2545 * using a running VBoxTray instance via balloon messages in the
2546 * Windows taskbar. */
2547 siInstaller.mArguments.push_back(Utf8Str("/post_installstatus"));
2548 /* Add optional installer command line arguments from the API to the
2549 * installer's startup info. */
2550 rc = addProcessArguments(siInstaller.mArguments, mArguments);
2551 AssertRC(rc);
2552 /* If the caller does not want to wait for out guest update process to end,
2553 * complete the progress object now so that the caller can do other work. */
2554 if (mFlags & AdditionsUpdateFlag_WaitForUpdateStartOnly)
2555 siInstaller.mFlags |= ProcessCreateFlag_WaitForProcessStartOnly;
2556 mFiles.push_back(ISOFile("VBOXWINDOWSADDITIONS.EXE",
2557 strUpdateDir + "VBoxWindowsAdditions.exe",
2558 ISOFILE_FLAG_COPY_FROM_ISO | ISOFILE_FLAG_EXECUTE, siInstaller));
2559 break;
2560 }
2561 case eOSType_Linux:
2562 /** @todo Add Linux support. */
2563 break;
2564 case eOSType_Solaris:
2565 /** @todo Add Solaris support. */
2566 break;
2567 default:
2568 AssertReleaseMsgFailed(("Unsupported guest type: %d\n", osType));
2569 break;
2570 }
2571 }
2572
2573 if (RT_SUCCESS(rc))
2574 {
2575 /* We want to spend 40% total for all copying operations. So roughly
2576 * calculate the specific percentage step of each copied file. */
2577 uint8_t uOffset = 20; /* Start at 20%. */
2578 uint8_t uStep = 40 / (uint8_t)mFiles.size(); Assert(mFiles.size() <= 10);
2579
2580 LogRel(("Copying over Guest Additions update files to the guest ...\n"));
2581
2582 std::vector<ISOFile>::const_iterator itFiles = mFiles.begin();
2583 while (itFiles != mFiles.end())
2584 {
2585 if (itFiles->fFlags & ISOFILE_FLAG_COPY_FROM_ISO)
2586 {
2587 bool fOptional = false;
2588 if (itFiles->fFlags & ISOFILE_FLAG_OPTIONAL)
2589 fOptional = true;
2590 rc = copyFileToGuest(pSession, &iso, itFiles->strSource, itFiles->strDest, fOptional);
2591 if (RT_FAILURE(rc))
2592 {
2593 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2594 Utf8StrFmt(GuestSession::tr("Error while copying file \"%s\" to \"%s\" on the guest: %Rrc"),
2595 itFiles->strSource.c_str(), itFiles->strDest.c_str(), rc));
2596 break;
2597 }
2598 }
2599
2600 rc = setProgress(uOffset);
2601 if (RT_FAILURE(rc))
2602 break;
2603 uOffset += uStep;
2604
2605 ++itFiles;
2606 }
2607 }
2608
2609 /* Done copying, close .ISO file. */
2610 RTIsoFsClose(&iso);
2611
2612 if (RT_SUCCESS(rc))
2613 {
2614 /* We want to spend 35% total for all copying operations. So roughly
2615 * calculate the specific percentage step of each copied file. */
2616 uint8_t uOffset = 60; /* Start at 60%. */
2617 uint8_t uStep = 35 / (uint8_t)mFiles.size(); Assert(mFiles.size() <= 10);
2618
2619 LogRel(("Executing Guest Additions update files ...\n"));
2620
2621 std::vector<ISOFile>::iterator itFiles = mFiles.begin();
2622 while (itFiles != mFiles.end())
2623 {
2624 if (itFiles->fFlags & ISOFILE_FLAG_EXECUTE)
2625 {
2626 rc = runFileOnGuest(pSession, itFiles->mProcInfo);
2627 if (RT_FAILURE(rc))
2628 break;
2629 }
2630
2631 rc = setProgress(uOffset);
2632 if (RT_FAILURE(rc))
2633 break;
2634 uOffset += uStep;
2635
2636 ++itFiles;
2637 }
2638 }
2639
2640 if (RT_SUCCESS(rc))
2641 {
2642 LogRel(("Automatic update of Guest Additions succeeded\n"));
2643 rc = setProgressSuccess();
2644 }
2645 }
2646 }
2647
2648 if (RT_FAILURE(rc))
2649 {
2650 if (rc == VERR_CANCELLED)
2651 {
2652 LogRel(("Automatic update of Guest Additions was canceled\n"));
2653
2654 hr = setProgressErrorMsg(VBOX_E_IPRT_ERROR,
2655 Utf8StrFmt(GuestSession::tr("Installation was canceled")));
2656 }
2657 else
2658 {
2659 Utf8Str strError = Utf8StrFmt("No further error information available (%Rrc)", rc);
2660 if (!mProgress.isNull()) /* Progress object is optional. */
2661 {
2662 com::ProgressErrorInfo errorInfo(mProgress);
2663 if ( errorInfo.isFullAvailable()
2664 || errorInfo.isBasicAvailable())
2665 {
2666 strError = errorInfo.getText();
2667 }
2668 }
2669
2670 LogRel(("Automatic update of Guest Additions failed: %s (%Rhrc)\n",
2671 strError.c_str(), hr));
2672 }
2673
2674 LogRel(("Please install Guest Additions manually\n"));
2675 }
2676
2677 /** @todo Clean up copied / left over installation files. */
2678
2679 LogFlowFuncLeaveRC(rc);
2680 return rc;
2681}
2682
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