-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathSamilCommunicator.h
107 lines (93 loc) · 3.6 KB
/
SamilCommunicator.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
#pragma once
#include <vector>
#include <SoftwareSerial.h>
#include <ESP8266WiFi.h>
#include "SettingsManager.h"
#define SAMIL_COMMS_ADDRES 0x00
#define SAMIL_COMMS_ADDRESS 0x00
#define PACKET_TIMEOUT 500 //0.5 sec packet timeout
#define OFFLINE_TIMEOUT 30000 //30 seconds no data -> inverter offline
#define DISCOVERY_INTERVAL 10000 //10 secs between discovery
#define INFO_INTERVAL 1000 //get inverter info every second
class SamilCommunicator
{
public:
struct SamilInverterInformation
{
char serialNumber[17]; //serial number (ascii) from inverter with zero appended
char address; //address provided by this software
bool addressConfirmed; //wether or not the address is confirmed by te inverter
unsigned long lastSeen; //when was the inverter last seen? If not seen for 30 seconds the inverter is marked offline.
bool isOnline; //is the inverter online (see above)
bool isDTSeries; //is tri phase inverter (get phase 2, 3 info)
//inverert info from inverter pdf. Updated by the inverter info command
float vpv1=0.0;
float vpv2=0.0;
float ipv1=0.0;
float ipv2 = 0.0;
float vac1 = 0.0;
float vac2 = 0.0;
float vac3 = 0.0;
float iac1 = 0.0;
float iac2 = 0.0;
float iac3 = 0.0;
float fac1 = 0.0;
float fac2 = 0.0;
float fac3 = 0.0;
short pac=0;
short workMode=0;
float temp = 0.0;
int errorMessage=0;
int eTotal=0;
int hTotal=0;
float tempFault = 0.0;
float pv1Fault = 0.0;
float pv2Fault = 0.0;
float line1VFault = 0.0;
float line2VFault = 0.0;
float line3VFault = 0.0;
float line1FFault = 0.0;
float line2FFault = 0.0;
float line3FFault = 0.0;
short gcfiFault=0;
float eDay = 0.0;
};
SamilCommunicator(SettingsManager * settingsManager, bool debugMode = false);
void start();
void stop();
void handle();
std::vector<SamilInverterInformation> getInvertersInfo();
~SamilCommunicator();
private:
static const int BufferSize = 96; // largest packet is 67 bytes long. Extra for receiving with sliding window
SoftwareSerial * samilSerial;
SettingsManager * settingsManager;
char headerBuffer[7];
char inputBuffer[BufferSize];
char outputBuffer[BufferSize];
bool debugMode;
unsigned long lastReceived = 0; //timeout detection
bool startPacketReceived = false; //start packet marker
char lastReceivedByte = 0; //packet start consist of 2 bytes to test. This holds the previous byte
int curReceivePtr = 0; //the ptr in our OutputBuffer when reading
int numToRead = 0; //number of bytes to read after the header is read.
unsigned long lastDiscoverySent = 0; //discovery needs to be sent every 10 secs.
unsigned long lastInfoUpdateSent = 0; //last info update sent to the registered inverters
char lastUsedAddress = 0; //last used address counter. When overflows will only allocate not used
std::vector<SamilCommunicator::SamilInverterInformation> inverters;
int sendData(unsigned int address, char controlCode, char functionCode, char dataLength, char * data);
void debugPrintHex(char cnt);
void sendDiscovery();
void checkOfflineInverters();
void checkIncomingData();
void parseIncomingData(char dataLength);
void handleRegistration(char * serialNumber, char length);
void handleRegistrationConfirmation(char address);
void handleIncomingInformation(char address, char dataLengthh, char * data);
float bytesToFloat(char * bt, char factor);
void askAllInvertersForInformation();
void askInverterForInformation(char address);
SamilCommunicator::SamilInverterInformation * getInverterInfoByAddress(char address);
void sendAllocateRegisterAddress(char * serialNumber, char Address);
//void sendRemoveRegistration(char address);
};