2020-05-01 04:42:11 +00:00
|
|
|
|
|
|
|
#include "RadioInterface.h"
|
2021-02-17 05:06:23 +00:00
|
|
|
#include "Channels.h"
|
2020-05-09 23:32:26 +00:00
|
|
|
#include "MeshRadio.h"
|
|
|
|
#include "MeshService.h"
|
2020-04-01 04:56:35 +00:00
|
|
|
#include "NodeDB.h"
|
2021-03-06 03:13:33 +00:00
|
|
|
#include "Router.h"
|
2021-03-30 15:34:13 +00:00
|
|
|
#include "assert.h"
|
|
|
|
#include "configuration.h"
|
2020-05-09 23:32:26 +00:00
|
|
|
#include "sleep.h"
|
2020-04-17 16:48:54 +00:00
|
|
|
#include <assert.h>
|
2020-04-01 04:56:35 +00:00
|
|
|
#include <pb_decode.h>
|
|
|
|
#include <pb_encode.h>
|
|
|
|
|
2020-09-25 19:52:08 +00:00
|
|
|
#define RDEF(name, freq, spacing, num_ch, power_limit) \
|
|
|
|
{ \
|
|
|
|
RegionCode_##name, num_ch, power_limit, freq, spacing, #name \
|
|
|
|
}
|
|
|
|
|
|
|
|
const RegionInfo regions[] = {
|
|
|
|
RDEF(US, 903.08f, 2.16f, 13, 0), RDEF(EU433, 433.175f, 0.2f, 8, 0), RDEF(EU865, 865.2f, 0.3f, 10, 0),
|
|
|
|
RDEF(CN, 470.0f, 2.0f, 20, 0),
|
|
|
|
RDEF(JP, 920.0f, 0.5f, 10, 13), // See https://github.com/meshtastic/Meshtastic-device/issues/346 power level 13
|
|
|
|
RDEF(ANZ, 916.0f, 0.5f, 20, 0), // AU/NZ channel settings 915-928MHz
|
|
|
|
RDEF(KR, 921.9f, 0.2f, 8, 0), // KR channel settings (KR920-923) Start from TTN download channel
|
|
|
|
// freq. (921.9f is for download, others are for uplink)
|
|
|
|
RDEF(TW, 923.0f, 0.2f, 10, 0), // TW channel settings (AS2 bandplan 923-925MHz)
|
2021-03-06 13:10:36 +00:00
|
|
|
RDEF(RU, 868.9f, 0.2f, 2, 20), // See notes below
|
2020-09-25 19:52:08 +00:00
|
|
|
RDEF(Unset, 903.08f, 2.16f, 13, 0) // Assume US freqs if unset, Must be last
|
|
|
|
};
|
|
|
|
|
2021-03-06 13:10:36 +00:00
|
|
|
/* Notes about the RU bandplan (from @denis-d in https://meshtastic.discourse.group/t/russian-band-plan-proposal/2786/2):
|
|
|
|
|
2021-03-30 15:34:13 +00:00
|
|
|
According to Annex 12 to GKRCh (National Radio Frequency Commission) decision № 18-46-03-1 (September 11th 2018)
|
|
|
|
https://digital.gov.ru/uploaded/files/prilozhenie-12-k-reshenyu-gkrch-18-46-03-1.pdf 1 We have 3 options for 868 MHz:
|
2021-03-06 13:10:36 +00:00
|
|
|
|
|
|
|
864,0 - 865,0 MHz ERP 25mW, Duty Cycle 0.1% (3.6 sec in hour) or LBT (Listen Before Talk), prohibited in airports.
|
|
|
|
866,0 - 868,0 MHz ERP 25mW, Duty Cycle 1% or LBT, PSD (Power Spectrum Density) 1000mW/MHz, prohibited in airports
|
|
|
|
868,7 - 869,2 MHz ERP 100mW, Duty Cycle 10% or LBT, no resctrictions
|
|
|
|
and according to RP2-1.0.2 LoRaWAN® Regional Parameters RP2-1.0.2 LoRaWAN® Regional Parameters - LoRa Alliance®
|
|
|
|
I propose to use following meshtastic bandplan:
|
|
|
|
1 channel - central frequency 868.9MHz 125kHz band
|
|
|
|
Protective band - 75kHz
|
|
|
|
2 channel - central frequency 869.1MHz 125kHz band
|
|
|
|
|
|
|
|
RDEF(RU, 868.9f, 0.2f, 2, 20)
|
|
|
|
*/
|
|
|
|
|
2020-10-21 09:27:13 +00:00
|
|
|
const RegionInfo *myRegion;
|
|
|
|
|
|
|
|
void initRegion()
|
|
|
|
{
|
2020-10-29 05:26:36 +00:00
|
|
|
const RegionInfo *r = regions;
|
|
|
|
for (; r->code != RegionCode_Unset && r->code != radioConfig.preferences.region; r++)
|
|
|
|
;
|
|
|
|
myRegion = r;
|
|
|
|
DEBUG_MSG("Wanted region %d, using %s\n", radioConfig.preferences.region, r->name);
|
|
|
|
|
2021-02-16 07:41:52 +00:00
|
|
|
myNodeInfo.num_bands = myRegion->numChannels; // Tell our android app how many channels we have
|
2020-10-21 09:27:13 +00:00
|
|
|
}
|
2020-09-25 19:52:08 +00:00
|
|
|
|
2020-05-09 23:32:26 +00:00
|
|
|
/**
|
|
|
|
* ## LoRaWAN for North America
|
|
|
|
|
|
|
|
LoRaWAN defines 64, 125 kHz channels from 902.3 to 914.9 MHz increments.
|
|
|
|
|
|
|
|
The maximum output power for North America is +30 dBM.
|
|
|
|
|
|
|
|
The band is from 902 to 928 MHz. It mentions channel number and its respective channel frequency. All the 13 channels are
|
|
|
|
separated by 2.16 MHz with respect to the adjacent channels. Channel zero starts at 903.08 MHz center frequency.
|
|
|
|
*/
|
|
|
|
|
2020-05-01 19:31:36 +00:00
|
|
|
// 1kb was too small
|
2020-05-01 19:11:04 +00:00
|
|
|
#define RADIO_STACK_SIZE 4096
|
|
|
|
|
2020-11-12 09:49:04 +00:00
|
|
|
/**
|
2020-11-14 02:07:25 +00:00
|
|
|
* Calculate airtime per
|
|
|
|
* https://www.rs-online.com/designspark/rel-assets/ds-assets/uploads/knowledge-items/application-notes-for-the-internet-of-things/LoRa%20Design%20Guide.pdf
|
2020-11-12 09:49:04 +00:00
|
|
|
* section 4
|
2020-11-14 02:07:25 +00:00
|
|
|
*
|
2020-11-12 09:49:04 +00:00
|
|
|
* @return num msecs for the packet
|
|
|
|
*/
|
2020-11-14 02:07:25 +00:00
|
|
|
uint32_t RadioInterface::getPacketTime(uint32_t pl)
|
2020-11-12 09:49:04 +00:00
|
|
|
{
|
2020-11-14 02:07:25 +00:00
|
|
|
float bandwidthHz = bw * 1000.0f;
|
2020-11-12 09:49:04 +00:00
|
|
|
bool headDisable = false; // we currently always use the header
|
|
|
|
float tSym = (1 << sf) / bandwidthHz;
|
2020-11-14 02:07:25 +00:00
|
|
|
|
|
|
|
bool lowDataOptEn = tSym > 16e-3 ? true : false; // Needed if symbol time is >16ms
|
|
|
|
|
|
|
|
float tPreamble = (preambleLength + 4.25f) * tSym;
|
2020-11-12 09:49:04 +00:00
|
|
|
float numPayloadSym =
|
2020-11-14 02:07:25 +00:00
|
|
|
8 + max(ceilf(((8.0f * pl - 4 * sf + 28 + 16 - 20 * headDisable) / (4 * (sf - 2 * lowDataOptEn))) * cr), 0.0f);
|
2020-11-12 09:49:04 +00:00
|
|
|
float tPayload = numPayloadSym * tSym;
|
|
|
|
float tPacket = tPreamble + tPayload;
|
|
|
|
|
2020-11-14 02:07:25 +00:00
|
|
|
uint32_t msecs = tPacket * 1000;
|
|
|
|
|
|
|
|
DEBUG_MSG("(bw=%d, sf=%d, cr=4/%d) packet symLen=%d ms, payloadSize=%u, time %d ms\n", (int)bw, sf, cr, (int)(tSym * 1000),
|
|
|
|
pl, msecs);
|
2020-11-12 09:49:04 +00:00
|
|
|
return msecs;
|
|
|
|
}
|
|
|
|
|
2020-11-14 02:07:25 +00:00
|
|
|
uint32_t RadioInterface::getPacketTime(MeshPacket *p)
|
|
|
|
{
|
2021-02-10 08:18:41 +00:00
|
|
|
assert(p->which_payloadVariant == MeshPacket_encrypted_tag); // It should have already been encoded by now
|
2020-11-14 02:07:25 +00:00
|
|
|
uint32_t pl = p->encrypted.size + sizeof(PacketHeader);
|
|
|
|
|
|
|
|
return getPacketTime(pl);
|
|
|
|
}
|
|
|
|
|
2020-11-12 09:49:04 +00:00
|
|
|
/** The delay to use for retransmitting dropped packets */
|
|
|
|
uint32_t RadioInterface::getRetransmissionMsec(const MeshPacket *p)
|
|
|
|
{
|
2021-03-30 15:39:51 +00:00
|
|
|
assert(shortPacketMsec); // Better be non zero
|
|
|
|
|
2020-11-14 02:07:25 +00:00
|
|
|
// was 20 and 22 secs respectively, but now with shortPacketMsec as 2269, this should give the same range
|
|
|
|
return random(9 * shortPacketMsec, 10 * shortPacketMsec);
|
2020-11-12 09:49:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/** The delay to use when we want to send something but the ether is busy */
|
|
|
|
uint32_t RadioInterface::getTxDelayMsec()
|
|
|
|
{
|
2020-11-14 02:07:25 +00:00
|
|
|
/** At the low end we want to pick a delay large enough that anyone who just completed sending (some other node)
|
|
|
|
* has had enough time to switch their radio back into receive mode.
|
|
|
|
*/
|
|
|
|
const uint32_t MIN_TX_WAIT_MSEC = 100;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* At the high end, this value is used to spread node attempts across time so when they are replying to a packet
|
|
|
|
* they don't both check that the airwaves are clear at the same moment. As long as they are off by some amount
|
|
|
|
* one of the two will be first to start transmitting and the other will see that. I bet 500ms is more than enough
|
|
|
|
* to guarantee this.
|
|
|
|
*/
|
|
|
|
// const uint32_t MAX_TX_WAIT_MSEC = 2000; // stress test would still fail occasionally with 1000
|
|
|
|
|
|
|
|
return random(MIN_TX_WAIT_MSEC, shortPacketMsec);
|
2020-11-12 09:49:04 +00:00
|
|
|
}
|
|
|
|
|
2020-06-14 22:30:42 +00:00
|
|
|
void printPacket(const char *prefix, const MeshPacket *p)
|
|
|
|
{
|
2021-03-06 03:13:33 +00:00
|
|
|
DEBUG_MSG("%s (id=0x%08x Fr0x%02x To0x%02x, WantAck%d, HopLim%d Ch0x%x", prefix, p->id, p->from & 0xff, p->to & 0xff,
|
|
|
|
p->want_ack, p->hop_limit, p->channel);
|
2021-02-10 08:18:41 +00:00
|
|
|
if (p->which_payloadVariant == MeshPacket_decoded_tag) {
|
2020-06-14 22:30:42 +00:00
|
|
|
auto &s = p->decoded;
|
2021-03-06 03:13:33 +00:00
|
|
|
|
2021-02-17 05:06:23 +00:00
|
|
|
DEBUG_MSG(" Portnum=%d", s.portnum);
|
|
|
|
|
2020-06-14 22:30:42 +00:00
|
|
|
if (s.want_response)
|
|
|
|
DEBUG_MSG(" WANTRESP");
|
|
|
|
|
|
|
|
if (s.source != 0)
|
|
|
|
DEBUG_MSG(" source=%08x", s.source);
|
|
|
|
|
|
|
|
if (s.dest != 0)
|
|
|
|
DEBUG_MSG(" dest=%08x", s.dest);
|
|
|
|
|
2021-03-30 15:34:13 +00:00
|
|
|
if (s.request_id)
|
2021-03-12 06:10:36 +00:00
|
|
|
DEBUG_MSG(" requestId=%0x", s.request_id);
|
|
|
|
|
2021-02-17 05:06:23 +00:00
|
|
|
/* now inside Data and therefore kinda opaque
|
2021-02-10 08:18:41 +00:00
|
|
|
if (s.which_ackVariant == SubPacket_success_id_tag)
|
|
|
|
DEBUG_MSG(" successId=%08x", s.ackVariant.success_id);
|
|
|
|
else if (s.which_ackVariant == SubPacket_fail_id_tag)
|
2021-02-17 05:06:23 +00:00
|
|
|
DEBUG_MSG(" failId=%08x", s.ackVariant.fail_id); */
|
2020-06-14 22:30:42 +00:00
|
|
|
} else {
|
|
|
|
DEBUG_MSG(" encrypted");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p->rx_time != 0) {
|
|
|
|
DEBUG_MSG(" rxtime=%u", p->rx_time);
|
|
|
|
}
|
2020-08-25 19:06:36 +00:00
|
|
|
if (p->rx_snr != 0.0) {
|
|
|
|
DEBUG_MSG(" rxSNR=%g", p->rx_snr);
|
|
|
|
}
|
2021-02-17 05:06:23 +00:00
|
|
|
if (p->priority != 0)
|
2021-02-11 11:00:17 +00:00
|
|
|
DEBUG_MSG(" priority=%d", p->priority);
|
2021-02-17 05:06:23 +00:00
|
|
|
|
2020-06-14 22:30:42 +00:00
|
|
|
DEBUG_MSG(")\n");
|
|
|
|
}
|
|
|
|
|
2020-10-21 09:27:13 +00:00
|
|
|
RadioInterface::RadioInterface()
|
2020-04-30 16:44:16 +00:00
|
|
|
{
|
2021-02-22 04:57:26 +00:00
|
|
|
assert(sizeof(PacketHeader) == 16); // make sure the compiler did what we expected
|
2020-05-09 23:32:26 +00:00
|
|
|
|
|
|
|
// Can't print strings this early - serial not setup yet
|
|
|
|
// DEBUG_MSG("Set meshradio defaults name=%s\n", channelSettings.name);
|
2020-04-30 16:44:16 +00:00
|
|
|
}
|
2020-04-01 04:56:35 +00:00
|
|
|
|
2021-03-30 15:34:13 +00:00
|
|
|
bool RadioInterface::reconfigure()
|
|
|
|
{
|
|
|
|
applyModemConfig();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:11:04 +00:00
|
|
|
bool RadioInterface::init()
|
|
|
|
{
|
2020-05-09 23:32:26 +00:00
|
|
|
DEBUG_MSG("Starting meshradio init...\n");
|
|
|
|
|
|
|
|
configChangedObserver.observe(&service.configChanged);
|
|
|
|
preflightSleepObserver.observe(&preflightSleep);
|
|
|
|
notifyDeepSleepObserver.observe(¬ifyDeepSleep);
|
|
|
|
|
|
|
|
// we now expect interfaces to operate in promiscous mode
|
|
|
|
// radioIf.setThisAddress(nodeDB.getNodeNum()); // Note: we must do this here, because the nodenum isn't inited at constructor
|
|
|
|
// time.
|
2020-10-21 09:27:13 +00:00
|
|
|
|
2021-03-30 15:34:13 +00:00
|
|
|
applyModemConfig();
|
|
|
|
|
2020-05-01 19:11:04 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-10-30 09:05:32 +00:00
|
|
|
int RadioInterface::notifyDeepSleepCb(void *unused)
|
|
|
|
{
|
|
|
|
sleep();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-05-09 23:32:26 +00:00
|
|
|
/** hash a string into an integer
|
|
|
|
*
|
|
|
|
* djb2 by Dan Bernstein.
|
|
|
|
* http://www.cse.yorku.ca/~oz/hash.html
|
|
|
|
*/
|
2020-09-16 04:05:57 +00:00
|
|
|
unsigned long hash(const char *str)
|
2020-05-09 23:32:26 +00:00
|
|
|
{
|
|
|
|
unsigned long hash = 5381;
|
|
|
|
int c;
|
|
|
|
|
|
|
|
while ((c = *str++) != 0)
|
|
|
|
hash = ((hash << 5) + hash) + (unsigned char)c; /* hash * 33 + c */
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
2021-02-10 05:59:00 +00:00
|
|
|
/**
|
|
|
|
* Save our frequency for later reuse.
|
|
|
|
*/
|
|
|
|
void RadioInterface::saveFreq(float freq)
|
|
|
|
{
|
|
|
|
savedFreq = freq;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save our channel for later reuse.
|
|
|
|
*/
|
|
|
|
void RadioInterface::saveChannelNum(uint32_t channel_num)
|
|
|
|
{
|
|
|
|
savedChannelNum = channel_num;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save our frequency for later reuse.
|
|
|
|
*/
|
|
|
|
float RadioInterface::getFreq()
|
|
|
|
{
|
|
|
|
return savedFreq;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save our channel for later reuse.
|
|
|
|
*/
|
|
|
|
uint32_t RadioInterface::getChannelNum()
|
|
|
|
{
|
|
|
|
return savedChannelNum;
|
|
|
|
}
|
|
|
|
|
2020-05-09 23:32:26 +00:00
|
|
|
/**
|
|
|
|
* Pull our channel settings etc... from protobufs to the dumb interface settings
|
|
|
|
*/
|
|
|
|
void RadioInterface::applyModemConfig()
|
|
|
|
{
|
|
|
|
// Set up default configuration
|
2020-08-12 17:42:25 +00:00
|
|
|
// No Sync Words in LORA mode
|
|
|
|
|
2021-02-16 07:41:52 +00:00
|
|
|
auto channelSettings = channels.getPrimary();
|
2020-11-14 02:07:25 +00:00
|
|
|
if (channelSettings.spread_factor == 0) {
|
|
|
|
switch (channelSettings.modem_config) {
|
|
|
|
case ChannelSettings_ModemConfig_Bw125Cr45Sf128: ///< Bw = 125 kHz, Cr = 4/5, Sf = 128chips/symbol, CRC on. Default medium
|
|
|
|
///< range
|
|
|
|
bw = 125;
|
|
|
|
cr = 5;
|
|
|
|
sf = 7;
|
|
|
|
break;
|
|
|
|
case ChannelSettings_ModemConfig_Bw500Cr45Sf128: ///< Bw = 500 kHz, Cr = 4/5, Sf = 128chips/symbol, CRC on. Fast+short
|
|
|
|
///< range
|
|
|
|
bw = 500;
|
|
|
|
cr = 5;
|
|
|
|
sf = 7;
|
|
|
|
break;
|
|
|
|
case ChannelSettings_ModemConfig_Bw31_25Cr48Sf512: ///< Bw = 31.25 kHz, Cr = 4/8, Sf = 512chips/symbol, CRC on. Slow+long
|
|
|
|
///< range
|
|
|
|
bw = 31.25;
|
|
|
|
cr = 8;
|
|
|
|
sf = 9;
|
|
|
|
break;
|
|
|
|
case ChannelSettings_ModemConfig_Bw125Cr48Sf4096:
|
|
|
|
bw = 125;
|
|
|
|
cr = 8;
|
|
|
|
sf = 12;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(0); // Unknown enum
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
sf = channelSettings.spread_factor;
|
|
|
|
cr = channelSettings.coding_rate;
|
|
|
|
bw = channelSettings.bandwidth;
|
|
|
|
|
|
|
|
if (bw == 31) // This parameter is not an integer
|
|
|
|
bw = 31.25;
|
|
|
|
}
|
|
|
|
|
2020-08-12 17:42:25 +00:00
|
|
|
power = channelSettings.tx_power;
|
2020-05-09 23:32:26 +00:00
|
|
|
|
2020-11-14 02:07:25 +00:00
|
|
|
shortPacketMsec = getPacketTime(sizeof(PacketHeader));
|
|
|
|
|
2020-09-25 19:52:08 +00:00
|
|
|
assert(myRegion); // Should have been found in init
|
|
|
|
|
2020-08-12 17:46:44 +00:00
|
|
|
// If user has manually specified a channel num, then use that, otherwise generate one by hashing the name
|
2021-02-16 07:41:52 +00:00
|
|
|
const char *channelName = channels.getName(channels.getPrimaryIndex());
|
|
|
|
int channel_num = channelSettings.channel_num ? channelSettings.channel_num - 1 : hash(channelName) % myRegion->numChannels;
|
2021-04-02 01:14:12 +00:00
|
|
|
freq = myRegion->freq + radioConfig.preferences.frequency_offset + myRegion->spacing * channel_num;
|
2020-05-09 23:32:26 +00:00
|
|
|
|
2021-02-10 05:59:00 +00:00
|
|
|
DEBUG_MSG("Set radio: name=%s, config=%u, ch=%d, power=%d\n", channelName, channelSettings.modem_config, channel_num, power);
|
2020-09-29 07:59:26 +00:00
|
|
|
DEBUG_MSG("Radio myRegion->freq: %f\n", myRegion->freq);
|
|
|
|
DEBUG_MSG("Radio myRegion->spacing: %f\n", myRegion->spacing);
|
|
|
|
DEBUG_MSG("Radio myRegion->numChannels: %d\n", myRegion->numChannels);
|
|
|
|
DEBUG_MSG("Radio channel_num: %d\n", channel_num);
|
|
|
|
DEBUG_MSG("Radio frequency: %f\n", freq);
|
2020-11-14 02:07:25 +00:00
|
|
|
DEBUG_MSG("Short packet time: %u msec\n", shortPacketMsec);
|
2021-02-10 05:59:00 +00:00
|
|
|
|
|
|
|
saveChannelNum(channel_num);
|
|
|
|
saveFreq(freq);
|
2020-05-09 23:32:26 +00:00
|
|
|
}
|
|
|
|
|
2020-09-16 01:54:50 +00:00
|
|
|
/**
|
|
|
|
* Some regulatory regions limit xmit power.
|
|
|
|
* This function should be called by subclasses after setting their desired power. It might lower it
|
|
|
|
*/
|
|
|
|
void RadioInterface::limitPower()
|
|
|
|
{
|
|
|
|
uint8_t maxPower = 255; // No limit
|
|
|
|
|
2020-09-25 19:52:08 +00:00
|
|
|
if (myRegion->powerLimit)
|
|
|
|
maxPower = myRegion->powerLimit;
|
|
|
|
|
2020-09-16 01:54:50 +00:00
|
|
|
if (power > maxPower) {
|
|
|
|
DEBUG_MSG("Lowering transmit power because of regulatory limits\n");
|
|
|
|
power = maxPower;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_MSG("Set radio: final power level=%d\n", power);
|
|
|
|
}
|
|
|
|
|
2020-04-01 04:56:35 +00:00
|
|
|
ErrorCode SimRadio::send(MeshPacket *p)
|
|
|
|
{
|
|
|
|
DEBUG_MSG("SimRadio.send\n");
|
2020-04-17 16:48:54 +00:00
|
|
|
packetPool.release(p);
|
2020-04-01 04:56:35 +00:00
|
|
|
return ERRNO_OK;
|
2020-04-17 16:48:54 +00:00
|
|
|
}
|
|
|
|
|
2020-04-18 21:22:24 +00:00
|
|
|
void RadioInterface::deliverToReceiver(MeshPacket *p)
|
2020-04-17 16:48:54 +00:00
|
|
|
{
|
|
|
|
assert(rxDest);
|
2020-04-18 21:22:24 +00:00
|
|
|
assert(rxDest->enqueue(p, 0)); // NOWAIT - fixme, if queue is full, delete older messages
|
2021-03-06 03:13:33 +00:00
|
|
|
|
|
|
|
// Nasty hack because our threading is primitive. interfaces shouldn't need to know about routers FIXME
|
|
|
|
if (router)
|
|
|
|
router->setReceivedMessage();
|
2020-04-30 01:46:32 +00:00
|
|
|
}
|
|
|
|
|
2020-04-30 16:44:16 +00:00
|
|
|
/***
|
|
|
|
* given a packet set sendingPacket and decode the protobufs into radiobuf. Returns # of payload bytes to send
|
|
|
|
*/
|
|
|
|
size_t RadioInterface::beginSending(MeshPacket *p)
|
|
|
|
{
|
|
|
|
assert(!sendingPacket);
|
|
|
|
|
|
|
|
// DEBUG_MSG("sending queued packet on mesh (txGood=%d,rxGood=%d,rxBad=%d)\n", rf95.txGood(), rf95.rxGood(), rf95.rxBad());
|
2021-02-10 08:18:41 +00:00
|
|
|
assert(p->which_payloadVariant == MeshPacket_encrypted_tag); // It should have already been encoded by now
|
2020-04-30 16:44:16 +00:00
|
|
|
|
2020-09-05 19:34:48 +00:00
|
|
|
lastTxStart = millis();
|
2020-04-30 16:44:16 +00:00
|
|
|
|
|
|
|
PacketHeader *h = (PacketHeader *)radiobuf;
|
|
|
|
|
|
|
|
h->from = p->from;
|
|
|
|
h->to = p->to;
|
|
|
|
h->id = p->id;
|
2021-02-22 04:57:26 +00:00
|
|
|
h->channel = p->channel;
|
2020-05-19 00:35:23 +00:00
|
|
|
assert(p->hop_limit <= HOP_MAX);
|
2020-05-19 17:27:28 +00:00
|
|
|
h->flags = p->hop_limit | (p->want_ack ? PACKET_FLAGS_WANT_ACK_MASK : 0);
|
2020-04-30 16:44:16 +00:00
|
|
|
|
|
|
|
// if the sender nodenum is zero, that means uninitialized
|
|
|
|
assert(h->from);
|
|
|
|
|
2020-05-10 00:51:20 +00:00
|
|
|
memcpy(radiobuf + sizeof(PacketHeader), p->encrypted.bytes, p->encrypted.size);
|
2020-04-30 16:44:16 +00:00
|
|
|
|
|
|
|
sendingPacket = p;
|
2020-05-10 00:51:20 +00:00
|
|
|
return p->encrypted.size + sizeof(PacketHeader);
|
2020-04-30 16:44:16 +00:00
|
|
|
}
|