Hardware/TBalancer/TBalancer.cs
author moel.mich
Sun, 28 Feb 2010 16:12:55 +0000
changeset 66 2b537d442924
parent 64 15181001ee61
child 73 8296f0ba87b1
permissions -rw-r--r--
The T-Balancer flow-meter time window seems to be scaled with a factor 4. Added the correction factor to get correct results.
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.Configuration;
moel@1
    41
using System.Drawing;
moel@1
    42
using System.IO;
moel@1
    43
using System.IO.Ports;
moel@1
    44
using System.Text;
moel@1
    45
moel@1
    46
namespace OpenHardwareMonitor.Hardware.TBalancer {
moel@1
    47
  public class TBalancer : IHardware {
moel@1
    48
moel@57
    49
    private string portName;
moel@1
    50
    private Image icon;
moel@1
    51
    private SerialPort serialPort;
moel@33
    52
    private byte protocolVersion;
moel@1
    53
    private Sensor[] digitalTemperatures = new Sensor[8];
moel@1
    54
    private Sensor[] analogTemperatures = new Sensor[4];
moel@57
    55
    private Sensor[] sensorhubTemperatures = new Sensor[6];
moel@57
    56
    private Sensor[] sensorhubFlows = new Sensor[2];
moel@1
    57
    private Sensor[] fans = new Sensor[4];
moel@57
    58
    private Sensor[] miniNGTemperatures = new Sensor[4];
moel@57
    59
    private Sensor[] miniNGFans = new Sensor[4];
moel@1
    60
    private List<ISensor> active = new List<ISensor>();
moel@1
    61
    private List<ISensor> deactivating = new List<ISensor>();
moel@57
    62
    private int[] primaryData = new int[0];
moel@57
    63
    private int[] alternativeData = new int[0];
moel@1
    64
moel@1
    65
    public const byte STARTFLAG = 100;
moel@57
    66
    public const byte ENDFLAG = 254;
moel@57
    67
moel@57
    68
    private delegate void MethodDelegate();
moel@57
    69
    private MethodDelegate alternativeRequest;    
moel@1
    70
moel@33
    71
    public TBalancer(string portName, byte protocolVersion) {
moel@57
    72
      this.portName = portName;
moel@57
    73
      this.icon = Utilities.EmbeddedResources.GetImage("bigng.png");
moel@33
    74
      this.protocolVersion = protocolVersion;
moel@1
    75
moel@63
    76
      ParameterDescription[] parameter = new ParameterDescription[] {
moel@63
    77
        new ParameterDescription("Offset", "Temperature offset.", 0)
moel@63
    78
      };
moel@57
    79
      int offset = 0;
moel@57
    80
      for (int i = 0; i < digitalTemperatures.Length; i++)
moel@57
    81
        digitalTemperatures[i] = new Sensor("Digital Sensor #" + (i + 1),
moel@63
    82
          offset + i, null, SensorType.Temperature, this, parameter);
moel@57
    83
      offset += digitalTemperatures.Length;
moel@57
    84
moel@57
    85
      for (int i = 0; i < analogTemperatures.Length; i++)
moel@57
    86
        analogTemperatures[i] = new Sensor("Analog Sensor #" + (i + 1),
moel@63
    87
          offset + i, null, SensorType.Temperature, this, parameter);
moel@57
    88
      offset += analogTemperatures.Length;
moel@57
    89
moel@57
    90
      for (int i = 0; i < sensorhubTemperatures.Length; i++)
moel@57
    91
        sensorhubTemperatures[i] = new Sensor("Sensorhub Sensor #" + (i + 1),
moel@63
    92
          offset + i, null, SensorType.Temperature, this, parameter);
moel@57
    93
      offset += sensorhubTemperatures.Length;
moel@57
    94
moel@57
    95
      for (int i = 0; i < sensorhubFlows.Length; i++)
moel@57
    96
        sensorhubFlows[i] = new Sensor("Flowmeter #" + (i + 1),
moel@63
    97
          offset + i, null, SensorType.Flow, this, new ParameterDescription[] {
moel@63
    98
            new ParameterDescription("Impulse Rate", 
moel@63
    99
              "The impulse rate of the flowmeter in pulses/L", 509)
moel@63
   100
          });
moel@57
   101
      offset += sensorhubFlows.Length;
moel@57
   102
moel@57
   103
      for (int i = 0; i < miniNGTemperatures.Length; i++)
moel@63
   104
        miniNGTemperatures[i] = new Sensor("miniNG #" + (i / 2 + 1) +
moel@63
   105
          " Sensor #" + (i % 2 + 1), offset + i, null, SensorType.Temperature, 
moel@63
   106
          this, parameter);
moel@57
   107
      offset += miniNGTemperatures.Length;
moel@57
   108
moel@57
   109
      alternativeRequest = new MethodDelegate(DelayedAlternativeRequest);
moel@57
   110
moel@1
   111
      try {
moel@1
   112
        serialPort = new SerialPort(portName, 19200, Parity.None, 8,
moel@1
   113
          StopBits.One);
moel@1
   114
        serialPort.Open();
moel@1
   115
        Update();
moel@57
   116
      } catch (IOException) { }      
moel@1
   117
    }
moel@1
   118
moel@1
   119
    private void ActivateSensor(Sensor sensor) {
moel@1
   120
      deactivating.Remove(sensor);
moel@1
   121
      if (!active.Contains(sensor)) {
moel@1
   122
        active.Add(sensor);
moel@15
   123
        if (SensorAdded != null)
moel@15
   124
          SensorAdded(sensor);
moel@1
   125
      }      
moel@1
   126
    }
moel@1
   127
moel@1
   128
    private void DeactivateSensor(Sensor sensor) {
moel@1
   129
      if (deactivating.Contains(sensor)) {
moel@1
   130
        active.Remove(sensor);
moel@1
   131
        deactivating.Remove(sensor);
moel@15
   132
        if (SensorRemoved != null)
moel@15
   133
          SensorRemoved(sensor);
moel@1
   134
      } else if (active.Contains(sensor)) {
moel@1
   135
        deactivating.Add(sensor);
moel@1
   136
      }     
moel@1
   137
    }
moel@1
   138
moel@57
   139
    private void ReadminiNG(int[] data, int number) {
moel@57
   140
      int offset = 1 + number * 65;
moel@57
   141
moel@57
   142
      if (data[offset + 61] != ENDFLAG)
moel@57
   143
        return;
moel@57
   144
moel@57
   145
      for (int i = 0; i < 2; i++) {
moel@57
   146
        Sensor sensor = miniNGTemperatures[number * 2 + i];
moel@57
   147
        if (data[offset + 7 + i] > 0) {
moel@63
   148
          sensor.Value = 0.5f * data[offset + 7 + i] + 
moel@63
   149
            sensor.Parameters[0].Value;
moel@57
   150
          ActivateSensor(sensor);
moel@57
   151
        } else {
moel@57
   152
          DeactivateSensor(sensor);
moel@57
   153
        }
moel@57
   154
      }
moel@57
   155
moel@57
   156
      for (int i = 0; i < 2; i++) {
moel@57
   157
        float maxRPM = 20.0f * data[offset + 44 + 2 * i];
moel@57
   158
moel@57
   159
        if (miniNGFans[number * 2 + i] == null)
moel@57
   160
          miniNGFans[number * 2 + i] = 
moel@57
   161
            new Sensor("miniNG #" + (number + 1) + " Fan #" + (i + 1),
moel@57
   162
            4 + number * 2 + i, maxRPM, SensorType.Fan, this);
moel@57
   163
        
moel@57
   164
        Sensor sensor = miniNGFans[number * 2 + i];
moel@57
   165
moel@57
   166
        sensor.Value = 20.0f * data[offset + 43 + 2 * i];
moel@57
   167
        ActivateSensor(sensor);
moel@57
   168
      }
moel@57
   169
    }
moel@57
   170
moel@1
   171
    private void ReadData() {
moel@1
   172
      int[] data = new int[285];
moel@1
   173
      for (int i = 0; i < data.Length; i++)
moel@1
   174
        data[i] = serialPort.ReadByte();
moel@1
   175
moel@57
   176
      if (data[0] != STARTFLAG) {
moel@1
   177
        serialPort.DiscardInBuffer();   
moel@1
   178
        return;
moel@1
   179
      }
moel@1
   180
moel@57
   181
      if (data[1] == 255) { // bigNG
moel@57
   182
moel@57
   183
        if (data[274] != protocolVersion) 
moel@57
   184
          return;
moel@57
   185
moel@57
   186
        this.primaryData = data;
moel@57
   187
moel@57
   188
        for (int i = 0; i < digitalTemperatures.Length; i++)
moel@57
   189
          if (data[238 + i] > 0) {
moel@63
   190
            digitalTemperatures[i].Value = 0.5f * data[238 + i] + 
moel@63
   191
              digitalTemperatures[i].Parameters[0].Value;
moel@57
   192
            ActivateSensor(digitalTemperatures[i]);
moel@57
   193
          } else {
moel@57
   194
            DeactivateSensor(digitalTemperatures[i]);
moel@57
   195
          }
moel@57
   196
moel@57
   197
        for (int i = 0; i < analogTemperatures.Length; i++)
moel@57
   198
          if (data[260 + i] > 0) {
moel@63
   199
            analogTemperatures[i].Value = 0.5f * data[260 + i] +
moel@63
   200
              analogTemperatures[i].Parameters[0].Value;
moel@57
   201
            ActivateSensor(analogTemperatures[i]);
moel@57
   202
          } else {
moel@57
   203
            DeactivateSensor(analogTemperatures[i]);
moel@57
   204
          }
moel@57
   205
moel@57
   206
        for (int i = 0; i < sensorhubTemperatures.Length; i++)
moel@57
   207
          if (data[246 + i] > 0) {
moel@63
   208
            sensorhubTemperatures[i].Value = 0.5f * data[246 + i] +
moel@63
   209
              sensorhubTemperatures[i].Parameters[0].Value;
moel@57
   210
            ActivateSensor(sensorhubTemperatures[i]);
moel@57
   211
          } else {
moel@57
   212
            DeactivateSensor(sensorhubTemperatures[i]);
moel@57
   213
          }
moel@57
   214
moel@57
   215
        for (int i = 0; i < sensorhubFlows.Length; i++)
moel@57
   216
          if (data[231 + i] > 0 && data[234] > 0) {
moel@66
   217
            float pulsesPerSecond = (data[231 + i] * 4.0f) / data[234];
moel@63
   218
            float pulsesPerLiter = sensorhubFlows[i].Parameters[0].Value;
moel@57
   219
            sensorhubFlows[i].Value = pulsesPerSecond * 3600 / pulsesPerLiter;
moel@57
   220
            ActivateSensor(sensorhubFlows[i]);
moel@57
   221
          } else {
moel@57
   222
            DeactivateSensor(sensorhubFlows[i]);
moel@57
   223
          }
moel@57
   224
moel@57
   225
        for (int i = 0; i < fans.Length; i++) {
moel@57
   226
          float maxRPM = 11.5f * ((data[149 + 2 * i] << 8) | data[148 + 2 * i]);
moel@57
   227
moel@57
   228
          if (fans[i] == null)
moel@57
   229
            fans[i] = new Sensor("Fan #" + (i + 1), i, maxRPM, SensorType.Fan,
moel@63
   230
              this, new ParameterDescription[] {
moel@63
   231
                new ParameterDescription("MaxRPM", 
moel@63
   232
                  "Maximum revolutions per minute (RPM) of the fan.", maxRPM)
moel@63
   233
              });
moel@57
   234
moel@63
   235
          if ((data[136] & (1 << i)) == 0) // pwm mode
moel@63
   236
            fans[i].Value = fans[i].Parameters[0].Value * 0.01f * data[156 + i];
moel@63
   237
          else // analog mode
moel@63
   238
            fans[i].Value = fans[i].Parameters[0].Value * 0.01f * data[141 + i]; 
moel@57
   239
          ActivateSensor(fans[i]);
moel@1
   240
        }
moel@1
   241
moel@57
   242
      } else if (data[1] == 253) { // miniNG #1
moel@57
   243
        this.alternativeData = data;
moel@1
   244
moel@57
   245
        ReadminiNG(data, 0);        
moel@57
   246
              
moel@57
   247
        if (data[66] == 252)  // miniNG #2
moel@57
   248
          ReadminiNG(data, 1);
moel@57
   249
      } 
moel@1
   250
    }
moel@1
   251
moel@1
   252
    public Image Icon {
moel@1
   253
      get { return icon; }
moel@1
   254
    }
moel@1
   255
moel@1
   256
    public string Name {
moel@1
   257
      get { return "T-Balancer bigNG"; }
moel@1
   258
    }
moel@1
   259
moel@1
   260
    public string Identifier {
moel@1
   261
      get { return "/bigng/" + 
moel@57
   262
        this.portName.TrimStart(new char[]{'/'}).ToLower(); }
moel@1
   263
    }
moel@1
   264
moel@64
   265
    public IHardware[] SubHardware {
moel@64
   266
      get { return new IHardware[0]; }
moel@64
   267
    }
moel@64
   268
moel@1
   269
    public ISensor[] Sensors {
moel@1
   270
      get { return active.ToArray(); }
moel@1
   271
    }
moel@1
   272
moel@1
   273
    public string GetReport() {
moel@1
   274
      StringBuilder r = new StringBuilder();
moel@1
   275
moel@1
   276
      r.AppendLine("T-Balancer bigNG");
moel@1
   277
      r.AppendLine();
moel@1
   278
      r.Append("Port Name: "); r.AppendLine(serialPort.PortName);
moel@1
   279
      r.AppendLine();
moel@57
   280
moel@57
   281
      r.AppendLine("Primary System Information Answer");
moel@1
   282
      r.AppendLine();
moel@1
   283
      r.AppendLine("       00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
moel@1
   284
      r.AppendLine();
moel@1
   285
      for (int i = 0; i <= 0x11; i++) {
moel@1
   286
        r.Append(" "); r.Append((i << 4).ToString("X3")); r.Append("  ");
moel@1
   287
        for (int j = 0; j <= 0xF; j++) {
moel@1
   288
          int index = ((i << 4) | j);
moel@57
   289
          if (index < primaryData.Length) {
moel@1
   290
            r.Append(" ");
moel@57
   291
            r.Append(primaryData[index].ToString("X2"));
moel@1
   292
          }          
moel@1
   293
        }
moel@1
   294
        r.AppendLine();
moel@1
   295
      }
moel@1
   296
      r.AppendLine();
moel@1
   297
moel@57
   298
      if (alternativeData.Length > 0) {
moel@57
   299
        r.AppendLine("Alternative System Information Answer");
moel@57
   300
        r.AppendLine();
moel@57
   301
        r.AppendLine("       00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
moel@57
   302
        r.AppendLine();
moel@57
   303
        for (int i = 0; i <= 0x11; i++) {
moel@57
   304
          r.Append(" "); r.Append((i << 4).ToString("X3")); r.Append("  ");
moel@57
   305
          for (int j = 0; j <= 0xF; j++) {
moel@57
   306
            int index = ((i << 4) | j);
moel@57
   307
            if (index < alternativeData.Length) {
moel@57
   308
              r.Append(" ");
moel@57
   309
              r.Append(alternativeData[index].ToString("X2"));
moel@57
   310
            }
moel@57
   311
          }
moel@57
   312
          r.AppendLine();
moel@57
   313
        }
moel@57
   314
        r.AppendLine();
moel@57
   315
      }
moel@57
   316
moel@1
   317
      return r.ToString();
moel@1
   318
    }
moel@1
   319
moel@57
   320
    private void DelayedAlternativeRequest() {
moel@57
   321
      System.Threading.Thread.Sleep(500);
moel@57
   322
      try {
moel@57
   323
        if (serialPort.IsOpen)
moel@57
   324
          serialPort.Write(new byte[] { 0x37 }, 0, 1);
moel@57
   325
      } catch (Exception) { }
moel@57
   326
    }
moel@57
   327
moel@1
   328
    public void Update() {
moel@1
   329
      while (serialPort.BytesToRead >= 285)
moel@1
   330
        ReadData();
moel@57
   331
      if (serialPort.BytesToRead == 1)
moel@57
   332
        serialPort.ReadByte();
moel@57
   333
moel@1
   334
      serialPort.Write(new byte[] { 0x38 }, 0, 1);
moel@57
   335
      alternativeRequest.BeginInvoke(null, null);
moel@1
   336
    }
moel@1
   337
moel@1
   338
    public void Close() {
moel@1
   339
      serialPort.Close();
moel@1
   340
    }
moel@1
   341
moel@1
   342
    public event SensorEventHandler SensorAdded;
moel@1
   343
    public event SensorEventHandler SensorRemoved;
moel@1
   344
  }
moel@1
   345
}