201 lines
6.1 KiB
C++
201 lines
6.1 KiB
C++
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Copyright (C) 2022 Thomas Basler and others
|
|
*/
|
|
#include "Hoymiles.h"
|
|
#include "inverters/HM_1CH.h"
|
|
#include "inverters/HM_2CH.h"
|
|
#include "inverters/HM_4CH.h"
|
|
#include "inverters/HMS_1CH.h"
|
|
#include "inverters/HMS_2CH.h"
|
|
#include "inverters/HMS_4CH.h"
|
|
#include <Arduino.h>
|
|
|
|
#define HOY_SEMAPHORE_TAKE() xSemaphoreTake(_xSemaphore, portMAX_DELAY)
|
|
#define HOY_SEMAPHORE_GIVE() xSemaphoreGive(_xSemaphore)
|
|
|
|
HoymilesClass Hoymiles;
|
|
|
|
void HoymilesClass::init(SPIClass* initialisedSpiBus, uint8_t pinCE, uint8_t pinIRQ)
|
|
{
|
|
_xSemaphore = xSemaphoreCreateMutex();
|
|
HOY_SEMAPHORE_GIVE(); // release before first use
|
|
|
|
_pollInterval = 0;
|
|
_radioNrf.reset(new HoymilesRadio_NRF());
|
|
_radioNrf->init(initialisedSpiBus, pinCE, pinIRQ);
|
|
|
|
_radioCmt.reset(new HoymilesRadio_CMT());
|
|
_radioCmt->init();
|
|
}
|
|
|
|
void HoymilesClass::loop()
|
|
{
|
|
HOY_SEMAPHORE_TAKE();
|
|
_radioNrf->loop();
|
|
_radioCmt->loop();
|
|
|
|
if (getNumInverters() > 0) {
|
|
if (millis() - _lastPoll > (_pollInterval * 1000)) {
|
|
static uint8_t inverterPos = 0;
|
|
|
|
if (_radioNrf->isIdle() && _radioCmt->isIdle()) {
|
|
std::shared_ptr<InverterAbstract> iv = getInverterByPos(inverterPos);
|
|
if (iv != nullptr) {
|
|
_messageOutput->print("Fetch inverter: ");
|
|
_messageOutput->println(iv->serial(), HEX);
|
|
|
|
iv->sendStatsRequest();
|
|
|
|
// Fetch event log
|
|
bool force = iv->EventLog()->getLastAlarmRequestSuccess() == CMD_NOK;
|
|
iv->sendAlarmLogRequest(force);
|
|
|
|
// Fetch limit
|
|
if ((iv->SystemConfigPara()->getLastLimitRequestSuccess() == CMD_NOK)
|
|
|| ((millis() - iv->SystemConfigPara()->getLastUpdateRequest() > HOY_SYSTEM_CONFIG_PARA_POLL_INTERVAL)
|
|
&& (millis() - iv->SystemConfigPara()->getLastUpdateCommand() > HOY_SYSTEM_CONFIG_PARA_POLL_MIN_DURATION))) {
|
|
_messageOutput->println("Request SystemConfigPara");
|
|
iv->sendSystemConfigParaRequest();
|
|
}
|
|
|
|
// Set limit if required
|
|
if (iv->SystemConfigPara()->getLastLimitCommandSuccess() == CMD_NOK) {
|
|
_messageOutput->println("Resend ActivePowerControl");
|
|
iv->resendActivePowerControlRequest();
|
|
}
|
|
|
|
// Set power status if required
|
|
if (iv->PowerCommand()->getLastPowerCommandSuccess() == CMD_NOK) {
|
|
_messageOutput->println("Resend PowerCommand");
|
|
iv->resendPowerControlRequest();
|
|
}
|
|
|
|
// Fetch dev info (but first fetch stats)
|
|
if (iv->Statistics()->getLastUpdate() > 0 && (iv->DevInfo()->getLastUpdateAll() == 0 || iv->DevInfo()->getLastUpdateSimple() == 0)) {
|
|
_messageOutput->println("Request device info");
|
|
iv->sendDevInfoRequest();
|
|
}
|
|
}
|
|
if (++inverterPos >= getNumInverters()) {
|
|
inverterPos = 0;
|
|
}
|
|
}
|
|
|
|
_lastPoll = millis();
|
|
}
|
|
}
|
|
|
|
HOY_SEMAPHORE_GIVE();
|
|
}
|
|
|
|
std::shared_ptr<InverterAbstract> HoymilesClass::addInverter(const char* name, uint64_t serial)
|
|
{
|
|
std::shared_ptr<InverterAbstract> i = nullptr;
|
|
if (HMS_4CH::isValidSerial(serial)) {
|
|
i = std::make_shared<HMS_4CH>(_radioCmt.get(), serial);
|
|
} else if (HMS_2CH::isValidSerial(serial)) {
|
|
i = std::make_shared<HMS_2CH>(_radioCmt.get(), serial);
|
|
} else if (HMS_1CH::isValidSerial(serial)) {
|
|
i = std::make_shared<HMS_1CH>(_radioCmt.get(), serial);
|
|
} else if (HM_4CH::isValidSerial(serial)) {
|
|
i = std::make_shared<HM_4CH>(_radioNrf.get(), serial);
|
|
} else if (HM_2CH::isValidSerial(serial)) {
|
|
i = std::make_shared<HM_2CH>(_radioNrf.get(), serial);
|
|
} else if (HM_1CH::isValidSerial(serial)) {
|
|
i = std::make_shared<HM_1CH>(_radioNrf.get(), serial);
|
|
}
|
|
|
|
if (i) {
|
|
i->setName(name);
|
|
i->init();
|
|
_inverters.push_back(std::move(i));
|
|
return _inverters.back();
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
std::shared_ptr<InverterAbstract> HoymilesClass::getInverterByPos(uint8_t pos)
|
|
{
|
|
if (pos >= _inverters.size()) {
|
|
return nullptr;
|
|
} else {
|
|
return _inverters[pos];
|
|
}
|
|
}
|
|
|
|
std::shared_ptr<InverterAbstract> HoymilesClass::getInverterBySerial(uint64_t serial)
|
|
{
|
|
for (uint8_t i = 0; i < _inverters.size(); i++) {
|
|
if (_inverters[i]->serial() == serial) {
|
|
return _inverters[i];
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
std::shared_ptr<InverterAbstract> HoymilesClass::getInverterByFragment(fragment_t* fragment)
|
|
{
|
|
if (fragment->len <= 4) {
|
|
return nullptr;
|
|
}
|
|
|
|
std::shared_ptr<InverterAbstract> inv;
|
|
for (uint8_t i = 0; i < _inverters.size(); i++) {
|
|
inv = _inverters[i];
|
|
serial_u p;
|
|
p.u64 = inv->serial();
|
|
|
|
if ((p.b[3] == fragment->fragment[1])
|
|
&& (p.b[2] == fragment->fragment[2])
|
|
&& (p.b[1] == fragment->fragment[3])
|
|
&& (p.b[0] == fragment->fragment[4])) {
|
|
|
|
return inv;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void HoymilesClass::removeInverterBySerial(uint64_t serial)
|
|
{
|
|
for (uint8_t i = 0; i < _inverters.size(); i++) {
|
|
if (_inverters[i]->serial() == serial) {
|
|
HOY_SEMAPHORE_TAKE();
|
|
_inverters.erase(_inverters.begin() + i);
|
|
HOY_SEMAPHORE_GIVE();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
size_t HoymilesClass::getNumInverters()
|
|
{
|
|
return _inverters.size();
|
|
}
|
|
|
|
HoymilesRadio_NRF* HoymilesClass::getRadioNrf()
|
|
{
|
|
return _radioNrf.get();
|
|
}
|
|
|
|
uint32_t HoymilesClass::PollInterval()
|
|
{
|
|
return _pollInterval;
|
|
}
|
|
|
|
void HoymilesClass::setPollInterval(uint32_t interval)
|
|
{
|
|
_pollInterval = interval;
|
|
}
|
|
|
|
void HoymilesClass::setMessageOutput(Print* output)
|
|
{
|
|
_messageOutput = output;
|
|
}
|
|
|
|
Print* HoymilesClass::getMessageOutput()
|
|
{
|
|
return _messageOutput;
|
|
} |