VirtualBox

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

Last change on this file since 23454 was 23223, checked in by vboxsync, 15 years ago

API: big medium handling change and lots of assorted other cleanups and fixes

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