VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxBalloonCtrl/VBoxBalloonCtrl.cpp@ 39731

Last change on this file since 39731 was 38636, checked in by vboxsync, 13 years ago

*,IPRT: Redid the ring-3 init to always convert the arguments to UTF-8.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 47.1 KB
Line 
1/* $Id: VBoxBalloonCtrl.cpp 38636 2011-09-05 13:49:45Z vboxsync $ */
2/** @file
3 * VBoxBalloonCtrl - VirtualBox Ballooning Control Service.
4 */
5
6/*
7 * Copyright (C) 2011 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#ifndef VBOX_ONLY_DOCS
23# include <VBox/com/com.h>
24# include <VBox/com/string.h>
25# include <VBox/com/Guid.h>
26# include <VBox/com/array.h>
27# include <VBox/com/ErrorInfo.h>
28# include <VBox/com/errorprint.h>
29
30# include <VBox/com/EventQueue.h>
31# include <VBox/com/listeners.h>
32# include <VBox/com/VirtualBox.h>
33#endif /* !VBOX_ONLY_DOCS */
34
35#include <VBox/err.h>
36#include <VBox/log.h>
37#include <VBox/version.h>
38
39#include <package-generated.h>
40
41#include <iprt/asm.h>
42#include <iprt/buildconfig.h>
43#include <iprt/critsect.h>
44#include <iprt/getopt.h>
45#include <iprt/initterm.h>
46#include <iprt/path.h>
47#include <iprt/process.h>
48#include <iprt/semaphore.h>
49#include <iprt/stream.h>
50#include <iprt/string.h>
51#include <iprt/system.h>
52
53#include <map>
54#include <string>
55#include <signal.h>
56
57#include "VBoxBalloonCtrl.h"
58
59using namespace com;
60
61/* When defined, use a global performance collector instead
62 * of a per-machine based one. */
63#define VBOX_BALLOONCTRL_GLOBAL_PERFCOL
64
65/** The critical section for keep our stuff in sync. */
66static RTCRITSECT g_MapCritSect;
67
68/** Set by the signal handler. */
69static volatile bool g_fCanceled = false;
70
71static uint32_t g_cHistory = 10; /* Enable log rotation, 10 files. */
72static uint32_t g_uHistoryFileTime = RT_SEC_1DAY; /* Max 1 day per file. */
73static uint64_t g_uHistoryFileSize = 100 * _1M; /* Max 100MB per file. */
74
75static bool g_fVerbose = false;
76
77#if defined(RT_OS_DARWIN) || defined(RT_OS_LINUX) || defined (RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
78/** Run in background. */
79static bool g_fDaemonize = false;
80#endif
81
82/**
83 * RTGetOpt-IDs for the command line.
84 */
85enum GETOPTDEF_BALLOONCTRL
86{
87 GETOPTDEF_BALLOONCTRL_BALLOOINC = 1000,
88 GETOPTDEF_BALLOONCTRL_BALLOONDEC,
89 GETOPTDEF_BALLOONCTRL_BALLOONLOWERLIMIT,
90 GETOPTDEF_BALLOONCTRL_BALLOONMAX
91};
92
93/**
94 * Command line arguments.
95 */
96static const RTGETOPTDEF g_aOptions[] = {
97#if defined(RT_OS_DARWIN) || defined(RT_OS_LINUX) || defined (RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
98 { "--background", 'b', RTGETOPT_REQ_NOTHING },
99#endif
100 /** For displayHelp(). */
101 { "--help", 'h', RTGETOPT_REQ_NOTHING },
102 /** Sets g_ulTimeoutMS. */
103 { "--interval", 'i', RTGETOPT_REQ_INT32 },
104 /** Sets g_ulMemoryBalloonIncrementMB. */
105 { "--balloon-inc", GETOPTDEF_BALLOONCTRL_BALLOOINC, RTGETOPT_REQ_INT32 },
106 /** Sets g_ulMemoryBalloonDecrementMB. */
107 { "--balloon-dec", GETOPTDEF_BALLOONCTRL_BALLOONDEC, RTGETOPT_REQ_INT32 },
108 { "--balloon-lower-limit", GETOPTDEF_BALLOONCTRL_BALLOONLOWERLIMIT, RTGETOPT_REQ_INT32 },
109 /** Global max. balloon limit. */
110 { "--balloon-max", GETOPTDEF_BALLOONCTRL_BALLOONMAX, RTGETOPT_REQ_INT32 },
111 { "--verbose", 'v', RTGETOPT_REQ_NOTHING },
112 { "--pidfile", 'P', RTGETOPT_REQ_STRING },
113 { "--logfile", 'F', RTGETOPT_REQ_STRING },
114 { "--logrotate", 'R', RTGETOPT_REQ_UINT32 },
115 { "--logsize", 'S', RTGETOPT_REQ_UINT64 },
116 { "--loginterval", 'I', RTGETOPT_REQ_UINT32 }
117};
118
119static unsigned long g_ulTimeoutMS = 30 * 1000; /* Default is 30 seconds timeout. */
120static unsigned long g_ulMemoryBalloonIncrementMB = 256;
121static unsigned long g_ulMemoryBalloonDecrementMB = 128;
122/** Global balloon limit is 0, so disabled. Can be overridden by a per-VM
123 * "VBoxInternal/Guest/BalloonSizeMax" value. */
124static unsigned long g_ulMemoryBalloonMaxMB = 0;
125static unsigned long g_ulLowerMemoryLimitMB = 64;
126
127/** Global objects. */
128static ComPtr<IVirtualBoxClient> g_pVirtualBoxClient = NULL;
129static ComPtr<IVirtualBox> g_pVirtualBox = NULL;
130static ComPtr<ISession> g_pSession = NULL;
131static ComPtr<IEventSource> g_pEventSource = NULL;
132static ComPtr<IEventSource> g_pEventSourceClient = NULL;
133static ComPtr<IEventListener> g_pVBoxEventListener = NULL;
134# ifdef VBOX_BALLOONCTRL_GLOBAL_PERFCOL
135static ComPtr<IPerformanceCollector> g_pPerfCollector = NULL;
136# endif
137static EventQueue *g_pEventQ = NULL;
138
139/** A machine's internal entry. */
140typedef struct VBOXBALLOONCTRL_MACHINE
141{
142 ComPtr<IMachine> machine;
143 unsigned long ulBalloonSizeMax;
144#ifndef VBOX_BALLOONCTRL_GLOBAL_PERFCOL
145 ComPtr<IPerformanceCollector> collector;
146#endif
147} VBOXBALLOONCTRL_MACHINE, *PVBOXBALLOONCTRL_MACHINE;
148typedef std::map<Bstr, VBOXBALLOONCTRL_MACHINE> mapVM;
149typedef std::map<Bstr, VBOXBALLOONCTRL_MACHINE>::iterator mapVMIter;
150typedef std::map<Bstr, VBOXBALLOONCTRL_MACHINE>::const_iterator mapVMIterConst;
151static mapVM g_mapVM;
152
153/* Prototypes. */
154#define serviceLogVerbose(a) if (g_fVerbose) { serviceLog a; }
155static void serviceLog(const char *pszFormat, ...);
156
157static bool machineIsRunning(MachineState_T enmState);
158static bool machineHandled(const Bstr &strUuid);
159static int machineAdd(const Bstr &strUuid);
160static int machineRemove(const Bstr &strUuid);
161static int machineUpdate(const Bstr &strUuid, MachineState_T enmState);
162
163static unsigned long balloonGetMaxSize(const ComPtr<IMachine> &rptrMachine);
164static bool balloonIsRequired(PVBOXBALLOONCTRL_MACHINE pMachine);
165static int balloonUpdate(const Bstr &strUuid, PVBOXBALLOONCTRL_MACHINE pMachine);
166
167static HRESULT balloonCtrlSetup();
168static void balloonCtrlShutdown();
169
170static HRESULT createGlobalObjects();
171static void deleteGlobalObjects();
172
173#ifdef RT_OS_WINDOWS
174/* Required for ATL. */
175static CComModule _Module;
176#endif
177
178/**
179 * Handler for global events.
180 */
181class VirtualBoxEventListener
182{
183 public:
184 VirtualBoxEventListener()
185 {
186 }
187
188 virtual ~VirtualBoxEventListener()
189 {
190 }
191
192 HRESULT init()
193 {
194 return S_OK;
195 }
196
197 void uninit()
198 {
199 }
200
201 STDMETHOD(HandleEvent)(VBoxEventType_T aType, IEvent *aEvent)
202 {
203 switch (aType)
204 {
205 case VBoxEventType_OnMachineRegistered:
206 {
207 ComPtr<IMachineRegisteredEvent> pEvent = aEvent;
208 Assert(pEvent);
209
210 Bstr uuid;
211 BOOL fRegistered;
212 HRESULT hr = pEvent->COMGETTER(Registered)(&fRegistered);
213 if (SUCCEEDED(hr))
214 hr = pEvent->COMGETTER(MachineId)(uuid.asOutParam());
215
216 if (SUCCEEDED(hr))
217 {
218 int rc = RTCritSectEnter(&g_MapCritSect);
219 if (RT_SUCCESS(rc))
220 {
221 if (fRegistered && machineHandled(uuid))
222 rc = machineAdd(uuid);
223 else if (!fRegistered)
224 rc = machineRemove(uuid);
225
226 int rc2 = RTCritSectLeave(&g_MapCritSect);
227 if (RT_SUCCESS(rc))
228 rc = rc2;
229 AssertRC(rc);
230 }
231 }
232 break;
233 }
234
235 case VBoxEventType_OnMachineStateChanged:
236 {
237 ComPtr<IMachineStateChangedEvent> pEvent = aEvent;
238 Assert(pEvent);
239
240 MachineState_T machineState;
241 Bstr uuid;
242
243 HRESULT hr = pEvent->COMGETTER(State)(&machineState);
244 if (SUCCEEDED(hr))
245 hr = pEvent->COMGETTER(MachineId)(uuid.asOutParam());
246
247 if (SUCCEEDED(hr))
248 {
249 int rc = RTCritSectEnter(&g_MapCritSect);
250 if (RT_SUCCESS(rc))
251 {
252 rc = machineUpdate(uuid, machineState);
253 int rc2 = RTCritSectLeave(&g_MapCritSect);
254 if (RT_SUCCESS(rc))
255 rc = rc2;
256 AssertRC(rc);
257 }
258 }
259 break;
260 }
261
262 case VBoxEventType_OnVBoxSVCAvailabilityChanged:
263 {
264 ComPtr<IVBoxSVCAvailabilityChangedEvent> pVSACEv = aEvent;
265 Assert(pVSACEv);
266 BOOL fAvailable = FALSE;
267 pVSACEv->COMGETTER(Available)(&fAvailable);
268 if (!fAvailable)
269 {
270 serviceLog("VBoxSVC became unavailable\n");
271
272 balloonCtrlShutdown();
273 deleteGlobalObjects();
274 }
275 else
276 {
277 serviceLog("VBoxSVC became available\n");
278 HRESULT hrc = createGlobalObjects();
279 if (FAILED(hrc))
280 serviceLog("Unable to re-create local COM objects (rc=%Rhrc)!\n", hrc);
281 else
282 {
283 hrc = balloonCtrlSetup();
284 if (FAILED(hrc))
285 serviceLog("Unable to re-set up ballooning (rc=%Rhrc)!\n", hrc);
286 }
287 }
288 break;
289 }
290
291 default:
292 /* Not handled event, just skip it. */
293 break;
294 }
295
296 return S_OK;
297 }
298
299 private:
300};
301typedef ListenerImpl<VirtualBoxEventListener> VirtualBoxEventListenerImpl;
302VBOX_LISTENER_DECLARE(VirtualBoxEventListenerImpl)
303
304
305/**
306 * Signal handler that sets g_fGuestCtrlCanceled.
307 *
308 * This can be executed on any thread in the process, on Windows it may even be
309 * a thread dedicated to delivering this signal. Do not doing anything
310 * unnecessary here.
311 */
312static void signalHandler(int iSignal)
313{
314 NOREF(iSignal);
315 ASMAtomicWriteBool(&g_fCanceled, true);
316
317 if (!g_pEventQ)
318 {
319 int rc = g_pEventQ->interruptEventQueueProcessing();
320 if (RT_FAILURE(rc))
321 serviceLog("Error: interruptEventQueueProcessing failed with rc=%Rrc\n", rc);
322 }
323}
324
325/**
326 * Installs a custom signal handler to get notified
327 * whenever the user wants to intercept the program.
328 */
329static void signalHandlerInstall()
330{
331 signal(SIGINT, signalHandler);
332#ifdef SIGBREAK
333 signal(SIGBREAK, signalHandler);
334#endif
335}
336
337/**
338 * Uninstalls a previously installed signal handler.
339 */
340static void signalHandlerUninstall()
341{
342 signal(SIGINT, SIG_DFL);
343#ifdef SIGBREAK
344 signal(SIGBREAK, SIG_DFL);
345#endif
346}
347
348/**
349 * Retrieves the current delta value
350 *
351 * @return long Delta (MB) of the balloon to be deflated (<0) or inflated (>0).
352 * @param ulCurrentDesktopBalloonSize The balloon's current size.
353 * @param ulDesktopFreeMemory The VM's current free memory.
354 * @param ulMaxBalloonSize The maximum balloon size (MB) it can inflate to.
355 */
356static long getlBalloonDelta(unsigned long ulCurrentDesktopBalloonSize, unsigned long ulDesktopFreeMemory, unsigned long ulMaxBalloonSize)
357{
358 if (ulCurrentDesktopBalloonSize > ulMaxBalloonSize)
359 return (ulMaxBalloonSize - ulCurrentDesktopBalloonSize);
360
361 long lBalloonDelta = 0;
362 if (ulDesktopFreeMemory < g_ulLowerMemoryLimitMB)
363 {
364 /* Guest is running low on memory, we need to
365 * deflate the balloon. */
366 lBalloonDelta = (g_ulMemoryBalloonDecrementMB * -1);
367
368 /* Ensure that the delta will not return a negative
369 * balloon size. */
370 if ((long)ulCurrentDesktopBalloonSize + lBalloonDelta < 0)
371 lBalloonDelta = 0;
372 }
373 else if (ulMaxBalloonSize > ulCurrentDesktopBalloonSize)
374 {
375 /* We want to inflate the balloon if we have room. */
376 long lIncrement = g_ulMemoryBalloonIncrementMB;
377 while (lIncrement >= 16 && (ulDesktopFreeMemory - lIncrement) < g_ulLowerMemoryLimitMB)
378 {
379 lIncrement = (lIncrement / 2);
380 }
381
382 if ((ulDesktopFreeMemory - lIncrement) > g_ulLowerMemoryLimitMB)
383 lBalloonDelta = lIncrement;
384 }
385 if (ulCurrentDesktopBalloonSize + lBalloonDelta > ulMaxBalloonSize)
386 lBalloonDelta = (ulMaxBalloonSize - ulCurrentDesktopBalloonSize);
387 return lBalloonDelta;
388}
389
390/**
391 * Indicates whether a VM is up and running (regardless of its running
392 * state, could be paused as well).
393 *
394 * @return bool Flag indicating whether the VM is running or not.
395 * @param enmState The VM's machine state to judge whether it's running or not.
396 */
397static bool machineIsRunning(MachineState_T enmState)
398{
399 switch (enmState)
400 {
401 case MachineState_Running:
402#if 0
403 /* Not required for ballooning. */
404 case MachineState_Teleporting:
405 case MachineState_LiveSnapshotting:
406 case MachineState_Paused:
407 case MachineState_TeleportingPausedVM:
408#endif
409 return true;
410 default:
411 break;
412 }
413 return false;
414}
415
416/**
417 * Determines whether the specified machine needs to be handled
418 * by this service.
419 *
420 * @return bool True if the machine needs handling, false if not.
421 * @param strUuid UUID of the specified machine.
422 */
423static bool machineHandled(const Bstr &strUuid)
424{
425 bool fHandled = false;
426
427 do
428 {
429 HRESULT rc;
430
431 ComPtr <IMachine> machine;
432 CHECK_ERROR_BREAK(g_pVirtualBox, FindMachine(strUuid.raw(), machine.asOutParam()));
433
434 MachineState_T machineState;
435 CHECK_ERROR_BREAK(machine, COMGETTER(State)(&machineState));
436
437 if ( balloonGetMaxSize(machine)
438 && machineIsRunning(machineState))
439 {
440 serviceLogVerbose(("Handling machine \"%ls\"\n", strUuid.raw()));
441 fHandled = true;
442 }
443 }
444 while (0);
445
446 return fHandled;
447}
448
449/**
450 * Adds a specified machine to the list (map) of handled machines.
451 *
452 * @return IPRT status code.
453 * @param strUuid UUID of the specified machine.
454 */
455static int machineAdd(const Bstr &strUuid)
456{
457 HRESULT rc;
458
459 do
460 {
461 ComPtr <IMachine> machine;
462 CHECK_ERROR_BREAK(g_pVirtualBox, FindMachine(strUuid.raw(), machine.asOutParam()));
463
464 MachineState_T machineState;
465 CHECK_ERROR_BREAK(machine, COMGETTER(State)(&machineState));
466
467 if ( !balloonGetMaxSize(machine)
468 || !machineIsRunning(machineState))
469 {
470 /* This machine does not need to be added, just skip it! */
471 break;
472 }
473
474 VBOXBALLOONCTRL_MACHINE m;
475 m.machine = machine;
476
477 /*
478 * Setup metrics.
479 */
480 com::SafeArray<BSTR> metricNames(1);
481 com::SafeIfaceArray<IUnknown> metricObjects(1);
482 com::SafeIfaceArray<IPerformanceMetric> metricAffected;
483
484 Bstr strMetricNames(L"Guest/RAM/Usage");
485 strMetricNames.cloneTo(&metricNames[0]);
486
487 m.machine.queryInterfaceTo(&metricObjects[0]);
488
489#ifdef VBOX_BALLOONCTRL_GLOBAL_PERFCOL
490 CHECK_ERROR_BREAK(g_pPerfCollector, SetupMetrics(ComSafeArrayAsInParam(metricNames),
491 ComSafeArrayAsInParam(metricObjects),
492 5 /* 5 seconds */,
493 1 /* One sample is enough */,
494 ComSafeArrayAsOutParam(metricAffected)));
495#else
496 CHECK_ERROR_BREAK(g_pVirtualBox, COMGETTER(PerformanceCollector)(m.collector.asOutParam()));
497 CHECK_ERROR_BREAK(m.collector, SetupMetrics(ComSafeArrayAsInParam(metricNames),
498 ComSafeArrayAsInParam(metricObjects),
499 5 /* 5 seconds */,
500 1 /* One sample is enough */,
501 ComSafeArrayAsOutParam(metricAffected)));
502#endif
503 /*
504 * Add machine to map.
505 */
506 mapVMIter it = g_mapVM.find(strUuid);
507 Assert(it == g_mapVM.end());
508
509 g_mapVM.insert(std::make_pair(strUuid, m));
510
511 serviceLogVerbose(("Added machine \"%ls\"\n", strUuid.raw()));
512
513 } while (0);
514
515 return SUCCEEDED(rc) ? VINF_SUCCESS : VERR_COM_IPRT_ERROR; /* @todo Find a better error! */
516}
517
518/**
519 * Removes a specified machine from the list of handled machines.
520 *
521 * @return IPRT status code.
522 * @param strUuid UUID of the specified machine.
523 */
524static int machineRemove(const Bstr &strUuid)
525{
526 int rc = RTCritSectEnter(&g_MapCritSect);
527 if (RT_SUCCESS(rc))
528 {
529 mapVMIter it = g_mapVM.find(strUuid);
530 if (it != g_mapVM.end())
531 {
532 /* Must log before erasing the iterator because of the UUID ref! */
533 serviceLogVerbose(("Removing machine \"%ls\"\n", strUuid.raw()));
534
535 /*
536 * Remove machine from map.
537 */
538 g_mapVM.erase(it);
539 }
540 else
541 {
542 serviceLogVerbose(("Warning: Removing not added machine \"%ls\"\n", strUuid.raw()));
543 rc = VERR_NOT_FOUND;
544 }
545
546 int rc2 = RTCritSectLeave(&g_MapCritSect);
547 if (RT_SUCCESS(rc))
548 rc = rc2;
549 }
550
551 return rc;
552}
553
554/**
555 * Updates a specified machine according to its current machine state.
556 * That currently also could mean that a machine gets removed if it doesn't
557 * fit in our criteria anymore or a machine gets added if we need to handle
558 * it now (and didn't before).
559 *
560 * @return IPRT status code.
561 * @param strUuid UUID of the specified machine.
562 * @param enmState The machine's current state.
563 */
564static int machineUpdate(const Bstr &strUuid, MachineState_T enmState)
565{
566 int rc = VINF_SUCCESS;
567
568 mapVMIter it = g_mapVM.find(strUuid);
569 if (it == g_mapVM.end())
570 {
571 if (machineHandled(strUuid))
572 {
573 rc = machineAdd(strUuid);
574 if (RT_SUCCESS(rc))
575 it = g_mapVM.find(strUuid);
576 }
577 else
578 {
579 serviceLogVerbose(("Machine \"%ls\" (state: %u) does not need to be updated\n",
580 strUuid.raw(), enmState));
581 }
582 }
583
584 if (it != g_mapVM.end())
585 {
586 /*
587 * Ballooning stuff - start.
588 */
589
590 /* Our actual ballooning criteria. */
591 if ( !balloonIsRequired(&it->second)
592 || !machineIsRunning(enmState))
593 {
594 /* Current machine is not suited for ballooning anymore -
595 * remove it from our map. */
596 rc = machineRemove(strUuid);
597 }
598 else
599 {
600 rc = balloonUpdate(strUuid, &it->second);
601 AssertRC(rc);
602 }
603 }
604
605 /*
606 * Ballooning stuff - end.
607 */
608
609 return rc;
610}
611
612/**
613 * Retrieves a metric from a specified machine.
614 *
615 * @return IPRT status code.
616 * @param pMachine Pointer to the machine's internal structure.
617 * @param strName Name of metric to retrieve.
618 * @param pulData Pointer to value to retrieve the actual metric value.
619 */
620static int getMetric(PVBOXBALLOONCTRL_MACHINE pMachine, const Bstr& strName, LONG *pulData)
621{
622 AssertPtrReturn(pMachine, VERR_INVALID_POINTER);
623 AssertPtrReturn(pulData, VERR_INVALID_POINTER);
624
625 /* Input. */
626 com::SafeArray<BSTR> metricNames(1);
627 com::SafeIfaceArray<IUnknown> metricObjects(1);
628 pMachine->machine.queryInterfaceTo(&metricObjects[0]);
629
630 /* Output. */
631 com::SafeArray<BSTR> retNames;
632 com::SafeIfaceArray<IUnknown> retObjects;
633 com::SafeArray<BSTR> retUnits;
634 com::SafeArray<ULONG> retScales;
635 com::SafeArray<ULONG> retSequenceNumbers;
636 com::SafeArray<ULONG> retIndices;
637 com::SafeArray<ULONG> retLengths;
638 com::SafeArray<LONG> retData;
639
640 /* Query current memory free. */
641 strName.cloneTo(&metricNames[0]);
642#ifdef VBOX_BALLOONCTRL_GLOBAL_PERFCOL
643 Assert(!g_pPerfCollector.isNull());
644 HRESULT hrc = g_pPerfCollector->QueryMetricsData(
645#else
646 Assert(!pMachine->collector.isNull());
647 HRESULT hrc = pMachine->collector->QueryMetricsData(
648#endif
649 ComSafeArrayAsInParam(metricNames),
650 ComSafeArrayAsInParam(metricObjects),
651 ComSafeArrayAsOutParam(retNames),
652 ComSafeArrayAsOutParam(retObjects),
653 ComSafeArrayAsOutParam(retUnits),
654 ComSafeArrayAsOutParam(retScales),
655 ComSafeArrayAsOutParam(retSequenceNumbers),
656 ComSafeArrayAsOutParam(retIndices),
657 ComSafeArrayAsOutParam(retLengths),
658 ComSafeArrayAsOutParam(retData));
659#if 0
660 /* Useful for metrics debugging. */
661 for (unsigned j = 0; j < retNames.size(); j++)
662 {
663 Bstr metricUnit(retUnits[j]);
664 Bstr metricName(retNames[j]);
665 RTPrintf("%-20ls ", metricName.raw());
666 const char *separator = "";
667 for (unsigned k = 0; k < retLengths[j]; k++)
668 {
669 if (retScales[j] == 1)
670 RTPrintf("%s%d %ls", separator, retData[retIndices[j] + k], metricUnit.raw());
671 else
672 RTPrintf("%s%d.%02d%ls", separator, retData[retIndices[j] + k] / retScales[j],
673 (retData[retIndices[j] + k] * 100 / retScales[j]) % 100, metricUnit.raw());
674 separator = ", ";
675 }
676 RTPrintf("\n");
677 }
678#endif
679
680 if (SUCCEEDED(hrc))
681 *pulData = retData.size() ? retData[retIndices[0]] : 0;
682
683 return SUCCEEDED(hrc) ? VINF_SUCCESS : VINF_NOT_SUPPORTED;
684}
685
686/**
687 * Determines the maximum balloon size to set for the specified machine.
688 *
689 * @return unsigned long Balloon size (in MB) to set, 0 if no ballooning required.
690 * @param rptrMachine Pointer to interface of specified machine.
691 */
692static unsigned long balloonGetMaxSize(const ComPtr<IMachine> &rptrMachine)
693{
694 /*
695 * Try to retrieve the balloon maximum size via the following order:
696 * - command line parameter ("--balloon-max")
697 * - per-VM parameter ("VBoxInternal/Guest/BalloonSizeMax")
698 * - global parameter ("VBoxInternal/Guest/BalloonSizeMax")
699 */
700 unsigned long ulBalloonMax = g_ulMemoryBalloonMaxMB; /* Use global limit as default. */
701 if (!ulBalloonMax) /* Not set by command line? */
702 {
703 /* Try per-VM approach. */
704 Bstr strValue;
705 HRESULT rc = rptrMachine->GetExtraData(Bstr("VBoxInternal/Guest/BalloonSizeMax").raw(),
706 strValue.asOutParam());
707 if ( SUCCEEDED(rc)
708 && !strValue.isEmpty())
709 {
710 ulBalloonMax = Utf8Str(strValue).toUInt32();
711 }
712 }
713 if (!ulBalloonMax) /* Still not set by per-VM value? */
714 {
715 /* Try global approach. */
716 Bstr strValue;
717 HRESULT rc = g_pVirtualBox->GetExtraData(Bstr("VBoxInternal/Guest/BalloonSizeMax").raw(),
718 strValue.asOutParam());
719 if ( SUCCEEDED(rc)
720 && !strValue.isEmpty())
721 {
722 ulBalloonMax = Utf8Str(strValue).toUInt32();
723 }
724 }
725 return ulBalloonMax;
726}
727
728/**
729 * Determines whether ballooning is required to the specified machine.
730 *
731 * @return bool True if ballooning is required, false if not.
732 * @param strUuid UUID of the specified machine.
733 */
734static bool balloonIsRequired(PVBOXBALLOONCTRL_MACHINE pMachine)
735{
736 AssertPtrReturn(pMachine, false);
737
738 /* Only do ballooning if we have a maximum balloon size set. */
739 pMachine->ulBalloonSizeMax = pMachine->machine.isNull()
740 ? 0 : balloonGetMaxSize(pMachine->machine);
741
742 return pMachine->ulBalloonSizeMax ? true : false;
743}
744
745/**
746 * Does the actual ballooning and assumes the machine is
747 * capable and ready for ballooning.
748 *
749 * @return IPRT status code.
750 * @param strUuid UUID of the specified machine.
751 * @param pMachine Pointer to the machine's internal structure.
752 */
753static int balloonUpdate(const Bstr &strUuid, PVBOXBALLOONCTRL_MACHINE pMachine)
754{
755 AssertPtrReturn(pMachine, VERR_INVALID_POINTER);
756
757 /*
758 * Get metrics collected at this point.
759 */
760 LONG lMemFree, lBalloonCur;
761 int vrc = getMetric(pMachine, L"Guest/RAM/Usage/Free", &lMemFree);
762 if (RT_SUCCESS(vrc))
763 vrc = getMetric(pMachine, L"Guest/RAM/Usage/Balloon", &lBalloonCur);
764
765 if (RT_SUCCESS(vrc))
766 {
767 /* If guest statistics are not up and running yet, skip this iteration
768 * and try next time. */
769 if (lMemFree <= 0)
770 {
771#ifdef DEBUG
772 serviceLogVerbose(("%ls: No metrics available yet!\n", strUuid.raw()));
773#endif
774 return VINF_SUCCESS;
775 }
776
777 lMemFree /= 1024;
778 lBalloonCur /= 1024;
779
780 serviceLogVerbose(("%ls: Balloon: %ld, Free mem: %ld, Max ballon: %ld\n",
781 strUuid.raw(),
782 lBalloonCur, lMemFree, pMachine->ulBalloonSizeMax));
783
784 /* Calculate current balloon delta. */
785 long lDelta = getlBalloonDelta(lBalloonCur, lMemFree, pMachine->ulBalloonSizeMax);
786 if (lDelta) /* Only do ballooning if there's really smth. to change ... */
787 {
788 lBalloonCur = lBalloonCur + lDelta;
789 Assert(lBalloonCur > 0);
790
791 serviceLog("%ls: %s balloon by %ld to %ld ...\n",
792 strUuid.raw(),
793 lDelta > 0 ? "Inflating" : "Deflating", lDelta, lBalloonCur);
794
795 HRESULT rc;
796
797 /* Open a session for the VM. */
798 CHECK_ERROR(pMachine->machine, LockMachine(g_pSession, LockType_Shared));
799
800 do
801 {
802 /* Get the associated console. */
803 ComPtr<IConsole> console;
804 CHECK_ERROR_BREAK(g_pSession, COMGETTER(Console)(console.asOutParam()));
805
806 ComPtr <IGuest> guest;
807 rc = console->COMGETTER(Guest)(guest.asOutParam());
808 if (SUCCEEDED(rc))
809 CHECK_ERROR_BREAK(guest, COMSETTER(MemoryBalloonSize)(lBalloonCur));
810 else
811 serviceLog("Error: Unable to set new balloon size %ld for machine \"%ls\", rc=%Rhrc",
812 lBalloonCur, strUuid.raw(), rc);
813 } while (0);
814
815 /* Unlock the machine again. */
816 g_pSession->UnlockMachine();
817 }
818 }
819 else
820 serviceLog("Error: Unable to retrieve metrics for machine \"%ls\", rc=%Rrc",
821 strUuid.raw(), vrc);
822 return vrc;
823}
824
825static void vmListDestroy()
826{
827 serviceLogVerbose(("Destroying VM list ...\n"));
828
829 int rc = RTCritSectEnter(&g_MapCritSect);
830 if (RT_SUCCESS(rc))
831 {
832 mapVMIter it = g_mapVM.begin();
833 while (it != g_mapVM.end())
834 {
835#ifndef VBOX_BALLOONCTRL_GLOBAL_PERFCOL
836 it->second.collector.setNull();
837#endif
838 it->second.machine.setNull();
839 it++;
840 }
841
842 g_mapVM.clear();
843
844 rc = RTCritSectLeave(&g_MapCritSect);
845 }
846 AssertRC(rc);
847}
848
849static int vmListBuild()
850{
851 serviceLogVerbose(("Building VM list ...\n"));
852
853 int rc = RTCritSectEnter(&g_MapCritSect);
854 if (RT_SUCCESS(rc))
855 {
856 /*
857 * Make sure the list is empty.
858 */
859 g_mapVM.clear();
860
861 /*
862 * Get the list of all _running_ VMs
863 */
864 com::SafeIfaceArray<IMachine> machines;
865 HRESULT hrc = g_pVirtualBox->COMGETTER(Machines)(ComSafeArrayAsOutParam(machines));
866 if (SUCCEEDED(hrc))
867 {
868 /*
869 * Iterate through the collection
870 */
871 for (size_t i = 0; i < machines.size(); ++i)
872 {
873 if (machines[i])
874 {
875 Bstr strUUID;
876 CHECK_ERROR_BREAK(machines[i], COMGETTER(Id)(strUUID.asOutParam()));
877
878 BOOL fAccessible;
879 CHECK_ERROR_BREAK(machines[i], COMGETTER(Accessible)(&fAccessible));
880 if (!fAccessible)
881 {
882 serviceLogVerbose(("Machine \"%ls\" is inaccessible, skipping\n",
883 strUUID.raw()));
884 continue;
885 }
886
887 rc = machineAdd(strUUID);
888 if (RT_FAILURE(rc))
889 break;
890 }
891 }
892
893 if (!machines.size())
894 serviceLogVerbose(("No machines to add found at the moment!\n"));
895 }
896
897 int rc2 = RTCritSectLeave(&g_MapCritSect);
898 if (RT_SUCCESS(rc))
899 rc = rc2;
900 }
901 return rc;
902}
903
904static int balloonCtrlCheck()
905{
906 static uint64_t uLast = UINT64_MAX;
907 uint64_t uNow = RTTimeProgramMilliTS() / g_ulTimeoutMS;
908 if (uLast == uNow)
909 return VINF_SUCCESS;
910 uLast = uNow;
911
912 int rc = RTCritSectEnter(&g_MapCritSect);
913 if (RT_SUCCESS(rc))
914 {
915 mapVMIter it = g_mapVM.begin();
916 while (it != g_mapVM.end())
917 {
918 MachineState_T machineState;
919 HRESULT hrc = it->second.machine->COMGETTER(State)(&machineState);
920 if (SUCCEEDED(hrc))
921 {
922 rc = machineUpdate(it->first /* UUID */, machineState);
923 if (RT_FAILURE(rc))
924 break;
925 }
926 it++;
927 }
928
929 int rc2 = RTCritSectLeave(&g_MapCritSect);
930 if (RT_SUCCESS(rc))
931 rc = rc2;
932 }
933
934 return rc;
935}
936
937static HRESULT balloonCtrlSetup()
938{
939 HRESULT rc = S_OK;
940
941 serviceLog("Setting up ballooning ...\n");
942
943 do
944 {
945 /*
946 * Setup metrics.
947 */
948#ifdef VBOX_BALLOONCTRL_GLOBAL_PERFCOL
949 CHECK_ERROR_BREAK(g_pVirtualBox, COMGETTER(PerformanceCollector)(g_pPerfCollector.asOutParam()));
950#endif
951
952 /*
953 * Build up initial VM list.
954 */
955 int vrc = vmListBuild();
956 if (RT_FAILURE(vrc))
957 {
958 rc = VBOX_E_IPRT_ERROR;
959 break;
960 }
961
962 } while (0);
963
964 return rc;
965}
966
967static void balloonCtrlShutdown()
968{
969 serviceLog("Shutting down ballooning ...\n");
970
971 vmListDestroy();
972
973#ifdef VBOX_BALLOONCTRL_GLOBAL_PERFCOL
974 g_pPerfCollector.setNull();
975#endif
976}
977
978static RTEXITCODE balloonCtrlMain(HandlerArg *a)
979{
980 HRESULT rc = S_OK;
981
982 do
983 {
984 int vrc = VINF_SUCCESS;
985
986 /* Initialize global weak references. */
987 g_pEventQ = com::EventQueue::getMainEventQueue();
988
989 RTCritSectInit(&g_MapCritSect);
990
991 /*
992 * Install signal handlers.
993 */
994 signal(SIGINT, signalHandler);
995 #ifdef SIGBREAK
996 signal(SIGBREAK, signalHandler);
997 #endif
998
999 /*
1000 * Setup the global event listeners:
1001 * - g_pEventSource for machine events
1002 * - g_pEventSourceClient for VBoxClient events (like VBoxSVC handling)
1003 */
1004 CHECK_ERROR_BREAK(g_pVirtualBox, COMGETTER(EventSource)(g_pEventSource.asOutParam()));
1005 CHECK_ERROR_BREAK(g_pVirtualBoxClient, COMGETTER(EventSource)(g_pEventSourceClient.asOutParam()));
1006
1007 ComObjPtr<VirtualBoxEventListenerImpl> vboxListenerImpl;
1008 vboxListenerImpl.createObject();
1009 vboxListenerImpl->init(new VirtualBoxEventListener());
1010
1011 com::SafeArray <VBoxEventType_T> eventTypes;
1012 eventTypes.push_back(VBoxEventType_OnMachineRegistered);
1013 eventTypes.push_back(VBoxEventType_OnMachineStateChanged);
1014 eventTypes.push_back(VBoxEventType_OnVBoxSVCAvailabilityChanged); /* Processed by g_pEventSourceClient. */
1015
1016 g_pVBoxEventListener = vboxListenerImpl;
1017 CHECK_ERROR_BREAK(g_pEventSource, RegisterListener(g_pVBoxEventListener, ComSafeArrayAsInParam(eventTypes), true /* Active listener */));
1018 CHECK_ERROR_BREAK(g_pEventSourceClient, RegisterListener(g_pVBoxEventListener, ComSafeArrayAsInParam(eventTypes), true /* Active listener */));
1019
1020 /*
1021 * Set up ballooning stuff.
1022 */
1023 rc = balloonCtrlSetup();
1024 if (FAILED(rc))
1025 break;
1026
1027 for (;;)
1028 {
1029 /*
1030 * Do the actual work.
1031 */
1032 vrc = balloonCtrlCheck();
1033 if (RT_FAILURE(vrc))
1034 {
1035 serviceLog("Error while doing ballooning control; rc=%Rrc\n", vrc);
1036 break;
1037 }
1038
1039 /*
1040 * Process pending events, then wait for new ones. Note, this
1041 * processes NULL events signalling event loop termination.
1042 */
1043 g_pEventQ->processEventQueue(g_ulTimeoutMS / 10);
1044
1045 if (g_fCanceled)
1046 {
1047 serviceLog("Signal caught, exiting ...\n");
1048 break;
1049 }
1050 }
1051
1052 signal(SIGINT, SIG_DFL);
1053 #ifdef SIGBREAK
1054 signal(SIGBREAK, SIG_DFL);
1055 #endif
1056
1057 /* VirtualBox callback unregistration. */
1058 if (g_pVBoxEventListener)
1059 {
1060 if (!g_pEventSource.isNull())
1061 CHECK_ERROR(g_pEventSource, UnregisterListener(g_pVBoxEventListener));
1062 g_pVBoxEventListener.setNull();
1063 }
1064
1065 g_pEventSource.setNull();
1066 g_pEventSourceClient.setNull();
1067
1068 balloonCtrlShutdown();
1069
1070 RTCritSectDelete(&g_MapCritSect);
1071
1072 if (RT_FAILURE(vrc))
1073 rc = VBOX_E_IPRT_ERROR;
1074
1075 } while (0);
1076
1077 return SUCCEEDED(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1078}
1079
1080static void serviceLog(const char *pszFormat, ...)
1081{
1082 va_list args;
1083 va_start(args, pszFormat);
1084 char *psz = NULL;
1085 RTStrAPrintfV(&psz, pszFormat, args);
1086 va_end(args);
1087
1088 LogRel(("%s", psz));
1089
1090 RTStrFree(psz);
1091}
1092
1093static void logHeaderFooter(PRTLOGGER pLoggerRelease, RTLOGPHASE enmPhase, PFNRTLOGPHASEMSG pfnLog)
1094{
1095 /* Some introductory information. */
1096 static RTTIMESPEC s_TimeSpec;
1097 char szTmp[256];
1098 if (enmPhase == RTLOGPHASE_BEGIN)
1099 RTTimeNow(&s_TimeSpec);
1100 RTTimeSpecToString(&s_TimeSpec, szTmp, sizeof(szTmp));
1101
1102 switch (enmPhase)
1103 {
1104 case RTLOGPHASE_BEGIN:
1105 {
1106 pfnLog(pLoggerRelease,
1107 "VirtualBox Ballooning Control Service %s r%u %s (%s %s) release log\n"
1108#ifdef VBOX_BLEEDING_EDGE
1109 "EXPERIMENTAL build " VBOX_BLEEDING_EDGE "\n"
1110#endif
1111 "Log opened %s\n",
1112 VBOX_VERSION_STRING, RTBldCfgRevision(), VBOX_BUILD_TARGET,
1113 __DATE__, __TIME__, szTmp);
1114
1115 int vrc = RTSystemQueryOSInfo(RTSYSOSINFO_PRODUCT, szTmp, sizeof(szTmp));
1116 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
1117 pfnLog(pLoggerRelease, "OS Product: %s\n", szTmp);
1118 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_RELEASE, szTmp, sizeof(szTmp));
1119 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
1120 pfnLog(pLoggerRelease, "OS Release: %s\n", szTmp);
1121 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_VERSION, szTmp, sizeof(szTmp));
1122 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
1123 pfnLog(pLoggerRelease, "OS Version: %s\n", szTmp);
1124 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
1125 pfnLog(pLoggerRelease, "OS Service Pack: %s\n", szTmp);
1126
1127 /* the package type is interesting for Linux distributions */
1128 char szExecName[RTPATH_MAX];
1129 char *pszExecName = RTProcGetExecutablePath(szExecName, sizeof(szExecName));
1130 pfnLog(pLoggerRelease,
1131 "Executable: %s\n"
1132 "Process ID: %u\n"
1133 "Package type: %s"
1134#ifdef VBOX_OSE
1135 " (OSE)"
1136#endif
1137 "\n",
1138 pszExecName ? pszExecName : "unknown",
1139 RTProcSelf(),
1140 VBOX_PACKAGE_STRING);
1141 break;
1142 }
1143
1144 case RTLOGPHASE_PREROTATE:
1145 pfnLog(pLoggerRelease, "Log rotated - Log started %s\n", szTmp);
1146 break;
1147
1148 case RTLOGPHASE_POSTROTATE:
1149 pfnLog(pLoggerRelease, "Log continuation - Log started %s\n", szTmp);
1150 break;
1151
1152 case RTLOGPHASE_END:
1153 pfnLog(pLoggerRelease, "End of log file - Log started %s\n", szTmp);
1154 break;
1155
1156 default:
1157 /* nothing */;
1158 }
1159}
1160
1161static void displayHelp()
1162{
1163 RTStrmPrintf(g_pStdErr, "\nUsage: VBoxBalloonCtrl [options]\n\nSupported options (default values in brackets):\n");
1164 for (unsigned i = 0;
1165 i < RT_ELEMENTS(g_aOptions);
1166 ++i)
1167 {
1168 std::string str(g_aOptions[i].pszLong);
1169 if (g_aOptions[i].iShort < 1000) /* Don't show short options which are defined by an ID! */
1170 {
1171 str += ", -";
1172 str += g_aOptions[i].iShort;
1173 }
1174 str += ":";
1175
1176 const char *pcszDescr = "";
1177
1178 switch (g_aOptions[i].iShort)
1179 {
1180 case 'h':
1181 pcszDescr = "Print this help message and exit.";
1182 break;
1183
1184 case 'i': /* Interval. */
1185 pcszDescr = "Sets the check interval in ms (30 seconds).";
1186 break;
1187
1188#if defined(RT_OS_DARWIN) || defined(RT_OS_LINUX) || defined (RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1189 case 'b':
1190 pcszDescr = "Run in background (daemon mode).";
1191 break;
1192#endif
1193 case GETOPTDEF_BALLOONCTRL_BALLOOINC:
1194 pcszDescr = "Sets the ballooning increment in MB (256 MB).";
1195 break;
1196
1197 case GETOPTDEF_BALLOONCTRL_BALLOONDEC:
1198 pcszDescr = "Sets the ballooning decrement in MB (128 MB).";
1199 break;
1200
1201 case GETOPTDEF_BALLOONCTRL_BALLOONLOWERLIMIT:
1202 pcszDescr = "Sets the ballooning lower limit in MB (64 MB).";
1203 break;
1204
1205 case GETOPTDEF_BALLOONCTRL_BALLOONMAX:
1206 pcszDescr = "Sets the balloon maximum limit in MB (0 MB).";
1207 break;
1208
1209 case 'P':
1210 pcszDescr = "Name of the PID file which is created when the daemon was started.";
1211 break;
1212
1213 case 'F':
1214 pcszDescr = "Name of file to write log to (no file).";
1215 break;
1216
1217 case 'R':
1218 pcszDescr = "Number of log files (0 disables log rotation).";
1219 break;
1220
1221 case 'S':
1222 pcszDescr = "Maximum size of a log file to trigger rotation (bytes).";
1223 break;
1224
1225 case 'I':
1226 pcszDescr = "Maximum time interval to trigger log rotation (seconds).";
1227 break;
1228 }
1229
1230 RTStrmPrintf(g_pStdErr, "%-23s%s\n", str.c_str(), pcszDescr);
1231 }
1232
1233 RTStrmPrintf(g_pStdErr, "\nUse environment variable VBOXBALLOONCTRL_RELEASE_LOG for logging options.\n"
1234 "Set \"VBoxInternal/Guest/BalloonSizeMax\" for a per-VM maximum ballooning size.\n");
1235}
1236
1237static void deleteGlobalObjects()
1238{
1239 serviceLogVerbose(("Deleting local objects ...\n"));
1240
1241 g_pSession.setNull();
1242 g_pVirtualBox.setNull();
1243}
1244
1245/**
1246 * Creates all global COM objects.
1247 *
1248 * @return HRESULT
1249 */
1250static HRESULT createGlobalObjects()
1251{
1252 serviceLogVerbose(("Creating local objects ...\n"));
1253
1254 HRESULT hrc = g_pVirtualBoxClient->COMGETTER(VirtualBox)(g_pVirtualBox.asOutParam());
1255 if (FAILED(hrc))
1256 {
1257 RTMsgError("Failed to get VirtualBox object (rc=%Rhrc)!", hrc);
1258 }
1259 else
1260 {
1261 hrc = g_pSession.createInprocObject(CLSID_Session);
1262 if (FAILED(hrc))
1263 RTMsgError("Failed to create a session object (rc=%Rhrc)!", hrc);
1264 }
1265
1266 return hrc;
1267}
1268
1269int main(int argc, char *argv[])
1270{
1271 /*
1272 * Before we do anything, init the runtime without loading
1273 * the support driver.
1274 */
1275 int rc = RTR3InitExe(argc, &argv, 0);
1276 if (RT_FAILURE(rc))
1277 return RTMsgInitFailure(rc);
1278
1279 RTPrintf(VBOX_PRODUCT " Balloon Control " VBOX_VERSION_STRING "\n"
1280 "(C) " VBOX_C_YEAR " " VBOX_VENDOR "\n"
1281 "All rights reserved.\n\n");
1282
1283 /*
1284 * Parse the global options
1285 */
1286 int c;
1287 const char *pszLogFile = NULL;
1288 const char *pszPidFile = NULL;
1289 RTGETOPTUNION ValueUnion;
1290 RTGETOPTSTATE GetState;
1291 RTGetOptInit(&GetState, argc, argv, g_aOptions, RT_ELEMENTS(g_aOptions), 1, 0 /*fFlags*/);
1292 while ((c = RTGetOpt(&GetState, &ValueUnion)))
1293 {
1294 switch (c)
1295 {
1296 case 'h':
1297 displayHelp();
1298 return 0;
1299
1300 case 'i': /* Interval. */
1301 g_ulTimeoutMS = ValueUnion.u32;
1302 if (g_ulTimeoutMS < 500)
1303 g_ulTimeoutMS = 500;
1304 break;
1305
1306 case 'v':
1307 g_fVerbose = true;
1308 break;
1309
1310#if defined(RT_OS_DARWIN) || defined(RT_OS_LINUX) || defined (RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1311 case 'b':
1312 g_fDaemonize = true;
1313 break;
1314#endif
1315 case 'V':
1316 RTPrintf("%sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr());
1317 return 0;
1318
1319 case GETOPTDEF_BALLOONCTRL_BALLOOINC:
1320 g_ulMemoryBalloonIncrementMB = ValueUnion.u32;
1321 break;
1322
1323 case GETOPTDEF_BALLOONCTRL_BALLOONDEC:
1324 g_ulMemoryBalloonDecrementMB = ValueUnion.u32;
1325 break;
1326
1327 case GETOPTDEF_BALLOONCTRL_BALLOONLOWERLIMIT:
1328 g_ulLowerMemoryLimitMB = ValueUnion.u32;
1329 break;
1330
1331 case GETOPTDEF_BALLOONCTRL_BALLOONMAX:
1332 g_ulMemoryBalloonMaxMB = ValueUnion.u32;
1333 break;
1334
1335 case 'P':
1336 pszPidFile = ValueUnion.psz;
1337 break;
1338
1339 case 'F':
1340 pszLogFile = ValueUnion.psz;
1341 break;
1342
1343 case 'R':
1344 g_cHistory = ValueUnion.u32;
1345 break;
1346
1347 case 'S':
1348 g_uHistoryFileSize = ValueUnion.u64;
1349 break;
1350
1351 case 'I':
1352 g_uHistoryFileTime = ValueUnion.u32;
1353 break;
1354
1355 default:
1356 rc = RTGetOptPrintError(c, &ValueUnion);
1357 return rc;
1358 }
1359 }
1360
1361 /* create release logger */
1362 PRTLOGGER pLoggerRelease;
1363 static const char * const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
1364 RTUINT fFlags = RTLOGFLAGS_PREFIX_THREAD | RTLOGFLAGS_PREFIX_TIME_PROG;
1365#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
1366 fFlags |= RTLOGFLAGS_USECRLF;
1367#endif
1368 char szError[RTPATH_MAX + 128] = "";
1369 rc = RTLogCreateEx(&pLoggerRelease, fFlags, "all",
1370 "VBOXBALLOONCTRL_RELEASE_LOG", RT_ELEMENTS(s_apszGroups), s_apszGroups, RTLOGDEST_STDOUT,
1371 logHeaderFooter, g_cHistory, g_uHistoryFileSize, g_uHistoryFileTime,
1372 szError, sizeof(szError), pszLogFile);
1373 if (RT_SUCCESS(rc))
1374 {
1375 /* register this logger as the release logger */
1376 RTLogRelSetDefaultInstance(pLoggerRelease);
1377
1378 /* Explicitly flush the log in case of VBOXWEBSRV_RELEASE_LOG=buffered. */
1379 RTLogFlush(pLoggerRelease);
1380 }
1381 else
1382 return RTMsgErrorExit(RTEXITCODE_FAILURE, "failed to open release log (%s, %Rrc)", szError, rc);
1383
1384#if defined(RT_OS_DARWIN) || defined(RT_OS_LINUX) || defined (RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1385 if (g_fDaemonize)
1386 {
1387 /* prepare release logging */
1388 char szLogFile[RTPATH_MAX];
1389
1390 rc = com::GetVBoxUserHomeDirectory(szLogFile, sizeof(szLogFile));
1391 if (RT_FAILURE(rc))
1392 return RTMsgErrorExit(RTEXITCODE_FAILURE, "could not get base directory for logging: %Rrc", rc);
1393 rc = RTPathAppend(szLogFile, sizeof(szLogFile), "vboxballoonctrl.log");
1394 if (RT_FAILURE(rc))
1395 return RTMsgErrorExit(RTEXITCODE_FAILURE, "could not construct logging path: %Rrc", rc);
1396
1397 rc = RTProcDaemonizeUsingFork(false /* fNoChDir */, false /* fNoClose */, pszPidFile);
1398 if (RT_FAILURE(rc))
1399 return RTMsgErrorExit(RTEXITCODE_FAILURE, "failed to daemonize, rc=%Rrc. exiting.", rc);
1400
1401 /* create release logger */
1402 PRTLOGGER pLoggerReleaseFile;
1403 static const char * const s_apszGroupsFile[] = VBOX_LOGGROUP_NAMES;
1404 RTUINT fFlagsFile = RTLOGFLAGS_PREFIX_THREAD | RTLOGFLAGS_PREFIX_TIME_PROG;
1405#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
1406 fFlagsFile |= RTLOGFLAGS_USECRLF;
1407#endif
1408 char szErrorFile[RTPATH_MAX + 128] = "";
1409 int vrc = RTLogCreateEx(&pLoggerReleaseFile, fFlagsFile, "all",
1410 "VBOXBALLOONCTRL_RELEASE_LOG", RT_ELEMENTS(s_apszGroupsFile), s_apszGroupsFile, RTLOGDEST_FILE,
1411 logHeaderFooter, g_cHistory, g_uHistoryFileSize, g_uHistoryFileTime,
1412 szErrorFile, sizeof(szErrorFile), szLogFile);
1413 if (RT_SUCCESS(vrc))
1414 {
1415 /* register this logger as the release logger */
1416 RTLogRelSetDefaultInstance(pLoggerReleaseFile);
1417
1418 /* Explicitly flush the log in case of VBOXBALLOONCTRL_RELEASE_LOG=buffered. */
1419 RTLogFlush(pLoggerReleaseFile);
1420 }
1421 else
1422 return RTMsgErrorExit(RTEXITCODE_FAILURE, "failed to open release log (%s, %Rrc)", szErrorFile, vrc);
1423 }
1424#endif
1425
1426#ifndef VBOX_ONLY_DOCS
1427 /*
1428 * Initialize COM.
1429 */
1430 using namespace com;
1431 HRESULT hrc = com::Initialize();
1432 if (FAILED(hrc))
1433 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Failed to initialize COM!");
1434
1435 hrc = g_pVirtualBoxClient.createInprocObject(CLSID_VirtualBoxClient);
1436 if (FAILED(hrc))
1437 {
1438 RTMsgError("failed to create the VirtualBoxClient object!");
1439 com::ErrorInfo info;
1440 if (!info.isFullAvailable() && !info.isBasicAvailable())
1441 {
1442 com::GluePrintRCMessage(hrc);
1443 RTMsgError("Most likely, the VirtualBox COM server is not running or failed to start.");
1444 }
1445 else
1446 com::GluePrintErrorInfo(info);
1447 return RTEXITCODE_FAILURE;
1448 }
1449
1450 hrc = createGlobalObjects();
1451 if (FAILED(hrc))
1452 return RTEXITCODE_FAILURE;
1453
1454 HandlerArg handlerArg = { argc, argv };
1455 RTEXITCODE rcExit = balloonCtrlMain(&handlerArg);
1456
1457 EventQueue::getMainEventQueue()->processEventQueue(0);
1458
1459 deleteGlobalObjects();
1460
1461 g_pVirtualBoxClient.setNull();
1462
1463 com::Shutdown();
1464
1465 return rcExit;
1466#else /* VBOX_ONLY_DOCS */
1467 return RTEXITCODE_SUCCESS;
1468#endif /* VBOX_ONLY_DOCS */
1469}
1470
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