Hardware/CPU/AMD10CPU.cs
author moel.mich
Sat, 20 Feb 2010 19:51:10 +0000
changeset 56 5cb7eb5bf628
parent 31 c4d1fb76a9e1
child 63 1a7c13ac7348
permissions -rw-r--r--
Improved Winbond temperature reading. Temperatures create by adding PECI Agent values (delta to TCC Activation Temperature) to a (possibly uncalibrated) TBase are not read. Direct reading temperatures from sensor report register if available. Added lower bound for temperatures on Winbond chips. Nvidia GPUs are now displayed even if they do not have any sensors.
moel@1
     1
/*
moel@1
     2
  
moel@1
     3
  Version: MPL 1.1/GPL 2.0/LGPL 2.1
moel@1
     4
moel@1
     5
  The contents of this file are subject to the Mozilla Public License Version
moel@1
     6
  1.1 (the "License"); you may not use this file except in compliance with
moel@1
     7
  the License. You may obtain a copy of the License at
moel@1
     8
 
moel@1
     9
  http://www.mozilla.org/MPL/
moel@1
    10
moel@1
    11
  Software distributed under the License is distributed on an "AS IS" basis,
moel@1
    12
  WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
moel@1
    13
  for the specific language governing rights and limitations under the License.
moel@1
    14
moel@1
    15
  The Original Code is the Open Hardware Monitor code.
moel@1
    16
moel@1
    17
  The Initial Developer of the Original Code is 
moel@1
    18
  Michael Möller <m.moeller@gmx.ch>.
moel@1
    19
  Portions created by the Initial Developer are Copyright (C) 2009-2010
moel@1
    20
  the Initial Developer. All Rights Reserved.
moel@1
    21
moel@1
    22
  Contributor(s):
moel@1
    23
moel@1
    24
  Alternatively, the contents of this file may be used under the terms of
moel@1
    25
  either the GNU General Public License Version 2 or later (the "GPL"), or
moel@1
    26
  the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
moel@1
    27
  in which case the provisions of the GPL or the LGPL are applicable instead
moel@1
    28
  of those above. If you wish to allow use of your version of this file only
moel@1
    29
  under the terms of either the GPL or the LGPL, and not to allow others to
moel@1
    30
  use your version of this file under the terms of the MPL, indicate your
moel@1
    31
  decision by deleting the provisions above and replace them with the notice
moel@1
    32
  and other provisions required by the GPL or the LGPL. If you do not delete
moel@1
    33
  the provisions above, a recipient may use your version of this file under
moel@1
    34
  the terms of any one of the MPL, the GPL or the LGPL.
moel@1
    35
 
moel@1
    36
*/
moel@1
    37
moel@1
    38
using System;
moel@1
    39
using System.Collections.Generic;
moel@1
    40
using System.Drawing;
moel@24
    41
using System.Diagnostics;
moel@1
    42
using System.Text;
moel@1
    43
moel@1
    44
