-
Notifications
You must be signed in to change notification settings - Fork 2
/
MSP.cpp
220 lines (181 loc) · 5.78 KB
/
MSP.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
/*
MSP.cpp
Copyright (c) 2017, Fabrizio Di Vittorio ([email protected])
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <Arduino.h>
#include "MSP.h"
void MSP::begin(Stream & stream, uint32_t timeout)
{
_stream = &stream;
_timeout = timeout;
}
void MSP::reset()
{
_stream->flush();
while (_stream->available() > 0)
_stream->read();
}
void MSP::send(uint8_t messageID, void * payload, uint8_t size)
{
_stream->write('$');
_stream->write('M');
_stream->write('<');
_stream->write(size);
_stream->write(messageID);
uint8_t checksum = size ^ messageID;
uint8_t * payloadPtr = (uint8_t*)payload;
for (uint8_t i = 0; i < size; ++i) {
uint8_t b = *(payloadPtr++);
checksum ^= b;
_stream->write(b);
}
_stream->write(checksum);
}
// timeout in milliseconds
bool MSP::recv(uint8_t * messageID, void * payload, uint8_t maxSize, uint8_t * recvSize)
{
uint32_t t0 = millis();
while (1) {
// read header
while (_stream->available() < 6)
if (millis() - t0 >= _timeout)
return false;
char header[3];
_stream->readBytes((char*)header, 3);
// check header
if (header[0] == '$' && header[1] == 'M' && header[2] == '>') {
// header ok, read payload size
*recvSize = _stream->read();
// read message ID (type)
*messageID = _stream->read();
uint8_t checksumCalc = *recvSize ^ *messageID;
// read payload
uint8_t * payloadPtr = (uint8_t*)payload;
uint8_t idx = 0;
while (idx < *recvSize) {
if (millis() - t0 >= _timeout)
return false;
if (_stream->available() > 0) {
uint8_t b = _stream->read();
checksumCalc ^= b;
if (idx < maxSize)
*(payloadPtr++) = b;
++idx;
}
}
// zero remaining bytes if *size < maxSize
for (; idx < maxSize; ++idx)
*(payloadPtr++) = 0;
// read and check checksum
while (_stream->available() == 0)
if (millis() - t0 >= _timeout)
return false;
uint8_t checksum = _stream->read();
if (checksumCalc == checksum) {
return true;
}
}
}
}
// wait for messageID
// recvSize can be NULL
bool MSP::waitFor(uint8_t messageID, void * payload, uint8_t maxSize, uint8_t * recvSize)
{
uint8_t recvMessageID;
uint8_t recvSizeValue;
uint32_t t0 = millis();
while (millis() - t0 < _timeout)
if (recv(&recvMessageID, payload, maxSize, (recvSize ? recvSize : &recvSizeValue)) && messageID == recvMessageID)
return true;
// timeout
return false;
}
// send a message and wait for the reply
// recvSize can be NULL
bool MSP::request(uint8_t messageID, void * payload, uint8_t maxSize, uint8_t * recvSize)
{
send(messageID, NULL, 0);
return waitFor(messageID, payload, maxSize, recvSize);
}
// send message and wait for ack
bool MSP::command(uint8_t messageID, void * payload, uint8_t size, bool waitACK)
{
send(messageID, payload, size);
// ack required
if (waitACK)
return waitFor(messageID, NULL, 0);
return true;
}
// map MSP_MODE_xxx to box ids
// mixed values from cleanflight and inav
static const uint8_t BOXIDS[30] PROGMEM = {
0, // 0: MSP_MODE_ARM
1, // 1: MSP_MODE_ANGLE
2, // 2: MSP_MODE_HORIZON
3, // 3: MSP_MODE_NAVALTHOLD (cleanflight BARO)
5, // 4: MSP_MODE_MAG
6, // 5: MSP_MODE_HEADFREE
7, // 6: MSP_MODE_HEADADJ
8, // 7: MSP_MODE_CAMSTAB
10, // 8: MSP_MODE_NAVRTH (cleanflight GPSHOME)
11, // 9: MSP_MODE_NAVPOSHOLD (cleanflight GPSHOLD)
12, // 10: MSP_MODE_PASSTHRU
13, // 11: MSP_MODE_BEEPERON
15, // 12: MSP_MODE_LEDLOW
16, // 13: MSP_MODE_LLIGHTS
19, // 14: MSP_MODE_OSD
20, // 15: MSP_MODE_TELEMETRY
21, // 16: MSP_MODE_GTUNE
22, // 17: MSP_MODE_SONAR
26, // 18: MSP_MODE_BLACKBOX
27, // 19: MSP_MODE_FAILSAFE
28, // 20: MSP_MODE_NAVWP (cleanflight AIRMODE)
29, // 21: MSP_MODE_AIRMODE (cleanflight DISABLE3DSWITCH)
30, // 22: MSP_MODE_HOMERESET (cleanflight FPVANGLEMIX)
31, // 23: MSP_MODE_GCSNAV (cleanflight BLACKBOXERASE)
32, // 24: MSP_MODE_HEADINGLOCK
33, // 25: MSP_MODE_SURFACE
34, // 26: MSP_MODE_FLAPERON
35, // 27: MSP_MODE_TURNASSIST
36, // 28: MSP_MODE_NAVLAUNCH
37, // 29: MSP_MODE_AUTOTRIM
};
// returns active mode (using MSP_STATUS and MSP_BOXIDS messages)
// see MSP_MODE_... for bits inside activeModes
bool MSP::getActiveModes(uint32_t * activeModes)
{
// request status ex
msp_status_t status;
if (request(MSP_STATUS, &status, sizeof(status))) {
// request permanent ids associated to boxes
uint8_t ids[sizeof(BOXIDS)];
uint8_t recvSize;
if (request(MSP_BOXIDS, ids, sizeof(ids), &recvSize)) {
// compose activeModes, converting BOXIDS to bit map (setting 1 if related flag in flightModeFlags is set)
*activeModes = 0;
for (uint8_t i = 0; i < recvSize; ++i) {
if (status.flightModeFlags & (1 << i)) {
for (uint8_t j = 0; j < sizeof(BOXIDS); ++j) {
if (pgm_read_byte(BOXIDS + j) == ids[i]) {
*activeModes |= 1 << j;
break;
}
}
}
}
return true;
}
}
return false;
}