VirtualBox

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

Last change on this file since 62493 was 62471, checked in by vboxsync, 8 years ago

Misc: scm

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