Hardware/TBalancer/TBalancer.cs
author moel.mich
Tue, 25 May 2010 18:57:28 +0000
changeset 127 76aaf45a01c7
parent 118 407f98562c3b
child 134 8b3b9b2e28e5
permissions -rw-r--r--
Added a workaround for the "You must keep the stream open for the lifetime of the Image." problem of the Image.FromStream method. This also reduced the overall memory usage (private working set).
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.Text;
moel@1
    43
moel@1
    44
namespace OpenHardwareMonitor.Hardware.TBalancer {
moel@1
    45
  public class TBalancer : IHardware {
moel@1
    46
moel@87
    47
    private int portIndex;
moel@87
    48
    private FT_HANDLE handle;
moel@1
    49
    private Image icon;
moel@33
    50
    private byte protocolVersion;
moel@1
    51
    private Sensor[] digitalTemperatures = new Sensor[8];
moel@1
    52
    private Sensor[] analogTemperatures = new Sensor[4];
moel@57
    53
    private Sensor[] sensorhubTemperatures = new Sensor[6];
moel@57
    54
    private Sensor[] sensorhubFlows = new Sensor[2];
moel@1
    55
    private Sensor[] fans = new Sensor[4];
moel@118
    56
    private Sensor[] controls = new Sensor[4];
moel@57
    57
    private Sensor[] miniNGTemperatures = new Sensor[4];
moel@57
    58
    private Sensor[] miniNGFans = new Sensor[4];
moel@1
    59
    private List<ISensor> active = new List<ISensor>();
moel@1
    60
    private List<ISensor> deactivating = new List<ISensor>();
moel@57
    61
    private int[] primaryData = new int[0];
moel@57
    62
    private int[] alternativeData = new int[0];
moel@1
    63
moel@1
    64
    public const byte STARTFLAG = 100;
moel@57
    65
    public const byte ENDFLAG = 254;
moel@57
    66
moel@57
    67
    private delegate void MethodDelegate();
moel@57
    68
    private MethodDelegate alternativeRequest;    
moel@1
    69
moel@87
    70
    public TBalancer(int portIndex, byte protocolVersion) {
moel@87
    71
      this.portIndex = portIndex;
moel@57
    72
      this.icon = Utilities.EmbeddedResources.GetImage("bigng.png");
moel@33
    73
      this.protocolVersion = protocolVersion;
moel@1
    74
moel@63
    75
      ParameterDescription[] parameter = new ParameterDescription[] {
moel@122
    76
        new ParameterDescription("Offset [°C]", "Temperature offset.", 0)
moel@63
    77
      };
moel@57
    78
      int offset = 0;
moel@57
    79
      for (int i = 0; i < digitalTemperatures.Length; i++)
moel@88
    80
        digitalTemperatures[i] = new Sensor("Digital Sensor " + i,
moel@63
    81
          offset + i, null, SensorType.Temperature, this, parameter);
moel@57
    82
      offset += digitalTemperatures.Length;
moel@57
    83
moel@57
    84
      for (int i = 0; i < analogTemperatures.Length; i++)
moel@88
    85
        analogTemperatures[i] = new Sensor("Analog Sensor " + (i + 1),
moel@63
    86
          offset + i, null, SensorType.Temperature, this, parameter);
moel@57
    87
      offset += analogTemperatures.Length;
moel@57
    88
moel@57
    89
      for (int i = 0; i < sensorhubTemperatures.Length; i++)
moel@88
    90
        sensorhubTemperatures[i] = new Sensor("Sensorhub Sensor " + i,
moel@63
    91
          offset + i, null, SensorType.Temperature, this, parameter);
moel@57
    92
      offset += sensorhubTemperatures.Length;
moel@57
    93
moel@57
    94
      for (int i = 0; i < miniNGTemperatures.Length; i++)
moel@63
    95
        miniNGTemperatures[i] = new Sensor("miniNG #" + (i / 2 + 1) +
moel@88
    96
          " Sensor " + (i % 2 + 1), offset + i, null, SensorType.Temperature, 
moel@63
    97
          this, parameter);
moel@57
    98
      offset += miniNGTemperatures.Length;
moel@57
    99
moel@118
   100
      for (int i = 0; i < sensorhubFlows.Length; i++)
moel@118
   101
        sensorhubFlows[i] = new Sensor("Flowmeter " + (i + 1),
moel@118
   102
          i, null, SensorType.Flow, this, new ParameterDescription[] {
moel@118
   103
            new ParameterDescription("Impulse Rate", 
moel@118
   104
              "The impulse rate of the flowmeter in pulses/L", 509)
moel@118
   105
          });
moel@118
   106
moel@118
   107
      for (int i = 0; i < controls.Length; i++) {
moel@118
   108
        controls[i] = new Sensor("Fan Channel " + i, i, null, 
moel@118
   109
          SensorType.Control, this, null);
moel@118
   110
        ActivateSensor(controls[i]);
moel@118
   111
      }
moel@118
   112
moel@57
   113
      alternativeRequest = new MethodDelegate(DelayedAlternativeRequest);
moel@57
   114
moel@87
   115
      Open();
moel@87
   116
      Update(); 
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@88
   161
            new Sensor("miniNG #" + (number + 1) + " Fan Channel " + (i + 1),
moel@109
   162
            4 + number * 2 + i, maxRPM, SensorType.Fan, this, null);
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@87
   174
        data[i] = FTD2XX.ReadByte(handle);
moel@1
   175
moel@57
   176
      if (data[0] != STARTFLAG) {
moel@87
   177
        FTD2XX.FT_Purge(handle, FT_PURGE.FT_PURGE_RX);   
moel@1
   178
        return;
moel@1
   179
      }
moel@1
   180
moel@73
   181
      if (data[1] == 255 || data[1] == 88) { // 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@86
   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@88
   229
            fans[i] = new Sensor("Fan Channel " + i, 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@118
   235
          float value;
moel@118
   236
          if ((data[136] & (1 << i)) == 0)  // pwm mode
moel@118
   237
            value = 0.02f * data[137 + i];
moel@63
   238
          else // analog mode
moel@118
   239
            value = 0.01f * data[141 + i];
moel@118
   240
          
moel@118
   241
          fans[i].Value = fans[i].Parameters[0].Value * value;
moel@57
   242
          ActivateSensor(fans[i]);
moel@118
   243
moel@118
   244
          controls[i].Value = 100 * value;          
moel@1
   245
        }
moel@1
   246
moel@57
   247
      } else if (data[1] == 253) { // miniNG #1
moel@57
   248
        this.alternativeData = data;
moel@1
   249
moel@57
   250
        ReadminiNG(data, 0);        
moel@57
   251
              
moel@57
   252
        if (data[66] == 252)  // miniNG #2
moel@57
   253
          ReadminiNG(data, 1);
moel@57
   254
      } 
moel@1
   255
    }
moel@1
   256
moel@1
   257
    public Image Icon {
moel@1
   258
      get { return icon; }
moel@1
   259
    }
moel@1
   260
moel@1
   261
    public string Name {
moel@1
   262
      get { return "T-Balancer bigNG"; }
moel@1
   263
    }
moel@1
   264
moel@109
   265
    public Identifier Identifier {
moel@109
   266
      get { return new Identifier("bigng", this.portIndex.ToString()); }
moel@1
   267
    }
moel@1
   268
moel@64
   269
    public IHardware[] SubHardware {
moel@64
   270
      get { return new IHardware[0]; }
moel@64
   271
    }
moel@64
   272
moel@1
   273
    public ISensor[] Sensors {
moel@1
   274
      get { return active.ToArray(); }
moel@1
   275
    }
moel@1
   276
moel@1
   277
    public string GetReport() {
moel@1
   278
      StringBuilder r = new StringBuilder();
moel@1
   279
moel@1
   280
      r.AppendLine("T-Balancer bigNG");
moel@1
   281
      r.AppendLine();
moel@87
   282
      r.Append("Port Index: "); r.AppendLine(portIndex.ToString());
moel@1
   283
      r.AppendLine();
moel@57
   284
moel@57
   285
      r.AppendLine("Primary System Information Answer");
moel@1
   286
      r.AppendLine();
moel@1
   287
      r.AppendLine("       00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
moel@1
   288
      r.AppendLine();
moel@1
   289
      for (int i = 0; i <= 0x11; i++) {
moel@1
   290
        r.Append(" "); r.Append((i << 4).ToString("X3")); r.Append("  ");
moel@1
   291
        for (int j = 0; j <= 0xF; j++) {
moel@1
   292
          int index = ((i << 4) | j);
moel@57
   293
          if (index < primaryData.Length) {
moel@1
   294
            r.Append(" ");
moel@57
   295
            r.Append(primaryData[index].ToString("X2"));
moel@1
   296
          }          
moel@1
   297
        }
moel@1
   298
        r.AppendLine();
moel@1
   299
      }
moel@1
   300
      r.AppendLine();
moel@1
   301
moel@57
   302
      if (alternativeData.Length > 0) {
moel@57
   303
        r.AppendLine("Alternative System Information Answer");
moel@57
   304
        r.AppendLine();
moel@57
   305
        r.AppendLine("       00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
moel@57
   306
        r.AppendLine();
moel@57
   307
        for (int i = 0; i <= 0x11; i++) {
moel@57
   308
          r.Append(" "); r.Append((i << 4).ToString("X3")); r.Append("  ");
moel@57
   309
          for (int j = 0; j <= 0xF; j++) {
moel@57
   310
            int index = ((i << 4) | j);
moel@57
   311
            if (index < alternativeData.Length) {
moel@57
   312
              r.Append(" ");
moel@57
   313
              r.Append(alternativeData[index].ToString("X2"));
moel@57
   314
            }
moel@57
   315
          }
moel@57
   316
          r.AppendLine();
moel@57
   317
        }
moel@57
   318
        r.AppendLine();
moel@57
   319
      }
moel@57
   320
moel@1
   321
      return r.ToString();
moel@1
   322
    }
moel@1
   323
moel@57
   324
    private void DelayedAlternativeRequest() {
moel@87
   325
      System.Threading.Thread.Sleep(500);      
moel@87
   326
      FTD2XX.Write(handle, new byte[] { 0x37 });
moel@57
   327
    }
moel@57
   328
moel@87
   329
    public void Open() {
moel@87
   330
      FTD2XX.FT_Open(portIndex, out handle); 
moel@87
   331
      FTD2XX.FT_SetBaudRate(handle, 19200);
moel@87
   332
      FTD2XX.FT_SetDataCharacteristics(handle, 8, 1, 0);
moel@87
   333
      FTD2XX.FT_SetFlowControl(handle, FT_FLOW_CONTROL.FT_FLOW_RTS_CTS, 0x11,
moel@87
   334
        0x13);
moel@87
   335
      FTD2XX.FT_SetTimeouts(handle, 1000, 1000);
moel@87
   336
      FTD2XX.FT_Purge(handle, FT_PURGE.FT_PURGE_ALL);
moel@87
   337
    }
moel@57
   338
moel@87
   339
    public void Update() {
moel@87
   340
      while (FTD2XX.BytesToRead(handle) >= 285)
moel@87
   341
        ReadData();
moel@87
   342
      if (FTD2XX.BytesToRead(handle) == 1)
moel@87
   343
        FTD2XX.ReadByte(handle);
moel@87
   344
moel@87
   345
      FTD2XX.Write(handle, new byte[] { 0x38 });
moel@87
   346
      alternativeRequest.BeginInvoke(null, null);
moel@1
   347
    }
moel@1
   348
moel@1
   349
    public void Close() {
moel@87
   350
      FTD2XX.FT_Close(handle);
moel@1
   351
    }
moel@1
   352
moel@1
   353
    public event SensorEventHandler SensorAdded;
moel@1
   354
    public event SensorEventHandler SensorRemoved;
moel@110
   355
moel@110
   356
    public void Accept(IVisitor visitor) {
moel@110
   357
      visitor.VisitHardware(this);
moel@110
   358
    }
moel@110
   359
moel@110
   360
    public void Traverse(IVisitor visitor) { }
moel@1
   361
  }
moel@1
   362
}