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:
Mon Jun 19 16:03:56 2017 -0500
Revision:
36:bf7b1b13d7da
Child:
38:3f9f04830555
xdot-library revision 3.0.0-rc7-1-ga5165ae and mbed-os revision mbed-os-5.4.7

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