Instrument Neutral Distributed Interface INDI  2.0.2
vantage.cpp
Go to the documentation of this file.
1 /*******************************************************************************
2  Copyright(c) 2015 Jasem Mutlaq. All rights reserved.
3 
4  INDI Davis Vantage Pro/Pro2/Vue Weather Driver
5 
6  This program is free software; you can redistribute it and/or modify it
7  under the terms of the GNU General Public License as published by the Free
8  Software Foundation; either version 2 of the License, or (at your option)
9  any later version.
10 
11  This program is distributed in the hope that it will be useful, but WITHOUT
12  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13  FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
14  more details.
15 
16  You should have received a copy of the GNU Library General Public License
17  along with this library; see the file COPYING.LIB. If not, write to
18  the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  Boston, MA 02110-1301, USA.
20 
21  The full GNU General Public License is included in this distribution in the
22  file called LICENSE.
23 *******************************************************************************/
24 
25 #include "vantage.h"
26 
27 #include "indicom.h"
29 
30 #include <memory>
31 #include <cstring>
32 #include <termios.h>
33 
34 #define VANTAGE_CMD 8
35 #define VANTAGE_RES 128
36 #define VANTAGE_TIMEOUT 2
37 
38 static uint16_t crc_table[] =
39 {
40  0x0000, 0x1021, 0x2042, 0x3063, 0x4084, 0x50a5, 0x60c6, 0x70e7, 0x8108, 0x9129, 0xa14a, 0xb16b, 0xc18c, 0xd1ad,
41  0xe1ce, 0xf1ef, 0x1231, 0x0210, 0x3273, 0x2252, 0x52b5, 0x4294, 0x72f7, 0x62d6, 0x9339, 0x8318, 0xb37b, 0xa35a,
42  0xd3bd, 0xc39c, 0xf3ff, 0xe3de, 0x2462, 0x3443, 0x0420, 0x1401, 0x64e6, 0x74c7, 0x44a4, 0x5485, 0xa56a, 0xb54b,
43  0x8528, 0x9509, 0xe5ee, 0xf5cf, 0xc5ac, 0xd58d, 0x3653, 0x2672, 0x1611, 0x0630, 0x76d7, 0x66f6, 0x5695, 0x46b4,
44  0xb75b, 0xa77a, 0x9719, 0x8738, 0xf7df, 0xe7fe, 0xd79d, 0xc7bc, 0x48c4, 0x58e5, 0x6886, 0x78a7, 0x0840, 0x1861,
45  0x2802, 0x3823, 0xc9cc, 0xd9ed, 0xe98e, 0xf9af, 0x8948, 0x9969, 0xa90a, 0xb92b, 0x5af5, 0x4ad4, 0x7ab7, 0x6a96,
46  0x1a71, 0x0a50, 0x3a33, 0x2a12, 0xdbfd, 0xcbdc, 0xfbbf, 0xeb9e, 0x9b79, 0x8b58, 0xbb3b, 0xab1a, 0x6ca6, 0x7c87,
47  0x4ce4, 0x5cc5, 0x2c22, 0x3c03, 0x0c60, 0x1c41, 0xedae, 0xfd8f, 0xcdec, 0xddcd, 0xad2a, 0xbd0b, 0x8d68, 0x9d49,
48  0x7e97, 0x6eb6, 0x5ed5, 0x4ef4, 0x3e13, 0x2e32, 0x1e51, 0x0e70, 0xff9f, 0xefbe, 0xdfdd, 0xcffc, 0xbf1b, 0xaf3a,
49  0x9f59, 0x8f78, 0x9188, 0x81a9, 0xb1ca, 0xa1eb, 0xd10c, 0xc12d, 0xf14e, 0xe16f, 0x1080, 0x00a1, 0x30c2, 0x20e3,
50  0x5004, 0x4025, 0x7046, 0x6067, 0x83b9, 0x9398, 0xa3fb, 0xb3da, 0xc33d, 0xd31c, 0xe37f, 0xf35e, 0x02b1, 0x1290,
51  0x22f3, 0x32d2, 0x4235, 0x5214, 0x6277, 0x7256, 0xb5ea, 0xa5cb, 0x95a8, 0x8589, 0xf56e, 0xe54f, 0xd52c, 0xc50d,
52  0x34e2, 0x24c3, 0x14a0, 0x0481, 0x7466, 0x6447, 0x5424, 0x4405, 0xa7db, 0xb7fa, 0x8799, 0x97b8, 0xe75f, 0xf77e,
53  0xc71d, 0xd73c, 0x26d3, 0x36f2, 0x0691, 0x16b0, 0x6657, 0x7676, 0x4615, 0x5634, 0xd94c, 0xc96d, 0xf90e, 0xe92f,
54  0x99c8, 0x89e9, 0xb98a, 0xa9ab, 0x5844, 0x4865, 0x7806, 0x6827, 0x18c0, 0x08e1, 0x3882, 0x28a3, 0xcb7d, 0xdb5c,
55  0xeb3f, 0xfb1e, 0x8bf9, 0x9bd8, 0xabbb, 0xbb9a, 0x4a75, 0x5a54, 0x6a37, 0x7a16, 0x0af1, 0x1ad0, 0x2ab3, 0x3a92,
56  0xfd2e, 0xed0f, 0xdd6c, 0xcd4d, 0xbdaa, 0xad8b, 0x9de8, 0x8dc9, 0x7c26, 0x6c07, 0x5c64, 0x4c45, 0x3ca2, 0x2c83,
57  0x1ce0, 0x0cc1, 0xef1f, 0xff3e, 0xcf5d, 0xdf7c, 0xaf9b, 0xbfba, 0x8fd9, 0x9ff8, 0x6e17, 0x7e36, 0x4e55, 0x5e74,
58  0x2e93, 0x3eb2, 0x0ed1, 0x1ef0,
59 };
60 
61 uint16_t crc16(const void *c_ptr, size_t len)
62 {
63  const uint8_t *c = (uint8_t *)c_ptr;
64  uint16_t crc = 0;
65 
66  while (len--)
67  crc = crc_table[((crc >> 8) ^ *c++)] ^ (crc << 8);
68 
69  return crc;
70 }
71 
72 // We declare an auto pointer to Vantage.
73 std::unique_ptr<Vantage> vantage(new Vantage());
74 
76 {
77  setVersion(1, 0);
78 }
79 
81 {
82  return (const char *)"Vantage";
83 }
84 
86 {
88 
89  addParameter("WEATHER_FORECAST", "Forecast", 0, 0, 15);
90  addParameter("WEATHER_TEMPERATURE", "Temperature (C)", -10, 30, 15);
91  addParameter("WEATHER_BAROMETER", "Barometer (mbar)", 20, 32.5, 15);
92  addParameter("WEATHER_WIND_SPEED", "Wind (kph)", 0, 20, 15);
93  addParameter("WEATHER_WIND_DIRECTION", "Wind Direction", 0, 360, 15);
94  addParameter("WEATHER_HUMIDITY", "Humidity %", 0, 100, 15);
95  addParameter("WEATHER_RAIN_RATE", "Rain (mm/h)", 0, 0, 15);
96  addParameter("WEATHER_SOLAR_RADIATION", "Solar Radiation (w/m^2)", 0, 10000, 15);
97 
98  setCriticalParameter("WEATHER_FORECAST");
99  setCriticalParameter("WEATHER_TEMPERATURE");
100  setCriticalParameter("WEATHER_WIND_SPEED");
101  setCriticalParameter("WEATHER_RAIN_RATE");
102 
103  addDebugControl();
104 
106 
107  return true;
108 }
109 
111 {
112  return ack();
113 }
114 
116 {
117  int nbytes_written = 0, nbytes_read = 0, rc = -1;
118  char errstr[MAXRBUF];
119  char command[VANTAGE_CMD];
120  char response[VANTAGE_RES];
121 
122  if (!wakeup())
123  return IPS_ALERT;
124 
125  strncpy(command, "LOOP 1", VANTAGE_CMD);
126  command[6] = 0;
127 
128  tcflush(PortFD, TCIOFLUSH);
129 
130  LOGF_DEBUG("CMD (%s)", command);
131 
132  command[6] = 0xA;
133 
134  if ((rc = tty_write(PortFD, command, 7, &nbytes_written)) != TTY_OK)
135  {
136  tty_error_msg(rc, errstr, MAXRBUF);
137  LOGF_ERROR("Loop error: %s.", errstr);
138  return IPS_ALERT;
139  }
140 
141  if ((rc = tty_read(PortFD, response, 1, VANTAGE_TIMEOUT, &nbytes_read)) != TTY_OK)
142  {
143  tty_error_msg(rc, errstr, MAXRBUF);
144  LOGF_ERROR("Loop error: %s.", errstr);
145  return IPS_ALERT;
146  }
147 
148  if (response[0] != 0x06)
149  {
150  LOGF_ERROR("Expecting 0x06, received %#X", response[0]);
151  return IPS_ALERT;
152  }
153 
154  if ((rc = tty_read(PortFD, response, 99, VANTAGE_TIMEOUT, &nbytes_read)) != TTY_OK)
155  {
156  tty_error_msg(rc, errstr, MAXRBUF);
157  LOGF_ERROR("Loop error: %s.", errstr);
158  return IPS_ALERT;
159  }
160 
161  uint16_t crc = crc16(response, 99);
162 
163  if (crc != 0)
164  {
165  LOG_ERROR("CRC check failed.");
166  return IPS_ALERT;
167  }
168 
169  uint8_t *loopData = (uint8_t *)response;
170 
171  LOGF_DEBUG("Packet Type (%d)", loopData[4]);
172 
173  uint8_t forecastValue = loopData[89];
174 
175  LOGF_DEBUG("Raw Forecast (%d)", forecastValue);
176 
177  switch (forecastValue)
178  {
179  // Clear
180  case 0x08:
181  LOG_INFO("Forecast: Mostly Clear.");
182  setParameterValue("WEATHER_FORECAST", 0);
183  break;
184 
185  case 0x06:
186  LOG_INFO("Forecast: Partly Cloudy.");
187  setParameterValue("WEATHER_FORECAST", 1);
188  break;
189 
190  case 0x02:
191  LOG_INFO("Forecast: Mostly Cloudy.");
192  setParameterValue("WEATHER_FORECAST", 2);
193  break;
194 
195  case 0x03:
196  LOG_INFO("Forecast: Mostly Cloudy. Rain within 12 hours.");
197  setParameterValue("WEATHER_FORECAST", 2);
198  break;
199 
200  case 0x12:
201  LOG_INFO("Forecast: Mostly Cloudy. Snow within 12 hours.");
202  setParameterValue("WEATHER_FORECAST", 2);
203  break;
204 
205  case 0x13:
206  LOG_INFO("Forecast: Mostly Cloudy. Rain or Snow within 12 hours.");
207  setParameterValue("WEATHER_FORECAST", 2);
208  break;
209 
210  case 0x07:
211  LOG_INFO("Forecast: Partly Cloudy. Rain within 12 hours.");
212  setParameterValue("WEATHER_FORECAST", 1);
213  break;
214 
215  case 0x16:
216  LOG_INFO("Forecast: Partly Cloudy. Snow within 12 hours.");
217  setParameterValue("WEATHER_FORECAST", 1);
218  break;
219 
220  case 0x17:
221  LOG_INFO("Forecast: Partly Cloudy. Rain or Snow within 12 hours.");
222  setParameterValue("WEATHER_FORECAST", 1);
223  break;
224  }
225 
226  // Inside Temperature
227  uint16_t temperatureValue = loopData[10] << 8 | loopData[9];
228 
229  setParameterValue("WEATHER_TEMPERATURE", ((temperatureValue / 10.0) - 32) / 1.8);
230 
231  LOGF_DEBUG("Raw Temperature (%d) [%#4X %#4X]", temperatureValue, loopData[9], loopData[10]);
232 
233  // Inside Humidity
234  uint8_t humidityValue = loopData[11];
235 
236  setParameterValue("WEATHER_HUMIDITY", humidityValue );
237 
238  LOGF_DEBUG("Raw Inside Humidity (%d) [%#X4]", humidityValue, loopData[11]);
239 
240  // Barometer
241  uint16_t barometerValue = loopData[8] << 8 | loopData[7];
242 
243  setParameterValue("WEATHER_BAROMETER", (barometerValue / 1000.0) * 33.8639);
244 
245  LOGF_DEBUG("Raw Barometer (%d) [%#4X %#4X]", barometerValue, loopData[7], loopData[8]);
246 
247  // Wind Speed
248  uint8_t windValue = loopData[14];
249 
250  LOGF_DEBUG("Raw Wind Speed (%d) [%#4X]", windValue, loopData[14]);
251 
252  setParameterValue("WEATHER_WIND_SPEED", windValue / 0.62137);
253 
254  // Wind Direction
255  uint16_t windDir = loopData[17] << 8 | loopData[16];
256 
257  LOGF_DEBUG("Raw Wind Direction (%d) [%#4X,%#4X]", windDir, loopData[16], loopData[17]);
258 
259  setParameterValue("WEATHER_WIND_DIRECTION", windDir);
260 
261  // Rain Rate
262  uint16_t rainRate = loopData[42] << 8 | loopData[41];
263 
264  LOGF_DEBUG("Raw Rain Rate (%d) [%#4X,%#4X]", rainRate, loopData[41], loopData[42]);
265 
266  setParameterValue("WEATHER_RAIN_RATE", rainRate / (100 * 0.039370));
267 
268  // Solar Radiation
269  uint16_t solarRadiation = loopData[45] << 8 | loopData[44];
270 
271  LOGF_DEBUG("Raw Solar Radiation (%d) [%#4X,%#4X]", solarRadiation, loopData[44], loopData[45]);
272 
273  if (solarRadiation == 32767)
274  solarRadiation = 0;
275 
276  setParameterValue("WEATHER_SOLAR_RADIATION", solarRadiation);
277 
278  return IPS_OK;
279 }
280 
281 bool Vantage::wakeup()
282 {
283  int nbytes_written = 0, nbytes_read = 0, rc = -1;
284  char errstr[MAXRBUF];
285  char command[VANTAGE_CMD];
286  char response[VANTAGE_RES];
287 
288  tcflush(PortFD, TCIOFLUSH);
289 
290  command[0] = 0xA;
291 
292  for (int i = 0; i < 3; i++)
293  {
294  LOGF_DEBUG("CMD (%#X)", command[0]);
295 
296  if ((rc = tty_write(PortFD, command, 1, &nbytes_written)) != TTY_OK)
297  {
298  tty_error_msg(rc, errstr, MAXRBUF);
299  LOGF_ERROR("Wakup error: %s.", errstr);
300  return false;
301  }
302 
303  if ((rc = tty_read_section(PortFD, response, 0xD, VANTAGE_TIMEOUT, &nbytes_read)) != TTY_OK)
304  continue;
305  else
306  {
307  if (nbytes_read == 2)
308  {
309  LOG_DEBUG("Console is awake.");
310  return true;
311  }
312  }
313  }
314 
315  return false;
316 }
317 
318 bool Vantage::ack()
319 {
320  int nbytes_written = 0, nbytes_read = 0, rc = -1;
321  char errstr[MAXRBUF];
322  char command[VANTAGE_CMD];
323  char response[VANTAGE_RES];
324 
325  if (!wakeup())
326  return false;
327 
328  command[0] = 'V';
329  command[1] = 'E';
330  command[2] = 'R';
331  command[3] = 0;
332 
333  tcflush(PortFD, TCIOFLUSH);
334 
335  LOGF_DEBUG("CMD (%s)", command);
336 
337  command[3] = 0xA;
338  command[4] = 0;
339 
340  if ((rc = tty_write(PortFD, command, 4, &nbytes_written)) != TTY_OK)
341  {
342  tty_error_msg(rc, errstr, MAXRBUF);
343  LOGF_ERROR("Ack error: %s.", errstr);
344  return false;
345  }
346 
347  if ((rc = tty_read_section(PortFD, response, 0xD, VANTAGE_TIMEOUT, &nbytes_read)) != TTY_OK)
348  {
349  tty_error_msg(rc, errstr, MAXRBUF);
350  LOGF_ERROR("Ack error: %s.", errstr);
351  return false;
352  }
353 
354  if (response[1] != 0xD)
355  {
356  LOGF_ERROR("Expecting 0xD, received %#X", response[1]);
357  return false;
358  }
359 
360  if ((rc = tty_read_section(PortFD, response, 0xD, VANTAGE_TIMEOUT, &nbytes_read)) != TTY_OK)
361  {
362  tty_error_msg(rc, errstr, MAXRBUF);
363  LOGF_ERROR("Ack error: %s.", errstr);
364  return false;
365  }
366 
367  response[nbytes_read - 2] = 0;
368 
369  if (strcmp(response, "OK") != 0)
370  {
371  LOGF_ERROR("Error response: %s", response);
372  return false;
373  }
374 
375  if ((rc = tty_read_section(PortFD, response, 0xD, VANTAGE_TIMEOUT, &nbytes_read)) != TTY_OK)
376  {
377  tty_error_msg(rc, errstr, MAXRBUF);
378  LOGF_ERROR("Ack error: %s.", errstr);
379  return false;
380  }
381 
382  response[nbytes_read - 2] = 0;
383 
384  LOGF_DEBUG("RES (%s)", response);
385 
386  return true;
387 }
void setDefaultBaudRate(BaudRate newRate)
setDefaultBaudRate Set default baud rate. The default baud rate is 9600 unless otherwise changed by t...
void setVersion(uint16_t vMajor, uint16_t vMinor)
Set driver version information to be defined in DRIVER_INFO property as vMajor.vMinor.
void addDebugControl()
Add Debug control to the driver.
void setParameterValue(std::string name, double value)
setParameterValue Update weather parameter value
bool setCriticalParameter(std::string param)
setCriticalParameter Set parameter that is considered critical to the operation of the observatory....
void addParameter(std::string name, std::string label, double numMinOk, double numMaxOk, double percWarning)
addParameter Add a physical weather measurable parameter to the weather driver. The weather value has...
Connection::Serial * serialConnection
Definition: indiweather.h:133
virtual bool initProperties() override
Initilize properties initial state and value. The child class must implement this function.
Definition: indiweather.cpp:41
Vantage()
Definition: vantage.cpp:75
virtual const char * getDefaultName() override
Definition: vantage.cpp:80
virtual IPState updateWeather() override
updateWeather Update weather conditions from device or service. The function should not change the st...
Definition: vantage.cpp:115
virtual bool initProperties() override
Initilize properties initial state and value. The child class must implement this function.
Definition: vantage.cpp:85
virtual bool Handshake() override
perform handshake with device to check communication
Definition: vantage.cpp:110
IPState
Property state.
Definition: indiapi.h:160
@ IPS_ALERT
Definition: indiapi.h:164
@ IPS_OK
Definition: indiapi.h:162
int tty_read_section(int fd, char *buf, char stop_char, int timeout, int *nbytes_read)
read buffer from terminal with a delimiter
Definition: indicom.c:566
int tty_write(int fd, const char *buf, int nbytes, int *nbytes_written)
Writes a buffer to fd.
Definition: indicom.c:424
int tty_read(int fd, char *buf, int nbytes, int timeout, int *nbytes_read)
read buffer from terminal
Definition: indicom.c:482
void tty_error_msg(int err_code, char *err_msg, int err_msg_len)
Retrieve the tty error message.
Definition: indicom.c:1167
Implementations for common driver routines.
@ TTY_OK
Definition: indicom.h:150
#define LOG_DEBUG(txt)
Definition: indilogger.h:75
#define LOGF_DEBUG(fmt,...)
Definition: indilogger.h:83
#define LOG_ERROR(txt)
Shorter logging macros. In order to use these macros, the function (or method) "getDeviceName()" must...
Definition: indilogger.h:72
#define LOGF_ERROR(fmt,...)
Definition: indilogger.h:80
#define LOG_INFO(txt)
Definition: indilogger.h:74
#define MAXRBUF
Definition: indiserver.cpp:102
std::unique_ptr< Vantage > vantage(new Vantage())
#define VANTAGE_TIMEOUT
Definition: vantage.cpp:36
#define VANTAGE_CMD
Definition: vantage.cpp:34
uint16_t crc16(const void *c_ptr, size_t len)
Definition: vantage.cpp:61
#define VANTAGE_RES
Definition: vantage.cpp:35