10 Commits

Author SHA1 Message Date
09d6afe5eb neuro fix 2024-12-11 17:50:16 +03:00
231b43d8bd tmp fix 2024-12-11 14:42:13 +03:00
a4d01ea360 fix state 2024-12-11 13:45:56 +03:00
a9f75262a5 debug 2024-12-09 17:24:34 +03:00
a78990110d upd 2024-12-09 16:03:47 +03:00
9934fda584 vanilla 2024-12-09 15:36:16 +03:00
89c91cb68f d 2024-12-09 15:06:46 +03:00
a95cc42233 upd 2024-12-09 14:57:43 +03:00
daf17f1ac6 upd 2024-12-05 17:56:48 +03:00
2fb1f546fa upd 2024-12-04 12:32:03 +03:00
7 changed files with 424 additions and 325 deletions

View File

@ -0,0 +1,150 @@
#include "EthernetMaket.h" // Предполагается, что этот файл подключает Ethernet3.h
#include <map>
// Настройки для W5500
#define W5500_CS_PIN PA2 // Пин для CS
#define SOCKET_NUMBER 0 // Используем сокет 0
#define SERVER_PORT 1337 // Порт на сервере
#define CLIENT_PORT 1337 // Порт клиента
#define ServerMac {0xDA, 0x7A, 0xFF, 0x00, 0x00, 0x00}
#define ServerIP {192, 168, 254, 254}
#define IniMac {0xDA, 0x7A, 0xF0, 0x00, 0x00, 0x00}
#define IniIP {192, 168, 254, 253}
uint8_t gateway[] = {192, 168, 0, 1}; // Шлюз
uint8_t subnet[] = {255, 255, 0, 0}; // Маска подсети
uint8_t mac[] = IniMac; // MAC-адрес
uint8_t localIP[] = IniIP; // Локальный IP-адрес
uint8_t serverMac[] = ServerMac; // MAC-адрес сервера
uint8_t serverIP[] = ServerIP; // IP-адрес сервера
uint8_t buf[512]; // Буфер для передачи данных
uint32_t previousMillis = 0; // Для хранения времени последней отправки
const long interval = 3750; // Интервал отправки данных (1.75 секунды)
uint32_t ttt;
EthernetServer server(SERVER_PORT);
struct EthernetMaketClientStruct
{
EthernetMaketClient clientMember; //
};
std::map<uint16_t, EthernetMaketClientStruct> clientMap;
uint32_t packetCounter = 0;
uint32_t data1;
uint32_t data2;
void setup()
{
Serial.begin(115200);
pinMode(LED_BUILTIN, OUTPUT);
Ethernet.setCsPin(W5500_CS_PIN);
Ethernet.init();
// Настройка Ethernet с указанием статических параметров
Ethernet.begin(mac, localIP, subnet, gateway, gateway);
EthernetMaketClientStruct c1;
EthernetMaketClientStruct c2;
clientMap.insert({1, c1});
clientMap.insert({2, c2});
}
void loop()
{
EthernetMaketClient::tick();
uint32_t currentMillis = millis();
if (currentMillis - ttt > 25)
{
digitalToggle(PC13);
ttt = currentMillis; // Обновляем время
}
if (currentMillis - previousMillis >= interval)
{
previousMillis = currentMillis;
Serial.println("connectNonBlock");
clientMap[1].clientMember.startConnection(serverIP, SERVER_PORT);
clientMap[2].clientMember.startConnection(serverIP, SERVER_PORT);
//* for single packets
Serial.print("Sending packet ONE: ");
data1 = packetCounter;
clientMap[1].clientMember.dataWrite((uint8_t *)&data1, sizeof(data1));
for (size_t j = 0; j < sizeof(data1); j++)
{
Serial.print(((uint8_t *)&data1)[j], HEX);
Serial.print(" ");
}
Serial.println();
packetCounter++;
data2 = packetCounter;
Serial.print("Sending packet TWO: ");
clientMap[2].clientMember.dataWrite((uint8_t *)&data2, sizeof(data2));
for (size_t j = 0; j < sizeof(data2); j++)
{
Serial.print(((uint8_t *)&data2)[j], HEX);
Serial.print(" ");
}
Serial.println();
packetCounter++;
}
if(clientMap[1].clientMember.available()>0){
byte response[16];
int len = clientMap[1].clientMember.read(response, sizeof(response));
if (len == 16)
{
// Проверка контрольной суммы
uint8_t checksum = 0;
for (int i = 0; i < 15; i++)
{
checksum += response[i];
}
Serial.print("RX: ");
for (int i = 0; i < 16; i++)
{
Serial.print(response[i], HEX);
Serial.print(" ");
}
if (response[15] == checksum)
{
Serial.println("Checksum is OK");
}
else
{
Serial.println("Checksum is ERROR");
}
clientMap[1].clientMember.activityUpdate(); // Обновляем время последней активности
}
else
{
Serial.println("Received an incomplete response");
}
}
}

