Hardware/LPC/F718XX.cs
author moel.mich
Thu, 04 Feb 2010 21:19:27 +0000
changeset 26 0e01b63e1fdc
parent 16 e9abdc6e05af
child 31 c4d1fb76a9e1
permissions -rw-r--r--
Release version 0.1.13. Replaced PerformanceCounter based CPU load sensors with an implementation using NtQuerySystemInformation. Fixed a temperature reading problem for W83627DHG chips (sensors with invalid value 127?C).
moel@16
     1
/*
moel@16
     2
  
moel@16
     3
  Version: MPL 1.1/GPL 2.0/LGPL 2.1
moel@16
     4
moel@16
     5
  The contents of this file are subject to the Mozilla Public License Version
moel@16
     6
  1.1 (the "License"); you may not use this file except in compliance with
moel@16
     7
  the License. You may obtain a copy of the License at
moel@16
     8
 
moel@16
     9
  http://www.mozilla.org/MPL/
moel@16
    10
moel@16
    11
  Software distributed under the License is distributed on an "AS IS" basis,
moel@16
    12
  WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
moel@16
    13
  for the specific language governing rights and limitations under the License.
moel@16
    14
moel@16
    15
  The Original Code is the Open Hardware Monitor code.
moel@16
    16
moel@16
    17
  The Initial Developer of the Original Code is 
moel@16
    18
  Michael Möller <m.moeller@gmx.ch>.
moel@16
    19
  Portions created by the Initial Developer are Copyright (C) 2009-2010
moel@16
    20
  the Initial Developer. All Rights Reserved.
moel@16
    21
moel@16
    22
  Contributor(s):
moel@16
    23
moel@16
    24
  Alternatively, the contents of this file may be used under the terms of
moel@16
    25
  either the GNU General Public License Version 2 or later (the "GPL"), or
moel@16
    26
  the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
moel@16
    27
  in which case the provisions of the GPL or the LGPL are applicable instead
moel@16
    28
  of those above. If you wish to allow use of your version of this file only
moel@16
    29
  under the terms of either the GPL or the LGPL, and not to allow others to
moel@16
    30
  use your version of this file under the terms of the MPL, indicate your
moel@16
    31
  decision by deleting the provisions above and replace them with the notice
moel@16
    32
  and other provisions required by the GPL or the LGPL. If you do not delete
moel@16
    33
  the provisions above, a recipient may use your version of this file under
moel@16
    34
  the terms of any one of the MPL, the GPL or the LGPL.
moel@16
    35
 
moel@16
    36
*/
moel@16
    37
moel@16
    38
using System;
moel@16
    39
using System.Collections.Generic;
moel@16
    40
using System.Drawing;
moel@16
    41
using System.Text;
moel@16
    42
moel@16
    43
