mirror of
https://github.com/Show-maket/EthernetMaket.git
synced 2025-05-04 15:20:18 +00:00
tcp client
This commit is contained in:
parent
4d3d44d03a
commit
b6e2fb1356
@ -32,6 +32,10 @@ EthernetMaketClient client;
|
|||||||
|
|
||||||
uint32_t packetCounter = 0;
|
uint32_t packetCounter = 0;
|
||||||
|
|
||||||
|
void onSend(){
|
||||||
|
Serial.println("Sending Successfully");
|
||||||
|
}
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
@ -46,11 +50,16 @@ void setup()
|
|||||||
// Ethernet.setRtCount(3);
|
// Ethernet.setRtCount(3);
|
||||||
|
|
||||||
// client.setNoDelayedACK(true); // Не ждать ответа от сервера
|
// client.setNoDelayedACK(true); // Не ждать ответа от сервера
|
||||||
|
|
||||||
|
client.setOnSendSuccess(onSend);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void loop()
|
void loop()
|
||||||
{
|
{
|
||||||
|
EthernetMaketClient::tick();
|
||||||
uint32_t currentMillis = millis();
|
uint32_t currentMillis = millis();
|
||||||
if (currentMillis - ttt > 75)
|
if (currentMillis - ttt > 75)
|
||||||
{
|
{
|
||||||
@ -61,12 +70,35 @@ void loop()
|
|||||||
if (currentMillis - previousMillis >= interval)
|
if (currentMillis - previousMillis >= interval)
|
||||||
{
|
{
|
||||||
previousMillis = currentMillis;
|
previousMillis = currentMillis;
|
||||||
// client.stop(); // Завершаем предыдущее соединение
|
|
||||||
connectStatus = CONNECT_START; // функция коннекта
|
|
||||||
// client.connect(serverIP, SERVER_PORT);
|
|
||||||
client.setData(buf, sizeof(buf));
|
|
||||||
client()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// client.stop(); // Завершаем предыдущее соединение
|
||||||
|
// connectStatus = CONNECT_START; // функция коннекта
|
||||||
|
// client.connect(serverIP, SERVER_PORT);
|
||||||
|
// client.setData(buf, sizeof(buf));
|
||||||
|
// client()
|
||||||
|
|
||||||
|
if (client.connectNonBlock(serverIP, SERVER_PORT) == CONNECT_CONNECTED)
|
||||||
|
{
|
||||||
|
for (size_t i = 0; i < 3; i++)
|
||||||
|
{
|
||||||
|
client.dataWrite((uint8_t *)&packetCounter, sizeof(packetCounter));
|
||||||
|
|
||||||
|
// Print the packet being sent
|
||||||
|
Serial.print("Sending packet: ");
|
||||||
|
for (size_t j = 0; j < sizeof(packetCounter); j++)
|
||||||
|
{
|
||||||
|
Serial.print(((uint8_t *)&packetCounter)[j], HEX);
|
||||||
|
Serial.print(" ");
|
||||||
|
}
|
||||||
|
Serial.println();
|
||||||
|
packetCounter++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial.println("Failed to connect to server");
|
||||||
|
client.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
@ -2,8 +2,32 @@
|
|||||||
#include "utility/w5500.h"
|
#include "utility/w5500.h"
|
||||||
#include "utility/socket.h"
|
#include "utility/socket.h"
|
||||||
|
|
||||||
|
std::vector<EthernetMaketClient*> EthernetMaketClient::instances;
|
||||||
|
|
||||||
|
|
||||||
|
EthernetMaketClient::EthernetMaketClient() {
|
||||||
|
_sock = MAX_SOCK_NUM;
|
||||||
|
instances.push_back(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
EthernetMaketClient::EthernetMaketClient(uint8_t sock) {
|
||||||
|
_sock = sock;
|
||||||
|
instances.push_back(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
EthernetMaketClient::~EthernetMaketClient() {
|
||||||
|
instances.erase(std::remove(instances.begin(), instances.end(), this), instances.end());
|
||||||
|
}
|
||||||
|
|
||||||
|
void EthernetMaketClient::tick() {
|
||||||
|
for (EthernetMaketClient* client : instances) {
|
||||||
|
client->tick_();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ConnectionStatusSimple EthernetMaketClient::connectNonBlock(IPAddress ip, uint16_t port) {
|
ConnectionStatusSimple EthernetMaketClient::connectNonBlock(IPAddress ip, uint16_t port) {
|
||||||
|
dstIP = ip;
|
||||||
|
dstPort = port;
|
||||||
if (_sock != MAX_SOCK_NUM) {
|
if (_sock != MAX_SOCK_NUM) {
|
||||||
// Serial.println(F("connectNonBlock"));
|
// Serial.println(F("connectNonBlock"));
|
||||||
uint8_t status_ = status();
|
uint8_t status_ = status();
|
||||||
@ -47,6 +71,10 @@ ConnectionStatusSimple EthernetMaketClient::connectNonBlock(IPAddress ip, uint16
|
|||||||
return CONNECT_CONNECTING;
|
return CONNECT_CONNECTING;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool EthernetMaketClient::isConnected() const {
|
||||||
|
return connectStatus == CONNECT_CONNECTED;
|
||||||
|
}
|
||||||
|
|
||||||
void EthernetMaketClient::disconnect(){
|
void EthernetMaketClient::disconnect(){
|
||||||
if (_sock == MAX_SOCK_NUM)
|
if (_sock == MAX_SOCK_NUM)
|
||||||
return;
|
return;
|
||||||
@ -60,7 +88,7 @@ void EthernetMaketClient::close(){
|
|||||||
_sock = MAX_SOCK_NUM;
|
_sock = MAX_SOCK_NUM;
|
||||||
}
|
}
|
||||||
|
|
||||||
void EthernetMaketClient::tick(){
|
void EthernetMaketClient::tick_(){
|
||||||
// Подключаемся к серверу
|
// Подключаемся к серверу
|
||||||
switch (connectStatus)
|
switch (connectStatus)
|
||||||
{
|
{
|
||||||
@ -88,26 +116,15 @@ void EthernetMaketClient::tick(){
|
|||||||
lastActivityTime = millis();
|
lastActivityTime = millis();
|
||||||
connectStatus = CONNECT_CONNECTED;
|
connectStatus = CONNECT_CONNECTED;
|
||||||
|
|
||||||
//! Передача пакетов
|
lastActivityTime = millis(); // Обновляем время последней активности
|
||||||
for (size_t i = 0; i < 3; i++)
|
|
||||||
{
|
|
||||||
client.write((uint8_t *)&packetCounter, sizeof(packetCounter));
|
|
||||||
|
|
||||||
// Печать отправленного пакета
|
|
||||||
Serial.print("Sending packet: ");
|
|
||||||
for (size_t j = 0; j < sizeof(packetCounter); j++)
|
|
||||||
{
|
|
||||||
Serial.print(((uint8_t *)&packetCounter)[j], HEX);
|
|
||||||
Serial.print(" ");
|
|
||||||
}
|
|
||||||
Serial.println();
|
|
||||||
packetCounter++;
|
|
||||||
lastActivityTime = millis(); // Обновляем время последней активности для каждого пакета
|
|
||||||
}
|
|
||||||
//! Передача завершена
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case CONNECT_CONNECTED:
|
case CONNECT_CONNECTED:
|
||||||
|
|
||||||
|
//! Передача пакетов
|
||||||
|
dataWrite();
|
||||||
|
//! Передача завершена
|
||||||
|
|
||||||
if (!connected())
|
if (!connected())
|
||||||
{
|
{
|
||||||
Serial.println("Client disconnected");
|
Serial.println("Client disconnected");
|
||||||
@ -123,40 +140,58 @@ void EthernetMaketClient::tick(){
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case CONNECT_STOP_START:
|
case CONNECT_STOP_START:
|
||||||
|
dstIP = IPAddress(0,0,0,0);
|
||||||
|
dstPort = 0;
|
||||||
disconnect();
|
disconnect();
|
||||||
stopStartTime = millis();
|
stopStartTime = millis();
|
||||||
connectStatus = CONNECT_STOP_WAITING;
|
connectStatus = CONNECT_STOP_WAITING;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CONNECT_STOP_WAITING:
|
case CONNECT_STOP_WAITING:
|
||||||
if (millis() - stopStartTime >= stopTimeout)
|
if (millis() - stopStartTime >= stopTimeout) {
|
||||||
{
|
close(); // Окончательное закрытие сокета после таймаута
|
||||||
close(); // Закрываем сокет после таймаута
|
connectStatus = (dataSize > 0) ? CONNECT_START : CONNECT_IDLE;
|
||||||
connectStatus = CONNECT_IDLE;
|
} else {
|
||||||
|
// Проверяем состояние сокета для завершения разрыва
|
||||||
|
uint8_t status_ = status();
|
||||||
|
if (status_ == SnSR::CLOSED || status_ == SnSR::CLOSE_WAIT) {
|
||||||
|
close(); // Закрываем при достижении состояния
|
||||||
|
connectStatus = (dataSize > 0) ? CONNECT_START : CONNECT_IDLE;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CONNECT_IDLE:
|
||||||
|
if(dataSize){
|
||||||
|
connectStatus = CONNECT_START;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
//********************************************************
|
||||||
switch (connectStatus)
|
switch (connectStatus)
|
||||||
{
|
{
|
||||||
case CONNECT_START:
|
case CONNECT_START:
|
||||||
case CONNECT_CONNECTING:
|
case CONNECT_CONNECTING:
|
||||||
|
if(dstIP == IPAddress(0,0,0,0) || dstPort == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
if (isNonBlocking)
|
if (isNonBlocking)
|
||||||
{
|
{
|
||||||
connectStatus = connectNonBlock(serverIP, SERVER_PORT);
|
connectStatus = connectNonBlock(dstIP, dstPort);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
connectStatus = (ConnectionStatusSimple)connect(serverIP, SERVER_PORT);
|
connectStatus = (ConnectionStatusSimple)connect(dstIP, dstPort);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
//********************************************************
|
||||||
|
|
||||||
//? Проверка наличия доступных данных от сервера
|
//? Проверка наличия доступных данных от сервера
|
||||||
if (available() > 0)
|
if (available() > 0)
|
||||||
@ -197,3 +232,64 @@ void EthernetMaketClient::tick(){
|
|||||||
}
|
}
|
||||||
//? -------------------------------------------------
|
//? -------------------------------------------------
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool EthernetMaketClient::dataWrite(){
|
||||||
|
bool ret = false;
|
||||||
|
if(dataSize) {
|
||||||
|
lastActivityTime = millis();
|
||||||
|
if(write((uint8_t *)&dataPtr, dataSize)){
|
||||||
|
dataPtr = nullptr;
|
||||||
|
dataSize = 0; // Очищаем размер данных
|
||||||
|
ret = true;
|
||||||
|
}
|
||||||
|
lastActivityTime = millis();
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool EthernetMaketClient::dataWrite(uint8_t* data, uint16_t dataSize, bool override) {
|
||||||
|
if (dataSize == 0 || override || (dataPtr == nullptr && this->dataSize == 0)) {
|
||||||
|
this->dataPtr = data;
|
||||||
|
this->dataSize = dataSize;
|
||||||
|
return dataWrite();
|
||||||
|
} else {
|
||||||
|
Serial.println("Data buffer is not empty. Write operation is ignored.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool EthernetMaketClient::send(uint8_t* data, uint16_t dataSize, IPAddress ip, uint16_t port) {
|
||||||
|
this->dataPtr = data;
|
||||||
|
this->dataSize = dataSize;
|
||||||
|
|
||||||
|
switch (connectStatus) {
|
||||||
|
case CONNECT_CONNECTED:
|
||||||
|
if(dataWrite()){
|
||||||
|
onSendSuccess();
|
||||||
|
}
|
||||||
|
// Если подключено, отправляем данные
|
||||||
|
return true;
|
||||||
|
|
||||||
|
case CONNECT_STOP_WAITING:
|
||||||
|
case CONNECT_STOP_START:
|
||||||
|
return false; // Ожидаем завершения разрыва
|
||||||
|
|
||||||
|
case CONNECT_FAIL:
|
||||||
|
case CONNECT_IDLE:
|
||||||
|
connectNonBlock(ip, port);
|
||||||
|
connectStatus = CONNECT_START; // Начинаем новое подключение
|
||||||
|
return false; // Подключение инициировано
|
||||||
|
|
||||||
|
default:
|
||||||
|
return false; // Пока ждём подключения
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void EthernetMaketClient::setOnSendSuccess(const std::function<void()>& callback) {
|
||||||
|
onSendSuccess = callback;
|
||||||
|
}
|
||||||
|
|
||||||
|
void EthernetMaketClient::resetOnSendSuccess() {
|
||||||
|
onSendSuccess = []() {};
|
||||||
|
}
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
#ifndef __MAKET_TCP_H__
|
#ifndef __MAKET_TCP_H__
|
||||||
#define __MAKET_TCP_H__
|
#define __MAKET_TCP_H__
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include <functional>
|
||||||
#include "../util/config.h"
|
#include "../util/config.h"
|
||||||
|
|
||||||
class EthernetMaketClient : public EthernetClient
|
class EthernetMaketClient : public EthernetClient {
|
||||||
{
|
|
||||||
protected:
|
protected:
|
||||||
uint32_t lastActivityTime = 0; // Время последней активности
|
uint32_t lastActivityTime = 0; // Время последней активности
|
||||||
const uint16_t timeout = 300; // Таймаут неактивности в миллисекундах
|
const uint16_t timeout = 300; // Таймаут неактивности в миллисекундах
|
||||||
@ -13,16 +14,35 @@ protected:
|
|||||||
uint32_t stopStartTime = 0; // Время начала состояния CONNECT_IDLE
|
uint32_t stopStartTime = 0; // Время начала состояния CONNECT_IDLE
|
||||||
const uint32_t stopTimeout = 300; // Таймаут для закрытия сокета в состоянии CONNECT_IDLE
|
const uint32_t stopTimeout = 300; // Таймаут для закрытия сокета в состоянии CONNECT_IDLE
|
||||||
|
|
||||||
|
std::function<void()> onSendSuccess = []() {};
|
||||||
|
|
||||||
bool isNonBlocking = true; // Флаг для определения режима подключения
|
bool isNonBlocking = true; // Флаг для определения режима подключения
|
||||||
ConnectionStatusSimple connectStatus = CONNECT_IDLE;
|
ConnectionStatusSimple connectStatus = CONNECT_IDLE;
|
||||||
|
|
||||||
public:
|
uint8_t* dataPtr;
|
||||||
using EthernetClient::EthernetClient;
|
uint16_t dataSize = 0;
|
||||||
|
IPAddress dstIP;
|
||||||
|
uint16_t dstPort;
|
||||||
|
bool dataWrite();
|
||||||
|
|
||||||
|
static std::vector<EthernetMaketClient*> instances;
|
||||||
|
void tick_();
|
||||||
|
public:
|
||||||
|
EthernetMaketClient();
|
||||||
|
EthernetMaketClient(uint8_t sock);
|
||||||
|
~EthernetMaketClient();
|
||||||
ConnectionStatusSimple connectNonBlock(IPAddress ip, uint16_t port);
|
ConnectionStatusSimple connectNonBlock(IPAddress ip, uint16_t port);
|
||||||
// void stop() override;
|
|
||||||
void disconnect();
|
void disconnect();
|
||||||
void close();
|
void close();
|
||||||
void tick();
|
|
||||||
|
static void tick();
|
||||||
|
bool send(uint8_t* data, uint16_t dataSize, IPAddress ip, uint16_t port);
|
||||||
|
|
||||||
|
bool isConnected() const;
|
||||||
|
|
||||||
|
void setOnSendSuccess(const std::function<void()>& callback);
|
||||||
|
void resetOnSendSuccess();
|
||||||
|
bool dataWrite(uint8_t* data, uint16_t dataSize, bool override = false);
|
||||||
};
|
};
|
||||||
#endif // __MAKET_TCP_H__
|
|
||||||
|
#endif // __MAKET_TCP_H__
|
||||||
|
Loading…
x
Reference in New Issue
Block a user