View File

@ -25,21 +25,25 @@ void serverHandler(EthernetClient client)
byte buf[512]; byte buf[512];
int len = client.read(buf, sizeof(buf)); int len = client.read(buf, sizeof(buf));
// Обработка каждого 4-байтового пакета Serial.print("sock: ");
for (int i = 0; i < len; i += 4) Serial.println(client.getSocketNumber());
{ processReceivedData(buf, len);
int packetLen = min(4, len - i); // Определяем длину текущего пакета
processReceivedData(buf + i, packetLen); // // Обработка каждого 4-байтового пакета
for (uint8_t j = 0; j < 3; j++) // for (int i = 0; i < len; i += 4)
{ // {
sendResponse(client, *(uint32_t *)(buf + i)); // int packetLen = min(4, len - i); // Определяем длину текущего пакета
} // processReceivedData(buf + i, packetLen);
packetCounter++; // for (uint8_t j = 0; j < 3; j++)
} // {
// sendResponse(client, *(uint32_t *)(buf + i));
// }
// packetCounter++;
// }
//!----------------- //!-----------------
Serial.println("DONE"); Serial.println("DONE");
client.stop(); // client.stop();
Serial.println(); Serial.println();
Serial.println(); Serial.println();
} }

View File

@ -12,88 +12,124 @@ void EthernetMaketServer::resetServerHandler()
serverHandler = [](EthernetMaketClient sclient) {}; serverHandler = [](EthernetMaketClient sclient) {};
} }
EthernetMaketClient EthernetMaketServer::available(int sock) // EthernetMaketClient EthernetMaketServer::available(int sock)
{ // {
accept(sock); // accept(sock);
EthernetMaketClient client(sock); // EthernetClient client(sock);
if (EthernetClass::_server_port[sock] == _port && // if (EthernetClass::_server_port[sock] == _port &&
(client.status() == SnSR::ESTABLISHED || // (client.status() == SnSR::ESTABLISHED ||
client.status() == SnSR::CLOSE_WAIT)) // client.status() == SnSR::CLOSE_WAIT))
{ // {
if (client.available()) // if (client.available())
{ // {
return client; // return std::move(EthernetMaketClient(client));
} // }
} // }
return EthernetMaketClient(MAX_SOCK_NUM); // return std::move(EthernetMaketClient(MAX_SOCK_NUM));
} // }
void EthernetMaketServer::accept(int sock) // void EthernetMaketServer::accept(int sock)
{ // {
int listening = 0; // int listening = 0;
EthernetClient client(sock); // EthernetClient client(sock);
if (EthernetClass::_server_port[sock] == _port) // if (EthernetClass::_server_port[sock] == _port)
{ // {
if (client.status() == SnSR::LISTEN) // if (client.status() == SnSR::LISTEN)
{ // {
listening = 1; // listening = 1;
} // }
else if (client.status() == SnSR::CLOSE_WAIT && !client.available()) // else if (client.status() == SnSR::CLOSE_WAIT && !client.available())
{ // {
client.stop(); // Serial.println("HARD STOP");
} // client.stop();
} // }
// }
if (!listening) // if (!listening)
{ // {
// begin(); // // begin();
begin(sock); // added // begin(sock); // added
} // }
} // }
void EthernetMaketServer::begin(int sock) // void EthernetMaketServer::begin(int sock)
{ // {
EthernetClient client(sock); // EthernetClient client(sock);
if (client.status() == SnSR::CLOSED) // if (client.status() == SnSR::CLOSED)
{ // {
socket(sock, SnMR::TCP, _port, 0); // socket(sock, SnMR::TCP, _port, 0);
listen(sock); // listen(sock);
EthernetClass::_server_port[sock] = _port; // EthernetClass::_server_port[sock] = _port;
} // serverSocket = sock;
} // return;
// }
void EthernetMaketServer::begin() // // Serial.println(F("EthernetMaketServer::begin() FAIL"));
{ // }
EthernetServer::begin();
}
// void EthernetMaketServer::begin()
// {
// for (int sock = 0; sock < MAX_SOCK_NUM; sock++) {
// EthernetClient client(sock);
// if (client.status() == SnSR::CLOSED) {
// socket(sock, SnMR::TCP, _port, 0);
// listen(sock);
// EthernetClass::_server_port[sock] = _port;
// serverSocket = sock;
// break;
// }
// }
// // Serial.println(F("EthernetMaketServer::begin() FAIL"));
// }
uint8_t buf[256];
void EthernetMaketServer::tick() void EthernetMaketServer::tick()
{ {
for (int sock = 0; sock < MAX_SOCK_NUM; sock++) // for (int sock = 0; sock < MAX_SOCK_NUM; sock++)
{ {
EthernetMaketClient sclient = available(sock); EthernetClient sclient = available(/* serverSocket */);
if (sclient.connected() && sclient.available() > 0) size_t dataSize;
if (sclient.connected() && (dataSize = sclient.available()) > 0)
{ {
serverHandler(sclient); // Serial.println("Receive!");
}
// // Закрытие соединения по таймеру неактивности // memset(buf,0x00,sizeof(buf));
// if (sclient.connected() && (millis() - lastActivityTime > timeout)) // sclient.read(buf, sizeof(buf));
// Serial.print("Received data: ");
// for (int i = 0; i < dataSize; i++)
// { // {
// Serial.println("Connection closed due to inactivity"); // Serial.print(buf[i], HEX);
// sclient.stop(); // Serial.print(" ");
// } // }
// Serial.println("\n");
// sclient.activityUpdate();
serverHandler(EthernetMaketClient(sclient));
}
// if (!wasConnected && sclient.connected()) // if (!wasConnected && sclient.connected())
// { // {
// wasConnected[sock] = true; // wasConnected = true;
// Serial.println("Client connected");
// } // }
// if (wasConnected && !sclient.connected()) // if (wasConnected && !sclient.connected())
// { // {
// wasConnected[sock] = false; // wasConnected = false;
// sclient.stop(); // sclient.stop();
// // begin(serverSocket);
// Serial.println("Client disconnected"); // Serial.println("Client disconnected");
// } // }
// // Закрытие соединения по таймеру неактивности
// if (sclient.connected() && (millis() - lastActivityTime > timeout))
// {
// Serial.println("Connection closed due to inactivity");
// sclient.close();
// // begin(serverSocket);
// }
} }
// sclient.tick_();
} }

