VirtualBox

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

Last change on this file since 10771 was 10770, checked in by vboxsync, 16 years ago

Main: Performance: Second m is redundant. Some locking-related todos.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.1 KB
Line 
1/* $Id: PerformanceImpl.cpp 10770 2008-07-21 09:02:08Z 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:avg",
42 "CPU/Load/User:min",
43 "CPU/Load/User:max",
44 "CPU/Load/Kernel:avg",
45 "CPU/Load/Kernel:min",
46 "CPU/Load/Kernel:max",
47 "CPU/Load/Idle:avg",
48 "CPU/Load/Idle:min",
49 "CPU/Load/Idle:max",
50 "CPU/MHz:avg",
51 "CPU/MHz:min",
52 "CPU/MHz:max",
53 "RAM/Usage/Total:avg",
54 "RAM/Usage/Total:min",
55 "RAM/Usage/Total:max",
56 "RAM/Usage/Used:avg",
57 "RAM/Usage/Used:min",
58 "RAM/Usage/Used:max",
59 "RAM/Usage/Free:avg",
60 "RAM/Usage/Free:min",
61 "RAM/Usage/Free:max",
62};
63
64////////////////////////////////////////////////////////////////////////////////
65// PerformanceCollector class
66////////////////////////////////////////////////////////////////////////////////
67
68// constructor / destructor
69////////////////////////////////////////////////////////////////////////////////
70
71PerformanceCollector::PerformanceCollector() : mMagic(0) {}
72
73PerformanceCollector::~PerformanceCollector() {}
74
75HRESULT PerformanceCollector::FinalConstruct()
76{
77 LogFlowThisFunc (("\n"));
78
79 return S_OK;
80}
81
82void PerformanceCollector::FinalRelease()
83{
84 LogFlowThisFunc (("\n"));
85}
86
87// public initializer/uninitializer for internal purposes only
88////////////////////////////////////////////////////////////////////////////////
89
90/**
91 * Initializes the PerformanceCollector object.
92 */
93HRESULT PerformanceCollector::init()
94{
95 /* Enclose the state transition NotReady->InInit->Ready */
96 AutoInitSpan autoInitSpan (this);
97 AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED);
98
99 LogFlowThisFuncEnter();
100
101 HRESULT rc = S_OK;
102
103 /* @todo Obviously other platforms must be added as well. */
104#ifdef RT_OS_SOLARIS
105 m.factory = new pm::MetricFactorySolaris();
106#endif
107#ifdef RT_OS_LINUX
108 m.factory = new pm::MetricFactoryLinux();
109#endif
110#ifdef RT_OS_WINDOWS
111 m.factory = new pm::MetricFactoryWin();
112#endif
113#ifdef RT_OS_OS2
114 m.factory = new pm::MetricFactoryOS2();
115#endif
116#ifdef RT_OS_DARWIN
117 m.factory = new pm::MetricFactoryDarwin();
118#endif
119
120 /* Let the sampler know it gets a valid collector. */
121 mMagic = MAGIC;
122
123 /* Start resource usage sampler */
124 int vrc = RTTimerCreate (&m.sampler, VBOX_USAGE_SAMPLER_MIN_INTERVAL,
125 &PerformanceCollector::staticSamplerCallback, this);
126 AssertMsgRC (vrc, ("Failed to create resource usage "
127 "sampling timer(%Rra)\n", vrc));
128 if (RT_FAILURE (vrc))
129 rc = E_FAIL;
130
131 if (SUCCEEDED (rc))
132 autoInitSpan.setSucceeded();
133
134 LogFlowThisFuncLeave();
135
136 return rc;
137}
138
139/**
140 * Uninitializes the PerformanceCollector object.
141 *
142 * Called either from FinalRelease() or by the parent when it gets destroyed.
143 */
144void PerformanceCollector::uninit()
145{
146 LogFlowThisFuncEnter();
147
148 /* Enclose the state transition Ready->InUninit->NotReady */
149 AutoUninitSpan autoUninitSpan (this);
150 if (autoUninitSpan.uninitDone())
151 {
152 LogFlowThisFunc (("Already uninitialized.\n"));
153 LogFlowThisFuncLeave();
154 return;
155 }
156
157 mMagic = 0;
158
159 /* Destroy resource usage sampler */
160 int vrc = RTTimerDestroy (m.sampler);
161 AssertMsgRC (vrc, ("Failed to destroy resource usage "
162 "sampling timer (%Rra)\n", vrc));
163 m.sampler = NULL;
164
165 delete m.factory;
166 m.factory = NULL;
167
168 LogFlowThisFuncLeave();
169}
170
171// IPerformanceCollector properties
172////////////////////////////////////////////////////////////////////////////////
173
174STDMETHODIMP
175PerformanceCollector::COMGETTER(MetricNames) (ComSafeArrayOut (BSTR, theMetricNames))
176{
177 if (ComSafeArrayOutIsNull (theMetricNames))
178 return E_POINTER;
179
180 AutoCaller autoCaller (this);
181 CheckComRCReturnRC (autoCaller.rc());
182
183 AutoReadLock alock (this);
184
185 com::SafeArray <BSTR> metricNames(RT_ELEMENTS(gMetricNames));
186 for (size_t i = 0; i < RT_ELEMENTS(gMetricNames); i++)
187 {
188 gMetricNames[i].detachTo(&metricNames[i]);
189 }
190 //gMetricNames.detachTo(ComSafeArrayOutArg (theMetricNames));
191 metricNames.detachTo (ComSafeArrayOutArg (theMetricNames));
192
193 return S_OK;
194}
195
196// IPerformanceCollector methods
197////////////////////////////////////////////////////////////////////////////////
198
199STDMETHODIMP
200PerformanceCollector::GetMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
201 ComSafeArrayIn (IUnknown *, objects),
202 ComSafeArrayOut (IPerformanceMetric *, outMetrics))
203{
204 //LogFlowThisFunc (("mState=%d, mType=%d\n", mState, mType));
205
206 HRESULT rc = S_OK;
207
208 AutoCaller autoCaller (this);
209 CheckComRCReturnRC (autoCaller.rc());
210
211 pm::Filter filter (ComSafeArrayInArg (metricNames),
212 ComSafeArrayInArg (objects));
213
214 AutoReadLock alock (this);
215
216 MetricList filteredMetrics;
217 MetricList::iterator it;
218 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
219 if (filter.match ((*it)->getObject(), (*it)->getName()))
220 filteredMetrics.push_back (*it);
221
222 com::SafeIfaceArray<IPerformanceMetric> retMetrics (filteredMetrics.size());
223 int i = 0;
224 for (it = filteredMetrics.begin(); it != filteredMetrics.end(); ++it)
225 {
226 ComObjPtr <PerformanceMetric> metric;
227 rc = metric.createObject();
228 if (SUCCEEDED (rc))
229 rc = metric->init (*it);
230 AssertComRCReturnRC (rc);
231 LogFlow (("PerformanceCollector::GetMetrics() store a metric at "
232 "retMetrics[%d]...\n", i));
233 metric.queryInterfaceTo (&retMetrics [++ i]);
234 }
235 retMetrics.detachTo (ComSafeArrayOutArg(outMetrics));
236 return rc;
237}
238
239STDMETHODIMP
240PerformanceCollector::SetupMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
241 ComSafeArrayIn (IUnknown *, objects),
242 ULONG aPeriod, ULONG aCount)
243{
244 AutoCaller autoCaller (this);
245 CheckComRCReturnRC (autoCaller.rc());
246
247 pm::Filter filter (ComSafeArrayInArg (metricNames),
248 ComSafeArrayInArg (objects));
249
250 AutoWriteLock alock (this);
251
252 BaseMetricList::iterator it;
253 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
254 if (filter.match((*it)->getObject(), (*it)->getName()))
255 {
256 (*it)->init(aPeriod, aCount);
257 (*it)->enable();
258 }
259
260 return S_OK;
261}
262
263STDMETHODIMP
264PerformanceCollector::EnableMetrics (ComSafeArrayIn (INPTR BSTR, metricNames),
265 ComSafeArrayIn (IUnknown *, objects))
266{
267 AutoCaller autoCaller (this);
268 CheckComRCReturnRC (autoCaller.rc());
269
270 pm::Filter filter (ComSafeArrayInArg (metricNames),
271 ComSafeArrayInArg (objects));
272
273 /// @todo (r=dmik) why read lock below? individual elements get modified!
274
275 AutoReadLock alock (this); /* Need a read lock to access mBaseMetrics */
276
277 BaseMetricList::iterator it;
278 for (it = m.baseMetrics.begin(); it != m.baseMetrics.end(); ++it)
279 if (filter.match((*it)->getObject(), (*it)->getName()))
280 (*it)->enable();
281
282 return S_OK;
283}
284
285STDMETHODIMP
286PerformanceCollector::DisableMetrics (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 /// @todo (r=dmik) why read lock below? individual elements get modified!
296
297 AutoReadLock alock (this);
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)->disable();
303
304 return S_OK;
305}
306
307STDMETHODIMP
308PerformanceCollector::QueryMetricsData (ComSafeArrayIn (INPTR BSTR, metricNames),
309 ComSafeArrayIn (IUnknown *, objects),
310 ComSafeArrayOut (BSTR, outMetricNames),
311 ComSafeArrayOut (IUnknown *, outObjects),
312 ComSafeArrayOut (ULONG, outDataIndices),
313 ComSafeArrayOut (ULONG, outDataLengths),
314 ComSafeArrayOut (LONG, outData))
315{
316 AutoCaller autoCaller (this);
317 CheckComRCReturnRC (autoCaller.rc());
318
319 AutoReadLock alock (this);
320
321 int i;
322 MetricList::const_iterator it;
323 /* Let's compute the size of the resulting flat array */
324 size_t flatSize = 0, numberOfMetrics = 0;
325 for (it = m.metrics.begin(); it != m.metrics.end(); ++it)
326 {
327 /* @todo Filtering goes here! */
328 flatSize += (*it)->getLength();
329 ++numberOfMetrics;
330 }
331 size_t flatIndex = 0;
332 com::SafeArray <BSTR> retNames (numberOfMetrics);
333 com::SafeIfaceArray <IUnknown> retObjects (numberOfMetrics);
334 com::SafeArray <ULONG> retIndices (numberOfMetrics);
335 com::SafeArray <ULONG> retLengths (numberOfMetrics);
336 com::SafeArray <LONG> retData (flatSize);
337
338 for (it = m.metrics.begin(), i = 0; it != m.metrics.end(); ++it)
339 {
340 /* @todo Filtering goes here! */
341 unsigned long *values, length;
342 /* @todo We may want to revise the query method to get rid of excessive alloc/memcpy calls. */
343 (*it)->query(&values, &length);
344 memcpy(retData.raw() + flatIndex, values, length * sizeof(*values));
345 Bstr tmp((*it)->getName());
346 tmp.detachTo(&retNames[i]);
347 (*it)->getObject().queryInterfaceTo (&retObjects[i]);
348 retLengths[i] = length;
349 retIndices[i] = flatIndex;
350 ++i;
351 flatIndex += length;
352 }
353
354 retNames.detachTo (ComSafeArrayOutArg (outMetricNames));
355 retObjects.detachTo (ComSafeArrayOutArg (outObjects));
356 retIndices.detachTo (ComSafeArrayOutArg (outDataIndices));
357 retLengths.detachTo (ComSafeArrayOutArg (outDataLengths));
358 retData.detachTo (ComSafeArrayOutArg (outData));
359 return S_OK;
360}
361
362// public methods for internal purposes
363///////////////////////////////////////////////////////////////////////////////
364
365void PerformanceCollector::registerBaseMetric (pm::BaseMetric *baseMetric)
366{
367 /// @todo (r=dmik) better always use AutoCaller unless you are 100% sure
368 /// the object is not uninitialized while you are doing something here
369
370 AutoWriteLock alock (this);
371 m.baseMetrics.push_back (baseMetric);
372}
373
374void PerformanceCollector::registerMetric (pm::Metric *metric)
375{
376 /// @todo (r=dmik) better always use AutoCaller unless you are 100% sure
377 /// the object is not uninitialized while you are doing something here
378
379 AutoWriteLock alock (this);
380 m.metrics.push_back (metric);
381}
382
383void PerformanceCollector::unregisterBaseMetricsFor (const ComPtr <IUnknown> &aObject)
384{
385 /// @todo (r=dmik) better always use AutoCaller unless you are 100% sure
386 /// the object is not uninitialized while you are doing something here
387
388 AutoWriteLock alock (this);
389 std::remove_if (m.baseMetrics.begin(), m.baseMetrics.end(),
390 std::bind2nd (std::mem_fun (&pm::BaseMetric::associatedWith),
391 aObject));
392}
393
394void PerformanceCollector::unregisterMetricsFor (const ComPtr <IUnknown> &aObject)
395{
396 /// @todo (r=dmik) better always use AutoCaller unless you are 100% sure
397 /// the object is not uninitialized while you are doing something here
398
399 AutoWriteLock alock (this);
400 std::remove_if (m.metrics.begin(), m.metrics.end(),
401 std::bind2nd (std::mem_fun (&pm::Metric::associatedWith),
402 aObject));
403}
404
405// private methods
406///////////////////////////////////////////////////////////////////////////////
407
408/* static */
409void PerformanceCollector::staticSamplerCallback (PRTTIMER pTimer, void *pvUser,
410 uint64_t iTick)
411{
412 AssertReturnVoid (pvUser != NULL);
413 PerformanceCollector *collector = static_cast <PerformanceCollector *> (pvUser);
414 Assert (collector->mMagic == MAGIC);
415 if (collector->mMagic == MAGIC)
416 {
417 collector->samplerCallback();
418 }
419}
420
421void PerformanceCollector::samplerCallback()
422{
423 AutoWriteLock alock (this);
424
425 uint64_t timestamp = RTTimeMilliTS();
426 std::for_each (m.baseMetrics.begin(), m.baseMetrics.end(),
427 std::bind2nd (std::mem_fun (&pm::BaseMetric::collectorBeat),
428 timestamp));
429}
430
431////////////////////////////////////////////////////////////////////////////////
432// PerformanceMetric class
433////////////////////////////////////////////////////////////////////////////////
434
435// constructor / destructor
436////////////////////////////////////////////////////////////////////////////////
437
438PerformanceMetric::PerformanceMetric() : mMetric(0)
439{
440}
441
442PerformanceMetric::~PerformanceMetric()
443{
444}
445
446HRESULT PerformanceMetric::FinalConstruct()
447{
448 LogFlowThisFunc (("\n"));
449
450 return S_OK;
451}
452
453void PerformanceMetric::FinalRelease()
454{
455 LogFlowThisFunc (("\n"));
456
457 uninit ();
458}
459
460// public initializer/uninitializer for internal purposes only
461////////////////////////////////////////////////////////////////////////////////
462
463HRESULT PerformanceMetric::init (pm::Metric *aMetric)
464{
465 mMetric = aMetric;
466 return S_OK;
467}
468
469void PerformanceMetric::uninit()
470{
471}
472
473STDMETHODIMP PerformanceMetric::COMGETTER(MetricName) (BSTR *aMetricName)
474{
475 /// @todo (r=dmik) why do all these getters not do AutoCaller and
476 /// AutoReadLock? Is the underlying metric a constant object?
477
478 Bstr tmp (mMetric->getName());
479 tmp.detachTo (aMetricName);
480 return S_OK;
481}
482
483STDMETHODIMP PerformanceMetric::COMGETTER(Object) (IUnknown **anObject)
484{
485 *anObject = mMetric->getObject();
486 return S_OK;
487}
488
489STDMETHODIMP PerformanceMetric::COMGETTER(Period) (ULONG *aPeriod)
490{
491 *aPeriod = mMetric->getPeriod();
492 return S_OK;
493}
494
495STDMETHODIMP PerformanceMetric::COMGETTER(Count) (ULONG *aCount)
496{
497 *aCount = mMetric->getLength();
498 return S_OK;
499}
500
501STDMETHODIMP PerformanceMetric::COMGETTER(Unit) (BSTR *aUnit)
502{
503 Bstr tmp (mMetric->getUnit());
504 tmp.detachTo(aUnit);
505 return S_OK;
506}
507
508STDMETHODIMP PerformanceMetric::COMGETTER(MinimumValue) (LONG *aMinValue)
509{
510 *aMinValue = mMetric->getMinValue();
511 return S_OK;
512}
513
514STDMETHODIMP PerformanceMetric::COMGETTER(MaximumValue) (LONG *aMaxValue)
515{
516 *aMaxValue = mMetric->getMaxValue();
517 return S_OK;
518}
519
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