namespace OpenHardwareMonitor.Hardware.CPU {
moel@1
    45
  
moel@31
    46
  public class AMD10CPU : Hardware, IHardware {
moel@1
    47
    private string name;
moel@1
    48
    private Image icon;
moel@1
    49
moel@1
    50
    private uint pciAddress;
moel@1
    51
moel@1
    52
    private Sensor coreTemperature;
moel@24
    53
    private Sensor totalLoad;
moel@24
    54
    private Sensor[] coreLoads;
moel@24
    55
moel@26
    56
    private CPULoad cpuLoad;
moel@1
    57
moel@1
    58
    private const ushort PCI_AMD_VENDOR_ID = 0x1022;
moel@1
    59
    private const ushort PCI_AMD_10H_MISCELLANEOUS_DEVICE_ID = 0x1203;
moel@42
    60
    private const ushort PCI_AMD_11H_MISCELLANEOUS_DEVICE_ID = 0x1303;
moel@1
    61
    private const uint REPORTED_TEMPERATURE_CONTROL_REGISTER = 0xA4;
moel@1
    62
moel@1
    63
    public AMD10CPU(string name, uint family, uint model, uint stepping, 
moel@1
    64
      uint[,] cpuidData, uint[,] cpuidExtData) {
moel@1
    65
      
moel@1
    66
      this.name = name;
moel@1
    67
      this.icon = Utilities.EmbeddedResources.GetImage("cpu.png");     
moel@1
    68
moel@1
    69
      uint coreCount = 1;
moel@1
    70
      if (cpuidExtData.GetLength(0) > 8)
moel@1
    71
        coreCount = (cpuidExtData[8, 2] & 0xFF) + 1;
moel@24
    72
moel@24
    73
      totalLoad = new Sensor("CPU Total", 0, SensorType.Load, this);
moel@24
    74
moel@24
    75
      coreLoads = new Sensor[coreCount];
moel@26
    76
      for (int i = 0; i < coreCount; i++) 
moel@24
    77
        coreLoads[i] = new Sensor("Core #" + (i + 1), i + 1,
moel@24
    78
          SensorType.Load, this);
moel@26
    79
moel@26
    80
      cpuLoad = new CPULoad(coreCount, 1);
moel@26
    81
      if (cpuLoad.IsAvailable) {
moel@26
    82
        foreach (Sensor sensor in coreLoads)
moel@26
    83
          ActivateSensor(sensor);
moel@26
    84
        ActivateSensor(totalLoad);
moel@26
    85
      }    
moel@1
    86
      
moel@1
    87
      // AMD family 10h processors support only one temperature sensor
moel@22
    88
      coreTemperature = new Sensor(
moel@22
    89
        "Core" + (coreCount > 1 ? " #1 - #" + coreCount : ""), 0, 
moel@22
    90
        SensorType.Temperature, this);
moel@1
    91
moel@1
    92
      pciAddress = WinRing0.FindPciDeviceById(PCI_AMD_VENDOR_ID, 
moel@1
    93
        PCI_AMD_10H_MISCELLANEOUS_DEVICE_ID, 0);
moel@42
    94
      if (pciAddress == 0xFFFFFFFF) 
moel@42
    95
        pciAddress = WinRing0.FindPciDeviceById(PCI_AMD_VENDOR_ID,
moel@42
    96
          PCI_AMD_11H_MISCELLANEOUS_DEVICE_ID, 0);
moel@42
    97
moel@1
    98
      Update();                   
moel@1
    99
    }
moel@1
   100
moel@1
   101
    public string Name {
moel@1
   102
      get { return name; }
moel@1
   103
    }
moel@1
   104
moel@1
   105
    public string Identifier {
moel@1
   106
      get { return "/amdcpu/0"; }
moel@1
   107
    }
moel@1
   108
moel@1
   109
    public Image Icon {
moel@1
   110
      get { return icon; }
moel@1
   111
    }
moel@1
   112
moel@1
   113
    public string GetReport() {
moel@1
   114
      return null;
moel@1
   115
    }
moel@1
   116
moel@1
   117
    public void Update() {
moel@42
   118
      if (pciAddress != 0xFFFFFFFF) {
moel@42
   119
        uint value;
moel@42
   120
        if (WinRing0.ReadPciConfigDwordEx(pciAddress,
moel@42
   121
          REPORTED_TEMPERATURE_CONTROL_REGISTER, out value)) {
moel@42
   122
          coreTemperature.Value = ((value >> 21) & 0x7FF) / 8.0f;
moel@42
   123
          ActivateSensor(coreTemperature);
moel@42
   124
        } else {
moel@42
   125
          DeactivateSensor(coreTemperature);
moel@42
   126
        }
moel@24
   127
      }
moel@24
   128
moel@26
   129
      if (cpuLoad.IsAvailable) {
moel@26
   130
        cpuLoad.Update();
moel@26
   131
        for (int i = 0; i < coreLoads.Length; i++)
moel@26
   132
          coreLoads[i].Value = cpuLoad.GetCoreLoad(i);
moel@26
   133
        totalLoad.Value = cpuLoad.GetTotalLoad();
moel@25
   134
      }
moel@1
   135
    }
moel@1
   136
  }
moel@1
   137
}