mirror of
https://github.com/chibicitiberiu/rainmeter-studio.git
synced 2024-02-24 04:33:31 +00:00
175 lines
5.0 KiB
C++
175 lines
5.0 KiB
C++
/*
|
|
Copyright (C) 2001 Kimmo Pekkola
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#pragma warning(disable: 4996)
|
|
|
|
#include "MeasureCPU.h"
|
|
#include "Rainmeter.h"
|
|
|
|
#define SystemBasicInformation 0
|
|
#define SystemPerformanceInformation 2
|
|
#define SystemTimeInformation 3
|
|
|
|
#define Li2Double(x) ((double)((x).HighPart) * 4.294967296E9 + (double)((x).LowPart))
|
|
|
|
// ntdll!NtQuerySystemInformation (NT specific!)
|
|
//
|
|
// The function copies the system information of the
|
|
// specified type into a buffer
|
|
//
|
|
// NTSYSAPI
|
|
// NTSTATUS
|
|
// NTAPI
|
|
// NtQuerySystemInformation(
|
|
// IN UINT SystemInformationClass, // information type
|
|
// OUT PVOID SystemInformation, // pointer to buffer
|
|
// IN ULONG SystemInformationLength, // buffer size in bytes
|
|
// OUT PULONG ReturnLength OPTIONAL // pointer to a 32-bit
|
|
// // variable that receives
|
|
// // the number of bytes
|
|
// // written to the buffer
|
|
// );
|
|
|
|
/*
|
|
** CMeasureCPU
|
|
**
|
|
** The constructor
|
|
**
|
|
*/
|
|
CMeasureCPU::CMeasureCPU(CMeterWindow* meterWindow) : CMeasure(meterWindow)
|
|
{
|
|
m_CPUFromRegistry = false;
|
|
m_MaxValue = 100.0;
|
|
m_MinValue = 0.0;
|
|
m_FirstTime = true;
|
|
|
|
m_NtQuerySystemInformation = NULL;
|
|
m_OldIdleTime.QuadPart = 0;
|
|
m_OldSystemTime.QuadPart = 0;
|
|
|
|
m_NtQuerySystemInformation = (PROCNTQSI)GetProcAddress(
|
|
GetModuleHandle(L"ntdll"),
|
|
"NtQuerySystemInformation"
|
|
);
|
|
|
|
GetSystemInfo(&m_SystemInfo);
|
|
}
|
|
|
|
/*
|
|
** ~CMeasureCPU
|
|
**
|
|
** The destructor
|
|
**
|
|
*/
|
|
CMeasureCPU::~CMeasureCPU()
|
|
{
|
|
if(m_CPUFromRegistry)
|
|
{
|
|
// Stop the counter if it was started
|
|
HKEY hkey;
|
|
DWORD dwDataSize;
|
|
DWORD dwType;
|
|
DWORD dwDummy;
|
|
|
|
RegOpenKeyEx(HKEY_DYN_DATA, L"PerfStats\\StopStat", 0, KEY_ALL_ACCESS, &hkey);
|
|
dwDataSize = sizeof(dwDummy);
|
|
RegQueryValueEx(hkey, L"KERNEL\\CPUUsage", NULL, &dwType, (LPBYTE)&dwDummy, &dwDataSize);
|
|
RegCloseKey(hkey);
|
|
}
|
|
}
|
|
|
|
/*
|
|
** Update
|
|
**
|
|
** Updates the current CPU utilization value. On NT the value is taken
|
|
** from the performance counters and on 9x we'll use the registry.
|
|
**
|
|
*/
|
|
bool CMeasureCPU::Update()
|
|
{
|
|
if (!CMeasure::PreUpdate()) return false;
|
|
|
|
if (CRainmeter::IsNT() != PLATFORM_9X)
|
|
{
|
|
if (m_NtQuerySystemInformation)
|
|
{
|
|
// This code is 'borrowed' from http://www.codepile.com/tric21.shtml
|
|
double dbIdleTime;
|
|
double dbSystemTime;
|
|
LONG status;
|
|
|
|
// get new system time
|
|
status = m_NtQuerySystemInformation(SystemTimeInformation, &m_SysTimeInfo, sizeof(m_SysTimeInfo), 0);
|
|
if (status != NO_ERROR) return false;
|
|
|
|
// get new CPU's idle time
|
|
status = m_NtQuerySystemInformation(SystemPerformanceInformation, &m_SysPerfInfo, sizeof(m_SysPerfInfo),NULL);
|
|
if (status != NO_ERROR) return false;
|
|
|
|
// if it's a first call - skip it
|
|
if(!m_FirstTime)
|
|
{
|
|
// CurrentValue = NewValue - OldValue
|
|
dbIdleTime = Li2Double(m_SysPerfInfo.liIdleTime) - Li2Double(m_OldIdleTime);
|
|
dbSystemTime = Li2Double(m_SysTimeInfo.liKeSystemTime) - Li2Double(m_OldSystemTime);
|
|
|
|
// CurrentCpuIdle = IdleTime / SystemTime
|
|
dbIdleTime = dbIdleTime / dbSystemTime;
|
|
|
|
// CurrentCpuUsage% = 100 - (CurrentCpuIdle * 100) / NumberOfProcessors
|
|
dbIdleTime = 100.0 - dbIdleTime * 100.0 / (double)m_SystemInfo.dwNumberOfProcessors + 0.5;
|
|
|
|
m_Value = min(dbIdleTime, 100.0);
|
|
m_Value = max(m_Value, 0.0);
|
|
}
|
|
|
|
// store new CPU's idle and system time
|
|
m_OldIdleTime = m_SysPerfInfo.liIdleTime;
|
|
m_OldSystemTime = m_SysTimeInfo.liKeSystemTime;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// It's a wintendo!
|
|
HKEY hkey;
|
|
DWORD dwDataSize;
|
|
DWORD dwType;
|
|
DWORD dwCpuUsage;
|
|
|
|
if(m_FirstTime)
|
|
{
|
|
RegOpenKeyEx(HKEY_DYN_DATA, L"PerfStats\\StartStat", 0, KEY_ALL_ACCESS, &hkey);
|
|
dwDataSize = sizeof(dwCpuUsage);
|
|
RegQueryValueEx(hkey, L"KERNEL\\CPUUsage", NULL, &dwType, (LPBYTE)&dwCpuUsage, &dwDataSize);
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
RegOpenKeyEx(HKEY_DYN_DATA, L"PerfStats\\StatData", 0, KEY_ALL_ACCESS, &hkey);
|
|
dwDataSize = sizeof(dwCpuUsage);
|
|
RegQueryValueEx(hkey, L"KERNEL\\CPUUsage", NULL, &dwType, (LPBYTE)&dwCpuUsage, &dwDataSize);
|
|
RegCloseKey(hkey);
|
|
|
|
m_Value = dwCpuUsage;
|
|
m_CPUFromRegistry = true;
|
|
}
|
|
|
|
m_FirstTime = false;
|
|
|
|
return PostUpdate();
|
|
}
|