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:
Tue Sep 18 08:25:41 2018 -0500
Revision:
127:db053d511848
Parent:
126:c17213d3156d
Child:
132:42d19971dfa9
xdot-library revision 3.1.0-28-g84dbee8 and mbed-os revision mbed-os-5.9.6

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