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