VirtualBox

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

Last change on this file since 12986 was 12973, checked in by vboxsync, 16 years ago

PerfAPI: New method queryMetricsDataEx()

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 26.1 KB
Line 
1/* $Id: PerformanceImpl.cpp 12973 2008-10-03 17:46:19Z 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 m.hal = pm::createHAL();
111
112 /* Let the sampler know it gets a valid collector. */
113 mMagic = MAGIC;
114
115 /* Start resource usage sampler */
116 int vrc = RTTimerLRCreate (&m.sampler, VBOX_USAGE_SAMPLER_MIN_INTERVAL,
117 &PerformanceCollector::staticSamplerCallback, this);
118 AssertMsgRC (vrc, ("Failed to create resource usage "
119 "sampling timer(%Rra)\n", vrc));
120 if (RT_FAILURE (vrc))
121 rc = E_FAIL;
122
123 if (SUCCEEDED (rc))
124 autoInitSpan.setSucceeded();
125
126 LogFlowThisFuncLeave();
127
128 return rc;
129}
130
131/**
132 * Uninitializes the PerformanceCollector object.
133 *
134 * Called either from FinalRelease() or by the parent when it gets destroyed.
135 */
136void PerformanceCollector::uninit()
137{
138 LogFlowThisFuncEnter();
139
140 /* Enclose the state transition Ready->InUninit->NotReady */
141 AutoUninitSpan autoUninitSpan (this);
142 if (autoUninitSpan.uninitDone())
143 {
144 LogFlowThisFunc (("Already uninitialized.\n"));
145 LogFlowThisFuncLeave();
146 return;
147 }
148
149 mMagic = 0;
150
151 /* Destroy resource usage sampler */
152 int vrc = RTTimerLRDestroy (m.sampler);
153 AssertMsgRC (vrc, ("Failed to destroy resource usage "
154 "sampling timer (%Rra)\n", vrc));
155 m.sampler = NULL;
156
157 //delete m.factory;
158 //m.factory = NULL;
159
160 delete m.hal;
161 m.hal = NULL;
162
163 LogFlowThisFuncLeave();
164}
165
166// IPerformanceCollector properties
167////////////////////////////////////////////////////////////////////////////////
168
169STDMETHODIMP
170PerformanceCollector::COMGETTER(MetricNames) (ComSafeArrayOut (BSTR, theMetricNames))
171{
172 if (ComSafeArrayOutIsNull (theMetricNames))
173 return E_POINTER;
174
175 AutoCaller autoCaller (this);
176 CheckComRCReturnRC (autoCaller.rc());
177
178 AutoReadLock alock (this);
179
180 com::SafeArray <BSTR> metricNames(RT_ELEMENTS(gMetricNames));
181 for (size_t i = 0; i < RT_ELEMENTS(gMetricNames); i++)
182 {
183 gMetricNames[i].detachTo(&metricNames[i]);
184 }
185 //gMetricNames.detachTo(ComSafeArrayOutArg (theMetricNames));
186 metricNames.detachTo (ComSafeArrayOutArg (theMetricNames));
187
188 return S_OK;
189}
190
191// IPerformanceCollector methods
192////////////////////////////////////////////////////////////////////////////////
193
194HRESULT PerformanceCollector::toIPerformanceMetric(pm::Metric *src, IPerformanceMetric **dst)
195{
196 ComObjPtr <PerformanceMetric> metric;
197 HRESULT rc = metric.createObject();
198 if (SUCCEEDED (rc))
199 rc = metric->init (src);
200 AssertComRCReturnRC (rc);
201 metric.queryInterfaceTo (dst);
202 return rc;
203}
204
205HRESULT PerformanceCollector::toIPerformanceMetric(pm::BaseMetric *src, IPerformanceMetric **dst)
206{
207 ComObjPtr <PerformanceMetric> metric;
208 HRESULT rc = metric.createObject();
209 if (SUCCEEDED (rc))
210 rc = metric->init (src);
211 AssertComRCReturnRC (rc);
212 metric.queryInterfaceTo (dst);
213 return rc;
214}
215
216STDMETHODIMP
217PerformanceCollector::GetMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
218 ComSafeArrayIn (IUnknown *, objects),
219 ComSafeArrayOut (IPerformanceMetric *, outMetrics))
220{
221 LogFlowThisFuncEnter();
222 //LogFlowThisFunc (("mState=%d, mType=%d\n", mState, mType));
223
224 HRESULT rc = S_OK;
225
226 AutoCaller autoCaller (this);
227 CheckComRCReturnRC (autoCaller.rc());
228
229 pm::Filter filter (ComSafeArrayInArg (metricNames),
230 ComSafeArrayInArg (objects));
231
232 AutoReadLock alock (this);
233
234 MetricList filteredMetrics;
235 MetricList::iterator it;
236 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
237 if (filter.match ((*it)->getObject(), (*it)->getName()))
238 filteredMetrics.push_back (*it);
239
240 com::SafeIfaceArray<IPerformanceMetric> retMetrics (filteredMetrics.size());
241 int i = 0;
242 for (it = filteredMetrics.begin(); it != filteredMetrics.end(); ++it)
243 {
244 ComObjPtr <PerformanceMetric> metric;
245 rc = metric.createObject();
246 if (SUCCEEDED (rc))
247 rc = metric->init (*it);
248 AssertComRCReturnRC (rc);
249 LogFlow (("PerformanceCollector::GetMetrics() store a metric at "
250 "retMetrics[%d]...\n", i));
251 metric.queryInterfaceTo (&retMetrics [i++]);
252 }
253 retMetrics.detachTo (ComSafeArrayOutArg(outMetrics));
254 LogFlowThisFuncLeave();
255 return rc;
256}
257
258STDMETHODIMP
259PerformanceCollector::SetupMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
260 ComSafeArrayIn (IUnknown *, objects),
261 ULONG aPeriod, ULONG aCount)
262{
263 LogFlowThisFuncEnter();
264 com::SafeIfaceArray <IPerformanceMetric> tmp;
265 return SetupMetricsInt(ComSafeArrayInArg(metricNames),
266 ComSafeArrayInArg(objects),
267 aPeriod, aCount,
268 false, ComSafeArrayAsOutParam(tmp));
269}
270
271STDMETHODIMP
272PerformanceCollector::SetupMetricsEx (ComSafeArrayIn (INPTR BSTR, metricNames),
273 ComSafeArrayIn (IUnknown *, objects),
274 ULONG aPeriod, ULONG aCount,
275 ComSafeArrayOut (IPerformanceMetric *,
276 outMetrics))
277{
278 LogFlowThisFuncEnter();
279 return SetupMetricsInt(ComSafeArrayInArg(metricNames),
280 ComSafeArrayInArg(objects),
281 aPeriod, aCount,
282 true, ComSafeArrayOutArg(outMetrics));
283}
284
285HRESULT
286PerformanceCollector::SetupMetricsInt (ComSafeArrayIn (INPTR BSTR, metricNames),
287 ComSafeArrayIn (IUnknown *, objects),
288 ULONG aPeriod, ULONG aCount,
289 bool reportAffected,
290 ComSafeArrayOut (IPerformanceMetric *,
291 outMetrics))
292{
293 AutoCaller autoCaller (this);
294 CheckComRCReturnRC (autoCaller.rc());
295
296 pm::Filter filter (ComSafeArrayInArg (metricNames),
297 ComSafeArrayInArg (objects));
298
299 AutoWriteLock alock (this);
300
301 HRESULT rc = S_OK;
302 BaseMetricList filteredMetrics;
303 BaseMetricList::iterator it;
304 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
305 if (filter.match((*it)->getObject(), (*it)->getName()))
306 {
307 LogFlow (("PerformanceCollector::SetupMetrics() setting period to %u,"
308 " count to %u for %s\n", aPeriod, aCount, (*it)->getName()));
309 (*it)->init(aPeriod, aCount);
310 if (aPeriod == 0 || aCount == 0)
311 {
312 LogFlow (("PerformanceCollector::SetupMetrics() disabling %s\n",
313 (*it)->getName()));
314 (*it)->disable();
315 }
316 else
317 {
318 LogFlow (("PerformanceCollector::SetupMetrics() enabling %s\n",
319 (*it)->getName()));
320 (*it)->enable();
321 }
322 if (reportAffected)
323 filteredMetrics.push_back(*it);
324 }
325
326 if (reportAffected)
327 {
328 com::SafeIfaceArray<IPerformanceMetric> retMetrics (filteredMetrics.size());
329 int i = 0;
330 for (it = filteredMetrics.begin();
331 it != filteredMetrics.end() && SUCCEEDED (rc); ++it)
332 rc = toIPerformanceMetric(*it, &retMetrics [i++]);
333 retMetrics.detachTo (ComSafeArrayOutArg(outMetrics));
334 }
335 LogFlowThisFuncLeave();
336 return rc;
337}
338
339STDMETHODIMP
340PerformanceCollector::EnableMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
341 ComSafeArrayIn (IUnknown *, objects))
342{
343 LogFlowThisFuncEnter();
344 com::SafeIfaceArray <IPerformanceMetric> tmp;
345 return EnableMetricsInt (ComSafeArrayInArg(metricNames),
346 ComSafeArrayInArg(objects),
347 false, ComSafeArrayAsOutParam(tmp));
348}
349
350STDMETHODIMP
351PerformanceCollector::EnableMetricsEx (ComSafeArrayIn (INPTR BSTR, metricNames),
352 ComSafeArrayIn (IUnknown *, objects),
353 ComSafeArrayOut (IPerformanceMetric *,
354 outMetrics))
355{
356 LogFlowThisFuncEnter();
357 return EnableMetricsInt (ComSafeArrayInArg(metricNames),
358 ComSafeArrayInArg(objects),
359 true, ComSafeArrayOutArg(outMetrics));
360}
361
362HRESULT
363PerformanceCollector::EnableMetricsInt (ComSafeArrayIn (INPTR BSTR, metricNames),
364 ComSafeArrayIn (IUnknown *, objects),
365 bool reportAffected,
366 ComSafeArrayOut (IPerformanceMetric *,
367 outMetrics))
368{
369 AutoCaller autoCaller (this);
370 CheckComRCReturnRC (autoCaller.rc());
371
372 pm::Filter filter (ComSafeArrayInArg (metricNames),
373 ComSafeArrayInArg (objects));
374
375 AutoWriteLock alock (this); /* Write lock is not needed atm since we are */
376 /* fiddling with enable bit only, but we */
377 /* care for those who come next :-). */
378
379 HRESULT rc = S_OK;
380 BaseMetricList filteredMetrics;
381 BaseMetricList::iterator it;
382 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
383 if (filter.match((*it)->getObject(), (*it)->getName()))
384 {
385 (*it)->enable();
386 if (reportAffected)
387 filteredMetrics.push_back(*it);
388 }
389
390 if (reportAffected)
391 {
392 com::SafeIfaceArray<IPerformanceMetric> retMetrics (filteredMetrics.size());
393 int i = 0;
394 for (it = filteredMetrics.begin();
395 it != filteredMetrics.end() && SUCCEEDED (rc); ++it)
396 rc = toIPerformanceMetric(*it, &retMetrics [i++]);
397 retMetrics.detachTo (ComSafeArrayOutArg(outMetrics));
398 }
399 LogFlowThisFuncLeave();
400 return rc;
401}
402
403STDMETHODIMP
404PerformanceCollector::DisableMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
405 ComSafeArrayIn (IUnknown *, objects))
406{
407 LogFlowThisFuncEnter();
408 com::SafeIfaceArray <IPerformanceMetric> tmp;
409 return DisableMetricsInt (ComSafeArrayInArg(metricNames),
410 ComSafeArrayInArg(objects),
411 false, ComSafeArrayAsOutParam(tmp));
412}
413
414STDMETHODIMP
415PerformanceCollector::DisableMetricsEx (ComSafeArrayIn (INPTR BSTR, metricNames),
416 ComSafeArrayIn (IUnknown *, objects),
417 ComSafeArrayOut (IPerformanceMetric *,
418 outMetrics))
419{
420 LogFlowThisFuncEnter();
421 return DisableMetricsInt (ComSafeArrayInArg(metricNames),
422 ComSafeArrayInArg(objects),
423 true, ComSafeArrayOutArg(outMetrics));
424}
425
426HRESULT
427PerformanceCollector::DisableMetricsInt (ComSafeArrayIn (INPTR BSTR, metricNames),
428 ComSafeArrayIn (IUnknown *, objects),
429 bool reportAffected,
430 ComSafeArrayOut (IPerformanceMetric *,
431 outMetrics))
432{
433 AutoCaller autoCaller (this);
434 CheckComRCReturnRC (autoCaller.rc());
435
436 pm::Filter filter (ComSafeArrayInArg (metricNames),
437 ComSafeArrayInArg (objects));
438
439 AutoWriteLock alock (this); /* Write lock is not needed atm since we are */
440 /* fiddling with enable bit only, but we */
441 /* care for those who come next :-). */
442
443 HRESULT rc = S_OK;
444 BaseMetricList filteredMetrics;
445 BaseMetricList::iterator it;
446 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
447 if (filter.match((*it)->getObject(), (*it)->getName()))
448 {
449 (*it)->disable();
450 if (reportAffected)
451 filteredMetrics.push_back(*it);
452 }
453
454 if (reportAffected)
455 {
456 com::SafeIfaceArray<IPerformanceMetric> retMetrics (filteredMetrics.size());
457 int i = 0;
458 for (it = filteredMetrics.begin();
459 it != filteredMetrics.end() && SUCCEEDED (rc); ++it)
460 rc = toIPerformanceMetric(*it, &retMetrics [i++]);
461 retMetrics.detachTo (ComSafeArrayOutArg(outMetrics));
462 }
463 LogFlowThisFuncLeave();
464 return rc;
465}
466
467STDMETHODIMP
468PerformanceCollector::QueryMetricsData (ComSafeArrayIn (INPTR BSTR, metricNames),
469 ComSafeArrayIn (IUnknown *, objects),
470 ComSafeArrayOut (BSTR, outMetricNames),
471 ComSafeArrayOut (IUnknown *, outObjects),
472 ComSafeArrayOut (ULONG, outDataIndices),
473 ComSafeArrayOut (ULONG, outDataLengths),
474 ComSafeArrayOut (LONG, outData))
475{
476 com::SafeArray <BSTR> tmpUnits;
477 com::SafeArray <ULONG> tmpScales;
478 com::SafeArray <ULONG> tmpSequenceNumbers;
479 return QueryMetricsDataEx(ComSafeArrayInArg(metricNames),
480 ComSafeArrayInArg(objects),
481 ComSafeArrayOutArg(outMetricNames),
482 ComSafeArrayOutArg(outObjects),
483 ComSafeArrayAsOutParam(tmpUnits),
484 ComSafeArrayAsOutParam(tmpScales),
485 ComSafeArrayAsOutParam(tmpSequenceNumbers),
486 ComSafeArrayOutArg(outDataIndices),
487 ComSafeArrayOutArg(outDataLengths),
488 ComSafeArrayOutArg(outData));
489}
490
491STDMETHODIMP
492PerformanceCollector::QueryMetricsDataEx (ComSafeArrayIn (INPTR BSTR, metricNames),
493 ComSafeArrayIn (IUnknown *, objects),
494 ComSafeArrayOut (BSTR, outMetricNames),
495 ComSafeArrayOut (IUnknown *, outObjects),
496 ComSafeArrayOut (BSTR, outUnits),
497 ComSafeArrayOut (ULONG, outScales),
498 ComSafeArrayOut (ULONG, outSequenceNumbers),
499 ComSafeArrayOut (ULONG, outDataIndices),
500 ComSafeArrayOut (ULONG, outDataLengths),
501 ComSafeArrayOut (LONG, outData))
502{
503 AutoCaller autoCaller (this);
504 CheckComRCReturnRC (autoCaller.rc());
505
506 pm::Filter filter (ComSafeArrayInArg (metricNames),
507 ComSafeArrayInArg (objects));
508
509 AutoReadLock alock (this);
510
511 /* Let's compute the size of the resulting flat array */
512 size_t flatSize = 0;
513 MetricList filteredMetrics;
514 MetricList::iterator it;
515 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
516 if (filter.match ((*it)->getObject(), (*it)->getName()))
517 {
518 filteredMetrics.push_back (*it);
519 flatSize += (*it)->getLength();
520 }
521
522 int i = 0;
523 size_t flatIndex = 0;
524 size_t numberOfMetrics = filteredMetrics.size();
525 com::SafeArray <BSTR> retNames (numberOfMetrics);
526 com::SafeIfaceArray <IUnknown> retObjects (numberOfMetrics);
527 com::SafeArray <BSTR> retUnits (numberOfMetrics);
528 com::SafeArray <ULONG> retScales (numberOfMetrics);
529 com::SafeArray <ULONG> retSequenceNumbers (numberOfMetrics);
530 com::SafeArray <ULONG> retIndices (numberOfMetrics);
531 com::SafeArray <ULONG> retLengths (numberOfMetrics);
532 com::SafeArray <LONG> retData (flatSize);
533
534 for (it = filteredMetrics.begin(); it != filteredMetrics.end(); ++it, ++i)
535 {
536 ULONG *values, length, sequenceNumber;
537 /* @todo We may want to revise the query method to get rid of excessive alloc/memcpy calls. */
538 (*it)->query(&values, &length, &sequenceNumber);
539 LogFlow (("PerformanceCollector::QueryMetricsData() querying metric %s "
540 "returned %d values.\n", (*it)->getName(), length));
541 memcpy(retData.raw() + flatIndex, values, length * sizeof(*values));
542 Bstr tmp((*it)->getName());
543 tmp.detachTo(&retNames[i]);
544 (*it)->getObject().queryInterfaceTo (&retObjects[i]);
545 tmp = (*it)->getUnit();
546 tmp.detachTo(&retUnits[i]);
547 retScales[i] = (*it)->getScale();
548 retSequenceNumbers[i] = sequenceNumber;
549 retLengths[i] = length;
550 retIndices[i] = flatIndex;
551 flatIndex += length;
552 }
553
554 retNames.detachTo (ComSafeArrayOutArg (outMetricNames));
555 retObjects.detachTo (ComSafeArrayOutArg (outObjects));
556 retUnits.detachTo (ComSafeArrayOutArg (outUnits));
557 retScales.detachTo (ComSafeArrayOutArg (outScales));
558 retSequenceNumbers.detachTo (ComSafeArrayOutArg (outSequenceNumbers));
559 retIndices.detachTo (ComSafeArrayOutArg (outDataIndices));
560 retLengths.detachTo (ComSafeArrayOutArg (outDataLengths));
561 retData.detachTo (ComSafeArrayOutArg (outData));
562 return S_OK;
563}
564
565// public methods for internal purposes
566///////////////////////////////////////////////////////////////////////////////
567
568void PerformanceCollector::registerBaseMetric (pm::BaseMetric *baseMetric)
569{
570 //LogFlowThisFuncEnter();
571 AutoCaller autoCaller (this);
572 if (!SUCCEEDED (autoCaller.rc())) return;
573
574 AutoWriteLock alock (this);
575 LogAleksey(("{%p} " LOG_FN_FMT ": obj=%p name=%s\n", this, __PRETTY_FUNCTION__, (void *)baseMetric->getObject(), baseMetric->getName()));
576 m.baseMetrics.push_back (baseMetric);
577 //LogFlowThisFuncLeave();
578}
579
580void PerformanceCollector::registerMetric (pm::Metric *metric)
581{
582 //LogFlowThisFuncEnter();
583 AutoCaller autoCaller (this);
584 if (!SUCCEEDED (autoCaller.rc())) return;
585
586 AutoWriteLock alock (this);
587 LogAleksey(("{%p} " LOG_FN_FMT ": obj=%p name=%s\n", this, __PRETTY_FUNCTION__, (void *)metric->getObject(), metric->getName()));
588 m.metrics.push_back (metric);
589 //LogFlowThisFuncLeave();
590}
591
592void PerformanceCollector::unregisterBaseMetricsFor (const ComPtr <IUnknown> &aObject)
593{
594 //LogFlowThisFuncEnter();
595 AutoCaller autoCaller (this);
596 if (!SUCCEEDED (autoCaller.rc())) return;
597
598 AutoWriteLock alock (this);
599 LogAleksey(("{%p} " LOG_FN_FMT ": before remove_if: m.baseMetrics.size()=%d\n", this, __PRETTY_FUNCTION__, m.baseMetrics.size()));
600 BaseMetricList::iterator it = std::remove_if (
601 m.baseMetrics.begin(), m.baseMetrics.end(), std::bind2nd (
602 std::mem_fun (&pm::BaseMetric::associatedWith), aObject));
603 m.baseMetrics.erase(it, m.baseMetrics.end());
604 LogAleksey(("{%p} " LOG_FN_FMT ": after remove_if: m.baseMetrics.size()=%d\n", this, __PRETTY_FUNCTION__, m.baseMetrics.size()));
605 //LogFlowThisFuncLeave();
606}
607
608void PerformanceCollector::unregisterMetricsFor (const ComPtr <IUnknown> &aObject)
609{
610 //LogFlowThisFuncEnter();
611 AutoCaller autoCaller (this);
612 if (!SUCCEEDED (autoCaller.rc())) return;
613
614 AutoWriteLock alock (this);
615 LogAleksey(("{%p} " LOG_FN_FMT ": obj=%p\n", this, __PRETTY_FUNCTION__, (void *)aObject));
616 MetricList::iterator it = std::remove_if (
617 m.metrics.begin(), m.metrics.end(), std::bind2nd (
618 std::mem_fun (&pm::Metric::associatedWith), aObject));
619 m.metrics.erase(it, m.metrics.end());
620 //LogFlowThisFuncLeave();
621}
622
623// private methods
624///////////////////////////////////////////////////////////////////////////////
625
626/* static */
627void PerformanceCollector::staticSamplerCallback (RTTIMERLR hTimerLR, void *pvUser,
628 uint64_t iTick)
629{
630 AssertReturnVoid (pvUser != NULL);
631 PerformanceCollector *collector = static_cast <PerformanceCollector *> (pvUser);
632 Assert (collector->mMagic == MAGIC);
633 if (collector->mMagic == MAGIC)
634 {
635 collector->samplerCallback();
636 }
637 NOREF (hTimerLR);
638}
639
640void PerformanceCollector::samplerCallback()
641{
642 Log4(("{%p} " LOG_FN_FMT ": ENTER\n", this, __PRETTY_FUNCTION__));
643 AutoWriteLock alock (this);
644
645 pm::CollectorHints hints;
646 uint64_t timestamp = RTTimeMilliTS();
647 BaseMetricList toBeCollected;
648 BaseMetricList::iterator it;
649 /* Compose the list of metrics being collected at this moment */
650 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); it++)
651 if ((*it)->collectorBeat(timestamp))
652 {
653 (*it)->preCollect(hints);
654 toBeCollected.push_back(*it);
655 }
656
657 if (toBeCollected.size() == 0)
658 return;
659
660 /* Let know the platform specific code what is being collected */
661 m.hal->preCollect(hints);
662
663 /* Finally, collect the data */
664 std::for_each (toBeCollected.begin(), toBeCollected.end(),
665 std::mem_fun (&pm::BaseMetric::collect));
666 Log4(("{%p} " LOG_FN_FMT ": LEAVE\n", this, __PRETTY_FUNCTION__));
667}
668
669////////////////////////////////////////////////////////////////////////////////
670// PerformanceMetric class
671////////////////////////////////////////////////////////////////////////////////
672
673// constructor / destructor
674////////////////////////////////////////////////////////////////////////////////
675
676PerformanceMetric::PerformanceMetric()
677{
678}
679
680PerformanceMetric::~PerformanceMetric()
681{
682}
683
684HRESULT PerformanceMetric::FinalConstruct()
685{
686 LogFlowThisFunc (("\n"));
687
688 return S_OK;
689}
690
691void PerformanceMetric::FinalRelease()
692{
693 LogFlowThisFunc (("\n"));
694
695 uninit ();
696}
697
698// public initializer/uninitializer for internal purposes only
699////////////////////////////////////////////////////////////////////////////////
700
701HRESULT PerformanceMetric::init (pm::Metric *aMetric)
702{
703 m.name = aMetric->getName();
704 m.object = aMetric->getObject();
705 m.description = aMetric->getDescription();
706 m.period = aMetric->getPeriod();
707 m.count = aMetric->getLength();
708 m.unit = aMetric->getUnit();
709 m.min = aMetric->getMinValue();
710 m.max = aMetric->getMaxValue();
711 return S_OK;
712}
713
714HRESULT PerformanceMetric::init (pm::BaseMetric *aMetric)
715{
716 m.name = aMetric->getName();
717 m.object = aMetric->getObject();
718 m.description = "";
719 m.period = aMetric->getPeriod();
720 m.count = aMetric->getLength();
721 m.unit = aMetric->getUnit();
722 m.min = aMetric->getMinValue();
723 m.max = aMetric->getMaxValue();
724 return S_OK;
725}
726
727void PerformanceMetric::uninit()
728{
729}
730
731STDMETHODIMP PerformanceMetric::COMGETTER(MetricName) (BSTR *aMetricName)
732{
733 /// @todo (r=dmik) why do all these getters not do AutoCaller and
734 /// AutoReadLock? Is the underlying metric a constant object?
735
736 m.name.cloneTo (aMetricName);
737 return S_OK;
738}
739
740STDMETHODIMP PerformanceMetric::COMGETTER(Object) (IUnknown **anObject)
741{
742 m.object.queryInterfaceTo(anObject);
743 return S_OK;
744}
745
746STDMETHODIMP PerformanceMetric::COMGETTER(Description) (BSTR *aDescription)
747{
748 m.description.cloneTo (aDescription);
749 return S_OK;
750}
751
752STDMETHODIMP PerformanceMetric::COMGETTER(Period) (ULONG *aPeriod)
753{
754 *aPeriod = m.period;
755 return S_OK;
756}
757
758STDMETHODIMP PerformanceMetric::COMGETTER(Count) (ULONG *aCount)
759{
760 *aCount = m.count;
761 return S_OK;
762}
763
764STDMETHODIMP PerformanceMetric::COMGETTER(Unit) (BSTR *aUnit)
765{
766 m.unit.cloneTo(aUnit);
767 return S_OK;
768}
769
770STDMETHODIMP PerformanceMetric::COMGETTER(MinimumValue) (LONG *aMinValue)
771{
772 *aMinValue = m.min;
773 return S_OK;
774}
775
776STDMETHODIMP PerformanceMetric::COMGETTER(MaximumValue) (LONG *aMaxValue)
777{
778 *aMaxValue = m.max;
779 return S_OK;
780}
781
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