VirtualBox

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

Last change on this file since 60756 was 60648, checked in by vboxsync, 9 years ago

VBoxBalloonCtrl: make gcc happy

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 27.9 KB
Line 
1/* $Id: VBoxModBallooning.cpp 60648 2016-04-22 13:13:18Z vboxsync $ */
2/** @file
3 * VBoxModBallooning - Module for handling the automatic ballooning of VMs.
4 */
5
6/*
7 * Copyright (C) 2011-2016 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/errorprint.h>
24#endif /* !VBOX_ONLY_DOCS */
25
26#include "VBoxWatchdogInternal.h"
27#include <iprt/system.h>
28
29using namespace com;
30
31#define VBOX_MOD_BALLOONING_NAME "balloon"
32
33/*********************************************************************************************************************************
34* Local Structures *
35*********************************************************************************************************************************/
36
37/**
38 * The module's RTGetOpt-IDs for the command line.
39 */
40enum GETOPTDEF_BALLOONCTRL
41{
42 GETOPTDEF_BALLOONCTRL_BALLOONINC = 2000,
43 GETOPTDEF_BALLOONCTRL_BALLOONDEC,
44 GETOPTDEF_BALLOONCTRL_BALLOONLOWERLIMIT,
45 GETOPTDEF_BALLOONCTRL_BALLOONMAX,
46 GETOPTDEF_BALLOONCTRL_BALLOONSAFETY,
47 GETOPTDEF_BALLOONCTRL_TIMEOUTMS,
48 GETOPTDEF_BALLOONCTRL_GROUPS
49};
50
51/**
52 * The module's command line arguments.
53 */
54static const RTGETOPTDEF g_aBalloonOpts[] = {
55 { "--balloon-dec", GETOPTDEF_BALLOONCTRL_BALLOONDEC, RTGETOPT_REQ_UINT32 },
56 { "--balloon-groups", GETOPTDEF_BALLOONCTRL_GROUPS, RTGETOPT_REQ_STRING },
57 { "--balloon-inc", GETOPTDEF_BALLOONCTRL_BALLOONINC, RTGETOPT_REQ_UINT32 },
58 { "--balloon-interval", GETOPTDEF_BALLOONCTRL_TIMEOUTMS, RTGETOPT_REQ_UINT32 },
59 { "--balloon-lower-limit", GETOPTDEF_BALLOONCTRL_BALLOONLOWERLIMIT, RTGETOPT_REQ_UINT32 },
60 { "--balloon-max", GETOPTDEF_BALLOONCTRL_BALLOONMAX, RTGETOPT_REQ_UINT32 },
61 { "--balloon-safety-margin", GETOPTDEF_BALLOONCTRL_BALLOONSAFETY, RTGETOPT_REQ_UINT32 }
62};
63
64/** The ballooning module's payload. */
65typedef struct VBOXWATCHDOG_BALLOONCTRL_PAYLOAD
66{
67 /** Last (most recent) ballooning size reported by the guest. */
68 unsigned long ulBalloonCurLast;
69 /** Last (most recent) ballooning request received. */
70 unsigned long ulBalloonReqLast;
71} VBOXWATCHDOG_BALLOONCTRL_PAYLOAD, *PVBOXWATCHDOG_BALLOONCTRL_PAYLOAD;
72
73/*********************************************************************************************************************************
74* Globals *
75*********************************************************************************************************************************/
76
77static unsigned long g_ulMemoryBalloonTimeoutMS = 30 * 1000;
78static unsigned long g_ulMemoryBalloonIncrementMB = 256;
79static unsigned long g_ulMemoryBalloonDecrementMB = 128;
80/** Command line: Global balloon limit (in MB) for all VMs. Default is 0, which means
81 * no global limit is set. See balloonGetMaxSize() for more information. */
82static unsigned long g_ulMemoryBalloonMaxMB = 0;
83static unsigned long g_ulMemoryBalloonLowerLimitMB = 128;
84static unsigned long g_ulMemoryBalloonSafetyMB = 1024;
85
86/*********************************************************************************************************************************
87* Local Function Prototypes *
88*********************************************************************************************************************************/
89static int balloonSetSize(PVBOXWATCHDOG_MACHINE pMachine, unsigned long ulBalloonCur);
90
91/**
92 * Retrieves the current delta value
93 *
94 * @return long Delta (MB) of the balloon to be deflated (<0) or inflated (>0).
95 * @param pMachine Pointer to the machine's internal structure.
96 * @param ulGuestMemFree The guest's current free memory (MB).
97 * @param ulBalloonOld The balloon's current (old) size (MB).
98 * @param ulBalloonNew The balloon's new size (MB).
99 * @param ulBalloonMax The maximum ballooning size (MB) it can inflate to.
100 */
101static long balloonGetDelta(PVBOXWATCHDOG_MACHINE pMachine,
102 unsigned long ulGuestMemFree,
103 unsigned long ulBalloonOld, unsigned long ulBalloonNew, unsigned long ulBalloonMax)
104{
105 serviceLogVerbose(("[%ls] ulGuestMemFree=%RU32, ulBalloonOld=%RU32, ulBalloonNew=%RU32, ulBalloonMax=%RU32\n",
106 pMachine->strName.raw(), ulGuestMemFree, ulBalloonOld, ulBalloonNew, ulBalloonMax));
107
108 /* Make sure that the requested new ballooning size does not
109 * exceed the maximum ballooning size (if set). */
110 if ( ulBalloonMax
111 && (ulBalloonNew > ulBalloonMax))
112 {
113 ulBalloonNew = ulBalloonMax;
114 }
115
116 long lBalloonDelta = 0;
117 if (ulGuestMemFree < g_ulMemoryBalloonLowerLimitMB)
118 {
119 /* Guest is running low on memory, we need to
120 * deflate the balloon. */
121 lBalloonDelta = (g_ulMemoryBalloonDecrementMB * -1);
122
123 /* Ensure that the delta will not return a negative
124 * balloon size. */
125 if ((long)ulBalloonOld + lBalloonDelta < 0)
126 lBalloonDelta = 0;
127 }
128 else if (ulBalloonNew > ulBalloonOld) /* Inflate. */
129 {
130 /* We want to inflate the balloon if we have room. */
131 unsigned long ulIncrement = g_ulMemoryBalloonIncrementMB;
132 while (ulIncrement >= 16 && (ulGuestMemFree - ulIncrement) < g_ulMemoryBalloonLowerLimitMB)
133 ulIncrement = (ulIncrement / 2);
134
135 if ((ulGuestMemFree - ulIncrement) > g_ulMemoryBalloonLowerLimitMB)
136 lBalloonDelta = (long)ulIncrement;
137
138 /* Make sure we're still within bounds. */
139 Assert(lBalloonDelta >= 0);
140 if (ulBalloonOld + lBalloonDelta > ulBalloonNew)
141 lBalloonDelta = RT_MIN(g_ulMemoryBalloonIncrementMB, ulBalloonNew - ulBalloonOld);
142 }
143 else if (ulBalloonNew < ulBalloonOld) /* Deflate. */
144 {
145 lBalloonDelta = RT_MIN(g_ulMemoryBalloonDecrementMB, ulBalloonOld - ulBalloonNew) * -1;
146 }
147
148 /* Limit the ballooning to the available host memory, leaving some free.
149 * If anything fails clamp the delta to 0. */
150 if (lBalloonDelta < 0)
151 {
152 uint64_t cbSafety = (uint64_t)g_ulMemoryBalloonSafetyMB * _1M;
153 uint64_t cbHostRamAvail = 0;
154 int vrc = RTSystemQueryAvailableRam(&cbHostRamAvail);
155 if (RT_SUCCESS(vrc))
156 {
157 if (cbHostRamAvail < cbSafety)
158 lBalloonDelta = 0;
159 else if ((uint64_t)(-lBalloonDelta) > (cbHostRamAvail - cbSafety) / _1M)
160 lBalloonDelta = -(long)((cbHostRamAvail - cbSafety) / _1M);
161 }
162 else
163 lBalloonDelta = 0;
164 }
165
166 return lBalloonDelta;
167}
168
169/**
170 * Determines the maximum balloon size to set for the specified machine.
171 *
172 * @return unsigned long Maximum ballooning size (in MB), 0 if no maximum set.
173 * @param pMachine Machine to determine maximum ballooning size for.
174 */
175static unsigned long balloonGetMaxSize(PVBOXWATCHDOG_MACHINE pMachine)
176{
177 const ComPtr<IMachine> &rptrMachine = pMachine->machine;
178
179 /*
180 * Is a maximum ballooning size set? Make sure we're within bounds.
181 *
182 * The maximum balloning size can be set
183 * - via global extra-data ("VBoxInternal/Guest/BalloonSizeMax")
184 * - via command line ("--balloon-max")
185 *
186 * Precedence from top to bottom.
187 */
188 unsigned long ulBalloonMax = 0;
189 char szSource[64];
190
191 Bstr strValue;
192 HRESULT hr = g_pVirtualBox->GetExtraData(Bstr("VBoxInternal/Guest/BalloonSizeMax").raw(),
193 strValue.asOutParam());
194 if ( SUCCEEDED(hr)
195 && strValue.isNotEmpty())
196 {
197 ulBalloonMax = Utf8Str(strValue).toUInt32();
198 if (g_fVerbose)
199 RTStrPrintf(szSource, sizeof(szSource), "global extra-data");
200 }
201
202 if (strValue.isEmpty())
203 {
204 Assert(ulBalloonMax == 0);
205
206 ulBalloonMax = g_ulMemoryBalloonMaxMB;
207 if (g_fVerbose)
208 RTStrPrintf(szSource, sizeof(szSource), "command line");
209 }
210
211 serviceLogVerbose(("[%ls] Maximum balloning size is (%s): %RU32MB\n", pMachine->strName.raw(), szSource, ulBalloonMax));
212 return ulBalloonMax;
213}
214
215/**
216 * Determines the current (set) balloon size of the specified machine.
217 *
218 * @return IPRT status code.
219 * @param pMachine Machine to determine maximum ballooning size for.
220 * @param pulBalloonCur Where to store the current (set) balloon size (in MB) on success.
221 */
222static int balloonGetCurrentSize(PVBOXWATCHDOG_MACHINE pMachine, unsigned long *pulBalloonCur)
223{
224 LONG lBalloonCur;
225 int vrc = getMetric(pMachine, L"Guest/RAM/Usage/Balloon", &lBalloonCur);
226 if (RT_SUCCESS(vrc))
227 {
228 lBalloonCur /= 1024; /* Convert to MB. */
229 if (pulBalloonCur)
230 *pulBalloonCur = (unsigned long)lBalloonCur;
231 }
232
233 return vrc;
234}
235
236/**
237 * Determines the requested balloon size to set for the specified machine.
238 *
239 * @return unsigned long Requested ballooning size (in MB), 0 if ballooning should be disabled.
240 * @param pMachine Machine to determine maximum ballooning size for.
241 */
242static unsigned long balloonGetRequestedSize(PVBOXWATCHDOG_MACHINE pMachine)
243{
244 const ComPtr<IMachine> &rptrMachine = pMachine->machine;
245
246 /*
247 * The maximum balloning size can be set
248 * - via per-VM extra-data ("VBoxInternal2/Watchdog/BalloonCtrl/BalloonSizeMax")
249 * - via per-VM extra-data (legacy) ("VBoxInternal/Guest/BalloonSizeMax")
250 *
251 * Precedence from top to bottom.
252 */
253 unsigned long ulBalloonReq = 0;
254 char szSource[64];
255
256 Bstr strValue;
257 HRESULT hr = rptrMachine->GetExtraData(Bstr("VBoxInternal2/Watchdog/BalloonCtrl/BalloonSizeMax").raw(),
258 strValue.asOutParam());
259 if ( SUCCEEDED(hr)
260 && strValue.isNotEmpty())
261 {
262 ulBalloonReq = Utf8Str(strValue).toUInt32();
263 if (g_fVerbose)
264 RTStrPrintf(szSource, sizeof(szSource), "per-VM extra-data");
265 }
266 else
267 {
268 hr = rptrMachine->GetExtraData(Bstr("VBoxInternal/Guest/BalloonSizeMax").raw(),
269 strValue.asOutParam());
270 if ( SUCCEEDED(hr)
271 && strValue.isNotEmpty())
272 {
273 ulBalloonReq = Utf8Str(strValue).toUInt32();
274 if (g_fVerbose)
275 RTStrPrintf(szSource, sizeof(szSource), "per-VM extra-data (legacy)");
276 }
277 }
278
279 if ( FAILED(hr)
280 || strValue.isEmpty())
281 {
282 ulBalloonReq = 0;
283 if (g_fVerbose)
284 RTStrPrintf(szSource, sizeof(szSource), "none (disabled)");
285 }
286
287 serviceLogVerbose(("[%ls] Requested balloning size is (%s): %RU32MB\n", pMachine->strName.raw(), szSource, ulBalloonReq));
288 return ulBalloonReq;
289}
290
291/**
292 * Determines whether ballooning for the specified machine is enabled or not.
293 * This can be specified on a per-VM basis or as a globally set value for all VMs.
294 *
295 * @return bool Whether ballooning is enabled or not.
296 * @param pMachine Machine to determine enable status for.
297 */
298static bool balloonIsEnabled(PVBOXWATCHDOG_MACHINE pMachine)
299{
300 const ComPtr<IMachine> &rptrMachine = pMachine->machine;
301
302 bool fEnabled = true; /* By default ballooning is enabled. */
303 char szSource[64];
304
305 Bstr strValue;
306 HRESULT hr = g_pVirtualBox->GetExtraData(Bstr("VBoxInternal/Guest/BalloonEnabled").raw(),
307 strValue.asOutParam());
308 if ( SUCCEEDED(hr)
309 && strValue.isNotEmpty())
310 {
311 if (g_fVerbose)
312 RTStrPrintf(szSource, sizeof(szSource), "global extra-data");
313 }
314 else
315 {
316 hr = rptrMachine->GetExtraData(Bstr("VBoxInternal2/Watchdog/BalloonCtrl/BalloonEnabled").raw(),
317 strValue.asOutParam());
318 if (SUCCEEDED(hr))
319 {
320 if (g_fVerbose)
321 RTStrPrintf(szSource, sizeof(szSource), "per-VM extra-data");
322 }
323 }
324
325 if (strValue.isNotEmpty())
326 {
327 fEnabled = RT_BOOL(Utf8Str(strValue).toUInt32());
328 serviceLogVerbose(("[%ls] Ballooning is forced to %s (%s)\n",
329 pMachine->strName.raw(), fEnabled ? "enabled" : "disabled", szSource));
330 }
331
332 return fEnabled;
333}
334
335/**
336 * Indicates whether ballooning on the specified machine state is
337 * possible -- this only is true if the machine is up and running.
338 *
339 * @return bool Flag indicating whether the VM is running or not.
340 * @param enmState The VM's machine state to judge whether it's running or not.
341 */
342static bool balloonIsPossible(MachineState_T enmState)
343{
344 switch (enmState)
345 {
346 case MachineState_Running:
347#if 0
348 /* Not required for ballooning. */
349 case MachineState_Teleporting:
350 case MachineState_LiveSnapshotting:
351 case MachineState_Paused:
352 case MachineState_TeleportingPausedVM:
353#endif
354 return true;
355 default:
356 break;
357 }
358 return false;
359}
360
361int balloonMachineSetup(const Bstr& strUuid)
362{
363 int vrc = VINF_SUCCESS;
364
365 do
366 {
367 PVBOXWATCHDOG_MACHINE pMachine = getMachine(strUuid);
368 AssertPtrBreakStmt(pMachine, vrc=VERR_INVALID_PARAMETER);
369
370 ComPtr<IMachine> m = pMachine->machine;
371
372 /*
373 * Setup metrics required for ballooning.
374 */
375 com::SafeArray<BSTR> metricNames(1);
376 com::SafeIfaceArray<IUnknown> metricObjects(1);
377 com::SafeIfaceArray<IPerformanceMetric> metricAffected;
378
379 Bstr strMetricNames(L"Guest/RAM/Usage");
380 strMetricNames.cloneTo(&metricNames[0]);
381
382 HRESULT rc = m.queryInterfaceTo(&metricObjects[0]);
383
384#ifdef VBOX_WATCHDOG_GLOBAL_PERFCOL
385 CHECK_ERROR_BREAK(g_pPerfCollector, SetupMetrics(ComSafeArrayAsInParam(metricNames),
386 ComSafeArrayAsInParam(metricObjects),
387 5 /* 5 seconds */,
388 1 /* One sample is enough */,
389 ComSafeArrayAsOutParam(metricAffected)));
390#else
391 ComPtr<IPerformanceCollector> coll = pMachine->collector;
392
393 CHECK_ERROR_BREAK(g_pVirtualBox, COMGETTER(PerformanceCollector)(coll.asOutParam()));
394 CHECK_ERROR_BREAK(coll, SetupMetrics(ComSafeArrayAsInParam(metricNames),
395 ComSafeArrayAsInParam(metricObjects),
396 5 /* 5 seconds */,
397 1 /* One sample is enough */,
398 ComSafeArrayAsOutParam(metricAffected)));
399#endif
400 if (FAILED(rc))
401 vrc = VERR_COM_IPRT_ERROR; /* @todo Find better rc! */
402
403 } while (0);
404
405 return vrc;
406}
407
408/**
409 * Does the actual ballooning and assumes the machine is
410 * capable and ready for ballooning.
411 *
412 * @return IPRT status code.
413 * @param pMachine Pointer to the machine's internal structure.
414 */
415static int balloonMachineUpdate(PVBOXWATCHDOG_MACHINE pMachine)
416{
417 AssertPtrReturn(pMachine, VERR_INVALID_POINTER);
418
419 /*
420 * Get metrics collected at this point.
421 */
422 LONG lGuestMemFree;
423 unsigned long ulBalloonCur;
424
425 int vrc = getMetric(pMachine, L"Guest/RAM/Usage/Free", &lGuestMemFree);
426 if (RT_SUCCESS(vrc))
427 vrc = balloonGetCurrentSize(pMachine, &ulBalloonCur);
428
429 if (RT_SUCCESS(vrc))
430 {
431 /* If guest statistics are not up and running yet, skip this iteration and try next time. */
432 if (lGuestMemFree <= 0)
433 {
434#ifdef DEBUG
435 serviceLogVerbose(("[%ls] No metrics available yet!\n", pMachine->strName.raw()));
436#endif
437 return VINF_SUCCESS;
438 }
439
440 lGuestMemFree /= 1024;
441
442 PVBOXWATCHDOG_BALLOONCTRL_PAYLOAD pData = (PVBOXWATCHDOG_BALLOONCTRL_PAYLOAD)
443 payloadFrom(pMachine, VBOX_MOD_BALLOONING_NAME);
444 AssertPtr(pData);
445
446 /* Determine if ballooning is enabled or disabled. */
447 bool fEnabled = balloonIsEnabled(pMachine);
448
449 /* Determine the current set maximum balloon size. */
450 unsigned long ulBalloonMax = balloonGetMaxSize(pMachine);
451
452 /* Determine the requested balloon size. */
453 unsigned long ulBalloonReq = balloonGetRequestedSize(pMachine);
454
455 serviceLogVerbose(("[%ls] Free RAM (MB): %RI32, Ballooning: Current=%RU32MB, Requested=%RU32MB, Maximum=%RU32MB\n",
456 pMachine->strName.raw(), lGuestMemFree, ulBalloonCur, ulBalloonReq, ulBalloonMax));
457
458 if ( ulBalloonMax
459 && (ulBalloonReq > ulBalloonMax))
460 {
461 if (pData->ulBalloonReqLast != ulBalloonReq)
462 serviceLog("[%ls] Warning: Requested ballooning size (%RU32MB) exceeds set maximum ballooning size (%RU32MB), limiting ...\n",
463 pMachine->strName.raw(), ulBalloonReq, ulBalloonMax);
464 }
465
466 /* Calculate current balloon delta. */
467 long lBalloonDelta = balloonGetDelta(pMachine,
468 (unsigned long)lGuestMemFree, ulBalloonCur, ulBalloonReq, ulBalloonMax);
469#ifdef DEBUG
470 serviceLogVerbose(("[%ls] lBalloonDelta=%RI32\n", pMachine->strName.raw(), lBalloonDelta));
471#endif
472 if (lBalloonDelta) /* Only do ballooning if there's really smth. to change ... */
473 {
474 ulBalloonCur = ulBalloonCur + lBalloonDelta;
475
476 if (fEnabled)
477 {
478 serviceLog("[%ls] %s balloon by %RU32MB to %RU32MB ...\n",
479 pMachine->strName.raw(), lBalloonDelta > 0 ? "Inflating" : "Deflating", RT_ABS(lBalloonDelta), ulBalloonCur);
480 vrc = balloonSetSize(pMachine, ulBalloonCur);
481 }
482 else
483 serviceLogVerbose(("[%ls] Requested %s balloon by %RU32MB to %RU32MB, but ballooning is disabled\n",
484 pMachine->strName.raw(), lBalloonDelta > 0 ? "inflating" : "deflating",
485 RT_ABS(lBalloonDelta), ulBalloonCur));
486 }
487
488 if (ulBalloonCur != pData->ulBalloonCurLast)
489 {
490 /* If ballooning is disabled, always bolt down the ballooning size to 0. */
491 if (!fEnabled)
492 {
493 serviceLogVerbose(("[%ls] Ballooning is disabled, forcing to 0\n", pMachine->strName.raw()));
494 int vrc2 = balloonSetSize(pMachine, 0);
495 if (RT_FAILURE(vrc2))
496 serviceLog("[%ls] Error disabling ballooning, rc=%Rrc\n", pMachine->strName.raw(), vrc2);
497 }
498 }
499
500 pData->ulBalloonCurLast = ulBalloonCur;
501 pData->ulBalloonReqLast = ulBalloonReq;
502 }
503 else
504 serviceLog("[%ls] Error retrieving metrics, rc=%Rrc\n", pMachine->strName.raw(), vrc);
505
506 return vrc;
507}
508
509static int balloonSetSize(PVBOXWATCHDOG_MACHINE pMachine, unsigned long ulBalloonCur)
510{
511 int vrc = VINF_SUCCESS;
512
513 serviceLogVerbose(("[%ls] Setting balloon size to %RU32MB ...\n", pMachine->strName.raw(), ulBalloonCur));
514
515 if (g_fDryrun)
516 return VINF_SUCCESS;
517
518 /* Open a session for the VM. */
519 HRESULT rc;
520 CHECK_ERROR_RET(pMachine->machine, LockMachine(g_pSession, LockType_Shared), VERR_ACCESS_DENIED);
521
522 do
523 {
524 /* Get the associated console. */
525 ComPtr<IConsole> console;
526 CHECK_ERROR_BREAK(g_pSession, COMGETTER(Console)(console.asOutParam()));
527
528 ComPtr <IGuest> guest;
529 rc = console->COMGETTER(Guest)(guest.asOutParam());
530 if (SUCCEEDED(rc))
531 CHECK_ERROR_BREAK(guest, COMSETTER(MemoryBalloonSize)((LONG)ulBalloonCur));
532 else
533 serviceLog("Error: Unable to set new balloon size %RU32 for machine '%ls', rc=%Rhrc\n",
534 ulBalloonCur, pMachine->strName.raw(), rc);
535 if (FAILED(rc))
536 vrc = VERR_COM_IPRT_ERROR;
537
538 } while (0);
539
540
541 /* Unlock the machine again. */
542 CHECK_ERROR_RET(g_pSession, UnlockMachine(), VERR_ACCESS_DENIED);
543
544 return vrc;
545}
546
547/* Callbacks. */
548static DECLCALLBACK(int) VBoxModBallooningPreInit(void)
549{
550 return VINF_SUCCESS;
551}
552
553static DECLCALLBACK(int) VBoxModBallooningOption(int argc, char *argv[], int *piConsumed)
554{
555 if (!argc) /* Take a shortcut. */
556 return -1;
557
558 AssertPtrReturn(argv, VERR_INVALID_POINTER);
559 AssertPtrReturn(piConsumed, VERR_INVALID_POINTER);
560
561 RTGETOPTSTATE GetState;
562 int rc = RTGetOptInit(&GetState, argc, argv,
563 g_aBalloonOpts, RT_ELEMENTS(g_aBalloonOpts),
564 0 /* First */, 0 /*fFlags*/);
565 if (RT_FAILURE(rc))
566 return rc;
567
568 rc = 0; /* Set default parsing result to valid. */
569
570 int c;
571 RTGETOPTUNION ValueUnion;
572 while ((c = RTGetOpt(&GetState, &ValueUnion)))
573 {
574 switch (c)
575 {
576 case GETOPTDEF_BALLOONCTRL_BALLOONDEC:
577 g_ulMemoryBalloonDecrementMB = ValueUnion.u32;
578 break;
579
580 case GETOPTDEF_BALLOONCTRL_BALLOONINC:
581 g_ulMemoryBalloonIncrementMB = ValueUnion.u32;
582 break;
583
584 case GETOPTDEF_BALLOONCTRL_GROUPS:
585 /** @todo Add ballooning groups cmd line arg. */
586 break;
587
588 case GETOPTDEF_BALLOONCTRL_BALLOONLOWERLIMIT:
589 g_ulMemoryBalloonLowerLimitMB = ValueUnion.u32;
590 break;
591
592 case GETOPTDEF_BALLOONCTRL_BALLOONMAX:
593 g_ulMemoryBalloonMaxMB = ValueUnion.u32;
594 break;
595
596 case GETOPTDEF_BALLOONCTRL_BALLOONSAFETY:
597 g_ulMemoryBalloonSafetyMB = ValueUnion.u32;
598 break;
599
600 /** @todo This option is a common module option! Put
601 * this into a utility function! */
602 case GETOPTDEF_BALLOONCTRL_TIMEOUTMS:
603 g_ulMemoryBalloonTimeoutMS = ValueUnion.u32;
604 if (g_ulMemoryBalloonTimeoutMS < 500)
605 g_ulMemoryBalloonTimeoutMS = 500;
606 break;
607
608 default:
609 rc = -1; /* We don't handle this option, skip. */
610 break;
611 }
612
613 /* At the moment we only process one option at a time. */
614 break;
615 }
616
617 *piConsumed += GetState.iNext - 1;
618
619 return rc;
620}
621
622static DECLCALLBACK(int) VBoxModBallooningInit(void)
623{
624 if (!g_ulMemoryBalloonTimeoutMS)
625 cfgGetValueULong(g_pVirtualBox, NULL /* Machine */,
626 "VBoxInternal2/Watchdog/BalloonCtrl/TimeoutMS", NULL /* Per-machine */,
627 &g_ulMemoryBalloonTimeoutMS, 30 * 1000 /* Default is 30 seconds timeout. */);
628
629 if (!g_ulMemoryBalloonIncrementMB)
630 cfgGetValueULong(g_pVirtualBox, NULL /* Machine */,
631 "VBoxInternal2/Watchdog/BalloonCtrl/BalloonIncrementMB", NULL /* Per-machine */,
632 &g_ulMemoryBalloonIncrementMB, 256);
633
634 if (!g_ulMemoryBalloonDecrementMB)
635 cfgGetValueULong(g_pVirtualBox, NULL /* Machine */,
636 "VBoxInternal2/Watchdog/BalloonCtrl/BalloonDecrementMB", NULL /* Per-machine */,
637 &g_ulMemoryBalloonDecrementMB, 128);
638
639 if (!g_ulMemoryBalloonLowerLimitMB)
640 cfgGetValueULong(g_pVirtualBox, NULL /* Machine */,
641 "VBoxInternal2/Watchdog/BalloonCtrl/BalloonLowerLimitMB", NULL /* Per-machine */,
642 &g_ulMemoryBalloonLowerLimitMB, 128);
643
644 return VINF_SUCCESS;
645}
646
647static DECLCALLBACK(int) VBoxModBallooningMain(void)
648{
649 static uint64_t s_msLast = RTTimeMilliTS();
650 uint64_t msDelta = RTTimeMilliTS() - s_msLast;
651 if (msDelta <= g_ulMemoryBalloonTimeoutMS)
652 return VINF_SUCCESS;
653
654 int rc = VINF_SUCCESS;
655
656 /** @todo Provide API for enumerating/working w/ machines inside a module! */
657 mapVMIter it = g_mapVM.begin();
658 while (it != g_mapVM.end())
659 {
660 MachineState_T state = getMachineState(&it->second);
661
662 /* Our actual ballooning criteria. */
663 if (balloonIsPossible(state))
664 {
665 rc = balloonMachineUpdate(&it->second);
666 AssertRC(rc);
667 }
668 if (RT_FAILURE(rc))
669 break;
670
671 ++it;
672 }
673
674 s_msLast = RTTimeMilliTS();
675 return rc;
676}
677
678static DECLCALLBACK(int) VBoxModBallooningStop(void)
679{
680 return VINF_SUCCESS;
681}
682
683static DECLCALLBACK(void) VBoxModBallooningTerm(void)
684{
685}
686
687static DECLCALLBACK(int) VBoxModBallooningOnMachineRegistered(const Bstr &strUuid)
688{
689 PVBOXWATCHDOG_MACHINE pMachine = getMachine(strUuid);
690 AssertPtrReturn(pMachine, VERR_INVALID_PARAMETER);
691
692 PVBOXWATCHDOG_BALLOONCTRL_PAYLOAD pData;
693 int rc = payloadAlloc(pMachine, VBOX_MOD_BALLOONING_NAME,
694 sizeof(VBOXWATCHDOG_BALLOONCTRL_PAYLOAD), (void**)&pData);
695 if (RT_SUCCESS(rc))
696 rc = balloonMachineUpdate(pMachine);
697
698 return rc;
699}
700
701static DECLCALLBACK(int) VBoxModBallooningOnMachineUnregistered(const Bstr &strUuid)
702{
703 PVBOXWATCHDOG_MACHINE pMachine = getMachine(strUuid);
704 AssertPtrReturn(pMachine, VERR_INVALID_PARAMETER);
705
706 payloadFree(pMachine, VBOX_MOD_BALLOONING_NAME);
707
708 return VINF_SUCCESS;
709}
710
711static DECLCALLBACK(int) VBoxModBallooningOnMachineStateChanged(const Bstr &strUuid,
712 MachineState_T enmState)
713{
714 PVBOXWATCHDOG_MACHINE pMachine = getMachine(strUuid);
715 /* Note: The machine state will change to "setting up" when machine gets deleted,
716 * so pMachine might be NULL here. */
717 if (!pMachine)
718 return VINF_SUCCESS;
719
720 return balloonMachineUpdate(pMachine);
721}
722
723static DECLCALLBACK(int) VBoxModBallooningOnServiceStateChanged(bool fAvailable)
724{
725 return VINF_SUCCESS;
726}
727
728/**
729 * The 'balloonctrl' module description.
730 */
731VBOXMODULE g_ModBallooning =
732{
733 /* pszName. */
734 VBOX_MOD_BALLOONING_NAME,
735 /* pszDescription. */
736 "Memory Ballooning Control",
737 /* pszDepends. */
738 NULL,
739 /* uPriority. */
740 0 /* Not used */,
741 /* pszUsage. */
742 " [--balloon-dec=<MB>] [--balloon-groups=<string>] [--balloon-inc=<MB>]\n"
743 " [--balloon-interval=<ms>] [--balloon-lower-limit=<MB>]\n"
744 " [--balloon-max=<MB>]\n",
745 /* pszOptions. */
746 "--balloon-dec Sets the ballooning decrement in MB (128 MB).\n"
747 "--balloon-groups Sets the VM groups for ballooning (all).\n"
748 "--balloon-inc Sets the ballooning increment in MB (256 MB).\n"
749 "--balloon-interval Sets the check interval in ms (30 seconds).\n"
750 "--balloon-lower-limit Sets the ballooning lower limit in MB (64 MB).\n"
751 "--balloon-max Sets the balloon maximum limit in MB (0 MB).\n"
752 " Specifying \"0\" means disabled ballooning.\n"
753#if 1
754 /* (Legacy) note. */
755 "Set \"VBoxInternal/Guest/BalloonSizeMax\" for a per-VM maximum ballooning size.\n"
756#endif
757 "--balloon-safety-margin Free memory when deflating a balloon in MB (1024 MB).\n"
758 ,
759 /* methods. */
760 VBoxModBallooningPreInit,
761 VBoxModBallooningOption,
762 VBoxModBallooningInit,
763 VBoxModBallooningMain,
764 VBoxModBallooningStop,
765 VBoxModBallooningTerm,
766 /* callbacks. */
767 VBoxModBallooningOnMachineRegistered,
768 VBoxModBallooningOnMachineUnregistered,
769 VBoxModBallooningOnMachineStateChanged,
770 VBoxModBallooningOnServiceStateChanged
771};
772
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