2009-02-10 18:37:48 +00:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2009-10-07 16:45:14 +00:00
|
|
|
#include "StdAfx.h"
|
2009-02-10 18:37:48 +00:00
|
|
|
#include "MeasureCPU.h"
|
|
|
|
#include "Rainmeter.h"
|
2010-09-11 19:39:45 +00:00
|
|
|
#include "System.h"
|
2010-02-13 03:07:34 +00:00
|
|
|
#include "Error.h"
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2010-02-13 03:07:34 +00:00
|
|
|
#define STATUS_SUCCESS 0
|
|
|
|
#define STATUS_INFO_LENGTH_MISMATCH 0xC0000004
|
|
|
|
|
|
|
|
#define SystemProcessorPerformanceInformation 8
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
//#define Li2Double(x) ((double)((x).HighPart) * 4.294967296E9 + (double)((x).LowPart))
|
|
|
|
#define Li2Double(x) ((double)((x).QuadPart))
|
2010-02-13 03:07:34 +00:00
|
|
|
#define Ft2Double(x) ((double)((x).dwHighDateTime) * 4.294967296E9 + (double)((x).dwLowDateTime))
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
PROCNTQSI CMeasureCPU::c_NtQuerySystemInformation = NULL;
|
|
|
|
PROCGST CMeasureCPU::c_GetSystemTimes = NULL;
|
|
|
|
int CMeasureCPU::c_NumOfProcessors = 0;
|
|
|
|
ULONG CMeasureCPU::c_BufferSize = 0;
|
|
|
|
|
2009-02-10 18:37:48 +00:00
|
|
|
// 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
|
|
|
|
**
|
|
|
|
*/
|
2011-01-29 00:11:01 +00:00
|
|
|
CMeasureCPU::CMeasureCPU(CMeterWindow* meterWindow) : CMeasure(meterWindow),
|
|
|
|
m_FirstTime(true),
|
2011-02-07 09:38:27 +00:00
|
|
|
m_Processor()
|
2009-02-10 18:37:48 +00:00
|
|
|
{
|
|
|
|
m_MaxValue = 100.0;
|
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
if (c_NtQuerySystemInformation == NULL)
|
|
|
|
{
|
|
|
|
c_NtQuerySystemInformation = (PROCNTQSI)GetProcAddress(GetModuleHandle(L"ntdll"), "NtQuerySystemInformation");
|
|
|
|
}
|
|
|
|
if (c_GetSystemTimes == NULL)
|
|
|
|
{
|
|
|
|
c_GetSystemTimes = (PROCGST)GetProcAddress(GetModuleHandle(L"kernel32"), "GetSystemTimes");
|
|
|
|
}
|
|
|
|
if (c_NumOfProcessors == 0)
|
|
|
|
{
|
|
|
|
SYSTEM_INFO systemInfo = {0};
|
|
|
|
GetSystemInfo(&systemInfo);
|
|
|
|
c_NumOfProcessors = (int)systemInfo.dwNumberOfProcessors;
|
|
|
|
}
|
2009-02-10 18:37:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
** ~CMeasureCPU
|
|
|
|
**
|
|
|
|
** The destructor
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
CMeasureCPU::~CMeasureCPU()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-02-13 03:07:34 +00:00
|
|
|
/*
|
|
|
|
** ReadConfig
|
|
|
|
**
|
|
|
|
** Reads the measure specific configs.
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
void CMeasureCPU::ReadConfig(CConfigParser& parser, const WCHAR* section)
|
|
|
|
{
|
|
|
|
CMeasure::ReadConfig(parser, section);
|
|
|
|
|
|
|
|
int processor = parser.ReadInt(section, L"Processor", 0);
|
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
if (processor < 0 || processor > c_NumOfProcessors)
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
2010-12-19 23:06:13 +00:00
|
|
|
LogWithArgs(LOG_WARNING, L"[%s] Invalid Processor: %i", section, processor);
|
2010-02-13 03:07:34 +00:00
|
|
|
|
|
|
|
processor = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (processor != m_Processor)
|
|
|
|
{
|
|
|
|
m_Processor = processor;
|
|
|
|
m_FirstTime = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_FirstTime)
|
|
|
|
{
|
2011-02-07 09:38:27 +00:00
|
|
|
if (m_Processor == 0 && c_GetSystemTimes == NULL)
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
2011-02-07 09:38:27 +00:00
|
|
|
m_OldTime.assign(c_NumOfProcessors * 2, 0.0);
|
2010-02-13 03:07:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_OldTime.assign(2, 0.0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-10 18:37:48 +00:00
|
|
|
/*
|
|
|
|
** Update
|
|
|
|
**
|
2010-11-11 20:24:59 +00:00
|
|
|
** Updates the current CPU utilization value.
|
2009-02-10 18:37:48 +00:00
|
|
|
**
|
|
|
|
*/
|
|
|
|
bool CMeasureCPU::Update()
|
|
|
|
{
|
|
|
|
if (!CMeasure::PreUpdate()) return false;
|
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
if (m_Processor == 0 && c_GetSystemTimes)
|
2009-02-10 18:37:48 +00:00
|
|
|
{
|
2010-11-11 20:24:59 +00:00
|
|
|
BOOL status;
|
|
|
|
FILETIME ftIdleTime, ftKernelTime, ftUserTime;
|
2010-02-13 03:07:34 +00:00
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
// get new CPU's idle/kernel/user time
|
2011-02-07 09:38:27 +00:00
|
|
|
status = c_GetSystemTimes(&ftIdleTime, &ftKernelTime, &ftUserTime);
|
2010-11-11 20:24:59 +00:00
|
|
|
if (status == 0) return false;
|
2010-02-13 03:07:34 +00:00
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
CalcUsage(Ft2Double(ftIdleTime),
|
|
|
|
Ft2Double(ftKernelTime) + Ft2Double(ftUserTime));
|
|
|
|
}
|
2011-02-07 09:38:27 +00:00
|
|
|
else if (c_NtQuerySystemInformation)
|
2010-11-11 20:24:59 +00:00
|
|
|
{
|
|
|
|
LONG status;
|
2011-02-07 09:38:27 +00:00
|
|
|
ULONG bufSize = c_BufferSize;
|
|
|
|
BYTE* buf = (bufSize > 0) ? new BYTE[bufSize] : NULL;
|
2010-02-13 03:07:34 +00:00
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
int loop = 0;
|
2010-02-13 03:07:34 +00:00
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
do
|
|
|
|
{
|
|
|
|
ULONG size = 0;
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
status = c_NtQuerySystemInformation(SystemProcessorPerformanceInformation, buf, bufSize, &size);
|
|
|
|
if (status == STATUS_SUCCESS || status != STATUS_INFO_LENGTH_MISMATCH) break;
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
else // status == STATUS_INFO_LENGTH_MISMATCH
|
2010-11-11 20:24:59 +00:00
|
|
|
{
|
|
|
|
if (size == 0) // Returned required buffer size is always 0 on Windows 2000/XP.
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
2010-11-11 20:24:59 +00:00
|
|
|
if (bufSize == 0)
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
2011-02-07 09:38:27 +00:00
|
|
|
bufSize = sizeof(SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION) * c_NumOfProcessors;
|
2010-02-13 03:07:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-11-11 20:24:59 +00:00
|
|
|
bufSize += sizeof(SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION);
|
2010-02-13 03:07:34 +00:00
|
|
|
}
|
|
|
|
}
|
2010-11-11 20:24:59 +00:00
|
|
|
else
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
2010-11-11 20:24:59 +00:00
|
|
|
if (size != bufSize)
|
|
|
|
{
|
|
|
|
bufSize = size;
|
|
|
|
}
|
|
|
|
else // ??
|
|
|
|
{
|
|
|
|
bufSize += sizeof(SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION);
|
|
|
|
}
|
2010-02-13 03:07:34 +00:00
|
|
|
}
|
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
delete [] buf;
|
2010-11-11 20:24:59 +00:00
|
|
|
buf = new BYTE[bufSize];
|
|
|
|
}
|
|
|
|
++loop;
|
2011-02-07 09:38:27 +00:00
|
|
|
} while (loop < 5);
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
if (status != STATUS_SUCCESS) // failed
|
|
|
|
{
|
2011-02-07 09:38:27 +00:00
|
|
|
delete [] buf;
|
2010-11-11 20:24:59 +00:00
|
|
|
return false;
|
|
|
|
}
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
if (bufSize != c_BufferSize)
|
|
|
|
{
|
|
|
|
// Store the new buffer size
|
|
|
|
c_BufferSize = bufSize;
|
|
|
|
}
|
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION* systemPerfInfo = (SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION*)buf;
|
2009-02-10 18:37:48 +00:00
|
|
|
|
2010-11-11 20:24:59 +00:00
|
|
|
if (m_Processor == 0)
|
|
|
|
{
|
|
|
|
CalcAverageUsage(systemPerfInfo);
|
2010-02-13 03:07:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-11-11 20:24:59 +00:00
|
|
|
int processor = m_Processor - 1;
|
|
|
|
|
|
|
|
CalcUsage(Li2Double(systemPerfInfo[processor].IdleTime),
|
|
|
|
Li2Double(systemPerfInfo[processor].KernelTime) + Li2Double(systemPerfInfo[processor].UserTime));
|
2009-02-10 18:37:48 +00:00
|
|
|
}
|
2010-11-11 20:24:59 +00:00
|
|
|
|
|
|
|
delete [] buf;
|
2009-02-10 18:37:48 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-11-11 20:24:59 +00:00
|
|
|
return false;
|
2009-02-10 18:37:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return PostUpdate();
|
|
|
|
}
|
2010-02-13 03:07:34 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
** CalcUsage
|
|
|
|
**
|
|
|
|
** Calculates the current CPU utilization value.
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
void CMeasureCPU::CalcUsage(double idleTime, double systemTime)
|
|
|
|
{
|
|
|
|
if (!m_FirstTime)
|
|
|
|
{
|
|
|
|
double dbCpuUsage;
|
|
|
|
|
|
|
|
// CurrentCpuUsage% = 100 - ((IdleTime / SystemTime) * 100)
|
|
|
|
dbCpuUsage = 100.0 - ((idleTime - m_OldTime[0]) / (systemTime - m_OldTime[1])) * 100.0;
|
|
|
|
|
|
|
|
dbCpuUsage = min(dbCpuUsage, 100.0);
|
|
|
|
m_Value = max(dbCpuUsage, 0.0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_FirstTime = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// store new CPU's idle and system time
|
|
|
|
m_OldTime[0] = idleTime;
|
|
|
|
m_OldTime[1] = systemTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
** CalcAverageUsage
|
|
|
|
**
|
|
|
|
** Calculates the current CPU average utilization value.
|
|
|
|
** This function is used if GetSystemTimes function is not available.
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
void CMeasureCPU::CalcAverageUsage(SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION* systemPerfInfo)
|
|
|
|
{
|
2011-02-07 09:38:27 +00:00
|
|
|
if (!m_FirstTime)
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
|
|
|
double dbIdleTimeDiff = 0, dbSystemTimeDiff = 0;
|
|
|
|
double dbCpuUsage;
|
|
|
|
|
2011-02-07 09:38:27 +00:00
|
|
|
for (int i = 0; i < c_NumOfProcessors; ++i)
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
|
|
|
double dbIdleTime, dbSystemTime;
|
|
|
|
|
|
|
|
dbIdleTime = Li2Double(systemPerfInfo[i].IdleTime);
|
|
|
|
dbSystemTime = Li2Double(systemPerfInfo[i].KernelTime) + Li2Double(systemPerfInfo[i].UserTime);
|
|
|
|
|
|
|
|
dbIdleTimeDiff += dbIdleTime - m_OldTime[i * 2 + 0];
|
|
|
|
dbSystemTimeDiff += dbSystemTime - m_OldTime[i * 2 + 1];
|
|
|
|
|
|
|
|
// store new CPU's idle and system time
|
|
|
|
m_OldTime[i * 2 + 0] = dbIdleTime;
|
|
|
|
m_OldTime[i * 2 + 1] = dbSystemTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
// CurrentCpuUsage% = 100 - ((IdleTime / SystemTime) * 100)
|
|
|
|
dbCpuUsage = 100.0 - (dbIdleTimeDiff / dbSystemTimeDiff) * 100.0;
|
|
|
|
|
|
|
|
dbCpuUsage = min(dbCpuUsage, 100.0);
|
|
|
|
m_Value = max(dbCpuUsage, 0.0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// store new CPU's idle and system time
|
2011-02-07 09:38:27 +00:00
|
|
|
for (int i = 0; i < c_NumOfProcessors; ++i)
|
2010-02-13 03:07:34 +00:00
|
|
|
{
|
|
|
|
m_OldTime[i * 2 + 0] = Li2Double(systemPerfInfo[i].IdleTime);
|
|
|
|
m_OldTime[i * 2 + 1] = Li2Double(systemPerfInfo[i].KernelTime) + Li2Double(systemPerfInfo[i].UserTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_FirstTime = false;
|
|
|
|
}
|
|
|
|
}
|