Bleeding edge development version of the xDot library for mbed 5. This version of the library is not guaranteed to be stable or well tested and should not be used in production or deployment scenarios.

Dependents:   Dot-Examples Dot-AT-Firmware Dot-Examples TEST_FF1705 ... more

The Dot library provides a LoRaWan certified stack for LoRa communication using MultiTech mDot and xDot devices. The stack is compatible with mbed 5.

Dot Library Version 3 Updates

Dot Library versions 3.x.x require a channel plan to be injected into the stack. Channel plans are included with the 3.x.x Dot Library releases. The following code snippet demonstrates how to create a channel plan and inject it into the stack.

#include "mDot.h"
#include "channel_plans.h"

int main() {
    ChannelPlan* plan = new lora::ChannelPlan_US915();
    assert(plan);
    mDot* dot = mDot::getInstance(plan);
    assert(dot);

    // ...
}

Dot devices must not be deployed with software using a different channel plan than the Dot's default plan! This functionality is for development and testing only!

Multicast Sessions

Multicast sessions and packet rx events in library. When in Class C mode Multicast downlinks can be received. Recieved packets should be filtered on address, counter value will be maintained in the session or can be set explicitly depending on Application support to share Multicast Address, Keys and Counters.

mDot.h

        /**
         * Add a multicast session address and keys
         * Downlink counter is set to 0
         * Up to 3 MULTICAST_SESSIONS can be set
         */
        int32_t setMulticastSession(uint8_t index, uint32_t addr, const uint8_t* nsk, const uint8_t* dsk);
 
        /**
         * Set a multicast session counter
         * Up to 3 MULTICAST_SESSIONS can be set
         */
        int32_t setMulticastDownlinkCounter(uint8_t index, uint32_t count);

mDotEvent.h

The address field was added to PacketRx event.

        virtual void PacketRx(uint8_t port, uint8_t *payload, uint16_t size, int16_t rssi, int8_t snr, lora::DownlinkControl ctrl, uint8_t slot, uint8_t retries, uint32_t address);

The name of the repository can be used to determine which device the stack was compiled for and if it's a development or production-ready build:

A changelog for the Dot library can be found here.

The Dot library version and the version of mbed-os it was compiled against can both be found in the commit message for that revision of the Dot library. Building your application with the same version of mbed-os as what was used to build the Dot library is highly recommended!

The Dot-Examples repository demonstrates how to use the Dot library in a custom application.

The mDot and xDot platform pages have lots of platform specific information and document potential issues, gotchas, etc, and provide instructions for getting started with development. Please take a look at the platform page before starting development as they should answer many questions you will have.

Committer:
Jenkins@KEILDM1.dc.multitech.prv
Date:
Wed Feb 28 10:31:10 2018 -0600
Revision:
86:8e7bfa8973b4
Parent:
81:2b275bd4c948
Child:
98:0fa5451750c3
xdot-library revision 3.1.0-rc1 and mbed-os revision mbed-os-5.7.5

Who changed what in which revision?

