VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxControl/VBoxControl.cpp@ 74251

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

VBoxControl: Added gzip, ls, tar and unzip commands.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 64.5 KB
Line 
1/* $Id: VBoxControl.cpp 70431 2018-01-02 15:14:30Z vboxsync $ */
2/** @file
3 * VBoxControl - Guest Additions Command Line Management Interface.
4 */
5
6/*
7 * Copyright (C) 2008-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include <iprt/alloca.h>
23#include <iprt/cpp/autores.h>
24#include <iprt/buildconfig.h>
25#include <iprt/getopt.h>
26#include <iprt/initterm.h>
27#include <iprt/mem.h>
28#include <iprt/message.h>
29#include <iprt/path.h>
30#include <iprt/string.h>
31#include <iprt/stream.h>
32#include <iprt/zip.h>
33#include <VBox/log.h>
34#include <VBox/version.h>
35#include <VBox/VBoxGuestLib.h>
36#ifdef RT_OS_WINDOWS
37# include <iprt/win/windows.h>
38#endif
39#ifdef VBOX_WITH_GUEST_PROPS
40# include <VBox/HostServices/GuestPropertySvc.h>
41#endif
42#ifdef VBOX_WITH_DPC_LATENCY_CHECKER
43# include <VBox/VBoxGuest.h>
44# include "../VBoxGuest/lib/VBoxGuestR3LibInternal.h" /* HACK ALERT! Using vbglR3DoIOCtl directly!! */
45#endif
46
47
48/*********************************************************************************************************************************
49* Global Variables *
50*********************************************************************************************************************************/
51/** The program name (derived from argv[0]). */
52char const *g_pszProgName = "";
53/** The current verbosity level. */
54int g_cVerbosity = 0;
55
56
57/** @name Displays the program usage message.
58 * @{
59 */
60
61/**
62 * Helper function that does indentation.
63 *
64 * @param pszLine Text.
65 * @param pszName Program name.
66 * @param pszCommand Command/option syntax.
67 */
68static void doUsage(char const *pszLine, char const *pszName = "", char const *pszCommand = "")
69{
70 /* Allow for up to 15 characters command name length (VBoxControl.exe) with
71 * perfect column alignment. Beyond that there's at least one space between
72 * the command if there are command line parameters. */
73 RTPrintf("%s %-*s%s%s\n",
74 pszName,
75 *pszLine ? 35 - strlen(pszName) : 1, pszCommand,
76 *pszLine ? " " : "", pszLine);
77}
78
79/** Enumerate the different parts of the usage we might want to print out */
80enum VBoxControlUsage
81{
82#ifdef RT_OS_WINDOWS
83 GET_VIDEO_ACCEL,
84 SET_VIDEO_ACCEL,
85 VIDEO_FLAGS,
86 LIST_CUST_MODES,
87 ADD_CUST_MODE,
88 REMOVE_CUST_MODE,
89 SET_VIDEO_MODE,
90#endif
91#ifdef VBOX_WITH_GUEST_PROPS
92 GUEST_PROP,
93#endif
94#ifdef VBOX_WITH_SHARED_FOLDERS
95 GUEST_SHAREDFOLDERS,
96#endif
97#if !defined(VBOX_CONTROL_TEST)
98 WRITE_CORE_DUMP,
99#endif
100 WRITE_LOG,
101 TAKE_SNAPSHOT,
102 SAVE_STATE,
103 SUSPEND,
104 POWER_OFF,
105 VERSION,
106 HELP,
107 USAGE_ALL = UINT32_MAX
108};
109
110static RTEXITCODE usage(enum VBoxControlUsage eWhich = USAGE_ALL)
111{
112 RTPrintf("Usage:\n\n");
113 doUsage("print version number and exit", g_pszProgName, "[-V|--version]");
114 doUsage("suppress the logo", g_pszProgName, "--nologo ...");
115 RTPrintf("\n");
116
117 /* Exclude the Windows bits from the test version. Anyone who needs to
118 test them can fix this. */
119#if defined(RT_OS_WINDOWS) && !defined(VBOX_CONTROL_TEST)
120 if (eWhich == GET_VIDEO_ACCEL || eWhich == USAGE_ALL)
121 doUsage("", g_pszProgName, "getvideoacceleration");
122 if (eWhich == SET_VIDEO_ACCEL || eWhich == USAGE_ALL)
123 doUsage("<on|off>", g_pszProgName, "setvideoacceleration");
124 if (eWhich == VIDEO_FLAGS || eWhich == USAGE_ALL)
125 doUsage("<get|set|clear|delete> [hex mask]", g_pszProgName, "videoflags");
126 if (eWhich == LIST_CUST_MODES || eWhich == USAGE_ALL)
127 doUsage("", g_pszProgName, "listcustommodes");
128 if (eWhich == ADD_CUST_MODE || eWhich == USAGE_ALL)
129 doUsage("<width> <height> <bpp>", g_pszProgName, "addcustommode");
130 if (eWhich == REMOVE_CUST_MODE || eWhich == USAGE_ALL)
131 doUsage("<width> <height> <bpp>", g_pszProgName, "removecustommode");
132 if (eWhich == SET_VIDEO_MODE || eWhich == USAGE_ALL)
133 doUsage("<width> <height> <bpp> <screen>", g_pszProgName, "setvideomode");
134#endif
135#ifdef VBOX_WITH_GUEST_PROPS
136 if (eWhich == GUEST_PROP || eWhich == USAGE_ALL)
137 {
138 doUsage("get <property> [--verbose]", g_pszProgName, "guestproperty");
139 doUsage("set <property> [<value> [--flags <flags>]]", g_pszProgName, "guestproperty");
140 doUsage("delete|unset <property>", g_pszProgName, "guestproperty");
141 doUsage("enumerate [--patterns <patterns>]", g_pszProgName, "guestproperty");
142 doUsage("wait <patterns>", g_pszProgName, "guestproperty");
143 doUsage("[--timestamp <last timestamp>]");
144 doUsage("[--timeout <timeout in ms>");
145 }
146#endif
147#ifdef VBOX_WITH_SHARED_FOLDERS
148 if (eWhich == GUEST_SHAREDFOLDERS || eWhich == USAGE_ALL)
149 {
150 doUsage("list [-automount]", g_pszProgName, "sharedfolder");
151 }
152#endif
153
154#if !defined(VBOX_CONTROL_TEST)
155 if (eWhich == WRITE_CORE_DUMP || eWhich == USAGE_ALL)
156 doUsage("", g_pszProgName, "writecoredump");
157#endif
158 if (eWhich == WRITE_LOG || eWhich == USAGE_ALL)
159 doUsage("", g_pszProgName, "writelog [-n|--no-newline] [--] <msg>");
160 if (eWhich == TAKE_SNAPSHOT || eWhich == USAGE_ALL)
161 doUsage("", g_pszProgName, "takesnapshot");
162 if (eWhich == SAVE_STATE || eWhich == USAGE_ALL)
163 doUsage("", g_pszProgName, "savestate");
164 if (eWhich == SUSPEND || eWhich == USAGE_ALL)
165 doUsage("", g_pszProgName, "suspend");
166 if (eWhich == POWER_OFF || eWhich == USAGE_ALL)
167 doUsage("", g_pszProgName, "poweroff");
168 if (eWhich == HELP || eWhich == USAGE_ALL)
169 doUsage("[command]", g_pszProgName, "help");
170 if (eWhich == VERSION || eWhich == USAGE_ALL)
171 doUsage("", g_pszProgName, "version");
172
173 return RTEXITCODE_SUCCESS;
174}
175
176/** @} */
177
178
179/**
180 * Implementation of the '--version' option.
181 *
182 * @returns RTEXITCODE_SUCCESS
183 */
184static RTEXITCODE printVersion(void)
185{
186 RTPrintf("%sr%u\n", VBOX_VERSION_STRING, RTBldCfgRevision());
187 return RTEXITCODE_SUCCESS;
188}
189
190
191/**
192 * Displays an error message.
193 *
194 * @returns RTEXITCODE_FAILURE.
195 * @param pszFormat The message text. No newline.
196 * @param ... Format arguments.
197 */
198static RTEXITCODE VBoxControlError(const char *pszFormat, ...)
199{
200 /** @todo prefix with current command. */
201 va_list va;
202 va_start(va, pszFormat);
203 RTMsgErrorV(pszFormat, va);
204 va_end(va);
205 return RTEXITCODE_FAILURE;
206}
207
208
209/**
210 * Displays a getopt error.
211 *
212 * @returns RTEXITCODE_FAILURE.
213 * @param ch The RTGetOpt return value.
214 * @param pValueUnion The RTGetOpt return data.
215 */
216static RTEXITCODE VBoxCtrlGetOptError(int ch, PCRTGETOPTUNION pValueUnion)
217{
218 /** @todo prefix with current command. */
219 return RTGetOptPrintError(ch, pValueUnion);
220}
221
222
223/**
224 * Displays an syntax error message.
225 *
226 * @returns RTEXITCODE_FAILURE.
227 * @param pszFormat The message text. No newline.
228 * @param ... Format arguments.
229 */
230static RTEXITCODE VBoxControlSyntaxError(const char *pszFormat, ...)
231{
232 /** @todo prefix with current command. */
233 va_list va;
234 va_start(va, pszFormat);
235 RTMsgErrorV(pszFormat, va);
236 va_end(va);
237 return RTEXITCODE_FAILURE;
238}
239
240#if defined(RT_OS_WINDOWS) && !defined(VBOX_CONTROL_TEST)
241
242decltype(ChangeDisplaySettingsExA) *g_pfnChangeDisplaySettingsExA;
243decltype(ChangeDisplaySettings) *g_pfnChangeDisplaySettingsA;
244decltype(EnumDisplaySettingsA) *g_pfnEnumDisplaySettingsA;
245
246static unsigned nextAdjacentRectXP(RECTL const *paRects, unsigned cRects, unsigned iRect)
247{
248 for (unsigned i = 0; i < cRects; i++)
249 if (paRects[iRect].right == paRects[i].left)
250 return i;
251 return ~0U;
252}
253
254static unsigned nextAdjacentRectXN(RECTL const *paRects, unsigned cRects, unsigned iRect)
255{
256 for (unsigned i = 0; i < cRects; i++)
257 if (paRects[iRect].left == paRects[i].right)
258 return i;
259 return ~0U;
260}
261
262static unsigned nextAdjacentRectYP(RECTL const *paRects, unsigned cRects, unsigned iRect)
263{
264 for (unsigned i = 0; i < cRects; i++)
265 if (paRects[iRect].bottom == paRects[i].top)
266 return i;
267 return ~0U;
268}
269
270unsigned nextAdjacentRectYN(RECTL const *paRects, unsigned cRects, unsigned iRect)
271{
272 for (unsigned i = 0; i < cRects; i++)
273 if (paRects[iRect].top == paRects[i].bottom)
274 return i;
275 return ~0U;
276}
277
278void resizeRect(RECTL *paRects, unsigned cRects, unsigned iPrimary, unsigned iResized, int NewWidth, int NewHeight)
279{
280 RECTL *paNewRects = (RECTL *)alloca(sizeof (RECTL) * cRects);
281 memcpy (paNewRects, paRects, sizeof(RECTL) * cRects);
282 paNewRects[iResized].right += NewWidth - (paNewRects[iResized].right - paNewRects[iResized].left);
283 paNewRects[iResized].bottom += NewHeight - (paNewRects[iResized].bottom - paNewRects[iResized].top);
284
285 /* Verify all pairs of originally adjacent rectangles for all 4 directions.
286 * If the pair has a "good" delta (that is the first rectangle intersects the second)
287 * at a direction and the second rectangle is not primary one (which can not be moved),
288 * move the second rectangle to make it adjacent to the first one.
289 */
290
291 /* X positive. */
292 unsigned iRect;
293 for (iRect = 0; iRect < cRects; iRect++)
294 {
295 /* Find the next adjacent original rect in x positive direction. */
296 unsigned iNextRect = nextAdjacentRectXP (paRects, cRects, iRect);
297 Log(("next %d -> %d\n", iRect, iNextRect));
298
299 if (iNextRect == ~0 || iNextRect == iPrimary)
300 {
301 continue;
302 }
303
304 /* Check whether there is an X intersection between these adjacent rects in the new rectangles
305 * and fix the intersection if delta is "good".
306 */
307 int delta = paNewRects[iRect].right - paNewRects[iNextRect].left;
308
309 if (delta > 0)
310 {
311 Log(("XP intersection right %d left %d, diff %d\n",
312 paNewRects[iRect].right, paNewRects[iNextRect].left,
313 delta));
314
315 paNewRects[iNextRect].left += delta;
316 paNewRects[iNextRect].right += delta;
317 }
318 }
319
320 /* X negative. */
321 for (iRect = 0; iRect < cRects; iRect++)
322 {
323 /* Find the next adjacent original rect in x negative direction. */
324 unsigned iNextRect = nextAdjacentRectXN (paRects, cRects, iRect);
325 Log(("next %d -> %d\n", iRect, iNextRect));
326
327 if (iNextRect == ~0 || iNextRect == iPrimary)
328 {
329 continue;
330 }
331
332 /* Check whether there is an X intersection between these adjacent rects in the new rectangles
333 * and fix the intersection if delta is "good".
334 */
335 int delta = paNewRects[iRect].left - paNewRects[iNextRect].right;
336
337 if (delta < 0)
338 {
339 Log(("XN intersection left %d right %d, diff %d\n",
340 paNewRects[iRect].left, paNewRects[iNextRect].right,
341 delta));
342
343 paNewRects[iNextRect].left += delta;
344 paNewRects[iNextRect].right += delta;
345 }
346 }
347
348 /* Y positive (in the computer sense, top->down). */
349 for (iRect = 0; iRect < cRects; iRect++)
350 {
351 /* Find the next adjacent original rect in y positive direction. */
352 unsigned iNextRect = nextAdjacentRectYP (paRects, cRects, iRect);
353 Log(("next %d -> %d\n", iRect, iNextRect));
354
355 if (iNextRect == ~0 || iNextRect == iPrimary)
356 {
357 continue;
358 }
359
360 /* Check whether there is an Y intersection between these adjacent rects in the new rectangles
361 * and fix the intersection if delta is "good".
362 */
363 int delta = paNewRects[iRect].bottom - paNewRects[iNextRect].top;
364
365 if (delta > 0)
366 {
367 Log(("YP intersection bottom %d top %d, diff %d\n",
368 paNewRects[iRect].bottom, paNewRects[iNextRect].top,
369 delta));
370
371 paNewRects[iNextRect].top += delta;
372 paNewRects[iNextRect].bottom += delta;
373 }
374 }
375
376 /* Y negative (in the computer sense, down->top). */
377 for (iRect = 0; iRect < cRects; iRect++)
378 {
379 /* Find the next adjacent original rect in x negative direction. */
380 unsigned iNextRect = nextAdjacentRectYN (paRects, cRects, iRect);
381 Log(("next %d -> %d\n", iRect, iNextRect));
382
383 if (iNextRect == ~0 || iNextRect == iPrimary)
384 {
385 continue;
386 }
387
388 /* Check whether there is an Y intersection between these adjacent rects in the new rectangles
389 * and fix the intersection if delta is "good".
390 */
391 int delta = paNewRects[iRect].top - paNewRects[iNextRect].bottom;
392
393 if (delta < 0)
394 {
395 Log(("YN intersection top %d bottom %d, diff %d\n",
396 paNewRects[iRect].top, paNewRects[iNextRect].bottom,
397 delta));
398
399 paNewRects[iNextRect].top += delta;
400 paNewRects[iNextRect].bottom += delta;
401 }
402 }
403
404 memcpy (paRects, paNewRects, sizeof (RECTL) * cRects);
405 return;
406}
407
408/* Returns TRUE to try again. */
409static BOOL ResizeDisplayDevice(ULONG Id, DWORD Width, DWORD Height, DWORD BitsPerPixel)
410{
411 BOOL fModeReset = (Width == 0 && Height == 0 && BitsPerPixel == 0);
412
413 DISPLAY_DEVICE DisplayDevice;
414 RT_ZERO(DisplayDevice);
415 DisplayDevice.cb = sizeof(DisplayDevice);
416
417 /* Find out how many display devices the system has */
418 DWORD NumDevices = 0;
419 DWORD i = 0;
420 while (EnumDisplayDevices(NULL, i, &DisplayDevice, 0))
421 {
422 Log(("[%d] %s\n", i, DisplayDevice.DeviceName));
423
424 if (DisplayDevice.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE)
425 {
426 Log(("Found primary device. err %d\n", GetLastError()));
427 NumDevices++;
428 }
429 else if (!(DisplayDevice.StateFlags & DISPLAY_DEVICE_MIRRORING_DRIVER))
430 {
431
432 Log(("Found secondary device. err %d\n", GetLastError()));
433 NumDevices++;
434 }
435
436 RT_ZERO(DisplayDevice);
437 DisplayDevice.cb = sizeof(DisplayDevice);
438 i++;
439 }
440
441 Log(("Found total %d devices. err %d\n", NumDevices, GetLastError()));
442
443 if (NumDevices == 0 || Id >= NumDevices)
444 {
445 Log(("Requested identifier %d is invalid. err %d\n", Id, GetLastError()));
446 return FALSE;
447 }
448
449 DISPLAY_DEVICE *paDisplayDevices = (DISPLAY_DEVICE *)alloca(sizeof (DISPLAY_DEVICE) * NumDevices);
450 DEVMODE *paDeviceModes = (DEVMODE *)alloca(sizeof (DEVMODE) * NumDevices);
451 RECTL *paRects = (RECTL *)alloca(sizeof (RECTL) * NumDevices);
452
453 /* Fetch information about current devices and modes. */
454 DWORD DevNum = 0;
455 DWORD DevPrimaryNum = 0;
456
457 RT_ZERO(DisplayDevice);
458 DisplayDevice.cb = sizeof(DISPLAY_DEVICE);
459
460 i = 0;
461 while (EnumDisplayDevices (NULL, i, &DisplayDevice, 0))
462 {
463 Log(("[%d(%d)] %s\n", i, DevNum, DisplayDevice.DeviceName));
464
465 BOOL fFetchDevice = FALSE;
466
467 if (DisplayDevice.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE)
468 {
469 Log(("Found primary device. err %d\n", GetLastError()));
470 DevPrimaryNum = DevNum;
471 fFetchDevice = TRUE;
472 }
473 else if (!(DisplayDevice.StateFlags & DISPLAY_DEVICE_MIRRORING_DRIVER))
474 {
475
476 Log(("Found secondary device. err %d\n", GetLastError()));
477 fFetchDevice = TRUE;
478 }
479
480 if (fFetchDevice)
481 {
482 if (DevNum >= NumDevices)
483 {
484 Log(("%d >= %d\n", NumDevices, DevNum));
485 return FALSE;
486 }
487
488 paDisplayDevices[DevNum] = DisplayDevice;
489
490 RT_BZERO(&paDeviceModes[DevNum], sizeof(DEVMODE));
491 paDeviceModes[DevNum].dmSize = sizeof(DEVMODE);
492 if (!g_pfnEnumDisplaySettingsA((LPSTR)DisplayDevice.DeviceName, ENUM_REGISTRY_SETTINGS, &paDeviceModes[DevNum]))
493 {
494 Log(("EnumDisplaySettings err %d\n", GetLastError()));
495 return FALSE;
496 }
497
498 Log(("%dx%d at %d,%d\n",
499 paDeviceModes[DevNum].dmPelsWidth,
500 paDeviceModes[DevNum].dmPelsHeight,
501 paDeviceModes[DevNum].dmPosition.x,
502 paDeviceModes[DevNum].dmPosition.y));
503
504 paRects[DevNum].left = paDeviceModes[DevNum].dmPosition.x;
505 paRects[DevNum].top = paDeviceModes[DevNum].dmPosition.y;
506 paRects[DevNum].right = paDeviceModes[DevNum].dmPosition.x + paDeviceModes[DevNum].dmPelsWidth;
507 paRects[DevNum].bottom = paDeviceModes[DevNum].dmPosition.y + paDeviceModes[DevNum].dmPelsHeight;
508 DevNum++;
509 }
510
511 RT_ZERO(DisplayDevice);
512 DisplayDevice.cb = sizeof(DISPLAY_DEVICE);
513 i++;
514 }
515
516 if (Width == 0)
517 Width = paRects[Id].right - paRects[Id].left;
518
519 if (Height == 0)
520 Height = paRects[Id].bottom - paRects[Id].top;
521
522 /* Check whether a mode reset or a change is requested. */
523 if ( !fModeReset
524 && paRects[Id].right - paRects[Id].left == (LONG)Width
525 && paRects[Id].bottom - paRects[Id].top == (LONG)Height
526 && paDeviceModes[Id].dmBitsPerPel == BitsPerPixel)
527 {
528 Log(("VBoxDisplayThread : already at desired resolution.\n"));
529 return FALSE;
530 }
531
532 resizeRect(paRects, NumDevices, DevPrimaryNum, Id, Width, Height);
533#ifdef LOG_ENABLED
534 for (i = 0; i < NumDevices; i++)
535 Log(("[%d]: %d,%d %dx%d\n",
536 i, paRects[i].left, paRects[i].top,
537 paRects[i].right - paRects[i].left,
538 paRects[i].bottom - paRects[i].top));
539#endif /* Log */
540
541 /* Without this, Windows will not ask the miniport for its
542 * mode table but uses an internal cache instead.
543 */
544 DEVMODE tempDevMode;
545 RT_ZERO(tempDevMode);
546 tempDevMode.dmSize = sizeof(DEVMODE);
547 g_pfnEnumDisplaySettingsA(NULL, 0xffffff, &tempDevMode);
548
549 /* Assign the new rectangles to displays. */
550 for (i = 0; i < NumDevices; i++)
551 {
552 paDeviceModes[i].dmPosition.x = paRects[i].left;
553 paDeviceModes[i].dmPosition.y = paRects[i].top;
554 paDeviceModes[i].dmPelsWidth = paRects[i].right - paRects[i].left;
555 paDeviceModes[i].dmPelsHeight = paRects[i].bottom - paRects[i].top;
556
557 paDeviceModes[i].dmFields = DM_POSITION | DM_PELSHEIGHT | DM_PELSWIDTH;
558
559 if ( i == Id
560 && BitsPerPixel != 0)
561 {
562 paDeviceModes[i].dmFields |= DM_BITSPERPEL;
563 paDeviceModes[i].dmBitsPerPel = BitsPerPixel;
564 }
565 Log(("calling pfnChangeDisplaySettingsEx %p\n", g_pfnChangeDisplaySettingsExA));
566 g_pfnChangeDisplaySettingsExA((LPSTR)paDisplayDevices[i].DeviceName,
567 &paDeviceModes[i], NULL, CDS_NORESET | CDS_UPDATEREGISTRY, NULL);
568 Log(("ChangeDisplaySettingsEx position err %d\n", GetLastError()));
569 }
570
571 /* A second call to ChangeDisplaySettings updates the monitor. */
572 LONG status = g_pfnChangeDisplaySettingsA(NULL, 0);
573 Log(("ChangeDisplaySettings update status %d\n", status));
574 if (status == DISP_CHANGE_SUCCESSFUL || status == DISP_CHANGE_BADMODE)
575 {
576 /* Successfully set new video mode or our driver can not set the requested mode. Stop trying. */
577 return FALSE;
578 }
579
580 /* Retry the request. */
581 return TRUE;
582}
583
584static DECLCALLBACK(RTEXITCODE) handleSetVideoMode(int argc, char *argv[])
585{
586 if (argc != 3 && argc != 4)
587 {
588 usage(SET_VIDEO_MODE);
589 return RTEXITCODE_FAILURE;
590 }
591
592 DWORD xres = atoi(argv[0]);
593 DWORD yres = atoi(argv[1]);
594 DWORD bpp = atoi(argv[2]);
595 DWORD scr = 0;
596 if (argc == 4)
597 scr = atoi(argv[3]);
598
599 HMODULE hmodUser = GetModuleHandle("user32.dll");
600 if (hmodUser)
601 {
602 /* ChangeDisplaySettingsExA was probably added in W2K, whereas ChangeDisplaySettingsA
603 and EnumDisplaySettingsA was added in NT 3.51. */
604 g_pfnChangeDisplaySettingsExA = (decltype(g_pfnChangeDisplaySettingsExA))GetProcAddress(hmodUser, "ChangeDisplaySettingsExA");
605 g_pfnChangeDisplaySettingsA = (decltype(g_pfnChangeDisplaySettingsA)) GetProcAddress(hmodUser, "ChangeDisplaySettingsA");
606 g_pfnEnumDisplaySettingsA = (decltype(g_pfnEnumDisplaySettingsA)) GetProcAddress(hmodUser, "EnumDisplaySettingsA");
607
608 Log(("VBoxService: g_pfnChangeDisplaySettingsExA=%p g_pfnChangeDisplaySettingsA=%p g_pfnEnumDisplaySettingsA=%p\n",
609 g_pfnChangeDisplaySettingsExA, g_pfnChangeDisplaySettingsA, g_pfnEnumDisplaySettingsA));
610
611 if ( g_pfnChangeDisplaySettingsExA
612 && g_pfnChangeDisplaySettingsA
613 && g_pfnEnumDisplaySettingsA)
614 {
615 /* The screen index is 0 based in the ResizeDisplayDevice call. */
616 scr = scr > 0 ? scr - 1 : 0;
617
618 /* Horizontal resolution must be a multiple of 8, round down. */
619 xres &= ~0x7;
620
621 RTPrintf("Setting resolution of display %d to %dx%dx%d ...", scr, xres, yres, bpp);
622 ResizeDisplayDevice(scr, xres, yres, bpp);
623 RTPrintf("done.\n");
624 }
625 else
626 VBoxControlError("Error retrieving API for display change!");
627 }
628 else
629 VBoxControlError("Error retrieving handle to user32.dll!");
630
631 return RTEXITCODE_SUCCESS;
632}
633
634static int checkVBoxVideoKey(HKEY hkeyVideo)
635{
636 RTUTF16 wszValue[128];
637 DWORD cbValue = sizeof(wszValue);
638 DWORD dwKeyType;
639 LONG status = RegQueryValueExW(hkeyVideo, L"Device Description", NULL, &dwKeyType, (LPBYTE)wszValue, &cbValue);
640 if (status == ERROR_SUCCESS)
641 {
642 /* WDDM has additional chars after "Adapter" */
643 static char s_szDeviceDescription[] = "VirtualBox Graphics Adapter";
644 wszValue[sizeof(s_szDeviceDescription) - 1] = '\0';
645 if (RTUtf16ICmpAscii(wszValue, s_szDeviceDescription) == 0)
646 return VINF_SUCCESS;
647 }
648
649 return VERR_NOT_FOUND;
650}
651
652static HKEY getVideoKey(bool writable)
653{
654 HKEY hkeyDeviceMap = 0;
655 LONG status = RegOpenKeyExA(HKEY_LOCAL_MACHINE, "HARDWARE\\DEVICEMAP\\VIDEO", 0, KEY_READ, &hkeyDeviceMap);
656 if (status != ERROR_SUCCESS || !hkeyDeviceMap)
657 {
658 VBoxControlError("Error opening video device map registry key!\n");
659 return 0;
660 }
661
662 HKEY hkeyVideo = 0;
663 ULONG iDevice;
664 DWORD dwKeyType;
665
666 /*
667 * Scan all '\Device\VideoX' REG_SZ keys to find VBox video driver entry.
668 * 'ObjectNumberList' REG_BINARY is an array of 32 bit device indexes (X).
669 */
670
671 /* Get the 'ObjectNumberList' */
672 ULONG cDevices = 0;
673 DWORD adwObjectNumberList[256];
674 DWORD cbValue = sizeof(adwObjectNumberList);
675 status = RegQueryValueExA(hkeyDeviceMap, "ObjectNumberList", NULL, &dwKeyType, (LPBYTE)&adwObjectNumberList[0], &cbValue);
676
677 if ( status == ERROR_SUCCESS
678 && dwKeyType == REG_BINARY)
679 cDevices = cbValue / sizeof(DWORD);
680 else
681 {
682 /* The list might not exists. Use 'MaxObjectNumber' REG_DWORD and build a list. */
683 DWORD dwMaxObjectNumber = 0;
684 cbValue = sizeof(dwMaxObjectNumber);
685 status = RegQueryValueExA(hkeyDeviceMap, "MaxObjectNumber", NULL, &dwKeyType, (LPBYTE)&dwMaxObjectNumber, &cbValue);
686 if ( status == ERROR_SUCCESS
687 && dwKeyType == REG_DWORD)
688 {
689 /* 'MaxObjectNumber' is inclusive. */
690 cDevices = RT_MIN(dwMaxObjectNumber + 1, RT_ELEMENTS(adwObjectNumberList));
691 for (iDevice = 0; iDevice < cDevices; iDevice++)
692 adwObjectNumberList[iDevice] = iDevice;
693 }
694 }
695
696 if (cDevices == 0)
697 {
698 /* Always try '\Device\Video0' as the old code did. Enum can be used in this case in principle. */
699 adwObjectNumberList[0] = 0;
700 cDevices = 1;
701 }
702
703 /* Scan device entries */
704 for (iDevice = 0; iDevice < cDevices; iDevice++)
705 {
706 char szValueName[64];
707 RTStrPrintf(szValueName, sizeof(szValueName), "\\Device\\Video%u", adwObjectNumberList[iDevice]);
708
709 char szVideoLocation[256];
710 cbValue = sizeof(szVideoLocation);
711 status = RegQueryValueExA(hkeyDeviceMap, szValueName, NULL, &dwKeyType, (LPBYTE)&szVideoLocation[0], &cbValue);
712
713 /* This value starts with '\REGISTRY\Machine' */
714 if ( status == ERROR_SUCCESS
715 && dwKeyType == REG_SZ
716 && _strnicmp(szVideoLocation, "\\REGISTRY\\Machine", 17) == 0)
717 {
718 status = RegOpenKeyExA(HKEY_LOCAL_MACHINE, &szVideoLocation[18], 0,
719 KEY_READ | (writable ? KEY_WRITE : 0), &hkeyVideo);
720 if (status == ERROR_SUCCESS)
721 {
722 int rc = checkVBoxVideoKey(hkeyVideo);
723 if (RT_SUCCESS(rc))
724 {
725 /* Found, return hkeyVideo to the caller. */
726 break;
727 }
728
729 RegCloseKey(hkeyVideo);
730 hkeyVideo = 0;
731 }
732 }
733 }
734
735 if (hkeyVideo == 0)
736 {
737 VBoxControlError("Error opening video registry key!\n");
738 }
739
740 RegCloseKey(hkeyDeviceMap);
741 return hkeyVideo;
742}
743
744static DECLCALLBACK(RTEXITCODE) handleGetVideoAcceleration(int argc, char *argv[])
745{
746 RT_NOREF2(argc, argv);
747 ULONG status;
748 HKEY hkeyVideo = getVideoKey(false);
749
750 if (hkeyVideo)
751 {
752 /* query the actual value */
753 DWORD fAcceleration = 1;
754 DWORD cbValue = sizeof(fAcceleration);
755 DWORD dwKeyType;
756 status = RegQueryValueExA(hkeyVideo, "EnableVideoAccel", NULL, &dwKeyType, (LPBYTE)&fAcceleration, &cbValue);
757 if (status != ERROR_SUCCESS)
758 RTPrintf("Video acceleration: default\n");
759 else
760 RTPrintf("Video acceleration: %s\n", fAcceleration ? "on" : "off");
761 RegCloseKey(hkeyVideo);
762 }
763 return RTEXITCODE_SUCCESS;
764}
765
766static DECLCALLBACK(RTEXITCODE) handleSetVideoAcceleration(int argc, char *argv[])
767{
768 ULONG status;
769 HKEY hkeyVideo;
770
771 /* must have exactly one argument: the new offset */
772 if ( (argc != 1)
773 || ( RTStrICmp(argv[0], "on")
774 && RTStrICmp(argv[0], "off")))
775 {
776 usage(SET_VIDEO_ACCEL);
777 return RTEXITCODE_FAILURE;
778 }
779
780 hkeyVideo = getVideoKey(true);
781
782 if (hkeyVideo)
783 {
784 int fAccel = 0;
785 if (RTStrICmp(argv[0], "on") == 0)
786 fAccel = 1;
787 /* set a new value */
788 status = RegSetValueExA(hkeyVideo, "EnableVideoAccel", 0, REG_DWORD, (LPBYTE)&fAccel, sizeof(fAccel));
789 if (status != ERROR_SUCCESS)
790 {
791 VBoxControlError("Error %d writing video acceleration status!\n", status);
792 }
793 RegCloseKey(hkeyVideo);
794 }
795 return RTEXITCODE_SUCCESS;
796}
797
798static DECLCALLBACK(RTEXITCODE) videoFlagsGet(void)
799{
800 HKEY hkeyVideo = getVideoKey(false);
801
802 if (hkeyVideo)
803 {
804 DWORD dwFlags = 0;
805 DWORD cbValue = sizeof(dwFlags);
806 DWORD dwKeyType;
807 ULONG status = RegQueryValueExA(hkeyVideo, "VBoxVideoFlags", NULL, &dwKeyType, (LPBYTE)&dwFlags, &cbValue);
808 if (status != ERROR_SUCCESS)
809 RTPrintf("Video flags: default\n");
810 else
811 RTPrintf("Video flags: 0x%08X\n", dwFlags);
812 RegCloseKey(hkeyVideo);
813 return RTEXITCODE_SUCCESS;
814 }
815
816 return RTEXITCODE_FAILURE;
817}
818
819static DECLCALLBACK(RTEXITCODE) videoFlagsDelete(void)
820{
821 HKEY hkeyVideo = getVideoKey(true);
822
823 if (hkeyVideo)
824 {
825 ULONG status = RegDeleteValueA(hkeyVideo, "VBoxVideoFlags");
826 if (status != ERROR_SUCCESS)
827 VBoxControlError("Error %d deleting video flags.\n", status);
828 RegCloseKey(hkeyVideo);
829 return RTEXITCODE_SUCCESS;
830 }
831
832 return RTEXITCODE_FAILURE;
833}
834
835static DECLCALLBACK(RTEXITCODE) videoFlagsModify(bool fSet, int argc, char *argv[])
836{
837 if (argc != 1)
838 {
839 VBoxControlError("Mask required.\n");
840 return RTEXITCODE_FAILURE;
841 }
842
843 uint32_t u32Mask = 0;
844 int rc = RTStrToUInt32Full(argv[0], 16, &u32Mask);
845 if (RT_FAILURE(rc))
846 {
847 VBoxControlError("Invalid video flags mask.\n");
848 return RTEXITCODE_FAILURE;
849 }
850
851 RTEXITCODE exitCode = RTEXITCODE_SUCCESS;
852
853 HKEY hkeyVideo = getVideoKey(true);
854 if (hkeyVideo)
855 {
856 DWORD dwFlags = 0;
857 DWORD cbValue = sizeof(dwFlags);
858 DWORD dwKeyType;
859 ULONG status = RegQueryValueExA(hkeyVideo, "VBoxVideoFlags", NULL, &dwKeyType, (LPBYTE)&dwFlags, &cbValue);
860 if (status != ERROR_SUCCESS)
861 dwFlags = 0;
862
863 dwFlags = fSet ? dwFlags | u32Mask : dwFlags & ~u32Mask;
864
865 status = RegSetValueExA(hkeyVideo, "VBoxVideoFlags", 0, REG_DWORD, (LPBYTE)&dwFlags, sizeof(dwFlags));
866 if (status != ERROR_SUCCESS)
867 {
868 VBoxControlError("Error %d writing video flags.\n", status);
869 exitCode = RTEXITCODE_FAILURE;
870 }
871
872 RegCloseKey(hkeyVideo);
873 }
874 else
875 {
876 exitCode = RTEXITCODE_FAILURE;
877 }
878
879 return exitCode;
880}
881
882static DECLCALLBACK(RTEXITCODE) handleVideoFlags(int argc, char *argv[])
883{
884 /* Must have a keyword and optional value (32 bit hex string). */
885 if (argc != 1 && argc != 2)
886 {
887 VBoxControlError("Invalid number of arguments.\n");
888 usage(VIDEO_FLAGS);
889 return RTEXITCODE_FAILURE;
890 }
891
892 RTEXITCODE exitCode = RTEXITCODE_SUCCESS;
893
894 if (RTStrICmp(argv[0], "get") == 0)
895 {
896 exitCode = videoFlagsGet();
897 }
898 else if (RTStrICmp(argv[0], "delete") == 0)
899 {
900 exitCode = videoFlagsDelete();
901 }
902 else if (RTStrICmp(argv[0], "set") == 0)
903 {
904 exitCode = videoFlagsModify(true, argc - 1, &argv[1]);
905 }
906 else if (RTStrICmp(argv[0], "clear") == 0)
907 {
908 exitCode = videoFlagsModify(false, argc - 1, &argv[1]);
909 }
910 else
911 {
912 VBoxControlError("Invalid command.\n");
913 exitCode = RTEXITCODE_FAILURE;
914 }
915
916 if (exitCode != RTEXITCODE_SUCCESS)
917 {
918 usage(VIDEO_FLAGS);
919 }
920
921 return exitCode;
922}
923
924#define MAX_CUSTOM_MODES 128
925
926/* the table of custom modes */
927struct
928{
929 DWORD xres;
930 DWORD yres;
931 DWORD bpp;
932} customModes[MAX_CUSTOM_MODES] = {0};
933
934void getCustomModes(HKEY hkeyVideo)
935{
936 ULONG status;
937 int curMode = 0;
938
939 /* null out the table */
940 RT_ZERO(customModes);
941
942 do
943 {
944 char valueName[20];
945 DWORD xres, yres, bpp = 0;
946 DWORD dwType;
947 DWORD dwLen = sizeof(DWORD);
948
949 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dWidth", curMode);
950 status = RegQueryValueExA(hkeyVideo, valueName, NULL, &dwType, (LPBYTE)&xres, &dwLen);
951 if (status != ERROR_SUCCESS)
952 break;
953 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dHeight", curMode);
954 status = RegQueryValueExA(hkeyVideo, valueName, NULL, &dwType, (LPBYTE)&yres, &dwLen);
955 if (status != ERROR_SUCCESS)
956 break;
957 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dBPP", curMode);
958 status = RegQueryValueExA(hkeyVideo, valueName, NULL, &dwType, (LPBYTE)&bpp, &dwLen);
959 if (status != ERROR_SUCCESS)
960 break;
961
962 /* check if the mode is OK */
963 if ( (xres > (1 << 16))
964 || (yres > (1 << 16))
965 || ( (bpp != 16)
966 && (bpp != 24)
967 && (bpp != 32)))
968 break;
969
970 /* add mode to table */
971 customModes[curMode].xres = xres;
972 customModes[curMode].yres = yres;
973 customModes[curMode].bpp = bpp;
974
975 ++curMode;
976
977 if (curMode >= MAX_CUSTOM_MODES)
978 break;
979 } while(1);
980}
981
982void writeCustomModes(HKEY hkeyVideo)
983{
984 ULONG status;
985 int tableIndex = 0;
986 int modeIndex = 0;
987
988 /* first remove all values */
989 for (int i = 0; i < MAX_CUSTOM_MODES; i++)
990 {
991 char valueName[20];
992 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dWidth", i);
993 RegDeleteValueA(hkeyVideo, valueName);
994 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dHeight", i);
995 RegDeleteValueA(hkeyVideo, valueName);
996 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dBPP", i);
997 RegDeleteValueA(hkeyVideo, valueName);
998 }
999
1000 do
1001 {
1002 if (tableIndex >= MAX_CUSTOM_MODES)
1003 break;
1004
1005 /* is the table entry present? */
1006 if ( (!customModes[tableIndex].xres)
1007 || (!customModes[tableIndex].yres)
1008 || (!customModes[tableIndex].bpp))
1009 {
1010 tableIndex++;
1011 continue;
1012 }
1013
1014 RTPrintf("writing mode %d (%dx%dx%d)\n", modeIndex, customModes[tableIndex].xres, customModes[tableIndex].yres, customModes[tableIndex].bpp);
1015 char valueName[20];
1016 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dWidth", modeIndex);
1017 status = RegSetValueExA(hkeyVideo, valueName, 0, REG_DWORD, (LPBYTE)&customModes[tableIndex].xres,
1018 sizeof(customModes[tableIndex].xres));
1019 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dHeight", modeIndex);
1020 RegSetValueExA(hkeyVideo, valueName, 0, REG_DWORD, (LPBYTE)&customModes[tableIndex].yres,
1021 sizeof(customModes[tableIndex].yres));
1022 RTStrPrintf(valueName, sizeof(valueName), "CustomMode%dBPP", modeIndex);
1023 RegSetValueExA(hkeyVideo, valueName, 0, REG_DWORD, (LPBYTE)&customModes[tableIndex].bpp,
1024 sizeof(customModes[tableIndex].bpp));
1025
1026 modeIndex++;
1027 tableIndex++;
1028
1029 } while(1);
1030
1031}
1032
1033static DECLCALLBACK(RTEXITCODE) handleListCustomModes(int argc, char *argv[])
1034{
1035 RT_NOREF1(argv);
1036 if (argc != 0)
1037 {
1038 usage(LIST_CUST_MODES);
1039 return RTEXITCODE_FAILURE;
1040 }
1041
1042 HKEY hkeyVideo = getVideoKey(false);
1043
1044 if (hkeyVideo)
1045 {
1046 getCustomModes(hkeyVideo);
1047 for (int i = 0; i < (sizeof(customModes) / sizeof(customModes[0])); i++)
1048 {
1049 if ( !customModes[i].xres
1050 || !customModes[i].yres
1051 || !customModes[i].bpp)
1052 continue;
1053
1054 RTPrintf("Mode: %d x %d x %d\n",
1055 customModes[i].xres, customModes[i].yres, customModes[i].bpp);
1056 }
1057 RegCloseKey(hkeyVideo);
1058 }
1059 return RTEXITCODE_SUCCESS;
1060}
1061
1062static DECLCALLBACK(RTEXITCODE) handleAddCustomMode(int argc, char *argv[])
1063{
1064 if (argc != 3)
1065 {
1066 usage(ADD_CUST_MODE);
1067 return RTEXITCODE_FAILURE;
1068 }
1069
1070 DWORD xres = atoi(argv[0]);
1071 DWORD yres = atoi(argv[1]);
1072 DWORD bpp = atoi(argv[2]);
1073
1074 /** @todo better check including xres mod 8 = 0! */
1075 if ( (xres > (1 << 16))
1076 || (yres > (1 << 16))
1077 || ( (bpp != 16)
1078 && (bpp != 24)
1079 && (bpp != 32)))
1080 {
1081 VBoxControlError("invalid mode specified!\n");
1082 return RTEXITCODE_FAILURE;
1083 }
1084
1085 HKEY hkeyVideo = getVideoKey(true);
1086
1087 if (hkeyVideo)
1088 {
1089 int i;
1090 int fModeExists = 0;
1091 getCustomModes(hkeyVideo);
1092 for (i = 0; i < MAX_CUSTOM_MODES; i++)
1093 {
1094 /* mode exists? */
1095 if ( customModes[i].xres == xres
1096 && customModes[i].yres == yres
1097 && customModes[i].bpp == bpp
1098 )
1099 {
1100 fModeExists = 1;
1101 }
1102 }
1103 if (!fModeExists)
1104 {
1105 for (i = 0; i < MAX_CUSTOM_MODES; i++)
1106 {
1107 /* item free? */
1108 if (!customModes[i].xres)
1109 {
1110 customModes[i].xres = xres;
1111 customModes[i].yres = yres;
1112 customModes[i].bpp = bpp;
1113 break;
1114 }
1115 }
1116 writeCustomModes(hkeyVideo);
1117 }
1118 RegCloseKey(hkeyVideo);
1119 }
1120 return RTEXITCODE_SUCCESS;
1121}
1122
1123static DECLCALLBACK(RTEXITCODE) handleRemoveCustomMode(int argc, char *argv[])
1124{
1125 if (argc != 3)
1126 {
1127 usage(REMOVE_CUST_MODE);
1128 return RTEXITCODE_FAILURE;
1129 }
1130
1131 DWORD xres = atoi(argv[0]);
1132 DWORD yres = atoi(argv[1]);
1133 DWORD bpp = atoi(argv[2]);
1134
1135 HKEY hkeyVideo = getVideoKey(true);
1136
1137 if (hkeyVideo)
1138 {
1139 getCustomModes(hkeyVideo);
1140 for (int i = 0; i < MAX_CUSTOM_MODES; i++)
1141 {
1142 /* correct item? */
1143 if ( (customModes[i].xres == xres)
1144 && (customModes[i].yres == yres)
1145 && (customModes[i].bpp == bpp))
1146 {
1147 RTPrintf("found mode at index %d\n", i);
1148 RT_ZERO(customModes[i]);
1149 break;
1150 }
1151 }
1152 writeCustomModes(hkeyVideo);
1153 RegCloseKey(hkeyVideo);
1154 }
1155
1156 return RTEXITCODE_SUCCESS;
1157}
1158
1159#endif /* RT_OS_WINDOWS */
1160
1161#ifdef VBOX_WITH_GUEST_PROPS
1162/**
1163 * Retrieves a value from the guest property store.
1164 * This is accessed through the "VBoxGuestPropSvc" HGCM service.
1165 *
1166 * @returns Command exit code.
1167 * @note see the command line API description for parameters
1168 */
1169static RTEXITCODE getGuestProperty(int argc, char **argv)
1170{
1171 bool fVerbose = false;
1172 if ( argc == 2
1173 && ( strcmp(argv[1], "-verbose") == 0
1174 || strcmp(argv[1], "--verbose") == 0)
1175 )
1176 fVerbose = true;
1177 else if (argc != 1)
1178 {
1179 usage(GUEST_PROP);
1180 return RTEXITCODE_FAILURE;
1181 }
1182
1183 uint32_t u32ClientId = 0;
1184 int rc = VINF_SUCCESS;
1185
1186 rc = VbglR3GuestPropConnect(&u32ClientId);
1187 if (RT_FAILURE(rc))
1188 VBoxControlError("Failed to connect to the guest property service, error %Rrc\n", rc);
1189
1190 /*
1191 * Here we actually retrieve the value from the host.
1192 */
1193 const char *pszName = argv[0];
1194 char *pszValue = NULL;
1195 uint64_t u64Timestamp = 0;
1196 char *pszFlags = NULL;
1197 /* The buffer for storing the data and its initial size. We leave a bit
1198 * of space here in case the maximum values are raised. */
1199 void *pvBuf = NULL;
1200 uint32_t cbBuf = GUEST_PROP_MAX_VALUE_LEN + GUEST_PROP_MAX_FLAGS_LEN + 1024;
1201 if (RT_SUCCESS(rc))
1202 {
1203 /* Because there is a race condition between our reading the size of a
1204 * property and the guest updating it, we loop a few times here and
1205 * hope. Actually this should never go wrong, as we are generous
1206 * enough with buffer space. */
1207 bool fFinished = false;
1208 for (unsigned i = 0; i < 10 && !fFinished; ++i)
1209 {
1210 void *pvTmpBuf = RTMemRealloc(pvBuf, cbBuf);
1211 if (NULL == pvTmpBuf)
1212 {
1213 rc = VERR_NO_MEMORY;
1214 VBoxControlError("Out of memory\n");
1215 }
1216 else
1217 {
1218 pvBuf = pvTmpBuf;
1219 rc = VbglR3GuestPropRead(u32ClientId, pszName, pvBuf, cbBuf,
1220 &pszValue, &u64Timestamp, &pszFlags,
1221 &cbBuf);
1222 }
1223 if (VERR_BUFFER_OVERFLOW == rc)
1224 /* Leave a bit of extra space to be safe */
1225 cbBuf += 1024;
1226 else
1227 fFinished = true;
1228 }
1229 if (VERR_TOO_MUCH_DATA == rc)
1230 VBoxControlError("Temporarily unable to retrieve the property\n");
1231 else if (RT_FAILURE(rc) && rc != VERR_NOT_FOUND)
1232 VBoxControlError("Failed to retrieve the property value, error %Rrc\n", rc);
1233 }
1234
1235 /*
1236 * And display it on the guest console.
1237 */
1238 if (VERR_NOT_FOUND == rc)
1239 RTPrintf("No value set!\n");
1240 else if (RT_SUCCESS(rc))
1241 {
1242 RTPrintf("Value: %s\n", pszValue);
1243 if (fVerbose)
1244 {
1245 RTPrintf("Timestamp: %lld ns\n", u64Timestamp);
1246 RTPrintf("Flags: %s\n", pszFlags);
1247 }
1248 }
1249
1250 if (u32ClientId != 0)
1251 VbglR3GuestPropDisconnect(u32ClientId);
1252 RTMemFree(pvBuf);
1253 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1254}
1255
1256
1257/**
1258 * Writes a value to the guest property store.
1259 * This is accessed through the "VBoxGuestPropSvc" HGCM service.
1260 *
1261 * @returns Command exit code.
1262 * @note see the command line API description for parameters
1263 */
1264static RTEXITCODE setGuestProperty(int argc, char *argv[])
1265{
1266 /*
1267 * Check the syntax. We can deduce the correct syntax from the number of
1268 * arguments.
1269 */
1270 bool fUsageOK = true;
1271 const char *pszName = NULL;
1272 const char *pszValue = NULL;
1273 const char *pszFlags = NULL;
1274 if (2 == argc)
1275 {
1276 pszValue = argv[1];
1277 }
1278 else if (3 == argc)
1279 fUsageOK = false;
1280 else if (4 == argc)
1281 {
1282 pszValue = argv[1];
1283 if ( strcmp(argv[2], "-flags") != 0
1284 && strcmp(argv[2], "--flags") != 0)
1285 fUsageOK = false;
1286 pszFlags = argv[3];
1287 }
1288 else if (argc != 1)
1289 fUsageOK = false;
1290 if (!fUsageOK)
1291 {
1292 usage(GUEST_PROP);
1293 return RTEXITCODE_FAILURE;
1294 }
1295 /* This is always needed. */
1296 pszName = argv[0];
1297
1298 /*
1299 * Do the actual setting.
1300 */
1301 uint32_t u32ClientId = 0;
1302 int rc = VINF_SUCCESS;
1303 rc = VbglR3GuestPropConnect(&u32ClientId);
1304 if (RT_FAILURE(rc))
1305 VBoxControlError("Failed to connect to the guest property service, error %Rrc\n", rc);
1306 else
1307 {
1308 if (pszFlags != NULL)
1309 rc = VbglR3GuestPropWrite(u32ClientId, pszName, pszValue, pszFlags);
1310 else
1311 rc = VbglR3GuestPropWriteValue(u32ClientId, pszName, pszValue);
1312 if (RT_FAILURE(rc))
1313 VBoxControlError("Failed to store the property value, error %Rrc\n", rc);
1314 }
1315
1316 if (u32ClientId != 0)
1317 VbglR3GuestPropDisconnect(u32ClientId);
1318 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1319}
1320
1321
1322/**
1323 * Deletes a guest property from the guest property store.
1324 * This is accessed through the "VBoxGuestPropSvc" HGCM service.
1325 *
1326 * @returns Command exit code.
1327 * @note see the command line API description for parameters
1328 */
1329static RTEXITCODE deleteGuestProperty(int argc, char *argv[])
1330{
1331 /*
1332 * Check the syntax. We can deduce the correct syntax from the number of
1333 * arguments.
1334 */
1335 bool fUsageOK = true;
1336 const char *pszName = NULL;
1337 if (argc < 1)
1338 fUsageOK = false;
1339 if (!fUsageOK)
1340 {
1341 usage(GUEST_PROP);
1342 return RTEXITCODE_FAILURE;
1343 }
1344 /* This is always needed. */
1345 pszName = argv[0];
1346
1347 /*
1348 * Do the actual setting.
1349 */
1350 uint32_t u32ClientId = 0;
1351 int rc = VbglR3GuestPropConnect(&u32ClientId);
1352 if (RT_FAILURE(rc))
1353 VBoxControlError("Failed to connect to the guest property service, error %Rrc\n", rc);
1354 else
1355 {
1356 rc = VbglR3GuestPropDelete(u32ClientId, pszName);
1357 if (RT_FAILURE(rc))
1358 VBoxControlError("Failed to delete the property value, error %Rrc\n", rc);
1359 }
1360
1361 if (u32ClientId != 0)
1362 VbglR3GuestPropDisconnect(u32ClientId);
1363 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1364}
1365
1366
1367/**
1368 * Enumerates the properties in the guest property store.
1369 * This is accessed through the "VBoxGuestPropSvc" HGCM service.
1370 *
1371 * @returns Command exit code.
1372 * @note see the command line API description for parameters
1373 */
1374static RTEXITCODE enumGuestProperty(int argc, char *argv[])
1375{
1376 /*
1377 * Check the syntax. We can deduce the correct syntax from the number of
1378 * arguments.
1379 */
1380 char const * const *papszPatterns = NULL;
1381 uint32_t cPatterns = 0;
1382 if ( argc > 1
1383 && ( strcmp(argv[0], "-patterns") == 0
1384 || strcmp(argv[0], "--patterns") == 0))
1385 {
1386 papszPatterns = (char const * const *)&argv[1];
1387 cPatterns = argc - 1;
1388 }
1389 else if (argc != 0)
1390 {
1391 usage(GUEST_PROP);
1392 return RTEXITCODE_FAILURE;
1393 }
1394
1395 /*
1396 * Do the actual enumeration.
1397 */
1398 uint32_t u32ClientId = 0;
1399 int rc = VbglR3GuestPropConnect(&u32ClientId);
1400 if (RT_SUCCESS(rc))
1401 {
1402 PVBGLR3GUESTPROPENUM pHandle;
1403 const char *pszName, *pszValue, *pszFlags;
1404 uint64_t u64Timestamp;
1405
1406 rc = VbglR3GuestPropEnum(u32ClientId, papszPatterns, cPatterns, &pHandle,
1407 &pszName, &pszValue, &u64Timestamp, &pszFlags);
1408 if (RT_SUCCESS(rc))
1409 {
1410 while (RT_SUCCESS(rc) && pszName)
1411 {
1412 RTPrintf("Name: %s, value: %s, timestamp: %lld, flags: %s\n",
1413 pszName, pszValue, u64Timestamp, pszFlags);
1414
1415 rc = VbglR3GuestPropEnumNext(pHandle, &pszName, &pszValue, &u64Timestamp, &pszFlags);
1416 if (RT_FAILURE(rc))
1417 VBoxControlError("Error while enumerating guest properties: %Rrc\n", rc);
1418 }
1419
1420 VbglR3GuestPropEnumFree(pHandle);
1421 }
1422 else if (VERR_NOT_FOUND == rc)
1423 RTPrintf("No properties found.\n");
1424 else
1425 VBoxControlError("Failed to enumerate the guest properties! Error: %Rrc\n", rc);
1426 VbglR3GuestPropDisconnect(u32ClientId);
1427 }
1428 else
1429 VBoxControlError("Failed to connect to the guest property service! Error: %Rrc\n", rc);
1430 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1431}
1432
1433
1434/**
1435 * Waits for notifications of changes to guest properties.
1436 * This is accessed through the "VBoxGuestPropSvc" HGCM service.
1437 *
1438 * @returns Command exit code.
1439 * @note see the command line API description for parameters
1440 */
1441static RTEXITCODE waitGuestProperty(int argc, char **argv)
1442{
1443 /*
1444 * Handle arguments
1445 */
1446 const char *pszPatterns = NULL;
1447 uint64_t u64TimestampIn = 0;
1448 uint32_t u32Timeout = RT_INDEFINITE_WAIT;
1449 bool fUsageOK = true;
1450 if (argc < 1)
1451 fUsageOK = false;
1452 pszPatterns = argv[0];
1453 for (int i = 1; fUsageOK && i < argc; ++i)
1454 {
1455 if ( strcmp(argv[i], "-timeout") == 0
1456 || strcmp(argv[i], "--timeout") == 0)
1457 {
1458 if ( i + 1 >= argc
1459 || RTStrToUInt32Full(argv[i + 1], 10, &u32Timeout)
1460 != VINF_SUCCESS
1461 )
1462 fUsageOK = false;
1463 else
1464 ++i;
1465 }
1466 else if ( strcmp(argv[i], "-timestamp") == 0
1467 || strcmp(argv[i], "--timestamp") == 0)
1468 {
1469 if ( i + 1 >= argc
1470 || RTStrToUInt64Full(argv[i + 1], 10, &u64TimestampIn)
1471 != VINF_SUCCESS
1472 )
1473 fUsageOK = false;
1474 else
1475 ++i;
1476 }
1477 else
1478 fUsageOK = false;
1479 }
1480 if (!fUsageOK)
1481 {
1482 usage(GUEST_PROP);
1483 return RTEXITCODE_FAILURE;
1484 }
1485
1486 /*
1487 * Connect to the service
1488 */
1489 uint32_t u32ClientId = 0;
1490 int rc = VINF_SUCCESS;
1491
1492 rc = VbglR3GuestPropConnect(&u32ClientId);
1493 if (RT_FAILURE(rc))
1494 VBoxControlError("Failed to connect to the guest property service, error %Rrc\n", rc);
1495
1496 /*
1497 * Retrieve the notification from the host
1498 */
1499 char *pszName = NULL;
1500 char *pszValue = NULL;
1501 uint64_t u64TimestampOut = 0;
1502 char *pszFlags = NULL;
1503 /* The buffer for storing the data and its initial size. We leave a bit
1504 * of space here in case the maximum values are raised. */
1505 void *pvBuf = NULL;
1506 uint32_t cbBuf = GUEST_PROP_MAX_NAME_LEN + GUEST_PROP_MAX_VALUE_LEN + GUEST_PROP_MAX_FLAGS_LEN + 1024;
1507 /* Because there is a race condition between our reading the size of a
1508 * property and the guest updating it, we loop a few times here and
1509 * hope. Actually this should never go wrong, as we are generous
1510 * enough with buffer space. */
1511 bool fFinished = false;
1512 for (unsigned i = 0; (RT_SUCCESS(rc) || rc == VERR_BUFFER_OVERFLOW) && !fFinished && i < 10; i++)
1513 {
1514 void *pvTmpBuf = RTMemRealloc(pvBuf, cbBuf);
1515 if (NULL == pvTmpBuf)
1516 {
1517 rc = VERR_NO_MEMORY;
1518 VBoxControlError("Out of memory\n");
1519 }
1520 else
1521 {
1522 pvBuf = pvTmpBuf;
1523 rc = VbglR3GuestPropWait(u32ClientId, pszPatterns, pvBuf, cbBuf,
1524 u64TimestampIn, u32Timeout,
1525 &pszName, &pszValue, &u64TimestampOut,
1526 &pszFlags, &cbBuf);
1527 }
1528 if (VERR_BUFFER_OVERFLOW == rc)
1529 /* Leave a bit of extra space to be safe */
1530 cbBuf += 1024;
1531 else
1532 fFinished = true;
1533 if (rc == VERR_TOO_MUCH_DATA)
1534 VBoxControlError("Temporarily unable to get a notification\n");
1535 else if (rc == VERR_INTERRUPTED)
1536 VBoxControlError("The request timed out or was interrupted\n");
1537#ifndef RT_OS_WINDOWS /* Windows guests do not do this right */
1538 else if (RT_FAILURE(rc) && rc != VERR_NOT_FOUND)
1539 VBoxControlError("Failed to get a notification, error %Rrc\n", rc);
1540#endif
1541 }
1542
1543 /*
1544 * And display it on the guest console.
1545 */
1546 if (VERR_NOT_FOUND == rc)
1547 RTPrintf("No value set!\n");
1548 else if (rc == VERR_BUFFER_OVERFLOW)
1549 RTPrintf("Internal error: unable to determine the size of the data!\n");
1550 else if (RT_SUCCESS(rc))
1551 {
1552 RTPrintf("Name: %s\n", pszName);
1553 RTPrintf("Value: %s\n", pszValue);
1554 RTPrintf("Timestamp: %lld ns\n", u64TimestampOut);
1555 RTPrintf("Flags: %s\n", pszFlags);
1556 }
1557
1558 if (u32ClientId != 0)
1559 VbglR3GuestPropDisconnect(u32ClientId);
1560 RTMemFree(pvBuf);
1561 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1562}
1563
1564
1565/**
1566 * Access the guest property store through the "VBoxGuestPropSvc" HGCM
1567 * service.
1568 *
1569 * @returns 0 on success, 1 on failure
1570 * @note see the command line API description for parameters
1571 */
1572static DECLCALLBACK(RTEXITCODE) handleGuestProperty(int argc, char *argv[])
1573{
1574 if (argc == 0)
1575 {
1576 usage(GUEST_PROP);
1577 return RTEXITCODE_FAILURE;
1578 }
1579 if (!strcmp(argv[0], "get"))
1580 return getGuestProperty(argc - 1, argv + 1);
1581 if (!strcmp(argv[0], "set"))
1582 return setGuestProperty(argc - 1, argv + 1);
1583 if (!strcmp(argv[0], "delete") || !strcmp(argv[0], "unset"))
1584 return deleteGuestProperty(argc - 1, argv + 1);
1585 if (!strcmp(argv[0], "enumerate"))
1586 return enumGuestProperty(argc - 1, argv + 1);
1587 if (!strcmp(argv[0], "wait"))
1588 return waitGuestProperty(argc - 1, argv + 1);
1589 /* unknown cmd */
1590 usage(GUEST_PROP);
1591 return RTEXITCODE_FAILURE;
1592}
1593#endif
1594
1595#ifdef VBOX_WITH_SHARED_FOLDERS
1596/**
1597 * Lists the Shared Folders provided by the host.
1598 */
1599static RTEXITCODE listSharedFolders(int argc, char **argv)
1600{
1601 bool fUsageOK = true;
1602 bool fOnlyShowAutoMount = false;
1603 if (argc == 1)
1604 {
1605 if ( !strcmp(argv[0], "-automount")
1606 || !strcmp(argv[0], "--automount"))
1607 fOnlyShowAutoMount = true;
1608 else
1609 fUsageOK = false;
1610 }
1611 else if (argc > 1)
1612 fUsageOK = false;
1613 if (!fUsageOK)
1614 {
1615 usage(GUEST_SHAREDFOLDERS);
1616 return RTEXITCODE_FAILURE;
1617 }
1618
1619 uint32_t u32ClientId;
1620 int rc = VbglR3SharedFolderConnect(&u32ClientId);
1621 if (RT_FAILURE(rc))
1622 VBoxControlError("Failed to connect to the shared folder service, error %Rrc\n", rc);
1623 else
1624 {
1625 PVBGLR3SHAREDFOLDERMAPPING paMappings;
1626 uint32_t cMappings;
1627 rc = VbglR3SharedFolderGetMappings(u32ClientId, fOnlyShowAutoMount, &paMappings, &cMappings);
1628 if (RT_SUCCESS(rc))
1629 {
1630 if (fOnlyShowAutoMount)
1631 RTPrintf("Auto-mounted Shared Folder mappings (%u):\n\n", cMappings);
1632 else
1633 RTPrintf("Shared Folder mappings (%u):\n\n", cMappings);
1634
1635 for (uint32_t i = 0; i < cMappings; i++)
1636 {
1637 char *pszName;
1638 rc = VbglR3SharedFolderGetName(u32ClientId, paMappings[i].u32Root, &pszName);
1639 if (RT_SUCCESS(rc))
1640 {
1641 RTPrintf("%02u - %s\n", i + 1, pszName);
1642 RTStrFree(pszName);
1643 }
1644 else
1645 VBoxControlError("Error while getting the shared folder name for root node = %u, rc = %Rrc\n",
1646 paMappings[i].u32Root, rc);
1647 }
1648 if (!cMappings)
1649 RTPrintf("No Shared Folders available.\n");
1650 VbglR3SharedFolderFreeMappings(paMappings);
1651 }
1652 else
1653 VBoxControlError("Error while getting the shared folder mappings, rc = %Rrc\n", rc);
1654 VbglR3SharedFolderDisconnect(u32ClientId);
1655 }
1656 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1657}
1658
1659/**
1660 * Handles Shared Folders control.
1661 *
1662 * @returns 0 on success, 1 on failure
1663 * @note see the command line API description for parameters
1664 * (r=bird: yeah, right. The API description contains nil about params)
1665 */
1666static DECLCALLBACK(RTEXITCODE) handleSharedFolder(int argc, char *argv[])
1667{
1668 if (argc == 0)
1669 {
1670 usage(GUEST_SHAREDFOLDERS);
1671 return RTEXITCODE_FAILURE;
1672 }
1673 if (!strcmp(argv[0], "list"))
1674 return listSharedFolders(argc - 1, argv + 1);
1675 /* else */
1676 usage(GUEST_SHAREDFOLDERS);
1677 return RTEXITCODE_FAILURE;
1678}
1679#endif
1680
1681#if !defined(VBOX_CONTROL_TEST)
1682/**
1683 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: writecoredump}
1684 */
1685static DECLCALLBACK(RTEXITCODE) handleWriteCoreDump(int argc, char *argv[])
1686{
1687 RT_NOREF2(argc, argv);
1688 int rc = VbglR3WriteCoreDump();
1689 if (RT_SUCCESS(rc))
1690 {
1691 RTPrintf("Guest core dump successful.\n");
1692 return RTEXITCODE_SUCCESS;
1693 }
1694 else
1695 {
1696 VBoxControlError("Error while taking guest core dump. rc=%Rrc\n", rc);
1697 return RTEXITCODE_FAILURE;
1698 }
1699}
1700#endif
1701
1702#ifdef VBOX_WITH_DPC_LATENCY_CHECKER
1703/**
1704 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: help}
1705 */
1706static DECLCALLBACK(RTEXITCODE) handleDpc(int argc, char *argv[])
1707{
1708 RT_NOREF(argc, argv);
1709 int rc = VERR_NOT_IMPLEMENTED;
1710# ifndef VBOX_CONTROL_TEST
1711 for (int i = 0; i < 30; i++)
1712 {
1713 VBGLREQHDR Req;
1714 VBGLREQHDR_INIT(&Req, DPC_LATENCY_CHECKER);
1715 rc = vbglR3DoIOCtl(VBGL_IOCTL_DPC_LATENCY_CHECKER, &Req, sizeof(Req));
1716 if (RT_SUCCESS(rc))
1717 RTPrintf("%d\n", i);
1718 else
1719 break;
1720 }
1721# endif
1722 if (RT_FAILURE(rc))
1723 return VBoxControlError("Error. rc=%Rrc\n", rc);
1724 RTPrintf("Samples collection completed.\n");
1725 return RTEXITCODE_SUCCESS;
1726}
1727#endif /* VBOX_WITH_DPC_LATENCY_CHECKER */
1728
1729
1730/**
1731 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: writelog}
1732 */
1733static DECLCALLBACK(RTEXITCODE) handleWriteLog(int argc, char *argv[])
1734{
1735 static const RTGETOPTDEF s_aOptions[] =
1736 {
1737 { "--no-newline", 'n', RTGETOPT_REQ_NOTHING },
1738 };
1739 bool fNoNewline = false;
1740
1741 RTGETOPTSTATE GetOptState;
1742 int rc = RTGetOptInit(&GetOptState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions),
1743 0 /*iFirst*/, RTGETOPTINIT_FLAGS_OPTS_FIRST);
1744 if (RT_SUCCESS(rc))
1745 {
1746 RTGETOPTUNION ValueUnion;
1747 int ch;
1748 while ((ch = RTGetOpt(&GetOptState, &ValueUnion)) != 0)
1749 {
1750 switch (ch)
1751 {
1752 case VINF_GETOPT_NOT_OPTION:
1753 {
1754 size_t cch = strlen(ValueUnion.psz);
1755 if ( fNoNewline
1756 || (cch > 0 && ValueUnion.psz[cch - 1] == '\n') )
1757 rc = VbglR3WriteLog(ValueUnion.psz, cch);
1758 else
1759 {
1760 char *pszDup = (char *)RTMemDupEx(ValueUnion.psz, cch, 2);
1761 if (RT_SUCCESS(rc))
1762 {
1763 pszDup[cch++] = '\n';
1764 pszDup[cch] = '\0';
1765 rc = VbglR3WriteLog(pszDup, cch);
1766 RTMemFree(pszDup);
1767 }
1768 else
1769 rc = VERR_NO_MEMORY;
1770 }
1771 if (RT_FAILURE(rc))
1772 return VBoxControlError("VbglR3WriteLog: %Rrc", rc);
1773 break;
1774 }
1775
1776 case 'n':
1777 fNoNewline = true;
1778 break;
1779
1780 case 'h': return usage(WRITE_LOG);
1781 case 'V': return printVersion();
1782 default:
1783 return VBoxCtrlGetOptError(ch, &ValueUnion);
1784 }
1785 }
1786 }
1787 else
1788 return VBoxControlError("RTGetOptInit: %Rrc", rc);
1789 return RTEXITCODE_SUCCESS;
1790}
1791
1792
1793/**
1794 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: takesnapshot}
1795 */
1796static DECLCALLBACK(RTEXITCODE) handleTakeSnapshot(int argc, char *argv[])
1797{
1798 RT_NOREF2(argc, argv); //VbglR3VmTakeSnapshot(argv[0], argv[1]);
1799 return VBoxControlError("not implemented");
1800}
1801
1802/**
1803 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: savestate}
1804 */
1805static DECLCALLBACK(RTEXITCODE) handleSaveState(int argc, char *argv[])
1806{
1807 RT_NOREF2(argc, argv); //VbglR3VmSaveState();
1808 return VBoxControlError("not implemented");
1809}
1810
1811/**
1812 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: suspend|pause}
1813 */
1814static DECLCALLBACK(RTEXITCODE) handleSuspend(int argc, char *argv[])
1815{
1816 RT_NOREF2(argc, argv); //VbglR3VmSuspend();
1817 return VBoxControlError("not implemented");
1818}
1819
1820/**
1821 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: poweroff|powerdown}
1822 */
1823static DECLCALLBACK(RTEXITCODE) handlePowerOff(int argc, char *argv[])
1824{
1825 RT_NOREF2(argc, argv); //VbglR3VmPowerOff();
1826 return VBoxControlError("not implemented");
1827}
1828
1829/**
1830 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: version}
1831 */
1832static DECLCALLBACK(RTEXITCODE) handleVersion(int argc, char *argv[])
1833{
1834 RT_NOREF1(argv);
1835 if (argc)
1836 return VBoxControlSyntaxError("getversion does not take any arguments");
1837 return printVersion();
1838}
1839
1840/**
1841 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: help}
1842 */
1843static DECLCALLBACK(RTEXITCODE) handleHelp(int argc, char *argv[])
1844{
1845 RT_NOREF2(argc, argv); /* ignore arguments for now. */
1846 usage();
1847 return RTEXITCODE_SUCCESS;
1848}
1849
1850/**
1851 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: ls}
1852 */
1853static DECLCALLBACK(RTEXITCODE) handleLs(int argc, char *argv[])
1854{
1855 return RTFsCmdLs(argc + 1, argv - 1);
1856}
1857
1858/**
1859 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: tar}
1860 */
1861static DECLCALLBACK(RTEXITCODE) handleTar(int argc, char *argv[])
1862{
1863 return RTZipTarCmd(argc + 1, argv - 1);
1864}
1865
1866/**
1867 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: tar}
1868 */
1869static DECLCALLBACK(RTEXITCODE) handleGzip(int argc, char *argv[])
1870{
1871 return RTZipGzipCmd(argc + 1, argv - 1);
1872}
1873
1874/**
1875 * @callback_method_impl{FNVBOXCTRLCMDHANDLER, Command: unzip}
1876 */
1877static DECLCALLBACK(RTEXITCODE) handleUnzip(int argc, char *argv[])
1878{
1879 return RTZipUnzipCmd(argc + 1, argv - 1);
1880}
1881
1882
1883/** command handler type */
1884typedef DECLCALLBACK(RTEXITCODE) FNVBOXCTRLCMDHANDLER(int argc, char *argv[]);
1885typedef FNVBOXCTRLCMDHANDLER *PFNVBOXCTRLCMDHANDLER;
1886
1887/** The table of all registered command handlers. */
1888struct COMMANDHANDLER
1889{
1890 const char *pszCommand;
1891 PFNVBOXCTRLCMDHANDLER pfnHandler;
1892 bool fNeedDevice;
1893} g_aCommandHandlers[] =
1894{
1895#if defined(RT_OS_WINDOWS) && !defined(VBOX_CONTROL_TEST)
1896 { "getvideoacceleration", handleGetVideoAcceleration, true },
1897 { "setvideoacceleration", handleSetVideoAcceleration, true },
1898 { "videoflags", handleVideoFlags, true },
1899 { "listcustommodes", handleListCustomModes, true },
1900 { "addcustommode", handleAddCustomMode, true },
1901 { "removecustommode", handleRemoveCustomMode, true },
1902 { "setvideomode", handleSetVideoMode, true },
1903#endif
1904#ifdef VBOX_WITH_GUEST_PROPS
1905 { "guestproperty", handleGuestProperty, true },
1906#endif
1907#ifdef VBOX_WITH_SHARED_FOLDERS
1908 { "sharedfolder", handleSharedFolder, true },
1909#endif
1910#if !defined(VBOX_CONTROL_TEST)
1911 { "writecoredump", handleWriteCoreDump, true },
1912#endif
1913#ifdef VBOX_WITH_DPC_LATENCY_CHECKER
1914 { "dpc", handleDpc, true },
1915#endif
1916 { "writelog", handleWriteLog, true },
1917 { "takesnapshot", handleTakeSnapshot, true },
1918 { "savestate", handleSaveState, true },
1919 { "suspend", handleSuspend, true },
1920 { "pause", handleSuspend, true },
1921 { "poweroff", handlePowerOff, true },
1922 { "powerdown", handlePowerOff, true },
1923 { "getversion", handleVersion, false },
1924 { "version", handleVersion, false },
1925 { "help", handleHelp, false },
1926 /* Hany tricks that doesn't cost much space: */
1927 { "gzip", handleGzip, false },
1928 { "ls", handleLs, false },
1929 { "tar", handleTar, false },
1930 { "unzip", handleUnzip, false },
1931};
1932
1933/** Main function */
1934int main(int argc, char **argv)
1935{
1936 /** The application's global return code */
1937 RTEXITCODE rcExit = RTEXITCODE_SUCCESS;
1938 /** An IPRT return code for local use */
1939 int rrc = VINF_SUCCESS;
1940 /** The index of the command line argument we are currently processing */
1941 int iArg = 1;
1942 /** Should we show the logo text? */
1943 bool fShowLogo = true;
1944 /** Should we print the usage after the logo? For the -help switch. */
1945 bool fDoHelp = false;
1946 /** Will we be executing a command or just printing information? */
1947 bool fOnlyInfo = false;
1948
1949 rrc = RTR3InitExe(argc, &argv, 0);
1950 if (RT_FAILURE(rrc))
1951 return RTMsgInitFailure(rrc);
1952
1953 /*
1954 * Start by handling command line switches
1955 */
1956 /** @todo RTGetOpt conversion of the whole file. */
1957 bool done = false; /**< Are we finished with handling switches? */
1958 while (!done && (iArg < argc))
1959 {
1960 if ( !strcmp(argv[iArg], "-V")
1961 || !strcmp(argv[iArg], "-v")
1962 || !strcmp(argv[iArg], "--version")
1963 || !strcmp(argv[iArg], "-version")
1964 )
1965 {
1966 /* Print version number, and do nothing else. */
1967 printVersion();
1968 fOnlyInfo = true;
1969 fShowLogo = false;
1970 done = true;
1971 }
1972 else if ( !strcmp(argv[iArg], "-nologo")
1973 || !strcmp(argv[iArg], "--nologo"))
1974 fShowLogo = false;
1975 else if ( !strcmp(argv[iArg], "-help")
1976 || !strcmp(argv[iArg], "--help"))
1977 {
1978 fOnlyInfo = true;
1979 fDoHelp = true;
1980 done = true;
1981 }
1982 else
1983 /* We have found an argument which isn't a switch. Exit to the
1984 * command processing bit. */
1985 done = true;
1986 if (!done)
1987 ++iArg;
1988 }
1989
1990 /*
1991 * Find the application name, show our logo if the user hasn't suppressed it,
1992 * and show the usage if the user asked us to
1993 */
1994 g_pszProgName = RTPathFilename(argv[0]);
1995 if (fShowLogo)
1996 RTPrintf(VBOX_PRODUCT " Guest Additions Command Line Management Interface Version "
1997 VBOX_VERSION_STRING "\n"
1998 "(C) 2008-" VBOX_C_YEAR " " VBOX_VENDOR "\n"
1999 "All rights reserved.\n\n");
2000 if (fDoHelp)
2001 usage();
2002
2003 /*
2004 * Now look for an actual command in the argument list and handle it.
2005 */
2006 if (!fOnlyInfo && rcExit == RTEXITCODE_SUCCESS)
2007 {
2008 if (argc > iArg)
2009 {
2010 /*
2011 * Try locate the command and execute it, complain if not found.
2012 */
2013 unsigned i;
2014 for (i = 0; i < RT_ELEMENTS(g_aCommandHandlers); i++)
2015 if (!strcmp(argv[iArg], g_aCommandHandlers[i].pszCommand))
2016 {
2017 if (g_aCommandHandlers[i].fNeedDevice)
2018 {
2019 rrc = VbglR3Init();
2020 if (RT_FAILURE(rrc))
2021 {
2022 VBoxControlError("Could not contact the host system. Make sure that you are running this\n"
2023 "application inside a VirtualBox guest system, and that you have sufficient\n"
2024 "user permissions.\n");
2025 rcExit = RTEXITCODE_FAILURE;
2026 }
2027 }
2028 if (rcExit == RTEXITCODE_SUCCESS)
2029 rcExit = g_aCommandHandlers[i].pfnHandler(argc - iArg - 1, argv + iArg + 1);
2030 break;
2031 }
2032 if (i >= RT_ELEMENTS(g_aCommandHandlers))
2033 {
2034 usage();
2035 rcExit = RTEXITCODE_SYNTAX;
2036 }
2037 }
2038 else
2039 {
2040 /* The user didn't specify a command. */
2041 usage();
2042 rcExit = RTEXITCODE_SYNTAX;
2043 }
2044 }
2045
2046 /*
2047 * And exit, returning the status
2048 */
2049 return rcExit;
2050}
2051
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