添加相关函数
This commit is contained in:
parent
6ec5ffd22f
commit
6605f0604a
120
include/serial.h
120
include/serial.h
@ -22,32 +22,32 @@ namespace ranges = std::ranges;
|
|||||||
namespace views = std::views;
|
namespace views = std::views;
|
||||||
|
|
||||||
namespace serial {
|
namespace serial {
|
||||||
template <typename... Types> struct _StrongType {};
|
template <typename... Types> struct _StrongType {};
|
||||||
|
|
||||||
template <typename T> struct _StrongType<T> {
|
template <typename T> struct _StrongType<T> {
|
||||||
using Type = T;
|
using Type = T;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Ta, typename Tb> struct _StrongType<Ta, Tb> {
|
template <typename Ta, typename Tb> struct _StrongType<Ta, Tb> {
|
||||||
using Type = decltype(true ? std::declval<Ta>() : std::declval<Tb>());
|
using Type = decltype(true ? std::declval<Ta>() : std::declval<Tb>());
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename... Types> struct _StrongType<T, Types...> {
|
template <typename T, typename... Types> struct _StrongType<T, Types...> {
|
||||||
using Type =
|
using Type =
|
||||||
typename _StrongType<T, typename _StrongType<Types...>::Type>::Type;
|
typename _StrongType<T, typename _StrongType<Types...>::Type>::Type;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename... Types>
|
template <typename... Types>
|
||||||
using _strongType_t = typename _StrongType<Types...>::Type;
|
using _strongType_t = typename _StrongType<Types...>::Type;
|
||||||
|
|
||||||
template <typename... Args> struct _IsCastable {};
|
template <typename... Args> struct _IsCastable {};
|
||||||
|
|
||||||
template <typename T> struct _IsCastable<T> {
|
template <typename T> struct _IsCastable<T> {
|
||||||
static constexpr bool value = true;
|
static constexpr bool value = true;
|
||||||
using Type = T;
|
using Type = T;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename U> struct _IsCastable<T, U> {
|
template <typename T, typename U> struct _IsCastable<T, U> {
|
||||||
using __TRUE = char;
|
using __TRUE = char;
|
||||||
using __FALSE = struct {
|
using __FALSE = struct {
|
||||||
char _[2];
|
char _[2];
|
||||||
@ -57,30 +57,32 @@ template <typename T, typename U> struct _IsCastable<T, U> {
|
|||||||
static constexpr bool value =
|
static constexpr bool value =
|
||||||
sizeof(__TEST(std::declval<T>())) == sizeof(__TRUE);
|
sizeof(__TEST(std::declval<T>())) == sizeof(__TRUE);
|
||||||
using Type = std::conditional_t<value, U, void>;
|
using Type = std::conditional_t<value, U, void>;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename... Args> struct _IsCastable<T, Args...> {
|
template <typename T, typename... Args> struct _IsCastable<T, Args...> {
|
||||||
static constexpr bool value =
|
static constexpr bool value =
|
||||||
_IsCastable<T, typename _IsCastable<Args...>::Type>::value;
|
_IsCastable<T, typename _IsCastable<Args...>::Type>::value;
|
||||||
using Type =
|
using Type =
|
||||||
std::conditional_t<value, typename _IsCastable<Args...>::Type, void>;
|
std::conditional_t<value, typename _IsCastable<Args...>::Type, void>;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
concept _SupportString = requires {
|
concept _SupportString = requires {
|
||||||
_IsCastable<T, const char *, char *, std::string, std::string_view>::value;
|
_IsCastable<T, const char*, char*, std::string, std::string_view>::value;
|
||||||
};
|
};
|
||||||
template <_SupportString T> [[maybe_unused]] std::string _to_string(T &&str) {
|
template <_SupportString T> [[maybe_unused]] std::string _to_string(T&& str) {
|
||||||
if constexpr (std::is_same_v<std::decay_t<T>, std::string>) {
|
if constexpr (std::is_same_v<std::decay_t<T>, std::string>) {
|
||||||
return str;
|
return str;
|
||||||
} else if constexpr (std::is_same_v<std::decay_t<T>, std::string_view>) {
|
}
|
||||||
|
else if constexpr (std::is_same_v<std::decay_t<T>, std::string_view>) {
|
||||||
return std::move(std::string(str.data()));
|
return std::move(std::string(str.data()));
|
||||||
} else if constexpr (std::is_same_v<std::decay_t<T>, const char *>) {
|
}
|
||||||
|
else if constexpr (std::is_same_v<std::decay_t<T>, const char*>) {
|
||||||
return std::string(str);
|
return std::string(str);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::vector<std::string> GetUsbPorts() {
|
static std::vector<std::string> GetUsbPorts() {
|
||||||
#if defined(_WIN32) || defined(_WIN64)
|
#if defined(_WIN32) || defined(_WIN64)
|
||||||
std::vector<std::string> portArray;
|
std::vector<std::string> portArray;
|
||||||
std::string comname;
|
std::string comname;
|
||||||
@ -102,50 +104,50 @@ static std::vector<std::string> GetUsbPorts() {
|
|||||||
#elif defined(__linux__)
|
#elif defined(__linux__)
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
enum class [[maybe_unused]] SerialErrorCode {
|
enum class [[maybe_unused]] SerialErrorCode {
|
||||||
SUCCESS,
|
SUCCESS,
|
||||||
TIMEOUT,
|
TIMEOUT,
|
||||||
SETTIMEOUTERROR,
|
SETTIMEOUTERROR,
|
||||||
WRITEINGERROR,
|
WRITEINGERROR,
|
||||||
READINGERROR,
|
READINGERROR,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum SerialDataBits {
|
enum SerialDataBits {
|
||||||
DATABIT_5 = 0,
|
DATABIT_5 = 0,
|
||||||
DATABIT_6,
|
DATABIT_6,
|
||||||
DATABIT_7,
|
DATABIT_7,
|
||||||
DATABIT_8,
|
DATABIT_8,
|
||||||
DATABIT_16,
|
DATABIT_16,
|
||||||
};
|
};
|
||||||
enum SerialStopBits {
|
enum SerialStopBits {
|
||||||
STOPBIT_1 = 0,
|
STOPBIT_1 = 0,
|
||||||
STOPBIT_1_5,
|
STOPBIT_1_5,
|
||||||
STOPBIT_2,
|
STOPBIT_2,
|
||||||
};
|
};
|
||||||
enum SerialParity {
|
enum SerialParity {
|
||||||
SERIAL_PARITY_NONE = 0,
|
SERIAL_PARITY_NONE = 0,
|
||||||
SERIAL_PARITY_EVEN,
|
SERIAL_PARITY_EVEN,
|
||||||
SERIAL_PARITY_ODD,
|
SERIAL_PARITY_ODD,
|
||||||
SERIAL_PARITY_MARK,
|
SERIAL_PARITY_MARK,
|
||||||
SERIAL_PARITY_SPACE,
|
SERIAL_PARITY_SPACE,
|
||||||
};
|
};
|
||||||
|
|
||||||
class Serial {
|
class Serial {
|
||||||
private:
|
private:
|
||||||
serialib ser;
|
serialib ser;
|
||||||
const char *endChar = "\r\n";
|
const char* endChar = "\r\n";
|
||||||
std::function<void(const std::string &)> logCallBack;
|
std::function<void(const std::string&)> logCallBack;
|
||||||
bool removeEcho = true; // 剔除回显
|
bool removeEcho = true; // 剔除回显
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Serial() = default;
|
Serial() = default;
|
||||||
~Serial() { CloseDevice(); }
|
~Serial() { CloseDevice(); }
|
||||||
Serial(const Serial &other) = delete;
|
Serial(const Serial& other) = delete;
|
||||||
Serial(Serial &&other) = delete;
|
Serial(Serial&& other) = delete;
|
||||||
Serial &operator=(const Serial &other) = delete;
|
Serial& operator=(const Serial& other) = delete;
|
||||||
Serial &operator=(Serial &&other) = delete;
|
Serial& operator=(Serial&& other) = delete;
|
||||||
|
|
||||||
auto SetRemoveEcho(bool remove) { removeEcho = remove; }
|
auto SetRemoveEcho(bool remove) { removeEcho = remove; }
|
||||||
auto SetDtr(bool flag) { return flag ? ser.setDTR() : ser.clearDTR(); }
|
auto SetDtr(bool flag) { return flag ? ser.setDTR() : ser.clearDTR(); }
|
||||||
@ -183,12 +185,13 @@ class Serial {
|
|||||||
static_cast<::SerialStopBits>(stopBits));
|
static_cast<::SerialStopBits>(stopBits));
|
||||||
if (code == 1) {
|
if (code == 1) {
|
||||||
return true;
|
return true;
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Log(const std::string &log) const {
|
void Log(const std::string& log) const {
|
||||||
if (logCallBack) {
|
if (logCallBack) {
|
||||||
auto msg = GetTimeNow() + " " + log;
|
auto msg = GetTimeNow() + " " + log;
|
||||||
logCallBack(msg);
|
logCallBack(msg);
|
||||||
@ -196,7 +199,7 @@ class Serial {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
SetLogCallBack(const std::function<void(const std::string &)> &callBack) {
|
SetLogCallBack(const std::function<void(const std::string&)>& callBack) {
|
||||||
logCallBack = callBack;
|
logCallBack = callBack;
|
||||||
}
|
}
|
||||||
void CloseDevice() {
|
void CloseDevice() {
|
||||||
@ -283,6 +286,33 @@ class Serial {
|
|||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
template <int timeout = 200, _SupportString T, _SupportString... Args>
|
||||||
|
std::string GetAtUntilAndReturn(T command, Args... expect)
|
||||||
|
{
|
||||||
|
auto endTime = std::chrono::system_clock::now() +
|
||||||
|
std::chrono::milliseconds(timeout);
|
||||||
|
ser.flushReceiver();
|
||||||
|
std::string resp = "";
|
||||||
|
std::string 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);
|
||||||
|
auto availableSize = ser.available();
|
||||||
|
auto buffer = std::make_unique<char[]>(availableSize + 1);
|
||||||
|
auto size = ser.readBytes(buffer.get(), availableSize, timeout);
|
||||||
|
buffer[size] = '\0';
|
||||||
|
auto str = std::string(buffer.get());
|
||||||
|
resp += str;
|
||||||
|
if (size > 0)
|
||||||
|
Log("Receive: " + str);
|
||||||
|
if (((str.find(expect) != std::string::npos) && ...)) {
|
||||||
|
return resp;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return resp;
|
||||||
|
}
|
||||||
|
};
|
||||||
} // namespace serial
|
} // namespace serial
|
||||||
#endif // SERIAL_H
|
#endif // SERIAL_H
|
||||||
|
Loading…
Reference in New Issue
Block a user