VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxManage/VBoxManageGuestCtrl.cpp@ 72440

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

Guest Control/VBoxManage: Check if progress object is valid in copy[to|from] commands.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 126.1 KB
Line 
1/* $Id: VBoxManageGuestCtrl.cpp 72006 2018-04-24 15:31:19Z vboxsync $ */
2/** @file
3 * VBoxManage - Implementation of guestcontrol command.
4 */
5
6/*
7 * Copyright (C) 2010-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#include "VBoxManage.h"
23#include "VBoxManageGuestCtrl.h"
24
25#ifndef VBOX_ONLY_DOCS
26
27#include <VBox/com/array.h>
28#include <VBox/com/com.h>
29#include <VBox/com/ErrorInfo.h>
30#include <VBox/com/errorprint.h>
31#include <VBox/com/listeners.h>
32#include <VBox/com/NativeEventQueue.h>
33#include <VBox/com/string.h>
34#include <VBox/com/VirtualBox.h>
35
36#include <VBox/err.h>
37#include <VBox/log.h>
38
39#include <iprt/asm.h>
40#include <iprt/dir.h>
41#include <iprt/file.h>
42#include <iprt/isofs.h>
43#include <iprt/getopt.h>
44#include <iprt/list.h>
45#include <iprt/path.h>
46#include <iprt/process.h> /* For RTProcSelf(). */
47#include <iprt/thread.h>
48#include <iprt/vfs.h>
49
50#include <map>
51#include <vector>
52
53#ifdef USE_XPCOM_QUEUE
54# include <sys/select.h>
55# include <errno.h>
56#endif
57
58#include <signal.h>
59
60#ifdef RT_OS_DARWIN
61# include <CoreFoundation/CFRunLoop.h>
62#endif
63
64using namespace com;
65
66
67/*********************************************************************************************************************************
68* Defined Constants And Macros *
69*********************************************************************************************************************************/
70#define GCTLCMD_COMMON_OPT_USER 999 /**< The --username option number. */
71#define GCTLCMD_COMMON_OPT_PASSWORD 998 /**< The --password option number. */
72#define GCTLCMD_COMMON_OPT_PASSWORD_FILE 997 /**< The --password-file option number. */
73#define GCTLCMD_COMMON_OPT_DOMAIN 996 /**< The --domain option number. */
74/** Common option definitions. */
75#define GCTLCMD_COMMON_OPTION_DEFS() \
76 { "--username", GCTLCMD_COMMON_OPT_USER, RTGETOPT_REQ_STRING }, \
77 { "--passwordfile", GCTLCMD_COMMON_OPT_PASSWORD_FILE, RTGETOPT_REQ_STRING }, \
78 { "--password", GCTLCMD_COMMON_OPT_PASSWORD, RTGETOPT_REQ_STRING }, \
79 { "--domain", GCTLCMD_COMMON_OPT_DOMAIN, RTGETOPT_REQ_STRING }, \
80 { "--quiet", 'q', RTGETOPT_REQ_NOTHING }, \
81 { "--verbose", 'v', RTGETOPT_REQ_NOTHING },
82
83/** Handles common options in the typical option parsing switch. */
84#define GCTLCMD_COMMON_OPTION_CASES(a_pCtx, a_ch, a_pValueUnion) \
85 case 'v': \
86 case 'q': \
87 case GCTLCMD_COMMON_OPT_USER: \
88 case GCTLCMD_COMMON_OPT_DOMAIN: \
89 case GCTLCMD_COMMON_OPT_PASSWORD: \
90 case GCTLCMD_COMMON_OPT_PASSWORD_FILE: \
91 { \
92 RTEXITCODE rcExitCommon = gctlCtxSetOption(a_pCtx, a_ch, a_pValueUnion); \
93 if (RT_UNLIKELY(rcExitCommon != RTEXITCODE_SUCCESS)) \
94 return rcExitCommon; \
95 } break
96
97
98/*********************************************************************************************************************************
99* Global Variables *
100*********************************************************************************************************************************/
101/** Set by the signal handler when current guest control
102 * action shall be aborted. */
103static volatile bool g_fGuestCtrlCanceled = false;
104
105
106/*********************************************************************************************************************************
107* Structures and Typedefs *
108*********************************************************************************************************************************/
109/**
110 * Listener declarations.
111 */
112VBOX_LISTENER_DECLARE(GuestFileEventListenerImpl)
113VBOX_LISTENER_DECLARE(GuestProcessEventListenerImpl)
114VBOX_LISTENER_DECLARE(GuestSessionEventListenerImpl)
115VBOX_LISTENER_DECLARE(GuestEventListenerImpl)
116
117
118/**
119 * Definition of a guestcontrol command, with handler and various flags.
120 */
121typedef struct GCTLCMDDEF
122{
123 /** The command name. */
124 const char *pszName;
125
126 /**
127 * Actual command handler callback.
128 *
129 * @param pCtx Pointer to command context to use.
130 */
131 DECLR3CALLBACKMEMBER(RTEXITCODE, pfnHandler, (struct GCTLCMDCTX *pCtx, int argc, char **argv));
132
133 /** The command usage flags. */
134 uint32_t fCmdUsage;
135 /** Command context flags (GCTLCMDCTX_F_XXX). */
136 uint32_t fCmdCtx;
137} GCTLCMD;
138/** Pointer to a const guest control command definition. */
139typedef GCTLCMDDEF const *PCGCTLCMDDEF;
140
141/** @name GCTLCMDCTX_F_XXX - Command context flags.
142 * @{
143 */
144/** No flags set. */
145#define GCTLCMDCTX_F_NONE 0
146/** Don't install a signal handler (CTRL+C trap). */
147#define GCTLCMDCTX_F_NO_SIGNAL_HANDLER RT_BIT(0)
148/** No guest session needed. */
149#define GCTLCMDCTX_F_SESSION_ANONYMOUS RT_BIT(1)
150/** @} */
151
152/**
153 * Context for handling a specific command.
154 */
155typedef struct GCTLCMDCTX
156{
157 HandlerArg *pArg;
158
159 /** Pointer to the command definition. */
160 PCGCTLCMDDEF pCmdDef;
161 /** The VM name or UUID. */
162 const char *pszVmNameOrUuid;
163
164 /** Whether we've done the post option parsing init already. */
165 bool fPostOptionParsingInited;
166 /** Whether we've locked the VM session. */
167 bool fLockedVmSession;
168 /** Whether to detach (@c true) or close the session. */
169 bool fDetachGuestSession;
170 /** Set if we've installed the signal handler. */
171 bool fInstalledSignalHandler;
172 /** The verbosity level. */
173 uint32_t cVerbose;
174 /** User name. */
175 Utf8Str strUsername;
176 /** Password. */
177 Utf8Str strPassword;
178 /** Domain. */
179 Utf8Str strDomain;
180 /** Pointer to the IGuest interface. */
181 ComPtr<IGuest> pGuest;
182 /** Pointer to the to be used guest session. */
183 ComPtr<IGuestSession> pGuestSession;
184 /** The guest session ID. */
185 ULONG uSessionID;
186
187} GCTLCMDCTX, *PGCTLCMDCTX;
188
189/**
190 * An entry for a source element, including an optional DOS-like wildcard (*,?).
191 */
192class SOURCEFILEENTRY
193{
194 public:
195
196 SOURCEFILEENTRY(const char *pszSource, const char *pszFilter)
197 : mSource(pszSource),
198 mFilter(pszFilter) {}
199
200 SOURCEFILEENTRY(const char *pszSource)
201 : mSource(pszSource)
202 {
203 Parse(pszSource);
204 }
205
206 Utf8Str GetSource() const
207 {
208 return mSource;
209 }
210
211 Utf8Str GetFilter() const
212 {
213 return mFilter;
214 }
215
216 private:
217
218 int Parse(const char *pszPath)
219 {
220 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
221
222 if ( !RTFileExists(pszPath)
223 && !RTDirExists(pszPath))
224 {
225 /* No file and no directory -- maybe a filter? */
226 char *pszFilename = RTPathFilename(pszPath);
227 if ( pszFilename
228 && strpbrk(pszFilename, "*?"))
229 {
230 /* Yep, get the actual filter part. */
231 mFilter = RTPathFilename(pszPath);
232 /* Remove the filter from actual sourcec directory name. */
233 RTPathStripFilename(mSource.mutableRaw());
234 mSource.jolt();
235 }
236 }
237
238 return VINF_SUCCESS; /** @todo */
239 }
240
241 private:
242
243 Utf8Str mSource;
244 Utf8Str mFilter;
245};
246typedef std::vector<SOURCEFILEENTRY> SOURCEVEC, *PSOURCEVEC;
247
248/**
249 * An entry for an element which needs to be copied/created to/on the guest.
250 */
251typedef struct DESTFILEENTRY
252{
253 DESTFILEENTRY(Utf8Str strFileName) : mFileName(strFileName) {}
254 Utf8Str mFileName;
255} DESTFILEENTRY, *PDESTFILEENTRY;
256/*
257 * Map for holding destination entries, whereas the key is the destination
258 * directory and the mapped value is a vector holding all elements for this directory.
259 */
260typedef std::map< Utf8Str, std::vector<DESTFILEENTRY> > DESTDIRMAP, *PDESTDIRMAP;
261typedef std::map< Utf8Str, std::vector<DESTFILEENTRY> >::iterator DESTDIRMAPITER, *PDESTDIRMAPITER;
262
263
264/**
265 * RTGetOpt-IDs for the guest execution control command line.
266 */
267enum GETOPTDEF_EXEC
268{
269 GETOPTDEF_EXEC_IGNOREORPHANEDPROCESSES = 1000,
270 GETOPTDEF_EXEC_NO_PROFILE,
271 GETOPTDEF_EXEC_OUTPUTFORMAT,
272 GETOPTDEF_EXEC_DOS2UNIX,
273 GETOPTDEF_EXEC_UNIX2DOS,
274 GETOPTDEF_EXEC_WAITFOREXIT,
275 GETOPTDEF_EXEC_WAITFORSTDOUT,
276 GETOPTDEF_EXEC_WAITFORSTDERR
277};
278
279enum kStreamTransform
280{
281 kStreamTransform_None = 0,
282 kStreamTransform_Dos2Unix,
283 kStreamTransform_Unix2Dos
284};
285#endif /* VBOX_ONLY_DOCS */
286
287
288void usageGuestControl(PRTSTREAM pStrm, const char *pcszSep1, const char *pcszSep2, uint32_t uSubCmd)
289{
290 const uint32_t fAnonSubCmds = USAGE_GSTCTRL_CLOSESESSION
291 | USAGE_GSTCTRL_LIST
292 | USAGE_GSTCTRL_CLOSEPROCESS
293 | USAGE_GSTCTRL_CLOSESESSION
294 | USAGE_GSTCTRL_UPDATEGA
295 | USAGE_GSTCTRL_WATCH;
296
297 /* 0 1 2 3 4 5 6 7 8XXXXXXXXXX */
298 /* 0123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890 */
299 if (~fAnonSubCmds & uSubCmd)
300 RTStrmPrintf(pStrm,
301 "%s guestcontrol %s <uuid|vmname> [--verbose|-v] [--quiet|-q]\n"
302 " [--username <name>] [--domain <domain>]\n"
303 " [--passwordfile <file> | --password <password>]\n%s",
304 pcszSep1, pcszSep2, uSubCmd == ~0U ? "\n" : "");
305 if (uSubCmd & USAGE_GSTCTRL_RUN)
306 RTStrmPrintf(pStrm,
307 " run [common-options]\n"
308 " [--exe <path to executable>] [--timeout <msec>]\n"
309 " [-E|--putenv <NAME>[=<VALUE>]] [--unquoted-args]\n"
310 " [--ignore-operhaned-processes] [--profile]\n"
311 " [--no-wait-stdout|--wait-stdout]\n"
312 " [--no-wait-stderr|--wait-stderr]\n"
313 " [--dos2unix] [--unix2dos]\n"
314 " -- <program/arg0> [argument1] ... [argumentN]]\n"
315 "\n");
316 if (uSubCmd & USAGE_GSTCTRL_START)
317 RTStrmPrintf(pStrm,
318 " start [common-options]\n"
319 " [--exe <path to executable>] [--timeout <msec>]\n"
320 " [-E|--putenv <NAME>[=<VALUE>]] [--unquoted-args]\n"
321 " [--ignore-operhaned-processes] [--profile]\n"
322 " -- <program/arg0> [argument1] ... [argumentN]]\n"
323 "\n");
324 if (uSubCmd & USAGE_GSTCTRL_COPYFROM)
325 RTStrmPrintf(pStrm,
326 " copyfrom [common-options]\n"
327 " [--follow] [-R|--recursive]\n"
328 " <guest-src0> [guest-src1 [...]] <host-dst>\n"
329 "\n"
330 " copyfrom [common-options]\n"
331 " [--follow] [-R|--recursive]\n"
332 " [--target-directory <host-dst-dir>]\n"
333 " <guest-src0> [guest-src1 [...]]\n"
334 "\n");
335 if (uSubCmd & USAGE_GSTCTRL_COPYTO)
336 RTStrmPrintf(pStrm,
337 " copyto [common-options]\n"
338 " [--follow] [-R|--recursive]\n"
339 " <host-src0> [host-src1 [...]] <guest-dst>\n"
340 "\n"
341 " copyto [common-options]\n"
342 " [--follow] [-R|--recursive]\n"
343 " [--target-directory <guest-dst>]\n"
344 " <host-src0> [host-src1 [...]]\n"
345 "\n");
346 if (uSubCmd & USAGE_GSTCTRL_MKDIR)
347 RTStrmPrintf(pStrm,
348 " mkdir|createdir[ectory] [common-options]\n"
349 " [--parents] [--mode <mode>]\n"
350 " <guest directory> [...]\n"
351 "\n");
352 if (uSubCmd & USAGE_GSTCTRL_RMDIR)
353 RTStrmPrintf(pStrm,
354 " rmdir|removedir[ectory] [common-options]\n"
355 " [-R|--recursive]\n"
356 " <guest directory> [...]\n"
357 "\n");
358 if (uSubCmd & USAGE_GSTCTRL_RM)
359 RTStrmPrintf(pStrm,
360 " removefile|rm [common-options] [-f|--force]\n"
361 " <guest file> [...]\n"
362 "\n");
363 if (uSubCmd & USAGE_GSTCTRL_MV)
364 RTStrmPrintf(pStrm,
365 " mv|move|ren[ame] [common-options]\n"
366 " <source> [source1 [...]] <dest>\n"
367 "\n");
368 if (uSubCmd & USAGE_GSTCTRL_MKTEMP)
369 RTStrmPrintf(pStrm,
370 " mktemp|createtemp[orary] [common-options]\n"
371 " [--secure] [--mode <mode>] [--tmpdir <directory>]\n"
372 " <template>\n"
373 "\n");
374 if (uSubCmd & USAGE_GSTCTRL_STAT)
375 RTStrmPrintf(pStrm,
376 " stat [common-options]\n"
377 " <file> [...]\n"
378 "\n");
379
380 /*
381 * Command not requiring authentication.
382 */
383 if (fAnonSubCmds & uSubCmd)
384 {
385 /* 0 1 2 3 4 5 6 7 8XXXXXXXXXX */
386 /* 0123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890 */
387 RTStrmPrintf(pStrm,
388 "%s guestcontrol %s <uuid|vmname> [--verbose|-v] [--quiet|-q]\n%s",
389 pcszSep1, pcszSep2, uSubCmd == ~0U ? "\n" : "");
390 if (uSubCmd & USAGE_GSTCTRL_LIST)
391 RTStrmPrintf(pStrm,
392 " list <all|sessions|processes|files> [common-opts]\n"
393 "\n");
394 if (uSubCmd & USAGE_GSTCTRL_CLOSEPROCESS)
395 RTStrmPrintf(pStrm,
396 " closeprocess [common-options]\n"
397 " < --session-id <ID>\n"
398 " | --session-name <name or pattern>\n"
399 " <PID1> [PID1 [...]]\n"
400 "\n");
401 if (uSubCmd & USAGE_GSTCTRL_CLOSESESSION)
402 RTStrmPrintf(pStrm,
403 " closesession [common-options]\n"
404 " < --all | --session-id <ID>\n"
405 " | --session-name <name or pattern> >\n"
406 "\n");
407 if (uSubCmd & USAGE_GSTCTRL_UPDATEGA)
408 RTStrmPrintf(pStrm,
409 " updatega|updateguestadditions|updateadditions\n"
410 " [--source <guest additions .ISO>]\n"
411 " [--wait-start] [common-options]\n"
412 " [-- [<argument1>] ... [<argumentN>]]\n"
413 "\n");
414 if (uSubCmd & USAGE_GSTCTRL_WATCH)
415 RTStrmPrintf(pStrm,
416 " watch [common-options]\n"
417 "\n");
418 }
419}
420
421#ifndef VBOX_ONLY_DOCS
422
423
424#ifdef RT_OS_WINDOWS
425static BOOL WINAPI gctlSignalHandler(DWORD dwCtrlType)
426{
427 bool fEventHandled = FALSE;
428 switch (dwCtrlType)
429 {
430 /* User pressed CTRL+C or CTRL+BREAK or an external event was sent
431 * via GenerateConsoleCtrlEvent(). */
432 case CTRL_BREAK_EVENT:
433 case CTRL_CLOSE_EVENT:
434 case CTRL_C_EVENT:
435 ASMAtomicWriteBool(&g_fGuestCtrlCanceled, true);
436 fEventHandled = TRUE;
437 break;
438 default:
439 break;
440 /** @todo Add other events here. */
441 }
442
443 return fEventHandled;
444}
445#else /* !RT_OS_WINDOWS */
446/**
447 * Signal handler that sets g_fGuestCtrlCanceled.
448 *
449 * This can be executed on any thread in the process, on Windows it may even be
450 * a thread dedicated to delivering this signal. Don't do anything
451 * unnecessary here.
452 */
453static void gctlSignalHandler(int iSignal)
454{
455 NOREF(iSignal);
456 ASMAtomicWriteBool(&g_fGuestCtrlCanceled, true);
457}
458#endif
459
460
461/**
462 * Installs a custom signal handler to get notified
463 * whenever the user wants to intercept the program.
464 *
465 * @todo Make this handler available for all VBoxManage modules?
466 */
467static int gctlSignalHandlerInstall(void)
468{
469 g_fGuestCtrlCanceled = false;
470
471 int rc = VINF_SUCCESS;
472#ifdef RT_OS_WINDOWS
473 if (!SetConsoleCtrlHandler((PHANDLER_ROUTINE)gctlSignalHandler, TRUE /* Add handler */))
474 {
475 rc = RTErrConvertFromWin32(GetLastError());
476 RTMsgError("Unable to install console control handler, rc=%Rrc\n", rc);
477 }
478#else
479 signal(SIGINT, gctlSignalHandler);
480 signal(SIGTERM, gctlSignalHandler);
481# ifdef SIGBREAK
482 signal(SIGBREAK, gctlSignalHandler);
483# endif
484#endif
485 return rc;
486}
487
488
489/**
490 * Uninstalls a previously installed signal handler.
491 */
492static int gctlSignalHandlerUninstall(void)
493{
494 int rc = VINF_SUCCESS;
495#ifdef RT_OS_WINDOWS
496 if (!SetConsoleCtrlHandler((PHANDLER_ROUTINE)NULL, FALSE /* Remove handler */))
497 {
498 rc = RTErrConvertFromWin32(GetLastError());
499 RTMsgError("Unable to uninstall console control handler, rc=%Rrc\n", rc);
500 }
501#else
502 signal(SIGINT, SIG_DFL);
503 signal(SIGTERM, SIG_DFL);
504# ifdef SIGBREAK
505 signal(SIGBREAK, SIG_DFL);
506# endif
507#endif
508 return rc;
509}
510
511
512/**
513 * Translates a process status to a human readable string.
514 */
515const char *gctlProcessStatusToText(ProcessStatus_T enmStatus)
516{
517 switch (enmStatus)
518 {
519 case ProcessStatus_Starting:
520 return "starting";
521 case ProcessStatus_Started:
522 return "started";
523 case ProcessStatus_Paused:
524 return "paused";
525 case ProcessStatus_Terminating:
526 return "terminating";
527 case ProcessStatus_TerminatedNormally:
528 return "successfully terminated";
529 case ProcessStatus_TerminatedSignal:
530 return "terminated by signal";
531 case ProcessStatus_TerminatedAbnormally:
532 return "abnormally aborted";
533 case ProcessStatus_TimedOutKilled:
534 return "timed out";
535 case ProcessStatus_TimedOutAbnormally:
536 return "timed out, hanging";
537 case ProcessStatus_Down:
538 return "killed";
539 case ProcessStatus_Error:
540 return "error";
541 default:
542 break;
543 }
544 return "unknown";
545}
546
547/**
548 * Translates a guest process wait result to a human readable string.
549 */
550const char *gctlProcessWaitResultToText(ProcessWaitResult_T enmWaitResult)
551{
552 switch (enmWaitResult)
553 {
554 case ProcessWaitResult_Start:
555 return "started";
556 case ProcessWaitResult_Terminate:
557 return "terminated";
558 case ProcessWaitResult_Status:
559 return "status changed";
560 case ProcessWaitResult_Error:
561 return "error";
562 case ProcessWaitResult_Timeout:
563 return "timed out";
564 case ProcessWaitResult_StdIn:
565 return "stdin ready";
566 case ProcessWaitResult_StdOut:
567 return "data on stdout";
568 case ProcessWaitResult_StdErr:
569 return "data on stderr";
570 case ProcessWaitResult_WaitFlagNotSupported:
571 return "waiting flag not supported";
572 default:
573 break;
574 }
575 return "unknown";
576}
577
578/**
579 * Translates a guest session status to a human readable string.
580 */
581const char *gctlGuestSessionStatusToText(GuestSessionStatus_T enmStatus)
582{
583 switch (enmStatus)
584 {
585 case GuestSessionStatus_Starting:
586 return "starting";
587 case GuestSessionStatus_Started:
588 return "started";
589 case GuestSessionStatus_Terminating:
590 return "terminating";
591 case GuestSessionStatus_Terminated:
592 return "terminated";
593 case GuestSessionStatus_TimedOutKilled:
594 return "timed out";
595 case GuestSessionStatus_TimedOutAbnormally:
596 return "timed out, hanging";
597 case GuestSessionStatus_Down:
598 return "killed";
599 case GuestSessionStatus_Error:
600 return "error";
601 default:
602 break;
603 }
604 return "unknown";
605}
606
607/**
608 * Translates a guest file status to a human readable string.
609 */
610const char *gctlFileStatusToText(FileStatus_T enmStatus)
611{
612 switch (enmStatus)
613 {
614 case FileStatus_Opening:
615 return "opening";
616 case FileStatus_Open:
617 return "open";
618 case FileStatus_Closing:
619 return "closing";
620 case FileStatus_Closed:
621 return "closed";
622 case FileStatus_Down:
623 return "killed";
624 case FileStatus_Error:
625 return "error";
626 default:
627 break;
628 }
629 return "unknown";
630}
631
632static int gctlPrintError(com::ErrorInfo &errorInfo)
633{
634 if ( errorInfo.isFullAvailable()
635 || errorInfo.isBasicAvailable())
636 {
637 /* If we got a VBOX_E_IPRT error we handle the error in a more gentle way
638 * because it contains more accurate info about what went wrong. */
639 if (errorInfo.getResultCode() == VBOX_E_IPRT_ERROR)
640 RTMsgError("%ls.", errorInfo.getText().raw());
641 else
642 {
643 RTMsgError("Error details:");
644 GluePrintErrorInfo(errorInfo);
645 }
646 return VERR_GENERAL_FAILURE; /** @todo */
647 }
648 AssertMsgFailedReturn(("Object has indicated no error (%Rhrc)!?\n", errorInfo.getResultCode()),
649 VERR_INVALID_PARAMETER);
650}
651
652static int gctlPrintError(IUnknown *pObj, const GUID &aIID)
653{
654 com::ErrorInfo ErrInfo(pObj, aIID);
655 return gctlPrintError(ErrInfo);
656}
657
658static int gctlPrintProgressError(ComPtr<IProgress> pProgress)
659{
660 int vrc = VINF_SUCCESS;
661 HRESULT rc;
662
663 do
664 {
665 BOOL fCanceled;
666 CHECK_ERROR_BREAK(pProgress, COMGETTER(Canceled)(&fCanceled));
667 if (!fCanceled)
668 {
669 LONG rcProc;
670 CHECK_ERROR_BREAK(pProgress, COMGETTER(ResultCode)(&rcProc));
671 if (FAILED(rcProc))
672 {
673 com::ProgressErrorInfo ErrInfo(pProgress);
674 vrc = gctlPrintError(ErrInfo);
675 }
676 }
677
678 } while(0);
679
680 AssertMsgStmt(SUCCEEDED(rc), ("Could not lookup progress information\n"), vrc = VERR_COM_UNEXPECTED);
681
682 return vrc;
683}
684
685
686
687/*
688 *
689 *
690 * Guest Control Command Context
691 * Guest Control Command Context
692 * Guest Control Command Context
693 * Guest Control Command Context
694 *
695 *
696 *
697 */
698
699
700/**
701 * Initializes a guest control command context structure.
702 *
703 * @returns RTEXITCODE_SUCCESS on success, RTEXITCODE_FAILURE on failure (after
704 * informing the user of course).
705 * @param pCtx The command context to init.
706 * @param pArg The handle argument package.
707 */
708static RTEXITCODE gctrCmdCtxInit(PGCTLCMDCTX pCtx, HandlerArg *pArg)
709{
710 RT_ZERO(*pCtx);
711 pCtx->pArg = pArg;
712
713 /*
714 * The user name defaults to the host one, if we can get at it.
715 */
716 char szUser[1024];
717 int rc = RTProcQueryUsername(RTProcSelf(), szUser, sizeof(szUser), NULL);
718 if ( RT_SUCCESS(rc)
719 && RTStrIsValidEncoding(szUser)) /* paranoia required on posix */
720 {
721 try
722 {
723 pCtx->strUsername = szUser;
724 }
725 catch (std::bad_alloc &)
726 {
727 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Out of memory");
728 }
729 }
730 /* else: ignore this failure. */
731
732 return RTEXITCODE_SUCCESS;
733}
734
735
736/**
737 * Worker for GCTLCMD_COMMON_OPTION_CASES.
738 *
739 * @returns RTEXITCODE_SUCCESS if the option was handled successfully. If not,
740 * an error message is printed and an appropriate failure exit code is
741 * returned.
742 * @param pCtx The guest control command context.
743 * @param ch The option char or ordinal.
744 * @param pValueUnion The option value union.
745 */
746static RTEXITCODE gctlCtxSetOption(PGCTLCMDCTX pCtx, int ch, PRTGETOPTUNION pValueUnion)
747{
748 RTEXITCODE rcExit = RTEXITCODE_SUCCESS;
749 switch (ch)
750 {
751 case GCTLCMD_COMMON_OPT_USER: /* User name */
752 if (!pCtx->pCmdDef || !(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS))
753 pCtx->strUsername = pValueUnion->psz;
754 else
755 RTMsgWarning("The --username|-u option is ignored by '%s'", pCtx->pCmdDef->pszName);
756 break;
757
758 case GCTLCMD_COMMON_OPT_PASSWORD: /* Password */
759 if (!pCtx->pCmdDef || !(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS))
760 {
761 if (pCtx->strPassword.isNotEmpty())
762 RTMsgWarning("Password is given more than once.");
763 pCtx->strPassword = pValueUnion->psz;
764 }
765 else
766 RTMsgWarning("The --password option is ignored by '%s'", pCtx->pCmdDef->pszName);
767 break;
768
769 case GCTLCMD_COMMON_OPT_PASSWORD_FILE: /* Password file */
770 if (!pCtx->pCmdDef || !(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS))
771 rcExit = readPasswordFile(pValueUnion->psz, &pCtx->strPassword);
772 else
773 RTMsgWarning("The --password-file|-p option is ignored by '%s'", pCtx->pCmdDef->pszName);
774 break;
775
776 case GCTLCMD_COMMON_OPT_DOMAIN: /* domain */
777 if (!pCtx->pCmdDef || !(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS))
778 pCtx->strDomain = pValueUnion->psz;
779 else
780 RTMsgWarning("The --domain option is ignored by '%s'", pCtx->pCmdDef->pszName);
781 break;
782
783 case 'v': /* --verbose */
784 pCtx->cVerbose++;
785 break;
786
787 case 'q': /* --quiet */
788 if (pCtx->cVerbose)
789 pCtx->cVerbose--;
790 break;
791
792 default:
793 AssertFatalMsgFailed(("ch=%d (%c)\n", ch, ch));
794 }
795 return rcExit;
796}
797
798
799/**
800 * Initializes the VM for IGuest operation.
801 *
802 * This opens a shared session to a running VM and gets hold of IGuest.
803 *
804 * @returns RTEXITCODE_SUCCESS on success. RTEXITCODE_FAILURE and user message
805 * on failure.
806 * @param pCtx The guest control command context.
807 * GCTLCMDCTX::pGuest will be set on success.
808 */
809static RTEXITCODE gctlCtxInitVmSession(PGCTLCMDCTX pCtx)
810{
811 HRESULT rc;
812 AssertPtr(pCtx);
813 AssertPtr(pCtx->pArg);
814
815 /*
816 * Find the VM and check if it's running.
817 */
818 ComPtr<IMachine> machine;
819 CHECK_ERROR(pCtx->pArg->virtualBox, FindMachine(Bstr(pCtx->pszVmNameOrUuid).raw(), machine.asOutParam()));
820 if (SUCCEEDED(rc))
821 {
822 MachineState_T enmMachineState;
823 CHECK_ERROR(machine, COMGETTER(State)(&enmMachineState));
824 if ( SUCCEEDED(rc)
825 && enmMachineState == MachineState_Running)
826 {
827 /*
828 * It's running. So, open a session to it and get the IGuest interface.
829 */
830 CHECK_ERROR(machine, LockMachine(pCtx->pArg->session, LockType_Shared));
831 if (SUCCEEDED(rc))
832 {
833 pCtx->fLockedVmSession = true;
834 ComPtr<IConsole> ptrConsole;
835 CHECK_ERROR(pCtx->pArg->session, COMGETTER(Console)(ptrConsole.asOutParam()));
836 if (SUCCEEDED(rc))
837 {
838 if (ptrConsole.isNotNull())
839 {
840 CHECK_ERROR(ptrConsole, COMGETTER(Guest)(pCtx->pGuest.asOutParam()));
841 if (SUCCEEDED(rc))
842 return RTEXITCODE_SUCCESS;
843 }
844 else
845 RTMsgError("Failed to get a IConsole pointer for the machine. Is it still running?\n");
846 }
847 }
848 }
849 else if (SUCCEEDED(rc))
850 RTMsgError("Machine \"%s\" is not running (currently %s)!\n",
851 pCtx->pszVmNameOrUuid, machineStateToName(enmMachineState, false));
852 }
853 return RTEXITCODE_FAILURE;
854}
855
856
857/**
858 * Creates a guest session with the VM.
859 *
860 * @retval RTEXITCODE_SUCCESS on success.
861 * @retval RTEXITCODE_FAILURE and user message on failure.
862 * @param pCtx The guest control command context.
863 * GCTCMDCTX::pGuestSession and GCTLCMDCTX::uSessionID
864 * will be set.
865 */
866static RTEXITCODE gctlCtxInitGuestSession(PGCTLCMDCTX pCtx)
867{
868 HRESULT rc;
869 AssertPtr(pCtx);
870 Assert(!(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS));
871 Assert(pCtx->pGuest.isNotNull());
872
873 /*
874 * Build up a reasonable guest session name. Useful for identifying
875 * a specific session when listing / searching for them.
876 */
877 char *pszSessionName;
878 if (RTStrAPrintf(&pszSessionName,
879 "[%RU32] VBoxManage Guest Control [%s] - %s",
880 RTProcSelf(), pCtx->pszVmNameOrUuid, pCtx->pCmdDef->pszName) < 0)
881 return RTMsgErrorExit(RTEXITCODE_FAILURE, "No enough memory for session name");
882
883 /*
884 * Create a guest session.
885 */
886 if (pCtx->cVerbose)
887 RTPrintf("Creating guest session as user '%s'...\n", pCtx->strUsername.c_str());
888 try
889 {
890 CHECK_ERROR(pCtx->pGuest, CreateSession(Bstr(pCtx->strUsername).raw(),
891 Bstr(pCtx->strPassword).raw(),
892 Bstr(pCtx->strDomain).raw(),
893 Bstr(pszSessionName).raw(),
894 pCtx->pGuestSession.asOutParam()));
895 }
896 catch (std::bad_alloc &)
897 {
898 RTMsgError("Out of memory setting up IGuest::CreateSession call");
899 rc = E_OUTOFMEMORY;
900 }
901 if (SUCCEEDED(rc))
902 {
903 /*
904 * Wait for guest session to start.
905 */
906 if (pCtx->cVerbose)
907 RTPrintf("Waiting for guest session to start...\n");
908 GuestSessionWaitResult_T enmWaitResult = GuestSessionWaitResult_None; /* Shut up MSC */
909 try
910 {
911 com::SafeArray<GuestSessionWaitForFlag_T> aSessionWaitFlags;
912 aSessionWaitFlags.push_back(GuestSessionWaitForFlag_Start);
913 CHECK_ERROR(pCtx->pGuestSession, WaitForArray(ComSafeArrayAsInParam(aSessionWaitFlags),
914 /** @todo Make session handling timeouts configurable. */
915 30 * 1000, &enmWaitResult));
916 }
917 catch (std::bad_alloc &)
918 {
919 RTMsgError("Out of memory setting up IGuestSession::WaitForArray call");
920 rc = E_OUTOFMEMORY;
921 }
922 if (SUCCEEDED(rc))
923 {
924 /* The WaitFlagNotSupported result may happen with GAs older than 4.3. */
925 if ( enmWaitResult == GuestSessionWaitResult_Start
926 || enmWaitResult == GuestSessionWaitResult_WaitFlagNotSupported)
927 {
928 /*
929 * Get the session ID and we're ready to rumble.
930 */
931 CHECK_ERROR(pCtx->pGuestSession, COMGETTER(Id)(&pCtx->uSessionID));
932 if (SUCCEEDED(rc))
933 {
934 if (pCtx->cVerbose)
935 RTPrintf("Successfully started guest session (ID %RU32)\n", pCtx->uSessionID);
936 RTStrFree(pszSessionName);
937 return RTEXITCODE_SUCCESS;
938 }
939 }
940 else
941 {
942 GuestSessionStatus_T enmSessionStatus;
943 CHECK_ERROR(pCtx->pGuestSession, COMGETTER(Status)(&enmSessionStatus));
944 RTMsgError("Error starting guest session (current status is: %s)\n",
945 SUCCEEDED(rc) ? gctlGuestSessionStatusToText(enmSessionStatus) : "<unknown>");
946 }
947 }
948 }
949
950 RTStrFree(pszSessionName);
951 return RTEXITCODE_FAILURE;
952}
953
954
955/**
956 * Completes the guest control context initialization after parsing arguments.
957 *
958 * Will validate common arguments, open a VM session, and if requested open a
959 * guest session and install the CTRL-C signal handler.
960 *
961 * It is good to validate all the options and arguments you can before making
962 * this call. However, the VM session, IGuest and IGuestSession interfaces are
963 * not availabe till after this call, so take care.
964 *
965 * @retval RTEXITCODE_SUCCESS on success.
966 * @retval RTEXITCODE_FAILURE and user message on failure.
967 * @param pCtx The guest control command context.
968 * GCTCMDCTX::pGuestSession and GCTLCMDCTX::uSessionID
969 * will be set.
970 * @remarks Can safely be called multiple times, will only do work once.
971 */
972static RTEXITCODE gctlCtxPostOptionParsingInit(PGCTLCMDCTX pCtx)
973{
974 if (pCtx->fPostOptionParsingInited)
975 return RTEXITCODE_SUCCESS;
976
977 /*
978 * Check that the user name isn't empty when we need it.
979 */
980 RTEXITCODE rcExit;
981 if ( (pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS)
982 || pCtx->strUsername.isNotEmpty())
983 {
984 /*
985 * Open the VM session and if required, a guest session.
986 */
987 rcExit = gctlCtxInitVmSession(pCtx);
988 if ( rcExit == RTEXITCODE_SUCCESS
989 && !(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS))
990 rcExit = gctlCtxInitGuestSession(pCtx);
991 if (rcExit == RTEXITCODE_SUCCESS)
992 {
993 /*
994 * Install signal handler if requested (errors are ignored).
995 */
996 if (!(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_NO_SIGNAL_HANDLER))
997 {
998 int rc = gctlSignalHandlerInstall();
999 pCtx->fInstalledSignalHandler = RT_SUCCESS(rc);
1000 }
1001 }
1002 }
1003 else
1004 rcExit = errorSyntaxEx(USAGE_GUESTCONTROL, pCtx->pCmdDef->fCmdUsage, "No user name specified!");
1005
1006 pCtx->fPostOptionParsingInited = rcExit == RTEXITCODE_SUCCESS;
1007 return rcExit;
1008}
1009
1010
1011/**
1012 * Cleans up the context when the command returns.
1013 *
1014 * This will close any open guest session, unless the DETACH flag is set.
1015 * It will also close any VM session that may be been established. Any signal
1016 * handlers we've installed will also be removed.
1017 *
1018 * Un-initializes the VM after guest control usage.
1019 * @param pCmdCtx Pointer to command context.
1020 */
1021static void gctlCtxTerm(PGCTLCMDCTX pCtx)
1022{
1023 HRESULT rc;
1024 AssertPtr(pCtx);
1025
1026 /*
1027 * Uninstall signal handler.
1028 */
1029 if (pCtx->fInstalledSignalHandler)
1030 {
1031 gctlSignalHandlerUninstall();
1032 pCtx->fInstalledSignalHandler = false;
1033 }
1034
1035 /*
1036 * Close, or at least release, the guest session.
1037 */
1038 if (pCtx->pGuestSession.isNotNull())
1039 {
1040 if ( !(pCtx->pCmdDef->fCmdCtx & GCTLCMDCTX_F_SESSION_ANONYMOUS)
1041 && !pCtx->fDetachGuestSession)
1042 {
1043 if (pCtx->cVerbose)
1044 RTPrintf("Closing guest session ...\n");
1045
1046 CHECK_ERROR(pCtx->pGuestSession, Close());
1047 }
1048 else if ( pCtx->fDetachGuestSession
1049 && pCtx->cVerbose)
1050 RTPrintf("Guest session detached\n");
1051
1052 pCtx->pGuestSession.setNull();
1053 }
1054
1055 /*
1056 * Close the VM session.
1057 */
1058 if (pCtx->fLockedVmSession)
1059 {
1060 Assert(pCtx->pArg->session.isNotNull());
1061 CHECK_ERROR(pCtx->pArg->session, UnlockMachine());
1062 pCtx->fLockedVmSession = false;
1063 }
1064}
1065
1066
1067
1068
1069
1070/*
1071 *
1072 *
1073 * Guest Control Command Handling.
1074 * Guest Control Command Handling.
1075 * Guest Control Command Handling.
1076 * Guest Control Command Handling.
1077 * Guest Control Command Handling.
1078 *
1079 *
1080 */
1081
1082
1083/** @name EXITCODEEXEC_XXX - Special run exit codes.
1084 *
1085 * Special exit codes for returning errors/information of a started guest
1086 * process to the command line VBoxManage was started from. Useful for e.g.
1087 * scripting.
1088 *
1089 * ASSUMING that all platforms have at least 7-bits for the exit code we can do
1090 * the following mapping:
1091 * - Guest exit code 0 is mapped to 0 on the host.
1092 * - Guest exit codes 1 thru 93 (0x5d) are displaced by 32, so that 1
1093 * becomes 33 (0x21) on the host and 93 becomes 125 (0x7d) on the host.
1094 * - Guest exit codes 94 (0x5e) and above are mapped to 126 (0x5e).
1095 *
1096 * We ASSUME that all VBoxManage status codes are in the range 0 thru 32.
1097 *
1098 * @note These are frozen as of 4.1.0.
1099 * @note The guest exit code mappings was introduced with 5.0 and the 'run'
1100 * command, they are/was not supported by 'exec'.
1101 * @sa gctlRunCalculateExitCode
1102 */
1103/** Process exited normally but with an exit code <> 0. */
1104#define EXITCODEEXEC_CODE ((RTEXITCODE)16)
1105#define EXITCODEEXEC_FAILED ((RTEXITCODE)17)
1106#define EXITCODEEXEC_TERM_SIGNAL ((RTEXITCODE)18)
1107#define EXITCODEEXEC_TERM_ABEND ((RTEXITCODE)19)
1108#define EXITCODEEXEC_TIMEOUT ((RTEXITCODE)20)
1109#define EXITCODEEXEC_DOWN ((RTEXITCODE)21)
1110/** Execution was interrupt by user (ctrl-c). */
1111#define EXITCODEEXEC_CANCELED ((RTEXITCODE)22)
1112/** The first mapped guest (non-zero) exit code. */
1113#define EXITCODEEXEC_MAPPED_FIRST 33
1114/** The last mapped guest (non-zero) exit code value (inclusive). */
1115#define EXITCODEEXEC_MAPPED_LAST 125
1116/** The number of exit codes from EXITCODEEXEC_MAPPED_FIRST to
1117 * EXITCODEEXEC_MAPPED_LAST. This is also the highest guest exit code number
1118 * we're able to map. */
1119#define EXITCODEEXEC_MAPPED_RANGE (93)
1120/** The guest exit code displacement value. */
1121#define EXITCODEEXEC_MAPPED_DISPLACEMENT 32
1122/** The guest exit code was too big to be mapped. */
1123#define EXITCODEEXEC_MAPPED_BIG ((RTEXITCODE)126)
1124/** @} */
1125
1126/**
1127 * Calculates the exit code of VBoxManage.
1128 *
1129 * @returns The exit code to return.
1130 * @param enmStatus The guest process status.
1131 * @param uExitCode The associated guest process exit code (where
1132 * applicable).
1133 * @param fReturnExitCodes Set if we're to use the 32-126 range for guest
1134 * exit codes.
1135 */
1136static RTEXITCODE gctlRunCalculateExitCode(ProcessStatus_T enmStatus, ULONG uExitCode, bool fReturnExitCodes)
1137{
1138 switch (enmStatus)
1139 {
1140 case ProcessStatus_TerminatedNormally:
1141 if (uExitCode == 0)
1142 return RTEXITCODE_SUCCESS;
1143 if (!fReturnExitCodes)
1144 return EXITCODEEXEC_CODE;
1145 if (uExitCode <= EXITCODEEXEC_MAPPED_RANGE)
1146 return (RTEXITCODE) (uExitCode + EXITCODEEXEC_MAPPED_DISPLACEMENT);
1147 return EXITCODEEXEC_MAPPED_BIG;
1148
1149 case ProcessStatus_TerminatedAbnormally:
1150 return EXITCODEEXEC_TERM_ABEND;
1151 case ProcessStatus_TerminatedSignal:
1152 return EXITCODEEXEC_TERM_SIGNAL;
1153
1154#if 0 /* see caller! */
1155 case ProcessStatus_TimedOutKilled:
1156 return EXITCODEEXEC_TIMEOUT;
1157 case ProcessStatus_Down:
1158 return EXITCODEEXEC_DOWN; /* Service/OS is stopping, process was killed. */
1159 case ProcessStatus_Error:
1160 return EXITCODEEXEC_FAILED;
1161
1162 /* The following is probably for detached? */
1163 case ProcessStatus_Starting:
1164 return RTEXITCODE_SUCCESS;
1165 case ProcessStatus_Started:
1166 return RTEXITCODE_SUCCESS;
1167 case ProcessStatus_Paused:
1168 return RTEXITCODE_SUCCESS;
1169 case ProcessStatus_Terminating:
1170 return RTEXITCODE_SUCCESS; /** @todo ???? */
1171#endif
1172
1173 default:
1174 AssertMsgFailed(("Unknown exit status (%u/%u) from guest process returned!\n", enmStatus, uExitCode));
1175 return RTEXITCODE_FAILURE;
1176 }
1177}
1178
1179
1180/**
1181 * Pumps guest output to the host.
1182 *
1183 * @return IPRT status code.
1184 * @param pProcess Pointer to appropriate process object.
1185 * @param hVfsIosDst Where to write the data.
1186 * @param uHandle Handle where to read the data from.
1187 * @param cMsTimeout Timeout (in ms) to wait for the operation to
1188 * complete.
1189 */
1190static int gctlRunPumpOutput(IProcess *pProcess, RTVFSIOSTREAM hVfsIosDst, ULONG uHandle, RTMSINTERVAL cMsTimeout)
1191{
1192 AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
1193 Assert(hVfsIosDst != NIL_RTVFSIOSTREAM);
1194
1195 int vrc;
1196
1197 SafeArray<BYTE> aOutputData;
1198 HRESULT hrc = pProcess->Read(uHandle, _64K, RT_MAX(cMsTimeout, 1), ComSafeArrayAsOutParam(aOutputData));
1199 if (SUCCEEDED(hrc))
1200 {
1201 size_t cbOutputData = aOutputData.size();
1202 if (cbOutputData == 0)
1203 vrc = VINF_SUCCESS;
1204 else
1205 {
1206 BYTE const *pbBuf = aOutputData.raw();
1207 AssertPtr(pbBuf);
1208
1209 vrc = RTVfsIoStrmWrite(hVfsIosDst, pbBuf, cbOutputData, true /*fBlocking*/, NULL);
1210 if (RT_FAILURE(vrc))
1211 RTMsgError("Unable to write output, rc=%Rrc\n", vrc);
1212 }
1213 }
1214 else
1215 vrc = gctlPrintError(pProcess, COM_IIDOF(IProcess));
1216 return vrc;
1217}
1218
1219
1220/**
1221 * Configures a host handle for pumping guest bits.
1222 *
1223 * @returns true if enabled and we successfully configured it.
1224 * @param fEnabled Whether pumping this pipe is configured.
1225 * @param enmHandle The IPRT standard handle designation.
1226 * @param pszName The name for user messages.
1227 * @param enmTransformation The transformation to apply.
1228 * @param phVfsIos Where to return the resulting I/O stream handle.
1229 */
1230static bool gctlRunSetupHandle(bool fEnabled, RTHANDLESTD enmHandle, const char *pszName,
1231 kStreamTransform enmTransformation, PRTVFSIOSTREAM phVfsIos)
1232{
1233 if (fEnabled)
1234 {
1235 int vrc = RTVfsIoStrmFromStdHandle(enmHandle, 0, true /*fLeaveOpen*/, phVfsIos);
1236 if (RT_SUCCESS(vrc))
1237 {
1238 if (enmTransformation != kStreamTransform_None)
1239 {
1240 RTMsgWarning("Unsupported %s line ending conversion", pszName);
1241 /** @todo Implement dos2unix and unix2dos stream filters. */
1242 }
1243 return true;
1244 }
1245 RTMsgWarning("Error getting %s handle: %Rrc", pszName, vrc);
1246 }
1247 return false;
1248}
1249
1250
1251/**
1252 * Returns the remaining time (in ms) based on the start time and a set
1253 * timeout value. Returns RT_INDEFINITE_WAIT if no timeout was specified.
1254 *
1255 * @return RTMSINTERVAL Time left (in ms).
1256 * @param u64StartMs Start time (in ms).
1257 * @param cMsTimeout Timeout value (in ms).
1258 */
1259static RTMSINTERVAL gctlRunGetRemainingTime(uint64_t u64StartMs, RTMSINTERVAL cMsTimeout)
1260{
1261 if (!cMsTimeout || cMsTimeout == RT_INDEFINITE_WAIT) /* If no timeout specified, wait forever. */
1262 return RT_INDEFINITE_WAIT;
1263
1264 uint64_t u64ElapsedMs = RTTimeMilliTS() - u64StartMs;
1265 if (u64ElapsedMs >= cMsTimeout)
1266 return 0;
1267
1268 return cMsTimeout - (RTMSINTERVAL)u64ElapsedMs;
1269}
1270
1271/**
1272 * Common handler for the 'run' and 'start' commands.
1273 *
1274 * @returns Command exit code.
1275 * @param pCtx Guest session context.
1276 * @param argc The argument count.
1277 * @param argv The argument vector for this command.
1278 * @param fRunCmd Set if it's 'run' clear if 'start'.
1279 * @param fHelp The help flag for the command.
1280 */
1281static RTEXITCODE gctlHandleRunCommon(PGCTLCMDCTX pCtx, int argc, char **argv, bool fRunCmd, uint32_t fHelp)
1282{
1283 RT_NOREF(fHelp);
1284 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
1285
1286 /*
1287 * Parse arguments.
1288 */
1289 enum kGstCtrlRunOpt
1290 {
1291 kGstCtrlRunOpt_IgnoreOrphanedProcesses = 1000,
1292 kGstCtrlRunOpt_NoProfile, /** @todo Deprecated and will be removed soon; use kGstCtrlRunOpt_Profile instead, if needed. */
1293 kGstCtrlRunOpt_Profile,
1294 kGstCtrlRunOpt_Dos2Unix,
1295 kGstCtrlRunOpt_Unix2Dos,
1296 kGstCtrlRunOpt_WaitForStdOut,
1297 kGstCtrlRunOpt_NoWaitForStdOut,
1298 kGstCtrlRunOpt_WaitForStdErr,
1299 kGstCtrlRunOpt_NoWaitForStdErr
1300 };
1301 static const RTGETOPTDEF s_aOptions[] =
1302 {
1303 GCTLCMD_COMMON_OPTION_DEFS()
1304 { "--putenv", 'E', RTGETOPT_REQ_STRING },
1305 { "--exe", 'e', RTGETOPT_REQ_STRING },
1306 { "--timeout", 't', RTGETOPT_REQ_UINT32 },
1307 { "--unquoted-args", 'u', RTGETOPT_REQ_NOTHING },
1308 { "--ignore-operhaned-processes", kGstCtrlRunOpt_IgnoreOrphanedProcesses, RTGETOPT_REQ_NOTHING },
1309 { "--no-profile", kGstCtrlRunOpt_NoProfile, RTGETOPT_REQ_NOTHING }, /** @todo Deprecated. */
1310 { "--profile", kGstCtrlRunOpt_Profile, RTGETOPT_REQ_NOTHING },
1311 /* run only: 6 - options */
1312 { "--dos2unix", kGstCtrlRunOpt_Dos2Unix, RTGETOPT_REQ_NOTHING },
1313 { "--unix2dos", kGstCtrlRunOpt_Unix2Dos, RTGETOPT_REQ_NOTHING },
1314 { "--no-wait-stdout", kGstCtrlRunOpt_NoWaitForStdOut, RTGETOPT_REQ_NOTHING },
1315 { "--wait-stdout", kGstCtrlRunOpt_WaitForStdOut, RTGETOPT_REQ_NOTHING },
1316 { "--no-wait-stderr", kGstCtrlRunOpt_NoWaitForStdErr, RTGETOPT_REQ_NOTHING },
1317 { "--wait-stderr", kGstCtrlRunOpt_WaitForStdErr, RTGETOPT_REQ_NOTHING },
1318 };
1319
1320 /** @todo stdin handling. */
1321
1322 int ch;
1323 RTGETOPTUNION ValueUnion;
1324 RTGETOPTSTATE GetState;
1325 int vrc = RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions) - (fRunCmd ? 0 : 6),
1326 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
1327 AssertRC(vrc);
1328
1329 com::SafeArray<ProcessCreateFlag_T> aCreateFlags;
1330 com::SafeArray<ProcessWaitForFlag_T> aWaitFlags;
1331 com::SafeArray<IN_BSTR> aArgs;
1332 com::SafeArray<IN_BSTR> aEnv;
1333 const char * pszImage = NULL;
1334 bool fWaitForStdOut = fRunCmd;
1335 bool fWaitForStdErr = fRunCmd;
1336 RTVFSIOSTREAM hVfsStdOut = NIL_RTVFSIOSTREAM;
1337 RTVFSIOSTREAM hVfsStdErr = NIL_RTVFSIOSTREAM;
1338 enum kStreamTransform enmStdOutTransform = kStreamTransform_None;
1339 enum kStreamTransform enmStdErrTransform = kStreamTransform_None;
1340 RTMSINTERVAL cMsTimeout = 0;
1341
1342 try
1343 {
1344 /* Wait for process start in any case. This is useful for scripting VBoxManage
1345 * when relying on its overall exit code. */
1346 aWaitFlags.push_back(ProcessWaitForFlag_Start);
1347
1348 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
1349 {
1350 /* For options that require an argument, ValueUnion has received the value. */
1351 switch (ch)
1352 {
1353 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
1354
1355 case 'E':
1356 if ( ValueUnion.psz[0] == '\0'
1357 || ValueUnion.psz[0] == '=')
1358 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_RUN,
1359 "Invalid argument variable[=value]: '%s'", ValueUnion.psz);
1360 aEnv.push_back(Bstr(ValueUnion.psz).raw());
1361 break;
1362
1363 case kGstCtrlRunOpt_IgnoreOrphanedProcesses:
1364 aCreateFlags.push_back(ProcessCreateFlag_IgnoreOrphanedProcesses);
1365 break;
1366
1367 case kGstCtrlRunOpt_NoProfile:
1368 /** @todo Deprecated, will be removed. */
1369 RTPrintf("Warning: Deprecated option \"--no-profile\" specified\n");
1370 break;
1371
1372 case kGstCtrlRunOpt_Profile:
1373 aCreateFlags.push_back(ProcessCreateFlag_Profile);
1374 break;
1375
1376 case 'e':
1377 pszImage = ValueUnion.psz;
1378 break;
1379
1380 case 'u':
1381 aCreateFlags.push_back(ProcessCreateFlag_UnquotedArguments);
1382 break;
1383
1384 /** @todo Add a hidden flag. */
1385
1386 case 't': /* Timeout */
1387 cMsTimeout = ValueUnion.u32;
1388 break;
1389
1390 /* run only options: */
1391 case kGstCtrlRunOpt_Dos2Unix:
1392 Assert(fRunCmd);
1393 enmStdErrTransform = enmStdOutTransform = kStreamTransform_Dos2Unix;
1394 break;
1395 case kGstCtrlRunOpt_Unix2Dos:
1396 Assert(fRunCmd);
1397 enmStdErrTransform = enmStdOutTransform = kStreamTransform_Unix2Dos;
1398 break;
1399
1400 case kGstCtrlRunOpt_WaitForStdOut:
1401 Assert(fRunCmd);
1402 fWaitForStdOut = true;
1403 break;
1404 case kGstCtrlRunOpt_NoWaitForStdOut:
1405 Assert(fRunCmd);
1406 fWaitForStdOut = false;
1407 break;
1408
1409 case kGstCtrlRunOpt_WaitForStdErr:
1410 Assert(fRunCmd);
1411 fWaitForStdErr = true;
1412 break;
1413 case kGstCtrlRunOpt_NoWaitForStdErr:
1414 Assert(fRunCmd);
1415 fWaitForStdErr = false;
1416 break;
1417
1418 case VINF_GETOPT_NOT_OPTION:
1419 aArgs.push_back(Bstr(ValueUnion.psz).raw());
1420 if (!pszImage)
1421 {
1422 Assert(aArgs.size() == 1);
1423 pszImage = ValueUnion.psz;
1424 }
1425 break;
1426
1427 default:
1428 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_RUN, ch, &ValueUnion);
1429
1430 } /* switch */
1431 } /* while RTGetOpt */
1432
1433 /* Must have something to execute. */
1434 if (!pszImage || !*pszImage)
1435 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_RUN, "No executable specified!");
1436
1437 /*
1438 * Finalize process creation and wait flags and input/output streams.
1439 */
1440 if (!fRunCmd)
1441 {
1442 aCreateFlags.push_back(ProcessCreateFlag_WaitForProcessStartOnly);
1443 Assert(!fWaitForStdOut);
1444 Assert(!fWaitForStdErr);
1445 }
1446 else
1447 {
1448 aWaitFlags.push_back(ProcessWaitForFlag_Terminate);
1449 fWaitForStdOut = gctlRunSetupHandle(fWaitForStdOut, RTHANDLESTD_OUTPUT, "stdout", enmStdOutTransform, &hVfsStdOut);
1450 if (fWaitForStdOut)
1451 {
1452 aCreateFlags.push_back(ProcessCreateFlag_WaitForStdOut);
1453 aWaitFlags.push_back(ProcessWaitForFlag_StdOut);
1454 }
1455 fWaitForStdErr = gctlRunSetupHandle(fWaitForStdErr, RTHANDLESTD_ERROR, "stderr", enmStdErrTransform, &hVfsStdErr);
1456 if (fWaitForStdErr)
1457 {
1458 aCreateFlags.push_back(ProcessCreateFlag_WaitForStdErr);
1459 aWaitFlags.push_back(ProcessWaitForFlag_StdErr);
1460 }
1461 }
1462 }
1463 catch (std::bad_alloc &)
1464 {
1465 return RTMsgErrorExit(RTEXITCODE_FAILURE, "VERR_NO_MEMORY\n");
1466 }
1467
1468 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
1469 if (rcExit != RTEXITCODE_SUCCESS)
1470 return rcExit;
1471
1472 HRESULT rc;
1473
1474 try
1475 {
1476 do
1477 {
1478 /* Get current time stamp to later calculate rest of timeout left. */
1479 uint64_t msStart = RTTimeMilliTS();
1480
1481 /*
1482 * Create the process.
1483 */
1484 if (pCtx->cVerbose)
1485 {
1486 if (cMsTimeout == 0)
1487 RTPrintf("Starting guest process ...\n");
1488 else
1489 RTPrintf("Starting guest process (within %ums)\n", cMsTimeout);
1490 }
1491 ComPtr<IGuestProcess> pProcess;
1492 CHECK_ERROR_BREAK(pCtx->pGuestSession, ProcessCreate(Bstr(pszImage).raw(),
1493 ComSafeArrayAsInParam(aArgs),
1494 ComSafeArrayAsInParam(aEnv),
1495 ComSafeArrayAsInParam(aCreateFlags),
1496 gctlRunGetRemainingTime(msStart, cMsTimeout),
1497 pProcess.asOutParam()));
1498
1499 /*
1500 * Explicitly wait for the guest process to be in a started state.
1501 */
1502 com::SafeArray<ProcessWaitForFlag_T> aWaitStartFlags;
1503 aWaitStartFlags.push_back(ProcessWaitForFlag_Start);
1504 ProcessWaitResult_T waitResult;
1505 CHECK_ERROR_BREAK(pProcess, WaitForArray(ComSafeArrayAsInParam(aWaitStartFlags),
1506 gctlRunGetRemainingTime(msStart, cMsTimeout), &waitResult));
1507
1508 ULONG uPID = 0;
1509 CHECK_ERROR_BREAK(pProcess, COMGETTER(PID)(&uPID));
1510 if (fRunCmd && pCtx->cVerbose)
1511 RTPrintf("Process '%s' (PID %RU32) started\n", pszImage, uPID);
1512 else if (!fRunCmd && pCtx->cVerbose)
1513 {
1514 /* Just print plain PID to make it easier for scripts
1515 * invoking VBoxManage. */
1516 RTPrintf("[%RU32 - Session %RU32]\n", uPID, pCtx->uSessionID);
1517 }
1518
1519 /*
1520 * Wait for process to exit/start...
1521 */
1522 RTMSINTERVAL cMsTimeLeft = 1; /* Will be calculated. */
1523 bool fReadStdOut = false;
1524 bool fReadStdErr = false;
1525 bool fCompleted = false;
1526 bool fCompletedStartCmd = false;
1527
1528 vrc = VINF_SUCCESS;
1529 while ( !fCompleted
1530 && cMsTimeLeft > 0)
1531 {
1532 cMsTimeLeft = gctlRunGetRemainingTime(msStart, cMsTimeout);
1533 CHECK_ERROR_BREAK(pProcess, WaitForArray(ComSafeArrayAsInParam(aWaitFlags),
1534 RT_MIN(500 /*ms*/, RT_MAX(cMsTimeLeft, 1 /*ms*/)),
1535 &waitResult));
1536 switch (waitResult)
1537 {
1538 case ProcessWaitResult_Start:
1539 fCompletedStartCmd = fCompleted = !fRunCmd; /* Only wait for startup if the 'start' command. */
1540 break;
1541 case ProcessWaitResult_StdOut:
1542 fReadStdOut = true;
1543 break;
1544 case ProcessWaitResult_StdErr:
1545 fReadStdErr = true;
1546 break;
1547 case ProcessWaitResult_Terminate:
1548 if (pCtx->cVerbose)
1549 RTPrintf("Process terminated\n");
1550 /* Process terminated, we're done. */
1551 fCompleted = true;
1552 break;
1553 case ProcessWaitResult_WaitFlagNotSupported:
1554 /* The guest does not support waiting for stdout/err, so
1555 * yield to reduce the CPU load due to busy waiting. */
1556 RTThreadYield();
1557 fReadStdOut = fReadStdErr = true;
1558 break;
1559 case ProcessWaitResult_Timeout:
1560 {
1561 /** @todo It is really unclear whether we will get stuck with the timeout
1562 * result here if the guest side times out the process and fails to
1563 * kill the process... To be on the save side, double the IPC and
1564 * check the process status every time we time out. */
1565 ProcessStatus_T enmProcStatus;
1566 CHECK_ERROR_BREAK(pProcess, COMGETTER(Status)(&enmProcStatus));
1567 if ( enmProcStatus == ProcessStatus_TimedOutKilled
1568 || enmProcStatus == ProcessStatus_TimedOutAbnormally)
1569 fCompleted = true;
1570 fReadStdOut = fReadStdErr = true;
1571 break;
1572 }
1573 case ProcessWaitResult_Status:
1574 /* ignore. */
1575 break;
1576 case ProcessWaitResult_Error:
1577 /* waitFor is dead in the water, I think, so better leave the loop. */
1578 vrc = VERR_CALLBACK_RETURN;
1579 break;
1580
1581 case ProcessWaitResult_StdIn: AssertFailed(); /* did ask for this! */ break;
1582 case ProcessWaitResult_None: AssertFailed(); /* used. */ break;
1583 default: AssertFailed(); /* huh? */ break;
1584 }
1585
1586 if (g_fGuestCtrlCanceled)
1587 break;
1588
1589 /*
1590 * Pump output as needed.
1591 */
1592 if (fReadStdOut)
1593 {
1594 cMsTimeLeft = gctlRunGetRemainingTime(msStart, cMsTimeout);
1595 int vrc2 = gctlRunPumpOutput(pProcess, hVfsStdOut, 1 /* StdOut */, cMsTimeLeft);
1596 if (RT_FAILURE(vrc2) && RT_SUCCESS(vrc))
1597 vrc = vrc2;
1598 fReadStdOut = false;
1599 }
1600 if (fReadStdErr)
1601 {
1602 cMsTimeLeft = gctlRunGetRemainingTime(msStart, cMsTimeout);
1603 int vrc2 = gctlRunPumpOutput(pProcess, hVfsStdErr, 2 /* StdErr */, cMsTimeLeft);
1604 if (RT_FAILURE(vrc2) && RT_SUCCESS(vrc))
1605 vrc = vrc2;
1606 fReadStdErr = false;
1607 }
1608 if ( RT_FAILURE(vrc)
1609 || g_fGuestCtrlCanceled)
1610 break;
1611
1612 /*
1613 * Process events before looping.
1614 */
1615 NativeEventQueue::getMainEventQueue()->processEventQueue(0);
1616 } /* while */
1617
1618 /*
1619 * Report status back to the user.
1620 */
1621 if (g_fGuestCtrlCanceled)
1622 {
1623 if (pCtx->cVerbose)
1624 RTPrintf("Process execution aborted!\n");
1625 rcExit = EXITCODEEXEC_CANCELED;
1626 }
1627 else if (fCompletedStartCmd)
1628 {
1629 if (pCtx->cVerbose)
1630 RTPrintf("Process successfully started!\n");
1631 rcExit = RTEXITCODE_SUCCESS;
1632 }
1633 else if (fCompleted)
1634 {
1635 ProcessStatus_T procStatus;
1636 CHECK_ERROR_BREAK(pProcess, COMGETTER(Status)(&procStatus));
1637 if ( procStatus == ProcessStatus_TerminatedNormally
1638 || procStatus == ProcessStatus_TerminatedAbnormally
1639 || procStatus == ProcessStatus_TerminatedSignal)
1640 {
1641 LONG lExitCode;
1642 CHECK_ERROR_BREAK(pProcess, COMGETTER(ExitCode)(&lExitCode));
1643 if (pCtx->cVerbose)
1644 RTPrintf("Exit code=%u (Status=%u [%s])\n",
1645 lExitCode, procStatus, gctlProcessStatusToText(procStatus));
1646
1647 rcExit = gctlRunCalculateExitCode(procStatus, lExitCode, true /*fReturnExitCodes*/);
1648 }
1649 else if ( procStatus == ProcessStatus_TimedOutKilled
1650 || procStatus == ProcessStatus_TimedOutAbnormally)
1651 {
1652 if (pCtx->cVerbose)
1653 RTPrintf("Process timed out (guest side) and %s\n",
1654 procStatus == ProcessStatus_TimedOutAbnormally
1655 ? "failed to terminate so far" : "was terminated");
1656 rcExit = EXITCODEEXEC_TIMEOUT;
1657 }
1658 else
1659 {
1660 if (pCtx->cVerbose)
1661 RTPrintf("Process now is in status [%s] (unexpected)\n", gctlProcessStatusToText(procStatus));
1662 rcExit = RTEXITCODE_FAILURE;
1663 }
1664 }
1665 else if (RT_FAILURE_NP(vrc))
1666 {
1667 if (pCtx->cVerbose)
1668 RTPrintf("Process monitor loop quit with vrc=%Rrc\n", vrc);
1669 rcExit = RTEXITCODE_FAILURE;
1670 }
1671 else
1672 {
1673 if (pCtx->cVerbose)
1674 RTPrintf("Process monitor loop timed out\n");
1675 rcExit = EXITCODEEXEC_TIMEOUT;
1676 }
1677
1678 } while (0);
1679 }
1680 catch (std::bad_alloc)
1681 {
1682 rc = E_OUTOFMEMORY;
1683 }
1684
1685 /*
1686 * Decide what to do with the guest session.
1687 *
1688 * If it's the 'start' command where detach the guest process after
1689 * starting, don't close the guest session it is part of, except on
1690 * failure or ctrl-c.
1691 *
1692 * For the 'run' command the guest process quits with us.
1693 */
1694 if (!fRunCmd && SUCCEEDED(rc) && !g_fGuestCtrlCanceled)
1695 pCtx->fDetachGuestSession = true;
1696
1697 /* Make sure we return failure on failure. */
1698 if (FAILED(rc) && rcExit == RTEXITCODE_SUCCESS)
1699 rcExit = RTEXITCODE_FAILURE;
1700 return rcExit;
1701}
1702
1703
1704static DECLCALLBACK(RTEXITCODE) gctlHandleRun(PGCTLCMDCTX pCtx, int argc, char **argv)
1705{
1706 return gctlHandleRunCommon(pCtx, argc, argv, true /*fRunCmd*/, USAGE_GSTCTRL_RUN);
1707}
1708
1709
1710static DECLCALLBACK(RTEXITCODE) gctlHandleStart(PGCTLCMDCTX pCtx, int argc, char **argv)
1711{
1712 return gctlHandleRunCommon(pCtx, argc, argv, false /*fRunCmd*/, USAGE_GSTCTRL_START);
1713}
1714
1715
1716static RTEXITCODE gctlHandleCopy(PGCTLCMDCTX pCtx, int argc, char **argv, bool fHostToGuest)
1717{
1718 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
1719
1720 /** @todo r=bird: This command isn't very unix friendly in general. mkdir
1721 * is much better (partly because it is much simpler of course). The main
1722 * arguments against this is that (1) all but two options conflicts with
1723 * what 'man cp' tells me on a GNU/Linux system, (2) wildchar matching is
1724 * done windows CMD style (though not in a 100% compatible way), and (3)
1725 * that only one source is allowed - efficiently sabotaging default
1726 * wildcard expansion by a unix shell. The best solution here would be
1727 * two different variant, one windowsy (xcopy) and one unixy (gnu cp). */
1728
1729 /*
1730 * IGuest::CopyToGuest is kept as simple as possible to let the developer choose
1731 * what and how to implement the file enumeration/recursive lookup, like VBoxManage
1732 * does in here.
1733 */
1734 enum GETOPTDEF_COPY
1735 {
1736 GETOPTDEF_COPY_FOLLOW = 1000,
1737 GETOPTDEF_COPY_TARGETDIR
1738 };
1739 static const RTGETOPTDEF s_aOptions[] =
1740 {
1741 GCTLCMD_COMMON_OPTION_DEFS()
1742 { "--follow", GETOPTDEF_COPY_FOLLOW, RTGETOPT_REQ_NOTHING },
1743 { "--recursive", 'R', RTGETOPT_REQ_NOTHING },
1744 { "--target-directory", GETOPTDEF_COPY_TARGETDIR, RTGETOPT_REQ_STRING }
1745 };
1746
1747 int ch;
1748 RTGETOPTUNION ValueUnion;
1749 RTGETOPTSTATE GetState;
1750 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
1751
1752 const char *pszDst = NULL;
1753 bool fFollow = false;
1754 bool fRecursive = false;
1755 uint32_t uUsage = fHostToGuest ? USAGE_GSTCTRL_COPYTO : USAGE_GSTCTRL_COPYFROM;
1756
1757 SOURCEVEC vecSources;
1758
1759 int vrc = VINF_SUCCESS;
1760 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
1761 {
1762 /* For options that require an argument, ValueUnion has received the value. */
1763 switch (ch)
1764 {
1765 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
1766
1767 case GETOPTDEF_COPY_FOLLOW:
1768 fFollow = true;
1769 break;
1770
1771 case 'R': /* Recursive processing */
1772 fRecursive = true;
1773 break;
1774
1775 case GETOPTDEF_COPY_TARGETDIR:
1776 pszDst = ValueUnion.psz;
1777 break;
1778
1779 case VINF_GETOPT_NOT_OPTION:
1780 /* Last argument and no destination specified with
1781 * --target-directory yet? Then use the current
1782 * (= last) argument as destination. */
1783 if ( GetState.argc == GetState.iNext
1784 && pszDst == NULL)
1785 pszDst = ValueUnion.psz;
1786 else
1787 {
1788 try
1789 { /* Save the source directory. */
1790 vecSources.push_back(SOURCEFILEENTRY(ValueUnion.psz));
1791 }
1792 catch (std::bad_alloc &)
1793 {
1794 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Out of memory");
1795 }
1796 }
1797 break;
1798
1799 default:
1800 return errorGetOptEx(USAGE_GUESTCONTROL, uUsage, ch, &ValueUnion);
1801 }
1802 }
1803
1804 if (!vecSources.size())
1805 return errorSyntaxEx(USAGE_GUESTCONTROL, uUsage, "No source(s) specified!");
1806
1807 if (pszDst == NULL)
1808 return errorSyntaxEx(USAGE_GUESTCONTROL, uUsage, "No destination specified!");
1809
1810 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
1811 if (rcExit != RTEXITCODE_SUCCESS)
1812 return rcExit;
1813
1814 /*
1815 * Done parsing arguments, do some more preparations.
1816 */
1817 if (pCtx->cVerbose)
1818 {
1819 if (fHostToGuest)
1820 RTPrintf("Copying from host to guest ...\n");
1821 else
1822 RTPrintf("Copying from guest to host ...\n");
1823 }
1824
1825 ComPtr<IProgress> pProgress;
1826 HRESULT rc = S_OK;
1827
1828 for (unsigned long s = 0; s < vecSources.size(); s++)
1829 {
1830 Utf8Str strSrc = vecSources[s].GetSource();
1831 Utf8Str strFilter = vecSources[s].GetFilter();
1832 RT_NOREF(strFilter);
1833 /* strFilter can be NULL if not set. */
1834
1835 if (fHostToGuest)
1836 {
1837 if (RTFileExists(strSrc.c_str()))
1838 {
1839 if (pCtx->cVerbose)
1840 RTPrintf("File '%s' -> '%s'\n", strSrc.c_str(), pszDst);
1841
1842 SafeArray<FileCopyFlag_T> copyFlags;
1843 rc = pCtx->pGuestSession->FileCopyToGuest(Bstr(strSrc).raw(), Bstr(pszDst).raw(),
1844 ComSafeArrayAsInParam(copyFlags), pProgress.asOutParam());
1845 }
1846 else if (RTDirExists(strSrc.c_str()))
1847 {
1848 if (pCtx->cVerbose)
1849 RTPrintf("Directory '%s' -> '%s'\n", strSrc.c_str(), pszDst);
1850
1851 SafeArray<DirectoryCopyFlag_T> copyFlags;
1852 copyFlags.push_back(DirectoryCopyFlag_CopyIntoExisting);
1853 rc = pCtx->pGuestSession->DirectoryCopyToGuest(Bstr(strSrc).raw(), Bstr(pszDst).raw(),
1854 ComSafeArrayAsInParam(copyFlags), pProgress.asOutParam());
1855 }
1856 else if (pCtx->cVerbose)
1857 RTPrintf("Warning: \"%s\" does not exist or is not a file/directory, skipping ...\n", strSrc.c_str());
1858 }
1859 else
1860 {
1861 /* We need to query the source type on the guest first in order to know which copy flavor we need. */
1862 ComPtr<IGuestFsObjInfo> pFsObjInfo;
1863 FsObjType_T enmObjType = FsObjType_Unknown; /* Shut up MSC */
1864 rc = pCtx->pGuestSession->FsObjQueryInfo(Bstr(strSrc).raw(), TRUE /* fFollowSymlinks */, pFsObjInfo.asOutParam());
1865 if (SUCCEEDED(rc))
1866 rc = pFsObjInfo->COMGETTER(Type)(&enmObjType);
1867 if (FAILED(rc))
1868 {
1869 if (pCtx->cVerbose)
1870 RTPrintf("Warning: Cannot stat for element '%s': No such element\n", strSrc.c_str());
1871 continue; /* Skip. */
1872 }
1873
1874 if (enmObjType == FsObjType_Directory)
1875 {
1876 if (pCtx->cVerbose)
1877 RTPrintf("Directory '%s' -> '%s'\n", strSrc.c_str(), pszDst);
1878
1879 SafeArray<DirectoryCopyFlag_T> copyFlags;
1880 copyFlags.push_back(DirectoryCopyFlag_CopyIntoExisting);
1881 rc = pCtx->pGuestSession->DirectoryCopyFromGuest(Bstr(strSrc).raw(), Bstr(pszDst).raw(),
1882 ComSafeArrayAsInParam(copyFlags), pProgress.asOutParam());
1883 }
1884 else if (enmObjType == FsObjType_File)
1885 {
1886 if (pCtx->cVerbose)
1887 RTPrintf("File '%s' -> '%s'\n", strSrc.c_str(), pszDst);
1888
1889 SafeArray<FileCopyFlag_T> copyFlags;
1890 rc = pCtx->pGuestSession->FileCopyFromGuest(Bstr(strSrc).raw(), Bstr(pszDst).raw(),
1891 ComSafeArrayAsInParam(copyFlags), pProgress.asOutParam());
1892 }
1893 else if (pCtx->cVerbose)
1894 RTPrintf("Warning: Skipping '%s': Not handled\n", strSrc.c_str());
1895 }
1896 }
1897
1898 if (FAILED(rc))
1899 {
1900 vrc = gctlPrintError(pCtx->pGuestSession, COM_IIDOF(IGuestSession));
1901 }
1902 else if (pProgress.isNotNull())
1903 {
1904 if (pCtx->cVerbose)
1905 rc = showProgress(pProgress);
1906 else
1907 rc = pProgress->WaitForCompletion(-1 /* No timeout */);
1908 if (SUCCEEDED(rc))
1909 CHECK_PROGRESS_ERROR(pProgress, ("File copy failed"));
1910 vrc = gctlPrintProgressError(pProgress);
1911 }
1912
1913 return RT_SUCCESS(vrc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1914}
1915
1916static DECLCALLBACK(RTEXITCODE) gctlHandleCopyFrom(PGCTLCMDCTX pCtx, int argc, char **argv)
1917{
1918 return gctlHandleCopy(pCtx, argc, argv, false /* Guest to host */);
1919}
1920
1921static DECLCALLBACK(RTEXITCODE) gctlHandleCopyTo(PGCTLCMDCTX pCtx, int argc, char **argv)
1922{
1923 return gctlHandleCopy(pCtx, argc, argv, true /* Host to guest */);
1924}
1925
1926static DECLCALLBACK(RTEXITCODE) handleCtrtMkDir(PGCTLCMDCTX pCtx, int argc, char **argv)
1927{
1928 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
1929
1930 static const RTGETOPTDEF s_aOptions[] =
1931 {
1932 GCTLCMD_COMMON_OPTION_DEFS()
1933 { "--mode", 'm', RTGETOPT_REQ_UINT32 },
1934 { "--parents", 'P', RTGETOPT_REQ_NOTHING }
1935 };
1936
1937 int ch;
1938 RTGETOPTUNION ValueUnion;
1939 RTGETOPTSTATE GetState;
1940 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
1941
1942 SafeArray<DirectoryCreateFlag_T> dirCreateFlags;
1943 uint32_t fDirMode = 0; /* Default mode. */
1944 uint32_t cDirsCreated = 0;
1945 RTEXITCODE rcExit = RTEXITCODE_SUCCESS;
1946
1947 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
1948 {
1949 /* For options that require an argument, ValueUnion has received the value. */
1950 switch (ch)
1951 {
1952 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
1953
1954 case 'm': /* Mode */
1955 fDirMode = ValueUnion.u32;
1956 break;
1957
1958 case 'P': /* Create parents */
1959 dirCreateFlags.push_back(DirectoryCreateFlag_Parents);
1960 break;
1961
1962 case VINF_GETOPT_NOT_OPTION:
1963 if (cDirsCreated == 0)
1964 {
1965 /*
1966 * First non-option - no more options now.
1967 */
1968 rcExit = gctlCtxPostOptionParsingInit(pCtx);
1969 if (rcExit != RTEXITCODE_SUCCESS)
1970 return rcExit;
1971 if (pCtx->cVerbose)
1972 RTPrintf("Creating %RU32 directories...\n", argc - GetState.iNext + 1);
1973 }
1974 if (g_fGuestCtrlCanceled)
1975 return RTMsgErrorExit(RTEXITCODE_FAILURE, "mkdir was interrupted by Ctrl-C (%u left)\n",
1976 argc - GetState.iNext + 1);
1977
1978 /*
1979 * Create the specified directory.
1980 *
1981 * On failure we'll change the exit status to failure and
1982 * continue with the next directory that needs creating. We do
1983 * this because we only create new things, and because this is
1984 * how /bin/mkdir works on unix.
1985 */
1986 cDirsCreated++;
1987 if (pCtx->cVerbose)
1988 RTPrintf("Creating directory \"%s\" ...\n", ValueUnion.psz);
1989 try
1990 {
1991 HRESULT rc;
1992 CHECK_ERROR(pCtx->pGuestSession, DirectoryCreate(Bstr(ValueUnion.psz).raw(),
1993 fDirMode, ComSafeArrayAsInParam(dirCreateFlags)));
1994 if (FAILED(rc))
1995 rcExit = RTEXITCODE_FAILURE;
1996 }
1997 catch (std::bad_alloc &)
1998 {
1999 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Out of memory\n");
2000 }
2001 break;
2002
2003 default:
2004 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MKDIR, ch, &ValueUnion);
2005 }
2006 }
2007
2008 if (!cDirsCreated)
2009 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MKDIR, "No directory to create specified!");
2010 return rcExit;
2011}
2012
2013
2014static DECLCALLBACK(RTEXITCODE) gctlHandleRmDir(PGCTLCMDCTX pCtx, int argc, char **argv)
2015{
2016 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2017
2018 static const RTGETOPTDEF s_aOptions[] =
2019 {
2020 GCTLCMD_COMMON_OPTION_DEFS()
2021 { "--recursive", 'R', RTGETOPT_REQ_NOTHING },
2022 };
2023
2024 int ch;
2025 RTGETOPTUNION ValueUnion;
2026 RTGETOPTSTATE GetState;
2027 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2028
2029 bool fRecursive = false;
2030 uint32_t cDirRemoved = 0;
2031 RTEXITCODE rcExit = RTEXITCODE_SUCCESS;
2032
2033 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
2034 {
2035 /* For options that require an argument, ValueUnion has received the value. */
2036 switch (ch)
2037 {
2038 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2039
2040 case 'R':
2041 fRecursive = true;
2042 break;
2043
2044 case VINF_GETOPT_NOT_OPTION:
2045 {
2046 if (cDirRemoved == 0)
2047 {
2048 /*
2049 * First non-option - no more options now.
2050 */
2051 rcExit = gctlCtxPostOptionParsingInit(pCtx);
2052 if (rcExit != RTEXITCODE_SUCCESS)
2053 return rcExit;
2054 if (pCtx->cVerbose)
2055 RTPrintf("Removing %RU32 directorie%ss...\n", argc - GetState.iNext + 1, fRecursive ? "trees" : "");
2056 }
2057 if (g_fGuestCtrlCanceled)
2058 return RTMsgErrorExit(RTEXITCODE_FAILURE, "rmdir was interrupted by Ctrl-C (%u left)\n",
2059 argc - GetState.iNext + 1);
2060
2061 cDirRemoved++;
2062 HRESULT rc;
2063 if (!fRecursive)
2064 {
2065 /*
2066 * Remove exactly one directory.
2067 */
2068 if (pCtx->cVerbose)
2069 RTPrintf("Removing directory \"%s\" ...\n", ValueUnion.psz);
2070 try
2071 {
2072 CHECK_ERROR(pCtx->pGuestSession, DirectoryRemove(Bstr(ValueUnion.psz).raw()));
2073 }
2074 catch (std::bad_alloc &)
2075 {
2076 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Out of memory\n");
2077 }
2078 }
2079 else
2080 {
2081 /*
2082 * Remove the directory and anything under it, that means files
2083 * and everything. This is in the tradition of the Windows NT
2084 * CMD.EXE "rmdir /s" operation, a tradition which jpsoft's TCC
2085 * strongly warns against (and half-ways questions the sense of).
2086 */
2087 if (pCtx->cVerbose)
2088 RTPrintf("Recursively removing directory \"%s\" ...\n", ValueUnion.psz);
2089 try
2090 {
2091 /** @todo Make flags configurable. */
2092 com::SafeArray<DirectoryRemoveRecFlag_T> aRemRecFlags;
2093 aRemRecFlags.push_back(DirectoryRemoveRecFlag_ContentAndDir);
2094
2095 ComPtr<IProgress> ptrProgress;
2096 CHECK_ERROR(pCtx->pGuestSession, DirectoryRemoveRecursive(Bstr(ValueUnion.psz).raw(),
2097 ComSafeArrayAsInParam(aRemRecFlags),
2098 ptrProgress.asOutParam()));
2099 if (SUCCEEDED(rc))
2100 {
2101 if (pCtx->cVerbose)
2102 rc = showProgress(ptrProgress);
2103 else
2104 rc = ptrProgress->WaitForCompletion(-1 /* indefinitely */);
2105 if (SUCCEEDED(rc))
2106 CHECK_PROGRESS_ERROR(ptrProgress, ("Directory deletion failed"));
2107 ptrProgress.setNull();
2108 }
2109 }
2110 catch (std::bad_alloc &)
2111 {
2112 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Out of memory during recursive rmdir\n");
2113 }
2114 }
2115
2116 /*
2117 * This command returns immediately on failure since it's destructive in nature.
2118 */
2119 if (FAILED(rc))
2120 return RTEXITCODE_FAILURE;
2121 break;
2122 }
2123
2124 default:
2125 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_RMDIR, ch, &ValueUnion);
2126 }
2127 }
2128
2129 if (!cDirRemoved)
2130 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_RMDIR, "No directory to remove specified!");
2131 return rcExit;
2132}
2133
2134static DECLCALLBACK(RTEXITCODE) gctlHandleRm(PGCTLCMDCTX pCtx, int argc, char **argv)
2135{
2136 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2137
2138 static const RTGETOPTDEF s_aOptions[] =
2139 {
2140 GCTLCMD_COMMON_OPTION_DEFS()
2141 { "--force", 'f', RTGETOPT_REQ_NOTHING, },
2142 };
2143
2144 int ch;
2145 RTGETOPTUNION ValueUnion;
2146 RTGETOPTSTATE GetState;
2147 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2148
2149 uint32_t cFilesDeleted = 0;
2150 RTEXITCODE rcExit = RTEXITCODE_SUCCESS;
2151 bool fForce = true;
2152
2153 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
2154 {
2155 /* For options that require an argument, ValueUnion has received the value. */
2156 switch (ch)
2157 {
2158 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2159
2160 case VINF_GETOPT_NOT_OPTION:
2161 if (cFilesDeleted == 0)
2162 {
2163 /*
2164 * First non-option - no more options now.
2165 */
2166 rcExit = gctlCtxPostOptionParsingInit(pCtx);
2167 if (rcExit != RTEXITCODE_SUCCESS)
2168 return rcExit;
2169 if (pCtx->cVerbose)
2170 RTPrintf("Removing %RU32 file(s)...\n", argc - GetState.iNext + 1);
2171 }
2172 if (g_fGuestCtrlCanceled)
2173 return RTMsgErrorExit(RTEXITCODE_FAILURE, "rm was interrupted by Ctrl-C (%u left)\n",
2174 argc - GetState.iNext + 1);
2175
2176 /*
2177 * Remove the specified file.
2178 *
2179 * On failure we will by default stop, however, the force option will
2180 * by unix traditions force us to ignore errors and continue.
2181 */
2182 cFilesDeleted++;
2183 if (pCtx->cVerbose)
2184 RTPrintf("Removing file \"%s\" ...\n", ValueUnion.psz);
2185 try
2186 {
2187 /** @todo How does IGuestSession::FsObjRemove work with read-only files? Do we
2188 * need to do some chmod or whatever to better emulate the --force flag? */
2189 HRESULT rc;
2190 CHECK_ERROR(pCtx->pGuestSession, FsObjRemove(Bstr(ValueUnion.psz).raw()));
2191 if (FAILED(rc) && !fForce)
2192 return RTEXITCODE_FAILURE;
2193 }
2194 catch (std::bad_alloc &)
2195 {
2196 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Out of memory\n");
2197 }
2198 break;
2199
2200 default:
2201 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_RM, ch, &ValueUnion);
2202 }
2203 }
2204
2205 if (!cFilesDeleted && !fForce)
2206 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_RM, "No file to remove specified!");
2207 return rcExit;
2208}
2209
2210static DECLCALLBACK(RTEXITCODE) gctlHandleMv(PGCTLCMDCTX pCtx, int argc, char **argv)
2211{
2212 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2213
2214 static const RTGETOPTDEF s_aOptions[] =
2215 {
2216 GCTLCMD_COMMON_OPTION_DEFS()
2217 };
2218
2219 int ch;
2220 RTGETOPTUNION ValueUnion;
2221 RTGETOPTSTATE GetState;
2222 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2223
2224 int vrc = VINF_SUCCESS;
2225
2226 bool fDryrun = false;
2227 std::vector< Utf8Str > vecSources;
2228 const char *pszDst = NULL;
2229 com::SafeArray<FsObjRenameFlag_T> aRenameFlags;
2230
2231 try
2232 {
2233 /** @todo Make flags configurable. */
2234 aRenameFlags.push_back(FsObjRenameFlag_NoReplace);
2235
2236 while ( (ch = RTGetOpt(&GetState, &ValueUnion))
2237 && RT_SUCCESS(vrc))
2238 {
2239 /* For options that require an argument, ValueUnion has received the value. */
2240 switch (ch)
2241 {
2242 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2243
2244 /** @todo Implement a --dryrun command. */
2245 /** @todo Implement rename flags. */
2246
2247 case VINF_GETOPT_NOT_OPTION:
2248 vecSources.push_back(Utf8Str(ValueUnion.psz));
2249 pszDst = ValueUnion.psz;
2250 break;
2251
2252 default:
2253 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MV, ch, &ValueUnion);
2254 }
2255 }
2256 }
2257 catch (std::bad_alloc)
2258 {
2259 vrc = VERR_NO_MEMORY;
2260 }
2261
2262 if (RT_FAILURE(vrc))
2263 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Failed to initialize, rc=%Rrc\n", vrc);
2264
2265 size_t cSources = vecSources.size();
2266 if (!cSources)
2267 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MV,
2268 "No source(s) to move specified!");
2269 if (cSources < 2)
2270 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MV,
2271 "No destination specified!");
2272
2273 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
2274 if (rcExit != RTEXITCODE_SUCCESS)
2275 return rcExit;
2276
2277 /* Delete last element, which now is the destination. */
2278 vecSources.pop_back();
2279 cSources = vecSources.size();
2280
2281 HRESULT rc = S_OK;
2282
2283 if (cSources > 1)
2284 {
2285 BOOL fExists = FALSE;
2286 rc = pCtx->pGuestSession->DirectoryExists(Bstr(pszDst).raw(), FALSE /*followSymlinks*/, &fExists);
2287 if (FAILED(rc) || !fExists)
2288 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Destination must be a directory when specifying multiple sources\n");
2289 }
2290
2291 /*
2292 * Rename (move) the entries.
2293 */
2294 if (pCtx->cVerbose)
2295 RTPrintf("Renaming %RU32 %s ...\n", cSources, cSources > 1 ? "entries" : "entry");
2296
2297 std::vector< Utf8Str >::iterator it = vecSources.begin();
2298 while ( (it != vecSources.end())
2299 && !g_fGuestCtrlCanceled)
2300 {
2301 Utf8Str strCurSource = (*it);
2302
2303 ComPtr<IGuestFsObjInfo> pFsObjInfo;
2304 FsObjType_T enmObjType = FsObjType_Unknown; /* Shut up MSC */
2305 rc = pCtx->pGuestSession->FsObjQueryInfo(Bstr(strCurSource).raw(), FALSE /*followSymlinks*/, pFsObjInfo.asOutParam());
2306 if (SUCCEEDED(rc))
2307 rc = pFsObjInfo->COMGETTER(Type)(&enmObjType);
2308 if (FAILED(rc))
2309 {
2310 if (pCtx->cVerbose)
2311 RTPrintf("Warning: Cannot stat for element \"%s\": No such element\n",
2312 strCurSource.c_str());
2313 ++it;
2314 continue; /* Skip. */
2315 }
2316
2317 if (pCtx->cVerbose)
2318 RTPrintf("Renaming %s \"%s\" to \"%s\" ...\n",
2319 enmObjType == FsObjType_Directory ? "directory" : "file",
2320 strCurSource.c_str(), pszDst);
2321
2322 if (!fDryrun)
2323 {
2324 if (enmObjType == FsObjType_Directory)
2325 {
2326 CHECK_ERROR_BREAK(pCtx->pGuestSession, FsObjRename(Bstr(strCurSource).raw(),
2327 Bstr(pszDst).raw(),
2328 ComSafeArrayAsInParam(aRenameFlags)));
2329
2330 /* Break here, since it makes no sense to rename mroe than one source to
2331 * the same directory. */
2332/** @todo r=bird: You are being kind of windowsy (or just DOSish) about the 'sense' part here,
2333 * while being totaly buggy about the behavior. 'VBoxGuest guestcontrol ren dir1 dir2 dstdir' will
2334 * stop after 'dir1' and SILENTLY ignore dir2. If you tried this on Windows, you'd see an error
2335 * being displayed. If you 'man mv' on a nearby unixy system, you'd see that they've made perfect
2336 * sense out of any situation with more than one source. */
2337 it = vecSources.end();
2338 break;
2339 }
2340 else
2341 CHECK_ERROR_BREAK(pCtx->pGuestSession, FsObjRename(Bstr(strCurSource).raw(),
2342 Bstr(pszDst).raw(),
2343 ComSafeArrayAsInParam(aRenameFlags)));
2344 }
2345
2346 ++it;
2347 }
2348
2349 if ( (it != vecSources.end())
2350 && pCtx->cVerbose)
2351 {
2352 RTPrintf("Warning: Not all sources were renamed\n");
2353 }
2354
2355 return FAILED(rc) ? RTEXITCODE_FAILURE : RTEXITCODE_SUCCESS;
2356}
2357
2358static DECLCALLBACK(RTEXITCODE) gctlHandleMkTemp(PGCTLCMDCTX pCtx, int argc, char **argv)
2359{
2360 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2361
2362 static const RTGETOPTDEF s_aOptions[] =
2363 {
2364 GCTLCMD_COMMON_OPTION_DEFS()
2365 { "--mode", 'm', RTGETOPT_REQ_UINT32 },
2366 { "--directory", 'D', RTGETOPT_REQ_NOTHING },
2367 { "--secure", 's', RTGETOPT_REQ_NOTHING },
2368 { "--tmpdir", 't', RTGETOPT_REQ_STRING }
2369 };
2370
2371 int ch;
2372 RTGETOPTUNION ValueUnion;
2373 RTGETOPTSTATE GetState;
2374 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2375
2376 Utf8Str strTemplate;
2377 uint32_t fMode = 0; /* Default mode. */
2378 bool fDirectory = false;
2379 bool fSecure = false;
2380 Utf8Str strTempDir;
2381
2382 DESTDIRMAP mapDirs;
2383
2384 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
2385 {
2386 /* For options that require an argument, ValueUnion has received the value. */
2387 switch (ch)
2388 {
2389 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2390
2391 case 'm': /* Mode */
2392 fMode = ValueUnion.u32;
2393 break;
2394
2395 case 'D': /* Create directory */
2396 fDirectory = true;
2397 break;
2398
2399 case 's': /* Secure */
2400 fSecure = true;
2401 break;
2402
2403 case 't': /* Temp directory */
2404 strTempDir = ValueUnion.psz;
2405 break;
2406
2407 case VINF_GETOPT_NOT_OPTION:
2408 {
2409 if (strTemplate.isEmpty())
2410 strTemplate = ValueUnion.psz;
2411 else
2412 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MKTEMP,
2413 "More than one template specified!\n");
2414 break;
2415 }
2416
2417 default:
2418 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MKTEMP, ch, &ValueUnion);
2419 }
2420 }
2421
2422 if (strTemplate.isEmpty())
2423 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MKTEMP,
2424 "No template specified!");
2425
2426 if (!fDirectory)
2427 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_MKTEMP,
2428 "Creating temporary files is currently not supported!");
2429
2430 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
2431 if (rcExit != RTEXITCODE_SUCCESS)
2432 return rcExit;
2433
2434 /*
2435 * Create the directories.
2436 */
2437 if (pCtx->cVerbose)
2438 {
2439 if (fDirectory && !strTempDir.isEmpty())
2440 RTPrintf("Creating temporary directory from template '%s' in directory '%s' ...\n",
2441 strTemplate.c_str(), strTempDir.c_str());
2442 else if (fDirectory)
2443 RTPrintf("Creating temporary directory from template '%s' in default temporary directory ...\n",
2444 strTemplate.c_str());
2445 else if (!fDirectory && !strTempDir.isEmpty())
2446 RTPrintf("Creating temporary file from template '%s' in directory '%s' ...\n",
2447 strTemplate.c_str(), strTempDir.c_str());
2448 else if (!fDirectory)
2449 RTPrintf("Creating temporary file from template '%s' in default temporary directory ...\n",
2450 strTemplate.c_str());
2451 }
2452
2453 HRESULT rc = S_OK;
2454 if (fDirectory)
2455 {
2456 Bstr directory;
2457 CHECK_ERROR(pCtx->pGuestSession, DirectoryCreateTemp(Bstr(strTemplate).raw(),
2458 fMode, Bstr(strTempDir).raw(),
2459 fSecure,
2460 directory.asOutParam()));
2461 if (SUCCEEDED(rc))
2462 RTPrintf("Directory name: %ls\n", directory.raw());
2463 }
2464 else
2465 {
2466 // else - temporary file not yet implemented
2467 /** @todo implement temporary file creation (we fend it off above, no
2468 * worries). */
2469 rc = E_FAIL;
2470 }
2471
2472 return FAILED(rc) ? RTEXITCODE_FAILURE : RTEXITCODE_SUCCESS;
2473}
2474
2475static DECLCALLBACK(RTEXITCODE) gctlHandleStat(PGCTLCMDCTX pCtx, int argc, char **argv)
2476{
2477 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2478
2479 static const RTGETOPTDEF s_aOptions[] =
2480 {
2481 GCTLCMD_COMMON_OPTION_DEFS()
2482 { "--dereference", 'L', RTGETOPT_REQ_NOTHING },
2483 { "--file-system", 'f', RTGETOPT_REQ_NOTHING },
2484 { "--format", 'c', RTGETOPT_REQ_STRING },
2485 { "--terse", 't', RTGETOPT_REQ_NOTHING }
2486 };
2487
2488 int ch;
2489 RTGETOPTUNION ValueUnion;
2490 RTGETOPTSTATE GetState;
2491 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2492
2493 DESTDIRMAP mapObjs;
2494
2495 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
2496 {
2497 /* For options that require an argument, ValueUnion has received the value. */
2498 switch (ch)
2499 {
2500 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2501
2502 case 'L': /* Dereference */
2503 case 'f': /* File-system */
2504 case 'c': /* Format */
2505 case 't': /* Terse */
2506 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_STAT,
2507 "Command \"%s\" not implemented yet!", ValueUnion.psz);
2508
2509 case VINF_GETOPT_NOT_OPTION:
2510 mapObjs[ValueUnion.psz]; /* Add element to check to map. */
2511 break;
2512
2513 default:
2514 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_STAT, ch, &ValueUnion);
2515 }
2516 }
2517
2518 size_t cObjs = mapObjs.size();
2519 if (!cObjs)
2520 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_STAT,
2521 "No element(s) to check specified!");
2522
2523 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
2524 if (rcExit != RTEXITCODE_SUCCESS)
2525 return rcExit;
2526
2527 HRESULT rc;
2528
2529 /*
2530 * Doing the checks.
2531 */
2532 DESTDIRMAPITER it = mapObjs.begin();
2533 while (it != mapObjs.end())
2534 {
2535 if (pCtx->cVerbose)
2536 RTPrintf("Checking for element \"%s\" ...\n", it->first.c_str());
2537
2538 ComPtr<IGuestFsObjInfo> pFsObjInfo;
2539 rc = pCtx->pGuestSession->FsObjQueryInfo(Bstr(it->first).raw(), FALSE /*followSymlinks*/, pFsObjInfo.asOutParam());
2540 if (FAILED(rc))
2541 {
2542 /* If there's at least one element which does not exist on the guest,
2543 * drop out with exitcode 1. */
2544 if (pCtx->cVerbose)
2545 RTPrintf("Cannot stat for element \"%s\": No such element\n",
2546 it->first.c_str());
2547 rcExit = RTEXITCODE_FAILURE;
2548 }
2549 else
2550 {
2551 FsObjType_T objType;
2552 pFsObjInfo->COMGETTER(Type)(&objType); /** @todo What about error checking? */
2553 switch (objType)
2554 {
2555 case FsObjType_File:
2556 RTPrintf("Element \"%s\" found: Is a file\n", it->first.c_str());
2557 break;
2558
2559 case FsObjType_Directory:
2560 RTPrintf("Element \"%s\" found: Is a directory\n", it->first.c_str());
2561 break;
2562
2563 case FsObjType_Symlink:
2564 RTPrintf("Element \"%s\" found: Is a symlink\n", it->first.c_str());
2565 break;
2566
2567 default:
2568 RTPrintf("Element \"%s\" found, type unknown (%ld)\n", it->first.c_str(), objType);
2569 break;
2570 }
2571
2572 /** @todo Show more information about this element. */
2573 }
2574
2575 ++it;
2576 }
2577
2578 return rcExit;
2579}
2580
2581static DECLCALLBACK(RTEXITCODE) gctlHandleUpdateAdditions(PGCTLCMDCTX pCtx, int argc, char **argv)
2582{
2583 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2584
2585 /*
2586 * Check the syntax. We can deduce the correct syntax from the number of
2587 * arguments.
2588 */
2589 Utf8Str strSource;
2590 com::SafeArray<IN_BSTR> aArgs;
2591 bool fWaitStartOnly = false;
2592
2593 static const RTGETOPTDEF s_aOptions[] =
2594 {
2595 GCTLCMD_COMMON_OPTION_DEFS()
2596 { "--source", 's', RTGETOPT_REQ_STRING },
2597 { "--wait-start", 'w', RTGETOPT_REQ_NOTHING }
2598 };
2599
2600 int ch;
2601 RTGETOPTUNION ValueUnion;
2602 RTGETOPTSTATE GetState;
2603 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2604
2605 int vrc = VINF_SUCCESS;
2606 while ( (ch = RTGetOpt(&GetState, &ValueUnion))
2607 && RT_SUCCESS(vrc))
2608 {
2609 switch (ch)
2610 {
2611 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2612
2613 case 's':
2614 strSource = ValueUnion.psz;
2615 break;
2616
2617 case 'w':
2618 fWaitStartOnly = true;
2619 break;
2620
2621 case VINF_GETOPT_NOT_OPTION:
2622 if (aArgs.size() == 0 && strSource.isEmpty())
2623 strSource = ValueUnion.psz;
2624 else
2625 aArgs.push_back(Bstr(ValueUnion.psz).raw());
2626 break;
2627
2628 default:
2629 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_UPDATEGA, ch, &ValueUnion);
2630 }
2631 }
2632
2633 if (pCtx->cVerbose)
2634 RTPrintf("Updating Guest Additions ...\n");
2635
2636 HRESULT rc = S_OK;
2637 while (strSource.isEmpty())
2638 {
2639 ComPtr<ISystemProperties> pProperties;
2640 CHECK_ERROR_BREAK(pCtx->pArg->virtualBox, COMGETTER(SystemProperties)(pProperties.asOutParam()));
2641 Bstr strISO;
2642 CHECK_ERROR_BREAK(pProperties, COMGETTER(DefaultAdditionsISO)(strISO.asOutParam()));
2643 strSource = strISO;
2644 break;
2645 }
2646
2647 /* Determine source if not set yet. */
2648 if (strSource.isEmpty())
2649 {
2650 RTMsgError("No Guest Additions source found or specified, aborting\n");
2651 vrc = VERR_FILE_NOT_FOUND;
2652 }
2653 else if (!RTFileExists(strSource.c_str()))
2654 {
2655 RTMsgError("Source \"%s\" does not exist!\n", strSource.c_str());
2656 vrc = VERR_FILE_NOT_FOUND;
2657 }
2658
2659 if (RT_SUCCESS(vrc))
2660 {
2661 if (pCtx->cVerbose)
2662 RTPrintf("Using source: %s\n", strSource.c_str());
2663
2664
2665 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
2666 if (rcExit != RTEXITCODE_SUCCESS)
2667 return rcExit;
2668
2669
2670 com::SafeArray<AdditionsUpdateFlag_T> aUpdateFlags;
2671 if (fWaitStartOnly)
2672 {
2673 aUpdateFlags.push_back(AdditionsUpdateFlag_WaitForUpdateStartOnly);
2674 if (pCtx->cVerbose)
2675 RTPrintf("Preparing and waiting for Guest Additions installer to start ...\n");
2676 }
2677
2678 ComPtr<IProgress> pProgress;
2679 CHECK_ERROR(pCtx->pGuest, UpdateGuestAdditions(Bstr(strSource).raw(),
2680 ComSafeArrayAsInParam(aArgs),
2681 /* Wait for whole update process to complete. */
2682 ComSafeArrayAsInParam(aUpdateFlags),
2683 pProgress.asOutParam()));
2684 if (FAILED(rc))
2685 vrc = gctlPrintError(pCtx->pGuest, COM_IIDOF(IGuest));
2686 else
2687 {
2688 if (pCtx->cVerbose)
2689 rc = showProgress(pProgress);
2690 else
2691 rc = pProgress->WaitForCompletion(-1 /* No timeout */);
2692
2693 if (SUCCEEDED(rc))
2694 CHECK_PROGRESS_ERROR(pProgress, ("Guest additions update failed"));
2695 vrc = gctlPrintProgressError(pProgress);
2696 if ( RT_SUCCESS(vrc)
2697 && pCtx->cVerbose)
2698 {
2699 RTPrintf("Guest Additions update successful\n");
2700 }
2701 }
2702 }
2703
2704 return RT_SUCCESS(vrc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
2705}
2706
2707static DECLCALLBACK(RTEXITCODE) gctlHandleList(PGCTLCMDCTX pCtx, int argc, char **argv)
2708{
2709 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2710
2711 static const RTGETOPTDEF s_aOptions[] =
2712 {
2713 GCTLCMD_COMMON_OPTION_DEFS()
2714 };
2715
2716 int ch;
2717 RTGETOPTUNION ValueUnion;
2718 RTGETOPTSTATE GetState;
2719 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2720
2721 bool fSeenListArg = false;
2722 bool fListAll = false;
2723 bool fListSessions = false;
2724 bool fListProcesses = false;
2725 bool fListFiles = false;
2726
2727 int vrc = VINF_SUCCESS;
2728 while ( (ch = RTGetOpt(&GetState, &ValueUnion))
2729 && RT_SUCCESS(vrc))
2730 {
2731 switch (ch)
2732 {
2733 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2734
2735 case VINF_GETOPT_NOT_OPTION:
2736 if ( !RTStrICmp(ValueUnion.psz, "sessions")
2737 || !RTStrICmp(ValueUnion.psz, "sess"))
2738 fListSessions = true;
2739 else if ( !RTStrICmp(ValueUnion.psz, "processes")
2740 || !RTStrICmp(ValueUnion.psz, "procs"))
2741 fListSessions = fListProcesses = true; /* Showing processes implies showing sessions. */
2742 else if (!RTStrICmp(ValueUnion.psz, "files"))
2743 fListSessions = fListFiles = true; /* Showing files implies showing sessions. */
2744 else if (!RTStrICmp(ValueUnion.psz, "all"))
2745 fListAll = true;
2746 else
2747 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_LIST,
2748 "Unknown list: '%s'", ValueUnion.psz);
2749 fSeenListArg = true;
2750 break;
2751
2752 default:
2753 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_UPDATEGA, ch, &ValueUnion);
2754 }
2755 }
2756
2757 if (!fSeenListArg)
2758 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_LIST, "Missing list name");
2759 Assert(fListAll || fListSessions);
2760
2761 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
2762 if (rcExit != RTEXITCODE_SUCCESS)
2763 return rcExit;
2764
2765
2766 /** @todo Do we need a machine-readable output here as well? */
2767
2768 HRESULT rc;
2769 size_t cTotalProcs = 0;
2770 size_t cTotalFiles = 0;
2771
2772 SafeIfaceArray <IGuestSession> collSessions;
2773 CHECK_ERROR(pCtx->pGuest, COMGETTER(Sessions)(ComSafeArrayAsOutParam(collSessions)));
2774 if (SUCCEEDED(rc))
2775 {
2776 size_t const cSessions = collSessions.size();
2777 if (cSessions)
2778 {
2779 RTPrintf("Active guest sessions:\n");
2780
2781 /** @todo Make this output a bit prettier. No time now. */
2782
2783 for (size_t i = 0; i < cSessions; i++)
2784 {
2785 ComPtr<IGuestSession> pCurSession = collSessions[i];
2786 if (!pCurSession.isNull())
2787 {
2788 do
2789 {
2790 ULONG uID;
2791 CHECK_ERROR_BREAK(pCurSession, COMGETTER(Id)(&uID));
2792 Bstr strName;
2793 CHECK_ERROR_BREAK(pCurSession, COMGETTER(Name)(strName.asOutParam()));
2794 Bstr strUser;
2795 CHECK_ERROR_BREAK(pCurSession, COMGETTER(User)(strUser.asOutParam()));
2796 GuestSessionStatus_T sessionStatus;
2797 CHECK_ERROR_BREAK(pCurSession, COMGETTER(Status)(&sessionStatus));
2798 RTPrintf("\n\tSession #%-3zu ID=%-3RU32 User=%-16ls Status=[%s] Name=%ls",
2799 i, uID, strUser.raw(), gctlGuestSessionStatusToText(sessionStatus), strName.raw());
2800 } while (0);
2801
2802 if ( fListAll
2803 || fListProcesses)
2804 {
2805 SafeIfaceArray <IGuestProcess> collProcesses;
2806 CHECK_ERROR_BREAK(pCurSession, COMGETTER(Processes)(ComSafeArrayAsOutParam(collProcesses)));
2807 for (size_t a = 0; a < collProcesses.size(); a++)
2808 {
2809 ComPtr<IGuestProcess> pCurProcess = collProcesses[a];
2810 if (!pCurProcess.isNull())
2811 {
2812 do
2813 {
2814 ULONG uPID;
2815 CHECK_ERROR_BREAK(pCurProcess, COMGETTER(PID)(&uPID));
2816 Bstr strExecPath;
2817 CHECK_ERROR_BREAK(pCurProcess, COMGETTER(ExecutablePath)(strExecPath.asOutParam()));
2818 ProcessStatus_T procStatus;
2819 CHECK_ERROR_BREAK(pCurProcess, COMGETTER(Status)(&procStatus));
2820
2821 RTPrintf("\n\t\tProcess #%-03zu PID=%-6RU32 Status=[%s] Command=%ls",
2822 a, uPID, gctlProcessStatusToText(procStatus), strExecPath.raw());
2823 } while (0);
2824 }
2825 }
2826
2827 cTotalProcs += collProcesses.size();
2828 }
2829
2830 if ( fListAll
2831 || fListFiles)
2832 {
2833 SafeIfaceArray <IGuestFile> collFiles;
2834 CHECK_ERROR_BREAK(pCurSession, COMGETTER(Files)(ComSafeArrayAsOutParam(collFiles)));
2835 for (size_t a = 0; a < collFiles.size(); a++)
2836 {
2837 ComPtr<IGuestFile> pCurFile = collFiles[a];
2838 if (!pCurFile.isNull())
2839 {
2840 do
2841 {
2842 ULONG idFile;
2843 CHECK_ERROR_BREAK(pCurFile, COMGETTER(Id)(&idFile));
2844 Bstr strName;
2845 CHECK_ERROR_BREAK(pCurFile, COMGETTER(FileName)(strName.asOutParam()));
2846 FileStatus_T fileStatus;
2847 CHECK_ERROR_BREAK(pCurFile, COMGETTER(Status)(&fileStatus));
2848
2849 RTPrintf("\n\t\tFile #%-03zu ID=%-6RU32 Status=[%s] Name=%ls",
2850 a, idFile, gctlFileStatusToText(fileStatus), strName.raw());
2851 } while (0);
2852 }
2853 }
2854
2855 cTotalFiles += collFiles.size();
2856 }
2857 }
2858 }
2859
2860 RTPrintf("\n\nTotal guest sessions: %zu\n", collSessions.size());
2861 if (fListAll || fListProcesses)
2862 RTPrintf("Total guest processes: %zu\n", cTotalProcs);
2863 if (fListAll || fListFiles)
2864 RTPrintf("Total guest files: %zu\n", cTotalFiles);
2865 }
2866 else
2867 RTPrintf("No active guest sessions found\n");
2868 }
2869
2870 if (FAILED(rc))
2871 rcExit = RTEXITCODE_FAILURE;
2872
2873 return rcExit;
2874}
2875
2876static DECLCALLBACK(RTEXITCODE) gctlHandleCloseProcess(PGCTLCMDCTX pCtx, int argc, char **argv)
2877{
2878 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
2879
2880 static const RTGETOPTDEF s_aOptions[] =
2881 {
2882 GCTLCMD_COMMON_OPTION_DEFS()
2883 { "--session-id", 'i', RTGETOPT_REQ_UINT32 },
2884 { "--session-name", 'n', RTGETOPT_REQ_STRING }
2885 };
2886
2887 int ch;
2888 RTGETOPTUNION ValueUnion;
2889 RTGETOPTSTATE GetState;
2890 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
2891
2892 std::vector < uint32_t > vecPID;
2893 ULONG ulSessionID = UINT32_MAX;
2894 Utf8Str strSessionName;
2895
2896 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
2897 {
2898 /* For options that require an argument, ValueUnion has received the value. */
2899 switch (ch)
2900 {
2901 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
2902
2903 case 'n': /* Session name (or pattern) */
2904 strSessionName = ValueUnion.psz;
2905 break;
2906
2907 case 'i': /* Session ID */
2908 ulSessionID = ValueUnion.u32;
2909 break;
2910
2911 case VINF_GETOPT_NOT_OPTION:
2912 {
2913 /* Treat every else specified as a PID to kill. */
2914 uint32_t uPid;
2915 int rc = RTStrToUInt32Ex(ValueUnion.psz, NULL, 0, &uPid);
2916 if ( RT_SUCCESS(rc)
2917 && rc != VWRN_TRAILING_CHARS
2918 && rc != VWRN_NUMBER_TOO_BIG
2919 && rc != VWRN_NEGATIVE_UNSIGNED)
2920 {
2921 if (uPid != 0)
2922 {
2923 try
2924 {
2925 vecPID.push_back(uPid);
2926 }
2927 catch (std::bad_alloc &)
2928 {
2929 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Out of memory");
2930 }
2931 }
2932 else
2933 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSEPROCESS, "Invalid PID value: 0");
2934 }
2935 else
2936 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSEPROCESS,
2937 "Error parsing PID value: %Rrc", rc);
2938 break;
2939 }
2940
2941 default:
2942 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSEPROCESS, ch, &ValueUnion);
2943 }
2944 }
2945
2946 if (vecPID.empty())
2947 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSEPROCESS,
2948 "At least one PID must be specified to kill!");
2949
2950 if ( strSessionName.isEmpty()
2951 && ulSessionID == UINT32_MAX)
2952 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSEPROCESS, "No session ID specified!");
2953
2954 if ( strSessionName.isNotEmpty()
2955 && ulSessionID != UINT32_MAX)
2956 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSEPROCESS,
2957 "Either session ID or name (pattern) must be specified");
2958
2959 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
2960 if (rcExit != RTEXITCODE_SUCCESS)
2961 return rcExit;
2962
2963 HRESULT rc = S_OK;
2964
2965 ComPtr<IGuestSession> pSession;
2966 ComPtr<IGuestProcess> pProcess;
2967 do
2968 {
2969 uint32_t uProcsTerminated = 0;
2970 bool fSessionFound = false;
2971
2972 SafeIfaceArray <IGuestSession> collSessions;
2973 CHECK_ERROR_BREAK(pCtx->pGuest, COMGETTER(Sessions)(ComSafeArrayAsOutParam(collSessions)));
2974 size_t cSessions = collSessions.size();
2975
2976 uint32_t uSessionsHandled = 0;
2977 for (size_t i = 0; i < cSessions; i++)
2978 {
2979 pSession = collSessions[i];
2980 Assert(!pSession.isNull());
2981
2982 ULONG uID; /* Session ID */
2983 CHECK_ERROR_BREAK(pSession, COMGETTER(Id)(&uID));
2984 Bstr strName;
2985 CHECK_ERROR_BREAK(pSession, COMGETTER(Name)(strName.asOutParam()));
2986 Utf8Str strNameUtf8(strName); /* Session name */
2987 if (strSessionName.isEmpty()) /* Search by ID. Slow lookup. */
2988 {
2989 fSessionFound = uID == ulSessionID;
2990 }
2991 else /* ... or by naming pattern. */
2992 {
2993 if (RTStrSimplePatternMatch(strSessionName.c_str(), strNameUtf8.c_str()))
2994 fSessionFound = true;
2995 }
2996
2997 if (fSessionFound)
2998 {
2999 AssertStmt(!pSession.isNull(), break);
3000 uSessionsHandled++;
3001
3002 SafeIfaceArray <IGuestProcess> collProcs;
3003 CHECK_ERROR_BREAK(pSession, COMGETTER(Processes)(ComSafeArrayAsOutParam(collProcs)));
3004
3005 size_t cProcs = collProcs.size();
3006 for (size_t p = 0; p < cProcs; p++)
3007 {
3008 pProcess = collProcs[p];
3009 Assert(!pProcess.isNull());
3010
3011 ULONG uPID; /* Process ID */
3012 CHECK_ERROR_BREAK(pProcess, COMGETTER(PID)(&uPID));
3013
3014 bool fProcFound = false;
3015 for (size_t a = 0; a < vecPID.size(); a++) /* Slow, but works. */
3016 {
3017 fProcFound = vecPID[a] == uPID;
3018 if (fProcFound)
3019 break;
3020 }
3021
3022 if (fProcFound)
3023 {
3024 if (pCtx->cVerbose)
3025 RTPrintf("Terminating process (PID %RU32) (session ID %RU32) ...\n",
3026 uPID, uID);
3027 CHECK_ERROR_BREAK(pProcess, Terminate());
3028 uProcsTerminated++;
3029 }
3030 else
3031 {
3032 if (ulSessionID != UINT32_MAX)
3033 RTPrintf("No matching process(es) for session ID %RU32 found\n",
3034 ulSessionID);
3035 }
3036
3037 pProcess.setNull();
3038 }
3039
3040 pSession.setNull();
3041 }
3042 }
3043
3044 if (!uSessionsHandled)
3045 RTPrintf("No matching session(s) found\n");
3046
3047 if (uProcsTerminated)
3048 RTPrintf("%RU32 %s terminated\n",
3049 uProcsTerminated, uProcsTerminated == 1 ? "process" : "processes");
3050
3051 } while (0);
3052
3053 pProcess.setNull();
3054 pSession.setNull();
3055
3056 return SUCCEEDED(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
3057}
3058
3059
3060static DECLCALLBACK(RTEXITCODE) gctlHandleCloseSession(PGCTLCMDCTX pCtx, int argc, char **argv)
3061{
3062 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
3063
3064 enum GETOPTDEF_SESSIONCLOSE
3065 {
3066 GETOPTDEF_SESSIONCLOSE_ALL = 2000
3067 };
3068 static const RTGETOPTDEF s_aOptions[] =
3069 {
3070 GCTLCMD_COMMON_OPTION_DEFS()
3071 { "--all", GETOPTDEF_SESSIONCLOSE_ALL, RTGETOPT_REQ_NOTHING },
3072 { "--session-id", 'i', RTGETOPT_REQ_UINT32 },
3073 { "--session-name", 'n', RTGETOPT_REQ_STRING }
3074 };
3075
3076 int ch;
3077 RTGETOPTUNION ValueUnion;
3078 RTGETOPTSTATE GetState;
3079 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
3080
3081 ULONG ulSessionID = UINT32_MAX;
3082 Utf8Str strSessionName;
3083
3084 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
3085 {
3086 /* For options that require an argument, ValueUnion has received the value. */
3087 switch (ch)
3088 {
3089 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
3090
3091 case 'n': /* Session name pattern */
3092 strSessionName = ValueUnion.psz;
3093 break;
3094
3095 case 'i': /* Session ID */
3096 ulSessionID = ValueUnion.u32;
3097 break;
3098
3099 case GETOPTDEF_SESSIONCLOSE_ALL:
3100 strSessionName = "*";
3101 break;
3102
3103 case VINF_GETOPT_NOT_OPTION:
3104 /** @todo Supply a CSV list of IDs or patterns to close?
3105 * break; */
3106 default:
3107 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSESESSION, ch, &ValueUnion);
3108 }
3109 }
3110
3111 if ( strSessionName.isEmpty()
3112 && ulSessionID == UINT32_MAX)
3113 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSESESSION,
3114 "No session ID specified!");
3115
3116 if ( !strSessionName.isEmpty()
3117 && ulSessionID != UINT32_MAX)
3118 return errorSyntaxEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_CLOSESESSION,
3119 "Either session ID or name (pattern) must be specified");
3120
3121 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
3122 if (rcExit != RTEXITCODE_SUCCESS)
3123 return rcExit;
3124
3125 HRESULT rc = S_OK;
3126
3127 do
3128 {
3129 bool fSessionFound = false;
3130 size_t cSessionsHandled = 0;
3131
3132 SafeIfaceArray <IGuestSession> collSessions;
3133 CHECK_ERROR_BREAK(pCtx->pGuest, COMGETTER(Sessions)(ComSafeArrayAsOutParam(collSessions)));
3134 size_t cSessions = collSessions.size();
3135
3136 for (size_t i = 0; i < cSessions; i++)
3137 {
3138 ComPtr<IGuestSession> pSession = collSessions[i];
3139 Assert(!pSession.isNull());
3140
3141 ULONG uID; /* Session ID */
3142 CHECK_ERROR_BREAK(pSession, COMGETTER(Id)(&uID));
3143 Bstr strName;
3144 CHECK_ERROR_BREAK(pSession, COMGETTER(Name)(strName.asOutParam()));
3145 Utf8Str strNameUtf8(strName); /* Session name */
3146
3147 if (strSessionName.isEmpty()) /* Search by ID. Slow lookup. */
3148 {
3149 fSessionFound = uID == ulSessionID;
3150 }
3151 else /* ... or by naming pattern. */
3152 {
3153 if (RTStrSimplePatternMatch(strSessionName.c_str(), strNameUtf8.c_str()))
3154 fSessionFound = true;
3155 }
3156
3157 if (fSessionFound)
3158 {
3159 cSessionsHandled++;
3160
3161 Assert(!pSession.isNull());
3162 if (pCtx->cVerbose)
3163 RTPrintf("Closing guest session ID=#%RU32 \"%s\" ...\n",
3164 uID, strNameUtf8.c_str());
3165 CHECK_ERROR_BREAK(pSession, Close());
3166 if (pCtx->cVerbose)
3167 RTPrintf("Guest session successfully closed\n");
3168
3169 pSession.setNull();
3170 }
3171 }
3172
3173 if (!cSessionsHandled)
3174 {
3175 RTPrintf("No guest session(s) found\n");
3176 rc = E_ABORT; /* To set exit code accordingly. */
3177 }
3178
3179 } while (0);
3180
3181 return SUCCEEDED(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
3182}
3183
3184
3185static DECLCALLBACK(RTEXITCODE) gctlHandleWatch(PGCTLCMDCTX pCtx, int argc, char **argv)
3186{
3187 AssertPtrReturn(pCtx, RTEXITCODE_FAILURE);
3188
3189 /*
3190 * Parse arguments.
3191 */
3192 static const RTGETOPTDEF s_aOptions[] =
3193 {
3194 GCTLCMD_COMMON_OPTION_DEFS()
3195 };
3196
3197 int ch;
3198 RTGETOPTUNION ValueUnion;
3199 RTGETOPTSTATE GetState;
3200 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
3201
3202 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
3203 {
3204 /* For options that require an argument, ValueUnion has received the value. */
3205 switch (ch)
3206 {
3207 GCTLCMD_COMMON_OPTION_CASES(pCtx, ch, &ValueUnion);
3208
3209 case VINF_GETOPT_NOT_OPTION:
3210 default:
3211 return errorGetOptEx(USAGE_GUESTCONTROL, USAGE_GSTCTRL_WATCH, ch, &ValueUnion);
3212 }
3213 }
3214
3215 /** @todo Specify categories to watch for. */
3216 /** @todo Specify a --timeout for waiting only for a certain amount of time? */
3217
3218 RTEXITCODE rcExit = gctlCtxPostOptionParsingInit(pCtx);
3219 if (rcExit != RTEXITCODE_SUCCESS)
3220 return rcExit;
3221
3222 HRESULT rc;
3223
3224 try
3225 {
3226 ComObjPtr<GuestEventListenerImpl> pGuestListener;
3227 do
3228 {
3229 /* Listener creation. */
3230 pGuestListener.createObject();
3231 pGuestListener->init(new GuestEventListener());
3232
3233 /* Register for IGuest events. */
3234 ComPtr<IEventSource> es;
3235 CHECK_ERROR_BREAK(pCtx->pGuest, COMGETTER(EventSource)(es.asOutParam()));
3236 com::SafeArray<VBoxEventType_T> eventTypes;
3237 eventTypes.push_back(VBoxEventType_OnGuestSessionRegistered);
3238 /** @todo Also register for VBoxEventType_OnGuestUserStateChanged on demand? */
3239 CHECK_ERROR_BREAK(es, RegisterListener(pGuestListener, ComSafeArrayAsInParam(eventTypes),
3240 true /* Active listener */));
3241 /* Note: All other guest control events have to be registered
3242 * as their corresponding objects appear. */
3243
3244 } while (0);
3245
3246 if (pCtx->cVerbose)
3247 RTPrintf("Waiting for events ...\n");
3248
3249 while (!g_fGuestCtrlCanceled)
3250 {
3251 /** @todo Timeout handling (see above)? */
3252 RTThreadSleep(10);
3253 }
3254
3255 if (pCtx->cVerbose)
3256 RTPrintf("Signal caught, exiting ...\n");
3257
3258 if (!pGuestListener.isNull())
3259 {
3260 /* Guest callback unregistration. */
3261 ComPtr<IEventSource> pES;
3262 CHECK_ERROR(pCtx->pGuest, COMGETTER(EventSource)(pES.asOutParam()));
3263 if (!pES.isNull())
3264 CHECK_ERROR(pES, UnregisterListener(pGuestListener));
3265 pGuestListener.setNull();
3266 }
3267 }
3268 catch (std::bad_alloc &)
3269 {
3270 rc = E_OUTOFMEMORY;
3271 }
3272
3273 return SUCCEEDED(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
3274}
3275
3276/**
3277 * Access the guest control store.
3278 *
3279 * @returns program exit code.
3280 * @note see the command line API description for parameters
3281 */
3282RTEXITCODE handleGuestControl(HandlerArg *pArg)
3283{
3284 AssertPtr(pArg);
3285
3286#ifdef DEBUG_andy_disabled
3287 if (RT_FAILURE(tstTranslatePath()))
3288 return RTEXITCODE_FAILURE;
3289#endif
3290
3291 /*
3292 * Command definitions.
3293 */
3294 static const GCTLCMDDEF s_aCmdDefs[] =
3295 {
3296 { "run", gctlHandleRun, USAGE_GSTCTRL_RUN, 0, },
3297 { "start", gctlHandleStart, USAGE_GSTCTRL_START, 0, },
3298 { "copyfrom", gctlHandleCopyFrom, USAGE_GSTCTRL_COPYFROM, 0, },
3299 { "copyto", gctlHandleCopyTo, USAGE_GSTCTRL_COPYTO, 0, },
3300
3301 { "mkdir", handleCtrtMkDir, USAGE_GSTCTRL_MKDIR, 0, },
3302 { "md", handleCtrtMkDir, USAGE_GSTCTRL_MKDIR, 0, },
3303 { "createdirectory", handleCtrtMkDir, USAGE_GSTCTRL_MKDIR, 0, },
3304 { "createdir", handleCtrtMkDir, USAGE_GSTCTRL_MKDIR, 0, },
3305
3306 { "rmdir", gctlHandleRmDir, USAGE_GSTCTRL_RMDIR, 0, },
3307 { "removedir", gctlHandleRmDir, USAGE_GSTCTRL_RMDIR, 0, },
3308 { "removedirectory", gctlHandleRmDir, USAGE_GSTCTRL_RMDIR, 0, },
3309
3310 { "rm", gctlHandleRm, USAGE_GSTCTRL_RM, 0, },
3311 { "removefile", gctlHandleRm, USAGE_GSTCTRL_RM, 0, },
3312 { "erase", gctlHandleRm, USAGE_GSTCTRL_RM, 0, },
3313 { "del", gctlHandleRm, USAGE_GSTCTRL_RM, 0, },
3314 { "delete", gctlHandleRm, USAGE_GSTCTRL_RM, 0, },
3315
3316 { "mv", gctlHandleMv, USAGE_GSTCTRL_MV, 0, },
3317 { "move", gctlHandleMv, USAGE_GSTCTRL_MV, 0, },
3318 { "ren", gctlHandleMv, USAGE_GSTCTRL_MV, 0, },
3319 { "rename", gctlHandleMv, USAGE_GSTCTRL_MV, 0, },
3320
3321 { "mktemp", gctlHandleMkTemp, USAGE_GSTCTRL_MKTEMP, 0, },
3322 { "createtemp", gctlHandleMkTemp, USAGE_GSTCTRL_MKTEMP, 0, },
3323 { "createtemporary", gctlHandleMkTemp, USAGE_GSTCTRL_MKTEMP, 0, },
3324
3325 { "stat", gctlHandleStat, USAGE_GSTCTRL_STAT, 0, },
3326
3327 { "closeprocess", gctlHandleCloseProcess, USAGE_GSTCTRL_CLOSEPROCESS, GCTLCMDCTX_F_SESSION_ANONYMOUS | GCTLCMDCTX_F_NO_SIGNAL_HANDLER, },
3328 { "closesession", gctlHandleCloseSession, USAGE_GSTCTRL_CLOSESESSION, GCTLCMDCTX_F_SESSION_ANONYMOUS | GCTLCMDCTX_F_NO_SIGNAL_HANDLER, },
3329 { "list", gctlHandleList, USAGE_GSTCTRL_LIST, GCTLCMDCTX_F_SESSION_ANONYMOUS | GCTLCMDCTX_F_NO_SIGNAL_HANDLER, },
3330 { "watch", gctlHandleWatch, USAGE_GSTCTRL_WATCH, GCTLCMDCTX_F_SESSION_ANONYMOUS | GCTLCMDCTX_F_NO_SIGNAL_HANDLER, },
3331
3332 {"updateguestadditions",gctlHandleUpdateAdditions, USAGE_GSTCTRL_UPDATEGA, GCTLCMDCTX_F_SESSION_ANONYMOUS | GCTLCMDCTX_F_NO_SIGNAL_HANDLER, },
3333 { "updateadditions", gctlHandleUpdateAdditions, USAGE_GSTCTRL_UPDATEGA, GCTLCMDCTX_F_SESSION_ANONYMOUS | GCTLCMDCTX_F_NO_SIGNAL_HANDLER, },
3334 { "updatega", gctlHandleUpdateAdditions, USAGE_GSTCTRL_UPDATEGA, GCTLCMDCTX_F_SESSION_ANONYMOUS | GCTLCMDCTX_F_NO_SIGNAL_HANDLER, },
3335 };
3336
3337 /*
3338 * VBoxManage guestcontrol [common-options] <VM> [common-options] <sub-command> ...
3339 *
3340 * Parse common options and VM name until we find a sub-command. Allowing
3341 * the user to put the user and password related options before the
3342 * sub-command makes it easier to edit the command line when doing several
3343 * operations with the same guest user account. (Accidentally, it also
3344 * makes the syntax diagram shorter and easier to read.)
3345 */
3346 GCTLCMDCTX CmdCtx;
3347 RTEXITCODE rcExit = gctrCmdCtxInit(&CmdCtx, pArg);
3348 if (rcExit == RTEXITCODE_SUCCESS)
3349 {
3350 static const RTGETOPTDEF s_CommonOptions[] = { GCTLCMD_COMMON_OPTION_DEFS() };
3351
3352 int ch;
3353 RTGETOPTUNION ValueUnion;
3354 RTGETOPTSTATE GetState;
3355 RTGetOptInit(&GetState, pArg->argc, pArg->argv, s_CommonOptions, RT_ELEMENTS(s_CommonOptions), 0, 0 /* No sorting! */);
3356
3357 while ((ch = RTGetOpt(&GetState, &ValueUnion)) != 0)
3358 {
3359 switch (ch)
3360 {
3361 GCTLCMD_COMMON_OPTION_CASES(&CmdCtx, ch, &ValueUnion);
3362
3363 case VINF_GETOPT_NOT_OPTION:
3364 /* First comes the VM name or UUID. */
3365 if (!CmdCtx.pszVmNameOrUuid)
3366 CmdCtx.pszVmNameOrUuid = ValueUnion.psz;
3367 /*
3368 * The sub-command is next. Look it up and invoke it.
3369 * Note! Currently no warnings about user/password options (like we'll do later on)
3370 * for GCTLCMDCTX_F_SESSION_ANONYMOUS commands. No reason to be too pedantic.
3371 */
3372 else
3373 {
3374 const char *pszCmd = ValueUnion.psz;
3375 uint32_t iCmd;
3376 for (iCmd = 0; iCmd < RT_ELEMENTS(s_aCmdDefs); iCmd++)
3377 if (strcmp(s_aCmdDefs[iCmd].pszName, pszCmd) == 0)
3378 {
3379 CmdCtx.pCmdDef = &s_aCmdDefs[iCmd];
3380
3381 rcExit = s_aCmdDefs[iCmd].pfnHandler(&CmdCtx, pArg->argc - GetState.iNext + 1,
3382 &pArg->argv[GetState.iNext - 1]);
3383
3384 gctlCtxTerm(&CmdCtx);
3385 return rcExit;
3386 }
3387 return errorSyntax(USAGE_GUESTCONTROL, "Unknown sub-command: '%s'", pszCmd);
3388 }
3389 break;
3390
3391 default:
3392 return errorGetOpt(USAGE_GUESTCONTROL, ch, &ValueUnion);
3393 }
3394 }
3395 if (CmdCtx.pszVmNameOrUuid)
3396 rcExit = errorSyntax(USAGE_GUESTCONTROL, "Missing sub-command");
3397 else
3398 rcExit = errorSyntax(USAGE_GUESTCONTROL, "Missing VM name and sub-command");
3399 }
3400 return rcExit;
3401}
3402#endif /* !VBOX_ONLY_DOCS */
3403
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