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