VirtualBox

source: vbox/trunk/src/VBox/Main/PerformanceImpl.cpp@ 11527

Last change on this file since 11527 was 11481, checked in by vboxsync, 16 years ago

PerfAPI: CPU/MHz counter name fix, proper handling of zero periods and counts

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.4 KB
Line 
1/* $Id: PerformanceImpl.cpp 11481 2008-08-19 12:13:53Z vboxsync $ */
2
3/** @file
4 *
5 * VBox Performance API COM Classes implementation
6 */
7
8/*
9 * Copyright (C) 2008 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23
24#if defined(RT_OS_WINDOWS)
25#elif defined(RT_OS_LINUX)
26#endif
27
28#include "PerformanceImpl.h"
29
30#include "Logging.h"
31
32#include <VBox/err.h>
33#include <iprt/process.h>
34
35#include <vector>
36#include <algorithm>
37#include <functional>
38
39static Bstr gMetricNames[] =
40{
41 "CPU/Load/User",
42 "CPU/Load/User:avg",
43 "CPU/Load/User:min",
44 "CPU/Load/User:max",
45 "CPU/Load/Kernel",
46 "CPU/Load/Kernel:avg",
47 "CPU/Load/Kernel:min",
48 "CPU/Load/Kernel:max",
49 "CPU/Load/Idle",
50 "CPU/Load/Idle:avg",
51 "CPU/Load/Idle:min",
52 "CPU/Load/Idle:max",
53 "CPU/MHz",
54 "CPU/MHz:avg",
55 "CPU/MHz:min",
56 "CPU/MHz:max",
57 "RAM/Usage/Total",
58 "RAM/Usage/Total:avg",
59 "RAM/Usage/Total:min",
60 "RAM/Usage/Total:max",
61 "RAM/Usage/Used",
62 "RAM/Usage/Used:avg",
63 "RAM/Usage/Used:min",
64 "RAM/Usage/Used:max",
65 "RAM/Usage/Free",
66 "RAM/Usage/Free:avg",
67 "RAM/Usage/Free:min",
68 "RAM/Usage/Free:max",
69};
70
71////////////////////////////////////////////////////////////////////////////////
72// PerformanceCollector class
73////////////////////////////////////////////////////////////////////////////////
74
75// constructor / destructor
76////////////////////////////////////////////////////////////////////////////////
77
78PerformanceCollector::PerformanceCollector() : mMagic(0) {}
79
80PerformanceCollector::~PerformanceCollector() {}
81
82HRESULT PerformanceCollector::FinalConstruct()
83{
84 LogFlowThisFunc (("\n"));
85
86 return S_OK;
87}
88
89void PerformanceCollector::FinalRelease()
90{
91 LogFlowThisFunc (("\n"));
92}
93
94// public initializer/uninitializer for internal purposes only
95////////////////////////////////////////////////////////////////////////////////
96
97/**
98 * Initializes the PerformanceCollector object.
99 */
100HRESULT PerformanceCollector::init()
101{
102 /* Enclose the state transition NotReady->InInit->Ready */
103 AutoInitSpan autoInitSpan (this);
104 AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED);
105
106 LogFlowThisFuncEnter();
107
108 HRESULT rc = S_OK;
109
110 /* @todo Obviously other platforms must be added as well. */
111#ifdef RT_OS_SOLARIS
112 m.factory = new pm::MetricFactorySolaris();
113#endif
114#ifdef RT_OS_LINUX
115 m.factory = new pm::MetricFactoryLinux();
116#endif
117#ifdef RT_OS_WINDOWS
118 m.factory = new pm::MetricFactoryWin();
119#endif
120#ifdef RT_OS_OS2
121 m.factory = new pm::MetricFactoryOS2();
122#endif
123#ifdef RT_OS_DARWIN
124 m.factory = new pm::MetricFactoryDarwin();
125#endif
126
127 /* Let the sampler know it gets a valid collector. */
128 mMagic = MAGIC;
129
130 /* Start resource usage sampler */
131 int vrc = RTTimerLRCreate (&m.sampler, VBOX_USAGE_SAMPLER_MIN_INTERVAL,
132 &PerformanceCollector::staticSamplerCallback, this);
133 AssertMsgRC (vrc, ("Failed to create resource usage "
134 "sampling timer(%Rra)\n", vrc));
135 if (RT_FAILURE (vrc))
136 rc = E_FAIL;
137
138 if (SUCCEEDED (rc))
139 autoInitSpan.setSucceeded();
140
141 LogFlowThisFuncLeave();
142
143 return rc;
144}
145
146/**
147 * Uninitializes the PerformanceCollector object.
148 *
149 * Called either from FinalRelease() or by the parent when it gets destroyed.
150 */
151void PerformanceCollector::uninit()
152{
153 LogFlowThisFuncEnter();
154
155 /* Enclose the state transition Ready->InUninit->NotReady */
156 AutoUninitSpan autoUninitSpan (this);
157 if (autoUninitSpan.uninitDone())
158 {
159 LogFlowThisFunc (("Already uninitialized.\n"));
160 LogFlowThisFuncLeave();
161 return;
162 }
163
164 mMagic = 0;
165
166 /* Destroy resource usage sampler */
167 int vrc = RTTimerLRDestroy (m.sampler);
168 AssertMsgRC (vrc, ("Failed to destroy resource usage "
169 "sampling timer (%Rra)\n", vrc));
170 m.sampler = NULL;
171
172 delete m.factory;
173 m.factory = NULL;
174
175 LogFlowThisFuncLeave();
176}
177
178// IPerformanceCollector properties
179////////////////////////////////////////////////////////////////////////////////
180
181STDMETHODIMP
182PerformanceCollector::COMGETTER(MetricNames) (ComSafeArrayOut (BSTR, theMetricNames))
183{
184 if (ComSafeArrayOutIsNull (theMetricNames))
185 return E_POINTER;
186
187 AutoCaller autoCaller (this);
188 CheckComRCReturnRC (autoCaller.rc());
189
190 AutoReadLock alock (this);
191
192 com::SafeArray <BSTR> metricNames(RT_ELEMENTS(gMetricNames));
193 for (size_t i = 0; i < RT_ELEMENTS(gMetricNames); i++)
194 {
195 gMetricNames[i].detachTo(&metricNames[i]);
196 }
197 //gMetricNames.detachTo(ComSafeArrayOutArg (theMetricNames));
198 metricNames.detachTo (ComSafeArrayOutArg (theMetricNames));
199
200 return S_OK;
201}
202
203// IPerformanceCollector methods
204////////////////////////////////////////////////////////////////////////////////
205
206STDMETHODIMP
207PerformanceCollector::GetMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
208 ComSafeArrayIn (IUnknown *, objects),
209 ComSafeArrayOut (IPerformanceMetric *, outMetrics))
210{
211 LogFlowThisFuncEnter();
212 //LogFlowThisFunc (("mState=%d, mType=%d\n", mState, mType));
213
214 HRESULT rc = S_OK;
215
216 AutoCaller autoCaller (this);
217 CheckComRCReturnRC (autoCaller.rc());
218
219 pm::Filter filter (ComSafeArrayInArg (metricNames),
220 ComSafeArrayInArg (objects));
221
222 AutoReadLock alock (this);
223
224 MetricList filteredMetrics;
225 MetricList::iterator it;
226 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
227 if (filter.match ((*it)->getObject(), (*it)->getName()))
228 filteredMetrics.push_back (*it);
229
230 com::SafeIfaceArray<IPerformanceMetric> retMetrics (filteredMetrics.size());
231 int i = 0;
232 for (it = filteredMetrics.begin(); it != filteredMetrics.end(); ++it)
233 {
234 ComObjPtr <PerformanceMetric> metric;
235 rc = metric.createObject();
236 if (SUCCEEDED (rc))
237 rc = metric->init (*it);
238 AssertComRCReturnRC (rc);
239 LogFlow (("PerformanceCollector::GetMetrics() store a metric at "
240 "retMetrics[%d]...\n", i));
241 metric.queryInterfaceTo (&retMetrics [i++]);
242 }
243 retMetrics.detachTo (ComSafeArrayOutArg(outMetrics));
244 LogFlowThisFuncLeave();
245 return rc;
246}
247
248STDMETHODIMP
249PerformanceCollector::SetupMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
250 ComSafeArrayIn (IUnknown *, objects),
251 ULONG aPeriod, ULONG aCount)
252{
253 AutoCaller autoCaller (this);
254 CheckComRCReturnRC (autoCaller.rc());
255
256 pm::Filter filter (ComSafeArrayInArg (metricNames),
257 ComSafeArrayInArg (objects));
258
259 AutoWriteLock alock (this);
260
261 BaseMetricList::iterator it;
262 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
263 if (filter.match((*it)->getObject(), (*it)->getName()))
264 {
265 LogFlow (("PerformanceCollector::SetupMetrics() setting period to %u,"
266 " count to %u for %s\n", aPeriod, aCount, (*it)->getName()));
267 (*it)->init(aPeriod, aCount);
268 if (aPeriod == 0 || aCount == 0)
269 {
270 LogFlow (("PerformanceCollector::SetupMetrics() disabling %s\n",
271 (*it)->getName()));
272 (*it)->disable();
273 }
274 else
275 {
276 LogFlow (("PerformanceCollector::SetupMetrics() enabling %s\n",
277 (*it)->getName()));
278 (*it)->enable();
279 }
280 }
281
282 return S_OK;
283}
284
285STDMETHODIMP
286PerformanceCollector::EnableMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
287 ComSafeArrayIn (IUnknown *, objects))
288{
289 AutoCaller autoCaller (this);
290 CheckComRCReturnRC (autoCaller.rc());
291
292 pm::Filter filter (ComSafeArrayInArg (metricNames),
293 ComSafeArrayInArg (objects));
294
295 AutoWriteLock alock (this); /* Write lock is not needed atm since we are */
296 /* fiddling with enable bit only, but we */
297 /* care for those who come next :-). */
298
299 BaseMetricList::iterator it;
300 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
301 if (filter.match((*it)->getObject(), (*it)->getName()))
302 (*it)->enable();
303
304 return S_OK;
305}
306
307STDMETHODIMP
308PerformanceCollector::DisableMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
309 ComSafeArrayIn (IUnknown *, objects))
310{
311 AutoCaller autoCaller (this);
312 CheckComRCReturnRC (autoCaller.rc());
313
314 pm::Filter filter (ComSafeArrayInArg (metricNames),
315 ComSafeArrayInArg (objects));
316
317 AutoWriteLock alock (this); /* Write lock is not needed atm since we are */
318 /* fiddling with enable bit only, but we */
319 /* care for those who come next :-). */
320
321 BaseMetricList::iterator it;
322 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
323 if (filter.match((*it)->getObject(), (*it)->getName()))
324 (*it)->disable();
325
326 return S_OK;
327}
328
329STDMETHODIMP
330PerformanceCollector::QueryMetricsData (ComSafeArrayIn (INPTR BSTR, metricNames),
331 ComSafeArrayIn (IUnknown *, objects),
332 ComSafeArrayOut (BSTR, outMetricNames),
333 ComSafeArrayOut (IUnknown *, outObjects),
334 ComSafeArrayOut (ULONG, outDataIndices),
335 ComSafeArrayOut (ULONG, outDataLengths),
336 ComSafeArrayOut (LONG, outData))
337{
338 AutoCaller autoCaller (this);
339 CheckComRCReturnRC (autoCaller.rc());
340
341 pm::Filter filter (ComSafeArrayInArg (metricNames),
342 ComSafeArrayInArg (objects));
343
344 AutoReadLock alock (this);
345
346 /* Let's compute the size of the resulting flat array */
347 size_t flatSize = 0;
348 MetricList filteredMetrics;
349 MetricList::iterator it;
350 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
351 if (filter.match ((*it)->getObject(), (*it)->getName()))
352 {
353 filteredMetrics.push_back (*it);
354 flatSize += (*it)->getLength();
355 }
356
357 int i = 0;
358 size_t flatIndex = 0;
359 size_t numberOfMetrics = filteredMetrics.size();
360 com::SafeArray <BSTR> retNames (numberOfMetrics);
361 com::SafeIfaceArray <IUnknown> retObjects (numberOfMetrics);
362 com::SafeArray <ULONG> retIndices (numberOfMetrics);
363 com::SafeArray <ULONG> retLengths (numberOfMetrics);
364 com::SafeArray <LONG> retData (flatSize);
365
366 for (it = filteredMetrics.begin(); it != filteredMetrics.end(); ++it, ++i)
367 {
368 /* @todo Filtering goes here! */
369 ULONG *values, length;
370 /* @todo We may want to revise the query method to get rid of excessive alloc/memcpy calls. */
371 (*it)->query(&values, &length);
372 LogFlow (("PerformanceCollector::QueryMetricsData() querying metric %s "
373 "returned %d values.\n", (*it)->getName(), length));
374 memcpy(retData.raw() + flatIndex, values, length * sizeof(*values));
375 Bstr tmp((*it)->getName());
376 tmp.detachTo(&retNames[i]);
377 (*it)->getObject().queryInterfaceTo (&retObjects[i]);
378 retLengths[i] = length;
379 retIndices[i] = flatIndex;
380 flatIndex += length;
381 }
382
383 retNames.detachTo (ComSafeArrayOutArg (outMetricNames));
384 retObjects.detachTo (ComSafeArrayOutArg (outObjects));
385 retIndices.detachTo (ComSafeArrayOutArg (outDataIndices));
386 retLengths.detachTo (ComSafeArrayOutArg (outDataLengths));
387 retData.detachTo (ComSafeArrayOutArg (outData));
388 return S_OK;
389}
390
391// public methods for internal purposes
392///////////////////////////////////////////////////////////////////////////////
393
394void PerformanceCollector::registerBaseMetric (pm::BaseMetric *baseMetric)
395{
396 //LogFlowThisFuncEnter();
397 AutoCaller autoCaller (this);
398 if (!SUCCEEDED (autoCaller.rc())) return;
399
400 AutoWriteLock alock (this);
401 LogAleksey(("{%p} " LOG_FN_FMT ": obj=%p name=%s\n", this, __PRETTY_FUNCTION__, (void *)baseMetric->getObject(), baseMetric->getName()));
402 m.baseMetrics.push_back (baseMetric);
403 //LogFlowThisFuncLeave();
404}
405
406void PerformanceCollector::registerMetric (pm::Metric *metric)
407{
408 //LogFlowThisFuncEnter();
409 AutoCaller autoCaller (this);
410 if (!SUCCEEDED (autoCaller.rc())) return;
411
412 AutoWriteLock alock (this);
413 LogAleksey(("{%p} " LOG_FN_FMT ": obj=%p name=%s\n", this, __PRETTY_FUNCTION__, (void *)metric->getObject(), metric->getName()));
414 m.metrics.push_back (metric);
415 //LogFlowThisFuncLeave();
416}
417
418void PerformanceCollector::unregisterBaseMetricsFor (const ComPtr <IUnknown> &aObject)
419{
420 //LogFlowThisFuncEnter();
421 AutoCaller autoCaller (this);
422 if (!SUCCEEDED (autoCaller.rc())) return;
423
424 AutoWriteLock alock (this);
425 LogAleksey(("{%p} " LOG_FN_FMT ": before remove_if: m.baseMetrics.size()=%d\n", this, __PRETTY_FUNCTION__, m.baseMetrics.size()));
426 BaseMetricList::iterator it = std::remove_if (
427 m.baseMetrics.begin(), m.baseMetrics.end(), std::bind2nd (
428 std::mem_fun (&pm::BaseMetric::associatedWith), aObject));
429 m.baseMetrics.erase(it, m.baseMetrics.end());
430 LogAleksey(("{%p} " LOG_FN_FMT ": after remove_if: m.baseMetrics.size()=%d\n", this, __PRETTY_FUNCTION__, m.baseMetrics.size()));
431 //LogFlowThisFuncLeave();
432}
433
434void PerformanceCollector::unregisterMetricsFor (const ComPtr <IUnknown> &aObject)
435{
436 //LogFlowThisFuncEnter();
437 AutoCaller autoCaller (this);
438 if (!SUCCEEDED (autoCaller.rc())) return;
439
440 AutoWriteLock alock (this);
441 LogAleksey(("{%p} " LOG_FN_FMT ": obj=%p\n", this, __PRETTY_FUNCTION__, (void *)aObject));
442 MetricList::iterator it = std::remove_if (
443 m.metrics.begin(), m.metrics.end(), std::bind2nd (
444 std::mem_fun (&pm::Metric::associatedWith), aObject));
445 m.metrics.erase(it, m.metrics.end());
446 //LogFlowThisFuncLeave();
447}
448
449// private methods
450///////////////////////////////////////////////////////////////////////////////
451
452/* static */
453void PerformanceCollector::staticSamplerCallback (RTTIMERLR hTimerLR, void *pvUser,
454 uint64_t iTick)
455{
456 AssertReturnVoid (pvUser != NULL);
457 PerformanceCollector *collector = static_cast <PerformanceCollector *> (pvUser);
458 Assert (collector->mMagic == MAGIC);
459 if (collector->mMagic == MAGIC)
460 {
461 collector->samplerCallback();
462 }
463 NOREF (hTimerLR);
464}
465
466void PerformanceCollector::samplerCallback()
467{
468 Log4(("{%p} " LOG_FN_FMT ": ENTER\n", this, __PRETTY_FUNCTION__));
469 AutoWriteLock alock (this);
470
471 uint64_t timestamp = RTTimeMilliTS();
472 std::for_each (m.baseMetrics.begin(), m.baseMetrics.end(),
473 std::bind2nd (std::mem_fun (&pm::BaseMetric::collectorBeat),
474 timestamp));
475 Log4(("{%p} " LOG_FN_FMT ": LEAVE\n", this, __PRETTY_FUNCTION__));
476}
477
478////////////////////////////////////////////////////////////////////////////////
479// PerformanceMetric class
480////////////////////////////////////////////////////////////////////////////////
481
482// constructor / destructor
483////////////////////////////////////////////////////////////////////////////////
484
485PerformanceMetric::PerformanceMetric()
486{
487}
488
489PerformanceMetric::~PerformanceMetric()
490{
491}
492
493HRESULT PerformanceMetric::FinalConstruct()
494{
495 LogFlowThisFunc (("\n"));
496
497 return S_OK;
498}
499
500void PerformanceMetric::FinalRelease()
501{
502 LogFlowThisFunc (("\n"));
503
504 uninit ();
505}
506
507// public initializer/uninitializer for internal purposes only
508////////////////////////////////////////////////////////////////////////////////
509
510HRESULT PerformanceMetric::init (pm::Metric *aMetric)
511{
512 m.name = aMetric->getName();
513 m.object = aMetric->getObject();
514 m.period = aMetric->getPeriod();
515 m.count = aMetric->getLength();
516 m.unit = aMetric->getUnit();
517 m.min = aMetric->getMinValue();
518 m.max = aMetric->getMaxValue();
519 return S_OK;
520}
521
522void PerformanceMetric::uninit()
523{
524}
525
526STDMETHODIMP PerformanceMetric::COMGETTER(MetricName) (BSTR *aMetricName)
527{
528 /// @todo (r=dmik) why do all these getters not do AutoCaller and
529 /// AutoReadLock? Is the underlying metric a constant object?
530
531 m.name.cloneTo (aMetricName);
532 return S_OK;
533}
534
535STDMETHODIMP PerformanceMetric::COMGETTER(Object) (IUnknown **anObject)
536{
537 m.object.queryInterfaceTo(anObject);
538 return S_OK;
539}
540
541STDMETHODIMP PerformanceMetric::COMGETTER(Period) (ULONG *aPeriod)
542{
543 *aPeriod = m.period;
544 return S_OK;
545}
546
547STDMETHODIMP PerformanceMetric::COMGETTER(Count) (ULONG *aCount)
548{
549 *aCount = m.count;
550 return S_OK;
551}
552
553STDMETHODIMP PerformanceMetric::COMGETTER(Unit) (BSTR *aUnit)
554{
555 m.unit.cloneTo(aUnit);
556 return S_OK;
557}
558
559STDMETHODIMP PerformanceMetric::COMGETTER(MinimumValue) (LONG *aMinValue)
560{
561 *aMinValue = m.min;
562 return S_OK;
563}
564
565STDMETHODIMP PerformanceMetric::COMGETTER(MaximumValue) (LONG *aMaxValue)
566{
567 *aMaxValue = m.max;
568 return S_OK;
569}
570
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