/*
xnrg_18_sdm72.ino - Eastron SDM72D-M Modbus energy meter support for Tasmota
Copyright (C) 2021 Gennaro Tortone, Theo Arends and Norbert Richter
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
#ifdef USE_ENERGY_SENSOR
#ifdef USE_SDM72
/*********************************************************************************************\
* Eastron SDM72 Modbus energy meter
\*********************************************************************************************/
#define XNRG_18 18
//#define SDM72_IMPEXP
// can be user defined in my_user_config.h
#ifndef SDM72_SPEED
#define SDM72_SPEED 9600 // default SDM72 Modbus address
#endif
// can be user defined in my_user_config.h
#ifndef SDM72_ADDR
#define SDM72_ADDR 1 // default SDM72 Modbus address
#endif
#include
TasmotaModbus *Sdm72Modbus;
const uint16_t sdm72_register[] {
0x0034, // 0 SDM72D_POWER [W]
0x0156, // 3 SDM72D_TOTAL_ACTIVE [kWh]
#ifdef SDM72_IMPEXP
0x0500, // 1 SDM72D_IMPORT_POWER [W]
0x0502, // 2 SDM72D_EXPORT_POWER [W]
0x0048, // 4 SDM72D_IMPORT_ACTIVE [kWh]
0x004A // 5 SDM72D_EXPORT_ACTIVE [kWh]
#endif // SDM72_IMPEXP
};
struct SDM72 {
float total_active = NAN;
#ifdef SDM72_IMPEXP
float import_power = 0;
float export_power = 0;
float import_active = 0;
#endif // SDM72_IMPEXP
uint8_t read_state = 0;
uint8_t send_retry = 0;
} Sdm72;
/*********************************************************************************************/
void Sdm72Every250ms(void)
{
bool data_ready = Sdm72Modbus->ReceiveReady();
if (data_ready) {
uint8_t buffer[14]; // At least 5 + (2 * 2) = 9
uint32_t error = Sdm72Modbus->ReceiveBuffer(buffer, 2);
AddLogBuffer(LOG_LEVEL_DEBUG_MORE, buffer, Sdm72Modbus->ReceiveCount());
if (error) {
AddLog(LOG_LEVEL_DEBUG, PSTR("SDM: SDM72 error %d"), error);
} else {
Energy.data_valid[0] = 0;
float value;
((uint8_t*)&value)[3] = buffer[3]; // Get float values
((uint8_t*)&value)[2] = buffer[4];
((uint8_t*)&value)[1] = buffer[5];
((uint8_t*)&value)[0] = buffer[6];
switch(Sdm72.read_state) {
case 0:
Energy.active_power[0] = value; // W
break;
case 1:
Sdm72.total_active = value; // kWh
break;
#ifdef SDM72_IMPEXP
case 2:
Sdm72.import_power = value; // W
break;
case 3:
Sdm72.export_power = value; // W
break;
case 4:
Energy.import_active[0] = value; // kWh
break;
case 5:
Energy.export_active[0] = value; // kWh
break;
#endif // SDM72_IMPEXP
}
++Sdm72.read_state %= nitems(sdm72_register);
if (0 == Sdm72.read_state && !isnan(Sdm72.total_active)) {
Energy.import_active[0] = Sdm72.total_active;
EnergyUpdateTotal();
}
}
} // end data ready
if (0 == Sdm72.send_retry || data_ready) {
Sdm72.send_retry = 5;
Sdm72Modbus->Send(SDM72_ADDR, 0x04, sdm72_register[Sdm72.read_state], 2);
} else {
Sdm72.send_retry--;
}
}
void Sdm72SnsInit(void)
{
Sdm72Modbus = new TasmotaModbus(Pin(GPIO_SDM72_RX), Pin(GPIO_SDM72_TX));
uint8_t result = Sdm72Modbus->Begin(SDM72_SPEED);
if (result) {
if (2 == result) {
ClaimSerial();
}
} else {
TasmotaGlobal.energy_driver = ENERGY_NONE;
}
}
void Sdm72DrvInit(void)
{
if (PinUsed(GPIO_SDM72_RX) && PinUsed(GPIO_SDM72_TX)) {
Energy.voltage_available = false;
Energy.current_available = false;
TasmotaGlobal.energy_driver = XNRG_18;
}
}
#ifdef SDM72_IMPEXP
/*
#ifdef USE_WEBSERVER
const char HTTP_ENERGY_SDM72[] PROGMEM =
"{s}" D_EXPORT_POWER "{m}%*_f " D_UNIT_WATT "{e}"
"{s}" D_IMPORT_POWER "{m}%*_f " D_UNIT_WATT "{e}";
#endif // USE_WEBSERVER
void Sdm72Show(bool json) {
if (isnan(Sdm72.total_active)) { return; }
if (json) {
ResponseAppend_P(PSTR(",\"" D_JSON_EXPORT_POWER "\":%*_f,\"" D_JSON_IMPORT_POWER "\":%*_f"),
Settings->flag2.wattage_resolution, &Sdm72.export_power,
Settings->flag2.wattage_resolution, &Sdm72.import_power);
#ifdef USE_WEBSERVER
} else {
WSContentSend_PD(HTTP_ENERGY_SDM72,
Settings->flag2.wattage_resolution, &Sdm72.export_power,
Settings->flag2.wattage_resolution, &Sdm72.import_power);
#endif // USE_WEBSERVER
}
}
*/
#ifdef USE_WEBSERVER
const char HTTP_ENERGY_SDM72[] PROGMEM =
"{s}" D_EXPORT_POWER "{m}%s" D_UNIT_WATT "{e}"
"{s}" D_IMPORT_POWER "{m}%s" D_UNIT_WATT "{e}";
#endif // USE_WEBSERVER
void Sdm72Show(bool json) {
if (isnan(Sdm72.total_active)) { return; }
char value_chr[TOPSZ];
char value2_chr[TOPSZ];
if (json) {
ResponseAppend_P(PSTR(",\"" D_JSON_EXPORT_POWER "\":%s,\"" D_JSON_IMPORT_POWER "\":%s"),
EnergyFormat(value_chr, &Sdm72.export_power, Settings->flag2.wattage_resolution),
EnergyFormat(value2_chr, &Sdm72.import_power, Settings->flag2.wattage_resolution));
#ifdef USE_WEBSERVER
} else {
WSContentSend_PD(HTTP_ENERGY_SDM72, WebEnergyFormat(value_chr, &Sdm72.export_power, Settings->flag2.wattage_resolution),
WebEnergyFormat(value2_chr, &Sdm72.import_power, Settings->flag2.wattage_resolution));
#endif // USE_WEBSERVER
}
}
#endif // SDM72_IMPEXP
/*********************************************************************************************\
* Interface
\*********************************************************************************************/
bool Xnrg18(uint8_t function)
{
bool result = false;
switch (function) {
case FUNC_EVERY_250_MSECOND:
Sdm72Every250ms();
break;
#ifdef SDM72_IMPEXP
case FUNC_JSON_APPEND:
Sdm72Show(1);
break;
#ifdef USE_WEBSERVER
#ifdef USE_ENERGY_COLUMN_GUI
case FUNC_WEB_COL_SENSOR:
Sdm72Show(0);
break;
#else // not USE_ENERGY_COLUMN_GUI
case FUNC_WEB_SENSOR:
Sdm72Show(0);
break;
#endif // USE_ENERGY_COLUMN_GUI
#endif // USE_WEBSERVER
#endif // SDM72_IMPEXP
case FUNC_INIT:
Sdm72SnsInit();
break;
case FUNC_PRE_INIT:
Sdm72DrvInit();
break;
}
return result;
}
#endif // USE_SDM72
#endif // USE_ENERGY_SENSOR