Hardware/CPU/IntelCPU.cs
author moel.mich
Wed, 03 Feb 2010 20:35:10 +0000
changeset 24 09ab31bee6bd
parent 23 1662dea7a261
child 25 ff3e6edc7113
permissions -rw-r--r--
Release version 0.1.11. Added support for load percentage sensor type. Added load sensors for CPUs and ATI GPUs.
     1 /*
     2   
     3   Version: MPL 1.1/GPL 2.0/LGPL 2.1
     4 
     5   The contents of this file are subject to the Mozilla Public License Version
     6   1.1 (the "License"); you may not use this file except in compliance with
     7   the License. You may obtain a copy of the License at
     8  
     9   http://www.mozilla.org/MPL/
    10 
    11   Software distributed under the License is distributed on an "AS IS" basis,
    12   WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
    13   for the specific language governing rights and limitations under the License.
    14 
    15   The Original Code is the Open Hardware Monitor code.
    16 
    17   The Initial Developer of the Original Code is 
    18   Michael Möller <m.moeller@gmx.ch>.
    19   Portions created by the Initial Developer are Copyright (C) 2009-2010
    20   the Initial Developer. All Rights Reserved.
    21 
    22   Contributor(s):
    23 
    24   Alternatively, the contents of this file may be used under the terms of
    25   either the GNU General Public License Version 2 or later (the "GPL"), or
    26   the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
    27   in which case the provisions of the GPL or the LGPL are applicable instead
    28   of those above. If you wish to allow use of your version of this file only
    29   under the terms of either the GPL or the LGPL, and not to allow others to
    30   use your version of this file under the terms of the MPL, indicate your
    31   decision by deleting the provisions above and replace them with the notice
    32   and other provisions required by the GPL or the LGPL. If you do not delete
    33   the provisions above, a recipient may use your version of this file under
    34   the terms of any one of the MPL, the GPL or the LGPL.
    35  
    36 */
    37 
    38 using System;
    39 using System.Collections.Generic;
    40 using System.Drawing;
    41 using System.Diagnostics;
    42 using System.Reflection;
    43 using System.Text;
    44 
    45 namespace OpenHardwareMonitor.Hardware.CPU {
    46   public class IntelCPU : IHardware {
    47 
    48     private string name;
    49     private Image icon;
    50 
    51     private Sensor[] coreTemperatures;
    52     private Sensor totalLoad;
    53     private Sensor[] coreLoads;
    54 
    55     private List<ISensor> active = new List<ISensor>();
    56 
    57     private float tjMax = 0;
    58     private uint logicalProcessors;
    59     private uint logicalProcessorsPerCore;
    60     private uint coreCount;
    61 
    62     private PerformanceCounter totalLoadCounter;
    63     private PerformanceCounter[] coreLoadCounters;
    64 
    65     private const uint IA32_THERM_STATUS_MSR = 0x019C;
    66     private const uint IA32_TEMPERATURE_TARGET = 0x01A2;
    67 
    68     public IntelCPU(string name, uint family, uint model, uint stepping, 
    69       uint[,] cpuidData, uint[,] cpuidExtData) {
    70       
    71       this.name = name;
    72       this.icon = Utilities.EmbeddedResources.GetImage("cpu.png");
    73             
    74       logicalProcessors = 0;
    75       if (cpuidData.GetLength(0) > 0x0B) {
    76         uint eax, ebx, ecx, edx;
    77         WinRing0.CpuidEx(0x0B, 0, out eax, out ebx, out ecx, out edx);
    78         logicalProcessorsPerCore = ebx & 0xFF;
    79         if (logicalProcessorsPerCore > 0) {
    80           WinRing0.CpuidEx(0x0B, 1, out eax, out ebx, out ecx, out edx);
    81           logicalProcessors = ebx & 0xFF;
    82         }   
    83       }
    84       if (logicalProcessors <= 0 && cpuidData.GetLength(0) > 0x04) {
    85         logicalProcessors = ((cpuidData[4, 0] >> 26) & 0x3F) + 1;
    86         logicalProcessorsPerCore = 1;
    87       }
    88       if (logicalProcessors <= 0) {
    89         logicalProcessors = 1;
    90         logicalProcessorsPerCore = 1;
    91       }
    92 
    93       coreCount = logicalProcessors / logicalProcessorsPerCore;
    94 
    95       switch (family) {
    96         case 0x06: {
    97           switch (model) {
    98             case 0x0F: // Intel Core 65nm
    99               switch (stepping) {
   100                 case 0x06: // B2
   101                   switch (coreCount) {
   102                     case 2:
   103                       tjMax = 80; break;
   104                     case 4:
   105                       tjMax = 90; break;
   106                     default:
   107                       tjMax = 85; break;
   108                   }
   109                   tjMax = 80; break;
   110                 case 0x0B: // G0
   111                   tjMax = 90; break;
   112                 case 0x0D: // M0
   113                   tjMax = 85; break;
   114                 default:
   115                   tjMax = 85; break;
   116               } break;            
   117             case 0x17: // Intel Core 45nm
   118               tjMax = 100; break;
   119             case 0x1C: // Intel Atom 
   120               tjMax = 90; break;
   121             case 0x1A:
   122               uint eax = 0, edx = 0;
   123               if (WinRing0.RdmsrPx(
   124                   IA32_TEMPERATURE_TARGET, ref eax, ref edx, (UIntPtr)1)) {
   125                 tjMax = (eax >> 16) & 0xFF;
   126               } else
   127                 tjMax = 100;
   128               break;
   129             default:
   130               tjMax = 100; break;
   131           }
   132         } break;
   133         default: tjMax = 100; break;
   134       }
   135 
   136       totalLoadCounter = new PerformanceCounter();
   137       totalLoadCounter.CategoryName = "Processor";
   138       totalLoadCounter.CounterName = "% Processor Time";
   139       totalLoadCounter.InstanceName = "_Total";
   140       totalLoad = new Sensor("CPU Total", 0, SensorType.Load, this);
   141 
   142       coreLoadCounters = new PerformanceCounter[
   143         coreCount * logicalProcessorsPerCore];
   144       for (int i = 0; i < coreLoadCounters.Length; i++) {
   145         coreLoadCounters[i] = new PerformanceCounter();
   146         coreLoadCounters[i].CategoryName = "Processor";
   147         coreLoadCounters[i].CounterName = "% Processor Time";
   148         coreLoadCounters[i].InstanceName = i.ToString();
   149       }
   150 
   151       coreTemperatures = new Sensor[coreCount];
   152       coreLoads = new Sensor[coreCount];
   153       for (int i = 0; i < coreTemperatures.Length; i++) {
   154         coreTemperatures[i] = new Sensor("Core #" + (i + 1), i, tjMax,
   155           SensorType.Temperature, this);
   156         coreLoads[i] = new Sensor("Core #" + (i + 1), i + 1, 
   157           SensorType.Load, this);
   158       }
   159 
   160       Update();                   
   161     }
   162 
   163     public string Name {
   164       get { return name; }
   165     }
   166 
   167     public string Identifier {
   168       get { return "/intelcpu/0"; }
   169     }
   170 
   171     public Image Icon {
   172       get { return icon; }
   173     }
   174 
   175     public ISensor[] Sensors {
   176       get { return active.ToArray(); }
   177     }
   178 
   179     public string GetReport() {
   180       StringBuilder r = new StringBuilder();
   181 
   182       r.AppendLine("Intel CPU");
   183       r.AppendLine();
   184       r.AppendFormat("Name: {0}{1}", name, Environment.NewLine);
   185       r.AppendFormat("Number of cores: {0}{1}", coreCount, 
   186         Environment.NewLine);
   187       r.AppendFormat("Threads per core: {0}{1}", logicalProcessorsPerCore,
   188         Environment.NewLine);
   189       r.AppendFormat("TjMax: {0}{1}", tjMax, Environment.NewLine);
   190       r.AppendLine();
   191 
   192       return r.ToString();
   193     }
   194 
   195     public void Update() {
   196 
   197       uint eax = 0, edx = 0;      
   198       for (int i = 0; i < coreTemperatures.Length; i++) {
   199         if (WinRing0.RdmsrPx(
   200           IA32_THERM_STATUS_MSR, ref eax, ref edx, 
   201             (UIntPtr)(1 << (int)(logicalProcessorsPerCore * i)))) 
   202         {
   203           // if reading is valid
   204           if ((eax & 0x80000000) != 0) {
   205             // get the dist from tjMax from bits 22:16
   206             coreTemperatures[i].Value = tjMax - ((eax & 0x007F0000) >> 16);
   207             ActivateSensor(coreTemperatures[i]);
   208           } else {
   209             DeactivateSensor(coreTemperatures[i]);
   210           }
   211         }        
   212       }
   213 
   214       totalLoad.Value = totalLoadCounter.NextValue();
   215       ActivateSensor(totalLoad);
   216 
   217       for (int i = 0; i < coreLoads.Length; i++) {
   218         float value = 0;
   219         for (int j = 0; j < logicalProcessorsPerCore; j++)
   220           value += coreLoadCounters[
   221             logicalProcessorsPerCore * i + j].NextValue();
   222         value /= logicalProcessorsPerCore;
   223         coreLoads[i].Value = value;
   224         ActivateSensor(coreLoads[i]);
   225       }
   226     }
   227 
   228     private void ActivateSensor(Sensor sensor) {
   229       if (!active.Contains(sensor)) {
   230         active.Add(sensor);
   231         if (SensorAdded != null)
   232           SensorAdded(sensor);
   233       }
   234     }
   235 
   236     private void DeactivateSensor(Sensor sensor) {
   237       if (active.Contains(sensor)) {
   238         active.Remove(sensor);
   239         if (SensorRemoved != null)
   240           SensorRemoved(sensor);
   241       }
   242     }
   243 
   244     public event SensorEventHandler SensorAdded;
   245     public event SensorEventHandler SensorRemoved;
   246   }
   247 }