# HG changeset patch # User moel.mich # Date 1266274709 0 # Node ID f847947b7f8e5fb3ce0b41c0462b8804aa1804fa # Parent eadad41d21a67177dea769cedb1eec755aaa9df9 Added core clock and bus speed support for Intel Core i5, i7 CPUs. diff -r eadad41d21a6 -r f847947b7f8e Hardware/CPU/IntelCPU.cs --- a/Hardware/CPU/IntelCPU.cs Sun Feb 14 21:09:08 2010 +0000 +++ b/Hardware/CPU/IntelCPU.cs Mon Feb 15 22:58:29 2010 +0000 @@ -48,6 +48,10 @@ private string name; private Image icon; + private uint family; + private uint model; + private uint stepping; + private Sensor[] coreTemperatures; private Sensor totalLoad; private Sensor[] coreLoads; @@ -63,17 +67,23 @@ private ulong lastCount; private long lastTime; + private uint maxNehalemMultiplier = 0; private const uint IA32_THERM_STATUS_MSR = 0x019C; private const uint IA32_TEMPERATURE_TARGET = 0x01A2; private const uint IA32_PERF_STATUS = 0x0198; + private const uint MSR_PLATFORM_INFO = 0xCE; public IntelCPU(string name, uint family, uint model, uint stepping, uint[,] cpuidData, uint[,] cpuidExtData) { this.name = name; this.icon = Utilities.EmbeddedResources.GetImage("cpu.png"); - + + this.family = family; + this.model = model; + this.stepping = stepping; + logicalProcessors = 0; if (cpuidData.GetLength(0) > 0x0B) { uint eax, ebx, ecx, edx; @@ -124,10 +134,11 @@ tjMax = 100; break; case 0x1C: // Intel Atom tjMax = 90; break; - case 0x1A: - uint eax = 0, edx = 0; - if (WinRing0.RdmsrPx( - IA32_TEMPERATURE_TARGET, ref eax, ref edx, (UIntPtr)1)) { + case 0x1A: // Intel Core i7 + case 0x1E: // Intel Core i5 + uint eax, edx; + if (WinRing0.Rdmsr(IA32_TEMPERATURE_TARGET, out eax, out edx)) + { tjMax = (eax >> 16) & 0xFF; } else tjMax = 100; @@ -139,6 +150,13 @@ default: tjMax = 100; break; } + if (family == 0x06 && model >= 0x1A) { // Core i5, i7 + uint eax, edx; + if (WinRing0.Rdmsr(MSR_PLATFORM_INFO, out eax, out edx)) { + maxNehalemMultiplier = (eax >> 8) & 0xff; + } + } + coreTemperatures = new Sensor[coreCount]; for (int i = 0; i < coreTemperatures.Length; i++) { coreTemperatures[i] = new Sensor("Core #" + (i + 1), i, tjMax, @@ -204,9 +222,9 @@ public void Update() { for (int i = 0; i < coreTemperatures.Length; i++) { - uint eax = 0, edx = 0; - if (WinRing0.RdmsrPx( - IA32_THERM_STATUS_MSR, ref eax, ref edx, + uint eax, edx; + if (WinRing0.RdmsrTx( + IA32_THERM_STATUS_MSR, out eax, out edx, (UIntPtr)(1 << (int)(logicalProcessorsPerCore * i)))) { // if reading is valid @@ -227,33 +245,41 @@ totalLoad.Value = cpuLoad.GetTotalLoad(); } - uint lsb = 0, msb = 0; - bool valid = WinRing0.RdtscPx(ref lsb, ref msb, (UIntPtr)1); + uint lsb, msb; + bool valid = WinRing0.RdtscTx(out lsb, out msb, (UIntPtr)1); long time = Stopwatch.GetTimestamp(); ulong count = ((ulong)msb << 32) | lsb; double delta = ((double)(time - lastTime)) / Stopwatch.Frequency; if (valid && delta > 0.5) { - double maxClock = (float)((count - lastCount) / (1e6 * delta)); + double maxClock = (count - lastCount) / (1e6 * delta); double busClock = 0; - uint eax, edx; + uint eax, edx; for (int i = 0; i < coreClocks.Length; i++) { - eax = 0; edx = 0; System.Threading.Thread.Sleep(1); - if (WinRing0.RdmsrPx(IA32_PERF_STATUS, ref eax, ref edx, + if (WinRing0.RdmsrTx(IA32_PERF_STATUS, out eax, out edx, (UIntPtr)(1 << (int)(logicalProcessorsPerCore * i)))) { - uint multiplier = (eax >> 8) & 0x1f; - uint maxMultiplier = (edx >> 8) & 0x1f; - // factor = multiplier * 2 to handle non integer multipliers - uint factor = (multiplier << 1) | ((eax >> 14) & 1); - uint maxFactor = (maxMultiplier << 1) | ((edx >> 14) & 1); - if (maxFactor > 0) { - coreClocks[i].Value = (float)(factor * maxClock / maxFactor); - busClock = (float)(2 * maxClock / maxFactor); + if (model < 0x1A) { // Core 2 + uint multiplier = (eax >> 8) & 0x1f; + uint maxMultiplier = (edx >> 8) & 0x1f; + // factor = multiplier * 2 to handle non integer multipliers + uint factor = (multiplier << 1) | ((eax >> 14) & 1); + uint maxFactor = (maxMultiplier << 1) | ((edx >> 14) & 1); + if (maxFactor > 0) { + coreClocks[i].Value = (float)(factor * maxClock / maxFactor); + busClock = (float)(2 * maxClock / maxFactor); + } + } else { // Core i5, i7 + uint nehalemMultiplier = eax & 0xff; + if (maxNehalemMultiplier > 0) { + coreClocks[i].Value = + (float)(nehalemMultiplier * maxClock / maxNehalemMultiplier); + busClock = (float)(maxClock / maxNehalemMultiplier); + } } - } else { + } else { // Intel Pentium 4 // if IA32_PERF_STATUS is not available, assume maxClock coreClocks[i].Value = (float)maxClock; - } + } } if (busClock > 0) { this.busClock.Value = (float)busClock; @@ -262,6 +288,6 @@ } lastCount = count; lastTime = time; - } - } + } + } } diff -r eadad41d21a6 -r f847947b7f8e Hardware/WinRing0.cs --- a/Hardware/WinRing0.cs Sun Feb 14 21:09:08 2010 +0000 +++ b/Hardware/WinRing0.cs Mon Feb 15 22:58:29 2010 +0000 @@ -83,8 +83,9 @@ out uint eax, out uint ebx, out uint ecx, out uint edx); public delegate bool CpuidExDelegate(uint index, uint ecxValue, out uint eax, out uint ebx, out uint ecx, out uint edx); - public delegate bool RdmsrPxDelegate(uint index, ref uint eax, ref uint edx, - UIntPtr processAffinityMask); + public delegate bool RdmsrDelegate(uint index, out uint eax, out uint edx); + public delegate bool RdmsrTxDelegate(uint index, out uint eax, out uint edx, + UIntPtr threadAffinityMask); public delegate byte ReadIoPortByteDelegate(ushort port); public delegate void WriteIoPortByteDelegate(ushort port, byte value); public delegate void SetPciMaxBusIndexDelegate(byte max); @@ -94,8 +95,8 @@ uint regAddress, out uint value); public delegate bool WritePciConfigDwordExDelegate(uint pciAddress, uint regAddress, uint value); - public delegate bool RdtscPxDelegate(ref uint eax, ref uint edx, - UIntPtr processAffinityMask); + public delegate bool RdtscTxDelegate(out uint eax, out uint edx, + UIntPtr threadAffinityMask); private static InitializeOlsDelegate InitializeOls; private static DeinitializeOlsDelegate DeinitializeOls; @@ -104,14 +105,15 @@ public static IsCpuidDelegate IsCpuid; public static CpuidDelegate Cpuid; public static CpuidExDelegate CpuidEx; - public static RdmsrPxDelegate RdmsrPx; + public static RdmsrDelegate Rdmsr; + public static RdmsrTxDelegate RdmsrTx; public static ReadIoPortByteDelegate ReadIoPortByte; public static WriteIoPortByteDelegate WriteIoPortByte; public static SetPciMaxBusIndexDelegate SetPciMaxBusIndex; public static FindPciDeviceByIdDelegate FindPciDeviceById; public static ReadPciConfigDwordExDelegate ReadPciConfigDwordEx; public static WritePciConfigDwordExDelegate WritePciConfigDwordEx; - public static RdtscPxDelegate RdtscPx; + public static RdtscTxDelegate RdtscTx; private static void GetDelegate(string entryPoint, out T newDelegate) where T : class @@ -131,14 +133,15 @@ GetDelegate("IsCpuid", out IsCpuid); GetDelegate("Cpuid", out Cpuid); GetDelegate("CpuidEx", out CpuidEx); - GetDelegate("RdmsrPx", out RdmsrPx); + GetDelegate("Rdmsr", out Rdmsr); + GetDelegate("RdmsrTx", out RdmsrTx); GetDelegate("ReadIoPortByte", out ReadIoPortByte); GetDelegate("WriteIoPortByte", out WriteIoPortByte); GetDelegate("SetPciMaxBusIndex", out SetPciMaxBusIndex); GetDelegate("FindPciDeviceById", out FindPciDeviceById); GetDelegate("ReadPciConfigDwordEx", out ReadPciConfigDwordEx); GetDelegate("WritePciConfigDwordEx", out WritePciConfigDwordEx); - GetDelegate("RdtscPx", out RdtscPx); + GetDelegate("RdtscTx", out RdtscTx); try { if (InitializeOls != null && InitializeOls()) diff -r eadad41d21a6 -r f847947b7f8e Properties/AssemblyInfo.cs --- a/Properties/AssemblyInfo.cs Sun Feb 14 21:09:08 2010 +0000 +++ b/Properties/AssemblyInfo.cs Mon Feb 15 22:58:29 2010 +0000 @@ -69,5 +69,5 @@ // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: // [assembly: AssemblyVersion("1.0.*")] -[assembly: AssemblyVersion("0.1.20.1")] -[assembly: AssemblyFileVersion("0.1.20.1")] +[assembly: AssemblyVersion("0.1.21.0")] +[assembly: AssemblyFileVersion("0.1.21.0")]