-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathtelegram.hpp
140 lines (131 loc) · 4.07 KB
/
telegram.hpp
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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
#ifndef _TELEGRAM_HPP_
#define _TELEGRAM_HPP_
#include <boost/asio.hpp>
#include <boost/locale.hpp>
#include <chrono>
#include <condition_variable>
#include <list>
#include <mutex>
#include <rapidjson/document.h>
#include <thread>
#include <vector>
namespace sepreference {
enum class TelegramPartType {
digital,
indicator,
uint4,
uint8,
uint16,
uint32,
int8,
int16,
int32,
string,
unknown
};
struct TelegramPart {
int startbit;
int endbit;
TelegramPartType type;
int factor;
size_t len;
int def;
uint32_t hysteresis;
uint32_t sent_val;
uint32_t new_val;
std::string name;
std::mutex mutex;
constexpr int size() {
switch (type) {
case TelegramPartType::digital:
return 1;
case TelegramPartType::indicator:
case TelegramPartType::uint4:
return 4;
case TelegramPartType::uint8:
case TelegramPartType::int8:
return 8;
case TelegramPartType::uint16:
case TelegramPartType::int16:
return 16;
case TelegramPartType::uint32:
case TelegramPartType::int32:
return 32;
case TelegramPartType::string:
return len * 8;
case TelegramPartType::unknown:
default:
return -1;
}
};
};
class Telegram {
private:
std::string ip;
int port;
std::list<std::unique_ptr<TelegramPart>> format;
int size;
int cycle;
std::mutex buf_mutex;
std::mutex comm_mutex;
uint8_t *buf;
std::atomic<bool> sending;
std::atomic<bool> thread_started;
std::condition_variable comm_condition;
std::unique_ptr<std::thread> sendthread;
int conv2be(int val, int size);
void valcopy(const uint32_t val, int startbit, int endbit);
void valcopy(const uint8_t *val, int startbit, int endbit);
public:
static const size_t maxBytesize = 65507;
Telegram(const rapidjson::Value &format);
void setSending(bool sending);
template <typename T> void updateValue(const std::string &name, T &val) {
for (auto &tp : format) {
std::lock_guard<std::mutex> l(tp->mutex);
if (tp->name == name) {
if (val < 0 && (tp->type == TelegramPartType::uint8 ||
tp->type == TelegramPartType::uint16 ||
tp->type == TelegramPartType::uint32))
val = 0 - val;
const T scaled_val = val * tp->factor;
tp->new_val = static_cast<uint32_t>(scaled_val);
uint32_t delta = std::max(tp->sent_val, tp->new_val) -
std::min(tp->sent_val, tp->new_val);
bool exceeded_hysteresis =
delta > tp->hysteresis || (scaled_val == 0 && delta > 0);
auto x = conv2be(tp->new_val, tp->size());
std::unique_lock<std::mutex> buf_lock(buf_mutex);
valcopy(x, tp->startbit, tp->endbit);
buf_lock.unlock();
if (exceeded_hysteresis) {
comm_condition.notify_one();
tp->sent_val = tp->new_val;
}
}
}
};
template <typename S>
void updateStringValue(const std::string &name, std::basic_string<S> &val) {
std::string utf8val = boost::locale::conv::utf_to_utf<char>(val);
for (auto &tp : format) {
std::lock_guard<std::mutex> l(tp->mutex);
if (tp->name == name) {
size_t actually_copied_bytes =
std::min(tp->len - 1, strlen(utf8val.c_str()));
size_t actually_copied_bits = actually_copied_bytes * 8;
int endbit = tp->startbit + actually_copied_bits - 1;
valcopy(reinterpret_cast<const uint8_t *>(utf8val.c_str()),
tp->startbit, endbit);
comm_condition.notify_one();
}
}
}
~Telegram() {
setSending(false);
delete[] buf;
buf = 0;
}
};
} // namespace sepreference
#endif