VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxService/VBoxServiceStats.cpp@ 34079

Last change on this file since 34079 was 32813, checked in by vboxsync, 14 years ago

VBoxService: Some division through zero checks.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 28.1 KB
Line 
1/* $Id: VBoxServiceStats.cpp 32813 2010-09-29 11:50:19Z vboxsync $ */
2/** @file
3 * VBoxStats - Guest statistics notification
4 */
5
6/*
7 * Copyright (C) 2006-2010 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* Header Files *
20*******************************************************************************/
21#if defined(RT_OS_WINDOWS)
22# ifdef TARGET_NT4
23# undef _WIN32_WINNT
24# define _WIN32_WINNT 0x501
25# endif
26# include <windows.h>
27# include <psapi.h>
28# include <winternl.h>
29
30#elif defined(RT_OS_LINUX)
31# include <iprt/ctype.h>
32# include <iprt/stream.h>
33# include <unistd.h>
34
35#elif defined(RT_OS_SOLARIS)
36# include <kstat.h>
37# include <sys/sysinfo.h>
38# include <unistd.h>
39#else
40/** @todo port me. */
41
42#endif
43
44#include <iprt/assert.h>
45#include <iprt/mem.h>
46#include <VBox/param.h>
47#include <iprt/semaphore.h>
48#include <iprt/string.h>
49#include <iprt/system.h>
50#include <iprt/time.h>
51#include <iprt/thread.h>
52#include <VBox/VBoxGuestLib.h>
53#include "VBoxServiceInternal.h"
54#include "VBoxServiceUtils.h"
55
56
57/*******************************************************************************
58* Structures and Typedefs *
59*******************************************************************************/
60typedef struct _VBOXSTATSCONTEXT
61{
62 RTMSINTERVAL cMsStatInterval;
63
64 uint64_t au64LastCpuLoad_Idle[VMM_MAX_CPU_COUNT];
65 uint64_t au64LastCpuLoad_Kernel[VMM_MAX_CPU_COUNT];
66 uint64_t au64LastCpuLoad_User[VMM_MAX_CPU_COUNT];
67 uint64_t au64LastCpuLoad_Nice[VMM_MAX_CPU_COUNT];
68
69#ifdef RT_OS_WINDOWS
70 NTSTATUS (WINAPI *pfnNtQuerySystemInformation)(SYSTEM_INFORMATION_CLASS SystemInformationClass, PVOID SystemInformation, ULONG SystemInformationLength, PULONG ReturnLength);
71 void (WINAPI *pfnGlobalMemoryStatusEx)(LPMEMORYSTATUSEX lpBuffer);
72 BOOL (WINAPI *pfnGetPerformanceInfo)(PPERFORMANCE_INFORMATION pPerformanceInformation, DWORD cb);
73#endif
74} VBOXSTATSCONTEXT;
75
76
77/*******************************************************************************
78* Global Variables *
79*******************************************************************************/
80static VBOXSTATSCONTEXT gCtx = {0};
81
82/** The semaphore we're blocking on. */
83static RTSEMEVENTMULTI g_VMStatEvent = NIL_RTSEMEVENTMULTI;
84
85
86/** @copydoc VBOXSERVICE::pfnPreInit */
87static DECLCALLBACK(int) VBoxServiceVMStatsPreInit(void)
88{
89 return VINF_SUCCESS;
90}
91
92
93/** @copydoc VBOXSERVICE::pfnOption */
94static DECLCALLBACK(int) VBoxServiceVMStatsOption(const char **ppszShort, int argc, char **argv, int *pi)
95{
96 NOREF(ppszShort);
97 NOREF(argc);
98 NOREF(argv);
99 NOREF(pi);
100 return VINF_SUCCESS;
101}
102
103
104/** @copydoc VBOXSERVICE::pfnInit */
105static DECLCALLBACK(int) VBoxServiceVMStatsInit(void)
106{
107 VBoxServiceVerbose(3, "VBoxServiceVMStatsInit\n");
108
109 int rc = RTSemEventMultiCreate(&g_VMStatEvent);
110 AssertRCReturn(rc, rc);
111
112 gCtx.cMsStatInterval = 0; /* default; update disabled */
113 RT_ZERO(gCtx.au64LastCpuLoad_Idle);
114 RT_ZERO(gCtx.au64LastCpuLoad_Kernel);
115 RT_ZERO(gCtx.au64LastCpuLoad_User);
116 RT_ZERO(gCtx.au64LastCpuLoad_Nice);
117
118 rc = VbglR3StatQueryInterval(&gCtx.cMsStatInterval);
119 if (RT_SUCCESS(rc))
120 VBoxServiceVerbose(3, "VBoxStatsInit: New statistics interval %u seconds\n", gCtx.cMsStatInterval);
121 else
122 VBoxServiceVerbose(3, "VBoxStatsInit: DeviceIoControl failed with %d\n", rc);
123
124#ifdef RT_OS_WINDOWS
125 /** @todo Use RTLdr instead of LoadLibrary/GetProcAddress here! */
126
127 /* NtQuerySystemInformation might be dropped in future releases, so load it dynamically as per Microsoft's recommendation */
128 HMODULE hMod = LoadLibrary("NTDLL.DLL");
129 if (hMod)
130 {
131 *(uintptr_t *)&gCtx.pfnNtQuerySystemInformation = (uintptr_t)GetProcAddress(hMod, "NtQuerySystemInformation");
132 if (gCtx.pfnNtQuerySystemInformation)
133 VBoxServiceVerbose(3, "VBoxStatsInit: gCtx.pfnNtQuerySystemInformation = %x\n", gCtx.pfnNtQuerySystemInformation);
134 else
135 {
136 VBoxServiceVerbose(3, "VBoxStatsInit: NTDLL.NtQuerySystemInformation not found!\n");
137 return VERR_SERVICE_DISABLED;
138 }
139 }
140
141 /* GlobalMemoryStatus is win2k and up, so load it dynamically */
142 hMod = LoadLibrary("KERNEL32.DLL");
143 if (hMod)
144 {
145 *(uintptr_t *)&gCtx.pfnGlobalMemoryStatusEx = (uintptr_t)GetProcAddress(hMod, "GlobalMemoryStatusEx");
146 if (gCtx.pfnGlobalMemoryStatusEx)
147 VBoxServiceVerbose(3, "VBoxStatsInit: gCtx.GlobalMemoryStatusEx = %x\n", gCtx.pfnGlobalMemoryStatusEx);
148 else
149 {
150 /** @todo Now fails in NT4; do we care? */
151 VBoxServiceVerbose(3, "VBoxStatsInit: KERNEL32.GlobalMemoryStatusEx not found!\n");
152 return VERR_SERVICE_DISABLED;
153 }
154 }
155 /* GetPerformanceInfo is xp and up, so load it dynamically */
156 hMod = LoadLibrary("PSAPI.DLL");
157 if (hMod)
158 {
159 *(uintptr_t *)&gCtx.pfnGetPerformanceInfo = (uintptr_t)GetProcAddress(hMod, "GetPerformanceInfo");
160 if (gCtx.pfnGetPerformanceInfo)
161 VBoxServiceVerbose(3, "VBoxStatsInit: gCtx.pfnGetPerformanceInfo= %x\n", gCtx.pfnGetPerformanceInfo);
162 /* failure is not fatal */
163 }
164#endif /* RT_OS_WINDOWS */
165
166 return VINF_SUCCESS;
167}
168
169
170/**
171 * Gathers VM statistics and reports them to the host.
172 */
173static void VBoxServiceVMStatsReport(void)
174{
175#if defined(RT_OS_WINDOWS)
176 SYSTEM_INFO systemInfo;
177 PSYSTEM_PROCESSOR_PERFORMANCE_INFORMATION pProcInfo;
178 MEMORYSTATUSEX memStatus;
179 uint32_t cbStruct;
180 DWORD cbReturned;
181
182 Assert(gCtx.pfnGlobalMemoryStatusEx && gCtx.pfnNtQuerySystemInformation);
183 if ( !gCtx.pfnGlobalMemoryStatusEx
184 || !gCtx.pfnNtQuerySystemInformation)
185 return;
186
187 /* Clear the report so we don't report garbage should NtQuerySystemInformation
188 behave in an unexpected manner. */
189 VMMDevReportGuestStats req;
190 RT_ZERO(req);
191
192 /* Query and report guest statistics */
193 GetSystemInfo(&systemInfo);
194
195 memStatus.dwLength = sizeof(memStatus);
196 gCtx.pfnGlobalMemoryStatusEx(&memStatus);
197
198 req.guestStats.u32PageSize = systemInfo.dwPageSize;
199 req.guestStats.u32PhysMemTotal = (uint32_t)(memStatus.ullTotalPhys / _4K);
200 req.guestStats.u32PhysMemAvail = (uint32_t)(memStatus.ullAvailPhys / _4K);
201 /* The current size of the committed memory limit, in bytes. This is physical
202 memory plus the size of the page file, minus a small overhead. */
203 req.guestStats.u32PageFileSize = (uint32_t)(memStatus.ullTotalPageFile / _4K) - req.guestStats.u32PhysMemTotal;
204 req.guestStats.u32MemoryLoad = memStatus.dwMemoryLoad;
205 req.guestStats.u32StatCaps = VBOX_GUEST_STAT_PHYS_MEM_TOTAL
206 | VBOX_GUEST_STAT_PHYS_MEM_AVAIL
207 | VBOX_GUEST_STAT_PAGE_FILE_SIZE
208 | VBOX_GUEST_STAT_MEMORY_LOAD;
209#ifdef VBOX_WITH_MEMBALLOON
210 req.guestStats.u32PhysMemBalloon = VBoxServiceBalloonQueryPages(_4K);
211 req.guestStats.u32StatCaps |= VBOX_GUEST_STAT_PHYS_MEM_BALLOON;
212#else
213 req.guestStats.u32PhysMemBalloon = 0;
214#endif
215
216 if (gCtx.pfnGetPerformanceInfo)
217 {
218 PERFORMANCE_INFORMATION perfInfo;
219
220 if (gCtx.pfnGetPerformanceInfo(&perfInfo, sizeof(perfInfo)))
221 {
222 req.guestStats.u32Processes = perfInfo.ProcessCount;
223 req.guestStats.u32Threads = perfInfo.ThreadCount;
224 req.guestStats.u32Handles = perfInfo.HandleCount;
225 req.guestStats.u32MemCommitTotal = perfInfo.CommitTotal; /* already in pages */
226 req.guestStats.u32MemKernelTotal = perfInfo.KernelTotal; /* already in pages */
227 req.guestStats.u32MemKernelPaged = perfInfo.KernelPaged; /* already in pages */
228 req.guestStats.u32MemKernelNonPaged = perfInfo.KernelNonpaged; /* already in pages */
229 req.guestStats.u32MemSystemCache = perfInfo.SystemCache; /* already in pages */
230 req.guestStats.u32StatCaps |= VBOX_GUEST_STAT_PROCESSES | VBOX_GUEST_STAT_THREADS | VBOX_GUEST_STAT_HANDLES
231 | VBOX_GUEST_STAT_MEM_COMMIT_TOTAL | VBOX_GUEST_STAT_MEM_KERNEL_TOTAL
232 | VBOX_GUEST_STAT_MEM_KERNEL_PAGED | VBOX_GUEST_STAT_MEM_KERNEL_NONPAGED
233 | VBOX_GUEST_STAT_MEM_SYSTEM_CACHE;
234 }
235 else
236 VBoxServiceVerbose(3, "VBoxServiceVMStatsReport: GetPerformanceInfo failed with %d\n", GetLastError());
237 }
238
239 /* Query CPU load information */
240 cbStruct = systemInfo.dwNumberOfProcessors * sizeof(SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION);
241 pProcInfo = (PSYSTEM_PROCESSOR_PERFORMANCE_INFORMATION)RTMemAlloc(cbStruct);
242 if (!pProcInfo)
243 return;
244
245 /* Unfortunately GetSystemTimes is XP SP1 and up only, so we need to use the semi-undocumented NtQuerySystemInformation */
246 NTSTATUS rc = gCtx.pfnNtQuerySystemInformation(SystemProcessorPerformanceInformation, pProcInfo, cbStruct, &cbReturned);
247 if ( !rc
248 && cbReturned == cbStruct)
249 {
250 if (gCtx.au64LastCpuLoad_Kernel == 0)
251 {
252 /* first time */
253 gCtx.au64LastCpuLoad_Idle[0] = pProcInfo->IdleTime.QuadPart;
254 gCtx.au64LastCpuLoad_Kernel[0] = pProcInfo->KernelTime.QuadPart;
255 gCtx.au64LastCpuLoad_User[0] = pProcInfo->UserTime.QuadPart;
256
257 Sleep(250);
258
259 rc = gCtx.pfnNtQuerySystemInformation(SystemProcessorPerformanceInformation, pProcInfo, cbStruct, &cbReturned);
260 Assert(!rc);
261 }
262
263 uint64_t deltaIdle = (pProcInfo->IdleTime.QuadPart - gCtx.au64LastCpuLoad_Idle[0]);
264 uint64_t deltaKernel = (pProcInfo->KernelTime.QuadPart - gCtx.au64LastCpuLoad_Kernel[0]);
265 uint64_t deltaUser = (pProcInfo->UserTime.QuadPart - gCtx.au64LastCpuLoad_User[0]);
266 deltaKernel -= deltaIdle; /* idle time is added to kernel time */
267 uint64_t ullTotalTime = deltaIdle + deltaKernel + deltaUser;
268 if (ullTotalTime == 0) /* Prevent division through zero. */
269 ullTotalTime = 1;
270
271 req.guestStats.u32CpuLoad_Idle = (uint32_t)(deltaIdle * 100 / ullTotalTime);
272 req.guestStats.u32CpuLoad_Kernel = (uint32_t)(deltaKernel* 100 / ullTotalTime);
273 req.guestStats.u32CpuLoad_User = (uint32_t)(deltaUser * 100 / ullTotalTime);
274
275 req.guestStats.u32StatCaps |= VBOX_GUEST_STAT_CPU_LOAD_IDLE | VBOX_GUEST_STAT_CPU_LOAD_KERNEL | VBOX_GUEST_STAT_CPU_LOAD_USER;
276
277 gCtx.au64LastCpuLoad_Idle[0] = pProcInfo->IdleTime.QuadPart;
278 gCtx.au64LastCpuLoad_Kernel[0] = pProcInfo->KernelTime.QuadPart;
279 gCtx.au64LastCpuLoad_User[0] = pProcInfo->UserTime.QuadPart;
280 /** @todo SMP: report details for each CPU? */
281 }
282
283 for (uint32_t i = 0; i < systemInfo.dwNumberOfProcessors; i++)
284 {
285 req.guestStats.u32CpuId = i;
286
287 rc = VbglR3StatReport(&req);
288 if (RT_SUCCESS(rc))
289 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: new statistics (CPU %u) reported successfully!\n", i);
290 else
291 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: DeviceIoControl (stats report) failed with %d\n", GetLastError());
292 }
293
294 RTMemFree(pProcInfo);
295
296#elif defined(RT_OS_LINUX)
297 VMMDevReportGuestStats req;
298 RT_ZERO(req);
299 PRTSTREAM pStrm;
300 char szLine[256];
301 char *psz;
302
303 int rc = RTStrmOpen("/proc/meminfo", "r", &pStrm);
304 if (RT_SUCCESS(rc))
305 {
306 uint64_t u64Kb;
307 uint64_t u64Total = 0, u64Free = 0, u64Buffers = 0, u64Cached = 0, u64PagedTotal = 0;
308 for (;;)
309 {
310 rc = RTStrmGetLine(pStrm, szLine, sizeof(szLine));
311 if (RT_FAILURE(rc))
312 break;
313 if (strstr(szLine, "MemTotal:") == szLine)
314 {
315 rc = RTStrToUInt64Ex(RTStrStripL(&szLine[9]), &psz, 0, &u64Kb);
316 if (RT_SUCCESS(rc))
317 u64Total = u64Kb * _1K;
318 }
319 else if (strstr(szLine, "MemFree:") == szLine)
320 {
321 rc = RTStrToUInt64Ex(RTStrStripL(&szLine[8]), &psz, 0, &u64Kb);
322 if (RT_SUCCESS(rc))
323 u64Free = u64Kb * _1K;
324 }
325 else if (strstr(szLine, "Buffers:") == szLine)
326 {
327 rc = RTStrToUInt64Ex(RTStrStripL(&szLine[8]), &psz, 0, &u64Kb);
328 if (RT_SUCCESS(rc))
329 u64Buffers = u64Kb * _1K;
330 }
331 else if (strstr(szLine, "Cached:") == szLine)
332 {
333 rc = RTStrToUInt64Ex(RTStrStripL(&szLine[7]), &psz, 0, &u64Kb);
334 if (RT_SUCCESS(rc))
335 u64Cached = u64Kb * _1K;
336 }
337 else if (strstr(szLine, "SwapTotal:") == szLine)
338 {
339 rc = RTStrToUInt64Ex(RTStrStripL(&szLine[10]), &psz, 0, &u64Kb);
340 if (RT_SUCCESS(rc))
341 u64PagedTotal = u64Kb * _1K;
342 }
343 }
344 req.guestStats.u32PhysMemTotal = u64Total / _4K;
345 req.guestStats.u32PhysMemAvail = (u64Free + u64Buffers + u64Cached) / _4K;
346 req.guestStats.u32MemSystemCache = (u64Buffers + u64Cached) / _4K;
347 req.guestStats.u32PageFileSize = u64PagedTotal / _4K;
348 RTStrmClose(pStrm);
349 }
350 else
351 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: memory info not available!\n");
352
353 req.guestStats.u32PageSize = getpagesize();
354 req.guestStats.u32StatCaps = VBOX_GUEST_STAT_PHYS_MEM_TOTAL
355 | VBOX_GUEST_STAT_PHYS_MEM_AVAIL
356 | VBOX_GUEST_STAT_PAGE_FILE_SIZE;
357#ifdef VBOX_WITH_MEMBALLOON
358 req.guestStats.u32PhysMemBalloon = VBoxServiceBalloonQueryPages(_4K);
359 req.guestStats.u32StatCaps |= VBOX_GUEST_STAT_PHYS_MEM_BALLOON;
360#else
361 req.guestStats.u32PhysMemBalloon = 0;
362#endif
363
364
365 /** @todo req.guestStats.u32Threads */
366 /** @todo req.guestStats.u32Processes */
367 /* req.guestStats.u32Handles doesn't make sense here. */
368 /** @todo req.guestStats.u32MemoryLoad */
369 /** @todo req.guestStats.u32MemCommitTotal */
370 /** @todo req.guestStats.u32MemKernelTotal */
371 /** @todo req.guestStats.u32MemKernelPaged, make any sense? = u32MemKernelTotal? */
372 /** @todo req.guestStats.u32MemKernelNonPaged, make any sense? = 0? */
373
374 bool fCpuInfoAvail = false;
375 rc = RTStrmOpen("/proc/stat", "r", &pStrm);
376 if (RT_SUCCESS(rc))
377 {
378 for (;;)
379 {
380 rc = RTStrmGetLine(pStrm, szLine, sizeof(szLine));
381 if (RT_FAILURE(rc))
382 break;
383 if ( strstr(szLine, "cpu") == szLine
384 && strlen(szLine) > 3
385 && RT_C_IS_DIGIT(szLine[3]))
386 {
387 uint32_t u32CpuId;
388 rc = RTStrToUInt32Ex(&szLine[3], &psz, 0, &u32CpuId);
389 if (u32CpuId < VMM_MAX_CPU_COUNT)
390 {
391 uint64_t u64User = 0;
392 if (RT_SUCCESS(rc))
393 rc = RTStrToUInt64Ex(RTStrStripL(psz), &psz, 0, &u64User);
394
395 uint64_t u64Nice = 0;
396 if (RT_SUCCESS(rc))
397 rc = RTStrToUInt64Ex(RTStrStripL(psz), &psz, 0, &u64Nice);
398
399 uint64_t u64System = 0;
400 if (RT_SUCCESS(rc))
401 rc = RTStrToUInt64Ex(RTStrStripL(psz), &psz, 0, &u64System);
402
403 uint64_t u64Idle = 0;
404 if (RT_SUCCESS(rc))
405 rc = RTStrToUInt64Ex(RTStrStripL(psz), &psz, 0, &u64Idle);
406
407 uint64_t u64DeltaIdle = u64Idle - gCtx.au64LastCpuLoad_Idle[u32CpuId];
408 uint64_t u64DeltaSystem = u64System - gCtx.au64LastCpuLoad_Kernel[u32CpuId];
409 uint64_t u64DeltaUser = u64User - gCtx.au64LastCpuLoad_User[u32CpuId];
410 uint64_t u64DeltaNice = u64Nice - gCtx.au64LastCpuLoad_Nice[u32CpuId];
411
412 uint64_t u64DeltaAll = u64DeltaIdle
413 + u64DeltaSystem
414 + u64DeltaUser
415 + u64DeltaNice;
416 if (u64DeltaAll == 0) /* Prevent division through zero. */
417 u64DeltaAll = 1;
418
419 gCtx.au64LastCpuLoad_Idle[u32CpuId] = u64Idle;
420 gCtx.au64LastCpuLoad_Kernel[u32CpuId] = u64System;
421 gCtx.au64LastCpuLoad_User[u32CpuId] = u64User;
422 gCtx.au64LastCpuLoad_Nice[u32CpuId] = u64Nice;
423
424 req.guestStats.u32CpuId = u32CpuId;
425 req.guestStats.u32CpuLoad_Idle = (uint32_t)(u64DeltaIdle * 100 / u64DeltaAll);
426 req.guestStats.u32CpuLoad_Kernel = (uint32_t)(u64DeltaSystem * 100 / u64DeltaAll);
427 req.guestStats.u32CpuLoad_User = (uint32_t)((u64DeltaUser
428 + u64DeltaNice) * 100 / u64DeltaAll);
429 req.guestStats.u32StatCaps |= VBOX_GUEST_STAT_CPU_LOAD_IDLE
430 | VBOX_GUEST_STAT_CPU_LOAD_KERNEL
431 | VBOX_GUEST_STAT_CPU_LOAD_USER;
432 fCpuInfoAvail = true;
433 rc = VbglR3StatReport(&req);
434 if (RT_SUCCESS(rc))
435 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: new statistics (CPU %u) reported successfully!\n", u32CpuId);
436 else
437 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: stats report failed with rc=%Rrc\n", rc);
438 }
439 else
440 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: skipping information for CPU%u\n", u32CpuId);
441 }
442 }
443 RTStrmClose(pStrm);
444 }
445 if (!fCpuInfoAvail)
446 {
447 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: CPU info not available!\n");
448 rc = VbglR3StatReport(&req);
449 if (RT_SUCCESS(rc))
450 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: new statistics reported successfully!\n");
451 else
452 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: stats report failed with rc=%Rrc\n", rc);
453 }
454
455#elif defined(RT_OS_SOLARIS)
456 VMMDevReportGuestStats req;
457 RT_ZERO(req);
458 kstat_ctl_t *pStatKern = kstat_open();
459 if (pStatKern)
460 {
461 /*
462 * Memory statistics.
463 */
464 uint64_t u64Total = 0, u64Free = 0, u64Buffers = 0, u64Cached = 0, u64PagedTotal = 0;
465 int rc = -1;
466 kstat_t *pStatPages = kstat_lookup(pStatKern, "unix", 0 /* instance */, "system_pages");
467 if (pStatPages)
468 {
469 rc = kstat_read(pStatKern, pStatPages, NULL /* optional-copy-buf */);
470 if (rc != -1)
471 {
472 kstat_named_t *pStat = NULL;
473 pStat = (kstat_named_t *)kstat_data_lookup(pStatPages, "pagestotal");
474 if (pStat)
475 u64Total = pStat->value.ul;
476
477 pStat = (kstat_named_t *)kstat_data_lookup(pStatPages, "freemem");
478 if (pStat)
479 u64Free = pStat->value.ul;
480 }
481 }
482
483 kstat_t *pStatZFS = kstat_lookup(pStatKern, "zfs", 0 /* instance */, "arcstats");
484 if (pStatZFS)
485 {
486 rc = kstat_read(pStatKern, pStatZFS, NULL /* optional-copy-buf */);
487 if (rc != -1)
488 {
489 kstat_named_t *pStat = (kstat_named_t *)kstat_data_lookup(pStatZFS, "size");
490 if (pStat)
491 u64Cached = pStat->value.ul;
492 }
493 }
494
495 /*
496 * The vminfo are accumulative counters updated every "N" ticks. Let's get the
497 * number of stat updates so far and use that to divide the swap counter.
498 */
499 kstat_t *pStatInfo = kstat_lookup(pStatKern, "unix", 0 /* instance */, "sysinfo");
500 if (pStatInfo)
501 {
502 sysinfo_t SysInfo;
503 rc = kstat_read(pStatKern, pStatInfo, &SysInfo);
504 if (rc != -1)
505 {
506 kstat_t *pStatVMInfo = kstat_lookup(pStatKern, "unix", 0 /* instance */, "vminfo");
507 if (pStatVMInfo)
508 {
509 vminfo_t VMInfo;
510 rc = kstat_read(pStatKern, pStatVMInfo, &VMInfo);
511 if (rc != -1)
512 {
513 Assert(SysInfo.updates != 0);
514 u64PagedTotal = VMInfo.swap_avail / SysInfo.updates;
515 }
516 }
517 }
518 }
519
520 req.guestStats.u32PhysMemTotal = u64Total; /* already in pages */
521 req.guestStats.u32PhysMemAvail = u64Free; /* already in pages */
522 req.guestStats.u32MemSystemCache = u64Cached / _4K;
523 req.guestStats.u32PageFileSize = u64PagedTotal; /* already in pages */
524 /** @todo req.guestStats.u32Threads */
525 /** @todo req.guestStats.u32Processes */
526 /** @todo req.guestStats.u32Handles -- ??? */
527 /** @todo req.guestStats.u32MemoryLoad */
528 /** @todo req.guestStats.u32MemCommitTotal */
529 /** @todo req.guestStats.u32MemKernelTotal */
530 /** @todo req.guestStats.u32MemKernelPaged */
531 /** @todo req.guestStats.u32MemKernelNonPaged */
532 req.guestStats.u32PageSize = getpagesize();
533
534 req.guestStats.u32StatCaps = VBOX_GUEST_STAT_PHYS_MEM_TOTAL
535 | VBOX_GUEST_STAT_PHYS_MEM_AVAIL
536 | VBOX_GUEST_STAT_PAGE_FILE_SIZE;
537#ifdef VBOX_WITH_MEMBALLOON
538 req.guestStats.u32PhysMemBalloon = VBoxServiceBalloonQueryPages(_4K);
539 req.guestStats.u32StatCaps |= VBOX_GUEST_STAT_PHYS_MEM_BALLOON;
540#else
541 req.guestStats.u32PhysMemBalloon = 0;
542#endif
543
544 /*
545 * CPU statistics.
546 */
547 cpu_stat_t StatCPU;
548 RT_ZERO(StatCPU);
549 kstat_t *pStatNode = NULL;
550 uint32_t cCPUs = 0;
551 bool fCpuInfoAvail = false;
552 for (pStatNode = pStatKern->kc_chain; pStatNode != NULL; pStatNode = pStatNode->ks_next)
553 {
554 if (!strcmp(pStatNode->ks_module, "cpu_stat"))
555 {
556 rc = kstat_read(pStatKern, pStatNode, &StatCPU);
557 if (rc == -1)
558 break;
559
560 uint64_t u64Idle = StatCPU.cpu_sysinfo.cpu[CPU_IDLE];
561 uint64_t u64User = StatCPU.cpu_sysinfo.cpu[CPU_USER];
562 uint64_t u64System = StatCPU.cpu_sysinfo.cpu[CPU_KERNEL];
563
564 uint64_t u64DeltaIdle = u64Idle - gCtx.au64LastCpuLoad_Idle[cCPUs];
565 uint64_t u64DeltaSystem = u64System - gCtx.au64LastCpuLoad_Kernel[cCPUs];
566 uint64_t u64DeltaUser = u64User - gCtx.au64LastCpuLoad_User[cCPUs];
567
568 uint64_t u64DeltaAll = u64DeltaIdle + u64DeltaSystem + u64DeltaUser;
569 if (u64DeltaAll == 0) /* Prevent division through zero. */
570 u64DeltaAll = 1;
571
572 gCtx.au64LastCpuLoad_Idle[cCPUs] = u64Idle;
573 gCtx.au64LastCpuLoad_Kernel[cCPUs] = u64System;
574 gCtx.au64LastCpuLoad_User[cCPUs] = u64User;
575
576 req.guestStats.u32CpuId = cCPUs;
577 req.guestStats.u32CpuLoad_Idle = (uint32_t)(u64DeltaIdle * 100 / u64DeltaAll);
578 req.guestStats.u32CpuLoad_Kernel = (uint32_t)(u64DeltaSystem * 100 / u64DeltaAll);
579 req.guestStats.u32CpuLoad_User = (uint32_t)(u64DeltaUser * 100 / u64DeltaAll);
580
581 req.guestStats.u32StatCaps |= VBOX_GUEST_STAT_CPU_LOAD_IDLE
582 | VBOX_GUEST_STAT_CPU_LOAD_KERNEL
583 | VBOX_GUEST_STAT_CPU_LOAD_USER;
584 fCpuInfoAvail = true;
585
586 rc = VbglR3StatReport(&req);
587 if (RT_SUCCESS(rc))
588 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: new statistics (CPU %u) reported successfully!\n", cCPUs);
589 else
590 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: stats report failed with rc=%Rrc\n", rc);
591 cCPUs++;
592 }
593 }
594
595 /*
596 * Report whatever statistics were collected.
597 */
598 if (!fCpuInfoAvail)
599 {
600 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: CPU info not available!\n");
601 rc = VbglR3StatReport(&req);
602 if (RT_SUCCESS(rc))
603 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: new statistics reported successfully!\n");
604 else
605 VBoxServiceVerbose(3, "VBoxStatsReportStatistics: stats report failed with rc=%Rrc\n", rc);
606 }
607
608 kstat_close(pStatKern);
609 }
610
611#else
612 /* todo: implement for other platforms. */
613
614#endif
615}
616
617/** @copydoc VBOXSERVICE::pfnWorker */
618DECLCALLBACK(int) VBoxServiceVMStatsWorker(bool volatile *pfShutdown)
619{
620 int rc = VINF_SUCCESS;
621
622 /* Start monitoring of the stat event change event. */
623 rc = VbglR3CtlFilterMask(VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST, 0);
624 if (RT_FAILURE(rc))
625 {
626 VBoxServiceVerbose(3, "VBoxServiceVMStatsWorker: VbglR3CtlFilterMask failed with %d\n", rc);
627 return rc;
628 }
629
630 /*
631 * Tell the control thread that it can continue
632 * spawning services.
633 */
634 RTThreadUserSignal(RTThreadSelf());
635
636 /*
637 * Now enter the loop retrieving runtime data continuously.
638 */
639 for (;;)
640 {
641 uint32_t fEvents = 0;
642 RTMSINTERVAL cWaitMillies;
643
644 /* Check if an update interval change is pending. */
645 rc = VbglR3WaitEvent(VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST, 0 /* no wait */, &fEvents);
646 if ( RT_SUCCESS(rc)
647 && (fEvents & VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST))
648 {
649 VbglR3StatQueryInterval(&gCtx.cMsStatInterval);
650 }
651
652 if (gCtx.cMsStatInterval)
653 {
654 VBoxServiceVMStatsReport();
655 cWaitMillies = gCtx.cMsStatInterval;
656 }
657 else
658 cWaitMillies = 3000;
659
660 /*
661 * Block for a while.
662 *
663 * The event semaphore takes care of ignoring interruptions and it
664 * allows us to implement service wakeup later.
665 */
666 if (*pfShutdown)
667 break;
668 int rc2 = RTSemEventMultiWait(g_VMStatEvent, cWaitMillies);
669 if (*pfShutdown)
670 break;
671 if (rc2 != VERR_TIMEOUT && RT_FAILURE(rc2))
672 {
673 VBoxServiceError("VBoxServiceVMStatsWorker: RTSemEventMultiWait failed; rc2=%Rrc\n", rc2);
674 rc = rc2;
675 break;
676 }
677 }
678
679 /* Cancel monitoring of the stat event change event. */
680 rc = VbglR3CtlFilterMask(0, VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST);
681 if (RT_FAILURE(rc))
682 VBoxServiceVerbose(3, "VBoxServiceVMStatsWorker: VbglR3CtlFilterMask failed with %d\n", rc);
683
684 RTSemEventMultiDestroy(g_VMStatEvent);
685 g_VMStatEvent = NIL_RTSEMEVENTMULTI;
686
687 VBoxServiceVerbose(3, "VBoxStatsThread: finished statistics change request thread\n");
688 return 0;
689}
690
691
692/** @copydoc VBOXSERVICE::pfnTerm */
693static DECLCALLBACK(void) VBoxServiceVMStatsTerm(void)
694{
695 VBoxServiceVerbose(3, "VBoxServiceVMStatsTerm\n");
696 return;
697}
698
699
700/** @copydoc VBOXSERVICE::pfnStop */
701static DECLCALLBACK(void) VBoxServiceVMStatsStop(void)
702{
703 RTSemEventMultiSignal(g_VMStatEvent);
704}
705
706
707/**
708 * The 'vminfo' service description.
709 */
710VBOXSERVICE g_VMStatistics =
711{
712 /* pszName. */
713 "vmstats",
714 /* pszDescription. */
715 "Virtual Machine Statistics",
716 /* pszUsage. */
717 NULL,
718 /* pszOptions. */
719 NULL,
720 /* methods */
721 VBoxServiceVMStatsPreInit,
722 VBoxServiceVMStatsOption,
723 VBoxServiceVMStatsInit,
724 VBoxServiceVMStatsWorker,
725 VBoxServiceVMStatsStop,
726 VBoxServiceVMStatsTerm
727};
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