Serial Wombat Arduino Library
Loading...
Searching...
No Matches
SerialWombat.cpp
Go to the documentation of this file.
1#include "Arduino.h"
2#include "SerialWombat.h"
3/*
4Copyright 2020-2022 Broadwell Consulting Inc.
5
6Serial Wombat is a registered trademark in the US of Broadwell Consulting Inc.
7
8Permission is hereby granted, free of charge, to any person obtaining a
9 * copy of this software and associated documentation files (the "Software"),
10 * to deal in the Software without restriction, including without limitation
11 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
12 * and/or sell copies of the Software, and to permit persons to whom the
13 * Software is furnished to do so, subject to the following conditions:
14
15The above copyright notice and this permission notice shall be included in
16 * all copies or substantial portions of the Software.
17
18THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
21 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
22 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
23 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
24 * OTHER DEALINGS IN THE SOFTWARE.
25*/
26
29
30
34
35
36
40
41
42
43
44
45int16_t SerialWombatChip::begin(uint8_t i2cAddress)
46{
47 //i2cInterface = &Wire;
48 //address = i2cAddress;
49 return begin(Wire, i2cAddress);
50}
51
52int SerialWombatChip::sendPacket(uint8_t tx[], bool retryIfEchoDoesntMatch)
53{
54 uint8_t rx[8];
55 return (sendPacket(tx,rx, retryIfEchoDoesntMatch));
56}
57
58int SerialWombatChip::sendPacket(uint8_t tx[], uint8_t rx[], bool retryIfEchoDoesntMatch, uint8_t startBytesToMatch, uint8_t endBytesToMatch)
59{
60 uint8_t retry = 4; //TODO communicationErrorRetries;
61
62
63 if (!retryIfEchoDoesntMatch)
64 {
65 retry = 1;
66 }
67
68 int result = 0;
69 for (int i = 0; i < 8; ++i)
70 {
71 rx[i] = 0;
72 }
73 while (retry)
74 {
75 result = sendPacket(tx, rx);
76
77 if (rx[0] == 'E')
78 {
79 return result;
80 }
81 uint8_t i;
82 bool success = true;
83 for (i = 0; i < startBytesToMatch; ++i)
84 {
85 if ( tx[i] != rx[i])
86 {
87 success = false;
88 }
89 }
90 for (i = 8- endBytesToMatch; i < 8; ++i)
91 {
92 if (tx[i] != rx[i])
93 {
94 success = false;
95 }
96 }
97 if (success)
98 {
99 return (result);
100 }
101 --retry;
102 delayMicroseconds(100);
103 {
104 //char echoTx[] = "!COM_ERR";
105 //sendPacket((uint8_t*)echoTx);
106 continue;
107 }
108 delayMicroseconds(100);
109
110 }
111
112 return(result);
113}
114
115int SerialWombatChip::sendPacket(uint8_t tx[], uint8_t rx[])
116{
117 if (sendReadyTime != 0 )
118 {
119 unsigned long currentTime = millis();
120 if (currentTime < sendReadyTime)
121 {
122 delay(sendReadyTime - currentTime);
123
124 }
125 sendReadyTime = 0;
126 initialize();
127 }
128
129 if (Serial != NULL)
130 {
131 if (_currentlyCommunicating)
132 {
133 //TODO return (-1 * SW_ERROR_REENTRANCY_NOT_SUPPORTED);
134 }
135 _currentlyCommunicating = true;
136 uint32_t milisStart = millis();
137 while (Serial->read() >= 0);
138 uint32_t milisFlush = millis();
139 Serial->write(tx, 8); //TODO add addressing, CRC
140 uint32_t millisWrite = millis();
141 int bytesRx = 0;
142
143 uint32_t timenow = millis();
144 while (bytesRx < 8 && timenow <= (millisWrite + 50))
145 {
146 int32_t data = Serial->read();
147 if (data >= 0)
148 {
149 rx[bytesRx] = data;
150 ++bytesRx;
151 }
152 timenow = millis();
153 }
154 uint32_t milisread = millis();
155 if (bytesRx < 8)
156 {
157 if (errorHandler != NULL)
158 {
159 errorHandler(SW_ERROR_LESS_THAN_8_BYTES_RETURNED,this);
160 }
161 ++errorCount;
162 _currentlyCommunicating = false;
164 }
165 if (rx[0] == 'E')
166 {
168 if (errorHandler != NULL)
169 {
170 errorHandler(lastErrorCode,this);
171 }
172 lastErrorCode *= -1;
173 ++errorCount;
174 _currentlyCommunicating = false;
175 return (lastErrorCode);
176 }
177 _currentlyCommunicating = false;
178 return (8);
179 }
180
181 if (i2cInterface != NULL)
182 {
183 int bytesWritten;
184 uint8_t retry = communicationErrorRetries;
185 int i2cResult;
186 int count = 8;
187 //while (tx[count - 1] == 0x55)
188 //{
189 // --count;
190 //}
191 if (_currentlyCommunicating)
192 {
193 //TODO return (-1 * SW_ERROR_REENTRANCY_NOT_SUPPORTED);
194 }
195 _currentlyCommunicating = true;
196 while (retry)
197 {
198 --retry;
199 i2cInterface->beginTransmission(address);
200 bytesWritten = i2cInterface->write(tx, 8);
201 i2cResult = i2cInterface->endTransmission();
202
203 if (bytesWritten == 8 && i2cResult == 0)
204 {
205 retry = 0;
206 }
207 else
208 {
209 delayMicroseconds(100);
210 char echoTx[] = "!COM_ERR";
211 /*
212 i2cInterface->beginTransmission(address);
213 bytesWritten = i2cInterface->write((uint8_t*)echoTx, 8);
214 i2cResult = i2cInterface->endTransmission();
215 */
216 continue;
217 }
218 }
219 //delay(3);
220 //Serial->setTimeout(10);
221 delayMicroseconds(100);
222 i2cInterface->requestFrom(address, (uint8_t)8);
223
224 count = 0;
225 int r = 0;
226 while (r >= 0 && count < 8)
227 {
228 r = i2cInterface->read();
229
230 if (r >= 0)
231 {
232 rx[count] = (uint8_t) r;
233 ++count;
234 }
235 else
236 {
237 break;
238 }
239 }
240
241 if (count < 8)
242 {
243 delayMicroseconds(100);
244 i2cInterface->requestFrom(address, (uint8_t)8);
245
246 count = 0;
247 while (i2cInterface->available() && count < 8)
248 {
249 int r = i2cInterface->read();
250
251 if (r >= 0)
252 {
253 rx[count] = (uint8_t)r;
254 ++count;
255 }
256 else
257 {
258 break;
259 }
260 }
261 if (count < 8)
262 {
263 if (errorHandler != NULL)
264 {
265 errorHandler(SW_ERROR_LESS_THAN_8_BYTES_RETURNED, this);
266 }
267 ++errorCount;
268 _currentlyCommunicating = false;
270 }
271 }
272 if (rx[0] == 'E')
273 {
275 if (errorHandler != NULL)
276 {
277 errorHandler(lastErrorCode,this);
278 }
279 lastErrorCode *= -1;
280 ++errorCount;
281 _currentlyCommunicating = false;
282 return (lastErrorCode);
283 }
284 }
285 _currentlyCommunicating = false;
286 return(0);
287
288}
290{
291
292 if (Serial != NULL)
293 {
294 Serial->write(tx, 8); //TODO add addressing, CRC
295 return (8);
296 }
297
298 if (i2cInterface != NULL)
299 {
300 int count = 8;
301 //while (tx[count - 1] == 0x55)
302 //{
303 // --count;
304 //}
305 i2cInterface->beginTransmission(address);
306 i2cInterface->write(tx, 8);
307 i2cInterface->endTransmission();
308 }
309 return(0);
310
311}
312
314{
315 uint8_t rx[8];
316
317 if (_asleep)
318 {
319 _asleep = false;
320 uint8_t txw[8] = { '!','!','!','!','!','!','!','!' }; //TODO adapt for UART
321 sendPacket(txw);
322 delayMicroseconds(200);
323 }
324
325 return(sendPacket(tx,rx));
326
327 if (sendReadyTime != 0)
328 {
329 unsigned long currentTime = millis();
330 if (currentTime < sendReadyTime)
331 {
332 delay(sendReadyTime - currentTime);
333
334 }
335 sendReadyTime = 0;
336 initialize();
337 }
338
339 if (Serial != NULL)
340 {
341 return Serial->write(tx, 8); //TODO add addressing, CRC
342 }
343
344 if (i2cInterface != NULL)
345 {
346 int count = 8;
347 // while (tx[count - 1] == 0x55)
348 //{
349 // --count;
350 //}
351 i2cInterface->beginTransmission(address);
352 i2cInterface->write(tx, 8);
353 i2cInterface->endTransmission();
354 }
355 return(0);
356
357}
358
360{
361 char s[10];
362 Serial.print("Comm Error on SW at addr ");
363 sprintf(s, "0x%02X ", sw->address);
364 Serial.print(s);
365 Serial.print(" Error code ");
366
367 Serial.print(error);
368
369 Serial.print(" Error count ");
370 Serial.print(sw->errorCount);
371
372 uint8_t errorCmd[8];
373 if (sw->readLastErrorCommand(errorCmd) >= 0)
374 {
375 Serial.print(" Command: ");
376 for (int i = 0; i < 8; ++i)
377 {
378 sprintf(s, "0x%02X ", errorCmd[i]);
379 Serial.print(s);
380 }
381 }
382
383 Serial.println();
384}
385
void SerialWombatSerialErrorHandlerBrief(uint16_t error, SerialWombatChip *sw)
a sample error handler that can be registered with registerErrorHandler to report protocol errors on ...
@ SW_ERROR_LESS_THAN_8_BYTES_RETURNED
(#48) The Serial Wombat Chip returned less than 8 bytes (Used at host level, not firmware)
Class for a Serial Wombat chip. Each Serial Wombat chip on a project should have its own instance.
uint8_t address
The I2C address of the SerialWombatChip instance.
uint32_t sendReadyTime
uint8_t communicationErrorRetries
How many times to retry a packet if communcation bus (such as I2C) error.
int sendPacketNoResponse(uint8_t tx[])
Send an 8 byte packet to the Serial Wombat chip, don't wait for a response.
int16_t lastErrorCode
The last error code returned as part of a protocol error message expressed as a positive integer.
int16_t begin(HardwareSerial &serial, bool reset=true)
initialize a Serial Wombat chip to use a Serial Interface.
int sendPacket(uint8_t tx[], uint8_t rx[])
Send an 8 byte packet to the Serial Wombat chip and wait for 8 bytes back.
uint16_t returnErrorCode(uint8_t *rx)
int16_t initialize()
uint16_t errorCount
Incremented every time a communication or command error is detected.
int16_t readLastErrorCommand(uint8_t *cmd)
Returns the last Serial Wombat command that produced a protocol error.