RadioLib
Universal wireless communication library for Arduino
LoRaWAN.h
1 #if !defined(_RADIOLIB_LORAWAN_H) && !RADIOLIB_EXCLUDE_LORAWAN
2 #define _RADIOLIB_LORAWAN_H
3 
4 #include "../../TypeDef.h"
5 #include "../PhysicalLayer/PhysicalLayer.h"
6 #include "../../utils/Cryptography.h"
7 
8 // activation mode
9 #define RADIOLIB_LORAWAN_MODE_OTAA (0x07AA)
10 #define RADIOLIB_LORAWAN_MODE_ABP (0x0AB9)
11 #define RADIOLIB_LORAWAN_MODE_NONE (0x0000)
12 
13 // operation mode
14 #define RADIOLIB_LORAWAN_CLASS_A (0x0A)
15 #define RADIOLIB_LORAWAN_CLASS_B (0x0B)
16 #define RADIOLIB_LORAWAN_CLASS_C (0x0C)
17 
18 // modulation type
19 #define RADIOLIB_LORAWAN_MODULATION_LORA (0)
20 #define RADIOLIB_LORAWAN_MODULATION_GFSK (1)
21 #define RADIOLIB_LORAWAN_MODULATION_LR_FHSS (2)
22 
23 // preamble format
24 #define RADIOLIB_LORAWAN_LORA_SYNC_WORD (0x34)
25 #define RADIOLIB_LORAWAN_LORA_PREAMBLE_LEN (8)
26 #define RADIOLIB_LORAWAN_GFSK_SYNC_WORD (0xC194C1)
27 #define RADIOLIB_LORAWAN_GFSK_PREAMBLE_LEN (5)
28 #define RADIOLIB_LORAWAN_LR_FHSS_SYNC_WORD (0x2C0F7995)
29 
30 // MAC header field encoding MSB LSB DESCRIPTION
31 #define RADIOLIB_LORAWAN_MHDR_MTYPE_JOIN_REQUEST (0x00 << 5) // 7 5 message type: join request
32 #define RADIOLIB_LORAWAN_MHDR_MTYPE_JOIN_ACCEPT (0x01 << 5) // 7 5 join accept
33 #define RADIOLIB_LORAWAN_MHDR_MTYPE_UNCONF_DATA_UP (0x02 << 5) // 7 5 unconfirmed data up
34 #define RADIOLIB_LORAWAN_MHDR_MTYPE_UNCONF_DATA_DOWN (0x03 << 5) // 7 5 unconfirmed data down
35 #define RADIOLIB_LORAWAN_MHDR_MTYPE_CONF_DATA_UP (0x04 << 5) // 7 5 confirmed data up
36 #define RADIOLIB_LORAWAN_MHDR_MTYPE_CONF_DATA_DOWN (0x05 << 5) // 7 5 confirmed data down
37 #define RADIOLIB_LORAWAN_MHDR_MTYPE_PROPRIETARY (0x07 << 5) // 7 5 proprietary
38 #define RADIOLIB_LORAWAN_MHDR_MTYPE_MASK (0x07 << 5) // 7 5 bitmask of all possible options
39 #define RADIOLIB_LORAWAN_MHDR_MAJOR_R1 (0x00 << 0) // 1 0 major version: LoRaWAN R1
40 
41 // frame control field encoding
42 #define RADIOLIB_LORAWAN_FCTRL_ADR_ENABLED (0x01 << 7) // 7 7 adaptive data rate: enabled
43 #define RADIOLIB_LORAWAN_FCTRL_ADR_DISABLED (0x00 << 7) // 7 7 disabled
44 #define RADIOLIB_LORAWAN_FCTRL_ADR_ACK_REQ (0x01 << 6) // 6 6 adaptive data rate ACK request
45 #define RADIOLIB_LORAWAN_FCTRL_ACK (0x01 << 5) // 5 5 confirmed message acknowledge
46 #define RADIOLIB_LORAWAN_FCTRL_FRAME_PENDING (0x01 << 4) // 4 4 downlink frame is pending
47 
48 // fPort field
49 #define RADIOLIB_LORAWAN_FPORT_MAC_COMMAND (0x00 << 0) // 7 0 payload contains MAC commands only
50 #define RADIOLIB_LORAWAN_FPORT_PAYLOAD_MIN (0x01 << 0) // 7 0 start of user-allowed fPort range
51 #define RADIOLIB_LORAWAN_FPORT_PAYLOAD_MAX (0xDF << 0) // 7 0 end of user-allowed fPort range
52 #define RADIOLIB_LORAWAN_FPORT_TS009 (0xE0 << 0) // 7 0 fPort used for TS009 testing
53 #define RADIOLIB_LORAWAN_FPORT_TS011 (0xE2 << 0) // 7 0 fPort used for TS011 Forwarding
54 #define RADIOLIB_LORAWAN_FPORT_RESERVED (0xE0 << 0) // 7 0 fPort values equal to and larger than this are reserved
55 
56 // data rate encoding
57 #define RADIOLIB_LORAWAN_DATA_RATE_MODEM (0x03 << 6) // 7 6 modem mask
58 #define RADIOLIB_LORAWAN_DATA_RATE_LORA (0x00 << 6) // 7 6 use LoRa modem
59 #define RADIOLIB_LORAWAN_DATA_RATE_FSK (0x01 << 6) // 7 6 use FSK modem
60 #define RADIOLIB_LORAWAN_DATA_RATE_LR_FHSS (0x02 << 6) // 7 6 use LR-FHSS modem
61 #define RADIOLIB_LORAWAN_DATA_RATE_SF (0x07 << 3) // 5 3 LoRa spreading factor mask
62 #define RADIOLIB_LORAWAN_DATA_RATE_SF_12 (0x05 << 3) // 5 3 LoRa spreading factor: SF12
63 #define RADIOLIB_LORAWAN_DATA_RATE_SF_11 (0x04 << 3) // 5 3 SF11
64 #define RADIOLIB_LORAWAN_DATA_RATE_SF_10 (0x03 << 3) // 5 3 SF10
65 #define RADIOLIB_LORAWAN_DATA_RATE_SF_9 (0x02 << 3) // 5 3 SF9
66 #define RADIOLIB_LORAWAN_DATA_RATE_SF_8 (0x01 << 3) // 5 3 SF8
67 #define RADIOLIB_LORAWAN_DATA_RATE_SF_7 (0x00 << 3) // 5 3 SF7
68 #define RADIOLIB_LORAWAN_DATA_RATE_BW (0x03 << 1) // 2 1 bandwith mask
69 #define RADIOLIB_LORAWAN_DATA_RATE_BW_125_KHZ (0x00 << 1) // 2 1 125 kHz
70 #define RADIOLIB_LORAWAN_DATA_RATE_BW_250_KHZ (0x01 << 1) // 2 1 250 kHz
71 #define RADIOLIB_LORAWAN_DATA_RATE_BW_500_KHZ (0x02 << 1) // 2 1 LoRa bandwidth: 500 kHz
72 #define RADIOLIB_LORAWAN_DATA_RATE_BW_137_KHZ (0x00 << 1) // 2 1 137 kHz
73 #define RADIOLIB_LORAWAN_DATA_RATE_BW_336_KHZ (0x01 << 1) // 2 1 336 kHz
74 #define RADIOLIB_LORAWAN_DATA_RATE_BW_1523_KHZ (0x02 << 1) // 2 1 LR-FHSS bandwidth: 1523 kHz
75 #define RADIOLIB_LORAWAN_DATA_RATE_CR_1_3 (0x00 << 0) // 0 0 LR-FHSS coding rate: 1/3
76 #define RADIOLIB_LORAWAN_DATA_RATE_CR_2_3 (0x01 << 0) // 0 0 2/3
77 #define RADIOLIB_LORAWAN_DATA_RATE_UNUSED (0xFF << 0) // 7 0 unused data rate
78 
79 // channels and channel plans
80 #define RADIOLIB_LORAWAN_UPLINK (0x00 << 0)
81 #define RADIOLIB_LORAWAN_DOWNLINK (0x01 << 0)
82 #define RADIOLIB_LORAWAN_DIR_RX2 (0x02 << 0)
83 #define RADIOLIB_LORAWAN_BAND_DYNAMIC (0)
84 #define RADIOLIB_LORAWAN_BAND_FIXED (1)
85 #define RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES (15)
86 #define RADIOLIB_LORAWAN_CHANNEL_INDEX_NONE (0xFF >> 0)
87 
88 // recommended default settings
89 #define RADIOLIB_LORAWAN_RECEIVE_DELAY_1_MS (1000)
90 #define RADIOLIB_LORAWAN_RECEIVE_DELAY_2_MS ((RADIOLIB_LORAWAN_RECEIVE_DELAY_1_MS) + 1000)
91 #define RADIOLIB_LORAWAN_RX1_DR_OFFSET (0)
92 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DELAY_1_MS (5000)
93 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DELAY_2_MS (6000)
94 #define RADIOLIB_LORAWAN_MAX_FCNT_GAP (16384)
95 #define RADIOLIB_LORAWAN_ADR_ACK_LIMIT_EXP (0x06)
96 #define RADIOLIB_LORAWAN_ADR_ACK_DELAY_EXP (0x05)
97 #define RADIOLIB_LORAWAN_RETRANSMIT_TIMEOUT_MIN_MS (1000)
98 #define RADIOLIB_LORAWAN_RETRANSMIT_TIMEOUT_MAX_MS (3000)
99 #define RADIOLIB_LORAWAN_POWER_STEP_SIZE_DBM (-2)
100 #define RADIOLIB_LORAWAN_REJOIN_MAX_COUNT_N (10) // send rejoin request 16384 uplinks
101 #define RADIOLIB_LORAWAN_REJOIN_MAX_TIME_N (15) // once every year, not actually implemented
102 
103 // join request message layout
104 #define RADIOLIB_LORAWAN_JOIN_REQUEST_LEN (23)
105 #define RADIOLIB_LORAWAN_JOIN_REQUEST_JOIN_EUI_POS (1)
106 #define RADIOLIB_LORAWAN_JOIN_REQUEST_DEV_EUI_POS (9)
107 #define RADIOLIB_LORAWAN_JOIN_REQUEST_DEV_NONCE_POS (17)
108 #define RADIOLIB_LORAWAN_JOIN_REQUEST_TYPE (0xFF)
109 #define RADIOLIB_LORAWAN_JOIN_REQUEST_TYPE_0 (0x00)
110 #define RADIOLIB_LORAWAN_JOIN_REQUEST_TYPE_1 (0x01)
111 #define RADIOLIB_LORAWAN_JOIN_REQUEST_TYPE_2 (0x02)
112 
113 // join accept message layout
114 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_MAX_LEN (33)
115 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_JOIN_NONCE_POS (1)
116 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_HOME_NET_ID_POS (4)
117 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DEV_ADDR_POS (7)
118 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DL_SETTINGS_POS (11)
119 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_RX_DELAY_POS (12)
120 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_POS (13)
121 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_LEN (16)
122 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_TYPE_POS (RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_POS + RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_LEN - 1)
123 
124 // join accept key derivation layout
125 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_AES_JOIN_NONCE_POS (1) // regular keys
126 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_AES_JOIN_EUI_POS (4)
127 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_AES_DEV_NONCE_POS (12)
128 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_AES_DEV_ADDR_POS (1) // relay keys
129 
130 // join accept message variables
131 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_R_1_0 (0x00 << 7) // 7 7 LoRaWAN revision: 1.0
132 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_R_1_1 (0x01 << 7) // 7 7 1.1
133 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_F_NWK_S_INT_KEY (0x01)
134 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_APP_S_KEY (0x02)
135 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_S_NWK_S_INT_KEY (0x03)
136 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_NWK_S_ENC_KEY (0x04)
137 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_JS_ENC_KEY (0x05)
138 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_JS_INT_KEY (0x06)
139 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_ROOT_WOR_S_KEY (0x01)
140 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_WOR_S_INT_KEY (0x01)
141 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_WOR_S_ENC_KEY (0x02)
142 
143 // frame header layout
144 #define RADIOLIB_LORAWAN_FHDR_LEN_START_OFFS (16)
145 #define RADIOLIB_LORAWAN_FHDR_DEV_ADDR_POS (RADIOLIB_LORAWAN_FHDR_LEN_START_OFFS + 1)
146 #define RADIOLIB_LORAWAN_FHDR_FCTRL_POS (RADIOLIB_LORAWAN_FHDR_LEN_START_OFFS + 5)
147 #define RADIOLIB_LORAWAN_FHDR_FCNT_POS (RADIOLIB_LORAWAN_FHDR_LEN_START_OFFS + 6)
148 #define RADIOLIB_LORAWAN_FHDR_FOPTS_POS (RADIOLIB_LORAWAN_FHDR_LEN_START_OFFS + 8)
149 #define RADIOLIB_LORAWAN_FHDR_FOPTS_LEN_MASK (0x0F)
150 #define RADIOLIB_LORAWAN_FHDR_FOPTS_MAX_LEN (15)
151 #define RADIOLIB_LORAWAN_FHDR_FPORT_POS(FOPTS) (RADIOLIB_LORAWAN_FHDR_LEN_START_OFFS + 8 + (FOPTS))
152 #define RADIOLIB_LORAWAN_FRAME_PAYLOAD_POS(FOPTS) (RADIOLIB_LORAWAN_FHDR_LEN_START_OFFS + 9 + (FOPTS))
153 #define RADIOLIB_LORAWAN_FRAME_LEN(PAYLOAD, FOPTS) (16 + 13 + (PAYLOAD) + (FOPTS))
154 
155 // payload encryption/MIC blocks common layout
156 #define RADIOLIB_LORAWAN_BLOCK_MAGIC_POS (0)
157 #define RADIOLIB_LORAWAN_BLOCK_CONF_FCNT_POS (1)
158 #define RADIOLIB_LORAWAN_BLOCK_DIR_POS (5)
159 #define RADIOLIB_LORAWAN_BLOCK_DEV_ADDR_POS (6)
160 #define RADIOLIB_LORAWAN_BLOCK_FCNT_POS (10)
161 
162 // payload encryption block layout
163 #define RADIOLIB_LORAWAN_ENC_BLOCK_MAGIC (0x01)
164 #define RADIOLIB_LORAWAN_ENC_BLOCK_COUNTER_ID_POS (4)
165 #define RADIOLIB_LORAWAN_ENC_BLOCK_COUNTER_POS (15)
166 
167 // payload MIC blocks layout
168 #define RADIOLIB_LORAWAN_MIC_BLOCK_MAGIC (0x49)
169 #define RADIOLIB_LORAWAN_MIC_BLOCK_LEN_POS (15)
170 #define RADIOLIB_LORAWAN_MIC_DATA_RATE_POS (3)
171 #define RADIOLIB_LORAWAN_MIC_CH_INDEX_POS (4)
172 
173 // maximum allowed dwell time on bands that implement dwell time limitations
174 #define RADIOLIB_LORAWAN_DWELL_TIME (400)
175 
176 // unused frame counter value
177 #define RADIOLIB_LORAWAN_FCNT_NONE (0xFFFFFFFF)
178 
179 // TR013 CSMA recommended values
180 #define RADIOLIB_LORAWAN_DIFS_DEFAULT (2)
181 #define RADIOLIB_LORAWAN_BACKOFF_MAX_DEFAULT (6)
182 #define RADIOLIB_LORAWAN_MAX_CHANGES_DEFAULT (4)
183 
184 // MAC commands
185 #define RADIOLIB_LORAWAN_NUM_MAC_COMMANDS (23)
186 
187 #define RADIOLIB_LORAWAN_MAC_RESET (0x01)
188 #define RADIOLIB_LORAWAN_MAC_LINK_CHECK (0x02)
189 #define RADIOLIB_LORAWAN_MAC_LINK_ADR (0x03)
190 #define RADIOLIB_LORAWAN_MAC_DUTY_CYCLE (0x04)
191 #define RADIOLIB_LORAWAN_MAC_RX_PARAM_SETUP (0x05)
192 #define RADIOLIB_LORAWAN_MAC_DEV_STATUS (0x06)
193 #define RADIOLIB_LORAWAN_MAC_NEW_CHANNEL (0x07)
194 #define RADIOLIB_LORAWAN_MAC_RX_TIMING_SETUP (0x08)
195 #define RADIOLIB_LORAWAN_MAC_TX_PARAM_SETUP (0x09)
196 #define RADIOLIB_LORAWAN_MAC_DL_CHANNEL (0x0A)
197 #define RADIOLIB_LORAWAN_MAC_REKEY (0x0B)
198 #define RADIOLIB_LORAWAN_MAC_ADR_PARAM_SETUP (0x0C)
199 #define RADIOLIB_LORAWAN_MAC_DEVICE_TIME (0x0D)
200 #define RADIOLIB_LORAWAN_MAC_FORCE_REJOIN (0x0E)
201 #define RADIOLIB_LORAWAN_MAC_REJOIN_PARAM_SETUP (0x0F)
202 #define RADIOLIB_LORAWAN_MAC_PROPRIETARY (0x80)
203 
204 // the length of internal MAC command queue - hopefully this is enough for most use cases
205 #define RADIOLIB_LORAWAN_MAC_COMMAND_QUEUE_SIZE (9)
206 
207 // the maximum number of simultaneously available channels
208 #define RADIOLIB_LORAWAN_NUM_AVAILABLE_CHANNELS (16)
209 
210 // maximum MAC command sizes
211 #define RADIOLIB_LORAWAN_MAX_MAC_COMMAND_LEN_DOWN (5)
212 #define RADIOLIB_LORAWAN_MAX_MAC_COMMAND_LEN_UP (2)
213 #define RADIOLIB_LORAWAN_MAX_NUM_ADR_COMMANDS (8)
214 
221  const uint8_t cid;
222 
224  const uint8_t lenDn;
225 
227  const uint8_t lenUp;
228 
230  const bool persist;
231 
233  const bool user;
234 };
235 
236 #define RADIOLIB_LORAWAN_MAC_COMMAND_NONE { .cid = 0, .lenDn = 0, .lenUp = 0, .persist = false, .user = false }
237 
238 constexpr LoRaWANMacCommand_t MacTable[RADIOLIB_LORAWAN_NUM_MAC_COMMANDS] = {
239  { RADIOLIB_LORAWAN_MAC_RESET, 1, 1, false, false },
240  { RADIOLIB_LORAWAN_MAC_LINK_CHECK, 2, 0, false, true },
241  { RADIOLIB_LORAWAN_MAC_LINK_ADR, 4, 1, false, false },
242  { RADIOLIB_LORAWAN_MAC_DUTY_CYCLE, 1, 0, false, false },
243  { RADIOLIB_LORAWAN_MAC_RX_PARAM_SETUP, 4, 1, true, false },
244  { RADIOLIB_LORAWAN_MAC_DEV_STATUS, 0, 2, false, false },
245  { RADIOLIB_LORAWAN_MAC_NEW_CHANNEL, 5, 1, false, false },
246  { RADIOLIB_LORAWAN_MAC_RX_TIMING_SETUP, 1, 0, true, false },
247  { RADIOLIB_LORAWAN_MAC_TX_PARAM_SETUP, 1, 0, true, false },
248  { RADIOLIB_LORAWAN_MAC_DL_CHANNEL, 4, 1, true, false },
249  { RADIOLIB_LORAWAN_MAC_REKEY, 1, 1, false, false },
250  { RADIOLIB_LORAWAN_MAC_ADR_PARAM_SETUP, 1, 0, false, false },
251  { RADIOLIB_LORAWAN_MAC_DEVICE_TIME, 5, 0, false, true },
252  { RADIOLIB_LORAWAN_MAC_FORCE_REJOIN, 2, 0, false, false },
253  { RADIOLIB_LORAWAN_MAC_REJOIN_PARAM_SETUP, 1, 1, false, false },
254  { RADIOLIB_LORAWAN_MAC_PROPRIETARY, 5, 0, false, true },
255 };
256 
257 #define RADIOLIB_LORAWAN_NONCES_VERSION_VAL (0x0001)
258 
259 enum LoRaWANSchemeBase_t {
260  RADIOLIB_LORAWAN_NONCES_START = 0x00,
261  RADIOLIB_LORAWAN_NONCES_VERSION = RADIOLIB_LORAWAN_NONCES_START, // 2 bytes
262  RADIOLIB_LORAWAN_NONCES_MODE = RADIOLIB_LORAWAN_NONCES_VERSION + sizeof(uint16_t), // 2 bytes
263  RADIOLIB_LORAWAN_NONCES_CLASS = RADIOLIB_LORAWAN_NONCES_MODE + sizeof(uint16_t), // 1 byte
264  RADIOLIB_LORAWAN_NONCES_PLAN = RADIOLIB_LORAWAN_NONCES_CLASS + sizeof(uint8_t), // 1 byte
265  RADIOLIB_LORAWAN_NONCES_CHECKSUM = RADIOLIB_LORAWAN_NONCES_PLAN + sizeof(uint8_t), // 2 bytes
266  RADIOLIB_LORAWAN_NONCES_DEV_NONCE = RADIOLIB_LORAWAN_NONCES_CHECKSUM + sizeof(uint16_t), // 2 bytes
267  RADIOLIB_LORAWAN_NONCES_JOIN_NONCE = RADIOLIB_LORAWAN_NONCES_DEV_NONCE + sizeof(uint16_t), // 3 bytes
268  RADIOLIB_LORAWAN_NONCES_ACTIVE = RADIOLIB_LORAWAN_NONCES_JOIN_NONCE + 3, // 1 byte
269  RADIOLIB_LORAWAN_NONCES_SIGNATURE = RADIOLIB_LORAWAN_NONCES_ACTIVE + sizeof(uint8_t), // 2 bytes
270  RADIOLIB_LORAWAN_NONCES_BUF_SIZE = RADIOLIB_LORAWAN_NONCES_SIGNATURE + sizeof(uint16_t) // Nonces buffer size
271 };
272 
273 enum LoRaWANSchemeSession_t {
274  RADIOLIB_LORAWAN_SESSION_START = 0x00,
275  RADIOLIB_LORAWAN_SESSION_NWK_SENC_KEY = RADIOLIB_LORAWAN_SESSION_START, // 16 bytes
276  RADIOLIB_LORAWAN_SESSION_APP_SKEY = RADIOLIB_LORAWAN_SESSION_NWK_SENC_KEY + RADIOLIB_AES128_KEY_SIZE, // 16 bytes
277  RADIOLIB_LORAWAN_SESSION_FNWK_SINT_KEY = RADIOLIB_LORAWAN_SESSION_APP_SKEY + RADIOLIB_AES128_KEY_SIZE, // 16 bytes
278  RADIOLIB_LORAWAN_SESSION_SNWK_SINT_KEY = RADIOLIB_LORAWAN_SESSION_FNWK_SINT_KEY + RADIOLIB_AES128_KEY_SIZE, // 16 bytes
279  RADIOLIB_LORAWAN_SESSION_DEV_ADDR = RADIOLIB_LORAWAN_SESSION_SNWK_SINT_KEY + RADIOLIB_AES128_KEY_SIZE, // 4 bytes
280  RADIOLIB_LORAWAN_SESSION_NONCES_SIGNATURE = RADIOLIB_LORAWAN_SESSION_DEV_ADDR + sizeof(uint32_t), // 2 bytes
281  RADIOLIB_LORAWAN_SESSION_A_FCNT_DOWN = RADIOLIB_LORAWAN_SESSION_NONCES_SIGNATURE + sizeof(uint16_t), // 4 bytes
282  RADIOLIB_LORAWAN_SESSION_CONF_FCNT_UP = RADIOLIB_LORAWAN_SESSION_A_FCNT_DOWN + sizeof(uint32_t), // 4 bytes
283  RADIOLIB_LORAWAN_SESSION_CONF_FCNT_DOWN = RADIOLIB_LORAWAN_SESSION_CONF_FCNT_UP + sizeof(uint32_t), // 4 bytes
284  RADIOLIB_LORAWAN_SESSION_RJ_COUNT0 = RADIOLIB_LORAWAN_SESSION_CONF_FCNT_DOWN + sizeof(uint32_t), // 2 bytes
285  RADIOLIB_LORAWAN_SESSION_RJ_COUNT1 = RADIOLIB_LORAWAN_SESSION_RJ_COUNT0 + sizeof(uint16_t), // 2 bytes
286  RADIOLIB_LORAWAN_SESSION_HOMENET_ID = RADIOLIB_LORAWAN_SESSION_RJ_COUNT1 + sizeof(uint16_t), // 4 bytes
287  RADIOLIB_LORAWAN_SESSION_VERSION = RADIOLIB_LORAWAN_SESSION_HOMENET_ID + sizeof(uint32_t), // 1 byte
288  RADIOLIB_LORAWAN_SESSION_DUTY_CYCLE = RADIOLIB_LORAWAN_SESSION_VERSION + sizeof(uint8_t), // 1 byte
289  RADIOLIB_LORAWAN_SESSION_RX_PARAM_SETUP = RADIOLIB_LORAWAN_SESSION_DUTY_CYCLE + 1, // 4 bytes
290  RADIOLIB_LORAWAN_SESSION_RX_TIMING_SETUP = RADIOLIB_LORAWAN_SESSION_RX_PARAM_SETUP + 4, // 1 byte
291  RADIOLIB_LORAWAN_SESSION_TX_PARAM_SETUP = RADIOLIB_LORAWAN_SESSION_RX_TIMING_SETUP + 1, // 1 byte
292  RADIOLIB_LORAWAN_SESSION_ADR_PARAM_SETUP = RADIOLIB_LORAWAN_SESSION_TX_PARAM_SETUP + 1, // 1 byte
293  RADIOLIB_LORAWAN_SESSION_REJOIN_PARAM_SETUP = RADIOLIB_LORAWAN_SESSION_ADR_PARAM_SETUP + 1, // 1 byte
294  RADIOLIB_LORAWAN_SESSION_BEACON_FREQ = RADIOLIB_LORAWAN_SESSION_REJOIN_PARAM_SETUP + 1, // 3 bytes
295  RADIOLIB_LORAWAN_SESSION_PING_SLOT_CHANNEL = RADIOLIB_LORAWAN_SESSION_BEACON_FREQ + 3, // 4 bytes
296  RADIOLIB_LORAWAN_SESSION_PERIODICITY = RADIOLIB_LORAWAN_SESSION_PING_SLOT_CHANNEL + 4, // 1 byte
297  RADIOLIB_LORAWAN_SESSION_LAST_TIME = RADIOLIB_LORAWAN_SESSION_PERIODICITY + 1, // 4 bytes
298  RADIOLIB_LORAWAN_SESSION_UL_CHANNELS = RADIOLIB_LORAWAN_SESSION_LAST_TIME + 4, // 16*5 bytes
299  RADIOLIB_LORAWAN_SESSION_DL_CHANNELS = RADIOLIB_LORAWAN_SESSION_UL_CHANNELS + RADIOLIB_LORAWAN_NUM_AVAILABLE_CHANNELS*5, // 16*4 bytes
300  RADIOLIB_LORAWAN_SESSION_MAC_QUEUE = RADIOLIB_LORAWAN_SESSION_DL_CHANNELS + RADIOLIB_LORAWAN_NUM_AVAILABLE_CHANNELS*4, // 15 bytes
301  RADIOLIB_LORAWAN_SESSION_MAC_QUEUE_LEN = RADIOLIB_LORAWAN_SESSION_MAC_QUEUE + 1, // 1 byte
302  RADIOLIB_LORAWAN_SESSION_N_FCNT_DOWN = RADIOLIB_LORAWAN_SESSION_MAC_QUEUE_LEN + RADIOLIB_LORAWAN_FHDR_FOPTS_MAX_LEN, // 4 bytes
303  RADIOLIB_LORAWAN_SESSION_ADR_FCNT = RADIOLIB_LORAWAN_SESSION_N_FCNT_DOWN + sizeof(uint32_t), // 4 bytes
304  RADIOLIB_LORAWAN_SESSION_LINK_ADR = RADIOLIB_LORAWAN_SESSION_ADR_FCNT + sizeof(uint32_t), // 14 bytes
305  RADIOLIB_LORAWAN_SESSION_AVAILABLE_CHANNELS = RADIOLIB_LORAWAN_SESSION_LINK_ADR + 14, // 2 bytes
306  RADIOLIB_LORAWAN_SESSION_FCNT_UP = RADIOLIB_LORAWAN_SESSION_AVAILABLE_CHANNELS + 2, // 4 bytes
307  RADIOLIB_LORAWAN_SESSION_SIGNATURE = RADIOLIB_LORAWAN_SESSION_FCNT_UP + sizeof(uint32_t), // 2 bytes
308  RADIOLIB_LORAWAN_SESSION_BUF_SIZE = RADIOLIB_LORAWAN_SESSION_SIGNATURE + sizeof(uint16_t) // Session buffer size
309 };
310 
318  bool enabled;
319 
321  uint8_t idx;
322 
324  uint32_t freq;
325 
327  uint8_t drMin;
328 
330  uint8_t drMax;
331 
333  uint8_t dr;
334 
336  bool available;
337 };
338 
339 // alias for unused channel
340 #define RADIOLIB_LORAWAN_CHANNEL_NONE { .enabled = false, .idx = RADIOLIB_LORAWAN_CHANNEL_INDEX_NONE, .freq = 0, \
341  .drMin = 0, .drMax = 0, .dr = RADIOLIB_LORAWAN_DATA_RATE_UNUSED, .available = false }
342 
350  uint8_t numChannels;
351 
353  uint32_t freqStart;
354 
356  uint32_t freqStep;
357 
359  uint8_t drMin;
360 
362  uint8_t drMax;
363 
365  uint8_t drJoinRequest;
366 };
367 
368 // alias for unused channel span
369 #define RADIOLIB_LORAWAN_CHANNEL_SPAN_NONE { .numChannels = 0, .freqStart = 0, .freqStep = 0, .drMin = 0, .drMax = 0, .drJoinRequest = RADIOLIB_LORAWAN_DATA_RATE_UNUSED }
370 
377  uint8_t bandNum;
378 
380  uint8_t bandType;
381 
383  uint32_t freqMin;
384 
386  uint32_t freqMax;
387 
389  uint8_t payloadLenMax[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES];
390 
392  int8_t powerMax;
393 
396 
399 
402 
405 
408 
411 
414 
416  uint8_t numTxSpans;
417 
420 
423 
424  uint8_t rx1DrTable[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES][8];
425 
428 
431 
434 
436  uint8_t dataRates[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES];
437 };
438 
439 // supported bands
440 extern const LoRaWANBand_t EU868;
441 extern const LoRaWANBand_t US915;
442 extern const LoRaWANBand_t EU433;
443 extern const LoRaWANBand_t AU915;
444 extern const LoRaWANBand_t CN500;
445 extern const LoRaWANBand_t AS923;
446 extern const LoRaWANBand_t AS923_2;
447 extern const LoRaWANBand_t AS923_3;
448 extern const LoRaWANBand_t AS923_4;
449 extern const LoRaWANBand_t KR920;
450 extern const LoRaWANBand_t IN865;
451 
456 enum LoRaWANBandNum_t {
457  BandEU868,
458  BandUS915,
459  BandEU433,
460  BandAU915,
461  BandCN500,
462  BandAS923,
463  BandAS923_2,
464  BandAS923_3,
465  BandAS923_4,
466  BandKR920,
467  BandIN865,
468  BandLast
469 };
470 
471 // provide easy access to the number of currently supported bands
472 #define RADIOLIB_LORAWAN_NUM_SUPPORTED_BANDS (BandLast - BandEU868)
473 
474 // array of currently supported bands
475 extern const LoRaWANBand_t* LoRaWANBands[];
476 
483  bool newSession = false;
484 
486  uint16_t devNonce = 0;
487 
489  uint32_t joinNonce = 0;
490 };
491 
498  uint8_t dir;
499 
501  bool confirmed;
502 
506 
508  uint8_t datarate;
509 
511  float freq;
512 
514  int16_t power;
515 
517  uint32_t fCnt;
518 
520  uint8_t fPort;
521 
523  uint8_t nbTrans;
524 };
525 
530 class LoRaWANNode {
531  public:
532 
539  LoRaWANNode(PhysicalLayer* phy, const LoRaWANBand_t* band, uint8_t subBand = 0);
540 
545  uint8_t* getBufferNonces();
546 
552  int16_t setBufferNonces(uint8_t* persistentBuffer);
553 
557  void clearSession();
558 
563  uint8_t* getBufferSession();
564 
570  int16_t setBufferSession(uint8_t* persistentBuffer);
571 
580  int16_t beginOTAA(uint64_t joinEUI, uint64_t devEUI, uint8_t* nwkKey, uint8_t* appKey);
581 
592  int16_t beginABP(uint32_t addr, uint8_t* fNwkSIntKey, uint8_t* sNwkSIntKey, uint8_t* nwkSEncKey, uint8_t* appSKey);
593 
600  virtual int16_t activateOTAA(uint8_t initialDr = RADIOLIB_LORAWAN_DATA_RATE_UNUSED, LoRaWANJoinEvent_t *joinEvent = NULL);
601 
608  int16_t activateABP(uint8_t initialDr = RADIOLIB_LORAWAN_DATA_RATE_UNUSED);
609 
611  bool isActivated();
612 
613  #if defined(RADIOLIB_BUILD_ARDUINO)
626  virtual int16_t sendReceive(String& strUp, uint8_t fPort, String& strDown, bool isConfirmed = false, LoRaWANEvent_t* eventUp = NULL, LoRaWANEvent_t* eventDown = NULL);
627  #endif
628 
640  virtual int16_t sendReceive(const char* strUp, uint8_t fPort, bool isConfirmed = false, LoRaWANEvent_t* eventUp = NULL, LoRaWANEvent_t* eventDown = NULL);
641 
655  virtual 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);
656 
669  virtual int16_t sendReceive(uint8_t* dataUp, size_t lenUp, uint8_t fPort = 1, bool isConfirmed = false, LoRaWANEvent_t* eventUp = NULL, LoRaWANEvent_t* eventDown = NULL);
670 
685  virtual 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);
686 
694  int16_t sendMacCommandReq(uint8_t cid);
695 
704  int16_t getMacLinkCheckAns(uint8_t* margin, uint8_t* gwCnt);
705 
715  int16_t getMacDeviceTimeAns(uint32_t* gpsEpoch, uint8_t* fraction, bool returnUnix = true);
716 
722  int16_t setDatarate(uint8_t drUp);
723 
729  int16_t setTxPower(int8_t txPower);
730 
737  int16_t setRx2Dr(uint8_t dr);
738 
743  void setADR(bool enable = true);
744 
751  void setDutyCycle(bool enable = true, RadioLibTime_t msPerHour = 0);
752 
759  void setDwellTime(bool enable, RadioLibTime_t msPerUplink = 0);
760 
768  void setCSMA(bool csmaEnabled, uint8_t maxChanges = 4, uint8_t backoffMax = 0, uint8_t difsSlots = 2);
769 
775  void setDeviceStatus(uint8_t battLevel);
776 
784  void scheduleTransmission(RadioLibTime_t tUplink);
785 
790  uint32_t getFCntUp();
791 
796  uint32_t getNFCntDown();
797 
802  uint32_t getAFCntDown();
803 
809  void resetFCntDown();
810 
815  uint32_t getDevAddr();
816 
822 
831 
834 
839  uint8_t getMaxPayloadLen();
840 
845  bool TS009 = false;
846 
860 
861 #if !RADIOLIB_GODMODE
862  protected:
863 #endif
864  PhysicalLayer* phyLayer = NULL;
865  const LoRaWANBand_t* band = NULL;
866 
867  // a buffer that holds all LW base parameters that should persist at all times!
868  uint8_t bufferNonces[RADIOLIB_LORAWAN_NONCES_BUF_SIZE] = { 0 };
869 
870  // a buffer that holds all LW session parameters that preferably persist, but can be afforded to get lost
871  uint8_t bufferSession[RADIOLIB_LORAWAN_SESSION_BUF_SIZE] = { 0 };
872 
873  uint8_t fOptsUp[RADIOLIB_LORAWAN_FHDR_FOPTS_MAX_LEN] = { 0 };
874  uint8_t fOptsDown[RADIOLIB_LORAWAN_FHDR_FOPTS_MAX_LEN] = { 0 };
875  uint8_t fOptsUpLen = 0;
876  uint8_t fOptsDownLen = 0;
877 
878  uint16_t lwMode = RADIOLIB_LORAWAN_MODE_NONE;
879  uint8_t lwClass = RADIOLIB_LORAWAN_CLASS_A;
880  bool isActive = false;
881 
882  uint64_t joinEUI = 0;
883  uint64_t devEUI = 0;
884  uint8_t nwkKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
885  uint8_t appKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
886 
887  // the following is either provided by the network server (OTAA)
888  // or directly entered by the user (ABP)
889  uint32_t devAddr = 0;
890  uint8_t appSKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
891  uint8_t fNwkSIntKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
892  uint8_t sNwkSIntKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
893  uint8_t nwkSEncKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
894  uint8_t jSIntKey[RADIOLIB_AES128_KEY_SIZE] = { 0 };
895 
896  uint16_t keyCheckSum = 0;
897 
898  // device-specific parameters, persistent through sessions
899  uint16_t devNonce = 0;
900  uint32_t joinNonce = 0;
901 
902  // session-specific parameters
903  uint32_t homeNetId = 0;
904  uint8_t adrLimitExp = RADIOLIB_LORAWAN_ADR_ACK_LIMIT_EXP;
905  uint8_t adrDelayExp = RADIOLIB_LORAWAN_ADR_ACK_DELAY_EXP;
906  uint8_t nbTrans = 1; // Number of allowed frame retransmissions
907  uint8_t txPowerSteps = 0;
908  uint8_t txPowerMax = 0;
909  uint32_t fCntUp = 0;
910  uint32_t aFCntDown = 0;
911  uint32_t nFCntDown = 0;
912  uint32_t confFCntUp = RADIOLIB_LORAWAN_FCNT_NONE;
913  uint32_t confFCntDown = RADIOLIB_LORAWAN_FCNT_NONE;
914  uint32_t adrFCnt = 0;
915 
916  // modulation of the currently configured channel
917  uint8_t modulation = RADIOLIB_LORAWAN_MODULATION_LORA;
918 
919  // ADR is enabled by default
920  bool adrEnabled = true;
921  bool adrAckReq = false;
922 
923  // duty cycle is set upon initialization and activated in regions that impose this
924  bool dutyCycleEnabled = false;
925  uint32_t dutyCycle = 0;
926 
927  // dwell time is set upon initialization and activated in regions that impose this
928  bool dwellTimeEnabledUp = false;
929  uint16_t dwellTimeUp = 0;
930  bool dwellTimeEnabledDn = false;
931  uint16_t dwellTimeDn = 0;
932 
933  RadioLibTime_t tUplink = 0; // scheduled uplink transmission time (internal clock)
934  RadioLibTime_t tDownlink = 0; // time at end of downlink reception
935 
936  // enable/disable CSMA for LoRaWAN
937  bool csmaEnabled = false;
938 
939  // maximum number of channel hops during CSMA
940  uint8_t maxChanges = RADIOLIB_LORAWAN_MAX_CHANGES_DEFAULT;
941 
942  // number of backoff slots to be checked after DIFS phase.
943  // A random BO avoids collisions in the case where two or more nodes start the CSMA
944  // process at the same time.
945  uint8_t backoffMax = RADIOLIB_LORAWAN_BACKOFF_MAX_DEFAULT;
946 
947  // number of CADs to estimate a clear CH
948  uint8_t difsSlots = RADIOLIB_LORAWAN_DIFS_DEFAULT;
949 
950  // available channel frequencies from list passed during OTA activation
951  LoRaWANChannel_t channelPlan[2][RADIOLIB_LORAWAN_NUM_AVAILABLE_CHANNELS];
952 
953  // currently configured channels for TX, RX1, RX2
954  LoRaWANChannel_t channels[3] = { RADIOLIB_LORAWAN_CHANNEL_NONE, RADIOLIB_LORAWAN_CHANNEL_NONE,
955  RADIOLIB_LORAWAN_CHANNEL_NONE };
956 
957  // delays between the uplink and RX1/2 windows
958  // the first field is meaningless, but is used for offsetting for Rx windows 1 and 2
959  RadioLibTime_t rxDelays[3] = { 0, RADIOLIB_LORAWAN_RECEIVE_DELAY_1_MS, RADIOLIB_LORAWAN_RECEIVE_DELAY_2_MS };
960 
961  // offset between TX and RX1 (such that RX1 has equal or lower DR)
962  uint8_t rx1DrOffset = 0;
963 
964  // LoRaWAN revision (1.0 vs 1.1)
965  uint8_t rev = 0;
966 
967  // Time on Air of last uplink
968  RadioLibTime_t lastToA = 0;
969 
970  // timestamp to measure the RX1/2 delay (from uplink end)
971  RadioLibTime_t rxDelayStart = 0;
972 
973  // timestamp when the Rx1/2 windows were closed (timeout or uplink received)
974  RadioLibTime_t rxDelayEnd = 0;
975 
976  // device status - battery level
977  uint8_t battLevel = 0xFF;
978 
979  // indicates whether an uplink has MAC commands as payload
980  bool isMACPayload = false;
981 
982  // save the selected sub-band in case this must be restored in ADR control
983  uint8_t subBand = 0;
984 
985  // allow port 226 for devices implementing TS011
986  bool TS011 = false;
987 
988  // this will reset the device credentials, so the device starts completely new
989  void clearNonces();
990 
991  // start a fresh session using default parameters
992  void createSession(uint16_t lwMode, uint8_t initialDr);
993 
994  // setup Join-Request payload
995  void composeJoinRequest(uint8_t* joinRequestMsg);
996 
997  // extract Join-Accept payload and start a new session
998  int16_t processJoinAccept(LoRaWANJoinEvent_t *joinEvent);
999 
1000  // a join-accept can piggy-back a set of channels or channel masks
1001  void processCFList(uint8_t* cfList);
1002 
1003  // check whether payload length and fport are allowed
1004  int16_t isValidUplink(uint8_t* len, uint8_t fPort);
1005 
1006  // perform ADR backoff
1007  void adrBackoff();
1008 
1009  // create an encrypted uplink buffer, composing metadata, user data and MAC data
1010  void composeUplink(uint8_t* in, uint8_t lenIn, uint8_t* out, uint8_t fPort, bool isConfirmed);
1011 
1012  // generate and set the MIC of an uplink buffer (depends on selected channels)
1013  void micUplink(uint8_t* inOut, uint8_t lenInOut);
1014 
1015  // transmit uplink buffer on a specified channel
1016  int16_t transmitUplink(LoRaWANChannel_t* chnl, uint8_t* in, uint8_t len, bool retrans);
1017 
1018  // wait for, open and listen during receive windows; only performs listening
1019  int16_t receiveCommon(uint8_t dir, const LoRaWANChannel_t* dlChannels, const RadioLibTime_t* dlDelays, uint8_t numWindows, RadioLibTime_t tReference);
1020 
1021  // extract downlink payload and process MAC commands
1022  int16_t parseDownlink(uint8_t* data, size_t* len, LoRaWANEvent_t* event = NULL);
1023 
1024  // execute mac command, return the number of processed bytes for sequential processing
1025  bool execMacCommand(uint8_t cid, uint8_t* optIn, uint8_t lenIn);
1026  bool execMacCommand(uint8_t cid, uint8_t* optIn, uint8_t lenIn, uint8_t* optOut);
1027 
1028  // possible override for additional MAC commands that are not in the base specification
1029  virtual bool derivedMacHandler(uint8_t cid, uint8_t* optIn, uint8_t lenIn, uint8_t* optOut);
1030 
1031  // pre-process a (set of) LinkAdrReq commands into one super-channel-mask + Tx/Dr/NbTrans fields
1032  void preprocessMacLinkAdr(uint8_t* mPtr, uint8_t cLen, uint8_t* mAdrOpt);
1033 
1034  // post-process a (set of) LinkAdrAns commands depending on LoRaWAN version
1035  void postprocessMacLinkAdr(uint8_t* ack, uint8_t cLen);
1036 
1037  // get the properties of a MAC command given a certain command ID
1038  int16_t getMacCommand(uint8_t cid, LoRaWANMacCommand_t* cmd);
1039 
1040  // possible override for additional MAC commands that are not in the base specification
1041  virtual int16_t derivedMacFinder(uint8_t cid, LoRaWANMacCommand_t* cmd);
1042 
1043  // get the length of a certain MAC command in a specific direction (up/down)
1044  // if inclusive is true, add one for the CID byte
1045  int16_t getMacLen(uint8_t cid, uint8_t* len, uint8_t dir, bool inclusive = false);
1046 
1047  // find out of a MAC command should persist destruction
1048  // in uplink direction, some commands must persist if no downlink is received
1049  // in downlink direction, the user-accessible MAC commands remain available for retrieval
1050  bool isPersistentMacCommand(uint8_t cid, uint8_t dir);
1051 
1052  // push MAC command to queue, done by copy
1053  int16_t pushMacCommand(uint8_t cid, uint8_t* cOcts, uint8_t* out, uint8_t* lenOut, uint8_t dir);
1054 
1055  // retrieve the payload of a certain MAC command, if present in the buffer
1056  int16_t getMacPayload(uint8_t cid, uint8_t* in, uint8_t lenIn, uint8_t* out, uint8_t dir);
1057 
1058  // delete a specific MAC command from queue, indicated by the command ID
1059  int16_t deleteMacCommand(uint8_t cid, uint8_t* inOut, uint8_t* lenInOut, uint8_t dir);
1060 
1061  // clear a MAC buffer, possible retaining persistent MAC commands
1062  void clearMacCommands(uint8_t* inOut, uint8_t* lenInOut, uint8_t dir);
1063 
1064  // configure the common physical layer properties (frequency, sync word etc.)
1065  int16_t setPhyProperties(const LoRaWANChannel_t* chnl, uint8_t dir, int8_t pwr, size_t pre = 0);
1066 
1067  // Performs CSMA as per LoRa Alliance Technical Recommendation 13 (TR-013).
1068  bool csmaChannelClear(uint8_t difs, uint8_t numBackoff);
1069 
1070  // perform a single CAD operation for the under SF/CH combination. Returns either busy or otherwise.
1071  bool cadChannelClear();
1072 
1073  // get or create a complete 80-bit channel mask for current configuration
1074  void getChannelPlanMask(uint64_t* chMaskGrp0123, uint32_t* chMaskGrp45);
1075 
1076  // setup uplink/downlink channel data rates and frequencies
1077  // for dynamic channels, there is a small set of predefined channels
1078  // in case of JoinRequest, add some optional extra frequencies
1079  void selectChannelPlanDyn(bool joinRequest = false);
1080 
1081  // setup uplink/downlink channel data rates and frequencies
1082  // for fixed bands, we only allow one sub-band at a time to be selected
1083  void selectChannelPlanFix();
1084 
1085  // get the number of available channels,
1086  // along with a 16-bit mask indicating which channels can be used next for uplink/downlink
1087  uint8_t getAvailableChannels(uint16_t* mask);
1088 
1089  // (re)set/restore which channels can be used next for uplink/downlink
1090  void setAvailableChannels(uint16_t mask);
1091 
1092  // select a set of random TX/RX channels for up- and downlink
1093  int16_t selectChannels();
1094 
1095  // apply a 96-bit channel mask
1096  bool applyChannelMask(uint64_t chMaskGrp0123, uint32_t chMaskGrp45);
1097 
1098 #if RADIOLIB_DEBUG_PROTOCOL
1099  // print the available channels through debug
1100  void printChannels();
1101 #endif
1102 
1103  // method to generate message integrity code
1104  uint32_t generateMIC(uint8_t* msg, size_t len, uint8_t* key);
1105 
1106  // method to verify message integrity code
1107  // it assumes that the MIC is the last 4 bytes of the message
1108  bool verifyMIC(uint8_t* msg, size_t len, uint8_t* key);
1109 
1110  // find the first usable data rate for the given band
1111  int16_t findDataRate(uint8_t dr, DataRate_t* dataRate);
1112 
1113  // function to encrypt and decrypt payloads (regular uplink/downlink)
1114  void processAES(const uint8_t* in, size_t len, uint8_t* key, uint8_t* out, uint32_t fCnt, uint8_t dir, uint8_t ctrId, bool counter);
1115 
1116  // 16-bit checksum method that takes a uint8_t array of even length and calculates the checksum
1117  static uint16_t checkSum16(const uint8_t *key, uint16_t keyLen);
1118 
1119  // check the integrity of a buffer using a 16-bit checksum located in the last two bytes of the buffer
1120  static int16_t checkBufferCommon(uint8_t *buffer, uint16_t size);
1121 
1122  // network-to-host conversion method - takes data from network packet and converts it to the host endians
1123  template<typename T>
1124  static T ntoh(uint8_t* buff, size_t size = 0);
1125 
1126  // host-to-network conversion method - takes data from host variable and and converts it to network packet endians
1127  template<typename T>
1128  static void hton(uint8_t* buff, T val, size_t size = 0);
1129 };
1130 
1131 #endif
LoRaWAN-compatible node (class A device).
Definition: LoRaWAN.h:530
void clearSession()
Clear an active session, so that the device will have to rejoin the network.
Definition: LoRaWAN.cpp:281
int16_t setBufferNonces(uint8_t *persistentBuffer)
Fill the internal buffer that holds the LW base parameters with a supplied buffer.
Definition: LoRaWAN.cpp:245
virtual int16_t sendReceive(const char *strUp, uint8_t fPort, 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:45
int16_t setDatarate(uint8_t drUp)
Set uplink datarate. This should not be used when ADR is enabled.
Definition: LoRaWAN.cpp:2530
void scheduleTransmission(RadioLibTime_t tUplink)
Set the exact time a transmission should occur. Note: this is the internal clock time....
Definition: LoRaWAN.cpp:2661
RadioLibTime_t scanGuard
Rx window padding in milliseconds according to the spec, the Rx window must be at least enough time t...
Definition: LoRaWAN.h:859
void setCSMA(bool csmaEnabled, uint8_t maxChanges=4, uint8_t backoffMax=0, uint8_t difsSlots=2)
Configures CSMA for LoRaWAN as per TR013, LoRa Alliance.
Definition: LoRaWAN.cpp:2643
void setDutyCycle(bool enable=true, RadioLibTime_t msPerHour=0)
Toggle adherence to dutyCycle limits to on or off.
Definition: LoRaWAN.cpp:2620
int16_t beginABP(uint32_t addr, uint8_t *fNwkSIntKey, uint8_t *sNwkSIntKey, uint8_t *nwkSEncKey, uint8_t *appSKey)
Set the device credentials and activation configuration.
Definition: LoRaWAN.cpp:613
RadioLibTime_t timeUntilUplink()
Returns time in milliseconds until next uplink is available under dutyCycle limits.
Definition: LoRaWAN.cpp:3157
void setDeviceStatus(uint8_t battLevel)
Set device status.
Definition: LoRaWAN.cpp:2657
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:3147
void setDwellTime(bool enable, RadioLibTime_t msPerUplink=0)
Toggle adherence to dwellTime limits to on or off.
Definition: LoRaWAN.cpp:2632
uint32_t getAFCntDown()
Returns the last application downlink's frame counter; also 0 if no application downlink occured yet.
Definition: LoRaWAN.cpp:2677
RadioLibTime_t getLastToA()
Get the Time-on-air of the last uplink message (in milliseconds).
Definition: LoRaWAN.cpp:2690
int16_t activateABP(uint8_t initialDr=RADIOLIB_LORAWAN_DATA_RATE_UNUSED)
Join network by restoring ABP session or performing over-the-air activation. In this procedure,...
Definition: LoRaWAN.cpp:954
bool TS009
TS009 Protocol Specification Verification switch (allows FPort 224 and cuts off uplink payload instea...
Definition: LoRaWAN.h:845
uint8_t * getBufferNonces()
Returns the pointer to the internal buffer that holds the LW base parameters.
Definition: LoRaWAN.cpp:230
uint8_t * getBufferSession()
Returns the pointer to the internal buffer that holds the LW session parameters.
Definition: LoRaWAN.cpp:436
bool isActivated()
Whether there is an ongoing session active.
Definition: LoRaWAN.cpp:1044
int16_t setTxPower(int8_t txPower)
Configure TX power of the radio module.
Definition: LoRaWAN.cpp:2563
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:2381
uint32_t getNFCntDown()
Returns the last network downlink's frame counter; also 0 if no network downlink occured yet.
Definition: LoRaWAN.cpp:2673
int16_t setBufferSession(uint8_t *persistentBuffer)
Fill the internal buffer that holds the LW session parameters with a supplied buffer.
Definition: LoRaWAN.cpp:460
LoRaWANNode(PhysicalLayer *phy, const LoRaWANBand_t *band, uint8_t subBand=0)
Default constructor.
Definition: LoRaWAN.cpp:9
uint8_t getMaxPayloadLen()
Returns the maximum allowed uplink payload size given the current MAC state. Most importantly,...
Definition: LoRaWAN.cpp:3166
virtual int16_t activateOTAA(uint8_t initialDr=RADIOLIB_LORAWAN_DATA_RATE_UNUSED, LoRaWANJoinEvent_t *joinEvent=NULL)
Join network by restoring OTAA session or performing over-the-air activation. By this procedure,...
Definition: LoRaWAN.cpp:867
int16_t beginOTAA(uint64_t joinEUI, uint64_t devEUI, uint8_t *nwkKey, uint8_t *appKey)
Set the device credentials and activation configuration.
Definition: LoRaWAN.cpp:584
uint32_t getFCntUp()
Returns the last uplink's frame counter; also 0 if no uplink occured yet.
Definition: LoRaWAN.cpp:2666
int16_t setRx2Dr(uint8_t dr)
Configure the Rx2 datarate for ABP mode. This should not be needed for LoRaWAN 1.1 as it is configure...
Definition: LoRaWAN.cpp:2589
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:2346
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:2370
void setADR(bool enable=true)
Toggle ADR to on or off.
Definition: LoRaWAN.cpp:2616
uint32_t getDevAddr()
Returns the DevAddr of the device, regardless of OTAA or ABP mode.
Definition: LoRaWAN.cpp:2686
void resetFCntDown()
Reset the downlink frame counters (application and network) This is unsafe and can possibly allow rep...
Definition: LoRaWAN.cpp:2681
Provides common interface for protocols that run on LoRa/FSK modules, such as RTTY or LoRaWAN....
Definition: PhysicalLayer.h:135
unsigned long RadioLibTime_t
Type used for durations in RadioLib.
Definition: TypeDef.h:616
Structure to save information about LoRaWAN band.
Definition: LoRaWAN.h:375
int8_t powerMax
Maximum allowed output power in this band in dBm.
Definition: LoRaWAN.h:392
bool txParamSupported
Whether this band implements the MAC command TxParamSetupReq.
Definition: LoRaWAN.h:407
LoRaWANChannel_t txAck[2]
Relay channels for ACK downlink.
Definition: LoRaWAN.h:433
RadioLibTime_t dutyCycle
Number of milliseconds per hour of allowed Time-on-Air.
Definition: LoRaWAN.h:398
uint8_t dataRates[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES]
The corresponding datarates, bandwidths and coding rates for DR index.
Definition: LoRaWAN.h:436
RadioLibTime_t dwellTimeUp
Maximum dwell time per uplink message in milliseconds.
Definition: LoRaWAN.h:401
LoRaWANChannel_t txWoR[2]
Relay channels for WoR uplink.
Definition: LoRaWAN.h:430
uint32_t freqMin
Minimum allowed frequency (coded in 100 Hz steps)
Definition: LoRaWAN.h:383
uint8_t payloadLenMax[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES]
Array of allowed maximum payload lengths for each data rate (global maximum)
Definition: LoRaWAN.h:389
LoRaWANChannel_t txFreqs[3]
A set of default uplink (TX) channels for dynamic bands.
Definition: LoRaWAN.h:410
LoRaWANChannel_t txJoinReq[3]
A set of possible extra channels for the Join-Request message for dynamic bands.
Definition: LoRaWAN.h:413
int8_t powerNumSteps
Number of power steps in this band.
Definition: LoRaWAN.h:395
LoRaWANChannelSpan_t txSpans[2]
Default uplink (TX) channel spans for fixed bands, including Join-Request parameters.
Definition: LoRaWAN.h:419
uint8_t bandNum
Identier for this band.
Definition: LoRaWAN.h:377
uint32_t freqMax
Maximum allowed frequency (coded in 100 Hz steps)
Definition: LoRaWAN.h:386
uint8_t bandType
Whether the channels are fixed per specification, or dynamically allocated through the network (plus ...
Definition: LoRaWAN.h:380
LoRaWANChannel_t rx2
Backup channel for downlink (RX2) window.
Definition: LoRaWAN.h:427
uint8_t numTxSpans
The number of TX channel spans for fixed bands.
Definition: LoRaWAN.h:416
RadioLibTime_t dwellTimeDn
Maximum dwell time per downlink message in milliseconds.
Definition: LoRaWAN.h:404
LoRaWANChannelSpan_t rx1Span
Default downlink (RX1) channel span for fixed bands.
Definition: LoRaWAN.h:422
IDs of all currently supported bands.
Structure to save information about LoRaWAN channels. To save space, adjacent channels are saved in "...
Definition: LoRaWAN.h:316
bool available
Whether this channel is available for channel selection.
Definition: LoRaWAN.h:336
uint8_t dr
Datarate currently in use on this channel.
Definition: LoRaWAN.h:333
uint32_t freq
The channel frequency (coded in 100 Hz steps)
Definition: LoRaWAN.h:324
uint8_t idx
The channel number, as specified by defaults or the network.
Definition: LoRaWAN.h:321
uint8_t drMin
Minimum allowed datarate for this channel.
Definition: LoRaWAN.h:327
bool enabled
Whether this channel is enabled (can be used) or is disabled.
Definition: LoRaWAN.h:318
uint8_t drMax
Maximum allowed datarate for this channel (inclusive)
Definition: LoRaWAN.h:330
Structure to save information about LoRaWAN channels. To save space, adjacent channels are saved in "...
Definition: LoRaWAN.h:348
uint8_t drJoinRequest
Allowed data rates for a join request message.
Definition: LoRaWAN.h:365
uint32_t freqStart
Center frequency of the first channel in span (coded in 100 Hz steps)
Definition: LoRaWAN.h:353
uint8_t numChannels
Total number of channels in the span.
Definition: LoRaWAN.h:350
uint8_t drMax
Maximum allowed datarate for all channels in this span (inclusive)
Definition: LoRaWAN.h:362
uint32_t freqStep
Frequency step between adjacent channels (coded in 100 Hz steps)
Definition: LoRaWAN.h:356
uint8_t drMin
Minimum allowed datarate for all channels in this span.
Definition: LoRaWAN.h:359
Structure to save extra information about uplink/downlink event.
Definition: LoRaWAN.h:496
float freq
Frequency in MHz.
Definition: LoRaWAN.h:511
bool confirmed
Whether the event is confirmed or not (e.g., confirmed uplink sent by user application)
Definition: LoRaWAN.h:501
uint8_t fPort
Port number.
Definition: LoRaWAN.h:520
int16_t power
Transmit power in dBm for uplink, or RSSI for downlink.
Definition: LoRaWAN.h:514
bool confirming
Whether the event is confirming a previous request (e.g., server downlink reply to confirmed uplink s...
Definition: LoRaWAN.h:505
uint8_t datarate
Datarate.
Definition: LoRaWAN.h:508
uint8_t dir
Event direction, one of RADIOLIB_LORAWAN_CHANNEL_DIR_*.
Definition: LoRaWAN.h:498
uint8_t nbTrans
Number of times this uplink was transmitted (ADR)
Definition: LoRaWAN.h:523
uint32_t fCnt
The appropriate frame counter - for different events, different frame counters will be reported!
Definition: LoRaWAN.h:517
Structure to save extra information about activation event.
Definition: LoRaWAN.h:481
bool newSession
Whether a new session was started.
Definition: LoRaWAN.h:483
uint32_t joinNonce
The received Join-Request JoinNonce value.
Definition: LoRaWAN.h:489
uint16_t devNonce
The transmitted Join-Request DevNonce value.
Definition: LoRaWAN.h:486
MAC command specification structure.
Definition: LoRaWAN.h:219
const uint8_t lenDn
Uplink message length.
Definition: LoRaWAN.h:224
const bool user
Whether this MAC command can be issued by the user or not.
Definition: LoRaWAN.h:233
const uint8_t cid
Command ID.
Definition: LoRaWAN.h:221
const uint8_t lenUp
Downlink message length.
Definition: LoRaWAN.h:227
const bool persist
Some commands must be resent until Class A downlink received.
Definition: LoRaWAN.h:230
Common data rate structure.
Definition: PhysicalLayer.h:71