VirtualBox

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

Last change on this file since 16840 was 16530, checked in by vboxsync, 16 years ago

Main: rework error macros everywhere; make error messages much more readable (esp. with VBoxManage); use shared function to actually print message; reduces size of VBoxManage debug build from 3.4 to 2.3 MB

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