UserRevisionLine numberNew contents of line
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 1 /**********************************************************************
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 2 * COPYRIGHT 2016 MULTI-TECH SYSTEMS, INC.
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 3 *
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 4 * ALL RIGHTS RESERVED BY AND FOR THE EXCLUSIVE BENEFIT OF
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 5 * MULTI-TECH SYSTEMS, INC.
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 6 *
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 7 * MULTI-TECH SYSTEMS, INC. - CONFIDENTIAL AND PROPRIETARY
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 8 * INFORMATION AND/OR TRADE SECRET.
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 9 *
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 10 * NOTICE: ALL CODE, PROGRAM, INFORMATION, SCRIPT, INSTRUCTION,
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 11 * DATA, AND COMMENT HEREIN IS AND SHALL REMAIN THE CONFIDENTIAL
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 12 * INFORMATION AND PROPERTY OF MULTI-TECH SYSTEMS, INC.
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 13 * USE AND DISCLOSURE THEREOF, EXCEPT AS STRICTLY AUTHORIZED IN A
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 14 * WRITTEN AGREEMENT SIGNED BY MULTI-TECH SYSTEMS, INC. IS PROHIBITED.
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 15 *
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 16 ***********************************************************************/
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 17
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 18 #include "ChannelPlan_US915.h"
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 19 #include "limits.h"
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 20
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 21 using namespace lora;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 22
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 23 const uint8_t ChannelPlan_US915::US915_TX_POWERS[] = { 30, 28, 26, 24, 22, 20, 18, 16, 14, 12, 10 };
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 24 const uint8_t ChannelPlan_US915::US915_RADIO_POWERS[] = { 3, 3, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 18, 19, 19 };
Jenkins@KEILDM1.dc.multitech.prv 62:4a8e7ac95a48 25 const uint8_t ChannelPlan_US915::US915_MAX_PAYLOAD_SIZE[] = { 11, 53, 125, 242, 242, 0, 0, 0, 53, 129, 242, 242, 242, 242, 0, 0 };
Jenkins@KEILDM1.dc.multitech.prv 52:2cfacf18f01a 26 const uint8_t ChannelPlan_US915::US915_MAX_PAYLOAD_SIZE_REPEATER[] = { 11, 53, 126, 222, 222, 0, 0, 0, 33, 109, 222, 222, 222, 222, 0, 0 };
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 27
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 28 ChannelPlan_US915::ChannelPlan_US915()
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 29 :
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 30 ChannelPlan(NULL, NULL)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 31 {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 32
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 33 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 34
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 35 ChannelPlan_US915::ChannelPlan_US915(Settings* settings)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 36 :
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 37 ChannelPlan(NULL, settings)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 38 {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 39
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 40 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 41
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 42 ChannelPlan_US915::ChannelPlan_US915(SxRadio* radio, Settings* settings)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 43 :
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 44 ChannelPlan(radio, settings)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 45 {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 46
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 47 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 48
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 49 ChannelPlan_US915::~ChannelPlan_US915() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 50
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 51 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 52
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 53 void ChannelPlan_US915::Init() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 54 _plan = US915;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 55 _planName = "US915";
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 56
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 57 _datarates.clear();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 58 _channels.clear();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 59 _dutyBands.clear();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 60
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 61 DutyBand band;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 62
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 63 band.Index = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 64 band.DutyCycle = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 65
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 66 Datarate dr;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 67
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 68 _maxTxPower = 30;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 69 _minTxPower = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 70
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 71 _minFrequency = US915_FREQ_MIN;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 72 _maxFrequency = US915_FREQ_MAX;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 73
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 74 TX_POWERS = US915_TX_POWERS;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 75 RADIO_POWERS = US915_RADIO_POWERS;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 76 MAX_PAYLOAD_SIZE = US915_MAX_PAYLOAD_SIZE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 77 MAX_PAYLOAD_SIZE_REPEATER = US915_MAX_PAYLOAD_SIZE_REPEATER;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 78
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 79 band.FrequencyMin = US915_FREQ_MIN;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 80 band.FrequencyMax = US915_FREQ_MAX;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 81
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 82 _freqUBase125k = US915_125K_FREQ_BASE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 83 _freqUStep125k = US915_125K_FREQ_STEP;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 84 _freqUBase500k = US915_500K_FREQ_BASE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 85 _freqUStep500k = US915_500K_FREQ_STEP;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 86 _freqDBase500k = US915_500K_DBASE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 87 _freqDStep500k = US915_500K_DSTEP;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 88 GetSettings()->Session.Rx2Frequency = US915_500K_DBASE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 89
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 90 _minDatarate = US915_MIN_DATARATE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 91 _maxDatarate = US915_MAX_DATARATE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 92 _minRx2Datarate = DR_8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 93 _maxRx2Datarate = DR_13;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 94 _minDatarateOffset = US915_MIN_DATARATE_OFFSET;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 95 _maxDatarateOffset = US915_MAX_DATARATE_OFFSET;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 96
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 97 _numChans125k = US915_125K_NUM_CHANS;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 98 _numChans500k = US915_500K_NUM_CHANS;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 99
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 100 logInfo("Initialize channels...");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 101
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 102 SetNumberOfChannels(US915_125K_NUM_CHANS + US915_500K_NUM_CHANS, false);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 103
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 104 dr.SpreadingFactor = SF_10;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 105
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 106 logInfo("Initialize datarates...");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 107
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 108 // Add DR0-3
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 109 while (dr.SpreadingFactor >= SF_7) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 110 AddDatarate(-1, dr);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 111 dr.SpreadingFactor--;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 112 dr.Index++;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 113 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 114
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 115 // Add DR4
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 116 dr.SpreadingFactor = SF_8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 117 dr.Bandwidth = BW_500;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 118 AddDatarate(-1, dr);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 119 dr.Index++;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 120
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 121 // Skip DR5-7 RFU
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 122 dr.SpreadingFactor = SF_INVALID;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 123 AddDatarate(-1, dr), dr.Index++;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 124 AddDatarate(-1, dr), dr.Index++;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 125 AddDatarate(-1, dr), dr.Index++;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 126
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 127 band.PowerMax = 30;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 128
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 129 band.TimeOffEnd = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 130
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 131 AddDutyBand(-1, band);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 132
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 133 GetSettings()->Session.Rx2DatarateIndex = DR_8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 134
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 135 // Add DR8-13
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 136 dr.SpreadingFactor = SF_12;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 137 while (dr.SpreadingFactor >= SF_7) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 138 AddDatarate(-1, dr);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 139 dr.SpreadingFactor--;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 140 dr.Index++;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 141 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 142
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 143 // Skip DR14-15 RFU
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 144 dr.SpreadingFactor = SF_INVALID;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 145 AddDatarate(-1, dr), AddDatarate(-1, dr);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 146
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 147 GetSettings()->Session.TxDatarate = DR_0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 148 GetSettings()->Session.TxPower = GetSettings()->Network.TxPower;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 149
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 150 SetFrequencySubBand(GetSettings()->Network.FrequencySubBand);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 151
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 152 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 153
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 154 uint8_t ChannelPlan_US915::HandleJoinAccept(const uint8_t* buffer, uint8_t size) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 155
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 156 if (size > 17) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 157 // TODO: Handle future channel mask settings
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 158 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 159
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 160 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 161 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 162
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 163 void ChannelPlan_US915::SetNumberOfChannels(uint8_t channels, bool resize) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 164 uint8_t newsize = ((channels - 1) / CHAN_MASK_SIZE) + 1;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 165
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 166 if (resize) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 167 _channels.resize(channels);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 168 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 169
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 170 _channelMask.resize(newsize, 0x0);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 171 _numChans = channels;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 172
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 173 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 174
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 175 bool ChannelPlan_US915::IsChannelEnabled(uint8_t channel) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 176 uint8_t index = channel / CHAN_MASK_SIZE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 177 uint8_t shift = channel % CHAN_MASK_SIZE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 178
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 179 assert(index < _channelMask.size() * CHAN_MASK_SIZE);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 180
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 181 // cannot shift over 32 bits
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 182 assert(shift < 32);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 183
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 184 // logDebug("index: %d shift %d cm: %04x bit: %04x enabled: %d", index, shift, _channelMask[index], (1 << shift), (_channelMask[index] & (1 << shift)) == (1 << shift));
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 185
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 186 return (_channelMask[index] & (1 << shift)) == (1 << shift);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 187 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 188
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 189 uint8_t ChannelPlan_US915::GetMinDatarate() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 190 if (GetSettings()->Network.Mode == lora::PEER_TO_PEER)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 191 return 8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 192 else
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 193 return _minDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 194 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 195
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 196 uint8_t ChannelPlan_US915::GetMaxDatarate() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 197 if (GetSettings()->Network.Mode == lora::PEER_TO_PEER)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 198 return 13;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 199 else
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 200 return _maxDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 201 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 202
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 203 uint8_t ChannelPlan_US915::SetRx1Offset(uint8_t offset) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 204 GetSettings()->Session.Rx1DatarateOffset = offset;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 205 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 206 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 207
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 208 uint8_t ChannelPlan_US915::SetRx2Frequency(uint32_t freq) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 209 GetSettings()->Session.Rx2Frequency = freq;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 210 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 211 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 212
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 213 uint8_t ChannelPlan_US915::SetRx2DatarateIndex(uint8_t index) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 214 GetSettings()->Session.Rx2DatarateIndex = index;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 215 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 216 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 217
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 218 uint8_t ChannelPlan_US915::SetTxConfig() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 219 logInfo("Configure radio for TX");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 220
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 221 uint8_t band = GetDutyBand(GetChannel(_txChannel).Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 222 Datarate txDr = GetDatarate(GetSettings()->Session.TxDatarate);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 223 int8_t max_pwr = _dutyBands[band].PowerMax;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 224 uint8_t chans_enabled = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 225
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 226 int8_t pwr = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 227
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 228 pwr = std::min < int8_t > (GetSettings()->Session.TxPower, max_pwr);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 229
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 230 // spec states that if < 50 125kHz channels are enabled, power is limited to 21dB conducted
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 231 chans_enabled += CountBits(_channelMask[0]);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 232 chans_enabled += CountBits(_channelMask[1]);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 233 chans_enabled += CountBits(_channelMask[2]);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 234 chans_enabled += CountBits(_channelMask[3]);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 235 if (chans_enabled < 50 && pwr > 21) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 236 pwr = 21;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 237 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 238
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 239 if (pwr + GetSettings()->Network.AntennaGain >= max_pwr + 6 && GetSettings()->Network.AntennaGain > 6) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 240 pwr -= (GetSettings()->Network.AntennaGain - 6);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 241 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 242
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 243 for (int i = 20; i >= 0; i--) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 244 if (RADIO_POWERS[i] <= pwr) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 245 pwr = i;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 246 break;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 247 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 248 if (i == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 249 pwr = i;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 250 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 251 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 252
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 253 logDebug("Session pwr: %d ant: %d max: %d", GetSettings()->Session.TxPower, GetSettings()->Network.AntennaGain, max_pwr);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 254 logDebug("Radio Power index: %d output: %d total: %d", pwr, RADIO_POWERS[pwr], RADIO_POWERS[pwr] + GetSettings()->Network.AntennaGain);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 255
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 256 uint32_t bw = txDr.Bandwidth;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 257 uint32_t sf = txDr.SpreadingFactor;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 258 uint8_t cr = txDr.Coderate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 259 uint8_t pl = txDr.PreambleLength;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 260 uint16_t fdev = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 261 bool crc = txDr.Crc;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 262 bool iq = txDr.TxIQ;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 263
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 264 if (GetSettings()->Network.DisableCRC == true)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 265 crc = false;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 266
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 267 SxRadio::RadioModems_t modem = SxRadio::MODEM_LORA;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 268
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 269 if (sf == SF_FSK) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 270 modem = SxRadio::MODEM_FSK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 271 sf = 50e3;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 272 fdev = 25e3;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 273 bw = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 274 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 275
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 276 GetRadio()->SetTxConfig(modem, pwr, fdev, bw, sf, cr, pl, false, crc, false, 0, iq, 3e3);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 277
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 278 logDebug("TX PWR: %u DR: %u SF: %u BW: %u CR: %u PL: %u CRC: %d IQ: %d", pwr, txDr.Index, sf, bw, cr, pl, crc, iq);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 279
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 280 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 281 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 282
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 283 uint8_t ChannelPlan_US915::SetRxConfig(uint8_t window, bool continuous) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 284
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 285 RxWindow rxw = GetRxWindow(window);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 286 GetRadio()->SetChannel(rxw.Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 287
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 288 Datarate rxDr = GetDatarate(rxw.DatarateIndex);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 289 uint32_t bw = rxDr.Bandwidth;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 290 uint32_t sf = rxDr.SpreadingFactor;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 291 uint8_t cr = rxDr.Coderate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 292 uint8_t pl = rxDr.PreambleLength;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 293 uint16_t sto = rxDr.SymbolTimeout();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 294 uint32_t afc = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 295 bool crc = false; // downlink does not use CRC according to LORAWAN
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 296
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 297 if (GetSettings()->Network.DisableCRC == true)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 298 crc = false;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 299
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 300 Datarate txDr = GetDatarate(GetSettings()->Session.TxDatarate);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 301 bool iq = txDr.RxIQ;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 302
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 303 if (P2PEnabled()) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 304 iq = txDr.TxIQ;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 305 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 306
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 307 SxRadio::RadioModems_t modem = SxRadio::MODEM_LORA;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 308
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 309 if (sf == SF_FSK) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 310 modem = SxRadio::MODEM_FSK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 311 sf = 50e3;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 312 cr = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 313 bw = 50e3;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 314 afc = 83333;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 315 iq = false;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 316 crc = true; // FSK must use CRC
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 317 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 318
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 319 // Disable printf's to actually receive packets, printing to debug may mess up the timing
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 320 // logTrace("Configure radio for RX%d on freq: %lu", window, rxw.Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 321 // logTrace("RX SF: %u BW: %u CR: %u PL: %u STO: %u CRC: %d IQ: %d", sf, bw, cr, pl, sto, crc, iq);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 322
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 323 GetRadio()->SetRxConfig(modem, bw, sf, cr, afc, pl, sto, false, 0, crc, false, 0, iq, continuous);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 324
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 325 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 326 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 327
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 328 uint8_t ChannelPlan_US915::AddChannel(int8_t index, Channel channel) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 329 logTrace("Add Channel %d : %lu : %02x %d", index, channel.Frequency, channel.DrRange.Value, _channels.size());
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 330
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 331 assert(index < (int) _channels.size());
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 332
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 333 if (index >= 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 334 _channels[index] = channel;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 335 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 336 _channels.push_back(channel);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 337 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 338
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 339 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 340 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 341
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 342 Channel ChannelPlan_US915::GetChannel(int8_t index) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 343 Channel chan;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 344 memset(&chan, 0, sizeof(Channel));
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 345
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 346 if (_channels.size() > 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 347 chan = _channels[index];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 348 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 349 if (index < 64) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 350 chan.Index = index;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 351 chan.DrRange.Fields.Min = _minDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 352 chan.DrRange.Fields.Max = _maxDatarate - 1;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 353 chan.Frequency = _freqUBase125k + (_freqUStep125k * index);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 354 } else if (index < 72) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 355 chan.Index = index;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 356 chan.DrRange.Fields.Min = _maxDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 357 chan.DrRange.Fields.Max = _maxDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 358 chan.Frequency = _freqUBase500k + (_freqUStep500k * (index - 64));
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 359 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 360 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 361
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 362 return chan;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 363 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 364
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 365 uint8_t ChannelPlan_US915::SetFrequencySubBand(uint8_t sub_band) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 366
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 367 _txFrequencySubBand = sub_band;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 368
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 369 if (sub_band > 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 370 SetChannelMask(0, 0x0000);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 371 SetChannelMask(1, 0x0000);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 372 SetChannelMask(2, 0x0000);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 373 SetChannelMask(3, 0x0000);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 374 SetChannelMask(4, 0x0000);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 375 SetChannelMask((sub_band - 1) / 2, (sub_band % 2) ? 0x00FF : 0xFF00);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 376 SetChannelMask(4, 1 << (sub_band - 1));
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 377 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 378 SetChannelMask(0, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 379 SetChannelMask(1, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 380 SetChannelMask(2, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 381 SetChannelMask(3, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 382 SetChannelMask(4, 0x00FF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 383 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 384
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 385 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 386 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 387
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 388
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 389 void ChannelPlan_US915::LogRxWindow(uint8_t wnd) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 390
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 391 RxWindow rxw = GetRxWindow(wnd);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 392 Datarate rxDr = GetDatarate(rxw.DatarateIndex);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 393 uint8_t bw = rxDr.Bandwidth;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 394 uint8_t sf = rxDr.SpreadingFactor;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 395 uint8_t cr = rxDr.Coderate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 396 uint8_t pl = rxDr.PreambleLength;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 397 uint16_t sto = rxDr.SymbolTimeout();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 398 bool crc = false; // downlink does not use CRC according to LORAWAN
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 399 bool iq = GetTxDatarate().RxIQ;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 400
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 401 logTrace("RX%d on freq: %lu", wnd, rxw.Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 402 logTrace("RX DR: %u SF: %u BW: %u CR: %u PL: %u STO: %u CRC: %d IQ: %d", rxDr.Index, sf, bw, cr, pl, sto, crc, iq);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 403 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 404
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 405 RxWindow ChannelPlan_US915::GetRxWindow(uint8_t window) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 406 RxWindow rxw;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 407 int index = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 408
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 409 if (P2PEnabled()) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 410 rxw.Frequency = GetSettings()->Network.TxFrequency;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 411 index = GetSettings()->Session.TxDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 412 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 413 if (window == 1) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 414 if (_txChannel < _numChans125k) {
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 415 if (GetSettings()->Network.Mode == lora::PRIVATE_MTS)
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 416 rxw.Frequency = _freqDBase500k + (_txChannel / 8) * _freqDStep500k;
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 417 else
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 418 rxw.Frequency = _freqDBase500k + (_txChannel % 8) * _freqDStep500k;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 419 } else
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 420 rxw.Frequency = _freqDBase500k + (_txChannel - _numChans125k) * _freqDStep500k;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 421
Jenkins@KEILDM1.dc.multitech.prv 56:3eb5210b5039 422 if (GetSettings()->Session.TxDatarate <= DR_4) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 423 index = GetSettings()->Session.TxDatarate + 10 - GetSettings()->Session.Rx1DatarateOffset;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 424
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 425 if (index < DR_8)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 426 index = DR_8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 427 if (index > DR_13)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 428 index = DR_13;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 429 } else if (GetSettings()->Session.TxDatarate >= DR_8) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 430 index = GetSettings()->Session.TxDatarate - GetSettings()->Session.Rx1DatarateOffset;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 431 if (index < DR_8)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 432 index = DR_8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 433 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 434 } else {
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 435 if (GetSettings()->Network.Mode == lora::PRIVATE_MTS)
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 436 if (_txChannel < _numChans125k) {
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 437 rxw.Frequency = _freqDBase500k + (_txChannel / 8) * _freqDStep500k;
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 438 } else {
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 439 rxw.Frequency = _freqDBase500k + (_txChannel % 8) * _freqDStep500k;
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 440 }
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 441 else
Jenkins@KEILDM1.dc.multitech.prv 81:2b275bd4c948 442 rxw.Frequency = GetSettings()->Session.Rx2Frequency;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 443 index = GetSettings()->Session.Rx2DatarateIndex;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 444 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 445 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 446
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 447 rxw.DatarateIndex = index;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 448
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 449 return rxw;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 450 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 451
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 452 uint8_t ChannelPlan_US915::HandleRxParamSetup(const uint8_t* payload, uint8_t index, uint8_t size, uint8_t& status) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 453 status = 0x07;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 454 int8_t datarate = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 455 int8_t drOffset = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 456 uint32_t freq = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 457
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 458 drOffset = payload[index++];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 459 datarate = drOffset & 0x0F;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 460 drOffset = (drOffset >> 4) & 0x07;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 461
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 462 freq = payload[index++];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 463 freq |= payload[index++] << 8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 464 freq |= payload[index++] << 16;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 465 freq *= 100;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 466
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 467 if (!CheckRfFrequency(freq)) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 468 logInfo("Freq KO");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 469 status &= 0xFE; // Channel frequency KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 470 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 471
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 472 if (datarate < _minRx2Datarate || datarate > _maxRx2Datarate) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 473 logInfo("DR KO");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 474 status &= 0xFD; // Datarate KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 475 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 476
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 477 if (drOffset < 0 || drOffset > _maxDatarateOffset) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 478 logInfo("DR Offset KO");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 479 status &= 0xFB; // Rx1DrOffset range KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 480 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 481
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 482 if ((status & 0x07) == 0x07) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 483 logInfo("RxParamSetup accepted Rx2DR: %d Rx2Freq: %d Rx1Offset: %d", datarate, freq, drOffset);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 484 SetRx2DatarateIndex(datarate);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 485 SetRx2Frequency(freq);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 486 SetRx1Offset(drOffset);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 487 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 488 logInfo("RxParamSetup rejected Rx2DR: %d Rx2Freq: %d Rx1Offset: %d", datarate, freq, drOffset);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 489 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 490
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 491 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 492 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 493
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 494 uint8_t ChannelPlan_US915::HandleNewChannel(const uint8_t* payload, uint8_t index, uint8_t size, uint8_t& status) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 495
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 496 // Not Supported in US915
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 497 status = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 498 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 499 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 500
Jenkins@KEILDM1.dc.multitech.prv 86:8e7bfa8973b4 501 uint8_t ChannelPlan_US915::HandleDownlinkChannelReq(const uint8_t* payload, uint8_t index, uint8_t size, uint8_t& status) {
Jenkins@KEILDM1.dc.multitech.prv 86:8e7bfa8973b4 502
Jenkins@KEILDM1.dc.multitech.prv 86:8e7bfa8973b4 503 // Not Supported in US915
Jenkins@KEILDM1.dc.multitech.prv 86:8e7bfa8973b4 504 status = 0;
Jenkins@KEILDM1.dc.multitech.prv 86:8e7bfa8973b4 505 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 86:8e7bfa8973b4 506 }
Jenkins@KEILDM1.dc.multitech.prv 86:8e7bfa8973b4 507
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 508 uint8_t ChannelPlan_US915::HandlePingSlotChannelReq(const uint8_t* payload, uint8_t index, uint8_t size, uint8_t& status) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 509
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 510 lora::CopyFreqtoInt(payload + index, _beaconRxChannel.Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 511 index += 3;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 512
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 513 if (_beaconRxChannel.Frequency != 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 514 _beaconRxChannel.DrRange.Value = payload[index];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 515 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 516 // TODO: set to default beacon rx channel
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 517 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 518
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 519 status = 0x03;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 520 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 521 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 522
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 523 uint8_t ChannelPlan_US915::HandleBeaconFrequencyReq(const uint8_t* payload, uint8_t index, uint8_t size, uint8_t& status) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 524
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 525 status = 0x03;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 526 Channel chParam;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 527
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 528 // Skip channel index
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 529 index++;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 530
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 531 lora::CopyFreqtoInt(payload + index, chParam.Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 532 index += 3;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 533 chParam.DrRange.Value = payload[index++];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 534
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 535 if (!GetRadio()->CheckRfFrequency(chParam.Frequency)) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 536 status &= 0xFE; // Channel frequency KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 537 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 538
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 539 if (chParam.DrRange.Fields.Min < chParam.DrRange.Fields.Max) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 540 status &= 0xFD; // Datarate range KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 541 } else if (chParam.DrRange.Fields.Min < _minDatarate || chParam.DrRange.Fields.Min > _maxDatarate) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 542 status &= 0xFD; // Datarate range KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 543 } else if (chParam.DrRange.Fields.Max < _minDatarate || chParam.DrRange.Fields.Max > _maxDatarate) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 544 status &= 0xFD; // Datarate range KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 545 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 546
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 547 if ((status & 0x03) == 0x03) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 548 _beaconChannel = chParam;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 549 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 550
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 551 if (_beaconChannel.Frequency == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 552 // TODO: Set to default
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 553 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 554
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 555 status = 0x01;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 556
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 557 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 558 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 559
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 560
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 561 uint8_t ChannelPlan_US915::HandleAdrCommand(const uint8_t* payload, uint8_t index, uint8_t size, uint8_t& status) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 562
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 563 uint8_t power = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 564 uint8_t datarate = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 565 uint16_t mask = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 566 uint8_t ctrl = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 567 uint8_t nbRep = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 568
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 569 status = 0x07;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 570 datarate = payload[index++];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 571 power = datarate & 0x0F;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 572 datarate = (datarate >> 4) & 0x0F;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 573
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 574 mask = payload[index++];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 575 mask |= payload[index++] << 8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 576
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 577 nbRep = payload[index++];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 578 ctrl = (nbRep >> 4) & 0x07;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 579 nbRep &= 0x0F;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 580
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 581 if (nbRep == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 582 nbRep = 1;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 583 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 584
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 585 if (datarate > _maxDatarate) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 586 status &= 0xFD; // Datarate KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 587 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 588 //
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 589 // Remark MaxTxPower = 0 and MinTxPower = 10
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 590 //
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 591 if (power < 0 || power > 10) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 592 status &= 0xFB; // TxPower KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 593 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 594
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 595 switch (ctrl) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 596 case 0:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 597 case 1:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 598 case 2:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 599 case 3:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 600 case 4:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 601 SetChannelMask(ctrl, mask);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 602 break;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 603
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 604 case 6:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 605 // enable all 125 kHz channels
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 606 SetChannelMask(0, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 607 SetChannelMask(1, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 608 SetChannelMask(2, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 609 SetChannelMask(3, 0xFFFF);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 610 SetChannelMask(4, mask);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 611 break;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 612
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 613 case 7:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 614 // disable all 125 kHz channels
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 615 SetChannelMask(0, 0x0);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 616 SetChannelMask(1, 0x0);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 617 SetChannelMask(2, 0x0);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 618 SetChannelMask(3, 0x0);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 619 SetChannelMask(4, mask);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 620 break;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 621
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 622 default:
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 623 logWarning("rejecting RFU or unknown control value %d", ctrl);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 624 status &= 0xFE; // ChannelMask KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 625 return LORA_ERROR;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 626 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 627
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 628 if (GetSettings()->Network.ADREnabled) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 629 GetSettings()->Session.TxDatarate = datarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 630 GetSettings()->Session.TxPower = TX_POWERS[power];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 631 GetSettings()->Session.Redundancy = nbRep;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 632 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 633 logDebug("ADR is disabled, DR and Power not changed.");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 634 status &= 0xFB; // TxPower KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 635 status &= 0xFD; // Datarate KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 636 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 637
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 638 logDebug("ADR DR: %u PWR: %u Ctrl: %02x Mask: %04x NbRep: %u Stat: %02x", datarate, power, ctrl, mask, nbRep, status);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 639
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 640 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 641 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 642
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 643 uint8_t ChannelPlan_US915::ValidateAdrConfiguration() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 644 uint8_t status = 0x07;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 645 uint8_t chans_enabled = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 646 uint8_t datarate = GetSettings()->Session.TxDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 647 uint8_t power = GetSettings()->Session.TxPower;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 648
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 649 if (!GetSettings()->Network.ADREnabled) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 650 logDebug("ADR disabled - no applied changes to validate");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 651 return status;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 652 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 653
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 654 if (datarate > _maxDatarate) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 655 logWarning("ADR Datarate KO - outside allowed range");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 656 status &= 0xFD; // Datarate KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 657 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 658 if (power < _minTxPower || power > _maxTxPower) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 659 logWarning("ADR TX Power KO - outside allowed range");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 660 status &= 0xFB; // TxPower KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 661 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 662
Jenkins@KEILDM1.dc.multitech.prv 38:3f9f04830555 663 // at least 2 125kHz channels must be enabled
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 664 chans_enabled += CountBits(_channelMask[0]);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 665 chans_enabled += CountBits(_channelMask[1]);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 666 chans_enabled += CountBits(_channelMask[2]);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 667 chans_enabled += CountBits(_channelMask[3]);
Jenkins@KEILDM1.dc.multitech.prv 38:3f9f04830555 668 // Semtech reference (LoRaMac-node) enforces at least 2 channels
Jenkins@KEILDM1.dc.multitech.prv 38:3f9f04830555 669 if (chans_enabled < 2) {
Jenkins@KEILDM1.dc.multitech.prv 38:3f9f04830555 670 logWarning("ADR Channel Mask KO - at least 2 125kHz channels must be enabled");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 671 status &= 0xFE; // ChannelMask KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 672 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 673
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 674 // if TXDR == 4 (SF8@500kHz) at least 1 500kHz channel must be enabled
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 675 if (datarate == DR_4 && (CountBits(_channelMask[4] & 0xFF) == 0)) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 676 logWarning("ADR Datarate KO - DR4 requires at least 1 500kHz channel enabled");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 677 status &= 0xFD; // Datarate KO
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 678 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 679
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 680 return status;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 681 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 682
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 683 uint32_t ChannelPlan_US915::GetTimeOffAir()
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 684 {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 685 if (GetSettings()->Test.DisableDutyCycle == lora::ON)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 686 return 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 687
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 688 uint32_t min = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 689 uint32_t now = _dutyCycleTimer.read_ms();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 690
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 691 if (GetSettings()->Session.AggregatedTimeOffEnd > 0 && GetSettings()->Session.AggregatedTimeOffEnd > now) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 692 min = std::max < uint32_t > (min, GetSettings()->Session.AggregatedTimeOffEnd - now);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 693 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 694
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 695 now = time(NULL);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 696 uint32_t join_time = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 697
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 698 if (GetSettings()->Session.JoinFirstAttempt != 0 && now < GetSettings()->Session.JoinTimeOffEnd) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 699 join_time = (GetSettings()->Session.JoinTimeOffEnd - now) * 1000;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 700 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 701
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 702 min = std::max < uint32_t > (join_time, min);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 703
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 704 return min;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 705 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 706
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 707 std::vector<uint32_t> lora::ChannelPlan_US915::GetChannels() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 708 std::vector < uint32_t > chans;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 709
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 710 if (GetSettings()->Network.FrequencySubBand > 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 711 uint8_t chans_per_group = 8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 712 size_t start = (GetSettings()->Network.FrequencySubBand - 1) * chans_per_group;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 713 for (int8_t i = start; i < int8_t(start + chans_per_group); i++) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 714 chans.push_back(GetChannel(i).Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 715 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 716 chans.push_back(GetChannel(_numChans125k + (GetSettings()->Network.FrequencySubBand - 1)).Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 717 chans.push_back(GetRxWindow(2).Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 718 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 719 for (int8_t i = 0; i < _numChans; i++) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 720 chans.push_back(GetChannel(i).Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 721 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 722 chans.push_back(GetRxWindow(2).Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 723 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 724
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 725 return chans;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 726 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 727
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 728 std::vector<uint8_t> lora::ChannelPlan_US915::GetChannelRanges() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 729 std::vector < uint8_t > ranges;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 730
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 731 if (GetSettings()->Network.FrequencySubBand > 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 732 uint8_t chans_per_group = 8;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 733 size_t start = (GetSettings()->Network.FrequencySubBand - 1) * chans_per_group;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 734 for (int8_t i = start; i < int8_t(start + chans_per_group); i++) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 735 ranges.push_back(GetChannel(i).DrRange.Value);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 736 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 737 ranges.push_back(GetChannel(_numChans125k + (GetSettings()->Network.FrequencySubBand - 1)).DrRange.Value);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 738 ranges.push_back(GetRxWindow(2).DatarateIndex);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 739 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 740 for (int8_t i = 0; i < _numChans; i++) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 741 ranges.push_back(GetChannel(i).DrRange.Value);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 742 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 743 ranges.push_back(GetRxWindow(2).DatarateIndex);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 744 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 745
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 746 ranges.push_back(GetRxWindow(2).DatarateIndex);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 747
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 748 return ranges;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 749
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 750 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 751
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 752 void lora::ChannelPlan_US915::EnableDefaultChannels() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 753 SetFrequencySubBand(GetFrequencySubBand());
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 754 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 755
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 756 uint8_t ChannelPlan_US915::GetNextChannel()
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 757 {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 758 if (GetSettings()->Session.AggregatedTimeOffEnd != 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 759 return LORA_AGGREGATED_DUTY_CYCLE;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 760 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 761
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 762 if (P2PEnabled() || GetSettings()->Network.TxFrequency != 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 763 logDebug("Using frequency %d", GetSettings()->Network.TxFrequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 764
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 765 if (GetSettings()->Test.DisableDutyCycle != lora::ON) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 766 int8_t band = GetDutyBand(GetSettings()->Network.TxFrequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 767 logDebug("band: %d freq: %d", band, GetSettings()->Network.TxFrequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 768 if (band != -1 && _dutyBands[band].TimeOffEnd != 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 769 return LORA_NO_CHANS_ENABLED;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 770 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 771 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 772
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 773 GetRadio()->SetChannel(GetSettings()->Network.TxFrequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 774 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 775 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 776
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 777 uint8_t start = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 778 uint8_t maxChannels = _numChans125k;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 779 uint8_t nbEnabledChannels = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 780 uint8_t *enabledChannels = new uint8_t[maxChannels];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 781
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 782 if (GetTxDatarate().Bandwidth == BW_500) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 783 maxChannels = _numChans500k;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 784 start = _numChans125k;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 785 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 786
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 787 // Search how many channels are enabled
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 788 DatarateRange range;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 789 uint8_t dr_index = GetSettings()->Session.TxDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 790 uint32_t now = _dutyCycleTimer.read_ms();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 791
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 792 for (size_t i = 0; i < _dutyBands.size(); i++) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 793 if (_dutyBands[i].TimeOffEnd < now || GetSettings()->Test.DisableDutyCycle == lora::ON) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 794 _dutyBands[i].TimeOffEnd = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 795 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 796 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 797
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 798 for (uint8_t i = start; i < start + maxChannels; i++) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 799 range = GetChannel(i).DrRange;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 800 // logDebug("chan: %d freq: %d range:%02x", i, GetChannel(i).Frequency, range.Value);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 801
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 802 if (IsChannelEnabled(i) && (dr_index >= range.Fields.Min && dr_index <= range.Fields.Max)) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 803 int8_t band = GetDutyBand(GetChannel(i).Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 804 // logDebug("band: %d freq: %d", band, _channels[i].Frequency);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 805 if (band != -1 && _dutyBands[band].TimeOffEnd == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 806 enabledChannels[nbEnabledChannels++] = i;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 807 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 808 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 809 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 810
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 811 if (GetTxDatarate().Bandwidth == BW_500) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 812 _dutyBands[0].PowerMax = 26;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 813 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 814 if (nbEnabledChannels < 50)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 815 _dutyBands[0].PowerMax = 21;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 816 else
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 817 _dutyBands[0].PowerMax = 30;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 818 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 819
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 820 logTrace("Number of available channels: %d", nbEnabledChannels);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 821
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 822 uint32_t freq = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 823 uint8_t sf = GetTxDatarate().SpreadingFactor;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 824 uint8_t bw = GetTxDatarate().Bandwidth;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 825 int16_t thres = DEFAULT_FREE_CHAN_RSSI_THRESHOLD;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 826
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 827 if (nbEnabledChannels == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 828 delete [] enabledChannels;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 829 return LORA_NO_CHANS_ENABLED;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 830 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 831
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 832 if (GetSettings()->Network.CADEnabled) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 833 // Search for free channel with ms timeout
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 834 int16_t timeout = 10000;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 835 Timer tmr;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 836 tmr.start();
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 837
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 838 for (uint8_t j = rand_r(0, nbEnabledChannels - 1); tmr.read_ms() < timeout; j++) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 839 freq = GetChannel(enabledChannels[j]).Frequency;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 840
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 841 if (GetRadio()->IsChannelFree(SxRadio::MODEM_LORA, freq, sf, thres, bw)) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 842 _txChannel = enabledChannels[j];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 843 break;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 844 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 845 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 846 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 847 uint8_t j = rand_r(0, nbEnabledChannels - 1);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 848 _txChannel = enabledChannels[j];
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 849 freq = GetChannel(_txChannel).Frequency;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 850 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 851
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 852 assert(freq != 0);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 853
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 854 logDebug("Using channel %d : %d", _txChannel, freq);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 855 GetRadio()->SetChannel(freq);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 856
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 857 delete [] enabledChannels;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 858 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 859 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 860
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 861 uint8_t lora::ChannelPlan_US915::GetJoinDatarate() {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 862 uint8_t dr = GetSettings()->Session.TxDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 863
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 864 if (GetSettings()->Test.DisableRandomJoinDatarate == lora::OFF) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 865 static bool altDatarate = false;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 866
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 867 if (GetSettings()->Network.FrequencySubBand == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 868 static uint16_t used_bands_125k = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 869 static uint16_t used_bands_500k = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 870 uint8_t frequency_sub_band = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 871
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 872 if (altDatarate) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 873 // 500k channel
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 874 if (CountBits(used_bands_500k) == 8) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 875 used_bands_500k = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 876 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 877 while ((frequency_sub_band = rand_r(1, 8)) && (used_bands_500k & (1 << (frequency_sub_band - 1))) != 0)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 878 ;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 879 used_bands_500k |= (1 << (frequency_sub_band - 1));
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 880 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 881 // 125k channel
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 882 if (CountBits(used_bands_125k) == 8) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 883 used_bands_125k = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 884 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 885 while ((frequency_sub_band = rand_r(1, 8)) && (used_bands_125k & (1 << (frequency_sub_band - 1))) != 0)
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 886 ;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 887 used_bands_125k |= (1 << (frequency_sub_band - 1));
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 888 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 889
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 890 logWarning("JoinDatarate setting frequency sub band to %d 125k: %04x 500k: %04x", frequency_sub_band, used_bands_125k, used_bands_500k);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 891 SetFrequencySubBand(frequency_sub_band);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 892 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 893
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 894 if (altDatarate && CountBits(_channelMask[4] > 0)) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 895 dr = lora::DR_4;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 896 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 897 dr = lora::DR_0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 898 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 899 altDatarate = !altDatarate;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 900 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 901
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 902 return dr;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 903 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 904
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 905 uint8_t lora::ChannelPlan_US915::CalculateJoinBackoff(uint8_t size) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 906
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 907 time_t now = time(NULL);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 908 uint32_t time_on_max = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 909 static uint32_t time_off_max = 15;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 910 uint32_t rand_time_off = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 911 static uint16_t join_cnt = 0;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 912
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 913 // TODO: calc time-off-max based on RTC time from JoinFirstAttempt, time-off-max is lost over sleep
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 914
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 915 if ((time_t)GetSettings()->Session.JoinTimeOffEnd > now) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 916 return LORA_JOIN_BACKOFF;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 917 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 918
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 919 uint32_t secs_since_first_attempt = (now - GetSettings()->Session.JoinFirstAttempt);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 920 uint16_t hours_since_first_attempt = secs_since_first_attempt / (60 * 60);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 921
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 922 join_cnt = (join_cnt+1) % 16;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 923
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 924 if (GetSettings()->Session.JoinFirstAttempt == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 925 /* 1 % duty-cycle for first hour
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 926 * 0.1 % next 10 hours
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 927 * 0.01 % upto 24 hours */
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 928 GetSettings()->Session.JoinFirstAttempt = now;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 929 GetSettings()->Session.JoinTimeOnAir += GetTimeOnAir(size);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 930 GetSettings()->Session.JoinTimeOffEnd = now + rand_r(GetSettings()->Network.JoinDelay + 2, GetSettings()->Network.JoinDelay + 3);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 931 } else if (join_cnt == 0) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 932 if (hours_since_first_attempt < 1) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 933 time_on_max = 36000;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 934 rand_time_off = rand_r(time_off_max - 1, time_off_max + 1);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 935 // time off max 1 hour
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 936 time_off_max = std::min < uint32_t > (time_off_max * 2, 60 * 60);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 937
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 938 if (GetSettings()->Session.JoinTimeOnAir < time_on_max) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 939 GetSettings()->Session.JoinTimeOnAir += GetTimeOnAir(size);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 940 GetSettings()->Session.JoinTimeOffEnd = now + rand_time_off;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 941 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 942 logWarning("Max time-on-air limit met for current join backoff period");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 943 GetSettings()->Session.JoinTimeOffEnd = GetSettings()->Session.JoinFirstAttempt + 60 * 60;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 944 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 945 } else if (hours_since_first_attempt < 11) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 946 if (GetSettings()->Session.JoinTimeOnAir < 36000) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 947 GetSettings()->Session.JoinTimeOnAir = 36000;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 948 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 949 time_on_max = 72000;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 950 rand_time_off = rand_r(time_off_max - 1, time_off_max + 1);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 951 // time off max 1 hour
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 952 time_off_max = std::min < uint32_t > (time_off_max * 2, 60 * 60);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 953
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 954 if (GetSettings()->Session.JoinTimeOnAir < time_on_max) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 955 GetSettings()->Session.JoinTimeOnAir += GetTimeOnAir(size);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 956 GetSettings()->Session.JoinTimeOffEnd = now + rand_time_off;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 957 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 958 logWarning("Max time-on-air limit met for current join backoff period");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 959 GetSettings()->Session.JoinTimeOffEnd = GetSettings()->Session.JoinFirstAttempt + 11 * 60 * 60;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 960 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 961 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 962 if (GetSettings()->Session.JoinTimeOnAir < 72000) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 963 GetSettings()->Session.JoinTimeOnAir = 72000;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 964 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 965 uint32_t join_time = 2500;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 966
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 967 // 16 join attempts is ~2754 ms, check if this is the third of the 24 hour period
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 968
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 969 time_on_max = 80700;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 970 time_off_max = 1 * 60 * 60; // 1 hour
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 971 rand_time_off = rand_r(time_off_max - 1, time_off_max + 1);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 972
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 973 if (GetSettings()->Session.JoinTimeOnAir < time_on_max - join_time) {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 974 GetSettings()->Session.JoinTimeOnAir += GetTimeOnAir(size);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 975 GetSettings()->Session.JoinTimeOffEnd = now + rand_time_off;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 976 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 977 logWarning("Max time-on-air limit met for current join backoff period");
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 978 // Reset the join time on air and set end of restriction to the next 24 hour period
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 979 GetSettings()->Session.JoinTimeOnAir = 72000;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 980 uint16_t days = (now - GetSettings()->Session.JoinFirstAttempt) / (24 * 60 * 60) + 1;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 981 logWarning("days : %d", days);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 982 GetSettings()->Session.JoinTimeOffEnd = GetSettings()->Session.JoinFirstAttempt + ((days * 24) + 11) * 60 * 60;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 983 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 984 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 985
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 986 logWarning("JoinBackoff: %lu seconds Time On Air: %lu / %lu", GetSettings()->Session.JoinTimeOffEnd - now, GetSettings()->Session.JoinTimeOnAir, time_on_max);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 987 } else {
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 988 GetSettings()->Session.JoinTimeOnAir += GetTimeOnAir(size);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 989 GetSettings()->Session.JoinTimeOffEnd = now + rand_r(GetSettings()->Network.JoinDelay + 2, GetSettings()->Network.JoinDelay + 3);
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 990 }
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 991
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 992 return LORA_OK;
Jenkins@KEILDM1.dc.multitech.prv 36:bf7b1b13d7da 993 }