1 #if !defined(_RADIOLIB_LW_H) && !RADIOLIB_EXCLUDE_LORAWAN
4 #include "../../TypeDef.h"
5 #include "../PhysicalLayer/PhysicalLayer.h"
6 #include "../../utils/Cryptography.h"
9 #define RADIOLIB_LW_MODE_OTAA (0x07AA)
10 #define RADIOLIB_LW_MODE_ABP (0x0AB9)
11 #define RADIOLIB_LW_MODE_NONE (0x0000)
14 #define RADIOLIB_LW_CLASS_A (0x0A)
15 #define RADIOLIB_LW_CLASS_B (0x0B)
16 #define RADIOLIB_LW_CLASS_C (0x0C)
19 #define RADIOLIB_LW_LORA_SYNC_WORD (0x34)
20 #define RADIOLIB_LW_LORA_PREAMBLE_LEN (8)
21 #define RADIOLIB_LW_GFSK_SYNC_WORD (0xC194C1)
22 #define RADIOLIB_LW_GFSK_PREAMBLE_LEN (5)
25 #define RADIOLIB_LW_MHDR_MTYPE_JOIN_REQUEST (0x00 << 5)
26 #define RADIOLIB_LW_MHDR_MTYPE_JOIN_ACCEPT (0x01 << 5)
27 #define RADIOLIB_LW_MHDR_MTYPE_UNCONF_DATA_UP (0x02 << 5)
28 #define RADIOLIB_LW_MHDR_MTYPE_UNCONF_DATA_DOWN (0x03 << 5)
29 #define RADIOLIB_LW_MHDR_MTYPE_CONF_DATA_UP (0x04 << 5)
30 #define RADIOLIB_LW_MHDR_MTYPE_CONF_DATA_DOWN (0x05 << 5)
31 #define RADIOLIB_LW_MHDR_MTYPE_PROPRIETARY (0x07 << 5)
32 #define RADIOLIB_LW_MHDR_MTYPE_MASK (0x07 << 5)
33 #define RADIOLIB_LW_MHDR_MAJOR_R1 (0x00 << 0)
36 #define RADIOLIB_LW_FCTRL_ADR_ENABLED (0x01 << 7)
37 #define RADIOLIB_LW_FCTRL_ADR_DISABLED (0x00 << 7)
38 #define RADIOLIB_LW_FCTRL_ADR_ACK_REQ (0x01 << 6)
39 #define RADIOLIB_LW_FCTRL_ACK (0x01 << 5)
40 #define RADIOLIB_LW_FCTRL_FRAME_PENDING (0x01 << 4)
43 #define RADIOLIB_LW_FPORT_MAC_COMMAND (0x00 << 0)
44 #define RADIOLIB_LW_FPORT_RESERVED (0xE0 << 0)
47 #define RADIOLIB_LW_LINK_CHECK_REQ (0x02 << 0)
48 #define RADIOLIB_LW_LINK_ADR_ANS (0x03 << 0)
49 #define RADIOLIB_LW_DUTY_CYCLE_ANS (0x04 << 0)
50 #define RADIOLIB_LW_RX_PARAM_SETUP_ANS (0x05 << 0)
51 #define RADIOLIB_LW_DEV_STATUS_ANS (0x06 << 0)
52 #define RADIOLIB_LW_NEW_CHANNEL_ANS (0x07 << 0)
53 #define RADIOLIB_LW_RX_TIMING_SETUP_ANS (0x08 << 0)
55 #define RADIOLIB_LW_NOPTS_LEN (8)
58 #define RADIOLIB_LW_DATA_RATE_FSK_50_K (0x01 << 7)
59 #define RADIOLIB_LW_DATA_RATE_SF_12 (0x06 << 4)
60 #define RADIOLIB_LW_DATA_RATE_SF_11 (0x05 << 4)
61 #define RADIOLIB_LW_DATA_RATE_SF_10 (0x04 << 4)
62 #define RADIOLIB_LW_DATA_RATE_SF_9 (0x03 << 4)
63 #define RADIOLIB_LW_DATA_RATE_SF_8 (0x02 << 4)
64 #define RADIOLIB_LW_DATA_RATE_SF_7 (0x01 << 4)
65 #define RADIOLIB_LW_DATA_RATE_BW_500_KHZ (0x00 << 2)
66 #define RADIOLIB_LW_DATA_RATE_BW_250_KHZ (0x01 << 2)
67 #define RADIOLIB_LW_DATA_RATE_BW_125_KHZ (0x02 << 2)
68 #define RADIOLIB_LW_DATA_RATE_BW_RESERVED (0x03 << 2)
69 #define RADIOLIB_LW_DATA_RATE_CR_4_5 (0x00 << 0)
70 #define RADIOLIB_LW_DATA_RATE_CR_4_6 (0x01 << 0)
71 #define RADIOLIB_LW_DATA_RATE_CR_4_7 (0x02 << 0)
72 #define RADIOLIB_LW_DATA_RATE_CR_4_8 (0x03 << 0)
73 #define RADIOLIB_LW_DATA_RATE_UNUSED (0xFF << 0)
75 #define RADIOLIB_LW_CHANNEL_DIR_UPLINK (0x00 << 0)
76 #define RADIOLIB_LW_CHANNEL_DIR_DOWNLINK (0x01 << 0)
77 #define RADIOLIB_LW_CHANNEL_DIR_BOTH (0x02 << 0)
78 #define RADIOLIB_LW_CHANNEL_DIR_NONE (0x03 << 0)
79 #define RADIOLIB_LW_BAND_DYNAMIC (0)
80 #define RADIOLIB_LW_BAND_FIXED (1)
81 #define RADIOLIB_LW_CHANNEL_NUM_DATARATES (15)
82 #define RADIOLIB_LW_CHANNEL_INDEX_NONE (0xFF >> 0)
85 #define RADIOLIB_LW_RECEIVE_DELAY_1_MS (1000)
86 #define RADIOLIB_LW_RECEIVE_DELAY_2_MS ((RADIOLIB_LW_RECEIVE_DELAY_1_MS) + 1000)
87 #define RADIOLIB_LW_RX1_DR_OFFSET (0)
88 #define RADIOLIB_LW_JOIN_ACCEPT_DELAY_1_MS (5000)
89 #define RADIOLIB_LW_JOIN_ACCEPT_DELAY_2_MS (6000)
90 #define RADIOLIB_LW_MAX_FCNT_GAP (16384)
91 #define RADIOLIB_LW_ADR_ACK_LIMIT_EXP (0x06)
92 #define RADIOLIB_LW_ADR_ACK_DELAY_EXP (0x05)
93 #define RADIOLIB_LW_RETRANSMIT_TIMEOUT_MIN_MS (1000)
94 #define RADIOLIB_LW_RETRANSMIT_TIMEOUT_MAX_MS (3000)
95 #define RADIOLIB_LW_POWER_STEP_SIZE_DBM (-2)
96 #define RADIOLIB_LW_REJOIN_MAX_COUNT_N (10)
97 #define RADIOLIB_LW_REJOIN_MAX_TIME_N (15)
100 #define RADIOLIB_LW_JOIN_REQUEST_LEN (23)
101 #define RADIOLIB_LW_JOIN_REQUEST_JOIN_EUI_POS (1)
102 #define RADIOLIB_LW_JOIN_REQUEST_DEV_EUI_POS (9)
103 #define RADIOLIB_LW_JOIN_REQUEST_DEV_NONCE_POS (17)
104 #define RADIOLIB_LW_JOIN_REQUEST_TYPE (0xFF)
105 #define RADIOLIB_LW_JOIN_REQUEST_TYPE_0 (0x00)
106 #define RADIOLIB_LW_JOIN_REQUEST_TYPE_1 (0x01)
107 #define RADIOLIB_LW_JOIN_REQUEST_TYPE_2 (0x02)
110 #define RADIOLIB_LW_JOIN_ACCEPT_MAX_LEN (33)
111 #define RADIOLIB_LW_JOIN_ACCEPT_JOIN_NONCE_POS (1)
112 #define RADIOLIB_LW_JOIN_ACCEPT_HOME_NET_ID_POS (4)
113 #define RADIOLIB_LW_JOIN_ACCEPT_DEV_ADDR_POS (7)
114 #define RADIOLIB_LW_JOIN_ACCEPT_JOIN_EUI_POS (4)
115 #define RADIOLIB_LW_JOIN_ACCEPT_DL_SETTINGS_POS (11)
116 #define RADIOLIB_LW_JOIN_ACCEPT_RX_DELAY_POS (12)
117 #define RADIOLIB_LW_JOIN_ACCEPT_DEV_NONCE_POS (12)
118 #define RADIOLIB_LW_JOIN_ACCEPT_CFLIST_POS (13)
119 #define RADIOLIB_LW_JOIN_ACCEPT_CFLIST_LEN (16)
120 #define RADIOLIB_LW_JOIN_ACCEPT_CFLIST_TYPE_POS (RADIOLIB_LW_JOIN_ACCEPT_CFLIST_POS + RADIOLIB_LW_JOIN_ACCEPT_CFLIST_LEN - 1)
123 #define RADIOLIB_LW_JOIN_ACCEPT_R_1_0 (0x00 << 7)
124 #define RADIOLIB_LW_JOIN_ACCEPT_R_1_1 (0x01 << 7)
125 #define RADIOLIB_LW_JOIN_ACCEPT_F_NWK_S_INT_KEY (0x01)
126 #define RADIOLIB_LW_JOIN_ACCEPT_APP_S_KEY (0x02)
127 #define RADIOLIB_LW_JOIN_ACCEPT_S_NWK_S_INT_KEY (0x03)
128 #define RADIOLIB_LW_JOIN_ACCEPT_NWK_S_ENC_KEY (0x04)
129 #define RADIOLIB_LW_JOIN_ACCEPT_JS_ENC_KEY (0x05)
130 #define RADIOLIB_LW_JOIN_ACCEPT_JS_INT_KEY (0x06)
133 #define RADIOLIB_LW_FHDR_LEN_START_OFFS (16)
134 #define RADIOLIB_LW_FHDR_DEV_ADDR_POS (RADIOLIB_LW_FHDR_LEN_START_OFFS + 1)
135 #define RADIOLIB_LW_FHDR_FCTRL_POS (RADIOLIB_LW_FHDR_LEN_START_OFFS + 5)
136 #define RADIOLIB_LW_FHDR_FCNT_POS (RADIOLIB_LW_FHDR_LEN_START_OFFS + 6)
137 #define RADIOLIB_LW_FHDR_FOPTS_POS (RADIOLIB_LW_FHDR_LEN_START_OFFS + 8)
138 #define RADIOLIB_LW_FHDR_FOPTS_LEN_MASK (0x0F)
139 #define RADIOLIB_LW_FHDR_FOPTS_MAX_LEN (15)
140 #define RADIOLIB_LW_FHDR_FPORT_POS(FOPTS) (RADIOLIB_LW_FHDR_LEN_START_OFFS + 8 + (FOPTS))
141 #define RADIOLIB_LW_FRAME_PAYLOAD_POS(FOPTS) (RADIOLIB_LW_FHDR_LEN_START_OFFS + 9 + (FOPTS))
142 #define RADIOLIB_LW_FRAME_LEN(PAYLOAD, FOPTS) (16 + 13 + (PAYLOAD) + (FOPTS))
145 #define RADIOLIB_LW_BLOCK_MAGIC_POS (0)
146 #define RADIOLIB_LW_BLOCK_CONF_FCNT_POS (1)
147 #define RADIOLIB_LW_BLOCK_DIR_POS (5)
148 #define RADIOLIB_LW_BLOCK_DEV_ADDR_POS (6)
149 #define RADIOLIB_LW_BLOCK_FCNT_POS (10)
152 #define RADIOLIB_LW_ENC_BLOCK_MAGIC (0x01)
153 #define RADIOLIB_LW_ENC_BLOCK_COUNTER_ID_POS (4)
154 #define RADIOLIB_LW_ENC_BLOCK_COUNTER_POS (15)
157 #define RADIOLIB_LW_MIC_BLOCK_MAGIC (0x49)
158 #define RADIOLIB_LW_MIC_BLOCK_LEN_POS (15)
159 #define RADIOLIB_LW_MIC_DATA_RATE_POS (3)
160 #define RADIOLIB_LW_MIC_CH_INDEX_POS (4)
163 #define RADIOLIB_LW_DWELL_TIME (400)
166 #define RADIOLIB_LW_FCNT_NONE (0xFFFFFFFF)
169 #define RADIOLIB_LW_NUM_MAC_COMMANDS (16)
171 #define RADIOLIB_LW_MAC_RESET (0x01)
172 #define RADIOLIB_LW_MAC_LINK_CHECK (0x02)
173 #define RADIOLIB_LW_MAC_LINK_ADR (0x03)
174 #define RADIOLIB_LW_MAC_DUTY_CYCLE (0x04)
175 #define RADIOLIB_LW_MAC_RX_PARAM_SETUP (0x05)
176 #define RADIOLIB_LW_MAC_DEV_STATUS (0x06)
177 #define RADIOLIB_LW_MAC_NEW_CHANNEL (0x07)
178 #define RADIOLIB_LW_MAC_RX_TIMING_SETUP (0x08)
179 #define RADIOLIB_LW_MAC_TX_PARAM_SETUP (0x09)
180 #define RADIOLIB_LW_MAC_DL_CHANNEL (0x0A)
181 #define RADIOLIB_LW_MAC_REKEY (0x0B)
182 #define RADIOLIB_LW_MAC_ADR_PARAM_SETUP (0x0C)
183 #define RADIOLIB_LW_MAC_DEVICE_TIME (0x0D)
184 #define RADIOLIB_LW_MAC_FORCE_REJOIN (0x0E)
185 #define RADIOLIB_LW_MAC_REJOIN_PARAM_SETUP (0x0F)
186 #define RADIOLIB_LW_MAC_PROPRIETARY (0x80)
189 #define RADIOLIB_LW_MAC_COMMAND_QUEUE_SIZE (9)
192 #define RADIOLIB_LW_NUM_AVAILABLE_CHANNELS (16)
195 #define RADIOLIB_LW_MAX_MAC_COMMAND_LEN_DOWN (5)
196 #define RADIOLIB_LW_MAX_MAC_COMMAND_LEN_UP (2)
197 #define RADIOLIB_LW_MAX_NUM_ADR_COMMANDS (8)
218 { 0x00, 0, 0,
false },
219 { RADIOLIB_LW_MAC_RESET, 1, 1,
false },
220 { RADIOLIB_LW_MAC_LINK_CHECK, 2, 0,
true },
221 { RADIOLIB_LW_MAC_LINK_ADR, 4, 1,
false },
222 { RADIOLIB_LW_MAC_DUTY_CYCLE, 1, 0,
false },
223 { RADIOLIB_LW_MAC_RX_PARAM_SETUP, 4, 1,
false },
224 { RADIOLIB_LW_MAC_DEV_STATUS, 0, 2,
false },
225 { RADIOLIB_LW_MAC_NEW_CHANNEL, 5, 1,
false },
226 { RADIOLIB_LW_MAC_RX_TIMING_SETUP, 1, 0,
false },
227 { RADIOLIB_LW_MAC_TX_PARAM_SETUP, 1, 0,
false },
228 { RADIOLIB_LW_MAC_DL_CHANNEL, 4, 1,
false },
229 { RADIOLIB_LW_MAC_REKEY, 1, 1,
false },
230 { RADIOLIB_LW_MAC_ADR_PARAM_SETUP, 1, 0,
false },
231 { RADIOLIB_LW_MAC_DEVICE_TIME, 5, 0,
true },
232 { RADIOLIB_LW_MAC_FORCE_REJOIN, 2, 0,
false },
233 { RADIOLIB_LW_MAC_REJOIN_PARAM_SETUP, 1, 1,
false },
234 { RADIOLIB_LW_MAC_PROPRIETARY, 5, 0,
true }
270 #define RADIOLIB_LW_NONCES_VERSION_VAL (0x0001)
272 enum LoRaWANSchemeBase_t {
273 RADIOLIB_LW_NONCES_START = 0x00,
274 RADIOLIB_LW_NONCES_VERSION = RADIOLIB_LW_NONCES_START,
275 RADIOLIB_LW_NONCES_MODE = RADIOLIB_LW_NONCES_VERSION +
sizeof(uint16_t),
276 RADIOLIB_LW_NONCES_CLASS = RADIOLIB_LW_NONCES_MODE +
sizeof(uint16_t),
277 RADIOLIB_LW_NONCES_PLAN = RADIOLIB_LW_NONCES_CLASS +
sizeof(uint8_t),
278 RADIOLIB_LW_NONCES_CHECKSUM = RADIOLIB_LW_NONCES_PLAN +
sizeof(uint8_t),
279 RADIOLIB_LW_NONCES_DEV_NONCE = RADIOLIB_LW_NONCES_CHECKSUM +
sizeof(uint16_t),
280 RADIOLIB_LW_NONCES_JOIN_NONCE = RADIOLIB_LW_NONCES_DEV_NONCE +
sizeof(uint16_t),
281 RADIOLIB_LW_NONCES_ACTIVE = RADIOLIB_LW_NONCES_JOIN_NONCE + 3,
282 RADIOLIB_LW_NONCES_SIGNATURE = RADIOLIB_LW_NONCES_ACTIVE +
sizeof(uint8_t),
283 RADIOLIB_LW_NONCES_BUF_SIZE = RADIOLIB_LW_NONCES_SIGNATURE +
sizeof(uint16_t)
286 enum LoRaWANSchemeSession_t {
287 RADIOLIB_LW_SESSION_START = 0x00,
288 RADIOLIB_LW_SESSION_NWK_SENC_KEY = RADIOLIB_LW_SESSION_START,
289 RADIOLIB_LW_SESSION_APP_SKEY = RADIOLIB_LW_SESSION_NWK_SENC_KEY + RADIOLIB_AES128_BLOCK_SIZE,
290 RADIOLIB_LW_SESSION_FNWK_SINT_KEY = RADIOLIB_LW_SESSION_APP_SKEY + RADIOLIB_AES128_BLOCK_SIZE,
291 RADIOLIB_LW_SESSION_SNWK_SINT_KEY = RADIOLIB_LW_SESSION_FNWK_SINT_KEY + RADIOLIB_AES128_BLOCK_SIZE,
292 RADIOLIB_LW_SESSION_DEV_ADDR = RADIOLIB_LW_SESSION_SNWK_SINT_KEY + RADIOLIB_AES128_BLOCK_SIZE,
293 RADIOLIB_LW_SESSION_NONCES_SIGNATURE = RADIOLIB_LW_SESSION_DEV_ADDR +
sizeof(uint32_t),
294 RADIOLIB_LW_SESSION_A_FCNT_DOWN = RADIOLIB_LW_SESSION_NONCES_SIGNATURE +
sizeof(uint16_t),
295 RADIOLIB_LW_SESSION_CONF_FCNT_UP = RADIOLIB_LW_SESSION_A_FCNT_DOWN +
sizeof(uint32_t),
296 RADIOLIB_LW_SESSION_CONF_FCNT_DOWN = RADIOLIB_LW_SESSION_CONF_FCNT_UP +
sizeof(uint32_t),
297 RADIOLIB_LW_SESSION_RJ_COUNT0 = RADIOLIB_LW_SESSION_CONF_FCNT_DOWN +
sizeof(uint32_t),
298 RADIOLIB_LW_SESSION_RJ_COUNT1 = RADIOLIB_LW_SESSION_RJ_COUNT0 +
sizeof(uint16_t),
299 RADIOLIB_LW_SESSION_HOMENET_ID = RADIOLIB_LW_SESSION_RJ_COUNT1 +
sizeof(uint16_t),
300 RADIOLIB_LW_SESSION_VERSION = RADIOLIB_LW_SESSION_HOMENET_ID +
sizeof(uint32_t),
301 RADIOLIB_LW_SESSION_DUTY_CYCLE = RADIOLIB_LW_SESSION_VERSION +
sizeof(uint8_t),
302 RADIOLIB_LW_SESSION_RX_PARAM_SETUP = RADIOLIB_LW_SESSION_DUTY_CYCLE + MacTable[RADIOLIB_LW_MAC_DUTY_CYCLE].lenDn,
303 RADIOLIB_LW_SESSION_RX_TIMING_SETUP = RADIOLIB_LW_SESSION_RX_PARAM_SETUP + MacTable[RADIOLIB_LW_MAC_RX_PARAM_SETUP].lenDn,
304 RADIOLIB_LW_SESSION_TX_PARAM_SETUP = RADIOLIB_LW_SESSION_RX_TIMING_SETUP + MacTable[RADIOLIB_LW_MAC_RX_TIMING_SETUP].lenDn,
305 RADIOLIB_LW_SESSION_ADR_PARAM_SETUP = RADIOLIB_LW_SESSION_TX_PARAM_SETUP + MacTable[RADIOLIB_LW_MAC_TX_PARAM_SETUP].lenDn,
306 RADIOLIB_LW_SESSION_REJOIN_PARAM_SETUP = RADIOLIB_LW_SESSION_ADR_PARAM_SETUP + MacTable[RADIOLIB_LW_MAC_ADR_PARAM_SETUP].lenDn,
307 RADIOLIB_LW_SESSION_BEACON_FREQ = RADIOLIB_LW_SESSION_REJOIN_PARAM_SETUP + MacTable[RADIOLIB_LW_MAC_REJOIN_PARAM_SETUP].lenDn,
308 RADIOLIB_LW_SESSION_PING_SLOT_CHANNEL = RADIOLIB_LW_SESSION_BEACON_FREQ + 3,
309 RADIOLIB_LW_SESSION_PERIODICITY = RADIOLIB_LW_SESSION_PING_SLOT_CHANNEL + 4,
310 RADIOLIB_LW_SESSION_LAST_TIME = RADIOLIB_LW_SESSION_PERIODICITY + 1,
311 RADIOLIB_LW_SESSION_UL_CHANNELS = RADIOLIB_LW_SESSION_LAST_TIME + 4,
312 RADIOLIB_LW_SESSION_DL_CHANNELS = RADIOLIB_LW_SESSION_UL_CHANNELS + 16*MacTable[RADIOLIB_LW_MAC_NEW_CHANNEL].lenDn,
313 RADIOLIB_LW_SESSION_MAC_QUEUE_UL = RADIOLIB_LW_SESSION_DL_CHANNELS + 16*MacTable[RADIOLIB_LW_MAC_DL_CHANNEL].lenDn,
315 RADIOLIB_LW_SESSION_ADR_FCNT = RADIOLIB_LW_SESSION_N_FCNT_DOWN +
sizeof(uint32_t),
316 RADIOLIB_LW_SESSION_LINK_ADR = RADIOLIB_LW_SESSION_ADR_FCNT +
sizeof(uint32_t),
317 RADIOLIB_LW_SESSION_FCNT_UP = RADIOLIB_LW_SESSION_LINK_ADR + MacTable[RADIOLIB_LW_MAC_LINK_ADR].
lenDn,
318 RADIOLIB_LW_SESSION_SIGNATURE = RADIOLIB_LW_SESSION_FCNT_UP +
sizeof(uint32_t),
319 RADIOLIB_LW_SESSION_BUF_SIZE = RADIOLIB_LW_SESSION_SIGNATURE +
sizeof(uint16_t)
345 #define RADIOLIB_LW_CHANNEL_NONE { .enabled = false, .idx = RADIOLIB_LW_CHANNEL_INDEX_NONE, .freq = 0, .drMin = 0, .drMax = 0 }
373 #define RADIOLIB_LW_CHANNEL_SPAN_NONE { .numChannels = 0, .freqStart = 0, .freqStep = 0, .drMin = 0, .drMax = 0, .joinRequestDataRate = RADIOLIB_LW_DATA_RATE_UNUSED }
458 #define RADIOLIB_LW_NUM_SUPPORTED_BANDS (BandLast - BandEU868)
551 int16_t
restore(uint16_t checkSum, uint16_t lwMode, uint8_t lwClass, uint8_t freqPlan);
564 int16_t
beginOTAA(uint64_t joinEUI, uint64_t devEUI, uint8_t* nwkKey, uint8_t* appKey,
bool force =
false, uint8_t joinDr = RADIOLIB_LW_DATA_RATE_UNUSED);
579 int16_t
beginABP(uint32_t addr, uint8_t* fNwkSIntKey, uint8_t* sNwkSIntKey, uint8_t* nwkSEncKey, uint8_t* appSKey,
bool force =
false, uint8_t initialDr = RADIOLIB_LW_DATA_RATE_UNUSED);
599 #if defined(RADIOLIB_BUILD_ARDUINO)
609 int16_t
uplink(String& str, uint8_t fPort,
bool isConfirmed =
false,
LoRaWANEvent_t* event = NULL);
621 int16_t
uplink(
const char* str, uint8_t fPort,
bool isConfirmed =
false,
LoRaWANEvent_t* event = NULL);
633 int16_t
uplink(uint8_t* data,
size_t len, uint8_t fPort,
bool isConfirmed =
false,
LoRaWANEvent_t* event = NULL);
635 #if defined(RADIOLIB_BUILD_ARDUINO)
664 #if defined(RADIOLIB_BUILD_ARDUINO)
709 int16_t
sendReceive(uint8_t* dataUp,
size_t lenUp, uint8_t fPort, uint8_t* dataDown,
size_t* lenDown,
bool isConfirmed =
false,
LoRaWANEvent_t* eventUp = NULL,
LoRaWANEvent_t* eventDown = NULL);
768 void setADR(
bool enable =
true);
831 int16_t
getMacDeviceTimeAns(uint32_t* gpsEpoch, uint8_t* fraction,
bool returnUnix =
true);
845 #if !RADIOLIB_GODMODE
851 static int16_t checkBufferCommon(uint8_t *buffer, uint16_t size);
853 void beginCommon(uint8_t initialDr);
856 uint8_t bufferNonces[RADIOLIB_LW_NONCES_BUF_SIZE] = { 0 };
859 uint8_t bufferSession[RADIOLIB_LW_SESSION_BUF_SIZE] = { 0 };
864 .commands = { { .cid = 0, .payload = { 0 }, .len = 0, .repeat = 0, } },
869 .commands = { { .cid = 0, .payload = { 0 }, .len = 0, .repeat = 0, } },
874 uint32_t devAddr = 0;
875 uint8_t appSKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
876 uint8_t fNwkSIntKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
877 uint8_t sNwkSIntKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
878 uint8_t nwkSEncKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
879 uint8_t jSIntKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
882 uint16_t devNonce = 0;
883 uint32_t joinNonce = 0;
886 uint32_t homeNetId = 0;
887 uint8_t adrLimitExp = RADIOLIB_LW_ADR_ACK_LIMIT_EXP;
888 uint8_t adrDelayExp = RADIOLIB_LW_ADR_ACK_DELAY_EXP;
890 uint8_t txPowerSteps = 0;
891 uint8_t txPowerMax = 0;
893 uint32_t aFCntDown = 0;
894 uint32_t nFCntDown = 0;
895 uint32_t confFCntUp = RADIOLIB_LW_FCNT_NONE;
896 uint32_t confFCntDown = RADIOLIB_LW_FCNT_NONE;
897 uint32_t adrFCnt = 0;
903 bool adrEnabled =
true;
906 bool dutyCycleEnabled =
false;
907 uint32_t dutyCycle = 0;
910 bool dwellTimeEnabledUp =
false;
911 uint16_t dwellTimeUp = 0;
912 bool dwellTimeEnabledDn =
false;
913 uint16_t dwellTimeDn = 0;
930 LoRaWANChannel_t currentChannels[2] = { RADIOLIB_LW_CHANNEL_NONE, RADIOLIB_LW_CHANNEL_NONE };
933 uint8_t dataRates[2] = { RADIOLIB_LW_DATA_RATE_UNUSED, RADIOLIB_LW_DATA_RATE_UNUSED };
948 RadioLibTime_t rxDelays[2] = { RADIOLIB_LW_RECEIVE_DELAY_1_MS, RADIOLIB_LW_RECEIVE_DELAY_2_MS };
951 uint8_t battLevel = 0xFF;
954 bool isMACPayload =
false;
960 int16_t downlinkCommon();
963 uint32_t generateMIC(uint8_t* msg,
size_t len, uint8_t* key);
967 bool verifyMIC(uint8_t* msg,
size_t len, uint8_t* key);
971 int16_t setPhyProperties(uint8_t dir);
976 int16_t setupChannelsDyn(
bool joinRequest =
false);
980 int16_t setupChannelsFix(uint8_t subBand);
983 int16_t processCFList(uint8_t* cfList);
986 int16_t selectChannels();
989 int16_t findDataRate(uint8_t dr,
DataRate_t* dataRate);
992 int16_t restoreChannels();
1005 bool applyChannelMaskDyn(uint8_t chMaskCntl, uint16_t chMask);
1008 bool applyChannelMaskFix(uint8_t chMaskCntl, uint16_t chMask);
1011 uint8_t getMacPayloadLength(uint8_t cid);
1019 void setCSMA(uint8_t backoffMax, uint8_t difsSlots,
bool enableCSMA =
false);
1028 void processAES(uint8_t* in,
size_t len, uint8_t* key, uint8_t* out, uint32_t fCnt, uint8_t dir, uint8_t ctrId,
bool counter);
1031 static uint16_t checkSum16(uint8_t *key, uint16_t keyLen);
1034 template<
typename T>
1035 static T ntoh(uint8_t* buff,
size_t size = 0);
1038 template<
typename T>
1039 static void hton(uint8_t* buff, T val,
size_t size = 0);
LoRaWAN-compatible node (class A device).
Definition: LoRaWAN.h:498
int16_t setBufferNonces(uint8_t *persistentBuffer)
Fill the internal buffer that holds the LW base parameters with a supplied buffer.
Definition: LoRaWAN.cpp:56
int16_t setDatarate(uint8_t drUp)
Set uplink datarate. This should not be used when ADR is enabled.
Definition: LoRaWAN.cpp:1928
int16_t uplink(const char *str, uint8_t fPort, bool isConfirmed=false, LoRaWANEvent_t *event=NULL)
Send a message to the server.
Definition: LoRaWAN.cpp:844
uint8_t maxPayloadDwellTime()
Returns the maximum payload given the currently present dwell time limits. WARNING: the addition of M...
Definition: LoRaWAN.cpp:2008
int16_t beginABP(uint32_t addr, uint8_t *fNwkSIntKey, uint8_t *sNwkSIntKey, uint8_t *nwkSEncKey, uint8_t *appSKey, bool force=false, uint8_t initialDr=RADIOLIB_LW_DATA_RATE_UNUSED)
Join network by performing activation by personalization. In this procedure, all necessary configurat...
Definition: LoRaWAN.cpp:724
void setDutyCycle(bool enable=true, RadioLibTime_t msPerHour=0)
Toggle adherence to dutyCycle limits to on or off.
Definition: LoRaWAN.cpp:1969
RadioLibTime_t timeUntilUplink()
Returns time in milliseconds until next uplink is available under dutyCycle limits.
Definition: LoRaWAN.cpp:1990
void setDeviceStatus(uint8_t battLevel)
Set device status.
Definition: LoRaWAN.cpp:1622
RadioLibTime_t dutyCycleInterval(RadioLibTime_t msPerHour, RadioLibTime_t airtime)
Calculate the minimum interval to adhere to a certain dutyCycle. This interval is based on the ToA of...
Definition: LoRaWAN.cpp:1980
void setDwellTime(bool enable, RadioLibTime_t msPerUplink=0)
Toggle adherence to dwellTime limits to on or off.
Definition: LoRaWAN.cpp:1999
uint32_t getAFCntDown()
Returns the last application downlink's frame counter; also 0 if no application downlink occured yet.
Definition: LoRaWAN.cpp:1638
RadioLibTime_t getLastToA()
Get the Time-on-air of the last uplink message.
Definition: LoRaWAN.cpp:2816
int16_t restore(uint16_t checkSum, uint16_t lwMode, uint8_t lwClass, uint8_t freqPlan)
Restore session by loading information from persistent storage.
Definition: LoRaWAN.cpp:129
int16_t saveSession()
Save the current state of the session to the session buffer.
Definition: LoRaWAN.cpp:804
uint8_t * getBufferNonces()
Returns the pointer to the internal buffer that holds the LW base parameters.
Definition: LoRaWAN.cpp:52
int16_t sendReceive(const char *strUp, uint8_t fPort, uint8_t *dataDown, size_t *lenDown, bool isConfirmed=false, LoRaWANEvent_t *eventUp=NULL, LoRaWANEvent_t *eventDown=NULL)
Send a message to the server and wait for a downlink during Rx1 and/or Rx2 window.
Definition: LoRaWAN.cpp:1602
int16_t beginOTAA(uint64_t joinEUI, uint64_t devEUI, uint8_t *nwkKey, uint8_t *appKey, bool force=false, uint8_t joinDr=RADIOLIB_LW_DATA_RATE_UNUSED)
Join network by performing over-the-air activation. By this procedure, the device will perform an exc...
Definition: LoRaWAN.cpp:449
uint8_t * getBufferSession()
Returns the pointer to the internal buffer that holds the LW session parameters.
Definition: LoRaWAN.cpp:74
void wipe()
Wipe internal persistent parameters. This will reset all counters and saved variables,...
Definition: LoRaWAN.cpp:47
int16_t setTxPower(int8_t txPower)
Configure TX power of the radio module.
Definition: LoRaWAN.cpp:2029
int16_t getMacDeviceTimeAns(uint32_t *gpsEpoch, uint8_t *fraction, bool returnUnix=true)
Returns the network time after requesting a DeviceTime MAC command. Returns 'true' if a network respo...
Definition: LoRaWAN.cpp:2795
uint32_t getNFCntDown()
Returns the last network downlink's frame counter; also 0 if no network downlink occured yet.
Definition: LoRaWAN.cpp:1634
uint64_t getDevAddr()
Returns the DevAddr of the device, regardless of OTAA or ABP mode.
Definition: LoRaWAN.cpp:2812
int16_t downlink(uint8_t *data, size_t *len, LoRaWANEvent_t *event=NULL)
Wait for downlink from the server in either RX1 or RX2 window.
Definition: LoRaWAN.cpp:1269
int16_t setBufferSession(uint8_t *persistentBuffer)
Fill the internal buffer that holds the LW session parameters with a supplied buffer.
Definition: LoRaWAN.cpp:81
LoRaWANNode(PhysicalLayer *phy, const LoRaWANBand_t *band, uint8_t subBand=0)
Default constructor.
Definition: LoRaWAN.cpp:30
uint8_t rx1DrOffset
Offset between TX and RX1 (such that RX1 has equal or lower DR)
Definition: LoRaWAN.h:502
uint32_t getFCntUp()
Returns the last uplink's frame counter; also 0 if no uplink occured yet.
Definition: LoRaWAN.cpp:1627
int16_t sendMacCommandReq(uint8_t cid)
Add a MAC command to the uplink queue. Only LinkCheck and DeviceTime are available to the user....
Definition: LoRaWAN.cpp:2093
int16_t getMacLinkCheckAns(uint8_t *margin, uint8_t *gwCnt)
Returns the quality of connectivity after requesting a LinkCheck MAC command. Returns 'true' if a net...
Definition: LoRaWAN.cpp:2784
void setADR(bool enable=true)
Toggle ADR to on or off.
Definition: LoRaWAN.cpp:1965
LoRaWANChannel_t rx2
RX2 channel properties - may be changed by MAC command.
Definition: LoRaWAN.h:505
void resetFCntDown()
Reset the downlink frame counters (application and network) This is unsafe and can possibly allow rep...
Definition: LoRaWAN.cpp:1642
bool isJoined()
Whether there is an ongoing session active.
Definition: LoRaWAN.cpp:800
Provides common interface for protocols that run on LoRa/FSK modules, such as RTTY or LoRaWAN....
Definition: PhysicalLayer.h:54
unsigned long RadioLibTime_t
Type used for durations in RadioLib.
Definition: TypeDef.h:586
Structure to save information about LoRaWAN band.
Definition: LoRaWAN.h:379
uint8_t dataRates[RADIOLIB_LW_CHANNEL_NUM_DATARATES]
The corresponding datarates, bandwidths and coding rates for DR index.
Definition: LoRaWAN.h:426
int8_t powerMax
Maximum allowed output power in this band in dBm.
Definition: LoRaWAN.h:390
RadioLibTime_t dutyCycle
Number of milliseconds per hour of allowed Time-on-Air.
Definition: LoRaWAN.h:396
uint8_t payloadLenMax[RADIOLIB_LW_CHANNEL_NUM_DATARATES]
Array of allowed maximum payload lengths for each data rate.
Definition: LoRaWAN.h:387
RadioLibTime_t dwellTimeUp
Maximum dwell time per uplink message in milliseconds.
Definition: LoRaWAN.h:399
LoRaWANChannel_t txFreqs[3]
A set of default uplink (TX) channels for frequency-type bands.
Definition: LoRaWAN.h:405
LoRaWANChannel_t txJoinReq[3]
A set of possible extra channels for the Join-Request message for frequency-type bands.
Definition: LoRaWAN.h:408
int8_t powerNumSteps
Number of power steps in this band.
Definition: LoRaWAN.h:393
LoRaWANChannelSpan_t txSpans[2]
Default uplink (TX) channel spans for mask-type bands, including Join-Request parameters.
Definition: LoRaWAN.h:414
uint8_t rx1DataRateBase
The base downlink data rate. Used to calculate data rate changes for adaptive data rate.
Definition: LoRaWAN.h:420
uint8_t bandNum
Identier for this band.
Definition: LoRaWAN.h:381
uint8_t bandType
Whether the channels are fixed per specification, or dynamically allocated through the network (plus ...
Definition: LoRaWAN.h:384
LoRaWANChannel_t rx2
Backup channel for downlink (RX2) window.
Definition: LoRaWAN.h:423
uint8_t numTxSpans
The number of TX channel spans for mask-type bands.
Definition: LoRaWAN.h:411
RadioLibTime_t dwellTimeDn
Maximum dwell time per downlink message in milliseconds.
Definition: LoRaWAN.h:402
LoRaWANChannelSpan_t rx1Span
Default downlink (RX1) channel span for mask-type bands.
Definition: LoRaWAN.h:417
IDs of all currently supported bands.
Structure to save information about LoRaWAN channels. To save space, adjacent channels are saved in "...
Definition: LoRaWAN.h:327
float freq
The channel frequency.
Definition: LoRaWAN.h:335
uint8_t idx
The channel number, as specified by defaults or the network.
Definition: LoRaWAN.h:332
uint8_t drMin
Minimum allowed datarate for this channel.
Definition: LoRaWAN.h:338
bool enabled
Whether this channel is enabled (can be used) or is disabled.
Definition: LoRaWAN.h:329
uint8_t drMax
Maximum allowed datarate for this channel (inclusive)
Definition: LoRaWAN.h:341
Structure to save information about LoRaWAN channels. To save space, adjacent channels are saved in "...
Definition: LoRaWAN.h:352
uint8_t joinRequestDataRate
Allowed data rates for a join request message.
Definition: LoRaWAN.h:369
float freqStart
Center frequency of the first channel in span.
Definition: LoRaWAN.h:357
uint8_t numChannels
Total number of channels in the span.
Definition: LoRaWAN.h:354
uint8_t drMax
Maximum allowed datarate for all channels in this span (inclusive)
Definition: LoRaWAN.h:366
float freqStep
Frequency step between adjacent channels.
Definition: LoRaWAN.h:360
uint8_t drMin
Minimum allowed datarate for all channels in this span.
Definition: LoRaWAN.h:363
Structure to save extra information about uplink/downlink event.
Definition: LoRaWAN.h:467
float freq
Frequency in MHz.
Definition: LoRaWAN.h:482
bool confirmed
Whether the event is confirmed or not (e.g., confirmed uplink sent by user application)
Definition: LoRaWAN.h:472
uint8_t fPort
Port number.
Definition: LoRaWAN.h:491
int16_t power
Transmit power in dBm for uplink, or RSSI for downlink.
Definition: LoRaWAN.h:485
bool confirming
Whether the event is confirming a previous request (e.g., server downlink reply to confirmed uplink s...
Definition: LoRaWAN.h:476
uint8_t datarate
Datarate.
Definition: LoRaWAN.h:479
uint8_t dir
Event direction, one of RADIOLIB_LW_CHANNEL_DIR_*.
Definition: LoRaWAN.h:469
uint32_t fCnt
The appropriate frame counter - for different events, different frame counters will be reported!
Definition: LoRaWAN.h:488
Structure to save information about MAC command.
Definition: LoRaWAN.h:241
uint8_t len
Length of the payload.
Definition: LoRaWAN.h:249
uint8_t cid
The command ID.
Definition: LoRaWAN.h:243
uint8_t repeat
Repetition counter (the command will be uplinked repeat + 1 times)
Definition: LoRaWAN.h:252
uint8_t payload[5]
Payload buffer (5 bytes is the longest possible)
Definition: LoRaWAN.h:246
Structure to hold information about a queue of MAC commands.
Definition: LoRaWAN.h:259
LoRaWANMacCommand_t commands[RADIOLIB_LW_MAC_COMMAND_QUEUE_SIZE]
MAC command buffer.
Definition: LoRaWAN.h:267
uint8_t numCommands
Number of commands in the queue.
Definition: LoRaWAN.h:261
uint8_t len
Total length of the queue.
Definition: LoRaWAN.h:264
MAC command specification structure.
Definition: LoRaWAN.h:203
const uint8_t cid
Command ID.
Definition: LoRaWAN.h:205
const uint8_t lenDn
Uplink message length.
Definition: LoRaWAN.h:208
const uint8_t lenUp
Downlink message length.
Definition: LoRaWAN.h:211
const bool user
Whether this MAC command can be issued by the user or not.
Definition: LoRaWAN.h:214
Common data rate structure.
Definition: PhysicalLayer.h:38