Bleeding edge development version of the xDot library for mbed 5. This version of the library is not guaranteed to be stable or well tested and should not be used in production or deployment scenarios.

Dependents:   Dot-Examples Dot-AT-Firmware Dot-Examples TEST_FF1705 ... more

The Dot library provides a LoRaWan certified stack for LoRa communication using MultiTech mDot and xDot devices. The stack is compatible with mbed 5.

Dot Library Version 3 Updates

Dot Library versions 3.x.x require a channel plan to be injected into the stack. Channel plans are included with the 3.x.x Dot Library releases. The following code snippet demonstrates how to create a channel plan and inject it into the stack.

#include "mDot.h"
#include "channel_plans.h"

int main() {
    ChannelPlan* plan = new lora::ChannelPlan_US915();
    assert(plan);
    mDot* dot = mDot::getInstance(plan);
    assert(dot);

    // ...
}

Dot devices must not be deployed with software using a different channel plan than the Dot's default plan! This functionality is for development and testing only!

Multicast Sessions

Multicast sessions and packet rx events in library. When in Class C mode Multicast downlinks can be received. Recieved packets should be filtered on address, counter value will be maintained in the session or can be set explicitly depending on Application support to share Multicast Address, Keys and Counters.

mDot.h

        /**
         * Add a multicast session address and keys
         * Downlink counter is set to 0
         * Up to 3 MULTICAST_SESSIONS can be set
         */
        int32_t setMulticastSession(uint8_t index, uint32_t addr, const uint8_t* nsk, const uint8_t* dsk);
 
        /**
         * Set a multicast session counter
         * Up to 3 MULTICAST_SESSIONS can be set
         */
        int32_t setMulticastDownlinkCounter(uint8_t index, uint32_t count);

mDotEvent.h

The address field was added to PacketRx event.

        virtual void PacketRx(uint8_t port, uint8_t *payload, uint16_t size, int16_t rssi, int8_t snr, lora::DownlinkControl ctrl, uint8_t slot, uint8_t retries, uint32_t address);

The name of the repository can be used to determine which device the stack was compiled for and if it's a development or production-ready build:

A changelog for the Dot library can be found here.

The Dot library version and the version of mbed-os it was compiled against can both be found in the commit message for that revision of the Dot library. Building your application with the same version of mbed-os as what was used to build the Dot library is highly recommended!

The Dot-Examples repository demonstrates how to use the Dot library in a custom application.

The mDot and xDot platform pages have lots of platform specific information and document potential issues, gotchas, etc, and provide instructions for getting started with development. Please take a look at the platform page before starting development as they should answer many questions you will have.

Committer:
Jenkins@KEILDM1.dc.multitech.prv
Date:
Fri Nov 09 14:56:50 2018 -0600
Revision:
133:4ca51f965419
Parent:
132:42d19971dfa9
xdot-library revision 3.1.0-57-g8e84ec6 and mbed-os revision mbed-os-5.10.2

Who changed what in which revision?

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