-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathuwbmodule.cpp
More file actions
435 lines (375 loc) · 13.3 KB
/
uwbmodule.cpp
File metadata and controls
435 lines (375 loc) · 13.3 KB
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
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
#include "uwbmodule.h"
AnchorData::AnchorData(const AnchorData &anchor){
id = anchor.id;
timestamp = anchor.timestamp;
x = anchor.x;
y = anchor.y;
z = anchor.z;
distance = anchor.distance;
rx_power = anchor.rx_power;
fp_power = anchor.fp_power;
fp_snr = anchor.fp_snr;
}
void UwbModule::setComport(QString port)
{
if(this->serial.isOpen()){
this->serial.close();
}
this->serial.setPortName(port);
this->serial.setBaudRate(UWB_BAUD_RATE, QSerialPort::AllDirections);
this->serial.setDataBits(QSerialPort::Data8);
this->serial.setFlowControl(QSerialPort::NoFlowControl);
this->serial.setStopBits(QSerialPort::OneStop);
this->serial.setParity(QSerialPort::NoParity);
this->openSuccess = this->serial.open(QIODevice::ReadWrite);
}
/**
* @brief UwbModule::readAllFields reads all fields from the UWB module and updates public members
*/
int UwbModule::readAllFields(){
QByteArray fields = QByteArray();
fields.append(char(SELF_ID));
fields.append(char(MODE));
fields.append(char(CHANNEL));
fields.append(char(NUMBER_OF_ANCHORS));
fields.append(char(SAMPLES_PER_RANGE));
fields.append(char(FIELD_X));
fields.append(char(FIELD_Y));
fields.append(char(FIELD_Z));
QByteArray rpy = this->readFields(fields); // formulate packet, send it, and read the response
if(rpy == nullptr)
return -1; // something went wrong
this->updateObject(rpy); // update our field data based on the reply
return 0;
}
/**
* @brief UwbModule::writeAllFields update the connected module from instance data
*/
void UwbModule::writeAllFields(){
QByteArray data = QByteArray();
int body_size = 0;
body_size += this->addFieldToBuffer(data, SELF_ID, this->id);
body_size += this->addFieldToBuffer(data, MODE, this->mode);
body_size += this->addFieldToBuffer(data, CHANNEL, this->channel);
body_size += this->addFieldToBuffer(data, NUMBER_OF_ANCHORS, this->numberOfAnchors);
body_size += this->addFieldToBuffer(data, SAMPLES_PER_RANGE, this->samplesPerRange);
body_size += this->addFieldToBuffer(data, FIELD_X, this->x);
body_size += this->addFieldToBuffer(data, FIELD_Y, this->y);
body_size += this->addFieldToBuffer(data, FIELD_Z, this->z);
//data[1] = char(body_size);
//data.append(char(STOP_BYTE));
int write_status = this->writeFields(data);
if(write_status != 0){
qDebug() << "Write error";
}else{
qDebug() << "Write success!";
}
QByteArray rpy;
while(this->serial.waitForReadyRead(100))
rpy.append(this->serial.readAll());
}
/**
* @brief UwbModule::readAnchors read and parse the list of anchor data from the UWB module
*/
void UwbModule::readAnchors(){
if(!this->openSuccess || !this->serial.isOpen())
return;
QByteArray cmd;
cmd.append(char(READ_ANCHORS));
cmd.append('\0');
cmd.append(char(STOP_BYTE));
this->serial.write(cmd);
QByteArray rpy;
while(this->serial.waitForReadyRead(250))
rpy.append(this->serial.readAll());
if(rpy[0] != char(READ_ANCHORS) || rpy[rpy.length()-1] != char(STOP_BYTE))
return;
int body_size = rpy[1];
const char* rpy_data = rpy.data();
this->anchors.clear();
for(int i = HDDR_SIZE; i < HDDR_SIZE + body_size; ){
AnchorData anchor;
anchor.id = uint32_t(rpy[i++]) & 0xFF;
anchor.timestamp = uint32_t( (rpy[i]) | (rpy[i+1] << 8) | (rpy[i+2] << 16) | (rpy[i+3] << 24) );
i += 4;
memcpy(&anchor.x, rpy_data + i, 4);
i += 4;
memcpy(&anchor.y, rpy_data + i, 4);
i += 4;
memcpy(&anchor.z, rpy_data + i, 4);
i += 4;
memcpy(&anchor.distance, rpy_data + i, 4);
i += 4;
memcpy(&anchor.rx_power, rpy_data + i, 4);
i += 4;
memcpy(&anchor.fp_power, rpy_data + i, 4);
i += 4;
memcpy(&anchor.fp_snr, rpy_data + i, 4);
i += 4;
qDebug() << anchor.toString();
this->anchors.append(anchor);
}
}
QString AnchorData::toString(){
QString retval;
retval.append(QString("%1 ").arg(this->id));
retval.append(QString("(%1, %2, %3) ")
.arg(this->x,0,'f',2).arg(this->y,0,'f',2).arg(this->z,0,'f',2));
retval.append(QString("%1m ").arg(double(this->distance),0,'f',2));
retval.append(QString("%1dBm").arg(double(this->rx_power),0,'f',1));
return retval;
}
QString UwbModule::anchorsToString(){
QString retval;
for(int i = 0; i < this->anchors.size(); i++){
retval.append(this->anchors[i].toString());
}
return retval;
}
/**
* @brief UwbModule::updateObject updates instance fields from the raw reply from the UWB module
* @param rpy The QByteArray read from the UWB module
*/
void UwbModule::updateObject(QByteArray& rpy_arr){
const char* rpy = rpy_arr.data(); // get the backbuffer
// make sure this is a reply to a READ command
if(rpy[0] != char(READ))
return;
int field = 0; // temporary
int ind;
for(ind = HDDR_SIZE; ind < rpy[1] + HDDR_SIZE; ind += FIELD_DATA_SIZE){
field = rpy[ind]; // field ID
ind++; // point to the field data
switch(field){
case SELF_ID:{
//memcpy(&this->id, rpy_data + ind, FIELD_DATA_SIZE);
this->id = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
break;
}case MODE:{
this->mode = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
if(this->mode == TAG_MODE){
this->modestring = QString("tag");
}else if(this->mode == ANCHOR_MODE){
this->modestring = QString("anchor");
}else{
this->modestring = QString("---");
}
break;
}case CHANNEL:{
this->channel = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
break;
}case SAMPLES_PER_RANGE:{
this->samplesPerRange = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
break;
}case NUMBER_OF_ANCHORS:{
this->numberOfAnchors = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
break;
}case FIELD_X:{
uint32_t temp = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
memcpy(&this->x, rpy + ind, FIELD_DATA_SIZE);
break;
}case FIELD_Y:{
uint32_t temp = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
memcpy(&this->y, rpy + ind, FIELD_DATA_SIZE);
break;
}case FIELD_Z:{
uint32_t temp = uint32_t( (rpy[ind]) | (rpy[ind+1] << 8) | (rpy[ind+2] << 16) | (rpy[ind+3] << 24) );
memcpy(&this->z, rpy + ind, FIELD_DATA_SIZE);
break;
}default:
break;
}
}
}
QString UwbModule::getFirmwareVersionString(char* data){
// [year].[month].[day], Rev [revision]
QString retval = QString::number(data[0], 10);
retval.append('.');
retval.append(QString::number(data[1], 10));
retval.append('.');
retval.append(QString::number(data[2], 10));
retval.append(", Rev ");
retval.append(QString::number(data[3], 10));
return retval;
}
/**
* @brief UwbModule::getId reads the ID from the UWB module and returns it as a QString
* @return QString base-10 representation of the module ID
*/
QString UwbModule::getId(){
QByteArray fields = QByteArray(1, '\0');
fields[0] = SELF_ID;
QByteArray rpy = this->readFields(fields);
char* rpy_buf = rpy.data();
uint32_t id = 0;
memcpy(&id, rpy_buf + HDDR_SIZE + 1, sizeof(id));
return QString::number(id, 10);
}
QString UwbModule::getMode(){
uint32_t mode = this->readFieldAsInt(MODE);
if(mode == TAG_MODE){
return "tag";
}else if(mode == ANCHOR_MODE){
return "anchor";
}else{
return "tag";
}
}
void UwbModule::setMode(QString s){
if(s.compare("tag") == 0){
this->modestring = "tag";
this->mode = uint32_t(TAG_MODE);
}else if(s.compare("anchor") == 0){
this->modestring = "anchor";
this->mode = uint32_t(ANCHOR_MODE);
}else{
this->modestring = "tag";
this->mode = uint32_t(TAG_MODE);
}
}
void UwbModule::setMode(int n){
if(n == TAG_MODE){
this->modestring = "tag";
this->mode = uint32_t(TAG_MODE);
}else if(n == ANCHOR_MODE){
this->modestring = "anchor";
this->mode = uint32_t(ANCHOR_MODE);
}else{
this->modestring = "tag";
this->mode = uint32_t(TAG_MODE);
}
}
int UwbModule::getChannel(){
return int(this->readFieldAsInt(CHANNEL));
}
int UwbModule::getSamplesPerRange(){
return int(this->readFieldAsInt(SAMPLES_PER_RANGE));
}
uint32_t UwbModule::readFieldAsInt(int fieldId){
QByteArray fields = QByteArray(1, char(fieldId));
QByteArray rpy = this->readFields(fields);
char* rpy_buf = rpy.data();
uint32_t retval = 0;
memcpy(&retval, rpy_buf+HDDR_SIZE+1, sizeof(retval));
return retval;
}
float UwbModule::readFieldAsFloat(int fieldId){
QByteArray fields = QByteArray(1, char(fieldId));
QByteArray rpy = this->readFields(fields);
char* rpy_buf = rpy.data();
float retval = 0;
memcpy(&retval, rpy_buf+HDDR_SIZE+1, sizeof(retval));
return retval;
}
/**
* @brief UwbModule::readFields
* @param fields array of field ID's to read
* @param nFields number of fields to read
* @retval QByteArray bytes read from the module
*/
QByteArray UwbModule::readFields(QByteArray fields){
if(!this->serial.isOpen()){
return nullptr;
}
int nFields = fields.length();
int size = 0;
if(nFields < MAX_CMD_SIZE - (HDDR_SIZE + 1)){
size = nFields;
}else{
size = MAX_CMD_SIZE - (HDDR_SIZE + 1);
}
QByteArray sendData = QByteArray(fields);
sendData.prepend(char(nFields & 0xFF));
sendData.prepend(char(READ));
sendData.append(char(STOP_BYTE & 0xFF));
QString dbg;
dbg.append("Send data (" + QString::number(sendData.length(), 10) + " bytes): ");
for(int i = 0; i < sendData.length(); i++){
dbg.append(QString::number(sendData[i], 16) + " ");
}
qDebug() << dbg;
this->serial.write(sendData);
QByteArray response;
// wait for the module to send some bytes
while(this->serial.waitForReadyRead(250))
response.append(this->serial.readAll());
dbg = QString("Reply data (" + QString::number(response.length(), 10) + " bytes): ");
for(int i = 0; i < response.length(); i++){
dbg.append(QString::number(response[i], 10) + " ");
}
qDebug() << dbg;
// char hddr[HDDR_SIZE];
// if(this->serial.read(hddr, HDDR_SIZE) != 0){
// return nullptr;
// }
// int readSize = hddr[1]; // size of the body plus stop byte
// char* body = (char*)malloc(readSize);
// if(this->serial.read(body, readSize) != 0)
// return nullptr;
// char stop[1];
// this->serial.read(stop, 1);
// response.append(hddr);
// response.append(body);
// response.append(stop);
return response;
}
/**
* @brief UwbModule::writeFields write the value of fieldData to the UWB module
* @param fieldData QBytesArray of the data to write
* @return 0 if success, -1 if fieldData is formatted wrong, -2 if something wierd happened
*/
int UwbModule::writeFields(QByteArray fieldData){
int retval = 0;
int dataLen = fieldData.length();
// make sure this makes sense
if(dataLen % CMD_WORD_SIZE != 0){
return -1; // fail
}
QByteArray sendData = QByteArray(fieldData);
sendData.prepend(char(dataLen));
sendData.prepend(char(CONFIG));
sendData.append(char(STOP_BYTE));
QString dbg;
dbg.append("Send data (" + QString::number(sendData.length(), 10) + " bytes): ");
for(int i = 0; i < sendData.length(); i++){
dbg.append(QString::number(sendData[i], 10) + " ");
}
qDebug() << dbg;
long long writeSize = this->serial.write(sendData);
if(writeSize == sendData.length()){
retval = 0; // success!
}else{
retval = -2; // hmm
}
return retval;
}
void UwbModule::saveConfig(){
QByteArray cmd;
cmd.append(char(SAVE_CONFIG));
cmd.append(char(0x00));
cmd.append(char(STOP_BYTE));
this->serial.write(cmd);
QByteArray rpy;
while(this->serial.waitForReadyRead(250))
rpy.append(this->serial.readAll());
}
int UwbModule::addFieldToBuffer(QByteArray& buffer, UwbConfigField fieldId, uint32_t value){
char* temp = new char[FIELD_DATA_SIZE];
memcpy(temp, &value, FIELD_DATA_SIZE);
buffer.append(char(fieldId));
buffer.append(temp[0]); // LSB copied first
buffer.append(temp[1]);
buffer.append(temp[2]);
buffer.append(temp[3]); // MSB copied last
return 1 + FIELD_DATA_SIZE;
}
int UwbModule::addFieldToBuffer(QByteArray& buffer, UwbConfigField fieldId, float value){
char* temp = new char[FIELD_DATA_SIZE];
memcpy(temp, &value, FIELD_DATA_SIZE);
buffer.append(char(fieldId));
buffer.append(temp[0]); // LSB copied first
buffer.append(temp[1]);
buffer.append(temp[2]);
buffer.append(temp[3]); // MSB copied last
return 1 + FIELD_DATA_SIZE;
}