Hardware/LPC/NCT677X.cs
author moel.mich
Sat, 16 Apr 2011 14:49:47 +0000
changeset 272 037a2d66082f
parent 246 59024371cd50
child 276 04905193c432
permissions -rw-r--r--
A small correction for AMD K8 (family 0Fh) core temperature reading, and fan reading on older revision of IT8712F chips. This should fix Issue 42 and Issue 194.
     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) 2010-2011
    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.Globalization;
    39 using System.Text;
    40 
    41 namespace OpenHardwareMonitor.Hardware.LPC {
    42   internal class NCT677X : ISuperIO {
    43 
    44     private readonly ushort port;
    45     private readonly byte revision;
    46 
    47     private readonly Chip chip;
    48 
    49     private readonly float?[] voltages = new float?[9];
    50     private readonly float?[] temperatures = new float?[3];
    51     private readonly float?[] fans = new float?[0];
    52 
    53     // Hardware Monitor
    54     private const uint ADDRESS_REGISTER_OFFSET = 0x05;
    55     private const uint DATA_REGISTER_OFFSET = 0x06;
    56     private const byte BANK_SELECT_REGISTER = 0x4E;
    57 
    58     private byte ReadByte(ushort address) {
    59       byte bank = (byte)(address >> 8);
    60       byte register = (byte)(address & 0xFF);
    61       Ring0.WriteIoPort(port + ADDRESS_REGISTER_OFFSET, BANK_SELECT_REGISTER);
    62       Ring0.WriteIoPort(port + DATA_REGISTER_OFFSET, bank);
    63       Ring0.WriteIoPort(port + ADDRESS_REGISTER_OFFSET, register);
    64       return Ring0.ReadIoPort(port + DATA_REGISTER_OFFSET);
    65     } 
    66 
    67     // Consts 
    68     private const ushort NUVOTON_VENDOR_ID = 0x5CA3;
    69 
    70     // Hardware Monitor Registers    
    71     private const ushort VENDOR_ID_HIGH_REGISTER = 0x804F;
    72     private const ushort VENDOR_ID_LOW_REGISTER = 0x004F;
    73     private const ushort VOLTAGE_VBAT_REG = 0x0551;
    74 
    75     private readonly ushort[] TEMPERATURE_REG = 
    76       { 0x150, 0x250, 0x27, 0x62B, 0x62C, 0x62D };
    77     private readonly ushort[] TEMPERATURE_HALF_REG = 
    78       { 0x151, 0x251, 0, 0x62E, 0x62E, 0x62E };    
    79     private readonly ushort[] TEMPERATURE_SRC_REG = 
    80       { 0x621, 0x622, 0x623, 0x624, 0x625, 0x626 };
    81     private readonly int[] TEMPERATURE_HALF_BIT = { 7, 7, -1, 0, 1, 2 };
    82     private readonly ushort[] VOLTAGE_REG = 
    83       { 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x550, 0x551 };
    84     private readonly ushort[] FAN_RPM_REG = 
    85       { 0x656, 0x658, 0x65A, 0x65C, 0x65E};
    86 
    87     private readonly int minFanRPM;
    88 
    89     private enum TemperatureSource : byte {
    90       SYSTIN = 1,
    91       CPUTIN = 2,
    92       AUXTIN = 3,
    93       SMBUSMASTER = 4,
    94       PECI0 = 5, 
    95       PECI1 = 6, 
    96       PECI2 = 7,
    97       PECI3 = 8,
    98       PECI4 = 9,
    99       PECI5 = 10,
   100       PECI6 = 11,
   101       PECI7 = 12,
   102       PCH_CHIP_CPU_MAX_TEMP = 13,
   103       PCH_CHIP_TEMP = 14,
   104       PCH_CPU_TEMP = 15,
   105       PCH_MCH_TEMP = 16, 
   106       PCH_DIM0_TEMP = 17,
   107       PCH_DIM1_TEMP = 18,
   108       PCH_DIM2_TEMP = 19,
   109       PCH_DIM3_TEMP = 20
   110     }
   111 
   112     public NCT677X(Chip chip, byte revision, ushort port) {
   113       this.chip = chip;
   114       this.revision = revision;
   115       this.port = port;
   116 
   117       if (!IsNuvotonVendor())
   118         return;
   119 
   120       switch (chip) {
   121         case LPC.Chip.NCT6771F:
   122           fans = new float?[4];
   123           // min value RPM value with 16-bit fan counter
   124           minFanRPM = (int)(1.35e6 / 0xFFFF);
   125           break;
   126         case LPC.Chip.NCT6776F:
   127           fans = new float?[5];
   128           // min value RPM value with 13-bit fan counter
   129           minFanRPM = (int)(1.35e6 / 0x1FFF);
   130           break;        
   131       }
   132     }
   133 
   134     private bool IsNuvotonVendor() {
   135       return ((ReadByte(VENDOR_ID_HIGH_REGISTER) << 8) |
   136         ReadByte(VENDOR_ID_LOW_REGISTER)) == NUVOTON_VENDOR_ID;
   137     }
   138 
   139     public byte? ReadGPIO(int index) {
   140       return null;
   141     }
   142 
   143     public void WriteGPIO(int index, byte value) { }
   144 
   145     public Chip Chip { get { return chip; } }
   146     public float?[] Voltages { get { return voltages; } }
   147     public float?[] Temperatures { get { return temperatures; } }
   148     public float?[] Fans { get { return fans; } }
   149 
   150     public void Update() {
   151       if (!Ring0.WaitIsaBusMutex(10))
   152         return;
   153 
   154       for (int i = 0; i < voltages.Length; i++) {
   155         float value = 0.008f * ReadByte(VOLTAGE_REG[i]);
   156         bool valid = value > 0;
   157 
   158         // check if battery voltage monitor is enabled
   159         if (valid && VOLTAGE_REG[i] == VOLTAGE_VBAT_REG) 
   160           valid = (ReadByte(0x005D) & 0x01) > 0;
   161 
   162         voltages[i] = valid ? value : (float?)null;
   163       }
   164 
   165       for (int i = 0; i < TEMPERATURE_REG.Length; i++) {
   166         int value = ((sbyte)ReadByte(TEMPERATURE_REG[i])) << 1;
   167         if (TEMPERATURE_HALF_BIT[i] > 0) {
   168           value |= ((ReadByte(TEMPERATURE_HALF_REG[i]) >>
   169             TEMPERATURE_HALF_BIT[i]) & 0x1);
   170         }
   171 
   172         TemperatureSource source = (TemperatureSource)
   173           ReadByte(TEMPERATURE_SRC_REG[i]);
   174 
   175         float? temperature = 0.5f * value;
   176         if (temperature > 125 || temperature < -55)
   177           temperature = null;
   178 
   179         switch (source) {
   180           case TemperatureSource.CPUTIN: temperatures[0] = temperature; break;
   181           case TemperatureSource.AUXTIN: temperatures[1] = temperature; break;
   182           case TemperatureSource.SYSTIN: temperatures[2] = temperature; break;
   183         }
   184       }
   185 
   186       for (int i = 0; i < fans.Length; i++) {
   187         byte high = ReadByte(FAN_RPM_REG[i]);
   188         byte low = ReadByte((ushort)(FAN_RPM_REG[i] + 1));
   189         int value = (high << 8) | low;
   190 
   191         fans[i] = value > minFanRPM ? value : 0;
   192       }
   193 
   194       Ring0.ReleaseIsaBusMutex();
   195     }
   196 
   197     public string GetReport() {
   198       StringBuilder r = new StringBuilder();
   199 
   200       r.AppendLine("LPC " + this.GetType().Name);
   201       r.AppendLine();
   202       r.Append("Chip ID: 0x"); r.AppendLine(chip.ToString("X"));
   203       r.Append("Chip revision: 0x");
   204       r.AppendLine(revision.ToString("X", CultureInfo.InvariantCulture));
   205       r.Append("Base Adress: 0x");
   206       r.AppendLine(port.ToString("X4", CultureInfo.InvariantCulture));
   207       r.AppendLine();
   208 
   209       if (!Ring0.WaitIsaBusMutex(100))
   210         return r.ToString();
   211 
   212       ushort[] addresses = new ushort[] { 
   213         0x000, 0x010, 0x020, 0x030, 0x040, 0x050, 0x060, 0x070,
   214         0x100, 0x110, 0x120, 0x130, 0x140, 0x150, 
   215         0x200,        0x220, 0x230, 0x240, 0x250,
   216         0x300,        0x320, 0x330, 0x340, 
   217         0x400, 0x410, 0x420,        0x440, 0x450, 0x460, 
   218         0x500,                             0x550, 
   219         0x600, 0x610 ,0x620, 0x630, 0x640, 0x650, 0x660, 0x670, 
   220         0xA00, 0xA10, 0xA20, 0xA30,        0xA50, 0xA60, 0xA70, 
   221         0xB00, 0xB10, 0xB20, 0xB30,        0xB50, 0xB60, 0xB70, 
   222         0xC00, 0xC10, 0xC20, 0xC30,        0xC50, 0xC60, 0xC70,
   223         0xD00, 0xD10, 0xD20, 0xD30,        0xD50, 0xD60, 
   224         0xE00, 0xE10, 0xE20, 0xE30, 
   225         0xF00, 0xF10, 0xF20, 0xF30};
   226 
   227       r.AppendLine("Hardware Monitor Registers");
   228       r.AppendLine();
   229       r.AppendLine("       00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
   230       r.AppendLine();
   231       foreach (ushort address in addresses) {
   232           r.Append(" ");
   233           r.Append(address.ToString("X3", CultureInfo.InvariantCulture));
   234           r.Append("  ");
   235           for (ushort j = 0; j <= 0xF; j++) {
   236             r.Append(" ");
   237             r.Append(ReadByte((ushort)(address | j)).ToString(
   238               "X2", CultureInfo.InvariantCulture));
   239           }
   240           r.AppendLine();
   241       }
   242       r.AppendLine();
   243 
   244       Ring0.ReleaseIsaBusMutex();
   245 
   246       return r.ToString();
   247     }
   248   }
   249 }