firmware/src/MeshRadio.cpp

163 lines
4.6 KiB
C++
Raw Normal View History

2020-02-01 16:59:16 +00:00
#include <SPI.h>
#include <RH_RF95.h>
2020-02-01 19:25:07 +00:00
#include <RHMesh.h>
#include <assert.h>
2020-02-01 16:59:16 +00:00
#include <pb_encode.h>
#include <pb_decode.h>
2020-02-01 16:59:16 +00:00
#include "MeshRadio.h"
#include "configuration.h"
2020-02-03 17:13:19 +00:00
#include "NodeDB.h"
2020-02-01 16:59:16 +00:00
// Change to 434.0 or other frequency, must match RX's freq!
#define RF95_FREQ 915.0
MeshRadio::MeshRadio(MemoryPool<MeshPacket> &_pool, PointerQueue<MeshPacket> &_rxDest)
: rf95(NSS_GPIO, DIO0_GPIO),
2020-02-03 17:13:19 +00:00
manager(rf95, nodeDB.getNodeNum()),
pool(_pool),
rxDest(_rxDest),
txQueue(MAX_TX_QUEUE)
{
}
2020-02-01 16:59:16 +00:00
bool MeshRadio::init()
{
pinMode(RESET_GPIO, OUTPUT); // Deassert reset
digitalWrite(RESET_GPIO, HIGH);
// pulse reset
digitalWrite(RESET_GPIO, LOW);
delay(10);
digitalWrite(RESET_GPIO, HIGH);
delay(10);
if (!manager.init())
{
Serial.println("LoRa radio init failed");
Serial.println("Uncomment '#define SERIAL_DEBUG' in RH_RF95.cpp for detailed debug info");
return false;
}
2020-02-01 16:59:16 +00:00
Serial.println("LoRa radio init OK!");
2020-02-01 19:25:07 +00:00
// Defaults after init are 434.0MHz, modulation GFSK_Rb250Fd250, +13dbM
if (!rf95.setFrequency(RF95_FREQ))
{
Serial.println("setFrequency failed");
while (1)
;
}
Serial.print("Set Freq to: ");
Serial.println(RF95_FREQ);
2020-02-01 19:25:07 +00:00
// Defaults after init are 434.0MHz, 13dBm, Bw = 125 kHz, Cr = 4/5, Sf = 128chips/symbol, CRC on
2020-02-01 19:25:07 +00:00
// The default transmitter power is 13dBm, using PA_BOOST.
// If you are using RFM95/96/97/98 modules which uses the PA_BOOST transmitter pin, then
// you can set transmitter powers from 5 to 23 dBm:
// FIXME - can we do this? It seems to be in the Heltec board.
rf95.setTxPower(23, false);
2020-02-02 02:45:27 +00:00
return true;
2020-02-01 19:25:07 +00:00
}
ErrorCode MeshRadio::send(MeshPacket *p)
{
2020-02-02 21:29:53 +00:00
Serial.println("enquing packet for sending on mesh");
return txQueue.enqueue(p, 0); // nowait
2020-02-01 16:59:16 +00:00
}
ErrorCode MeshRadio::sendTo(NodeNum dest, const uint8_t *buf, size_t len)
{
Serial.printf("mesh sendTo %d bytes to %d\n", len, dest);
// FIXME - for now we do all packets as broadcast
dest = NODENUM_BROADCAST;
2020-02-03 03:08:04 +00:00
assert(len <= 255); // Make sure we don't overflow the tiny max packet size
2020-02-02 21:29:53 +00:00
// Note: we don't use sendToWait here because we don't want to wait and for the time being don't require
// reliable delivery
// return manager.sendtoWait((uint8_t *) buf, len, dest);
return manager.sendto((uint8_t *)buf, len, dest) ? ERRNO_OK : ERRNO_UNKNOWN;
}
2020-02-01 16:59:16 +00:00
2020-02-03 03:08:04 +00:00
/// enqueue a received packet in rxDest
void MeshRadio::handleReceive(MeshPacket *mp)
{
int res = rxDest.enqueue(mp, 0); // NOWAIT - fixme, if queue is full, delete older messages
assert(res == pdTRUE);
}
void MeshRadio::loop()
2020-02-01 16:59:16 +00:00
{
// FIXME read from radio with recvfromAckTimeout
2020-02-01 19:25:07 +00:00
#if 0
static int16_t packetnum = 0; // packet counter, we increment per xmission
2020-02-02 00:05:12 +00:00
2020-02-01 16:59:16 +00:00
char radiopacket[20] = "Hello World # ";
2020-02-01 19:25:07 +00:00
sprintf(radiopacket, "hello %d", packetnum++);
2020-02-01 16:59:16 +00:00
assert(sendTo(NODENUM_BROADCAST, (uint8_t *)radiopacket, sizeof(radiopacket)) == ERRNO_OK);
#endif
2020-02-02 21:29:53 +00:00
/// A temporary buffer used for sending/receving packets, sized to hold the biggest buffer we might need
static uint8_t radiobuf[SubPacket_size];
uint8_t rxlen;
uint8_t srcaddr, destaddr, id, flags;
// Poll to see if we've received a packet
if (manager.recvfromAckTimeout(radiobuf, &rxlen, 0, &srcaddr, &destaddr, &id, &flags))
{
// We received a packet
Serial.printf("Received packet from mesh src=%d,dest=%d,id=%d,len=%d\n", srcaddr, destaddr, id, rxlen);
MeshPacket *mp = pool.allocZeroed();
assert(mp); // FIXME
SubPacket *p = &mp->payload;
mp->from = srcaddr;
mp->to = destaddr;
pb_istream_t stream = pb_istream_from_buffer(radiobuf, rxlen);
if (!pb_decode(&stream, SubPacket_fields, p))
{
Serial.printf("Error: can't decode SubPacket %s\n", PB_GET_ERROR(&stream));
pool.release(mp);
}
else
{
// parsing was successful, queue for our recipient
mp->has_payload = true;
2020-02-03 03:08:04 +00:00
handleReceive(mp);
2020-02-02 21:29:53 +00:00
}
}
2020-02-02 21:29:53 +00:00
// Poll to see if we need to send any packets
MeshPacket *txp = txQueue.dequeuePtr(0); // nowait
if (txp)
{
Serial.println("sending queued packet on mesh");
assert(txp->has_payload);
pb_ostream_t stream = pb_ostream_from_buffer(radiobuf, sizeof(radiobuf));
if (!pb_encode(&stream, SubPacket_fields, &txp->payload))
{
Serial.printf("Error: can't encode SubPacket %s\n", PB_GET_ERROR(&stream));
}
else
{
int res = sendTo(txp->to, radiobuf, stream.bytes_written);
assert(res == ERRNO_OK);
}
2020-02-03 03:08:04 +00:00
bool loopbackTest = false; // if true we will pretend to receive any packets we just sent
if (loopbackTest)
handleReceive(txp);
else
pool.release(txp);
2020-02-02 21:29:53 +00:00
}
}