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:
Fri Nov 09 14:56:50 2018 -0600
Revision:
133:4ca51f965419
Parent:
132:42d19971dfa9
xdot-library revision 3.1.0-57-g8e84ec6 and mbed-os revision mbed-os-5.10.2

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