/* $Id: Performance.h 43629 2012-10-12 09:26:07Z vboxsync $ */ /** @file * VirtualBox Main - Performance Classes declaration. */ /* * Copyright (C) 2008 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ #ifndef ___performance_h #define ___performance_h #include #include #include #include #include #include #include #include #include /* For std::fun_ptr in testcase */ #include #include #include /* Forward decl. */ class Machine; namespace pm { /* CPU load is measured in 1/1000 of per cent. */ const uint64_t PM_CPU_LOAD_MULTIPLIER = UINT64_C(100000); /* Network load is measured in 1/1000 of per cent. */ const uint64_t PM_NETWORK_LOAD_MULTIPLIER = UINT64_C(100000); /* Disk load is measured in 1/1000 of per cent. */ const uint64_t PM_DISK_LOAD_MULTIPLIER = UINT64_C(100000); /* Sampler precision in milliseconds. */ const uint64_t PM_SAMPLER_PRECISION_MS = 50; /* Sub Metrics **********************************************************/ class CircularBuffer { public: CircularBuffer() : mData(0), mLength(0), mEnd(0), mWrapped(false) {}; ~CircularBuffer() { if (mData) RTMemFree(mData); }; void init(ULONG length); ULONG length(); ULONG getSequenceNumber() { return mSequenceNumber; } void put(ULONG value); void copyTo(ULONG *data); private: ULONG *mData; ULONG mLength; ULONG mEnd; ULONG mSequenceNumber; bool mWrapped; }; class SubMetric : public CircularBuffer { public: SubMetric(com::Utf8Str name, const char *description) : mName(name), mDescription(description) {}; void query(ULONG *data); const char *getName() { return mName.c_str(); }; const char *getDescription() { return mDescription; }; private: const com::Utf8Str mName; const char *mDescription; }; enum { COLLECT_NONE = 0x0, COLLECT_CPU_LOAD = 0x1, COLLECT_RAM_USAGE = 0x2, COLLECT_GUEST_STATS = 0x4 }; typedef int HintFlags; typedef std::pair ProcessFlagsPair; class CollectorHints { public: typedef std::list ProcessList; CollectorHints() : mHostFlags(COLLECT_NONE) {} void collectHostCpuLoad() { mHostFlags |= COLLECT_CPU_LOAD; } void collectHostRamUsage() { mHostFlags |= COLLECT_RAM_USAGE; } void collectHostRamVmm() { mHostFlags |= COLLECT_GUEST_STATS; } void collectProcessCpuLoad(RTPROCESS process) { findProcess(process).second |= COLLECT_CPU_LOAD; } void collectProcessRamUsage(RTPROCESS process) { findProcess(process).second |= COLLECT_RAM_USAGE; } void collectGuestStats(RTPROCESS process) { findProcess(process).second |= COLLECT_GUEST_STATS; } bool isHostCpuLoadCollected() const { return (mHostFlags & COLLECT_CPU_LOAD) != 0; } bool isHostRamUsageCollected() const { return (mHostFlags & COLLECT_RAM_USAGE) != 0; } bool isHostRamVmmCollected() const { return (mHostFlags & COLLECT_GUEST_STATS) != 0; } bool isProcessCpuLoadCollected(RTPROCESS process) { return (findProcess(process).second & COLLECT_CPU_LOAD) != 0; } bool isProcessRamUsageCollected(RTPROCESS process) { return (findProcess(process).second & COLLECT_RAM_USAGE) != 0; } bool isGuestStatsCollected(RTPROCESS process) { return (findProcess(process).second & COLLECT_GUEST_STATS) != 0; } void getProcesses(std::vector& processes) const { processes.clear(); processes.reserve(mProcesses.size()); for (ProcessList::const_iterator it = mProcesses.begin(); it != mProcesses.end(); it++) processes.push_back(it->first); } const ProcessList& getProcessFlags() const { return mProcesses; } private: HintFlags mHostFlags; ProcessList mProcesses; ProcessFlagsPair& findProcess(RTPROCESS process) { ProcessList::iterator it; for (it = mProcesses.begin(); it != mProcesses.end(); it++) if (it->first == process) return *it; /* Not found -- add new */ mProcesses.push_back(ProcessFlagsPair(process, COLLECT_NONE)); return mProcesses.back(); } }; /* Guest Collector Classes *********************************/ /* * WARNING! The bits in the following masks must correspond to parameters * of CollectorGuest::updateStats(). */ typedef enum { GUESTSTATMASK_NONE = 0x00000000, GUESTSTATMASK_CPUUSER = 0x00000001, GUESTSTATMASK_CPUKERNEL = 0x00000002, GUESTSTATMASK_CPUIDLE = 0x00000004, GUESTSTATMASK_MEMTOTAL = 0x00000008, GUESTSTATMASK_MEMFREE = 0x00000010, GUESTSTATMASK_MEMBALLOON = 0x00000020, GUESTSTATMASK_MEMSHARED = 0x00000040, GUESTSTATMASK_MEMCACHE = 0x00000080, GUESTSTATMASK_PAGETOTAL = 0x00000100, GUESTSTATMASK_ALLOCVMM = 0x00000200, GUESTSTATMASK_FREEVMM = 0x00000400, GUESTSTATMASK_BALOONVMM = 0x00000800, GUESTSTATMASK_SHAREDVMM = 0x00001000 } GUESTSTATMASK; const ULONG GUESTSTATS_CPULOAD = GUESTSTATMASK_CPUUSER|GUESTSTATMASK_CPUKERNEL|GUESTSTATMASK_CPUIDLE; const ULONG GUESTSTATS_RAMUSAGE = GUESTSTATMASK_MEMTOTAL|GUESTSTATMASK_MEMFREE|GUESTSTATMASK_MEMBALLOON| GUESTSTATMASK_MEMSHARED|GUESTSTATMASK_MEMCACHE| GUESTSTATMASK_PAGETOTAL; const ULONG GUESTSTATS_VMMRAM = GUESTSTATMASK_ALLOCVMM|GUESTSTATMASK_FREEVMM| GUESTSTATMASK_BALOONVMM|GUESTSTATMASK_SHAREDVMM; const ULONG GUESTSTATS_ALL = GUESTSTATS_CPULOAD|GUESTSTATS_RAMUSAGE|GUESTSTATS_VMMRAM; class CollectorGuest; class CollectorGuestRequest { public: CollectorGuestRequest() : mCGuest(0) {}; virtual ~CollectorGuestRequest() {}; void setGuest(CollectorGuest *aGuest) { mCGuest = aGuest; }; CollectorGuest *getGuest() { return mCGuest; }; virtual int execute() = 0; virtual void debugPrint(void *aObject, const char *aFunction, const char *aText) = 0; protected: CollectorGuest *mCGuest; const char *mDebugName; }; class CGRQEnable : public CollectorGuestRequest { public: CGRQEnable(ULONG aMask) : mMask(aMask) {}; int execute(); void debugPrint(void *aObject, const char *aFunction, const char *aText); private: ULONG mMask; }; class CGRQDisable : public CollectorGuestRequest { public: CGRQDisable(ULONG aMask) : mMask(aMask) {}; int execute(); void debugPrint(void *aObject, const char *aFunction, const char *aText); private: ULONG mMask; }; class CGRQAbort : public CollectorGuestRequest { public: CGRQAbort() {}; int execute(); void debugPrint(void *aObject, const char *aFunction, const char *aText); }; class CollectorGuestQueue { public: CollectorGuestQueue(); ~CollectorGuestQueue(); void push(CollectorGuestRequest* rq); CollectorGuestRequest* pop(); private: RTCLockMtx mLockMtx; RTSEMEVENT mEvent; std::queue mQueue; }; class CollectorGuestManager; class CollectorGuest { public: CollectorGuest(Machine *machine, RTPROCESS process); ~CollectorGuest(); void setManager(CollectorGuestManager *aManager) { mManager = aManager; }; bool isUnregistered() { return mUnregistered; }; bool isEnabled() { return mEnabled != 0; }; bool isValid(ULONG mask) { return (mValid & mask) == mask; }; void invalidate(ULONG mask) { mValid &= ~mask; }; void unregister() { mUnregistered = true; }; void updateStats(ULONG aValidStats, ULONG aCpuUser, ULONG aCpuKernel, ULONG aCpuIdle, ULONG aMemTotal, ULONG aMemFree, ULONG aMemBalloon, ULONG aMemShared, ULONG aMemCache, ULONG aPageTotal, ULONG aAllocVMM, ULONG aFreeVMM, ULONG aBalloonedVMM, ULONG aSharedVMM); int enable(ULONG mask); int disable(ULONG mask); int enqueueRequest(CollectorGuestRequest *aRequest); int enableInternal(ULONG mask); int disableInternal(ULONG mask); const com::Utf8Str& getVMName() const { return mMachineName; }; RTPROCESS getProcess() { return mProcess; }; ULONG getCpuUser() { return mCpuUser; }; ULONG getCpuKernel() { return mCpuKernel; }; ULONG getCpuIdle() { return mCpuIdle; }; ULONG getMemTotal() { return mMemTotal; }; ULONG getMemFree() { return mMemFree; }; ULONG getMemBalloon() { return mMemBalloon; }; ULONG getMemShared() { return mMemShared; }; ULONG getMemCache() { return mMemCache; }; ULONG getPageTotal() { return mPageTotal; }; ULONG getAllocVMM() { return mAllocVMM; }; ULONG getFreeVMM() { return mFreeVMM; }; ULONG getBalloonedVMM() { return mBalloonedVMM; }; ULONG getSharedVMM() { return mSharedVMM; }; private: int enableVMMStats(bool mCollectVMMStats); CollectorGuestManager *mManager; bool mUnregistered; ULONG mEnabled; ULONG mValid; Machine *mMachine; com::Utf8Str mMachineName; RTPROCESS mProcess; ComPtr mConsole; ComPtr mGuest; ULONG mCpuUser; ULONG mCpuKernel; ULONG mCpuIdle; ULONG mMemTotal; ULONG mMemFree; ULONG mMemBalloon; ULONG mMemShared; ULONG mMemCache; ULONG mPageTotal; ULONG mAllocVMM; ULONG mFreeVMM; ULONG mBalloonedVMM; ULONG mSharedVMM; }; typedef std::list CollectorGuestList; class CollectorGuestManager { public: CollectorGuestManager(); ~CollectorGuestManager(); void registerGuest(CollectorGuest* pGuest); void unregisterGuest(CollectorGuest* pGuest); CollectorGuest *getVMMStatsProvider() { return mVMMStatsProvider; }; void preCollect(CollectorHints& hints, uint64_t iTick); void destroyUnregistered(); int enqueueRequest(CollectorGuestRequest *aRequest); CollectorGuest *getBlockedGuest() { return mGuestBeingCalled; }; static DECLCALLBACK(int) requestProcessingThread(RTTHREAD aThread, void *pvUser); private: RTTHREAD mThread; CollectorGuestList mGuests; CollectorGuest *mVMMStatsProvider; CollectorGuestQueue mQueue; CollectorGuest *mGuestBeingCalled; }; /* Collector Hardware Abstraction Layer *********************************/ class CollectorHAL { public: CollectorHAL() {}; virtual ~CollectorHAL() { }; virtual int preCollect(const CollectorHints& /* hints */, uint64_t /* iTick */) { return VINF_SUCCESS; } /** Returns averaged CPU usage in 1/1000th per cent across all host's CPUs. */ virtual int getHostCpuLoad(ULONG *user, ULONG *kernel, ULONG *idle); /** Returns the average frequency in MHz across all host's CPUs. */ virtual int getHostCpuMHz(ULONG *mhz); /** Returns the amount of physical memory in kilobytes. */ virtual int getHostMemoryUsage(ULONG *total, ULONG *used, ULONG *available); /** Returns file system counters in megabytes. */ virtual int getHostFilesystemUsage(const char *name, ULONG *total, ULONG *used, ULONG *available); /** Returns CPU usage in 1/1000th per cent by a particular process. */ virtual int getProcessCpuLoad(RTPROCESS process, ULONG *user, ULONG *kernel); /** Returns the amount of memory used by a process in kilobytes. */ virtual int getProcessMemoryUsage(RTPROCESS process, ULONG *used); /** Returns CPU usage counters in platform-specific units. */ virtual int getRawHostCpuLoad(uint64_t *user, uint64_t *kernel, uint64_t *idle); /** Returns received and transmitted bytes. */ virtual int getRawHostNetworkLoad(const char *name, uint64_t *rx, uint64_t *tx); /** Returns disk usage counters in platform-specific units. */ virtual int getRawHostDiskLoad(const char *name, uint64_t *disk_ms, uint64_t *total_ms); /** Returns process' CPU usage counter in platform-specific units. */ virtual int getRawProcessCpuLoad(RTPROCESS process, uint64_t *user, uint64_t *kernel, uint64_t *total); }; extern CollectorHAL *createHAL(); typedef std::list DiskList; extern int getDiskListByFs(const char *name, DiskList& list); /* Base Metrics *********************************************************/ class BaseMetric { public: BaseMetric(CollectorHAL *hal, const com::Utf8Str name, ComPtr object) : mPeriod(0), mLength(0), mHAL(hal), mName(name), mObject(object), mLastSampleTaken(0), mEnabled(false), mUnregistered(false) {}; virtual ~BaseMetric() {}; virtual void init(ULONG period, ULONG length) = 0; virtual void preCollect(CollectorHints& hints, uint64_t iTick) = 0; virtual void collect() = 0; virtual const char *getUnit() = 0; virtual ULONG getMinValue() = 0; virtual ULONG getMaxValue() = 0; virtual ULONG getScale() = 0; bool collectorBeat(uint64_t nowAt); virtual int enable() { mEnabled = true; return S_OK; }; virtual int disable() { mEnabled = false; return S_OK; }; void unregister() { mUnregistered = true; }; bool isUnregistered() { return mUnregistered; }; bool isEnabled() { return mEnabled; }; ULONG getPeriod() { return mPeriod; }; ULONG getLength() { return mLength; }; const char *getName() { return mName.c_str(); }; ComPtr getObject() { return mObject; }; bool associatedWith(ComPtr object) { return mObject == object; }; protected: ULONG mPeriod; ULONG mLength; CollectorHAL *mHAL; const com::Utf8Str mName; ComPtr mObject; uint64_t mLastSampleTaken; bool mEnabled; bool mUnregistered; }; class BaseGuestMetric : public BaseMetric { public: BaseGuestMetric(CollectorGuest *cguest, const char *name, ComPtr object) : BaseMetric(NULL, name, object), mCGuest(cguest) {}; protected: CollectorGuest *mCGuest; }; class HostCpuLoad : public BaseMetric { public: HostCpuLoad(CollectorHAL *hal, ComPtr object, SubMetric *user, SubMetric *kernel, SubMetric *idle) : BaseMetric(hal, "CPU/Load", object), mUser(user), mKernel(kernel), mIdle(idle) {}; ~HostCpuLoad() { delete mUser; delete mKernel; delete mIdle; }; void init(ULONG period, ULONG length); void collect(); const char *getUnit() { return "%"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return PM_CPU_LOAD_MULTIPLIER; }; ULONG getScale() { return PM_CPU_LOAD_MULTIPLIER / 100; } protected: SubMetric *mUser; SubMetric *mKernel; SubMetric *mIdle; }; class HostCpuLoadRaw : public HostCpuLoad { public: HostCpuLoadRaw(CollectorHAL *hal, ComPtr object, SubMetric *user, SubMetric *kernel, SubMetric *idle) : HostCpuLoad(hal, object, user, kernel, idle), mUserPrev(0), mKernelPrev(0), mIdlePrev(0) {}; void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); private: uint64_t mUserPrev; uint64_t mKernelPrev; uint64_t mIdlePrev; }; class HostCpuMhz : public BaseMetric { public: HostCpuMhz(CollectorHAL *hal, ComPtr object, SubMetric *mhz) : BaseMetric(hal, "CPU/MHz", object), mMHz(mhz) {}; ~HostCpuMhz() { delete mMHz; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& /* hints */, uint64_t /* iTick */) {} void collect(); const char *getUnit() { return "MHz"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return INT32_MAX; }; ULONG getScale() { return 1; } private: SubMetric *mMHz; }; class HostRamUsage : public BaseMetric { public: HostRamUsage(CollectorHAL *hal, ComPtr object, SubMetric *total, SubMetric *used, SubMetric *available) : BaseMetric(hal, "RAM/Usage", object), mTotal(total), mUsed(used), mAvailable(available) {}; ~HostRamUsage() { delete mTotal; delete mUsed; delete mAvailable; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); const char *getUnit() { return "kB"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return INT32_MAX; }; ULONG getScale() { return 1; } private: SubMetric *mTotal; SubMetric *mUsed; SubMetric *mAvailable; }; class HostNetworkLoadRaw : public BaseMetric { public: HostNetworkLoadRaw(CollectorHAL *hal, ComPtr object, com::Utf8Str name, com::Utf8Str shortname, com::Utf8Str ifname, uint32_t speed, SubMetric *rx, SubMetric *tx) : BaseMetric(hal, name, object), mShortName(shortname), mInterfaceName(ifname), mRx(rx), mTx(tx), mRxPrev(0), mTxPrev(0), mRc(VINF_SUCCESS) { mSpeed = (uint64_t)speed * (1000000/8); /* Convert to bytes/sec */ }; ~HostNetworkLoadRaw() { delete mRx; delete mTx; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); const char *getUnit() { return "%"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return PM_NETWORK_LOAD_MULTIPLIER; }; ULONG getScale() { return PM_NETWORK_LOAD_MULTIPLIER / 100; } private: com::Utf8Str mShortName; com::Utf8Str mInterfaceName; SubMetric *mRx; SubMetric *mTx; uint64_t mRxPrev; uint64_t mTxPrev; uint64_t mSpeed; int mRc; }; class HostFilesystemUsage : public BaseMetric { public: HostFilesystemUsage(CollectorHAL *hal, ComPtr object, com::Utf8Str name, com::Utf8Str fsname, SubMetric *total, SubMetric *used, SubMetric *available) : BaseMetric(hal, name, object), mFsName(fsname), mTotal(total), mUsed(used), mAvailable(available) {}; ~HostFilesystemUsage() { delete mTotal; delete mUsed; delete mAvailable; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); const char *getUnit() { return "mB"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return INT32_MAX; }; ULONG getScale() { return 1; } private: com::Utf8Str mFsName; SubMetric *mTotal; SubMetric *mUsed; SubMetric *mAvailable; }; class HostDiskLoadRaw : public BaseMetric { public: HostDiskLoadRaw(CollectorHAL *hal, ComPtr object, com::Utf8Str name, com::Utf8Str diskname, SubMetric *util) : BaseMetric(hal, name, object), mDiskName(diskname), mUtil(util), mDiskPrev(0), mTotalPrev(0) {}; ~HostDiskLoadRaw() { delete mUtil; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); const char *getUnit() { return "%"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return PM_DISK_LOAD_MULTIPLIER; }; ULONG getScale() { return PM_DISK_LOAD_MULTIPLIER / 100; } private: com::Utf8Str mDiskName; SubMetric *mUtil; uint64_t mDiskPrev; uint64_t mTotalPrev; }; #ifndef VBOX_COLLECTOR_TEST_CASE class HostRamVmm : public BaseMetric { public: HostRamVmm(CollectorGuestManager *gm, ComPtr object, SubMetric *allocVMM, SubMetric *freeVMM, SubMetric *balloonVMM, SubMetric *sharedVMM) : BaseMetric(NULL, "RAM/VMM", object), mCollectorGuestManager(gm), mAllocVMM(allocVMM), mFreeVMM(freeVMM), mBalloonVMM(balloonVMM), mSharedVMM(sharedVMM), mAllocCurrent(0), mFreeCurrent(0), mBalloonedCurrent(0), mSharedCurrent(0) {}; ~HostRamVmm() { delete mAllocVMM; delete mFreeVMM; delete mBalloonVMM; delete mSharedVMM; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); int enable(); int disable(); const char *getUnit() { return "kB"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return INT32_MAX; }; ULONG getScale() { return 1; } private: CollectorGuestManager *mCollectorGuestManager; SubMetric *mAllocVMM; SubMetric *mFreeVMM; SubMetric *mBalloonVMM; SubMetric *mSharedVMM; ULONG mAllocCurrent; ULONG mFreeCurrent; ULONG mBalloonedCurrent; ULONG mSharedCurrent; }; #endif /* VBOX_COLLECTOR_TEST_CASE */ class MachineCpuLoad : public BaseMetric { public: MachineCpuLoad(CollectorHAL *hal, ComPtr object, RTPROCESS process, SubMetric *user, SubMetric *kernel) : BaseMetric(hal, "CPU/Load", object), mProcess(process), mUser(user), mKernel(kernel) {}; ~MachineCpuLoad() { delete mUser; delete mKernel; }; void init(ULONG period, ULONG length); void collect(); const char *getUnit() { return "%"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return PM_CPU_LOAD_MULTIPLIER; }; ULONG getScale() { return PM_CPU_LOAD_MULTIPLIER / 100; } protected: RTPROCESS mProcess; SubMetric *mUser; SubMetric *mKernel; }; class MachineCpuLoadRaw : public MachineCpuLoad { public: MachineCpuLoadRaw(CollectorHAL *hal, ComPtr object, RTPROCESS process, SubMetric *user, SubMetric *kernel) : MachineCpuLoad(hal, object, process, user, kernel), mHostTotalPrev(0), mProcessUserPrev(0), mProcessKernelPrev(0) {}; void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); private: uint64_t mHostTotalPrev; uint64_t mProcessUserPrev; uint64_t mProcessKernelPrev; }; class MachineRamUsage : public BaseMetric { public: MachineRamUsage(CollectorHAL *hal, ComPtr object, RTPROCESS process, SubMetric *used) : BaseMetric(hal, "RAM/Usage", object), mProcess(process), mUsed(used) {}; ~MachineRamUsage() { delete mUsed; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); const char *getUnit() { return "kB"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return INT32_MAX; }; ULONG getScale() { return 1; } private: RTPROCESS mProcess; SubMetric *mUsed; }; #ifndef VBOX_COLLECTOR_TEST_CASE class GuestCpuLoad : public BaseGuestMetric { public: GuestCpuLoad(CollectorGuest *cguest, ComPtr object, SubMetric *user, SubMetric *kernel, SubMetric *idle) : BaseGuestMetric(cguest, "Guest/CPU/Load", object), mUser(user), mKernel(kernel), mIdle(idle) {}; ~GuestCpuLoad() { delete mUser; delete mKernel; delete mIdle; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); int enable(); int disable(); const char *getUnit() { return "%"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return PM_CPU_LOAD_MULTIPLIER; }; ULONG getScale() { return PM_CPU_LOAD_MULTIPLIER / 100; } protected: SubMetric *mUser; SubMetric *mKernel; SubMetric *mIdle; }; class GuestRamUsage : public BaseGuestMetric { public: GuestRamUsage(CollectorGuest *cguest, ComPtr object, SubMetric *total, SubMetric *free, SubMetric *balloon, SubMetric *shared, SubMetric *cache, SubMetric *pagedtotal) : BaseGuestMetric(cguest, "Guest/RAM/Usage", object), mTotal(total), mFree(free), mBallooned(balloon), mCache(cache), mPagedTotal(pagedtotal), mShared(shared) {}; ~GuestRamUsage() { delete mTotal; delete mFree; delete mBallooned; delete mShared; delete mCache; delete mPagedTotal; }; void init(ULONG period, ULONG length); void preCollect(CollectorHints& hints, uint64_t iTick); void collect(); int enable(); int disable(); const char *getUnit() { return "kB"; }; ULONG getMinValue() { return 0; }; ULONG getMaxValue() { return INT32_MAX; }; ULONG getScale() { return 1; } private: SubMetric *mTotal, *mFree, *mBallooned, *mCache, *mPagedTotal, *mShared; }; #endif /* VBOX_COLLECTOR_TEST_CASE */ /* Aggregate Functions **************************************************/ class Aggregate { public: virtual ULONG compute(ULONG *data, ULONG length) = 0; virtual const char *getName() = 0; }; class AggregateAvg : public Aggregate { public: virtual ULONG compute(ULONG *data, ULONG length); virtual const char *getName(); }; class AggregateMin : public Aggregate { public: virtual ULONG compute(ULONG *data, ULONG length); virtual const char *getName(); }; class AggregateMax : public Aggregate { public: virtual ULONG compute(ULONG *data, ULONG length); virtual const char *getName(); }; /* Metric Class *********************************************************/ class Metric { public: Metric(BaseMetric *baseMetric, SubMetric *subMetric, Aggregate *aggregate) : mName(subMetric->getName()), mBaseMetric(baseMetric), mSubMetric(subMetric), mAggregate(aggregate) { if (mAggregate) { mName.append(":"); mName.append(mAggregate->getName()); } } ~Metric() { delete mAggregate; } bool associatedWith(ComPtr object) { return getObject() == object; }; const char *getName() { return mName.c_str(); }; ComPtr getObject() { return mBaseMetric->getObject(); }; const char *getDescription() { return mAggregate ? "" : mSubMetric->getDescription(); }; const char *getUnit() { return mBaseMetric->getUnit(); }; ULONG getMinValue() { return mBaseMetric->getMinValue(); }; ULONG getMaxValue() { return mBaseMetric->getMaxValue(); }; ULONG getPeriod() { return mBaseMetric->getPeriod(); }; ULONG getLength() { return mAggregate ? 1 : mBaseMetric->getLength(); }; ULONG getScale() { return mBaseMetric->getScale(); } void query(ULONG **data, ULONG *count, ULONG *sequenceNumber); private: RTCString mName; BaseMetric *mBaseMetric; SubMetric *mSubMetric; Aggregate *mAggregate; }; /* Filter Class *********************************************************/ class Filter { public: Filter(ComSafeArrayIn(IN_BSTR, metricNames), ComSafeArrayIn(IUnknown * , objects)); Filter(const com::Utf8Str name, const ComPtr &aObject); static bool patternMatch(const char *pszPat, const char *pszName, bool fSeenColon = false); bool match(const ComPtr object, const RTCString &name) const; private: void init(ComSafeArrayIn(IN_BSTR, metricNames), ComSafeArrayIn(IUnknown * , objects)); typedef std::pair, const RTCString> FilterElement; typedef std::list ElementList; ElementList mElements; void processMetricList(const com::Utf8Str &name, const ComPtr object); }; } #endif /* ___performance_h */ /* vi: set tabstop=4 shiftwidth=4 expandtab: */