Hardware/LPC/IT87XX.cs
author moel.mich
Tue, 17 Aug 2010 21:44:02 +0000
changeset 170 31858ba46e9c
parent 166 fa9dfbfc4145
child 182 4801e9eaf979
permissions -rw-r--r--
Fixed Issue 97.
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@166
    40
using System.Globalization;
moel@16
    41
using System.Text;
moel@16
    42
moel@16
    43
namespace OpenHardwareMonitor.Hardware.LPC {
moel@165
    44
  internal class IT87XX : ISuperIO {
moel@130
    45
       
moel@16
    46
    private ushort address;
moel@130
    47
    private Chip chip;
moel@145
    48
    private byte version;
moel@16
    49
moel@78
    50
    private readonly ushort addressReg;
moel@78
    51
    private readonly ushort dataReg;
moel@78
    52
moel@130
    53
    private float?[] voltages = new float?[0];
moel@130
    54
    private float?[] temperatures = new float?[0];
moel@130
    55
    private float?[] fans = new float?[0];
moel@170
    56
moel@170
    57
    private readonly float voltageGain;
moel@16
    58
   
moel@16
    59
    // Consts
moel@16
    60
    private const byte ITE_VENDOR_ID = 0x90;
moel@16
    61
       
moel@16
    62
    // Environment Controller
moel@16
    63
    private const byte ADDRESS_REGISTER_OFFSET = 0x05;
moel@16
    64
    private const byte DATA_REGISTER_OFFSET = 0x06;
moel@16
    65
moel@16
    66
    // Environment Controller Registers    
moel@16
    67
    private const byte CONFIGURATION_REGISTER = 0x00;
moel@16
    68
    private const byte TEMPERATURE_BASE_REG = 0x29;
moel@16
    69
    private const byte VENDOR_ID_REGISTER = 0x58;
moel@16
    70
    private const byte FAN_TACHOMETER_16_BIT_ENABLE_REGISTER = 0x0c;
moel@16
    71
    private byte[] FAN_TACHOMETER_REG = 
moel@16
    72
      new byte[] { 0x0d, 0x0e, 0x0f, 0x80, 0x82 };
moel@16
    73
    private byte[] FAN_TACHOMETER_EXT_REG =
moel@16
    74
      new byte[] { 0x18, 0x19, 0x1a, 0x81, 0x83 };
moel@78
    75
    private const byte VOLTAGE_BASE_REG = 0x20;
moel@78
    76
moel@78
    77
    private byte ReadByte(byte register, out bool valid) {
moel@78
    78
      WinRing0.WriteIoPortByte(addressReg, register);
moel@78
    79
      byte value = WinRing0.ReadIoPortByte(dataReg);
moel@78
    80
      valid = register == WinRing0.ReadIoPortByte(addressReg);
moel@78
    81
      return value;
moel@144
    82
    } 
moel@16
    83
moel@145
    84
    public IT87XX(Chip chip, ushort address, byte version) {
moel@16
    85
      
moel@16
    86
      this.address = address;
moel@130
    87
      this.chip = chip;
moel@145
    88
      this.version = version;
moel@78
    89
      this.addressReg = (ushort)(address + ADDRESS_REGISTER_OFFSET);
moel@78
    90
      this.dataReg = (ushort)(address + DATA_REGISTER_OFFSET);
moel@16
    91
      
moel@16
    92
      // Check vendor id
moel@78
    93
      bool valid;
moel@78
    94
      byte vendorId = ReadByte(VENDOR_ID_REGISTER, out valid);       
moel@78
    95
      if (!valid || vendorId != ITE_VENDOR_ID)
moel@16
    96
        return;
moel@16
    97
moel@16
    98
      // Bit 0x10 of the configuration register should always be 1
moel@78
    99
      if ((ReadByte(CONFIGURATION_REGISTER, out valid) & 0x10) == 0)
moel@78
   100
        return;
moel@78
   101
      if (!valid)
moel@16
   102
        return;
moel@16
   103
moel@130
   104
      voltages = new float?[9];
moel@130
   105
      temperatures = new float?[3];
moel@130
   106
      fans = new float?[5];
moel@170
   107
moel@170
   108
      // The IT8721F uses a 12mV resultion ADC, all others 16mV
moel@170
   109
      if (chip == Chip.IT8721F) {
moel@170
   110
        voltageGain = 0.012f;
moel@170
   111
      } else {
moel@170
   112
        voltageGain = 0.016f;
moel@170
   113
      }
moel@16
   114
    }
moel@16
   115
moel@130
   116
    public Chip Chip { get { return chip; } }
moel@130
   117
    public float?[] Voltages { get { return voltages; } }
moel@130
   118
    public float?[] Temperatures { get { return temperatures; } }
moel@130
   119
    public float?[] Fans { get { return fans; } }
moel@16
   120
moel@130
   121
    public string GetReport() {
moel@16
   122
      StringBuilder r = new StringBuilder();
moel@16
   123
moel@31
   124
      r.AppendLine("LPC " + this.GetType().Name);
moel@16
   125
      r.AppendLine();
moel@16
   126
      r.Append("Chip ID: 0x"); r.AppendLine(chip.ToString("X"));
moel@166
   127
      r.Append("Chip Version: 0x"); r.AppendLine(
moel@166
   128
        version.ToString("X", CultureInfo.InvariantCulture));
moel@166
   129
      r.Append("Base Address: 0x"); r.AppendLine(
moel@166
   130
        address.ToString("X4", CultureInfo.InvariantCulture));
moel@16
   131
      r.AppendLine();
moel@162
   132
moel@163
   133
      if (!WinRing0.WaitIsaBusMutex(100))
moel@162
   134
        return r.ToString();
moel@162
   135
moel@16
   136
      r.AppendLine("Environment Controller Registers");
moel@16
   137
      r.AppendLine();
moel@16
   138
      r.AppendLine("      00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
moel@16
   139
      r.AppendLine();
moel@16
   140
      for (int i = 0; i <= 0xA; i++) {
moel@166
   141
        r.Append(" "); 
moel@166
   142
        r.Append((i << 4).ToString("X2", CultureInfo.InvariantCulture)); 
moel@166
   143
        r.Append("  ");
moel@16
   144
        for (int j = 0; j <= 0xF; j++) {
moel@16
   145
          r.Append(" ");
moel@78
   146
          bool valid;
moel@78
   147
          byte value = ReadByte((byte)((i << 4) | j), out valid);
moel@78
   148
          if (valid)
moel@166
   149
            r.Append(value.ToString("X2", CultureInfo.InvariantCulture));
moel@78
   150
          else
moel@78
   151
            r.Append("??");
moel@16
   152
        }
moel@16
   153
        r.AppendLine();
moel@16
   154
      }
moel@16
   155
      r.AppendLine();
moel@16
   156
moel@162
   157
      WinRing0.ReleaseIsaBusMutex();
moel@162
   158
moel@16
   159
      return r.ToString();
moel@16
   160
    }
moel@16
   161
moel@130
   162
    public void Update() {
moel@163
   163
      if (!WinRing0.WaitIsaBusMutex(10))
moel@162
   164
        return;
moel@16
   165
moel@130
   166
      for (int i = 0; i < voltages.Length; i++) {
moel@78
   167
        bool valid;
moel@170
   168
        
moel@170
   169
        float value = 
moel@170
   170
          voltageGain * ReadByte((byte)(VOLTAGE_BASE_REG + i), out valid);   
moel@170
   171
moel@130
   172
        if (!valid)
moel@130
   173
          continue;
moel@130
   174
        if (value > 0)
moel@130
   175
          voltages[i] = value;  
moel@130
   176
        else
moel@130
   177
          voltages[i] = null;
moel@130
   178
      }
moel@130
   179
moel@130
   180
      for (int i = 0; i < temperatures.Length; i++) {
moel@130
   181
        bool valid;
moel@130
   182
        sbyte value = (sbyte)ReadByte(
moel@130
   183
          (byte)(TEMPERATURE_BASE_REG + i), out valid);
moel@78
   184
        if (!valid)
moel@78
   185
          continue;
moel@78
   186
moel@130
   187
        if (value < sbyte.MaxValue && value > 0)
moel@130
   188
          temperatures[i] = value;
moel@130
   189
        else
moel@130
   190
          temperatures[i] = null;       
moel@16
   191
      }
moel@16
   192
moel@130
   193
      for (int i = 0; i < fans.Length; i++) {
moel@78
   194
        bool valid;
moel@130
   195
        int value = ReadByte(FAN_TACHOMETER_REG[i], out valid);
moel@78
   196
        if (!valid) 
moel@78
   197
          continue;
moel@130
   198
        value |= ReadByte(FAN_TACHOMETER_EXT_REG[i], out valid) << 8;
moel@78
   199
        if (!valid)
moel@78
   200
          continue;
moel@16
   201
moel@36
   202
        if (value > 0x3f) {
moel@130
   203
          fans[i] = (value < 0xffff) ? 1.35e6f / ((value) * 2) : 0;
moel@16
   204
        } else {
moel@130
   205
          fans[i] = null;
moel@16
   206
        }
moel@162
   207
      }
moel@162
   208
moel@162
   209
      WinRing0.ReleaseIsaBusMutex();
moel@126
   210
    }
moel@130
   211
  } 
moel@16
   212
}