View File

@ -8,19 +8,22 @@ class EthernetMaketClient;
class EthernetMaketServer : public EthernetServer class EthernetMaketServer : public EthernetServer
{ {
private: private:
EthernetClient available(); // EthernetClient available();
void accept(int sock); // void accept(int sock);
std::function<void(EthernetMaketClient)> serverHandler = [](EthernetMaketClient sclient) {}; std::function<void(EthernetMaketClient)> serverHandler = [](EthernetMaketClient sclient) {};
bool wasConnected[MAX_SOCK_NUM]; SOCKET serverSocket = MAX_SOCK_NUM;
bool wasConnected;
// EthernetMaketClient sclient;
public: public:
using EthernetServer::EthernetServer; using EthernetServer::EthernetServer;
EthernetMaketClient available(int sock); // EthernetMaketClient available(int sock);
void setServerHandler(std::function<void(EthernetMaketClient)> handler); void setServerHandler(std::function<void(EthernetMaketClient)> handler);
void resetServerHandler(); void resetServerHandler();
void begin() override; // void begin() override;
void begin(int sock); // void begin(int sock);
void tick(); void tick();
}; };

View File

@ -2,283 +2,195 @@
#include "utility/w5500.h" #include "utility/w5500.h"
#include "utility/socket.h" #include "utility/socket.h"
std::list<EthernetMaketClient*>& EthernetMaketClient::getInstances() {
static std::list<EthernetMaketClient*> instances; // Список создаётся при первом доступе
return instances;
}
EthernetMaketClient::EthernetMaketClient() { EthernetMaketClient::EthernetMaketClient() {
_sock = MAX_SOCK_NUM; _sock = MAX_SOCK_NUM;
getInstances().push_back(this);
} }
EthernetMaketClient::EthernetMaketClient(uint8_t sock) { EthernetMaketClient::EthernetMaketClient(uint8_t sock) {
_sock = sock; _sock = sock;
getInstances().push_back(this);
} }
EthernetMaketClient::~EthernetMaketClient() { EthernetMaketClient::EthernetMaketClient(EthernetClient &client) {
getInstances().remove(this); _sock = client.getSocketNumber();
} }
void EthernetMaketClient::tick() { EthernetMaketClient& EthernetMaketClient::operator=(const EthernetMaketClient& other) {
for (EthernetMaketClient* client : getInstances()) { if (this == &other) {
client->tick_(); return *this; // Защита от самоприсваивания
} }
this->lastActivityTime = other.lastActivityTime;
this->startConnectionTime = other.startConnectionTime;
this->stopStartTime = other.stopStartTime;
this->isNonBlocking = other.isNonBlocking;
this->connectStatus = other.connectStatus;
this->dstIP = other.dstIP;
this->dstPort = other.dstPort;
this->dataPtr = other.dataPtr;
this->dataSize = other.dataSize;
this->onSendSuccess = other.onSendSuccess;
this->_sock = other._sock;
const_cast<EthernetMaketClient&>(other).connectStatus = CONNECT_IDLE;
return *this;
} }
ConnectionStatusSimple EthernetMaketClient::connectNonBlock(IPAddress ip, uint16_t port) { ConnectionStatusSimple EthernetMaketClient::connectNonBlock(IPAddress ip, uint16_t port) {
dstIP = ip; dstIP = ip;
dstPort = port; dstPort = port;
// if(connectStatus == CONNECT_IDLE){
// connectStatus = CONNECT_START;
// }
if (_sock != MAX_SOCK_NUM) { if (_sock != MAX_SOCK_NUM) {
uint8_t status_ = status(); uint8_t status_ = status();
switch (status_) { switch (status_) {
case SnSR::CLOSED:
Serial.println(F("connectNonBlock(): CONNECT_FAIL (CLOSED)"));
_sock = MAX_SOCK_NUM;
return CONNECT_FAIL;
break;
case SnSR::ESTABLISHED: case SnSR::ESTABLISHED:
Serial.println(F("connectNonBlock(): CONNECT_SUCCESS (ESTABLISHED)")); connectStatus = CONNECT_SUCCESS;
return CONNECT_SUCCESS; return CONNECT_SUCCESS;
case SnSR::CLOSED:
case SnSR::CLOSE_WAIT:
close(); // Освобождаем сокет перед попыткой нового подключения
break; break;
default: default:
// Serial.println(F("connectNonBlock(): CONNECT_CONNECTING (Default)"));
return CONNECT_CONNECTING; return CONNECT_CONNECTING;
break;
} }
} }
for (int i = 0; i < MAX_SOCK_NUM; i++) { for (int i = 0; i < MAX_SOCK_NUM; i++) {
uint8_t s = w5500.readSnSR(i); uint8_t s = w5500.readSnSR(i);
if (s == SnSR::CLOSED || s == SnSR::FIN_WAIT || s == SnSR::CLOSE_WAIT) { if (s == SnSR::CLOSED || s == SnSR::CLOSE_WAIT) {
_sock = i; _sock = i;
break; break;
} }
} }
if (_sock == MAX_SOCK_NUM) { if (_sock == MAX_SOCK_NUM) {
Serial.println(F("connectNonBlock(): CONNECT_FAIL (No available socket)")); connectStatus = CONNECT_FAIL;
return CONNECT_FAIL; return CONNECT_FAIL;
} }
Serial.print("sock: ");
Serial.println(_sock);
_srcport++; _srcport++;
if (_srcport == 0) _srcport = 1024; if (_srcport == 0) _srcport = 1024;
socket(_sock, SnMR::TCP, _srcport, 0); socket(_sock, SnMR::TCP, _srcport, 0);
if (!::connect(_sock, rawIPAddress(ip), port)) { if (!::connect(_sock, rawIPAddress(ip), port)) {
Serial.println(F("connectNonBlock(): CONNECT_FAIL (Connect failed)"));
_sock = MAX_SOCK_NUM; _sock = MAX_SOCK_NUM;
connectStatus = CONNECT_FAIL;
return CONNECT_FAIL; return CONNECT_FAIL;
} }
Serial.println(F("connectNonBlock(): CONNECT_CONNECTING (Connection initiated)")); connectStatus = CONNECT_CONNECTING;
return CONNECT_CONNECTING; return CONNECT_CONNECTING;
} }
void EthernetMaketClient::tick_() {
if (_sock == MAX_SOCK_NUM) {
connectStatus = CONNECT_IDLE;
return;
}
bool EthernetMaketClient::isConnected() const { uint8_t s = w5500.readSnSR(_sock);
return connectStatus == CONNECT_CONNECTED;
switch (s) {
case SnSR::CLOSED:
connectStatus = (connectStatus == CONNECT_CONNECTING) ? CONNECT_FAIL : CONNECT_STOP_WAITING;
close();
break;
case SnSR::ESTABLISHED:
if (connectStatus != CONNECT_CONNECTED) {
connectStatus = CONNECT_SUCCESS;
}
break;
case SnSR::CLOSE_WAIT:
connectStatus = CONNECT_STOP_START;
disconnect();
break;
case SnSR::FIN_WAIT:
case SnSR::CLOSING:
case SnSR::TIME_WAIT:
case SnSR::LAST_ACK:
close();
connectStatus = CONNECT_STOP_WAITING;
break;
default:
break;
}
// Проверка активности и таймаутов
if (connectStatus == CONNECT_CONNECTED) {
if (millis() - lastActivityTime > timeout) {
connectStatus = CONNECT_STOP_START;
disconnect();
}
} else if (connectStatus == CONNECT_CONNECTING) {
if (millis() - startConnectionTime >= connectionTimeout) {
connectStatus = CONNECT_FAIL;
close();
}
}
} }
void EthernetMaketClient::disconnect() { void EthernetMaketClient::disconnect() {
if (_sock == MAX_SOCK_NUM) if (_sock != MAX_SOCK_NUM) {
return;
::disconnect(_sock); ::disconnect(_sock);
} }
}
void EthernetMaketClient::close() { void EthernetMaketClient::close() {
if (_sock == MAX_SOCK_NUM) if (_sock != MAX_SOCK_NUM) {
return;
::close(_sock); ::close(_sock);
_sock = MAX_SOCK_NUM; _sock = MAX_SOCK_NUM;
} }
}
void EthernetMaketClient::tick_(){ void EthernetMaketClient::startConnection(IPAddress ip, uint16_t port, bool nonBlock) {
switch (connectStatus) dstIP = ip;
{ dstPort = port;
case CONNECT_START: isNonBlocking = nonBlock;
Serial.println("CONNECT_START"); connectStatus = CONNECT_START;
startConnectionTime = millis(); startConnectionTime = millis();
connectStatus = CONNECT_CONNECTING;
close();
Serial.println("CONNECT_CONNECTING...");
break;
case CONNECT_CONNECTING:
if (millis() - startConnectionTime >= connectionTimeout)
{
connectStatus = CONNECT_FAIL;
break;
}
break;
case CONNECT_FAIL:
Serial.println("CONNECT_FAIL");
Serial.print("Failed to connect to server in ");
Serial.println(isNonBlocking ? "non-blocking mode" : "blocking mode");
Serial.println("\n");
connectStatus = CONNECT_STOP_START;
break;
case CONNECT_SUCCESS:
Serial.println("CONNECT_SUCCESS");
Serial.print("Connected to server in ");
Serial.println(isNonBlocking ? "non-blocking mode" : "blocking mode");
lastActivityTime = millis();
connectStatus = CONNECT_CONNECTED;
lastActivityTime = millis(); // Обновляем время последней активности
break;
case CONNECT_CONNECTED:
//! Передача пакетов
dataWrite();
//! Передача завершена
if (!connected())
{
Serial.println("Client disconnected");
connectStatus = CONNECT_STOP_START;
}
// Закрытие соединения по таймеру неактивности
if (connected() && (millis() - lastActivityTime > timeout))
{
Serial.println("Connection closed due to inactivity");
connectStatus = CONNECT_STOP_START;
}
break;
case CONNECT_STOP_START:
Serial.println("CONNECT_STOP_START");
dstIP = IPAddress(0,0,0,0);
dstPort = 0;
disconnect();
stopStartTime = millis();
connectStatus = CONNECT_STOP_WAITING;
Serial.println("CONNECT_STOP_WAITING");
break;
case CONNECT_STOP_WAITING:
if (millis() - stopStartTime >= stopTimeout) {
close(); // Окончательное закрытие сокета после таймаута
connectStatus = CONNECT_IDLE;
} else {
// Проверяем состояние сокета для завершения разрыва
uint8_t status_ = status();
if (status_ == SnSR::CLOSED || status_ == SnSR::CLOSE_WAIT) {
close(); // Закрываем при достижении состояния
connectStatus = CONNECT_IDLE;
}
}
if(connectStatus == CONNECT_IDLE) {
Serial.println("CONNECT_IDLE\n\n");
}
break;
case CONNECT_IDLE:
// if(dataSize){
// connectStatus = CONNECT_START;
// }
break;
default:
break;
}
//********************************************************
switch (connectStatus)
{
case CONNECT_START:
case CONNECT_CONNECTING:
if (isNonBlocking)
{
connectStatus = connectNonBlock(dstIP, dstPort);
}
else
{
connectStatus = (ConnectionStatusSimple)connect(dstIP, dstPort);
}
break;
default:
break;
}
//********************************************************
// //? Проверка наличия доступных данных от сервера
// if (available() > 0)
// {
// }
// //? -------------------------------------------------
} }
bool EthernetMaketClient::dataWrite() { bool EthernetMaketClient::dataWrite() {
bool ret = false; if (!canWriteIntoConnection(dataSize)) return false;
if(dataSize) {
lastActivityTime = millis();
if (write(dataPtr, dataSize)) { if (write(dataPtr, dataSize)) {
dataPtr = nullptr; clearData();
dataSize = 0; // Очищаем размер данных
ret = true;
}
lastActivityTime = millis(); lastActivityTime = millis();
}
if(ret){
onSendSuccess(); onSendSuccess();
return true;
} }
return ret;
return false;
} }
void EthernetMaketClient::dataWrite(uint8_t* data, uint16_t dataSize, bool override) { void EthernetMaketClient::dataWrite(uint8_t* data, uint16_t dataSize, bool override) {
if (this->dataSize == 0 || override) { if (this->dataPtr == nullptr || override) {
this->dataPtr = data; this->dataPtr = data;
this->dataSize = dataSize; this->dataSize = dataSize;
if(isConnected()){ activityUpdate();
dataWrite();
}
} else { } else {
Serial.println("Data buffer is not empty. Write operation is ignored."); Serial.println("Data buffer is not empty. Write operation is ignored.");
} }
} }
void EthernetMaketClient::clearData() {
this->dataPtr = nullptr;
this->dataSize = 0;
}
// bool EthernetMaketClient::send(uint8_t* data, uint16_t dataSize, IPAddress ip, uint16_t port) { bool EthernetMaketClient::isDataEmpty() const {
// this->dataPtr = data; return this->dataPtr == nullptr;
// this->dataSize = dataSize; }
// dstIP = ip;
// dstPort = port;
// switch (connectStatus) { bool EthernetMaketClient::canWriteIntoConnection(size_t size) const {
// case CONNECT_CONNECTED: return (connectStatus == CONNECT_CONNECTED) && (w5500.getTXFreeSize(_sock) >= size);
// if(dataWrite()){ }
// onSendSuccess();
// }
// // Если подключено, отправляем данные
// return true;
// case CONNECT_STOP_WAITING: void EthernetMaketClient::tick() {
// case CONNECT_STOP_START: forEach([](EthernetMaketClient &client) {
// return false; // Ожидаем завершения разрыва client.tick_();
});
// // case CONNECT_FAIL: }
// case CONNECT_IDLE:
// connectNonBlock(ip, port);
// connectStatus = CONNECT_START; // Начинаем новое подключение
// return false; // Подключение инициировано
// default:
// return false; // Пока ждём подключения
// }
// return false;
// }
void EthernetMaketClient::setOnSendSuccess(const std::function<void()>& callback) { void EthernetMaketClient::setOnSendSuccess(const std::function<void()>& callback) {
onSendSuccess = callback; onSendSuccess = callback;
@ -292,40 +204,10 @@ void EthernetMaketClient::stop(){
connectStatus = CONNECT_STOP_START; connectStatus = CONNECT_STOP_START;
} }
void EthernetMaketClient::startConnection(IPAddress ip, uint16_t port, bool nonBlock){ ConnectionStatusSimple EthernetMaketClient::getConnectionStatus() const {
dstIP = ip; return connectStatus;
dstPort = port; }
isNonBlocking = nonBlock;
connectStatus = CONNECT_START;
};
void EthernetMaketClient::activityUpdate() { void EthernetMaketClient::activityUpdate() {
lastActivityTime = millis(); lastActivityTime = millis();
} }
EthernetMaketClient& EthernetMaketClient::operator=(const EthernetMaketClient& other) {
if (this == &other) {
return *this; // Защита от самоприсваивания
}
// Копируем все данные
this->lastActivityTime = other.lastActivityTime;
this->startConnectionTime = other.startConnectionTime;
this->stopStartTime = other.stopStartTime;
this->isNonBlocking = other.isNonBlocking;
this->connectStatus = other.connectStatus;
// this->timeout = other.timeout;
// this->connectionTimeout = other.connectionTimeout;
// this->stopTimeout = other.stopTimeout;
this->dstIP = other.dstIP;
this->dstPort = other.dstPort;
this->dataPtr = other.dataPtr;
this->dataSize = other.dataSize;
this->onSendSuccess = other.onSendSuccess;
// Отключаем исполнение старого клиента
const_cast<EthernetMaketClient&>(other).connectStatus = CONNECT_IDLE;
return *this;
}

View File

@ -4,8 +4,9 @@
#include <list> #include <list>
#include <functional> #include <functional>
#include "../util/config.h" #include "../util/config.h"
#include "InstanceManager.h"
class EthernetMaketClient : public EthernetClient { class EthernetMaketClient : public EthernetClient, public InstanceManager<EthernetMaketClient> {
protected: protected:
uint32_t lastActivityTime = 0; // Время последней активности uint32_t lastActivityTime = 0; // Время последней активности
const uint16_t timeout = 300; // Таймаут неактивности в миллисекундах const uint16_t timeout = 300; // Таймаут неактивности в миллисекундах
@ -25,15 +26,32 @@ protected:
bool dataWrite(); bool dataWrite();
static std::list<EthernetMaketClient*>& getInstances(); // Ленивое создание списка // static std::list<EthernetMaketClient*>& getInstances(); // Ленивое создание списка
void tick_();
ConnectionStatusSimple connectStatus = CONNECT_IDLE; ConnectionStatusSimple connectStatus = CONNECT_IDLE;
public: public:
EthernetMaketClient(); EthernetMaketClient();
EthernetMaketClient(uint8_t sock); EthernetMaketClient(uint8_t sock);
~EthernetMaketClient(); EthernetMaketClient(EthernetClient &client);
// // Move constructor
// EthernetMaketClient(EthernetMaketClient&& other) noexcept
// : EthernetClient(std::move(other)), // Move base class
// lastActivityTime(other.lastActivityTime),
// startConnectionTime(other.startConnectionTime),
// stopStartTime(other.stopStartTime),
// onSendSuccess(std::move(other.onSendSuccess)),
// isNonBlocking(other.isNonBlocking),
// dataPtr(other.dataPtr),
// dataSize(other.dataSize),
// dstIP(std::move(other.dstIP)),
// dstPort(other.dstPort),
// connectStatus(other.connectStatus)
// {
// other.dataPtr = nullptr; // Reset the source object's pointer
// other.dataSize = 0;
// }
EthernetMaketClient& operator=(const EthernetMaketClient& other); EthernetMaketClient& operator=(const EthernetMaketClient& other);
@ -41,18 +59,22 @@ public:
void startConnection(IPAddress ip, uint16_t port, bool nonBlock = true); void startConnection(IPAddress ip, uint16_t port, bool nonBlock = true);
void dataWrite(uint8_t* data, uint16_t dataSize, bool override = false); void dataWrite(uint8_t* data, uint16_t dataSize, bool override = false);
// bool send(uint8_t* data, uint16_t dataSize, IPAddress ip, uint16_t port); void clearData();
bool isDataEmpty() const;
void disconnect(); void disconnect();
void close(); void close();
void stop(); void stop();
bool isConnected() const; bool canWriteIntoConnection(size_t = 0) const;
static void tick(); static void tick();
void tick_();
void setOnSendSuccess(const std::function<void()>& callback); void setOnSendSuccess(const std::function<void()>& callback);
void resetOnSendSuccess(); void resetOnSendSuccess();
ConnectionStatusSimple getConnectionStatus() const;
void activityUpdate(); void activityUpdate();
}; };

2
todo
View File

@ -1 +1,3 @@
[] Проверить и отладить UDP [] Проверить и отладить UDP
[] 8 статических клиентов для управления таймингами и сессией