VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/PerformanceImpl.cpp@ 67078

Last change on this file since 67078 was 65201, checked in by vboxsync, 8 years ago

Main/PerformanceImpl: Invalidate the magic after the timer was destroyed or we risk running into the assertion when the timer kicks in between destroying the magic and destroying the timer which is quite annoying

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 27.6 KB
Line 
1/* $Id: PerformanceImpl.cpp 65201 2017-01-09 12:39:39Z vboxsync $ */
2
3/** @file
4 *
5 * VBox Performance API COM Classes implementation
6 */
7
8/*
9 * Copyright (C) 2008-2016 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20/*
21 * Rules of engagement:
22 * 1) All performance objects must be destroyed by PerformanceCollector only!
23 * 2) All public methods of PerformanceCollector must be protected with
24 * read or write lock.
25 * 3) samplerCallback only uses the write lock during the third phase
26 * which pulls data into SubMetric objects. This is where object destruction
27 * and all list modifications are done. The pre-collection phases are
28 * run without any locks which is only possible because:
29 * 4) Public methods of PerformanceCollector as well as pre-collection methods
30 cannot modify lists or destroy objects, and:
31 * 5) Pre-collection methods cannot modify metric data.
32 */
33
34#include "PerformanceImpl.h"
35
36#include "AutoCaller.h"
37#include "Logging.h"
38
39#include <iprt/process.h>
40
41#include <VBox/err.h>
42#include <VBox/settings.h>
43
44#include <vector>
45#include <algorithm>
46#include <functional>
47
48#include "Performance.h"
49
50static const char *g_papcszMetricNames[] =
51{
52 "CPU/Load/User",
53 "CPU/Load/User:avg",
54 "CPU/Load/User:min",
55 "CPU/Load/User:max",
56 "CPU/Load/Kernel",
57 "CPU/Load/Kernel:avg",
58 "CPU/Load/Kernel:min",
59 "CPU/Load/Kernel:max",
60 "CPU/Load/Idle",
61 "CPU/Load/Idle:avg",
62 "CPU/Load/Idle:min",
63 "CPU/Load/Idle:max",
64 "CPU/MHz",
65 "CPU/MHz:avg",
66 "CPU/MHz:min",
67 "CPU/MHz:max",
68 "Net/*/Load/Rx",
69 "Net/*/Load/Rx:avg",
70 "Net/*/Load/Rx:min",
71 "Net/*/Load/Rx:max",
72 "Net/*/Load/Tx",
73 "Net/*/Load/Tx:avg",
74 "Net/*/Load/Tx:min",
75 "Net/*/Load/Tx:max",
76 "RAM/Usage/Total",
77 "RAM/Usage/Total:avg",
78 "RAM/Usage/Total:min",
79 "RAM/Usage/Total:max",
80 "RAM/Usage/Used",
81 "RAM/Usage/Used:avg",
82 "RAM/Usage/Used:min",
83 "RAM/Usage/Used:max",
84 "RAM/Usage/Free",
85 "RAM/Usage/Free:avg",
86 "RAM/Usage/Free:min",
87 "RAM/Usage/Free:max",
88 "RAM/VMM/Used",
89 "RAM/VMM/Used:avg",
90 "RAM/VMM/Used:min",
91 "RAM/VMM/Used:max",
92 "RAM/VMM/Free",
93 "RAM/VMM/Free:avg",
94 "RAM/VMM/Free:min",
95 "RAM/VMM/Free:max",
96 "RAM/VMM/Ballooned",
97 "RAM/VMM/Ballooned:avg",
98 "RAM/VMM/Ballooned:min",
99 "RAM/VMM/Ballooned:max",
100 "RAM/VMM/Shared",
101 "RAM/VMM/Shared:avg",
102 "RAM/VMM/Shared:min",
103 "RAM/VMM/Shared:max",
104 "Guest/CPU/Load/User",
105 "Guest/CPU/Load/User:avg",
106 "Guest/CPU/Load/User:min",
107 "Guest/CPU/Load/User:max",
108 "Guest/CPU/Load/Kernel",
109 "Guest/CPU/Load/Kernel:avg",
110 "Guest/CPU/Load/Kernel:min",
111 "Guest/CPU/Load/Kernel:max",
112 "Guest/CPU/Load/Idle",
113 "Guest/CPU/Load/Idle:avg",
114 "Guest/CPU/Load/Idle:min",
115 "Guest/CPU/Load/Idle:max",
116 "Guest/RAM/Usage/Total",
117 "Guest/RAM/Usage/Total:avg",
118 "Guest/RAM/Usage/Total:min",
119 "Guest/RAM/Usage/Total:max",
120 "Guest/RAM/Usage/Free",
121 "Guest/RAM/Usage/Free:avg",
122 "Guest/RAM/Usage/Free:min",
123 "Guest/RAM/Usage/Free:max",
124 "Guest/RAM/Usage/Balloon",
125 "Guest/RAM/Usage/Balloon:avg",
126 "Guest/RAM/Usage/Balloon:min",
127 "Guest/RAM/Usage/Balloon:max",
128 "Guest/RAM/Usage/Shared",
129 "Guest/RAM/Usage/Shared:avg",
130 "Guest/RAM/Usage/Shared:min",
131 "Guest/RAM/Usage/Shared:max",
132 "Guest/RAM/Usage/Cache",
133 "Guest/RAM/Usage/Cache:avg",
134 "Guest/RAM/Usage/Cache:min",
135 "Guest/RAM/Usage/Cache:max",
136 "Guest/Pagefile/Usage/Total",
137 "Guest/Pagefile/Usage/Total:avg",
138 "Guest/Pagefile/Usage/Total:min",
139 "Guest/Pagefile/Usage/Total:max",
140};
141
142////////////////////////////////////////////////////////////////////////////////
143// PerformanceCollector class
144////////////////////////////////////////////////////////////////////////////////
145
146// constructor / destructor
147////////////////////////////////////////////////////////////////////////////////
148
149PerformanceCollector::PerformanceCollector()
150 : mMagic(0), mUnknownGuest("unknown guest")
151{
152}
153
154PerformanceCollector::~PerformanceCollector() {}
155
156HRESULT PerformanceCollector::FinalConstruct()
157{
158 LogFlowThisFunc(("\n"));
159
160 return BaseFinalConstruct();
161}
162
163void PerformanceCollector::FinalRelease()
164{
165 LogFlowThisFunc(("\n"));
166 BaseFinalRelease();
167}
168
169// public initializer/uninitializer for internal purposes only
170////////////////////////////////////////////////////////////////////////////////
171
172/**
173 * Initializes the PerformanceCollector object.
174 */
175HRESULT PerformanceCollector::init()
176{
177 /* Enclose the state transition NotReady->InInit->Ready */
178 AutoInitSpan autoInitSpan(this);
179 AssertReturn(autoInitSpan.isOk(), E_FAIL);
180
181 LogFlowThisFuncEnter();
182
183 HRESULT rc = S_OK;
184
185 m.hal = pm::createHAL();
186 m.gm = new pm::CollectorGuestManager;
187
188 /* Let the sampler know it gets a valid collector. */
189 mMagic = PERFORMANCE_METRIC_MAGIC;
190
191 /* Start resource usage sampler */
192 int vrc = RTTimerLRCreate(&m.sampler, VBOX_USAGE_SAMPLER_MIN_INTERVAL,
193 &PerformanceCollector::staticSamplerCallback, this);
194 AssertMsgRC(vrc, ("Failed to create resource usage sampling timer(%Rra)\n", vrc));
195 if (RT_FAILURE(vrc))
196 rc = E_FAIL;
197
198 if (SUCCEEDED(rc))
199 autoInitSpan.setSucceeded();
200
201 LogFlowThisFuncLeave();
202
203 return rc;
204}
205
206/**
207 * Uninitializes the PerformanceCollector object.
208 *
209 * Called either from FinalRelease() or by the parent when it gets destroyed.
210 */
211void PerformanceCollector::uninit()
212{
213 LogFlowThisFuncEnter();
214
215 /* Enclose the state transition Ready->InUninit->NotReady */
216 AutoUninitSpan autoUninitSpan(this);
217 if (autoUninitSpan.uninitDone())
218 {
219 LogFlowThisFunc(("Already uninitialized.\n"));
220 LogFlowThisFuncLeave();
221 return;
222 }
223
224 /* Destroy unregistered metrics */
225 BaseMetricList::iterator it;
226 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end();)
227 if ((*it)->isUnregistered())
228 {
229 delete *it;
230 it = m.baseMetrics.erase(it);
231 }
232 else
233 ++it;
234 Assert(m.baseMetrics.size() == 0);
235 /*
236 * Now when we have destroyed all base metrics that could
237 * try to pull data from unregistered CollectorGuest objects
238 * it is safe to destroy them as well.
239 */
240 m.gm->destroyUnregistered();
241
242 /* Destroy resource usage sampler */
243 int vrc = RTTimerLRDestroy(m.sampler);
244 AssertMsgRC(vrc, ("Failed to destroy resource usage sampling timer (%Rra)\n", vrc));
245 m.sampler = NULL;
246
247 /* Invalidate the magic now. */
248 mMagic = 0;
249
250 //delete m.factory;
251 //m.factory = NULL;
252
253 delete m.gm;
254 m.gm = NULL;
255 delete m.hal;
256 m.hal = NULL;
257
258 LogFlowThisFuncLeave();
259}
260
261// IPerformanceCollector properties
262////////////////////////////////////////////////////////////////////////////////
263
264HRESULT PerformanceCollector::getMetricNames(std::vector<com::Utf8Str> &aMetricNames)
265{
266 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
267
268 aMetricNames.resize(RT_ELEMENTS(g_papcszMetricNames));
269 for (size_t i = 0; i < RT_ELEMENTS(g_papcszMetricNames); i++)
270 aMetricNames[i] = g_papcszMetricNames[i];
271
272 return S_OK;
273}
274
275// IPerformanceCollector methods
276////////////////////////////////////////////////////////////////////////////////
277
278HRESULT PerformanceCollector::toIPerformanceMetric(pm::Metric *src, ComPtr<IPerformanceMetric> &dst)
279{
280 ComObjPtr<PerformanceMetric> metric;
281 HRESULT rc = metric.createObject();
282 if (SUCCEEDED(rc))
283 rc = metric->init(src);
284 AssertComRCReturnRC(rc);
285 dst = metric;
286 return rc;
287}
288
289HRESULT PerformanceCollector::toIPerformanceMetric(pm::BaseMetric *src, ComPtr<IPerformanceMetric> &dst)
290{
291 ComObjPtr<PerformanceMetric> metric;
292 HRESULT rc = metric.createObject();
293 if (SUCCEEDED(rc))
294 rc = metric->init(src);
295 AssertComRCReturnRC(rc);
296 dst = metric;
297 return rc;
298}
299
300const Utf8Str& PerformanceCollector::getFailedGuestName()
301{
302 pm::CollectorGuest *pGuest = m.gm->getBlockedGuest();
303 if (pGuest)
304 return pGuest->getVMName();
305 return mUnknownGuest;
306}
307
308HRESULT PerformanceCollector::getMetrics(const std::vector<com::Utf8Str> &aMetricNames,
309 const std::vector<ComPtr<IUnknown> > &aObjects,
310 std::vector<ComPtr<IPerformanceMetric> > &aMetrics)
311{
312 HRESULT rc = S_OK;
313
314 pm::Filter filter(aMetricNames, aObjects);
315
316 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
317
318 MetricList filteredMetrics;
319 MetricList::iterator it;
320 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
321 if (filter.match((*it)->getObject(), (*it)->getName()))
322 filteredMetrics.push_back(*it);
323
324 aMetrics.resize(filteredMetrics.size());
325 int i = 0;
326 for (it = filteredMetrics.begin(); it != filteredMetrics.end(); ++it)
327 {
328 ComObjPtr<PerformanceMetric> metric;
329 rc = metric.createObject();
330 if (SUCCEEDED(rc))
331 rc = metric->init(*it);
332 AssertComRCReturnRC(rc);
333 LogFlow(("PerformanceCollector::GetMetrics() store a metric at retMetrics[%d]...\n", i));
334 aMetrics[i++] = metric;
335 }
336 return rc;
337}
338
339HRESULT PerformanceCollector::setupMetrics(const std::vector<com::Utf8Str> &aMetricNames,
340 const std::vector<ComPtr<IUnknown> > &aObjects,
341 ULONG aPeriod,
342 ULONG aCount,
343 std::vector<ComPtr<IPerformanceMetric> > &aAffectedMetrics)
344{
345 pm::Filter filter(aMetricNames, aObjects);
346
347 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
348
349 HRESULT rc = S_OK;
350 BaseMetricList filteredMetrics;
351 BaseMetricList::iterator it;
352 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
353 if (filter.match((*it)->getObject(), (*it)->getName()))
354 {
355 LogFlow(("PerformanceCollector::SetupMetrics() setting period to %u, count to %u for %s\n",
356 aPeriod, aCount, (*it)->getName()));
357 (*it)->init(aPeriod, aCount);
358 if (aPeriod == 0 || aCount == 0)
359 {
360 LogFlow(("PerformanceCollector::SetupMetrics() disabling %s\n",
361 (*it)->getName()));
362 rc = (*it)->disable();
363 if (FAILED(rc))
364 break;
365 }
366 else
367 {
368 LogFlow(("PerformanceCollector::SetupMetrics() enabling %s\n",
369 (*it)->getName()));
370 rc = (*it)->enable();
371 if (FAILED(rc))
372 break;
373 }
374 filteredMetrics.push_back(*it);
375 }
376
377 aAffectedMetrics.resize(filteredMetrics.size());
378 int i = 0;
379 for (it = filteredMetrics.begin();
380 it != filteredMetrics.end() && SUCCEEDED(rc); ++it)
381 rc = toIPerformanceMetric(*it, aAffectedMetrics[i++]);
382
383 if (FAILED(rc))
384 return setError(E_FAIL, "Failed to setup metrics for '%s'",
385 getFailedGuestName().c_str());
386 return rc;
387}
388
389HRESULT PerformanceCollector::enableMetrics(const std::vector<com::Utf8Str> &aMetricNames,
390 const std::vector<ComPtr<IUnknown> > &aObjects,
391 std::vector<ComPtr<IPerformanceMetric> > &aAffectedMetrics)
392{
393 pm::Filter filter(aMetricNames, aObjects);
394
395 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); /* Write lock is not needed atm since we are */
396 /* fiddling with enable bit only, but we */
397 /* care for those who come next :-). */
398
399 HRESULT rc = S_OK;
400 BaseMetricList filteredMetrics;
401 BaseMetricList::iterator it;
402 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
403 if (filter.match((*it)->getObject(), (*it)->getName()))
404 {
405 rc = (*it)->enable();
406 if (FAILED(rc))
407 break;
408 filteredMetrics.push_back(*it);
409 }
410
411 aAffectedMetrics.resize(filteredMetrics.size());
412 int i = 0;
413 for (it = filteredMetrics.begin();
414 it != filteredMetrics.end() && SUCCEEDED(rc); ++it)
415 rc = toIPerformanceMetric(*it, aAffectedMetrics[i++]);
416
417 LogFlowThisFuncLeave();
418
419 if (FAILED(rc))
420 return setError(E_FAIL, "Failed to enable metrics for '%s'",
421 getFailedGuestName().c_str());
422 return rc;
423}
424
425HRESULT PerformanceCollector::disableMetrics(const std::vector<com::Utf8Str> &aMetricNames,
426 const std::vector<ComPtr<IUnknown> > &aObjects,
427 std::vector<ComPtr<IPerformanceMetric> > &aAffectedMetrics)
428{
429 pm::Filter filter(aMetricNames, aObjects);
430
431 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); /* Write lock is not needed atm since we are */
432 /* fiddling with enable bit only, but we */
433 /* care for those who come next :-). */
434
435 HRESULT rc = S_OK;
436 BaseMetricList filteredMetrics;
437 BaseMetricList::iterator it;
438 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
439 if (filter.match((*it)->getObject(), (*it)->getName()))
440 {
441 rc = (*it)->disable();
442 if (FAILED(rc))
443 break;
444 filteredMetrics.push_back(*it);
445 }
446
447 aAffectedMetrics.resize(filteredMetrics.size());
448 int i = 0;
449 for (it = filteredMetrics.begin();
450 it != filteredMetrics.end() && SUCCEEDED(rc); ++it)
451 rc = toIPerformanceMetric(*it, aAffectedMetrics[i++]);
452
453 LogFlowThisFuncLeave();
454
455 if (FAILED(rc))
456 return setError(E_FAIL, "Failed to disable metrics for '%s'",
457 getFailedGuestName().c_str());
458 return rc;
459}
460
461HRESULT PerformanceCollector::queryMetricsData(const std::vector<com::Utf8Str> &aMetricNames,
462 const std::vector<ComPtr<IUnknown> > &aObjects,
463 std::vector<com::Utf8Str> &aReturnMetricNames,
464 std::vector<ComPtr<IUnknown> > &aReturnObjects,
465 std::vector<com::Utf8Str> &aReturnUnits,
466 std::vector<ULONG> &aReturnScales,
467 std::vector<ULONG> &aReturnSequenceNumbers,
468 std::vector<ULONG> &aReturnDataIndices,
469 std::vector<ULONG> &aReturnDataLengths,
470 std::vector<LONG> &aReturnData)
471{
472 pm::Filter filter(aMetricNames, aObjects);
473
474 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
475
476 /* Let's compute the size of the resulting flat array */
477 size_t flatSize = 0;
478 MetricList filteredMetrics;
479 MetricList::iterator it;
480 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
481 if (filter.match((*it)->getObject(), (*it)->getName()))
482 {
483 filteredMetrics.push_back(*it);
484 flatSize += (*it)->getLength();
485 }
486
487 int i = 0;
488 size_t flatIndex = 0;
489 size_t numberOfMetrics = filteredMetrics.size();
490 aReturnMetricNames.resize(numberOfMetrics);
491 aReturnObjects.resize(numberOfMetrics);
492 aReturnUnits.resize(numberOfMetrics);
493 aReturnScales.resize(numberOfMetrics);
494 aReturnSequenceNumbers.resize(numberOfMetrics);
495 aReturnDataIndices.resize(numberOfMetrics);
496 aReturnDataLengths.resize(numberOfMetrics);
497 aReturnData.resize(flatSize);
498
499 for (it = filteredMetrics.begin(); it != filteredMetrics.end(); ++it, ++i)
500 {
501 ULONG *values, length, sequenceNumber;
502 /** @todo We may want to revise the query method to get rid of excessive alloc/memcpy calls. */
503 (*it)->query(&values, &length, &sequenceNumber);
504 LogFlow(("PerformanceCollector::QueryMetricsData() querying metric %s returned %d values.\n",
505 (*it)->getName(), length));
506 memcpy(&aReturnData[flatIndex], values, length * sizeof(*values));
507 RTMemFree(values);
508 aReturnMetricNames[i] = (*it)->getName();
509 aReturnObjects[i] = (*it)->getObject();
510 aReturnUnits[i] = (*it)->getUnit();
511 aReturnScales[i] = (*it)->getScale();
512 aReturnSequenceNumbers[i] = sequenceNumber;
513 aReturnDataIndices[i] = (ULONG)flatIndex;
514 aReturnDataLengths[i] = length;
515 flatIndex += length;
516 }
517
518 return S_OK;
519}
520
521// public methods for internal purposes
522///////////////////////////////////////////////////////////////////////////////
523
524void PerformanceCollector::registerBaseMetric(pm::BaseMetric *baseMetric)
525{
526 //LogFlowThisFuncEnter();
527 AutoCaller autoCaller(this);
528 if (!SUCCEEDED(autoCaller.rc())) return;
529
530 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
531 Log7(("{%p} " LOG_FN_FMT ": obj=%p name=%s\n", this, __PRETTY_FUNCTION__,
532 (void *)baseMetric->getObject(), baseMetric->getName()));
533 m.baseMetrics.push_back(baseMetric);
534 //LogFlowThisFuncLeave();
535}
536
537void PerformanceCollector::registerMetric(pm::Metric *metric)
538{
539 //LogFlowThisFuncEnter();
540 AutoCaller autoCaller(this);
541 if (!SUCCEEDED(autoCaller.rc())) return;
542
543 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
544 Log7(("{%p} " LOG_FN_FMT ": obj=%p name=%s\n", this, __PRETTY_FUNCTION__, (void *)metric->getObject(), metric->getName()));
545 m.metrics.push_back(metric);
546 //LogFlowThisFuncLeave();
547}
548
549void PerformanceCollector::unregisterBaseMetricsFor(const ComPtr<IUnknown> &aObject, const Utf8Str name)
550{
551 //LogFlowThisFuncEnter();
552 AutoCaller autoCaller(this);
553 if (!SUCCEEDED(autoCaller.rc())) return;
554
555 pm::Filter filter(name, aObject);
556
557 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
558 int n = 0;
559 BaseMetricList::iterator it;
560 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
561 if (filter.match((*it)->getObject(), (*it)->getName()))
562 {
563 (*it)->unregister();
564 ++n;
565 }
566 Log7(("{%p} " LOG_FN_FMT ": obj=%p, name=%s, marked %d metrics\n",
567 this, __PRETTY_FUNCTION__, (void *)aObject, name.c_str(), n));
568 //LogFlowThisFuncLeave();
569}
570
571void PerformanceCollector::unregisterMetricsFor(const ComPtr<IUnknown> &aObject, const Utf8Str name)
572{
573 //LogFlowThisFuncEnter();
574 AutoCaller autoCaller(this);
575 if (!SUCCEEDED(autoCaller.rc())) return;
576
577 pm::Filter filter(name, aObject);
578
579 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
580 Log7(("{%p} " LOG_FN_FMT ": obj=%p, name=%s\n", this, __PRETTY_FUNCTION__, (void *)aObject, name.c_str()));
581 MetricList::iterator it;
582 for (it = m.metrics.begin(); it != m.metrics.end();)
583 if (filter.match((*it)->getObject(), (*it)->getName()))
584 {
585 delete *it;
586 it = m.metrics.erase(it);
587 }
588 else
589 ++it;
590 //LogFlowThisFuncLeave();
591}
592
593void PerformanceCollector::registerGuest(pm::CollectorGuest* pGuest)
594{
595 AutoCaller autoCaller(this);
596 if (!SUCCEEDED(autoCaller.rc())) return;
597
598 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
599 m.gm->registerGuest(pGuest);
600}
601
602void PerformanceCollector::unregisterGuest(pm::CollectorGuest* pGuest)
603{
604 AutoCaller autoCaller(this);
605 if (!SUCCEEDED(autoCaller.rc())) return;
606
607 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
608 m.gm->unregisterGuest(pGuest);
609}
610
611void PerformanceCollector::suspendSampling()
612{
613 AutoCaller autoCaller(this);
614 if (!SUCCEEDED(autoCaller.rc())) return;
615
616 int rc = RTTimerLRStop(m.sampler);
617 if ( RT_FAILURE(rc)
618 && rc != VERR_TIMER_SUSPENDED) /* calling suspendSampling() successively shouldn't assert. See @bugref{3495}. */
619 AssertMsgFailed(("PerformanceCollector::suspendSampling(): RTTimerLRStop returned %Rrc\n", rc));
620}
621
622void PerformanceCollector::resumeSampling()
623{
624 AutoCaller autoCaller(this);
625 if (!SUCCEEDED(autoCaller.rc())) return;
626
627 int rc = RTTimerLRStart(m.sampler, 0);
628 if ( RT_FAILURE(rc)
629 && rc != VERR_TIMER_ACTIVE) /* calling resumeSampling() successively shouldn't assert. See @bugref{3495}. */
630 AssertMsgFailed(("PerformanceCollector::resumeSampling(): RTTimerLRStart returned %Rrc\n", rc));
631}
632
633
634// private methods
635///////////////////////////////////////////////////////////////////////////////
636
637/* static */
638DECLCALLBACK(void) PerformanceCollector::staticSamplerCallback(RTTIMERLR hTimerLR, void *pvUser,
639 uint64_t iTick)
640{
641 AssertReturnVoid(pvUser != NULL);
642 PerformanceCollector *collector = static_cast <PerformanceCollector *> (pvUser);
643 Assert(collector->mMagic == PERFORMANCE_METRIC_MAGIC);
644 if (collector->mMagic == PERFORMANCE_METRIC_MAGIC)
645 collector->samplerCallback(iTick);
646
647 NOREF(hTimerLR);
648}
649
650/*
651 * Metrics collection is a three stage process:
652 * 1) Pre-collection (hinting)
653 * At this stage we compose the list of all metrics to be collected
654 * If any metrics cannot be collected separately or if it is more
655 * efficient to collect several metric at once, these metrics should
656 * use hints to mark that they will need to be collected.
657 * 2) Pre-collection (bulk)
658 * Using hints set at stage 1 platform-specific HAL
659 * instance collects all marked host-related metrics.
660 * Hinted guest-related metrics then get collected by CollectorGuestManager.
661 * 3) Collection
662 * Metrics that are collected individually get collected and stored. Values
663 * saved in HAL and CollectorGuestManager are extracted and stored to
664 * individual metrics.
665 */
666void PerformanceCollector::samplerCallback(uint64_t iTick)
667{
668 Log4(("{%p} " LOG_FN_FMT ": ENTER\n", this, __PRETTY_FUNCTION__));
669 /* No locking until stage 3!*/
670
671 pm::CollectorHints hints;
672 uint64_t timestamp = RTTimeMilliTS();
673 BaseMetricList toBeCollected;
674 BaseMetricList::iterator it;
675 /* Compose the list of metrics being collected at this moment */
676 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
677 if ((*it)->collectorBeat(timestamp))
678 {
679 (*it)->preCollect(hints, iTick);
680 toBeCollected.push_back(*it);
681 }
682
683 if (toBeCollected.size() == 0)
684 {
685 Log4(("{%p} " LOG_FN_FMT ": LEAVE (nothing to collect)\n", this, __PRETTY_FUNCTION__));
686 return;
687 }
688
689 /* Let know the platform specific code what is being collected */
690 m.hal->preCollect(hints, iTick);
691#if 0
692 /* Guest stats are now pushed by guests themselves */
693 /* Collect the data in bulk from all hinted guests */
694 m.gm->preCollect(hints, iTick);
695#endif
696
697 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
698 /*
699 * Before we can collect data we need to go through both lists
700 * again to see if any base metrics are marked as unregistered.
701 * Those should be destroyed now.
702 */
703 Log7(("{%p} " LOG_FN_FMT ": before remove_if: toBeCollected.size()=%d\n", this, __PRETTY_FUNCTION__, toBeCollected.size()));
704 toBeCollected.remove_if(std::mem_fun(&pm::BaseMetric::isUnregistered));
705 Log7(("{%p} " LOG_FN_FMT ": after remove_if: toBeCollected.size()=%d\n", this, __PRETTY_FUNCTION__, toBeCollected.size()));
706 Log7(("{%p} " LOG_FN_FMT ": before remove_if: m.baseMetrics.size()=%d\n", this, __PRETTY_FUNCTION__, m.baseMetrics.size()));
707 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end();)
708 if ((*it)->isUnregistered())
709 {
710 delete *it;
711 it = m.baseMetrics.erase(it);
712 }
713 else
714 ++it;
715 Log7(("{%p} " LOG_FN_FMT ": after remove_if: m.baseMetrics.size()=%d\n", this, __PRETTY_FUNCTION__, m.baseMetrics.size()));
716 /*
717 * Now when we have destroyed all base metrics that could
718 * try to pull data from unregistered CollectorGuest objects
719 * it is safe to destroy them as well.
720 */
721 m.gm->destroyUnregistered();
722
723 /* Finally, collect the data */
724 std::for_each(toBeCollected.begin(), toBeCollected.end(),
725 std::mem_fun(&pm::BaseMetric::collect));
726 Log4(("{%p} " LOG_FN_FMT ": LEAVE\n", this, __PRETTY_FUNCTION__));
727}
728
729////////////////////////////////////////////////////////////////////////////////
730// PerformanceMetric class
731////////////////////////////////////////////////////////////////////////////////
732
733// constructor / destructor
734////////////////////////////////////////////////////////////////////////////////
735
736PerformanceMetric::PerformanceMetric()
737{
738}
739
740PerformanceMetric::~PerformanceMetric()
741{
742}
743
744HRESULT PerformanceMetric::FinalConstruct()
745{
746 LogFlowThisFunc(("\n"));
747
748 return BaseFinalConstruct();
749}
750
751void PerformanceMetric::FinalRelease()
752{
753 LogFlowThisFunc(("\n"));
754
755 uninit();
756
757 BaseFinalRelease();
758}
759
760// public initializer/uninitializer for internal purposes only
761////////////////////////////////////////////////////////////////////////////////
762
763HRESULT PerformanceMetric::init(pm::Metric *aMetric)
764{
765 /* Enclose the state transition NotReady->InInit->Ready */
766 AutoInitSpan autoInitSpan(this);
767 AssertReturn(autoInitSpan.isOk(), E_FAIL);
768
769 m.name = aMetric->getName();
770 m.object = aMetric->getObject();
771 m.description = aMetric->getDescription();
772 m.period = aMetric->getPeriod();
773 m.count = aMetric->getLength();
774 m.unit = aMetric->getUnit();
775 m.min = aMetric->getMinValue();
776 m.max = aMetric->getMaxValue();
777
778 autoInitSpan.setSucceeded();
779 return S_OK;
780}
781
782HRESULT PerformanceMetric::init(pm::BaseMetric *aMetric)
783{
784 /* Enclose the state transition NotReady->InInit->Ready */
785 AutoInitSpan autoInitSpan(this);
786 AssertReturn(autoInitSpan.isOk(), E_FAIL);
787
788 m.name = aMetric->getName();
789 m.object = aMetric->getObject();
790 m.description = "";
791 m.period = aMetric->getPeriod();
792 m.count = aMetric->getLength();
793 m.unit = aMetric->getUnit();
794 m.min = aMetric->getMinValue();
795 m.max = aMetric->getMaxValue();
796
797 autoInitSpan.setSucceeded();
798 return S_OK;
799}
800
801void PerformanceMetric::uninit()
802{
803 /* Enclose the state transition Ready->InUninit->NotReady */
804 AutoUninitSpan autoUninitSpan(this);
805 if (autoUninitSpan.uninitDone())
806 {
807 LogFlowThisFunc(("Already uninitialized.\n"));
808 LogFlowThisFuncLeave();
809 return;
810 }
811}
812
813HRESULT PerformanceMetric::getMetricName(com::Utf8Str &aMetricName)
814{
815 /* this is const, no need to lock */
816 aMetricName = m.name;
817 return S_OK;
818}
819
820HRESULT PerformanceMetric::getObject(ComPtr<IUnknown> &aObject)
821{
822 /* this is const, no need to lock */
823 aObject = m.object;
824 return S_OK;
825}
826
827HRESULT PerformanceMetric::getDescription(com::Utf8Str &aDescription)
828{
829 /* this is const, no need to lock */
830 aDescription = m.description;
831 return S_OK;
832}
833
834HRESULT PerformanceMetric::getPeriod(ULONG *aPeriod)
835{
836 /* this is const, no need to lock */
837 *aPeriod = m.period;
838 return S_OK;
839}
840
841HRESULT PerformanceMetric::getCount(ULONG *aCount)
842{
843 /* this is const, no need to lock */
844 *aCount = m.count;
845 return S_OK;
846}
847
848HRESULT PerformanceMetric::getUnit(com::Utf8Str &aUnit)
849{
850 /* this is const, no need to lock */
851 aUnit = m.unit;
852 return S_OK;
853}
854
855HRESULT PerformanceMetric::getMinimumValue(LONG *aMinimumValue)
856{
857 /* this is const, no need to lock */
858 *aMinimumValue = m.min;
859 return S_OK;
860}
861
862HRESULT PerformanceMetric::getMaximumValue(LONG *aMaximumValue)
863{
864 /* this is const, no need to lock */
865 *aMaximumValue = m.max;
866 return S_OK;
867}
868/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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