2020-04-27 15:10:17 +00:00
|
|
|
#include "StreamAPI.h"
|
2020-04-29 00:43:16 +00:00
|
|
|
#include "configuration.h"
|
2020-04-27 15:45:39 +00:00
|
|
|
|
|
|
|
#define START1 0x94
|
|
|
|
#define START2 0xc3
|
|
|
|
#define HEADER_LEN 4
|
|
|
|
|
|
|
|
void StreamAPI::loop()
|
|
|
|
{
|
|
|
|
writeStream();
|
|
|
|
readStream();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Read any rx chars from the link and call handleToRadio
|
|
|
|
*/
|
|
|
|
void StreamAPI::readStream()
|
|
|
|
{
|
|
|
|
while (stream->available()) { // Currently we never want to block
|
|
|
|
uint8_t c = stream->read();
|
|
|
|
|
|
|
|
// Use the read pointer for a little state machine, first look for framing, then length bytes, then payload
|
|
|
|
size_t ptr = rxPtr++; // assume we will probably advance the rxPtr
|
|
|
|
|
|
|
|
rxBuf[ptr] = c; // store all bytes (including framing)
|
|
|
|
|
|
|
|
if (ptr == 0) { // looking for START1
|
|
|
|
if (c != START1)
|
|
|
|
rxPtr = 0; // failed to find framing
|
|
|
|
} else if (ptr == 1) { // looking for START2
|
|
|
|
if (c != START2)
|
|
|
|
rxPtr = 0; // failed to find framing
|
|
|
|
} else if (ptr >= HEADER_LEN) { // we have at least read our 4 byte framing
|
2020-05-10 04:02:56 +00:00
|
|
|
uint32_t len = (rxBuf[2] << 8) + rxBuf[3]; // big endian 16 bit length follows framing
|
2020-04-27 15:45:39 +00:00
|
|
|
|
|
|
|
if (ptr == HEADER_LEN) {
|
|
|
|
// we _just_ finished our 4 byte header, validate length now (note: a length of zero is a valid
|
|
|
|
// protobuf also)
|
|
|
|
if (len > MAX_TO_FROM_RADIO_SIZE)
|
|
|
|
rxPtr = 0; // length is bogus, restart search for framing
|
|
|
|
}
|
|
|
|
|
2020-04-28 01:52:57 +00:00
|
|
|
if (rxPtr != 0 && ptr + 1 == len + HEADER_LEN) {
|
2020-04-27 15:45:39 +00:00
|
|
|
// If we didn't just fail the packet and we now have the right # of bytes, parse it
|
|
|
|
handleToRadio(rxBuf + HEADER_LEN, len);
|
2020-04-28 01:52:57 +00:00
|
|
|
rxPtr = 0; // start over again
|
2020-04-27 15:45:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* call getFromRadio() and deliver encapsulated packets to the Stream
|
|
|
|
*/
|
|
|
|
void StreamAPI::writeStream()
|
|
|
|
{
|
2020-04-27 16:36:39 +00:00
|
|
|
if (canWrite) {
|
|
|
|
uint32_t len;
|
|
|
|
do {
|
|
|
|
// Send every packet we can
|
|
|
|
len = getFromRadio(txBuf + HEADER_LEN);
|
2020-04-29 00:43:16 +00:00
|
|
|
emitTxBuffer(len);
|
2020-04-27 16:36:39 +00:00
|
|
|
} while (len);
|
|
|
|
}
|
2020-04-29 00:43:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send the current txBuffer over our stream
|
|
|
|
*/
|
|
|
|
void StreamAPI::emitTxBuffer(size_t len)
|
|
|
|
{
|
|
|
|
if (len != 0) {
|
|
|
|
txBuf[0] = START1;
|
|
|
|
txBuf[1] = START2;
|
|
|
|
txBuf[2] = (len >> 8) & 0xff;
|
|
|
|
txBuf[3] = len & 0xff;
|
|
|
|
|
|
|
|
stream->write(txBuf, len + HEADER_LEN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void StreamAPI::emitRebooted()
|
|
|
|
{
|
|
|
|
// In case we send a FromRadio packet
|
|
|
|
memset(&fromRadioScratch, 0, sizeof(fromRadioScratch));
|
|
|
|
fromRadioScratch.which_variant = FromRadio_rebooted_tag;
|
|
|
|
fromRadioScratch.variant.rebooted = true;
|
|
|
|
|
|
|
|
DEBUG_MSG("Emitting reboot packet for serial shell\n");
|
|
|
|
emitTxBuffer(pb_encode_to_bytes(txBuf + HEADER_LEN, FromRadio_size, FromRadio_fields, &fromRadioScratch));
|
2020-04-27 15:45:39 +00:00
|
|
|
}
|