diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/app/bme680-max44009-logger/Makefile.inc | 11 | ||||
-rwxr-xr-x | src/app/bme680-max44009-logger/bme680-max44009-client.py | 185 | ||||
-rw-r--r-- | src/app/bme680-max44009-logger/main.cc | 104 |
3 files changed, 300 insertions, 0 deletions
diff --git a/src/app/bme680-max44009-logger/Makefile.inc b/src/app/bme680-max44009-logger/Makefile.inc new file mode 100644 index 0000000..5b3e8bb --- /dev/null +++ b/src/app/bme680-max44009-logger/Makefile.inc @@ -0,0 +1,11 @@ +# vim:ft=make +# +# Copyright 2020 Daniel Friesel +# +# SPDX-License-Identifier: CC0-1.0 + +ifdef app + override loop = 1 + override arch_drivers += ,i2c + override drivers += ,bme680,max44009 +endif diff --git a/src/app/bme680-max44009-logger/bme680-max44009-client.py b/src/app/bme680-max44009-logger/bme680-max44009-client.py new file mode 100755 index 0000000..3d836f1 --- /dev/null +++ b/src/app/bme680-max44009-logger/bme680-max44009-client.py @@ -0,0 +1,185 @@ +#!/usr/bin/env python3 + +import json +import paho.mqtt.client as mqtt +import re +import requests +import serial +import serial.threaded +import sys +import time + +location = "wohnzimmer" + + +class SerialReader(serial.threaded.Protocol): + """ + Character- to line-wise data buffer for serial interfaces. + + Reads in new data whenever it becomes available and exposes a line-based + interface to applications. + """ + + def __init__(self, callback): + """Create a new SerialReader object.""" + self.callback = callback + self.recv_buf = "" + + def __call__(self): + return self + + def data_received(self, data): + """Append newly received serial data to the line buffer.""" + try: + str_data = data.decode("UTF-8") + self.recv_buf += str_data + + # We may get anything between \r\n, \n\r and simple \n newlines. + # We assume that \n is always present and use str.strip to remove leading/trailing \r symbols + # Note: Do not call str.strip on lines[-1]! Otherwise, lines may be mangled + lines = self.recv_buf.split("\n") + if len(lines) > 1: + self.recv_buf = lines[-1] + for line in lines[:-1]: + self.callback(str.strip(line)) + + except UnicodeDecodeError: + pass + # sys.stderr.write('UART output contains garbage: {data}\n'.format(data = data)) + + +class SerialMonitor: + """SerialMonitor captures serial output for a specific amount of time.""" + + def __init__(self, port: str, baud: int, callback): + """ + Create a new SerialMonitor connected to port at the specified baud rate. + + Communication uses no parity, no flow control, and one stop bit. + Data collection starts immediately. + """ + self.ser = serial.serial_for_url(port, do_not_open=True) + self.ser.baudrate = baud + self.ser.parity = "N" + self.ser.rtscts = False + self.ser.xonxoff = False + + try: + self.ser.open() + except serial.SerialException as e: + sys.stderr.write( + "Could not open serial port {}: {}\n".format(self.ser.name, e) + ) + sys.exit(1) + + self.reader = SerialReader(callback=callback) + self.worker = serial.threaded.ReaderThread(self.ser, self.reader) + self.worker.start() + + def close(self): + """Close serial connection.""" + self.worker.stop() + self.ser.close() + + +if __name__ == "__main__": + + mqtt = mqtt.Client() + mqtt.connect("mqtt.derf0.net") + + step = 0 + got_data = False + max_accel = 0 + max_magnet = 0 + vcc = 0 + temperature = 0 + humidity = 0 + pressure = 0 + gas = 0 + brightness = 0 + + def parse_line(line): + + global got_data + global vcc + global temperature + global humidity + global pressure + global gas + global brightness + + match = re.match("BME680 temperature: ([^ ]+)", line) + if match: + temperature = float(match.group(1)) + + match = re.match("BME680 humidity: ([^ ]+)", line) + if match: + humidity = float(match.group(1)) + + match = re.match("BME680 pressure: ([^ ]+)", line) + if match: + pressure = float(match.group(1)) + + match = re.match("BME680 gas resistance: ([^ ]+)", line) + if match: + gas = match.group(1) + + match = re.match("VCC: ([^ ]+)", line) + if match: + vcc = int(match.group(1)) + + match = re.match("MAX44009: ([^ ]+)", line) + if match: + got_data = True + brightness = float(match.group(1)) + + requests.post( + "http://influxdb.derf0.net:8086/write?db=sensors", + f"bme680,area=hm17,location={location} temperature_celsius={temperature},humidity_relpercent={humidity},pressure_hpa={pressure},air_quality_ohm={gas}", + ) + requests.post( + "http://influxdb.derf0.net:8086/write?db=sensors", + f"max44009,area=hm17,location={location} illuminance_lux={brightness}", + ) + + mqtt.publish( + f"sensor/hm17/{location}/brightness_lux", + brightness + ) + mqtt.publish( + f"sensor/hm17/{location}/bme680", + json.dumps( + { + "temperature_celsius": round(temperature, 1), + "humidity_percent": round(humidity, 1), + "pressure_hpa": pressure, + "iaq_ohm": gas, + } + ), + ) + + temperature = None + humidity = None + pressure = None + gas = None + vcc = None + brightness = None + + monitor = SerialMonitor("/dev/ttyUSB0", 57600, parse_line) + + try: + while True: + time.sleep(5) + + step += 1 + + if step == 4: + if not got_data: + print("Error: received no data for 20 seconds", file=sys.stderr) + sys.exit(1) + got_data = False + step = 0 + + except KeyboardInterrupt: + monitor.close() + mqtt.disconnect() diff --git a/src/app/bme680-max44009-logger/main.cc b/src/app/bme680-max44009-logger/main.cc new file mode 100644 index 0000000..a8d4c2e --- /dev/null +++ b/src/app/bme680-max44009-logger/main.cc @@ -0,0 +1,104 @@ +/* + * Copyright 2021 Daniel Friesel + * + * SPDX-License-Identifier: BSD-2-Clause + */ +#include "arch.h" +#include "driver/gpio.h" +#include "driver/stdout.h" +#if defined(MULTIPASS_ARCH_HAS_I2C) && !defined(DRIVER_SOFTI2C) +#include "driver/i2c.h" +#else +#include "driver/soft_i2c.h" +#endif +#include "driver/bme680.h" +#include "driver/bme680_util.h" +#include "driver/max44009.h" + +struct bme680_field_data data; + +void loop(void) +{ + static unsigned char i = 0; + + if ((i == 1) && (ADCSRA & _BV(ADIF))) { + uint8_t adcr_l = ADCL; + uint8_t adcr_h = ADCH; + uint16_t adcr = adcr_l + (adcr_h << 8); + uint16_t vcc = 1100L * 1023 / adcr; + + TIFR1 |= _BV(TOV1); + ADCSRA |= _BV(ADIF); + + kout << "VCC: " << vcc << endl; + } + + if (i == 0) { + bme680.setSensorMode(); + } + else if (i == 1) { + if (bme680.getSensorData(&data) == 0) { + kout << "BME680 temperature: " << (float)data.temperature / 100 << " degC" << endl; + kout << "BME680 humidity: " << (float)data.humidity / 1000 << " %" << endl; + kout << "BME680 pressure: " << (float)data.pressure / 100 << " hPa" << endl; + kout << "BME680 gas resistance: " << data.gas_resistance << endl; + } + kout << "MAX44009: "; + kout.printf_float(max44009.getLux()); + kout << " lx" << endl; + } + + i = (i + 1) % 20; +} + +int main(void) +{ + unsigned short i = 0; + + arch.setup(); + gpio.setup(); + kout.setup(); + + // One ADC conversion per four seconds + TCCR0A = 0; + TCCR0B = _BV(CS12) | _BV(CS10); + + // Measure internal 1.1V bandgap using VCC as reference on each Timer 0 overflow + ADMUX = _BV(REFS0) | 0x0e; + ADCSRB = _BV(ADTS2); + ADCSRA = _BV(ADEN) | _BV(ADATE) | _BV(ADPS2) | _BV(ADPS1); + + if (i2c.setup() != 0) { + kout << "I2C setup failed" << endl; + return 1; + } + + kout << "I2C setup OK" << endl; + + bme680.intf = BME680_I2C_INTF; + bme680.read = bme680_i2c_read; + bme680.write = bme680_i2c_write; + bme680.delay_ms = bme680_delay_ms; + + bme680.amb_temp = 25; + + int8_t rslt = BME680_OK; + rslt = bme680.init(); + kout << "BME680 init " << rslt << endl; + + bme680.power_mode = BME680_FORCED_MODE; + bme680.tph_sett.os_hum = BME680_OS_2X; + bme680.tph_sett.os_pres = BME680_OS_16X; + bme680.tph_sett.os_temp = BME680_OS_2X; + + bme680.gas_sett.run_gas = BME680_ENABLE_GAS_MEAS; + bme680.gas_sett.heatr_dur = 100; + bme680.gas_sett.heatr_temp = 300; + bme680.setSensorSettings(BME680_OST_SEL | BME680_OSP_SEL | BME680_OSH_SEL | BME680_GAS_SENSOR_SEL); + + arch.delay_ms(200); + + arch.idle_loop(); + + return 0; +} |