/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright 2013 - 2025, nymea GmbH * Contact: contact@nymea.io * * This fileDescriptor is part of nymea. * This project including source code and documentation is protected by * copyright law, and remains the property of nymea GmbH. All rights, including * reproduction, publication, editing and translation, are reserved. The use of * this project is subject to the terms of a license agreement to be concluded * with nymea GmbH in accordance with the terms of use of nymea GmbH, available * under https://nymea.io/license * * GNU Lesser General Public License Usage * Alternatively, this project may be redistributed and/or modified under the * terms of the GNU Lesser General Public License as published by the Free * Software Foundation; version 3. This project 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this project. If not, see . * * For any further details and any questions please contact us under * contact@nymea.io or see our FAQ/Licensing Information on * https://nymea.io/license/faq * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "sunspecflowbatterymodulemodel.h" #include "sunspecconnection.h" SunSpecFlowBatteryModuleModelRepeatingBlock::SunSpecFlowBatteryModuleModelRepeatingBlock(quint16 blockIndex, quint16 blockSize, quint16 modbusStartRegister, SunSpecFlowBatteryModuleModel *parent) : SunSpecModelRepeatingBlock(blockIndex, blockSize, modbusStartRegister, parent) { m_parentModel = parent; m_byteOrder = parent->byteOrder(); initDataPoints(); } QString SunSpecFlowBatteryModuleModelRepeatingBlock::name() const { return "stack"; } SunSpecFlowBatteryModuleModel *SunSpecFlowBatteryModuleModelRepeatingBlock::parentModel() const { return m_parentModel; } quint16 SunSpecFlowBatteryModuleModelRepeatingBlock::stackPointsToBeDetermined() const { return m_stackPointsToBeDetermined; } void SunSpecFlowBatteryModuleModelRepeatingBlock::initDataPoints() { SunSpecDataPoint stackPointsToBeDeterminedDataPoint; stackPointsToBeDeterminedDataPoint.setName("StackTBD"); stackPointsToBeDeterminedDataPoint.setLabel("Stack Points To Be Determined"); stackPointsToBeDeterminedDataPoint.setMandatory(true); stackPointsToBeDeterminedDataPoint.setSize(1); stackPointsToBeDeterminedDataPoint.setAddressOffset(0); stackPointsToBeDeterminedDataPoint.setSunSpecDataType("uint16"); stackPointsToBeDeterminedDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(stackPointsToBeDeterminedDataPoint.name(), stackPointsToBeDeterminedDataPoint); } void SunSpecFlowBatteryModuleModelRepeatingBlock::processBlockData() { // Update properties according to the data point type if (m_dataPoints.value("StackTBD").isValid()) m_stackPointsToBeDetermined = m_dataPoints.value("StackTBD").toUInt16(); qCDebug(dcSunSpecModelData()) << this; } SunSpecFlowBatteryModuleModel::SunSpecFlowBatteryModuleModel(SunSpecConnection *connection, quint16 modbusStartRegister, quint16 modelLength, SunSpecDataPoint::ByteOrder byteOrder, QObject *parent) : SunSpecModel(connection, modbusStartRegister, 808, modelLength, byteOrder, parent) { m_modelBlockType = SunSpecModel::ModelBlockTypeFixedAndRepeating; initDataPoints(); connect(this, &SunSpecModel::initFinished, this, &SunSpecFlowBatteryModuleModel::setupRepeatingBlocks); } SunSpecFlowBatteryModuleModel::~SunSpecFlowBatteryModuleModel() { } QString SunSpecFlowBatteryModuleModel::name() const { return "flow_battery_module"; } QString SunSpecFlowBatteryModuleModel::description() const { return QString(); } QString SunSpecFlowBatteryModuleModel::label() const { return "Flow Battery Module Model"; } quint16 SunSpecFlowBatteryModuleModel::modulePointsToBeDetermined() const { return m_modulePointsToBeDetermined; } void SunSpecFlowBatteryModuleModel::initDataPoints() { SunSpecDataPoint modelIdDataPoint; modelIdDataPoint.setName("ID"); modelIdDataPoint.setLabel("Model ID"); modelIdDataPoint.setDescription("Model identifier"); modelIdDataPoint.setMandatory(true); modelIdDataPoint.setSize(1); modelIdDataPoint.setAddressOffset(0); modelIdDataPoint.setSunSpecDataType("uint16"); modelIdDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(modelIdDataPoint.name(), modelIdDataPoint); SunSpecDataPoint modelLengthDataPoint; modelLengthDataPoint.setName("L"); modelLengthDataPoint.setLabel("Model Length"); modelLengthDataPoint.setDescription("Model length"); modelLengthDataPoint.setMandatory(true); modelLengthDataPoint.setSize(1); modelLengthDataPoint.setAddressOffset(1); modelLengthDataPoint.setSunSpecDataType("uint16"); modelLengthDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(modelLengthDataPoint.name(), modelLengthDataPoint); SunSpecDataPoint modulePointsToBeDeterminedDataPoint; modulePointsToBeDeterminedDataPoint.setName("ModuleTBD"); modulePointsToBeDeterminedDataPoint.setLabel("Module Points To Be Determined"); modulePointsToBeDeterminedDataPoint.setMandatory(true); modulePointsToBeDeterminedDataPoint.setSize(1); modulePointsToBeDeterminedDataPoint.setAddressOffset(2); modulePointsToBeDeterminedDataPoint.setBlockOffset(0); modulePointsToBeDeterminedDataPoint.setSunSpecDataType("uint16"); modulePointsToBeDeterminedDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(modulePointsToBeDeterminedDataPoint.name(), modulePointsToBeDeterminedDataPoint); } void SunSpecFlowBatteryModuleModel::processBlockData() { // Update properties according to the data point type if (m_dataPoints.value("ModuleTBD").isValid()) m_modulePointsToBeDetermined = m_dataPoints.value("ModuleTBD").toUInt16(); qCDebug(dcSunSpecModelData()) << this; } void SunSpecFlowBatteryModuleModel::setupRepeatingBlocks() { if (!m_repeatingBlocks.isEmpty()) { foreach (SunSpecModelRepeatingBlock *block, m_repeatingBlocks) { block->deleteLater(); } m_repeatingBlocks.clear(); } const auto headerLength = 2; const auto repeatingBlocksDataSize = m_blockData.size() - headerLength - m_fixedBlockLength; if (repeatingBlocksDataSize % m_repeatingBlockLength != 0) { qCWarning(dcSunSpecModelData()) << "Unexpected repeating block data size:" << repeatingBlocksDataSize << "(repeating block size:" << m_repeatingBlockLength << ", extra bytes:" << repeatingBlocksDataSize % m_repeatingBlockLength << "). Repeating blocks will not be handled!"; return; } const auto numberOfBlocks = repeatingBlocksDataSize / m_repeatingBlockLength; const auto repeatingBlocksOffset = m_fixedBlockLength + headerLength; for (int i = 0; i < numberOfBlocks; ++i) { const auto blockStartRegister = static_cast(modbusStartRegister() + repeatingBlocksOffset + m_repeatingBlockLength * i); const auto block = new SunSpecFlowBatteryModuleModelRepeatingBlock(i, m_repeatingBlockLength, blockStartRegister, this); m_repeatingBlocks.append(block); } } QDebug operator<<(QDebug debug, SunSpecFlowBatteryModuleModel *model) { debug.nospace().noquote() << "SunSpecFlowBatteryModuleModel(Model: " << model->modelId() << ", Register: " << model->modbusStartRegister() << ", Length: " << model->modelLength() << ")\n"; debug.nospace().noquote() << " - " << model->dataPoints().value("ModuleTBD") << "-->"; if (model->dataPoints().value("ModuleTBD").isValid()) { debug.nospace().noquote() << model->modulePointsToBeDetermined() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } return debug.space().quote(); }