* Optimize Sun data calculation * Remove not required enum * Split config struct into different sub structs * Feature: Allow configuration of LWT QoS * Made resetreason methods static * Feature: Implement offset cache for "YieldDay" Thanks to @broth-itk for the idea! Fix: #1258 #1397 * Add Esp32-Stick-PoE-A * remove broken LilyGO_T_ETH_POE config, use device profile instead * Feature: High resolution Icon and PWA (Progressive Web App) functionality Fix: #1289 * webapp: Update dependencies * Initialize TaskScheduler * Migrate SunPosition to TaskScheduler * Migrate Datastore to TaskScheduler * Migrate MqttHandleInverterTotal to TaskSchedule * Migrate MqttHandleHass to TaskScheduler * Migrate MqttHandleDtu to TaskScheduler * Migrate MqttHandleInverter to TaskScheduler * Migrate LedSingle to TaskScheduler * Migrate NetworkSettings to TaskScheduler * Migrate InverterSettings to TaskScheduler * Migrate MessageOutput to TaskScheduler * Migrate Display_Graphic to TaskScheduler * Migrate WebApi to TaskScheduler * Split InverterSettings into multiple tasks * Calculate SunPosition only every 5 seconds * Split LedSingle into multiple tasks * Upgrade espMqttClient from 1.4.5 to 1.5.0 * Doc: Correct amount of MPP-Tracker * Added HMT-1600-4T and HMT-1800-4T to DevInfoParser Fix #1524 * Adjusted inverter names for HMS-1600/1800/2000-4T * Add channel count to description of detected inverter type (DevInfoParser) * Adjust device web api endpoint for dynamic led count * Feature: Added ability to change the brightness of the LEDs Based on the idea of @moritzlerch with several modifications like pwmTable and structure * webapp: Update dependencies * Update olikraus/U8g2 from 2.35.7 to 2.35.8 * Remove not required onWebsocketEvent * Remove code nesting * Introduce several const statements * Remove not required AsyncEventSource * Doc: Added byte specification to each command * Feature: Added basic Grid Profile parser which shows the used profile and version Other values are still outstanding. * Optimize AlarmLogParser to save memory * Add libfrozen to project to create constexpr maps * Feature: First version of GridProfile Parser which shows all values contained in the profile. * webapp: Update dependencies * Apply better variable names * Remove not required casts * Add additional compiler flags to prevent errors * Add const statement to several variables * Replace NULL by nullptr * Update bblanchon/ArduinoJson from 6.21.3 to 6.21.4 * Add const keyword to method parameters * Add const keyword to methods * Use references instead of pointers whenver possible * Adjust member variable names in MqttSettings * Adjust member variable names in NetworkSettings * webapp: Update timezone database to latest version * webapp: Beautify and unify form footers * Feature: Allow setting of an inverter limit of 0% and 0W Thanks to @madmartin in #1270 * Feature: Allow links in device profiles These links will be shown on the hardware settings page. * Doc: Added hint regarding HMS-xxxx-xT-NA inverters * Feature: Added DeviceProfile for CASmo-DTU Based on #1565 * Upgrade actions/upload-artifact from v3 to v4 * Upgrade actions/download-artifact from v3 to v4 * webapp: add app.js.gz * Gridprofileparser: Added latest known values Thanks to @stefan123t and @noone2k * webapp: Fix lint errors * Feature: Add DTU to Home Assistant Auto Discovery This is based on PR 1365 from @CFenner with several fixes and optimizations * Fix: Remove debug output as it floods the console * Fix: Gridprofileparser: Add additional error handling if profile is unknown * webapp: add app.js.gz * Fix: Offset cache for "YieldDay" did not work correctly * webapp: update dependencies * webapp: add app.js.gz * Fix: yarn.lock was outdated * Fix: yarn build error * Fix: Reset Yield day correction in combination with Zero Yield Day on Midnight lead to wrong values. * Fix: Allow negative values in GridProfileParser * Correct variable name * Fix #1579: Static IP in Ethernet mode did not work correctly * Feature: Added diagram to display This is based on the idea of @Henrik-Ingenieur and was discussed in #1504 * webapp: update dependencies * webapp: add app.js.gz --------- Co-authored-by: Thomas Basler <thomas@familie-basler.net> Co-authored-by: Pierre Kancir <pierre.kancir.emn@gmail.com>
238 lines
7.6 KiB
C++
238 lines
7.6 KiB
C++
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Copyright (C) 2023 Thomas Basler and others
|
|
*/
|
|
#include "HoymilesRadio_CMT.h"
|
|
#include "Hoymiles.h"
|
|
#include "crc.h"
|
|
#include <FunctionalInterrupt.h>
|
|
|
|
#define HOY_BOOT_FREQ 868000000 // Hoymiles boot/init frequency after power up inverter or connection lost for 15 min
|
|
#define HOY_BASE_FREQ 860000000
|
|
// offset from initalized CMT base frequency to Hoy base frequency in channels
|
|
#define CMT_BASE_CH_OFFSET860 ((CMT_BASE_FREQ - HOY_BASE_FREQ) / CMT2300A_ONE_STEP_SIZE / FH_OFFSET)
|
|
|
|
// frequency can not be lower than actual initailized base freq
|
|
#define MIN_FREQ_KHZ ((HOY_BASE_FREQ + (CMT_BASE_CH_OFFSET860 >= 1 ? CMT_BASE_CH_OFFSET860 : 1) * CMT2300A_ONE_STEP_SIZE * FH_OFFSET) / 1000)
|
|
|
|
// =923500, 0xFF does not work
|
|
#define MAX_FREQ_KHZ ((HOY_BASE_FREQ + 0xFE * CMT2300A_ONE_STEP_SIZE * FH_OFFSET) / 1000)
|
|
|
|
float HoymilesRadio_CMT::getFrequencyFromChannel(const uint8_t channel)
|
|
{
|
|
return (CMT_BASE_FREQ + (CMT_BASE_CH_OFFSET860 + channel) * FH_OFFSET * CMT2300A_ONE_STEP_SIZE) / 1000000.0;
|
|
}
|
|
|
|
uint8_t HoymilesRadio_CMT::getChannelFromFrequency(const uint32_t freq_kHz)
|
|
{
|
|
if ((freq_kHz % 250) != 0) {
|
|
Hoymiles.getMessageOutput()->printf("%.3f MHz is not divisible by 250 kHz!\r\n", freq_kHz / 1000.0);
|
|
return 0xFF; // ERROR
|
|
}
|
|
if (freq_kHz < MIN_FREQ_KHZ || freq_kHz > MAX_FREQ_KHZ) {
|
|
Hoymiles.getMessageOutput()->printf("%.2f MHz is out of Hoymiles/CMT range! (%.2f MHz - %.2f MHz)\r\n",
|
|
freq_kHz / 1000.0, MIN_FREQ_KHZ / 1000.0, MAX_FREQ_KHZ / 1000.0);
|
|
return 0xFF; // ERROR
|
|
}
|
|
if (freq_kHz < 863000 || freq_kHz > 870000) {
|
|
Hoymiles.getMessageOutput()->printf("!!! caution: %.2f MHz is out of EU legal range! (863 - 870 MHz)\r\n",
|
|
freq_kHz / 1000.0);
|
|
}
|
|
return (freq_kHz * 1000 - CMT_BASE_FREQ) / CMT2300A_ONE_STEP_SIZE / FH_OFFSET - CMT_BASE_CH_OFFSET860; // frequency to channel
|
|
}
|
|
|
|
bool HoymilesRadio_CMT::cmtSwitchDtuFreq(const uint32_t to_freq_kHz)
|
|
{
|
|
const uint8_t toChannel = getChannelFromFrequency(to_freq_kHz);
|
|
if (toChannel == 0xFF) {
|
|
return false;
|
|
}
|
|
|
|
_radio->setChannel(toChannel);
|
|
|
|
return true;
|
|
}
|
|
|
|
void HoymilesRadio_CMT::init(const int8_t pin_sdio, const int8_t pin_clk, const int8_t pin_cs, const int8_t pin_fcs, const int8_t pin_gpio2, const int8_t pin_gpio3)
|
|
{
|
|
_dtuSerial.u64 = 0;
|
|
|
|
_radio.reset(new CMT2300A(pin_sdio, pin_clk, pin_cs, pin_fcs));
|
|
|
|
_radio->begin();
|
|
|
|
cmtSwitchDtuFreq(_inverterTargetFrequency); // start dtu at work freqency, for fast Rx if inverter is already on and frequency switched
|
|
|
|
if (!_radio->isChipConnected()) {
|
|
Hoymiles.getMessageOutput()->println("CMT: Connection error!!");
|
|
return;
|
|
}
|
|
Hoymiles.getMessageOutput()->println("CMT: Connection successful");
|
|
|
|
if (pin_gpio2 >= 0) {
|
|
attachInterrupt(digitalPinToInterrupt(pin_gpio2), std::bind(&HoymilesRadio_CMT::handleInt1, this), RISING);
|
|
_gpio2_configured = true;
|
|
}
|
|
|
|
if (pin_gpio3 >= 0) {
|
|
attachInterrupt(digitalPinToInterrupt(pin_gpio3), std::bind(&HoymilesRadio_CMT::handleInt2, this), RISING);
|
|
_gpio3_configured = true;
|
|
}
|
|
|
|
_isInitialized = true;
|
|
}
|
|
|
|
void HoymilesRadio_CMT::loop()
|
|
{
|
|
if (!_isInitialized) {
|
|
return;
|
|
}
|
|
|
|
if (!_gpio3_configured) {
|
|
if (_radio->rxFifoAvailable()) { // read INT2, PKT_OK flag
|
|
_packetReceived = true;
|
|
}
|
|
}
|
|
|
|
if (_packetReceived) {
|
|
Hoymiles.getVerboseMessageOutput()->println("Interrupt received");
|
|
while (_radio->available()) {
|
|
if (!(_rxBuffer.size() > FRAGMENT_BUFFER_SIZE)) {
|
|
fragment_t f;
|
|
memset(f.fragment, 0xcc, MAX_RF_PAYLOAD_SIZE);
|
|
f.len = _radio->getDynamicPayloadSize();
|
|
f.channel = _radio->getChannel();
|
|
f.rssi = _radio->getRssiDBm();
|
|
f.wasReceived = false;
|
|
f.mainCmd = 0x00;
|
|
if (f.len > MAX_RF_PAYLOAD_SIZE) {
|
|
f.len = MAX_RF_PAYLOAD_SIZE;
|
|
}
|
|
_radio->read(f.fragment, f.len);
|
|
_rxBuffer.push(f);
|
|
} else {
|
|
Hoymiles.getMessageOutput()->println("CMT: Buffer full");
|
|
_radio->flush_rx();
|
|
}
|
|
}
|
|
_radio->flush_rx();
|
|
_packetReceived = false;
|
|
|
|
} else {
|
|
// Perform package parsing only if no packages are received
|
|
if (!_rxBuffer.empty()) {
|
|
fragment_t f = _rxBuffer.back();
|
|
if (checkFragmentCrc(f)) {
|
|
|
|
const serial_u dtuId = convertSerialToRadioId(_dtuSerial);
|
|
|
|
// The CMT RF module does not filter foreign packages by itself.
|
|
// Has to be done manually here.
|
|
if (memcmp(&f.fragment[5], &dtuId.b[1], 4) == 0) {
|
|
|
|
std::shared_ptr<InverterAbstract> inv = Hoymiles.getInverterByFragment(f);
|
|
|
|
if (nullptr != inv) {
|
|
// Save packet in inverter rx buffer
|
|
Hoymiles.getVerboseMessageOutput()->printf("RX %.2f MHz --> ", getFrequencyFromChannel(f.channel));
|
|
dumpBuf(f.fragment, f.len, false);
|
|
Hoymiles.getVerboseMessageOutput()->printf("| %d dBm\r\n", f.rssi);
|
|
|
|
inv->addRxFragment(f.fragment, f.len);
|
|
} else {
|
|
Hoymiles.getMessageOutput()->println("Inverter Not found!");
|
|
}
|
|
}
|
|
|
|
} else {
|
|
Hoymiles.getMessageOutput()->println("Frame kaputt"); // ;-)
|
|
}
|
|
|
|
// Remove paket from buffer even it was corrupted
|
|
_rxBuffer.pop();
|
|
}
|
|
}
|
|
|
|
handleReceivedPackage();
|
|
}
|
|
|
|
void HoymilesRadio_CMT::setPALevel(const int8_t paLevel)
|
|
{
|
|
if (!_isInitialized) {
|
|
return;
|
|
}
|
|
|
|
if (_radio->setPALevel(paLevel)) {
|
|
Hoymiles.getMessageOutput()->printf("CMT TX power set to %d dBm\r\n", paLevel);
|
|
} else {
|
|
Hoymiles.getMessageOutput()->printf("CMT TX power %d dBm is not defined! (min: -10 dBm, max: 20 dBm)\r\n", paLevel);
|
|
}
|
|
}
|
|
|
|
void HoymilesRadio_CMT::setInverterTargetFrequency(const uint32_t frequency)
|
|
{
|
|
_inverterTargetFrequency = frequency;
|
|
if (!_isInitialized) {
|
|
return;
|
|
}
|
|
cmtSwitchDtuFreq(_inverterTargetFrequency);
|
|
}
|
|
|
|
uint32_t HoymilesRadio_CMT::getInverterTargetFrequency() const
|
|
{
|
|
return _inverterTargetFrequency;
|
|
}
|
|
|
|
bool HoymilesRadio_CMT::isConnected() const
|
|
{
|
|
if (!_isInitialized) {
|
|
return false;
|
|
}
|
|
return _radio->isChipConnected();
|
|
}
|
|
|
|
uint32_t HoymilesRadio_CMT::getMinFrequency()
|
|
{
|
|
return MIN_FREQ_KHZ;
|
|
}
|
|
|
|
uint32_t HoymilesRadio_CMT::getMaxFrequency()
|
|
{
|
|
return MAX_FREQ_KHZ;
|
|
}
|
|
|
|
void ARDUINO_ISR_ATTR HoymilesRadio_CMT::handleInt1()
|
|
{
|
|
_packetSent = true;
|
|
}
|
|
|
|
void ARDUINO_ISR_ATTR HoymilesRadio_CMT::handleInt2()
|
|
{
|
|
_packetReceived = true;
|
|
}
|
|
|
|
void HoymilesRadio_CMT::sendEsbPacket(CommandAbstract& cmd)
|
|
{
|
|
cmd.incrementSendCount();
|
|
|
|
cmd.setRouterAddress(DtuSerial().u64);
|
|
|
|
_radio->stopListening();
|
|
|
|
if (cmd.getDataPayload()[0] == 0x56) { // @todo(tbnobody) Bad hack to identify ChannelChange Command
|
|
cmtSwitchDtuFreq(HOY_BOOT_FREQ / 1000);
|
|
}
|
|
|
|
Hoymiles.getVerboseMessageOutput()->printf("TX %s %.2f MHz --> ",
|
|
cmd.getCommandName().c_str(), getFrequencyFromChannel(_radio->getChannel()));
|
|
cmd.dumpDataPayload(Hoymiles.getVerboseMessageOutput());
|
|
|
|
if (!_radio->write(cmd.getDataPayload(), cmd.getDataSize())) {
|
|
Hoymiles.getMessageOutput()->println("TX SPI Timeout");
|
|
}
|
|
cmtSwitchDtuFreq(_inverterTargetFrequency);
|
|
_radio->startListening();
|
|
_busyFlag = true;
|
|
_rxTimeout.set(cmd.getTimeout());
|
|
}
|