serial/include/serial.h

146 lines
3.8 KiB
C
Raw Normal View History

2024-09-23 19:50:16 +08:00
#ifndef SERIAL_H
#define SERIAL_H
#include <cstring>
#include <type_traits>
#include <iostream>
#include <string>
#include <chrono>
#include <optional>
#include <thread>
#include <functional>
#include "../third/serialib.h"
using namespace std::literals::chrono_literals;
template<typename T >
concept SupportString = requires{
std::is_same_v<T, const char*>;
std::is_same_v<T, std::string>;
};
enum class
[[maybe_unused]]
ErrorCode{
SUCCESS,
TIMEOUT,
SETTIMEOUTERROR,
WRITEINGERROR,
READINGERROR,
};
class Serial
{
private:
serialib ser;
const char* endChar = "\r\n";
std::function<void(const std::string&)> logCallBack;
public:
Serial() = default;
std::string GetTimeNow(){
auto now = std::chrono::system_clock::now();
auto now_c = std::chrono::system_clock::to_time_t(now);
2024-09-24 16:59:08 +08:00
char buffer[32];
ctime_s(buffer, 32, &now_c);
return std::string(buffer);
2024-09-23 19:50:16 +08:00
}
template<SupportString T>
bool OpenDevice(T PortName, unsigned int bauds)
{
int code;
if constexpr (std::is_same_v<T, std::string>){
code = ser.openDevice(PortName.c_str(), bauds);
}
else{
code = ser.openDevice(PortName, bauds);
}
if(code == 1){
return true;
}else{
return false;
}
}
void Log(const std::string& log){
if(logCallBack){
auto msg = GetTimeNow() + " "+ log + "\n";
logCallBack(msg);
}
}
void SetLogCallBack(std::function<void(const std::string&)> callBack){
logCallBack = callBack;
}
void CloseDevice(){
ser.closeDevice();
}
~Serial() = default;
template<SupportString T>
std::optional<std::string> GetAtResponse(T command, int timeout = 50){
ser.flushReceiver();
std::string reallyCommand;
std::string response;
if constexpr (std::is_same_v<T, std::string>){
reallyCommand = command + endChar;
}
else{
reallyCommand = std::string(command) + endChar;
}
ser.writeString(reallyCommand.c_str());
Log("Send: " + reallyCommand);
std::this_thread::sleep_for(10ms);
// char buffer[ser.available()] = {0};
2024-09-24 16:59:08 +08:00
auto availableSize = ser.available();
char* buffer = new char[availableSize+1];
2024-09-24 16:59:08 +08:00
std::memset(buffer, 0, availableSize);
auto size = ser.readBytes(buffer, availableSize, timeout);
2024-09-23 19:50:16 +08:00
if(size > 0){
2024-09-24 16:59:08 +08:00
buffer[size] = '\0';
2024-09-23 19:50:16 +08:00
response = std::string(buffer);
Log("Receive: " + response);
delete[] buffer;
return response;
}
delete[] buffer;
return std::nullopt;
}
template<SupportString T>
bool GetAtUntil(T command, T expect = "OK",int timeout = 50){
auto endTime = std::chrono::system_clock::now() + std::chrono::milliseconds(timeout);
ser.flushReceiver();
std::string reallyCommand;
if constexpr (std::is_same_v<T, std::string>){
reallyCommand = command + endChar;
}
else{
reallyCommand = std::string(command) + endChar;
}
ser.writeString(reallyCommand.c_str());
Log("Send : " + reallyCommand);
while(std::chrono::system_clock::now() < endTime){
std::this_thread::sleep_for(10ms);
2024-09-24 16:59:08 +08:00
auto availableSize = ser.available();
auto buffer = new char[availableSize+1];
2024-09-24 16:59:08 +08:00
auto size = ser.readBytes(buffer, availableSize, timeout);
2024-09-23 19:50:16 +08:00
auto str = std::string(buffer);
delete[] buffer;
if(size > 0)
Log("Receive: "+str);
if(str.find(expect) != std::string::npos){
return true;
}
}
return false;
}
};
#endif // SERIAL_H