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

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

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

Dot Library Version 3 Updates

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

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

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

    // ...
}

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

Multicast Sessions

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

mDot.h

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

mDotEvent.h

The address field was added to PacketRx event.

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

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

A changelog for the Dot library can be found here.

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

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

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

Committer:
Jenkins@KEILDM1.dc.multitech.prv
Date:
Thu Aug 30 09:06:17 2018 -0500
Revision:
121:b7c80d8c4eb2
Parent:
119:2eaeda989f93
Child:
126:c17213d3156d
xdot-library revision 3.1.0-class-b-alpha-1-15-g0c33f0a and mbed-os revision mbed-os-5.7.7

Who changed what in which revision?

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