mirror of
https://github.com/meshtastic/firmware.git
synced 2025-08-02 20:05:52 +00:00

Some checks failed
CI / setup (check) (push) Has been cancelled
CI / setup (esp32) (push) Has been cancelled
CI / setup (esp32c3) (push) Has been cancelled
CI / setup (esp32c6) (push) Has been cancelled
CI / setup (esp32s3) (push) Has been cancelled
CI / setup (nrf52840) (push) Has been cancelled
CI / setup (rp2040) (push) Has been cancelled
CI / setup (rp2350) (push) Has been cancelled
CI / setup (stm32) (push) Has been cancelled
CI / version (push) Has been cancelled
CI / build-debian-src (push) Has been cancelled
CI / package-pio-deps-native-tft (push) Has been cancelled
CI / test-native (push) Has been cancelled
CI / docker-deb-amd64 (push) Has been cancelled
CI / docker-deb-amd64-tft (push) Has been cancelled
CI / docker-alp-amd64 (push) Has been cancelled
CI / docker-alp-amd64-tft (push) Has been cancelled
CI / docker-deb-arm64 (push) Has been cancelled
CI / docker-deb-armv7 (push) Has been cancelled
CI / check (push) Has been cancelled
CI / build-esp32 (push) Has been cancelled
CI / build-esp32s3 (push) Has been cancelled
CI / build-esp32c3 (push) Has been cancelled
CI / build-esp32c6 (push) Has been cancelled
CI / build-nrf52840 (push) Has been cancelled
CI / build-rp2040 (push) Has been cancelled
CI / build-rp2350 (push) Has been cancelled
CI / build-stm32 (push) Has been cancelled
CI / gather-artifacts (esp32) (push) Has been cancelled
CI / gather-artifacts (esp32c3) (push) Has been cancelled
CI / gather-artifacts (esp32c6) (push) Has been cancelled
CI / gather-artifacts (esp32s3) (push) Has been cancelled
CI / gather-artifacts (nrf52840) (push) Has been cancelled
CI / gather-artifacts (rp2040) (push) Has been cancelled
CI / gather-artifacts (rp2350) (push) Has been cancelled
CI / gather-artifacts (stm32) (push) Has been cancelled
CI / release-artifacts (push) Has been cancelled
CI / release-firmware (esp32) (push) Has been cancelled
CI / release-firmware (esp32c3) (push) Has been cancelled
CI / release-firmware (esp32c6) (push) Has been cancelled
CI / release-firmware (esp32s3) (push) Has been cancelled
CI / release-firmware (nrf52840) (push) Has been cancelled
CI / release-firmware (rp2040) (push) Has been cancelled
CI / release-firmware (rp2350) (push) Has been cancelled
CI / release-firmware (stm32) (push) Has been cancelled
CI / publish-firmware (push) Has been cancelled
* treewide: make 'ifdef DEBUG_PORT' guards also take into account DEBUG_MUTE * stm32wl: Add a guard against having debug prints turned on without PIO_FRAMEWORK_ARDUINO_NANOLIB_FLOAT_PRINTF defined --------- Co-authored-by: Ben Meadors <benmmeadors@gmail.com>
236 lines
9.5 KiB
C++
236 lines
9.5 KiB
C++
#include "configuration.h"
|
||
|
||
#if HAS_TELEMETRY && !MESHTASTIC_EXCLUDE_ENVIRONMENTAL_SENSOR && __has_include("Adafruit_PM25AQI.h")
|
||
|
||
#include "../mesh/generated/meshtastic/telemetry.pb.h"
|
||
#include "AirQualityTelemetry.h"
|
||
#include "Default.h"
|
||
#include "MeshService.h"
|
||
#include "NodeDB.h"
|
||
#include "PowerFSM.h"
|
||
#include "RTC.h"
|
||
#include "Router.h"
|
||
#include "detect/ScanI2CTwoWire.h"
|
||
#include "main.h"
|
||
#include <Throttle.h>
|
||
|
||
#ifndef PMSA003I_WARMUP_MS
|
||
// from the PMSA003I datasheet:
|
||
// "Stable data should be got at least 30 seconds after the sensor wakeup
|
||
// from the sleep mode because of the fan’s performance."
|
||
#define PMSA003I_WARMUP_MS 30000
|
||
#endif
|
||
|
||
int32_t AirQualityTelemetryModule::runOnce()
|
||
{
|
||
/*
|
||
Uncomment the preferences below if you want to use the module
|
||
without having to configure it from the PythonAPI or WebUI.
|
||
*/
|
||
|
||
// moduleConfig.telemetry.air_quality_enabled = 1;
|
||
|
||
if (!(moduleConfig.telemetry.air_quality_enabled)) {
|
||
// If this module is not enabled, and the user doesn't want the display screen don't waste any OSThread time on it
|
||
return disable();
|
||
}
|
||
|
||
if (firstTime) {
|
||
// This is the first time the OSThread library has called this function, so do some setup
|
||
firstTime = false;
|
||
|
||
if (moduleConfig.telemetry.air_quality_enabled) {
|
||
LOG_INFO("Air quality Telemetry: init");
|
||
|
||
#ifdef PMSA003I_ENABLE_PIN
|
||
// put the sensor to sleep on startup
|
||
pinMode(PMSA003I_ENABLE_PIN, OUTPUT);
|
||
digitalWrite(PMSA003I_ENABLE_PIN, LOW);
|
||
#endif /* PMSA003I_ENABLE_PIN */
|
||
|
||
if (!aqi.begin_I2C()) {
|
||
#ifndef I2C_NO_RESCAN
|
||
LOG_WARN("Could not establish i2c connection to AQI sensor. Rescan");
|
||
// rescan for late arriving sensors. AQI Module starts about 10 seconds into the boot so this is plenty.
|
||
uint8_t i2caddr_scan[] = {PMSA0031_ADDR};
|
||
uint8_t i2caddr_asize = 1;
|
||
auto i2cScanner = std::unique_ptr<ScanI2CTwoWire>(new ScanI2CTwoWire());
|
||
#if defined(I2C_SDA1)
|
||
i2cScanner->scanPort(ScanI2C::I2CPort::WIRE1, i2caddr_scan, i2caddr_asize);
|
||
#endif
|
||
i2cScanner->scanPort(ScanI2C::I2CPort::WIRE, i2caddr_scan, i2caddr_asize);
|
||
auto found = i2cScanner->find(ScanI2C::DeviceType::PMSA0031);
|
||
if (found.type != ScanI2C::DeviceType::NONE) {
|
||
nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_PMSA003I].first = found.address.address;
|
||
nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_PMSA003I].second =
|
||
i2cScanner->fetchI2CBus(found.address);
|
||
return setStartDelay();
|
||
}
|
||
#endif
|
||
return disable();
|
||
}
|
||
return setStartDelay();
|
||
}
|
||
return disable();
|
||
} else {
|
||
// if we somehow got to a second run of this module with measurement disabled, then just wait forever
|
||
if (!moduleConfig.telemetry.air_quality_enabled)
|
||
return disable();
|
||
|
||
switch (state) {
|
||
#ifdef PMSA003I_ENABLE_PIN
|
||
case State::IDLE:
|
||
// sensor is in standby; fire it up and sleep
|
||
LOG_DEBUG("runOnce(): state = idle");
|
||
digitalWrite(PMSA003I_ENABLE_PIN, HIGH);
|
||
state = State::ACTIVE;
|
||
|
||
return PMSA003I_WARMUP_MS;
|
||
#endif /* PMSA003I_ENABLE_PIN */
|
||
case State::ACTIVE:
|
||
// sensor is already warmed up; grab telemetry and send it
|
||
LOG_DEBUG("runOnce(): state = active");
|
||
|
||
if (((lastSentToMesh == 0) ||
|
||
!Throttle::isWithinTimespanMs(lastSentToMesh, Default::getConfiguredOrDefaultMsScaled(
|
||
moduleConfig.telemetry.air_quality_interval,
|
||
default_telemetry_broadcast_interval_secs, numOnlineNodes))) &&
|
||
airTime->isTxAllowedChannelUtil(config.device.role != meshtastic_Config_DeviceConfig_Role_SENSOR) &&
|
||
airTime->isTxAllowedAirUtil()) {
|
||
sendTelemetry();
|
||
lastSentToMesh = millis();
|
||
} else if (service->isToPhoneQueueEmpty()) {
|
||
// Just send to phone when it's not our time to send to mesh yet
|
||
// Only send while queue is empty (phone assumed connected)
|
||
sendTelemetry(NODENUM_BROADCAST, true);
|
||
}
|
||
|
||
#ifdef PMSA003I_ENABLE_PIN
|
||
// put sensor back to sleep
|
||
digitalWrite(PMSA003I_ENABLE_PIN, LOW);
|
||
state = State::IDLE;
|
||
#endif /* PMSA003I_ENABLE_PIN */
|
||
|
||
return sendToPhoneIntervalMs;
|
||
default:
|
||
return disable();
|
||
}
|
||
}
|
||
}
|
||
|
||
bool AirQualityTelemetryModule::handleReceivedProtobuf(const meshtastic_MeshPacket &mp, meshtastic_Telemetry *t)
|
||
{
|
||
if (t->which_variant == meshtastic_Telemetry_air_quality_metrics_tag) {
|
||
#if defined(DEBUG_PORT) && !defined(DEBUG_MUTE)
|
||
const char *sender = getSenderShortName(mp);
|
||
|
||
LOG_INFO("(Received from %s): pm10_standard=%i, pm25_standard=%i, pm100_standard=%i", sender,
|
||
t->variant.air_quality_metrics.pm10_standard, t->variant.air_quality_metrics.pm25_standard,
|
||
t->variant.air_quality_metrics.pm100_standard);
|
||
|
||
LOG_INFO(" | PM1.0(Environmental)=%i, PM2.5(Environmental)=%i, PM10.0(Environmental)=%i",
|
||
t->variant.air_quality_metrics.pm10_environmental, t->variant.air_quality_metrics.pm25_environmental,
|
||
t->variant.air_quality_metrics.pm100_environmental);
|
||
#endif
|
||
// release previous packet before occupying a new spot
|
||
if (lastMeasurementPacket != nullptr)
|
||
packetPool.release(lastMeasurementPacket);
|
||
|
||
lastMeasurementPacket = packetPool.allocCopy(mp);
|
||
}
|
||
|
||
return false; // Let others look at this message also if they want
|
||
}
|
||
|
||
bool AirQualityTelemetryModule::getAirQualityTelemetry(meshtastic_Telemetry *m)
|
||
{
|
||
if (!aqi.read(&data)) {
|
||
LOG_WARN("Skip send measurements. Could not read AQIn");
|
||
return false;
|
||
}
|
||
|
||
m->time = getTime();
|
||
m->which_variant = meshtastic_Telemetry_air_quality_metrics_tag;
|
||
m->variant.air_quality_metrics.has_pm10_standard = true;
|
||
m->variant.air_quality_metrics.pm10_standard = data.pm10_standard;
|
||
m->variant.air_quality_metrics.has_pm25_standard = true;
|
||
m->variant.air_quality_metrics.pm25_standard = data.pm25_standard;
|
||
m->variant.air_quality_metrics.has_pm100_standard = true;
|
||
m->variant.air_quality_metrics.pm100_standard = data.pm100_standard;
|
||
|
||
m->variant.air_quality_metrics.has_pm10_environmental = true;
|
||
m->variant.air_quality_metrics.pm10_environmental = data.pm10_env;
|
||
m->variant.air_quality_metrics.has_pm25_environmental = true;
|
||
m->variant.air_quality_metrics.pm25_environmental = data.pm25_env;
|
||
m->variant.air_quality_metrics.has_pm100_environmental = true;
|
||
m->variant.air_quality_metrics.pm100_environmental = data.pm100_env;
|
||
|
||
LOG_INFO("Send: PM1.0(Standard)=%i, PM2.5(Standard)=%i, PM10.0(Standard)=%i", m->variant.air_quality_metrics.pm10_standard,
|
||
m->variant.air_quality_metrics.pm25_standard, m->variant.air_quality_metrics.pm100_standard);
|
||
|
||
LOG_INFO(" | PM1.0(Environmental)=%i, PM2.5(Environmental)=%i, PM10.0(Environmental)=%i",
|
||
m->variant.air_quality_metrics.pm10_environmental, m->variant.air_quality_metrics.pm25_environmental,
|
||
m->variant.air_quality_metrics.pm100_environmental);
|
||
|
||
return true;
|
||
}
|
||
|
||
meshtastic_MeshPacket *AirQualityTelemetryModule::allocReply()
|
||
{
|
||
if (currentRequest) {
|
||
auto req = *currentRequest;
|
||
const auto &p = req.decoded;
|
||
meshtastic_Telemetry scratch;
|
||
meshtastic_Telemetry *decoded = NULL;
|
||
memset(&scratch, 0, sizeof(scratch));
|
||
if (pb_decode_from_bytes(p.payload.bytes, p.payload.size, &meshtastic_Telemetry_msg, &scratch)) {
|
||
decoded = &scratch;
|
||
} else {
|
||
LOG_ERROR("Error decoding AirQualityTelemetry module!");
|
||
return NULL;
|
||
}
|
||
// Check for a request for air quality metrics
|
||
if (decoded->which_variant == meshtastic_Telemetry_air_quality_metrics_tag) {
|
||
meshtastic_Telemetry m = meshtastic_Telemetry_init_zero;
|
||
if (getAirQualityTelemetry(&m)) {
|
||
LOG_INFO("Air quality telemetry reply to request");
|
||
return allocDataProtobuf(m);
|
||
} else {
|
||
return NULL;
|
||
}
|
||
}
|
||
}
|
||
return NULL;
|
||
}
|
||
|
||
bool AirQualityTelemetryModule::sendTelemetry(NodeNum dest, bool phoneOnly)
|
||
{
|
||
meshtastic_Telemetry m = meshtastic_Telemetry_init_zero;
|
||
if (getAirQualityTelemetry(&m)) {
|
||
meshtastic_MeshPacket *p = allocDataProtobuf(m);
|
||
p->to = dest;
|
||
p->decoded.want_response = false;
|
||
if (config.device.role == meshtastic_Config_DeviceConfig_Role_SENSOR)
|
||
p->priority = meshtastic_MeshPacket_Priority_RELIABLE;
|
||
else
|
||
p->priority = meshtastic_MeshPacket_Priority_BACKGROUND;
|
||
|
||
// release previous packet before occupying a new spot
|
||
if (lastMeasurementPacket != nullptr)
|
||
packetPool.release(lastMeasurementPacket);
|
||
|
||
lastMeasurementPacket = packetPool.allocCopy(*p);
|
||
if (phoneOnly) {
|
||
LOG_INFO("Send packet to phone");
|
||
service->sendToPhone(p);
|
||
} else {
|
||
LOG_INFO("Send packet to mesh");
|
||
service->sendToMesh(p, RX_SRC_LOCAL, true);
|
||
}
|
||
return true;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
#endif |