CubeCellRepeater/main.cpp

232 lines
7.6 KiB
C++
Raw Normal View History

2020-09-18 19:14:51 +00:00
#include <Arduino.h> // needed for platform.io
#include "mesh.pb.h"
#include "CubeCell_NeoPixel.h"
#include "MeshRadio.h"
// CONFIGURATION: change values in MeshRadio.h !
#include <radio.h>
#define VERBOSE // define to SILENT to turn off serial messages
#define BLINK // define to NOBLINK to turn off LED signaling
2020-09-14 14:25:07 +00:00
2020-09-21 18:26:49 +00:00
typedef struct {
uint32_t to, from, id;
uint8_t flags; // The bottom three bits of flags are use to store hop_limit, bit 4 is the WANT_ACK flag
2020-09-21 18:26:49 +00:00
} PacketHeader;
void TxDone( void );
void TxTimeout( void );
void RxDone( uint8_t *payload, uint16_t size, int16_t rssi, int8_t snr );
2020-09-18 19:14:51 +00:00
void ConfigureRadio( ChannelSettings ChanSet );
unsigned long hash(char *str);
2020-09-21 18:26:49 +00:00
MeshPacket thePacket;
ChannelSettings ChanSet;
2020-09-21 18:26:49 +00:00
static RadioEvents_t RadioEvents;
uint32_t receivedID[ID_BUFFER_SIZE];
int receivedCount = -1;
CubeCell_NeoPixel pixels(1, RGB, NEO_GRB + NEO_KHZ800);
2020-09-14 14:25:07 +00:00
void setup() {
#ifndef NOBLINK
pinMode(Vext,OUTPUT);
digitalWrite(Vext,LOW); //SET POWER
pixels.begin(); // INITIALIZE NeoPixel strip object (REQUIRED)
pixels.clear(); // Set all pixel colors to 'off'
pixels.show();
#endif
#ifndef SILENT
2020-09-14 14:25:07 +00:00
Serial.begin(115200);
Serial.println("\nSetting up Radio:");
#endif
2020-09-21 18:26:49 +00:00
RadioEvents.TxDone = TxDone;
RadioEvents.TxTimeout = TxTimeout;
RadioEvents.RxDone = RxDone;
2020-09-14 14:25:07 +00:00
Radio.Init( &RadioEvents );
2020-09-21 18:26:49 +00:00
Radio.Sleep();
2020-09-21 19:56:31 +00:00
memcpy(ChanSet.name, MESHTASTIC_NAME, 12);
ChanSet.channel_num = hash( MESHTASTIC_NAME ) % NUM_CHANNELS; // see MeshRadio.h
ChanSet.tx_power = TX_OUTPUT_POWER;
ChanSet.psk = MESHTASTIC_PSK;
/* FYI:
"bandwidth":
[0: 125 kHz, 1: 250 kHz, 2: 500 kHz, 3: 62.5kHz, 4: 41.67kHz, 5: 31.25kHz, 6: 20.83kHz, 7: 15.63kHz, 8: 10.42kHz, 9: 7.81kHz]
"speed":
0: ChannelSettings_ModemConfig_Bw125Cr45Sf128 aka short range
1: ChannelSettings_ModemConfig_Bw500Cr45Sf128 aka medium range
2: ChannelSettings_ModemConfig_Bw31_25Cr48Sf512 aka long range
3: ChannelSettings_ModemConfig_Bw125Cr48Sf4096 aka very long range
"coding rate":
[1: 4/5, 2: 4/6, 3: 4/7, 4: 4/8]
*/
switch ( MESHTASTIC_SPEED ){
case 0: { // short range
ChanSet.bandwidth = 0; // 125 kHz
ChanSet.coding_rate = 1; // = 4/5
ChanSet.spread_factor = 7;
break;
}
case 1: { // medium range
ChanSet.bandwidth = 2; // 500 kHz
ChanSet.coding_rate = 1; // = 4/5
ChanSet.spread_factor = 7;
break;
}
case 2: { // long range
ChanSet.bandwidth = 5; // 31.25 kHz
ChanSet.coding_rate = 4; // = 4/8
ChanSet.spread_factor = 9;
break;
}
case 3: { // very long range
ChanSet.bandwidth = 0; // 125 kHz
ChanSet.coding_rate = 4; // = 4/8
ChanSet.spread_factor = 12;
break;
}
default:{ // default setting is medium range
ChanSet.bandwidth = 2; // 500 kHz
ChanSet.coding_rate = 1; // = 4/5
ChanSet.spread_factor = 7;
}
}
2020-09-18 19:14:51 +00:00
ConfigureRadio( ChanSet );
#ifndef SILENT
2020-09-21 18:26:49 +00:00
Serial.println("..done! Switch to Receive Mode.\n");
#endif
2020-09-21 18:26:49 +00:00
Radio.Rx(0); // initial mode = receive
2020-09-14 14:25:07 +00:00
}
2020-09-21 18:26:49 +00:00
void loop( )
2020-09-14 14:25:07 +00:00
{
2020-09-21 18:26:49 +00:00
lowPowerHandler( );
2020-09-14 14:25:07 +00:00
Radio.IrqProcess( );
}
2020-09-21 18:26:49 +00:00
void TxDone( void )
2020-09-14 14:25:07 +00:00
{
2020-09-21 18:26:49 +00:00
Radio.Sleep( );
#ifndef NOBLINK
pixels.clear( );
pixels.show( );
#endif
#ifndef SILENT
Serial.println(".done! Switch to Receive Mode.");
#endif
2020-09-21 18:26:49 +00:00
Radio.Rx( 0 ); // switch to receive mode
2020-09-14 14:25:07 +00:00
}
2020-09-21 18:26:49 +00:00
void TxTimeout( void )
2020-09-14 14:25:07 +00:00
{
2020-09-21 18:26:49 +00:00
Radio.Sleep( );
#ifndef NOBLINK
pixels.clear( );
pixels.show( );
#endif
#ifndef SILENT
Serial.println(".failed (TX Timeout)! Switch to Receive Mode.");
#endif
2020-09-21 18:26:49 +00:00
Radio.Rx( 0 ); // switch to receive mode
2020-09-14 14:25:07 +00:00
}
2020-09-21 18:26:49 +00:00
void RxDone( uint8_t *payload, uint16_t size, int16_t rssi, int8_t snr )
2020-09-14 14:25:07 +00:00
{
2020-09-21 18:26:49 +00:00
Radio.Sleep( );
if ( size > MAX_PAYLOAD_LENGTH ) size = MAX_PAYLOAD_LENGTH;
if ( !(size > sizeof(PacketHeader)) ) {
#ifndef SILENT
Serial.printf("\nReceived packet! (Size %i bytes, RSSI %i, SNR %i)\n", size, rssi, snr);
Serial.println("Packet to small (No MeshPacket), discard. Switch to Receive Mode.");
#endif
Radio.Rx( 0 );
return;
}
PacketHeader * h = (PacketHeader *)payload;
MeshPacket * p = &thePacket;
p->to = h->to;
p->from = h->from;
p->id = h->id;
p->hop_limit = h->flags && 0b00000111;
p->want_ack = h->flags && 0b00001000;
p->which_payload = MeshPacket_encrypted_tag;
p->encrypted.size= size - sizeof(PacketHeader);
memcpy(p->encrypted.bytes, payload + sizeof(PacketHeader), p->encrypted.size);
2020-09-14 14:25:07 +00:00
#ifndef SILENT
Serial.printf("\nReceived packet! (Size %i bytes, RSSI %i, SNR %i)\n", size, rssi, snr);
Serial.print("TO: ");
2020-09-21 18:26:49 +00:00
(thePacket.to == UINT32_MAX) ? Serial.print( "BROADCAST" ) : Serial.print( thePacket.to, HEX );
2020-09-14 14:25:07 +00:00
Serial.print(" FROM: ");
2020-09-21 18:26:49 +00:00
Serial.print(thePacket.from, HEX);
2020-09-14 14:25:07 +00:00
2020-09-21 18:26:49 +00:00
Serial.print(" Packet ID: ");
Serial.print(thePacket.id, HEX);
2020-09-14 14:25:07 +00:00
2020-09-21 18:26:49 +00:00
Serial.print(" Flags: WANT_ACK=");
Serial.print( (thePacket.want_ack) ? "YES " : "NO ");
Serial.printf("HOP_LIMIT=%i\n", thePacket.hop_limit);
2020-09-14 14:25:07 +00:00
Serial.print("Payload: ");
2020-09-21 18:26:49 +00:00
for(int i=0; i < p->encrypted.size; i++){
Serial.print(p->encrypted.bytes[i], HEX);
2020-09-14 14:25:07 +00:00
Serial.print(" ");
}
Serial.println();
#endif
2020-09-21 18:26:49 +00:00
bool found = false;
2020-09-14 14:25:07 +00:00
for(int i = 0; i < ID_BUFFER_SIZE; i++){
2020-09-21 18:26:49 +00:00
if (receivedID[i] == thePacket.id) found = true;
2020-09-14 14:25:07 +00:00
}
if (!found){
2020-09-21 18:26:49 +00:00
// will repeat package
#ifndef SILENT
Serial.print("Sending packet..");
Serial.printf("(Size: %i)..", size);
#endif
2020-09-18 19:14:51 +00:00
(receivedCount < (ID_BUFFER_SIZE - 1) ) ? receivedCount++ : receivedCount = 0; // if counter = max, overwrite first entry in list, reset counter
2020-09-21 18:26:49 +00:00
receivedID[receivedCount] = thePacket.id; // add ID to received list
#ifndef NOBLINK
pixels.setPixelColor( 0, RGB_RED ); // send mode
pixels.show();
#endif
2020-09-21 18:26:49 +00:00
Radio.Send( payload, size );
2020-09-18 19:14:51 +00:00
}
2020-09-14 14:25:07 +00:00
else{
#ifndef SILENT
2020-09-21 18:26:49 +00:00
Serial.println("PacketID known, will not repeat again.");
2020-09-14 14:25:07 +00:00
#endif
Radio.Rx( 0 ); // switch to Receive Mode
2020-09-14 14:25:07 +00:00
}
}
2020-09-18 19:14:51 +00:00
// hash a string into an integer - djb2 by Dan Bernstein. -
// http://www.cse.yorku.ca/~oz/hash.html
unsigned long hash(char *str)
{
unsigned long hash = 5381;
int c;
while ((c = *str++) != 0)
2020-09-21 18:26:49 +00:00
hash = ((hash << 5) + hash) + (unsigned char) c; // hash * 33 + c //
return hash;
}
2020-09-18 19:14:51 +00:00
void ConfigureRadio( ChannelSettings ChanSet )
{
uint32_t freq = (CH0 + CH_SPACING * ChanSet.channel_num)*1E6;
#ifndef SILENT
Serial.printf("\nSetting frequency to %i Hz (meshtastic channel %i) .. \n",freq,ChanSet.channel_num );
2020-09-21 19:56:31 +00:00
Serial.printf("Channel name is %s .. \n", ChanSet.name );
Serial.printf("Setting bandwidth to index %i ..\n",ChanSet.bandwidth);
Serial.printf("Setting CodeRate to index %i .. \n", ChanSet.coding_rate);
Serial.printf("Setting SpreadingFactor to %i ..\n",ChanSet.spread_factor);
#endif
2020-09-18 19:14:51 +00:00
Radio.SetChannel( freq );
Radio.SetTxConfig( MODEM_LORA, TX_OUTPUT_POWER, 0, ChanSet.bandwidth, ChanSet.spread_factor, ChanSet.coding_rate,
LORA_PREAMBLE_LENGTH, false, true, false, 0, false, 20000 );
Radio.SetRxConfig( MODEM_LORA, ChanSet.bandwidth, ChanSet.spread_factor, ChanSet.coding_rate, 0, LORA_PREAMBLE_LENGTH,
LORA_SYMBOL_TIMEOUT, false , 0, true, false, 0, false, true );
}