NVAPI interop struct packing improved.
3 Version: MPL 1.1/GPL 2.0/LGPL 2.1
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
9 http://www.mozilla.org/MPL/
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.
15 The Original Code is the Open Hardware Monitor code.
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) 2009-2010
20 the Initial Developer. All Rights Reserved.
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.
39 using System.Collections.Generic;
40 using System.Configuration;
43 using System.IO.Ports;
46 namespace OpenHardwareMonitor.Hardware.TBalancer {
47 public class TBalancer : IHardware {
49 private string portName;
51 private SerialPort serialPort;
52 private byte protocolVersion;
53 private Sensor[] digitalTemperatures = new Sensor[8];
54 private Sensor[] analogTemperatures = new Sensor[4];
55 private Sensor[] sensorhubTemperatures = new Sensor[6];
56 private Sensor[] sensorhubFlows = new Sensor[2];
57 private Sensor[] fans = new Sensor[4];
58 private Sensor[] miniNGTemperatures = new Sensor[4];
59 private Sensor[] miniNGFans = new Sensor[4];
60 private List<ISensor> active = new List<ISensor>();
61 private List<ISensor> deactivating = new List<ISensor>();
62 private int[] primaryData = new int[0];
63 private int[] alternativeData = new int[0];
65 public const byte STARTFLAG = 100;
66 public const byte ENDFLAG = 254;
68 private delegate void MethodDelegate();
69 private MethodDelegate alternativeRequest;
71 public TBalancer(string portName, byte protocolVersion) {
72 this.portName = portName;
73 this.icon = Utilities.EmbeddedResources.GetImage("bigng.png");
74 this.protocolVersion = protocolVersion;
77 for (int i = 0; i < digitalTemperatures.Length; i++)
78 digitalTemperatures[i] = new Sensor("Digital Sensor #" + (i + 1),
79 offset + i, SensorType.Temperature, this);
80 offset += digitalTemperatures.Length;
82 for (int i = 0; i < analogTemperatures.Length; i++)
83 analogTemperatures[i] = new Sensor("Analog Sensor #" + (i + 1),
84 offset + i, SensorType.Temperature, this);
85 offset += analogTemperatures.Length;
87 for (int i = 0; i < sensorhubTemperatures.Length; i++)
88 sensorhubTemperatures[i] = new Sensor("Sensorhub Sensor #" + (i + 1),
89 offset + i, SensorType.Temperature, this);
90 offset += sensorhubTemperatures.Length;
92 for (int i = 0; i < sensorhubFlows.Length; i++)
93 sensorhubFlows[i] = new Sensor("Flowmeter #" + (i + 1),
94 offset + i, SensorType.Flow, this);
95 offset += sensorhubFlows.Length;
97 for (int i = 0; i < miniNGTemperatures.Length; i++)
98 miniNGTemperatures[i] = new Sensor("miniNG #" + (i / 2 + 1) +
99 " Sensor #" + (i % 2 + 1), offset + i, SensorType.Temperature, this);
100 offset += miniNGTemperatures.Length;
102 alternativeRequest = new MethodDelegate(DelayedAlternativeRequest);
105 serialPort = new SerialPort(portName, 19200, Parity.None, 8,
109 } catch (IOException) { }
112 private void ActivateSensor(Sensor sensor) {
113 deactivating.Remove(sensor);
114 if (!active.Contains(sensor)) {
116 if (SensorAdded != null)
121 private void DeactivateSensor(Sensor sensor) {
122 if (deactivating.Contains(sensor)) {
123 active.Remove(sensor);
124 deactivating.Remove(sensor);
125 if (SensorRemoved != null)
126 SensorRemoved(sensor);
127 } else if (active.Contains(sensor)) {
128 deactivating.Add(sensor);
132 private void ReadminiNG(int[] data, int number) {
133 int offset = 1 + number * 65;
135 if (data[offset + 61] != ENDFLAG)
138 for (int i = 0; i < 2; i++) {
139 Sensor sensor = miniNGTemperatures[number * 2 + i];
140 if (data[offset + 7 + i] > 0) {
141 sensor.Value = 0.5f * data[offset + 7 + i];
142 ActivateSensor(sensor);
144 DeactivateSensor(sensor);
148 for (int i = 0; i < 2; i++) {
149 float maxRPM = 20.0f * data[offset + 44 + 2 * i];
151 if (miniNGFans[number * 2 + i] == null)
152 miniNGFans[number * 2 + i] =
153 new Sensor("miniNG #" + (number + 1) + " Fan #" + (i + 1),
154 4 + number * 2 + i, maxRPM, SensorType.Fan, this);
156 Sensor sensor = miniNGFans[number * 2 + i];
158 sensor.Value = 20.0f * data[offset + 43 + 2 * i];
159 ActivateSensor(sensor);
163 private void ReadData() {
164 int[] data = new int[285];
165 for (int i = 0; i < data.Length; i++)
166 data[i] = serialPort.ReadByte();
168 if (data[0] != STARTFLAG) {
169 serialPort.DiscardInBuffer();
173 if (data[1] == 255) { // bigNG
175 if (data[274] != protocolVersion)
178 this.primaryData = data;
180 for (int i = 0; i < digitalTemperatures.Length; i++)
181 if (data[238 + i] > 0) {
182 digitalTemperatures[i].Value = 0.5f * data[238 + i];
183 ActivateSensor(digitalTemperatures[i]);
185 DeactivateSensor(digitalTemperatures[i]);
188 for (int i = 0; i < analogTemperatures.Length; i++)
189 if (data[260 + i] > 0) {
190 analogTemperatures[i].Value = 0.5f * data[260 + i];
191 ActivateSensor(analogTemperatures[i]);
193 DeactivateSensor(analogTemperatures[i]);
196 for (int i = 0; i < sensorhubTemperatures.Length; i++)
197 if (data[246 + i] > 0) {
198 sensorhubTemperatures[i].Value = 0.5f * data[246 + i];
199 ActivateSensor(sensorhubTemperatures[i]);
201 DeactivateSensor(sensorhubTemperatures[i]);
204 for (int i = 0; i < sensorhubFlows.Length; i++)
205 if (data[231 + i] > 0 && data[234] > 0) {
206 float pulsesPerSecond = ((float)data[231 + i]) / data[234];
207 const float pulsesPerLiter = 509;
208 sensorhubFlows[i].Value = pulsesPerSecond * 3600 / pulsesPerLiter;
209 ActivateSensor(sensorhubFlows[i]);
211 DeactivateSensor(sensorhubFlows[i]);
214 for (int i = 0; i < fans.Length; i++) {
215 float maxRPM = 11.5f * ((data[149 + 2 * i] << 8) | data[148 + 2 * i]);
218 fans[i] = new Sensor("Fan #" + (i + 1), i, maxRPM, SensorType.Fan,
221 if ((data[136] & (1 << i)) == 0)
222 fans[i].Value = maxRPM * 0.01f * data[156 + i]; // pwm mode
224 fans[i].Value = maxRPM * 0.01f * data[141 + i]; // analog mode
225 ActivateSensor(fans[i]);
228 } else if (data[1] == 253) { // miniNG #1
229 this.alternativeData = data;
233 if (data[66] == 252) // miniNG #2
243 get { return "T-Balancer bigNG"; }
246 public string Identifier {
247 get { return "/bigng/" +
248 this.portName.TrimStart(new char[]{'/'}).ToLower(); }
251 public ISensor[] Sensors {
252 get { return active.ToArray(); }
255 public string GetReport() {
256 StringBuilder r = new StringBuilder();
258 r.AppendLine("T-Balancer bigNG");
260 r.Append("Port Name: "); r.AppendLine(serialPort.PortName);
263 r.AppendLine("Primary System Information Answer");
265 r.AppendLine(" 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
267 for (int i = 0; i <= 0x11; i++) {
268 r.Append(" "); r.Append((i << 4).ToString("X3")); r.Append(" ");
269 for (int j = 0; j <= 0xF; j++) {
270 int index = ((i << 4) | j);
271 if (index < primaryData.Length) {
273 r.Append(primaryData[index].ToString("X2"));
280 if (alternativeData.Length > 0) {
281 r.AppendLine("Alternative System Information Answer");
283 r.AppendLine(" 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
285 for (int i = 0; i <= 0x11; i++) {
286 r.Append(" "); r.Append((i << 4).ToString("X3")); r.Append(" ");
287 for (int j = 0; j <= 0xF; j++) {
288 int index = ((i << 4) | j);
289 if (index < alternativeData.Length) {
291 r.Append(alternativeData[index].ToString("X2"));
302 private void DelayedAlternativeRequest() {
303 System.Threading.Thread.Sleep(500);
305 if (serialPort.IsOpen)
306 serialPort.Write(new byte[] { 0x37 }, 0, 1);
307 } catch (Exception) { }
310 public void Update() {
311 while (serialPort.BytesToRead >= 285)
313 if (serialPort.BytesToRead == 1)
314 serialPort.ReadByte();
316 serialPort.Write(new byte[] { 0x38 }, 0, 1);
317 alternativeRequest.BeginInvoke(null, null);
320 public void Close() {
324 public event SensorEventHandler SensorAdded;
325 public event SensorEventHandler SensorRemoved;