VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxHeadless/VBoxHeadless.cpp@ 21216

Last change on this file since 21216 was 20928, checked in by vboxsync, 16 years ago

API/others: Renamed IConsole::discardSavedState to IConsole::forgetSavedState, added parameter. Deleted old IConsole::powerDown, renamed IConsole::powerDownAsync to IConsole::powerDown (as promised for 2.1). Implemented perl sample code for registering a hard disk. Cleaned up constant formatting in the API docs. Updated SDK changelog. Renamed com/errorprint2.h to com/errorprint.h, added a few assertion variants. Eliminated com/errorprint_legacy.h. Adjusted all files using the affected headers and APIs. Renamed tstHeadless2 to tstHeadless.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 29.5 KB
Line 
1/** @file
2 *
3 * VBox frontends: VBoxHeadless (headless frontend):
4 * Headless server executable
5 */
6
7/*
8 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#include <VBox/com/com.h>
24#include <VBox/com/string.h>
25#include <VBox/com/Guid.h>
26#include <VBox/com/ErrorInfo.h>
27#include <VBox/com/errorprint.h>
28#include <VBox/com/EventQueue.h>
29
30#include <VBox/com/VirtualBox.h>
31
32using namespace com;
33
34#define LOG_GROUP LOG_GROUP_GUI
35
36#include <VBox/log.h>
37#include <VBox/version.h>
38#ifdef VBOX_WITH_VRDP
39# include <VBox/vrdpapi.h>
40#endif
41#include <iprt/ctype.h>
42#include <iprt/initterm.h>
43#include <iprt/stream.h>
44#include <iprt/ldr.h>
45#include <iprt/getopt.h>
46#include <iprt/env.h>
47#include <VBox/err.h>
48#include <VBox/VBoxVideo.h>
49
50#ifdef VBOX_FFMPEG
51#include <cstdlib>
52#include <cerrno>
53#include "VBoxHeadless.h"
54#include <iprt/env.h>
55#include <iprt/param.h>
56#include <iprt/process.h>
57#include <VBox/sup.h>
58#endif
59
60//#define VBOX_WITH_SAVESTATE_ON_SIGNAL
61#ifdef VBOX_WITH_SAVESTATE_ON_SIGNAL
62#include <signal.h>
63#endif
64
65#ifdef VBOX_WITH_VRDP
66# include "Framebuffer.h"
67#endif
68
69////////////////////////////////////////////////////////////////////////////////
70
71#define LogError(m,rc) \
72 do { \
73 Log (("VBoxHeadless: ERROR: " m " [rc=0x%08X]\n", rc)); \
74 RTPrintf ("%s\n", m); \
75 } while (0)
76
77////////////////////////////////////////////////////////////////////////////////
78
79/* global weak references (for event handlers) */
80static ComPtr <ISession, ComWeakRef> gSession;
81static ComPtr <IConsole, ComWeakRef> gConsole;
82static EventQueue *gEventQ = NULL;
83
84////////////////////////////////////////////////////////////////////////////////
85
86/**
87 * State change event.
88 */
89class StateChangeEvent : public Event
90{
91public:
92 StateChangeEvent (MachineState_T state) : mState (state) {}
93protected:
94 void *handler()
95 {
96 LogFlow (("VBoxHeadless: StateChangeEvent: %d\n", mState));
97 /* post the termination event if the machine has been PoweredDown/Saved/Aborted */
98 if (mState < MachineState_Running)
99 gEventQ->postEvent (NULL);
100 return 0;
101 }
102private:
103 MachineState_T mState;
104};
105
106/**
107 * Callback handler for machine events.
108 */
109class ConsoleCallback : VBOX_SCRIPTABLE_IMPL(IConsoleCallback)
110{
111public:
112
113 ConsoleCallback ()
114 {
115#ifndef VBOX_WITH_XPCOM
116 refcnt = 0;
117#endif
118 }
119
120 virtual ~ConsoleCallback() {}
121
122 NS_DECL_ISUPPORTS
123
124#ifndef VBOX_WITH_XPCOM
125 STDMETHOD_(ULONG, AddRef)()
126 {
127 return ::InterlockedIncrement(&refcnt);
128 }
129 STDMETHOD_(ULONG, Release)()
130 {
131 long cnt = ::InterlockedDecrement(&refcnt);
132 if (cnt == 0)
133 delete this;
134 return cnt;
135 }
136 STDMETHOD(QueryInterface) (REFIID riid , void **ppObj)
137 {
138 if (riid == IID_IUnknown)
139 {
140 *ppObj = this;
141 AddRef();
142 return S_OK;
143 }
144 if (riid == IID_IConsoleCallback)
145 {
146 *ppObj = this;
147 AddRef();
148 return S_OK;
149 }
150 *ppObj = NULL;
151 return E_NOINTERFACE;
152 }
153#endif
154
155 STDMETHOD(OnMousePointerShapeChange) (BOOL visible, BOOL alpha, ULONG xHot, ULONG yHot,
156 ULONG width, ULONG height, BYTE *shape)
157 {
158 return S_OK;
159 }
160
161 STDMETHOD(OnMouseCapabilityChange) (BOOL supportsAbsolute, BOOL needsHostCursor)
162 {
163 /* Emit absolute mouse event to actually enable the host mouse cursor. */
164 if (supportsAbsolute && gConsole)
165 {
166 ComPtr<IMouse> mouse;
167 gConsole->COMGETTER(Mouse)(mouse.asOutParam());
168 if (mouse)
169 {
170 mouse->PutMouseEventAbsolute(-1, -1, 0, 0);
171 }
172 }
173 return S_OK;
174 }
175
176 STDMETHOD(OnKeyboardLedsChange)(BOOL fNumLock, BOOL fCapsLock, BOOL fScrollLock)
177 {
178 return S_OK;
179 }
180
181 STDMETHOD(OnStateChange) (MachineState_T machineState)
182 {
183 gEventQ->postEvent (new StateChangeEvent (machineState));
184 return S_OK;
185 }
186
187 STDMETHOD(OnExtraDataChange) (BSTR key)
188 {
189 return S_OK;
190 }
191
192 STDMETHOD(OnAdditionsStateChange)()
193 {
194 return S_OK;
195 }
196
197 STDMETHOD(OnDVDDriveChange)()
198 {
199 return S_OK;
200 }
201
202 STDMETHOD(OnFloppyDriveChange)()
203 {
204 return S_OK;
205 }
206
207 STDMETHOD(OnNetworkAdapterChange) (INetworkAdapter *aNetworkAdapter)
208 {
209 return S_OK;
210 }
211
212 STDMETHOD(OnSerialPortChange) (ISerialPort *aSerialPort)
213 {
214 return S_OK;
215 }
216
217 STDMETHOD(OnParallelPortChange) (IParallelPort *aParallelPort)
218 {
219 return S_OK;
220 }
221
222 STDMETHOD(OnVRDPServerChange)()
223 {
224 return S_OK;
225 }
226
227 STDMETHOD(OnUSBControllerChange)()
228 {
229 return S_OK;
230 }
231
232 STDMETHOD(OnStorageControllerChange)()
233 {
234 return S_OK;
235 }
236
237 STDMETHOD(OnUSBDeviceStateChange) (IUSBDevice *aDevice, BOOL aAttached,
238 IVirtualBoxErrorInfo *aError)
239 {
240 return S_OK;
241 }
242
243 STDMETHOD(OnSharedFolderChange) (Scope_T aScope)
244 {
245 return S_OK;
246 }
247
248 STDMETHOD(OnRuntimeError)(BOOL fatal, IN_BSTR id, IN_BSTR message)
249 {
250 return S_OK;
251 }
252
253 STDMETHOD(OnCanShowWindow)(BOOL *canShow)
254 {
255 if (!canShow)
256 return E_POINTER;
257 /* Headless windows should not be shown */
258 *canShow = FALSE;
259 return S_OK;
260 }
261
262 STDMETHOD(OnShowWindow) (ULONG64 *winId)
263 {
264 /* OnCanShowWindow() always returns FALSE, so this call should never
265 * happen. */
266 AssertFailed();
267 if (!winId)
268 return E_POINTER;
269 *winId = 0;
270 return E_NOTIMPL;
271 }
272
273private:
274
275#ifndef VBOX_WITH_XPCOM
276 long refcnt;
277#endif
278};
279
280#ifdef VBOX_WITH_XPCOM
281NS_DECL_CLASSINFO (ConsoleCallback)
282NS_IMPL_THREADSAFE_ISUPPORTS1_CI (ConsoleCallback, IConsoleCallback)
283#endif
284
285#ifdef VBOX_WITH_SAVESTATE_ON_SIGNAL
286static void SaveState(int sig)
287{
288 ComPtr <IProgress> progress = NULL;
289
290/** @todo Deal with nested signals, multithreaded signal dispatching (esp. on windows),
291 * and multiple signals (both SIGINT and SIGTERM in some order).
292 * Consider processing the signal request asynchronously since there are lots of things
293 * which aren't safe (like RTPrintf and printf IIRC) in a signal context. */
294
295 RTPrintf("Signal received, saving state.\n");
296
297 HRESULT rc = gConsole->SaveState(progress.asOutParam());
298 if (FAILED(S_OK))
299 {
300 RTPrintf("Error saving state! rc = 0x%x\n", rc);
301 return;
302 }
303 Assert(progress);
304 LONG cPercent = 0;
305
306 RTPrintf("0%%");
307 RTStrmFlush(g_pStdOut);
308 for (;;)
309 {
310 BOOL fCompleted = false;
311 rc = progress->COMGETTER(Completed)(&fCompleted);
312 if (FAILED(rc) || fCompleted)
313 break;
314 ULONG cPercentNow;
315 rc = progress->COMGETTER(Percent)(&cPercentNow);
316 if (FAILED(rc))
317 break;
318 if ((cPercentNow / 10) != (cPercent / 10))
319 {
320 cPercent = cPercentNow;
321 RTPrintf("...%d%%", cPercentNow);
322 RTStrmFlush(g_pStdOut);
323 }
324
325 /* wait */
326 rc = progress->WaitForCompletion(100);
327 }
328
329 HRESULT lrc;
330 rc = progress->COMGETTER(ResultCode)(&lrc);
331 if (FAILED(rc))
332 lrc = ~0;
333 if (!lrc)
334 {
335 RTPrintf(" -- Saved the state successfully.\n");
336 RTThreadYield();
337 }
338 else
339 RTPrintf("-- Error saving state, lrc=%d (%#x)\n", lrc, lrc);
340
341}
342#endif /* VBOX_WITH_SAVESTATE_ON_SIGNAL */
343
344////////////////////////////////////////////////////////////////////////////////
345
346static void show_usage()
347{
348 RTPrintf("Usage:\n"
349 " -s, -startvm, --startvm <name|uuid> Start given VM (required argument)\n"
350#ifdef VBOX_WITH_VRDP
351 " -v, -vrdp, --vrdp on|off|config Enable (default) or disable the VRDP\n"
352 " server or don't change the setting\n"
353 " -p, -vrdpport, --vrdpport <port> Port number the VRDP server will bind\n"
354 " to\n"
355 " -a, -vrdpaddress, --vrdpaddress <ip> Interface IP the VRDP will bind to \n"
356#endif
357#ifdef VBOX_FFMPEG
358 " -c, -capture, --capture Record the VM screen output to a file\n"
359 " -w, --width Frame width when recording\n"
360 " -h, --height Frame height when recording\n"
361 " -r, --bitrate Recording bit rate when recording\n"
362 " -f, --filename File name when recording. The codec\n"
363 " used will be chosen based on the\n"
364 " file extension\n"
365#endif
366 "\n");
367}
368
369#ifdef VBOX_FFMPEG
370/**
371 * Parse the environment for variables which can influence the FFMPEG settings.
372 * purely for backwards compatibility.
373 * @param pulFrameWidth may be updated with a desired frame width
374 * @param pulFrameHeight may be updated with a desired frame height
375 * @param pulBitRate may be updated with a desired bit rate
376 * @param ppszFileName may be updated with a desired file name
377 */
378static void parse_environ(unsigned long *pulFrameWidth, unsigned long *pulFrameHeight,
379 unsigned long *pulBitRate, const char **ppszFileName)
380{
381 const char *pszEnvTemp;
382
383 if ((pszEnvTemp = RTEnvGet("VBOX_CAPTUREWIDTH")) != 0)
384 {
385 errno = 0;
386 unsigned long ulFrameWidth = strtoul(pszEnvTemp, 0, 10);
387 if (errno != 0)
388 LogError("VBoxHeadless: ERROR: invalid VBOX_CAPTUREWIDTH environment variable", 0);
389 else
390 *pulFrameWidth = ulFrameWidth;
391 }
392 if ((pszEnvTemp = RTEnvGet("VBOX_CAPTUREHEIGHT")) != 0)
393 {
394 errno = 0;
395 unsigned long ulFrameHeight = strtoul(pszEnvTemp, 0, 10);
396 if (errno != 0)
397 LogError("VBoxHeadless: ERROR: invalid VBOX_CAPTUREHEIGHT environment variable", 0);
398 else
399 *pulFrameHeight = ulFrameHeight;
400 }
401 if ((pszEnvTemp = RTEnvGet("VBOX_CAPTUREBITRATE")) != 0)
402 {
403 errno = 0;
404 unsigned long ulBitRate = strtoul(pszEnvTemp, 0, 10);
405 if (errno != 0)
406 LogError("VBoxHeadless: ERROR: invalid VBOX_CAPTUREBITRATE environment variable", 0);
407 else
408 *pulBitRate = ulBitRate;
409 }
410 if ((pszEnvTemp = RTEnvGet("VBOX_CAPTUREFILE")) != 0)
411 *ppszFileName = pszEnvTemp;
412}
413#endif /* VBOX_FFMPEG defined */
414
415/**
416 * Entry point.
417 */
418extern "C" DECLEXPORT (int) TrustedMain (int argc, char **argv, char **envp)
419{
420#ifdef VBOX_WITH_VRDP
421 ULONG vrdpPort = ~0U;
422 const char *vrdpAddress = NULL;
423 const char *vrdpEnabled = NULL;
424#endif
425 unsigned fRawR0 = ~0U;
426 unsigned fRawR3 = ~0U;
427 unsigned fPATM = ~0U;
428 unsigned fCSAM = ~0U;
429#ifdef VBOX_FFMPEG
430 unsigned fFFMPEG = 0;
431 unsigned long ulFrameWidth = 800;
432 unsigned long ulFrameHeight = 600;
433 unsigned long ulBitRate = 300000;
434 char pszMPEGFile[RTPATH_MAX];
435 const char *pszFileNameParam = "VBox-%d.vob";
436#endif /* VBOX_FFMPEG */
437
438 /* Make sure that DISPLAY is unset, so that X11 bits do not get initialised
439 * on X11-using OSes. */
440 /** @todo this should really be taken care of in Main. */
441 RTEnvUnset("DISPLAY");
442
443 LogFlow (("VBoxHeadless STARTED.\n"));
444 RTPrintf ("VirtualBox Headless Interface %s\n"
445 "(C) 2008-2009 Sun Microsystems, Inc.\n"
446 "All rights reserved.\n\n",
447 VBOX_VERSION_STRING);
448
449 Bstr id;
450 /* the below cannot be Bstr because on Linux Bstr doesn't work until XPCOM (nsMemory) is initialized */
451 const char *name = NULL;
452
453#ifdef VBOX_FFMPEG
454 /* Parse the environment */
455 parse_environ(&ulFrameWidth, &ulFrameHeight, &ulBitRate, &pszFileNameParam);
456#endif
457
458 enum eHeadlessOptions
459 {
460 OPT_RAW_R0 = 0x100,
461 OPT_NO_RAW_R0,
462 OPT_RAW_R3,
463 OPT_NO_RAW_R3,
464 OPT_PATM,
465 OPT_NO_PATM,
466 OPT_CSAM,
467 OPT_NO_CSAM,
468 OPT_COMMENT,
469 };
470
471 static const RTGETOPTDEF s_aOptions[] =
472 {
473 { "-startvm", 's', RTGETOPT_REQ_STRING },
474 { "--startvm", 's', RTGETOPT_REQ_STRING },
475#ifdef VBOX_WITH_VRDP
476 { "-vrdpport", 'p', RTGETOPT_REQ_UINT32 },
477 { "--vrdpport", 'p', RTGETOPT_REQ_UINT32 },
478 { "-vrdpaddress", 'a', RTGETOPT_REQ_STRING },
479 { "--vrdpaddress", 'a', RTGETOPT_REQ_STRING },
480 { "-vrdp", 'v', RTGETOPT_REQ_STRING },
481 { "--vrdp", 'v', RTGETOPT_REQ_STRING },
482#endif /* VBOX_WITH_VRDP defined */
483 { "-rawr0", OPT_RAW_R0, 0 },
484 { "--rawr0", OPT_RAW_R0, 0 },
485 { "-norawr0", OPT_NO_RAW_R0, 0 },
486 { "--norawr0", OPT_NO_RAW_R0, 0 },
487 { "-rawr3", OPT_RAW_R3, 0 },
488 { "--rawr3", OPT_RAW_R3, 0 },
489 { "-norawr3", OPT_NO_RAW_R3, 0 },
490 { "--norawr3", OPT_NO_RAW_R3, 0 },
491 { "-patm", OPT_PATM, 0 },
492 { "--patm", OPT_PATM, 0 },
493 { "-nopatm", OPT_NO_PATM, 0 },
494 { "--nopatm", OPT_NO_PATM, 0 },
495 { "-csam", OPT_CSAM, 0 },
496 { "--csam", OPT_CSAM, 0 },
497 { "-nocsam", OPT_NO_CSAM, 0 },
498 { "--nocsam", OPT_NO_CSAM, 0 },
499#ifdef VBOX_FFMPEG
500 { "-capture", 'c', 0 },
501 { "--capture", 'c', 0 },
502 { "--width", 'w', RTGETOPT_REQ_UINT32 },
503 { "--height", 'h', RTGETOPT_REQ_UINT32 },
504 { "--bitrate", 'r', RTGETOPT_REQ_UINT32 },
505 { "--filename", 'f', RTGETOPT_REQ_STRING },
506#endif /* VBOX_FFMPEG defined */
507 { "-comment", OPT_COMMENT, RTGETOPT_REQ_STRING },
508 { "--comment", OPT_COMMENT, RTGETOPT_REQ_STRING }
509 };
510
511 // parse the command line
512 int ch;
513 RTGETOPTUNION ValueUnion;
514 RTGETOPTSTATE GetState;
515 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, 0 /* fFlags */);
516 while ((ch = RTGetOpt(&GetState, &ValueUnion)))
517 {
518 switch(ch)
519 {
520 case 's':
521 id = asGuidStr(ValueUnion.psz);
522 /* If the argument was not a UUID, then it must be a name. */
523 if (!id)
524 name = ValueUnion.psz;
525 break;
526#ifdef VBOX_WITH_VRDP
527 case 'p':
528 vrdpPort = ValueUnion.u32;
529 break;
530 case 'a':
531 vrdpAddress = ValueUnion.psz;
532 break;
533 case 'v':
534 vrdpEnabled = ValueUnion.psz;
535 break;
536#endif /* VBOX_WITH_VRDP defined */
537 case OPT_RAW_R0:
538 fRawR0 = true;
539 break;
540 case OPT_NO_RAW_R0:
541 fRawR0 = false;
542 break;
543 case OPT_RAW_R3:
544 fRawR3 = true;
545 break;
546 case OPT_NO_RAW_R3:
547 fRawR3 = false;
548 break;
549 case OPT_PATM:
550 fPATM = true;
551 break;
552 case OPT_NO_PATM:
553 fPATM = false;
554 break;
555 case OPT_CSAM:
556 fCSAM = true;
557 break;
558 case OPT_NO_CSAM:
559 fCSAM = false;
560 break;
561#ifdef VBOX_FFMPEG
562 case 'c':
563 fFFMPEG = true;
564 break;
565 case 'w':
566 ulFrameWidth = ValueUnion.u32;
567 break;
568 case 'h':
569 ulFrameHeight = ValueUnion.u32;
570 break;
571 case 'r':
572 ulBitRate = ValueUnion.u32;
573 break;
574 case 'f':
575 pszFileNameParam = ValueUnion.psz;
576 break;
577#endif /* VBOX_FFMPEG defined */
578 case VINF_GETOPT_NOT_OPTION:
579 RTPrintf("Invalid parameter '%s'\n\n", ValueUnion.psz);
580 show_usage();
581 return -1;
582 case OPT_COMMENT:
583 /* nothing to do */
584 break;
585 default:
586 if (ch > 0)
587 {
588 if (RT_C_IS_PRINT(ch))
589 RTPrintf("Invalid option -%c\n\n", ch);
590 else
591 RTPrintf("Invalid option case %i\n\n", ch);
592 }
593 else if (ch == VERR_GETOPT_UNKNOWN_OPTION)
594 RTPrintf("Unknown option: %s\n\n", ValueUnion.psz);
595 else if (ValueUnion.pDef)
596 RTPrintf("%s: %Rrs\n\n", ValueUnion.pDef->pszLong, ch);
597 else
598 RTPrintf("Error: %Rrs\n\n", ch);
599 show_usage();
600 return -1;
601 }
602 }
603
604#ifdef VBOX_FFMPEG
605 if (ulFrameWidth < 512 || ulFrameWidth > 2048 || ulFrameWidth % 2)
606 {
607 LogError("VBoxHeadless: ERROR: please specify an even frame width between 512 and 2048", 0);
608 return -1;
609 }
610 if (ulFrameHeight < 384 || ulFrameHeight > 1536 || ulFrameHeight % 2)
611 {
612 LogError("VBoxHeadless: ERROR: please specify an even frame height between 384 and 1536", 0);
613 return -1;
614 }
615 if (ulBitRate < 300000 || ulBitRate > 1000000)
616 {
617 LogError("VBoxHeadless: ERROR: please specify an even bitrate between 300000 and 1000000", 0);
618 return -1;
619 }
620 /* Make sure we only have %d or %u (or none) in the file name specified */
621 char *pcPercent = (char*)strchr(pszFileNameParam, '%');
622 if (pcPercent != 0 && *(pcPercent + 1) != 'd' && *(pcPercent + 1) != 'u')
623 {
624 LogError("VBoxHeadless: ERROR: Only %%d and %%u are allowed in the capture file name.", -1);
625 return -1;
626 }
627 /* And no more than one % in the name */
628 if (pcPercent != 0 && strchr(pcPercent + 1, '%') != 0)
629 {
630 LogError("VBoxHeadless: ERROR: Only one format modifier is allowed in the capture file name.", -1);
631 return -1;
632 }
633 RTStrPrintf(&pszMPEGFile[0], RTPATH_MAX, pszFileNameParam, RTProcSelf());
634#endif /* defined VBOX_FFMPEG */
635
636 if (!id && !name)
637 {
638 show_usage();
639 return -1;
640 }
641
642 HRESULT rc;
643
644 rc = com::Initialize();
645 if (FAILED(rc))
646 {
647 RTPrintf("VBoxHeadless: ERROR: failed to initialize COM!\n");
648 return rc;
649 }
650
651 do
652 {
653 ComPtr<IVirtualBox> virtualBox;
654 ComPtr<ISession> session;
655
656 rc = virtualBox.createLocalObject(CLSID_VirtualBox);
657 if (FAILED(rc))
658 RTPrintf("VBoxHeadless: ERROR: failed to create the VirtualBox object!\n");
659 else
660 {
661 rc = session.createInprocObject(CLSID_Session);
662 if (FAILED(rc))
663 RTPrintf("VBoxHeadless: ERROR: failed to create a session object!\n");
664 }
665
666 if (FAILED(rc))
667 {
668 com::ErrorInfo info;
669 if (!info.isFullAvailable() && !info.isBasicAvailable())
670 {
671 com::GluePrintRCMessage(rc);
672 RTPrintf("Most likely, the VirtualBox COM server is not running or failed to start.\n");
673 }
674 else
675 GluePrintErrorInfo(info);
676 break;
677 }
678
679 /* find ID by name */
680 if (!id)
681 {
682 ComPtr <IMachine> m;
683 rc = virtualBox->FindMachine (Bstr (name), m.asOutParam());
684 if (FAILED (rc))
685 {
686 LogError ("Invalid machine name!\n", rc);
687 break;
688 }
689 m->COMGETTER(Id) (id.asOutParam());
690 AssertComRC (rc);
691 if (FAILED (rc))
692 break;
693 }
694
695 Log (("VBoxHeadless: Opening a session with machine (id={%s})...\n",
696 Utf8Str(id).raw()));
697
698 // open a session
699 CHECK_ERROR_BREAK(virtualBox, OpenSession (session, id));
700
701 /* get the console */
702 ComPtr <IConsole> console;
703 CHECK_ERROR_BREAK(session, COMGETTER (Console) (console.asOutParam()));
704
705 /* get the machine */
706 ComPtr <IMachine> machine;
707 CHECK_ERROR_BREAK(console, COMGETTER(Machine) (machine.asOutParam()));
708
709 ComPtr <IDisplay> display;
710 CHECK_ERROR_BREAK(console, COMGETTER(Display) (display.asOutParam()));
711
712#ifdef VBOX_FFMPEG
713 IFramebuffer *pFramebuffer = 0;
714 RTLDRMOD hLdrFFmpegFB;
715 PFNREGISTERFFMPEGFB pfnRegisterFFmpegFB;
716
717 if (fFFMPEG)
718 {
719 int rrc = VINF_SUCCESS, rcc = S_OK;
720
721 Log2(("VBoxHeadless: loading VBoxFFmpegFB shared library\n"));
722 rrc = SUPR3HardenedLdrLoadAppPriv("VBoxFFmpegFB", &hLdrFFmpegFB);
723
724 if (RT_SUCCESS(rrc))
725 {
726 Log2(("VBoxHeadless: looking up symbol VBoxRegisterFFmpegFB\n"));
727 rrc = RTLdrGetSymbol(hLdrFFmpegFB, "VBoxRegisterFFmpegFB",
728 reinterpret_cast<void **>(&pfnRegisterFFmpegFB));
729 if (RT_FAILURE(rrc))
730 LogError("Failed to load the video capture extension, possibly due to a damaged file\n", rrc);
731 }
732 else
733 LogError("Failed to load the video capture extension\n", rrc);
734 if (RT_SUCCESS(rrc))
735 {
736 Log2(("VBoxHeadless: calling pfnRegisterFFmpegFB\n"));
737 rcc = pfnRegisterFFmpegFB(ulFrameWidth, ulFrameHeight, ulBitRate,
738 pszMPEGFile, &pFramebuffer);
739 if (rcc != S_OK)
740 LogError("Failed to initialise video capturing - make sure that the file format\n"
741 "you wish to use is supported on your system\n", rcc);
742 }
743 if (RT_SUCCESS(rrc) && (S_OK == rcc))
744 {
745 Log2(("VBoxHeadless: Registering framebuffer\n"));
746 pFramebuffer->AddRef();
747 display->SetFramebuffer(VBOX_VIDEO_PRIMARY_SCREEN, pFramebuffer);
748 }
749 if (!RT_SUCCESS(rrc) || (rcc != S_OK))
750 rc = E_FAIL;
751 }
752 if (rc != S_OK)
753 {
754 return -1;
755 }
756#endif /* defined(VBOX_FFMPEG) */
757
758 ULONG cMonitors = 1;
759 machine->COMGETTER(MonitorCount)(&cMonitors);
760
761#ifdef VBOX_WITH_VRDP
762 unsigned uScreenId;
763 for (uScreenId = 0; uScreenId < cMonitors; uScreenId++)
764 {
765#ifdef VBOX_FFMPEG
766 if (fFFMPEG && uScreenId == 0)
767 {
768 /* Already registered. */
769 continue;
770 }
771#endif /* defined(VBOX_FFMPEG) */
772 VRDPFramebuffer *pFramebuffer = new VRDPFramebuffer ();
773 if (!pFramebuffer)
774 {
775 RTPrintf("Error: could not create framebuffer object %d\n", uScreenId);
776 return -1;
777 }
778 pFramebuffer->AddRef();
779 display->SetFramebuffer(uScreenId, pFramebuffer);
780 }
781#endif
782
783 /* get the machine debugger (isn't necessarily available) */
784 ComPtr <IMachineDebugger> machineDebugger;
785 console->COMGETTER(Debugger)(machineDebugger.asOutParam());
786 if (machineDebugger)
787 {
788 Log(("Machine debugger available!\n"));
789 }
790
791 if (fRawR0 != ~0U)
792 {
793 if (!machineDebugger)
794 {
795 RTPrintf("Error: No debugger object; -%srawr0 cannot be executed!\n", fRawR0 ? "" : "no");
796 break;
797 }
798 machineDebugger->COMSETTER(RecompileSupervisor)(!fRawR0);
799 }
800 if (fRawR3 != ~0U)
801 {
802 if (!machineDebugger)
803 {
804 RTPrintf("Error: No debugger object; -%srawr3 cannot be executed!\n", fRawR0 ? "" : "no");
805 break;
806 }
807 machineDebugger->COMSETTER(RecompileUser)(!fRawR3);
808 }
809 if (fPATM != ~0U)
810 {
811 if (!machineDebugger)
812 {
813 RTPrintf("Error: No debugger object; -%spatm cannot be executed!\n", fRawR0 ? "" : "no");
814 break;
815 }
816 machineDebugger->COMSETTER(PATMEnabled)(fPATM);
817 }
818 if (fCSAM != ~0U)
819 {
820 if (!machineDebugger)
821 {
822 RTPrintf("Error: No debugger object; -%scsam cannot be executed!\n", fRawR0 ? "" : "no");
823 break;
824 }
825 machineDebugger->COMSETTER(CSAMEnabled)(fCSAM);
826 }
827
828 /* create an event queue */
829 EventQueue eventQ;
830
831 /* initialize global references */
832 gSession = session;
833 gConsole = console;
834 gEventQ = &eventQ;
835
836 /* register a callback for machine events */
837 {
838 ConsoleCallback *callback = new ConsoleCallback ();
839 callback->AddRef();
840 CHECK_ERROR(console, RegisterCallback (callback));
841 callback->Release();
842 if (FAILED (rc))
843 break;
844 }
845
846#ifdef VBOX_WITH_VRDP
847 /* default is to enable the RDP server (backward compatibility) */
848 BOOL fVRDPEnable = true;
849 BOOL fVRDPEnabled;
850 ComPtr <IVRDPServer> vrdpServer;
851 CHECK_ERROR_BREAK(machine, COMGETTER (VRDPServer) (vrdpServer.asOutParam()));
852 CHECK_ERROR_BREAK(vrdpServer, COMGETTER(Enabled) (&fVRDPEnabled));
853
854 if (vrdpEnabled != NULL)
855 {
856 /* -vrdp on|off|config */
857 if (!strcmp(vrdpEnabled, "off") || !strcmp(vrdpEnabled, "disable"))
858 fVRDPEnable = false;
859 else if (!strcmp(vrdpEnabled, "config"))
860 {
861 if (!fVRDPEnabled)
862 fVRDPEnable = false;
863 }
864 else if (strcmp(vrdpEnabled, "on") && strcmp(vrdpEnabled, "enable"))
865 {
866 RTPrintf("-vrdp requires an argument (on|off|config)\n");
867 break;
868 }
869 }
870
871 if (fVRDPEnable)
872 {
873 Log (("VBoxHeadless: Enabling VRDP server...\n"));
874
875 /* set VRDP port if requested by the user */
876 if (vrdpPort != ~0U)
877 CHECK_ERROR_BREAK(vrdpServer, COMSETTER(Port)(vrdpPort));
878 else
879 CHECK_ERROR_BREAK(vrdpServer, COMGETTER(Port)(&vrdpPort));
880 /* set VRDP address if requested by the user */
881 if (vrdpAddress != NULL)
882 {
883 CHECK_ERROR_BREAK(vrdpServer, COMSETTER(NetAddress)(Bstr(vrdpAddress)));
884 }
885 /* enable VRDP server (only if currently disabled) */
886 if (!fVRDPEnabled)
887 {
888 CHECK_ERROR_BREAK(vrdpServer, COMSETTER(Enabled) (TRUE));
889 }
890 }
891 else
892 {
893 /* disable VRDP server (only if currently enabled */
894 if (fVRDPEnabled)
895 {
896 CHECK_ERROR_BREAK(vrdpServer, COMSETTER(Enabled) (FALSE));
897 }
898 }
899#endif
900 Log (("VBoxHeadless: Powering up the machine...\n"));
901#ifdef VBOX_WITH_VRDP
902 if (fVRDPEnable)
903 RTPrintf("Listening on port %d\n", !vrdpPort ? VRDP_DEFAULT_PORT : vrdpPort);
904#endif
905
906 ComPtr <IProgress> progress;
907 CHECK_ERROR_BREAK(console, PowerUp (progress.asOutParam()));
908
909 /* wait for result because there can be errors */
910 if (SUCCEEDED(progress->WaitForCompletion (-1)))
911 {
912 LONG progressRc;
913 progress->COMGETTER(ResultCode)(&progressRc);
914 rc = progressRc;
915 if (FAILED(progressRc))
916 {
917 com::ProgressErrorInfo info(progress);
918 if (info.isBasicAvailable())
919 {
920 RTPrintf("Error: failed to start machine. Error message: %lS\n", info.getText().raw());
921 }
922 else
923 {
924 RTPrintf("Error: failed to start machine. No error message available!\n");
925 }
926 }
927 }
928
929#ifdef VBOX_WITH_SAVESTATE_ON_SIGNAL
930 signal(SIGINT, SaveState);
931 signal(SIGTERM, SaveState);
932#endif
933
934 Log (("VBoxHeadless: Waiting for PowerDown...\n"));
935
936 Event *e;
937
938 while (eventQ.waitForEvent (&e) && e)
939 eventQ.handleEvent (e);
940
941 Log (("VBoxHeadless: event loop has terminated...\n"));
942
943#ifdef VBOX_FFMPEG
944 if (pFramebuffer)
945 {
946 pFramebuffer->Release ();
947 Log(("Released framebuffer\n"));
948 pFramebuffer = NULL;
949 }
950#endif /* defined(VBOX_FFMPEG) */
951
952 /* we don't have to disable VRDP here because we don't save the settings of the VM */
953
954 /*
955 * Close the session. This will also uninitialize the console and
956 * unregister the callback we've registered before.
957 */
958 Log (("VBoxHeadless: Closing the session...\n"));
959 session->Close();
960 }
961 while (0);
962
963 com::Shutdown();
964
965 LogFlow (("VBoxHeadless FINISHED.\n"));
966
967 return rc;
968}
969
970
971#ifndef VBOX_WITH_HARDENING
972/**
973 * Main entry point.
974 */
975int main (int argc, char **argv, char **envp)
976{
977 // initialize VBox Runtime
978 int rc = RTR3InitAndSUPLib();
979 if (RT_FAILURE(rc))
980 {
981 RTPrintf("VBoxHeadless: Runtime Error:\n"
982 " %Rrc -- %Rrf\n", rc, rc);
983 switch (rc)
984 {
985 case VERR_VM_DRIVER_NOT_INSTALLED:
986 RTPrintf("Cannot access the kernel driver. Make sure the kernel module has been \n"
987 "loaded successfully. Aborting ...\n");
988 break;
989 default:
990 break;
991 }
992 return 1;
993 }
994
995 return TrustedMain (argc, argv, envp);
996}
997#endif /* !VBOX_WITH_HARDENING */
998
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