-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdata_parser.cpp
289 lines (240 loc) · 9.69 KB
/
data_parser.cpp
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
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
#include "data_parser.h"
#include "xbusmessageid.h"
#include "xsdataidentifier.h"
DataParser::DataParser()
{
m_xspacket = new XsDataPacket();
//Debug print:
// String str = "m_packet_length = 0x" + String(m_packet_length, HEX);
// Serial.println(str);
}
DataParser::~DataParser() {
delete m_xspacket;
}
XsDataPacket* DataParser::getXsDatePacket() {
return m_xspacket;
}
bool DataParser::isLeapYear(uint16_t year) {
return (year % 4 == 0 && (year % 100 != 0 || year % 400 == 0));
}
// Compute the number of days since the Unix epoch (1st Jan 1970) to the given date.
uint32_t DataParser::dateToEpochDays(uint16_t year, uint8_t month, uint8_t day) {
static const uint8_t daysInMonth[] = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
uint32_t days = 0;
// Days from 1970 to the last year
for (uint16_t y = 1970; y < year; ++y) {
days += isLeapYear(y) ? 366 : 365;
}
// Days from January to the last complete month
for (uint8_t m = 1; m < month; ++m) {
days += daysInMonth[m - 1];
if (m == 2 && isLeapYear(year)) {
days += 1; // February in a leap year
}
}
// Days in the current month
days += day - 1;
return days;
}
// Convert date and time to epoch time in seconds
double DataParser::dateTimeToEpoch(uint16_t year, uint8_t month, uint8_t day,
uint8_t hour, uint8_t minute, uint8_t second, uint32_t nanosecond) {
uint32_t days = dateToEpochDays(year, month, day);
uint64_t totalSeconds = days * 86400 + hour * 3600 + minute * 60 + second;
double totalSecondsWithNanos = totalSeconds + (nanosecond * 1e-9);
return totalSecondsWithNanos;
}
void DataParser::parseDataPacket(const uint8_t* data) {
m_packet_length = data[3] + 5;
size_t input_size = m_packet_length;
if (input_size < 3) return; // Ensure there's enough data to start parsing
// packet = FA FF 36 16 10 60 04 09 3A 7C B7 20 30 0C BF 49 1A 52 BE C3 6A 10 C3 2E E2 EF 3E
size_t offset = 2; // Start after the presumed packet header FA FF
// Check for MTData2 preamble
if (data[offset] == XMID_MtData2)
{
offset++; // move past MTData2 identifier
if (offset >= input_size) return; // Safety check
uint8_t dataLengthMT = data[offset]; // data length for MTData2
offset++; // move to the first data type in MTData2
if (offset >= input_size) return; // Safety check
size_t endMTData2 = offset + dataLengthMT; // Calculate the end of the MTData2 block
if (endMTData2 > input_size) return; // Ensure we don't read past the buffer
while (offset < endMTData2)
{
if (offset + 2 > input_size) return; // Ensure space for dataType and dataLength
offset += 2; // Move past the 2-byte dataType
uint8_t dataLength = data[offset];
offset++; // Move past dataLength byte
if (offset + dataLength > input_size) return; // Ensure we don't read past the buffer
const uint8_t* dataPtr = &data[offset - 3]; // Adjust this to point to the start of dataType
parseMTData2(dataPtr, dataLength + 3);
offset += dataLength; // Move past the data to the next data type
}
}
}
void DataParser::dataswapendian(uint8_t *data, int len)
{
// Check if the length is either 2 or a multiple of 4
if (len % 4 != 0 && len != 2)
{
Serial.println("Length must be 2 or a multiple of 4");
return;
}
// Handle the case where the length is exactly 2
if (len == 2)
{
uint8_t temp = data[0];
data[0] = data[1];
data[1] = temp;
return;
}
// Loop through each 4-byte group for lengths that are multiples of 4
for (int i = 0; i < len; i += 4)
{
// Swap bytes within the group
uint8_t temp;
temp = data[i];
data[i] = data[i + 3];
data[i + 3] = temp;
temp = data[i + 1];
data[i + 1] = data[i + 2];
data[i + 2] = temp;
}
}
double DataParser::parseFP1632(const uint8_t *data)
{
int32_t fpfrac;
int16_t fpint;
// Note: memcpy is used to prevent issues caused by alignment requirements
memcpy(&fpfrac, data, sizeof(fpfrac));
memcpy(&fpint, data + 4, sizeof(fpint));
dataswapendian(reinterpret_cast<uint8_t*>(&fpfrac), sizeof(fpfrac));
dataswapendian(reinterpret_cast<uint8_t*>(&fpint), sizeof(fpint));
int64_t fp_i64 = (static_cast<int64_t>(fpint) << 32) | (static_cast<int64_t>(fpfrac) & 0xffffffff);
double rv_d = static_cast<double>(fp_i64) / 4294967296.0;
return rv_d;
}
void DataParser::parseMTData2(uint8_t *data, uint8_t datalength)
{
int offset = 0;
while (offset < datalength)
{
uint16_t dataId = (static_cast<uint16_t>(data[offset]) << 8) | data[offset + 1];
offset += 3; // i.e 10 60 04 0a 14 33 17 ==> 0a 14 33 17
switch (dataId)
{
case XDI_PacketCounter:
dataswapendian(&data[offset], 2);
m_xspacket->packetCounter = *reinterpret_cast<uint16_t *>(&data[offset]);
m_xspacket->packetCounterAvailable = true;
offset += 2;
break;
case XDI_SampleTimeFine: // Sample time fine
dataswapendian(&data[offset], 4);
m_xspacket->sampleTimeFine = *reinterpret_cast<uint32_t *>(&data[offset]);
m_xspacket->sampleTimeFineAvailable = true;
offset += 4;
break;
case XDI_UtcTime: // UTC Time stamp
{
dataswapendian(&data[offset], 4);
uint32_t nanosec = *reinterpret_cast<uint32_t *>(&data[offset]);
offset += 4;
dataswapendian(&data[offset], 2);
uint16_t year = *reinterpret_cast<uint16_t *>(&data[offset]);
offset += 2;
uint8_t month = *reinterpret_cast<uint8_t *>(&data[offset]);
offset++;
uint8_t day = *reinterpret_cast<uint8_t *>(&data[offset]);
offset++;
uint8_t hour = *reinterpret_cast<uint8_t *>(&data[offset]);
offset++;
uint8_t minute = *reinterpret_cast<uint8_t *>(&data[offset]);
offset++;
uint8_t sec = *reinterpret_cast<uint8_t *>(&data[offset]);
offset++;
// uint8_t flags = *reinterpret_cast<uint8_t *>(&data[offset]);
offset++;
uint16_t tm_year = year - 1900; // Years since 1900
uint8_t tm_mon = month - 1; // Months since January
// Convert to time_t (seconds since 1st Jan 1970)
m_xspacket->utcTime = dateTimeToEpoch(tm_year, tm_mon, day, hour, minute, sec, nanosec);
m_xspacket->utcTimeAvailable = true;
break;
}
case XDI_EulerAngles: // Euler Angles
dataswapendian(&data[offset], 12);
memcpy(m_xspacket->euler, &data[offset], 12);
m_xspacket->eulerAvailable = true;
offset += 12;
break;
case XDI_Quaternion: // Quaternion.
dataswapendian(&data[offset], 16);
memcpy(m_xspacket->quat, &data[offset], 16);
m_xspacket->quaternionAvailable = true;
m_xspacket->convertQuatToEuler();
offset += 16;
break;
case XDI_Acceleration: // Acceleration
dataswapendian(&data[offset], 12);
memcpy(m_xspacket->acc, &data[offset], 12);
m_xspacket->accAvailable = true;
offset += 12;
break;
case XDI_FreeAcceleration: // Free Acceleration
dataswapendian(&data[offset], 12);
memcpy(m_xspacket->freeacc, &data[offset], 12);
m_xspacket->freeaccAvailable = true;
offset += 12;
break;
case XDI_RateOfTurn: // Rate of Turn
dataswapendian(&data[offset], 12);
memcpy(m_xspacket->gyro, &data[offset], 12);
m_xspacket->gyroAvailable = true;
offset += 12;
break;
case 0x5042: // Latitude Longitude, FP16.32
m_xspacket->latlon[0] = parseFP1632(&data[offset]);
offset += 6;
m_xspacket->latlon[1] = parseFP1632(&data[offset]);
m_xspacket->latlonAvailable = true;
offset += 6;
break;
case 0x5022: // AltitudeEllipsoid, FP16.32
m_xspacket->altitude = parseFP1632(&data[offset]);
m_xspacket->altitudeAvailable = true;
offset += 6;
break;
case XDI_MagneticField: // Magnetic Field
dataswapendian(&data[offset], 12);
memcpy(m_xspacket->mag, &data[offset], 12);
m_xspacket->magAvailable = true;
offset += 12;
break;
case XDI_StatusWord: // StatusWord
dataswapendian(&data[offset], 4);
memcpy(&m_xspacket->statusWord, &data[offset], 4);
m_xspacket->statusWordAvailable = true;
offset += 4;
break;
case 0xD012: // Velocity, FP16.32
m_xspacket->vel[0] = parseFP1632(&data[offset]);
offset += 6;
m_xspacket->vel[1] = parseFP1632(&data[offset]);
offset += 6;
m_xspacket->vel[2] = parseFP1632(&data[offset]);
offset += 6;
m_xspacket->velocityAvailable = true;
break;
default:
String error_str = "Unrecognized data ID: 0x" + String(dataId, HEX) + " at offset " + String(offset) + ". Following bytes: ";
for (int i = 0; i < min(static_cast<int>(datalength - offset), 5); i++)
{ // print up to next 5 bytes
error_str += String(data[offset + i], HEX);
}
Serial.println(error_str);
break;
}
}
}