]> mj.ucw.cz Git - home-hw.git/blob - bsb/daemon/burrow-bsbd.c
Merge branch 'master' of ssh://git.ucw.cz/home/mj/GIT/home-hw
[home-hw.git] / bsb / daemon / burrow-bsbd.c
1 /*
2  *      Boiler System Bus Interface Daemon
3  *
4  *      (c) 2020 Martin Mares <mj@ucw.cz>
5  */
6
7 #include <ucw/lib.h>
8 #include <ucw/log.h>
9 #include <ucw/opt.h>
10 #include <ucw/string.h>
11 #include <ucw/unaligned.h>
12
13 #include <stdarg.h>
14 #include <stdio.h>
15 #include <stdint.h>
16 #include <stdlib.h>
17 #include <string.h>
18 #include <syslog.h>
19 #include <time.h>
20 #include <unistd.h>
21
22 #include <libusb.h>
23 #include <mosquitto.h>
24
25 typedef unsigned char byte;
26 typedef uint32_t u32;
27 typedef unsigned int uint;
28
29 #include "../firmware/interface.h"
30
31 /*** MQTT ***/
32
33 static struct mosquitto *mosq;
34 static bool mqtt_connected;
35
36 static void mqtt_publish(const char *topic, const char *fmt, ...)
37 {
38         va_list args;
39         va_start(args, fmt);
40
41         if (mqtt_connected) {
42                 char m[256];
43                 int l = vsnprintf(m, sizeof(m), fmt, args);
44                 int err = mosquitto_publish(mosq, NULL, topic, l, m, 0, true);
45                 if (err != MOSQ_ERR_SUCCESS)
46                         msg(L_ERROR, "Mosquitto: Publish failed, error=%d", err);
47         }
48
49         va_end(args);
50 }
51
52 static void mqtt_publish_ephemeral(const char *topic, const char *fmt, ...)
53 {
54         va_list args;
55         va_start(args, fmt);
56
57         if (mqtt_connected) {
58                 char m[256];
59                 int l = vsnprintf(m, sizeof(m), fmt, args);
60                 int err = mosquitto_publish(mosq, NULL, topic, l, m, 0, false);
61                 if (err != MOSQ_ERR_SUCCESS)
62                         msg(L_ERROR, "Mosquitto: Publish failed, error=%d", err);
63         }
64
65         va_end(args);
66 }
67
68 static void mqtt_conn_callback(struct mosquitto *mosq UNUSED, void *obj UNUSED, int status)
69 {
70         if (!status) {
71                 msg(L_DEBUG, "MQTT: Connection established");
72                 mqtt_connected = true;
73                 mqtt_publish("status/bsb", "ok");
74         } else if (mqtt_connected) {
75                 msg(L_DEBUG, "MQTT: Connection lost");
76                 mqtt_connected = false;
77         }
78 }
79
80 static void mqtt_log_callback(struct mosquitto *mosq UNUSED, void *obj UNUSED, int level, const char *message)
81 {
82         msg(L_DEBUG, "MQTT(%d): %s", level, message);
83 }
84
85 static void mqtt_init(void)
86 {
87         mosquitto_lib_init();
88
89         mosq = mosquitto_new("bsbd", 1, NULL);
90         if (!mosq)
91                 die("Mosquitto: Initialization failed");
92
93         mosquitto_connect_callback_set(mosq, mqtt_conn_callback);
94         mosquitto_log_callback_set(mosq, mqtt_log_callback);
95
96         if (mosquitto_will_set(mosq, "status/bsb", 4, "dead", 0, true) != MOSQ_ERR_SUCCESS)
97                 die("Mosquitto: Unable to set will");
98
99         if (mosquitto_connect_async(mosq, "127.0.0.1", 1883, 60) != MOSQ_ERR_SUCCESS)
100                 die("Mosquitto: Unable to connect");
101
102         if (mosquitto_loop_start(mosq))
103                 die("Mosquitto: Cannot start service thread");
104 }
105
106 /*** USB ***/
107
108 static struct libusb_context *usb_ctxt;
109 static struct libusb_device_handle *devh;
110
111 static void usb_error(const char *msg, ...)
112 {
113         va_list args;
114         va_start(args, msg);
115         ucw_vmsg(L_ERROR, msg, args);
116         va_end(args);
117
118         if (devh) {
119                 libusb_close(devh);
120                 devh = NULL;
121         }
122 }
123
124 static void open_device(void)
125 {
126         int err;
127         libusb_device **devlist;
128         ssize_t devn = libusb_get_device_list(usb_ctxt, &devlist);
129         if (devn < 0)
130                 die("Cannot enumerate USB devices: error %d", (int) devn);
131
132         for (ssize_t i=0; i<devn; i++) {
133                 struct libusb_device_descriptor desc;
134                 libusb_device *dev = devlist[i];
135                 if (!libusb_get_device_descriptor(dev, &desc)) {
136                         if (desc.idVendor == BSB_USB_VENDOR && desc.idProduct == BSB_USB_PRODUCT) {
137                                 msg(L_INFO, "Found device at usb%d.%d", libusb_get_bus_number(dev), libusb_get_device_address(dev));
138
139                                 if (err = libusb_open(dev, &devh)) {
140                                         usb_error("Cannot open device: error %d", err);
141                                         goto out;
142                                 }
143                                 libusb_reset_device(devh);
144                                 if (err = libusb_claim_interface(devh, 0)) {
145                                         usb_error("Cannot claim interface: error %d", err);
146                                         goto out;
147                                 }
148
149                                 goto out;
150                         }
151                 }
152         }
153
154 out:
155         libusb_free_device_list(devlist, 1);
156 }
157
158 static void init_usb(void)
159 {
160         int err;
161         if (err = libusb_init(&usb_ctxt))
162                 die("Cannot initialize libusb: error %d", err);
163         // libusb_set_debug(usb_ctxt, 3);
164         open_device();
165
166 }
167
168 /*** Protocol ***/
169
170 static const char * const stat_names[] = {
171 #define P(x) #x,
172         BSB_STATS
173 #undef P
174 };
175
176 static int get_s16_be(const byte *x)
177 {
178         int val = get_u16_be(x);
179         if (val >= 0x8000)
180                 return val - 0x10000;
181         else
182                 return val;
183 }
184
185 static void process_stats(time_t t, byte *resp, uint len)
186 {
187         for (uint i=0; i < ARRAY_SIZE(stat_names) && 4*i + 3 < (uint) len; i++) {
188                 char item[64];
189                 snprintf(item, sizeof(item), "bsb/stats/%s", stat_names[i]);
190                 mqtt_publish(item, "%u %lld", (uint) get_u32_le(resp+4*i), (long long) t);
191         }
192 }
193
194 static void process_info(time_t t, byte *p, uint len)
195 {
196         if (len < 4)
197                 return;
198
199         u32 addr = get_u32_be(p);
200         p += 4;
201         len -= 4;
202
203         switch (addr) {
204                 case 0x0500006b:
205                         if (len >= 2) {
206                                 int err = get_u16_be(p);
207                                 mqtt_publish("burrow/heating/error", "%d %lld", err, (long long) t);
208                         }
209                         break;
210                 case 0x0500006c:
211                         if (len >= 7) {
212                                 mqtt_publish("burrow/heating/clock", "%04d-%02d-%02dT%02d:%02d %lld",
213                                         get_u16_be(p) + 1900, p[2], p[3],
214                                         p[5], p[6],
215                                         (long long) t);
216                         }
217                         break;
218                 case 0x05000219:
219                         if (len >= 4) {
220                                 int temp = get_s16_be(p);
221                                 uint press = get_u16_be(p + 2);
222                                 mqtt_publish("burrow/heating/outside-temp", "%.2f %lld", temp / 64., (long long) t);
223                                 mqtt_publish("burrow/heating/water-pressure", "%.1f %lld", press / 10., (long long) t);
224                         }
225                         break;
226                 case 0x05000229:
227                         // AGU.2 status
228                         if (len >= 2) {
229                                 int temp = get_s16_be(p);
230                                 mqtt_publish("burrow/heating/circuit1/mix-temp", "%.2f %lld", temp / 64., (long long) t);
231                         }
232                         break;
233                 case 0x05040227:
234                         // AGU.2 control
235                         if (len >= 4) {
236                                 uint m = get_u16_be(p);
237                                 mqtt_publish("burrow/heating/circuit1/mix-valve", "%u %lld", m, (long long) t);
238                                 mqtt_publish("burrow/heating/circuit1/pump", "%u %lld", (p[3] == 3), (long long) t);
239                         }
240                         break;
241                 case 0x2d000211:
242                         // Circuit 1 status
243                         if (len >= 10) {
244                                 mqtt_publish("burrow/heating/circuit1/active", "%u %lld", (p[8] != 0), (long long) t);
245                         }
246                         break;
247                 case 0x2e000211:
248                         // Circuit 2 status
249                         if (len >= 10) {
250                                 mqtt_publish("burrow/heating/circuit2/active", "%u %lld", (p[8] != 0), (long long) t);
251                         }
252                         break;
253                 case 0x31000212:
254                         // Hot water status
255                         if (len >= 3) {
256                                 mqtt_publish("burrow/heating/water/active", "%u %lld", (p[1] != 0), (long long) t);
257                         }
258                         break;
259                 case 0x3d2d0215:
260                         // Room 1 status
261                         if (len >= 2) {
262                                 int temp = get_s16_be(p);
263                                 mqtt_publish("burrow/heating/circuit1/room-temp", "%.2f %lld", temp / 64., (long long) t);
264                         }
265                         break;
266                 case 0x3e2e0215:
267                         // Room 2 status
268                         if (len >= 2) {
269                                 int temp = get_s16_be(p);
270                                 mqtt_publish("burrow/heating/circuit2/room-temp", "%.2f %lld", temp / 64., (long long) t);
271                         }
272                         break;
273         }
274 }
275
276 static void process_answer(time_t t, byte *p, uint len)
277 {
278         if (len < 4)
279                 return;
280
281         u32 addr = get_u32_be(p);
282         p += 4;
283         len -= 4;
284
285         switch (addr) {
286         }
287 }
288
289 static void process_frame(time_t t, byte *pkt, uint len)
290 {
291         if (!len) {
292                 msg(L_ERROR, "Received empty frame");
293                 return;
294         }
295
296         byte status = *pkt++;
297         len--;
298
299         if (!len) {
300                 msg(L_ERROR, "Frame transmit status: %u", status);
301                 return;
302         }
303
304         if (len < 6) {
305                 msg(L_ERROR, "Received truncated frame");
306                 return;
307         }
308
309         char hex[3*len + 1];
310         mem_to_hex(hex, pkt, len, ' ');
311         mqtt_publish_ephemeral("bsb/frame", "%s", hex, t);
312
313         msg(L_DEBUG, "<< %s", hex);
314
315         byte *body = pkt + BF_BODY;
316         uint body_len = len - BF_BODY - 2;      // 2 for CRC
317
318         switch (pkt[BF_OP]) {
319                 case BSB_OP_INFO:
320                         process_info(t, body, body_len);
321                         break;
322                 case BSB_OP_ANSWER:
323                         process_answer(t, body, body_len);
324                         break;
325         }
326 }
327
328 static int use_daemon;
329 static int use_debug;
330
331 static struct opt_section options = {
332         OPT_ITEMS {
333                 OPT_HELP("A daemon for controlling the air conditioning controller via MQTT"),
334                 OPT_HELP(""),
335                 OPT_HELP("Options:"),
336                 OPT_BOOL('d', "debug", use_debug, 0, "\tLog debugging messages"),
337                 OPT_BOOL(0, "daemon", use_daemon, 0, "\tDaemonize"),
338                 OPT_HELP_OPTION,
339                 OPT_CONF_OPTIONS,
340                 OPT_END
341         }
342 };
343
344 int main(int argc UNUSED, char **argv)
345 {
346         log_init(argv[0]);
347         opt_parse(&options, argv+1);
348
349         if (use_daemon) {
350                 struct log_stream *ls = log_new_syslog("daemon", LOG_PID);
351                 log_set_default_stream(ls);
352         }
353         if (!use_debug)
354                 log_default_stream()->levels &= ~(1U << L_DEBUG);
355
356         mqtt_init();
357         init_usb();
358
359         time_t now = time(NULL);
360         time_t last_stats = 0;
361         // time_t last_query = now;
362         int err, received;
363         byte resp[64];
364
365         for (;;) {
366                 if (!devh) {
367                         msg(L_INFO, "Waiting for device to appear...");
368                         while (!devh) {
369                                 sleep(5);
370                                 open_device();
371                         }
372                 }
373
374                 now = time(NULL);
375                 if (last_stats + 60 < now) {
376                         if ((received = libusb_control_transfer(devh, 0xc0, 0x00, 0, 0, resp, sizeof(resp), 1000)) < 0) {
377                                 usb_error("Receive failed: error %d", received);
378                                 continue;
379                         }
380
381                         process_stats(now, resp, received);
382                         last_stats = now;
383                 }
384
385 #if 0
386                 if (last_query + 10 < now) {
387                         byte pkt[] = { 0xdc, 0xc2, 0x00, 0x0b, 0x06, 0x3d, 0x2e, 0x11, 0x25, 0x00, 0x00 };
388                         if (err = libusb_bulk_transfer(devh, 0x01, pkt, sizeof(pkt), &received, 2000)) {
389                                 usb_error("Send failed: error %d", err);
390                                 continue;
391                         } else {
392                                 // msg(L_DEBUG"Send OK: %d bytes", received);
393                         }
394                         last_query = now;
395                 }
396 #endif
397
398                 if (err = libusb_interrupt_transfer(devh, 0x82, resp, 64, &received, 1000)) {
399                         if (err != LIBUSB_ERROR_TIMEOUT)
400                                 usb_error("Receive failed: error %d", err);
401                         continue;
402                 }
403
404                 process_frame(now, resp, received);
405         }
406 }