namespace OpenHardwareMonitor.Hardware.LPC {
moel@16
    44
  public class F718XX  : IHardware {
moel@16
    45
moel@16
    46
    private string name;
moel@16
    47
    private Image icon;
moel@16
    48
moel@16
    49
    private Chip chip;
moel@16
    50
    private ushort address;
moel@16
    51
moel@16
    52
    private List<ISensor> active = new List<ISensor>();
moel@16
    53
moel@16
    54
    private Sensor[] temperatures;
moel@16
    55
    private Sensor[] fans;
moel@16
    56
    private Sensor[] voltages;
moel@16
    57
    private float[] voltageGains;
moel@16
    58
moel@16
    59
    // Hardware Monitor
moel@16
    60
    private const byte ADDRESS_REGISTER_OFFSET = 0x05;
moel@16
    61
    private const byte DATA_REGISTER_OFFSET = 0x06;
moel@16
    62
moel@16
    63
    // Hardware Monitor Registers
moel@16
    64
    private const byte VOLTAGE_BASE_REG = 0x20;
moel@16
    65
    private const byte TEMPERATURE_BASE_REG = 0x72;
moel@16
    66
    private byte[] FAN_TACHOMETER_REG = new byte[] { 0xA0, 0xB0, 0xC0, 0xD0 };
moel@16
    67
    
moel@16
    68
    private byte ReadByte(byte register) {
moel@16
    69
      WinRing0.WriteIoPortByte(
moel@16
    70
        (ushort)(address + ADDRESS_REGISTER_OFFSET), register);
moel@16
    71
      return WinRing0.ReadIoPortByte((ushort)(address + DATA_REGISTER_OFFSET));
moel@16
    72
    }
moel@16
    73
moel@16
    74
    public F718XX(Chip chip, ushort address) {
moel@16
    75
      this.chip = chip;
moel@16
    76
      this.address = address;
moel@16
    77
moel@16
    78
      switch (chip) {
moel@16
    79
        case Chip.F71862: name = "Fintek F71862"; break;
moel@16
    80
        case Chip.F71869: name = "Fintek F71869"; break;
moel@16
    81
        case Chip.F71882: name = "Fintek F71882"; break;
moel@16
    82
        case Chip.F71889: name = "Fintek F71889"; break;
moel@16
    83
        default: return;
moel@16
    84
      }
moel@16
    85
moel@16
    86
      temperatures = new Sensor[3];
moel@16
    87
      for (int i = 0; i < temperatures.Length; i++)
moel@16
    88
        temperatures[i] = new Sensor("Temperature #" + (i + 1), i,
moel@16
    89
          SensorType.Temperature, this);
moel@16
    90
moel@16
    91
      fans = new Sensor[4];
moel@16
    92
      for (int i = 0; i < fans.Length; i++)
moel@16
    93
        fans[i] = new Sensor("Fan #" + (i + 1), i, SensorType.Fan, this);
moel@16
    94
moel@16
    95
      voltageGains = new float[] { 1, 0.5f, 1, 1, 1, 1, 1, 1, 1 };
moel@16
    96
      voltages = new Sensor[4];
moel@16
    97
      voltages[0] = new Sensor("VCC3V", 0, SensorType.Voltage, this);
moel@16
    98
      voltages[1] = new Sensor("CPU VCore", 1, SensorType.Voltage, this);      
moel@16
    99
      voltages[2] = new Sensor("VSB3V", 7, SensorType.Voltage, this);
moel@16
   100
      voltages[3] = new Sensor("Battery", 8, SensorType.Voltage, this);
moel@16
   101
moel@16
   102
      this.icon = Utilities.EmbeddedResources.GetImage("chip.png");
moel@16
   103
    }
moel@16
   104
moel@16
   105
    public string Name {
moel@16
   106
      get { return name; }
moel@16
   107
    }
moel@16
   108
moel@16
   109
    public string Identifier {
moel@18
   110
      get { return "/lpc/" + chip.ToString().ToLower(); }
moel@16
   111
    }
moel@16
   112
moel@16
   113
    public Image Icon {
moel@16
   114
      get { return icon; }
moel@16
   115
    }
moel@16
   116
moel@16
   117
    public ISensor[] Sensors {
moel@16
   118
      get { return active.ToArray(); }
moel@16
   119
    }
moel@16
   120
moel@16
   121
    public string GetReport() {
moel@16
   122
      StringBuilder r = new StringBuilder();
moel@16
   123
moel@16
   124
      r.AppendLine("LPC F718XX");
moel@16
   125
      r.AppendLine();
moel@16
   126
      r.Append("Base Adress: 0x"); r.AppendLine(address.ToString("X4"));
moel@16
   127
      r.AppendLine();
moel@16
   128
      r.AppendLine("Hardware Monitor Registers");
moel@16
   129
      r.AppendLine();
moel@16
   130
moel@16
   131
      r.AppendLine("      00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
moel@16
   132
      r.AppendLine();
moel@16
   133
      for (int i = 0; i <= 0xF; i++) {
moel@16
   134
        r.Append(" "); r.Append((i << 4).ToString("X2")); r.Append("  ");
moel@16
   135
        for (int j = 0; j <= 0xF; j++) {
moel@16
   136
          r.Append(" ");
moel@16
   137
          r.Append(ReadByte((byte)((i << 4) | j)).ToString("X2"));
moel@16
   138
        }
moel@16
   139
        r.AppendLine();
moel@16
   140
      }
moel@16
   141
      r.AppendLine();
moel@16
   142
      return r.ToString();
moel@16
   143
    }
moel@16
   144
moel@16
   145
    public void Update() {
moel@16
   146
moel@16
   147
      foreach (Sensor sensor in voltages) {
moel@16
   148
        int value = ReadByte((byte)(VOLTAGE_BASE_REG + sensor.Index));
moel@16
   149
        sensor.Value = voltageGains[sensor.Index] * 0.001f * (value << 4);
moel@16
   150
        if (sensor.Value > 0)
moel@16
   151
          ActivateSensor(sensor);
moel@16
   152
        else
moel@16
   153
          DeactivateSensor(sensor);
moel@16
   154
      }
moel@16
   155
moel@16
   156
      foreach (Sensor sensor in temperatures) {
moel@16
   157
        sbyte value = (sbyte)ReadByte((byte)(
moel@16
   158
          TEMPERATURE_BASE_REG + 2 * sensor.Index));
moel@16
   159
        sensor.Value = value;
moel@16
   160
        if (value < sbyte.MaxValue && value > 0)
moel@16
   161
          ActivateSensor(sensor);
moel@16
   162
        else
moel@16
   163
          DeactivateSensor(sensor);
moel@16
   164
      }
moel@16
   165
moel@16
   166
      foreach (Sensor sensor in fans) {
moel@16
   167
        int value = ReadByte(FAN_TACHOMETER_REG[sensor.Index]) << 8;
moel@16
   168
        value |= ReadByte((byte)(FAN_TACHOMETER_REG[sensor.Index] + 1));
moel@16
   169
moel@16
   170
        if (value > 0) {
moel@16
   171
          sensor.Value = (value < 0x0fff) ? 1.5e6f / value : 0;
moel@16
   172
          ActivateSensor(sensor);
moel@16
   173
        } else {
moel@16
   174
          DeactivateSensor(sensor);
moel@16
   175
        }
moel@16
   176
      }      
moel@16
   177
    }
moel@16
   178
moel@16
   179
    private void ActivateSensor(Sensor sensor) {
moel@16
   180
      if (!active.Contains(sensor)) {
moel@16
   181
        active.Add(sensor);
moel@16
   182
        if (SensorAdded != null)
moel@16
   183
          SensorAdded(sensor);
moel@16
   184
      }
moel@16
   185
    }
moel@16
   186
moel@16
   187
    private void DeactivateSensor(Sensor sensor) {
moel@16
   188
      if (active.Contains(sensor)) {
moel@16
   189
        active.Remove(sensor);
moel@16
   190
        if (SensorRemoved != null)
moel@16
   191
          SensorRemoved(sensor);
moel@16
   192
      }
moel@16
   193
    }
moel@16
   194
moel@16
   195
    public event SensorEventHandler SensorAdded;
moel@16
   196
    public event SensorEventHandler SensorRemoved;
moel@16
   197
moel@16
   198
  }
moel@16
   199
}