RadioLib
Universal wireless communication library for Arduino
LoRaWAN.h
1 #if !defined(_RADIOLIB_LORAWAN_H) && !defined(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 // preamble format
9 #define RADIOLIB_LORAWAN_LORA_SYNC_WORD (0x34)
10 #define RADIOLIB_LORAWAN_LORA_PREAMBLE_LEN (8)
11 #define RADIOLIB_LORAWAN_GFSK_SYNC_WORD (0xC194C1)
12 #define RADIOLIB_LORAWAN_GFSK_PREAMBLE_LEN (5)
13 
14 // MAC header field encoding MSB LSB DESCRIPTION
15 #define RADIOLIB_LORAWAN_MHDR_MTYPE_JOIN_REQUEST (0x00 << 5) // 7 5 message type: join request
16 #define RADIOLIB_LORAWAN_MHDR_MTYPE_JOIN_ACCEPT (0x01 << 5) // 7 5 join accept
17 #define RADIOLIB_LORAWAN_MHDR_MTYPE_UNCONF_DATA_UP (0x02 << 5) // 7 5 unconfirmed data up
18 #define RADIOLIB_LORAWAN_MHDR_MTYPE_UNCONF_DATA_DOWN (0x03 << 5) // 7 5 unconfirmed data down
19 #define RADIOLIB_LORAWAN_MHDR_MTYPE_CONF_DATA_UP (0x04 << 5) // 7 5 confirmed data up
20 #define RADIOLIB_LORAWAN_MHDR_MTYPE_CONF_DATA_DOWN (0x05 << 5) // 7 5 confirmed data down
21 #define RADIOLIB_LORAWAN_MHDR_MTYPE_PROPRIETARY (0x07 << 5) // 7 5 proprietary
22 #define RADIOLIB_LORAWAN_MHDR_MTYPE_MASK (0x07 << 5) // 7 5 bitmask of all possible options
23 #define RADIOLIB_LORAWAN_MHDR_MAJOR_R1 (0x00 << 0) // 1 0 major version: LoRaWAN R1
24 
25 // frame control field encoding
26 #define RADIOLIB_LORAWAN_FCTRL_ADR_ENABLED (0x01 << 7) // 7 7 adaptive data rate: enabled
27 #define RADIOLIB_LORAWAN_FCTRL_ADR_DISABLED (0x00 << 7) // 7 7 disabled
28 #define RADIOLIB_LORAWAN_FCTRL_ADR_ACK_REQ (0x01 << 6) // 6 6 adaptive data rate ACK request
29 #define RADIOLIB_LORAWAN_FCTRL_ACK (0x01 << 5) // 5 5 confirmed message acknowledge
30 #define RADIOLIB_LORAWAN_FCTRL_FRAME_PENDING (0x01 << 4) // 4 4 downlink frame is pending
31 
32 // port field
33 #define RADIOLIB_LORAWAN_FPORT_MAC_COMMAND (0x00 << 0) // 7 0 payload contains MAC commands only
34 #define RADIOLIB_LORAWAN_FPORT_RESERVED (0xE0 << 0) // 7 0 reserved port values
35 
36 // MAC commands - only those sent from end-device to gateway
37 #define RADIOLIB_LORAWAN_LINK_CHECK_REQ (0x02 << 0) // 7 0 MAC command: request to check connectivity to network
38 #define RADIOLIB_LORAWAN_LINK_ADR_ANS (0x03 << 0) // 7 0 answer to ADR change
39 #define RADIOLIB_LORAWAN_DUTY_CYCLE_ANS (0x04 << 0) // 7 0 answer to duty cycle change
40 #define RADIOLIB_LORAWAN_RX_PARAM_SETUP_ANS (0x05 << 0) // 7 0 answer to reception slot setup request
41 #define RADIOLIB_LORAWAN_DEV_STATUS_ANS (0x06 << 0) // 7 0 device status information
42 #define RADIOLIB_LORAWAN_NEW_CHANNEL_ANS (0x07 << 0) // 7 0 acknowledges change of a radio channel
43 #define RADIOLIB_LORAWAN_RX_TIMING_SETUP_ANS (0x08 << 0) // 7 0 acknowledges change of a reception slots timing
44 
45 #define RADIOLIB_LORAWAN_NOPTS_LEN (8)
46 
47 // data rat encoding
48 #define RADIOLIB_LORAWAN_DATA_RATE_SF_12 (0x06 << 4) // 7 4 LoRaWAN spreading factor: SF12
49 #define RADIOLIB_LORAWAN_DATA_RATE_SF_11 (0x05 << 4) // 7 4 SF11
50 #define RADIOLIB_LORAWAN_DATA_RATE_SF_10 (0x04 << 4) // 7 4 SF10
51 #define RADIOLIB_LORAWAN_DATA_RATE_SF_9 (0x03 << 4) // 7 4 SF9
52 #define RADIOLIB_LORAWAN_DATA_RATE_SF_8 (0x02 << 4) // 7 4 SF8
53 #define RADIOLIB_LORAWAN_DATA_RATE_SF_7 (0x01 << 4) // 7 4 SF7
54 #define RADIOLIB_LORAWAN_DATA_RATE_FSK_50_K (0x00 << 4) // 7 4 FSK @ 50 kbps
55 #define RADIOLIB_LORAWAN_DATA_RATE_BW_500_KHZ (0x00 << 0) // 3 0 LoRaWAN bandwidth: 500 kHz
56 #define RADIOLIB_LORAWAN_DATA_RATE_BW_250_KHZ (0x01 << 0) // 3 0 250 kHz
57 #define RADIOLIB_LORAWAN_DATA_RATE_BW_125_KHZ (0x02 << 0) // 3 0 125 kHz
58 #define RADIOLIB_LORAWAN_DATA_RATE_UNUSED (0xFF << 0) // 7 0 unused data rate
59 
60 #define RADIOLIB_LORAWAN_CHANNEL_DIR_UPLINK (0x00 << 0)
61 #define RADIOLIB_LORAWAN_CHANNEL_DIR_DOWNLINK (0x01 << 0)
62 #define RADIOLIB_LORAWAN_CHANNEL_DIR_BOTH (0x02 << 0)
63 #define RADIOLIB_LORAWAN_CHANNEL_DIR_NONE (0x03 << 0)
64 #define RADIOLIB_LORAWAN_CFLIST_TYPE_FREQUENCIES (0)
65 #define RADIOLIB_LORAWAN_CFLIST_TYPE_MASK (1)
66 #define RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES (16)
67 
68 // recommended default settings
69 #define RADIOLIB_LORAWAN_RECEIVE_DELAY_1_MS (1000)
70 #define RADIOLIB_LORAWAN_RECEIVE_DELAY_2_MS ((RADIOLIB_LORAWAN_RECEIVE_DELAY_1_MS) + 1000)
71 #define RADIOLIB_LORAWAN_RX1_DR_OFFSET (0)
72 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DELAY_1_MS (5000)
73 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DELAY_2_MS (6000)
74 #define RADIOLIB_LORAWAN_MAX_FCNT_GAP (16384)
75 #define RADIOLIB_LORAWAN_ADR_ACK_LIMIT (64)
76 #define RADIOLIB_LORAWAN_ADR_ACK_DELAY (32)
77 #define RADIOLIB_LORAWAN_RETRANSMIT_TIMEOUT_MIN_MS (1000)
78 #define RADIOLIB_LORAWAN_RETRANSMIT_TIMEOUT_MAX_MS (3000)
79 #define RADIOLIB_LORAWAN_POWER_STEP_SIZE_DBM (-2)
80 
81 // join request message layout
82 #define RADIOLIB_LORAWAN_JOIN_REQUEST_LEN (23)
83 #define RADIOLIB_LORAWAN_JOIN_REQUEST_JOIN_EUI_POS (1)
84 #define RADIOLIB_LORAWAN_JOIN_REQUEST_DEV_EUI_POS (9)
85 #define RADIOLIB_LORAWAN_JOIN_REQUEST_DEV_NONCE_POS (17)
86 
87 // join accept message layout
88 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_MAX_LEN (33)
89 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_JOIN_NONCE_POS (1)
90 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_HOME_NET_ID_POS (4)
91 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DEV_ADDR_POS (7)
92 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_DL_SETTINGS_POS (11)
93 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_RX_DELAY_POS (12)
94 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_POS (13)
95 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_LEN (16)
96 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_TYPE_POS (RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_POS + RADIOLIB_LORAWAN_JOIN_ACCEPT_CFLIST_LEN - 1)
97 
98 // join accept message variables
99 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_R_1_0 (0x00 << 7) // 7 7 LoRaWAN revision: 1.0
100 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_R_1_1 (0x01 << 7) // 7 7 1.1
101 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_F_NWK_S_INT_KEY (0x01)
102 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_APP_S_KEY (0x02)
103 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_S_NWK_S_INT_KEY (0x03)
104 #define RADIOLIB_LORAWAN_JOIN_ACCEPT_NWK_S_ENC_KEY (0x04)
105 
106 // uplink message layout
107 #define RADIOLIB_LORAWAN_UPLINK_LEN(PAYLOAD, FOPTS) (16 + 13 + (PAYLOAD) + (FOPTS))
108 #define RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS (16)
109 #define RADIOLIB_LORAWAN_UPLINK_DEV_ADDR_POS (RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS + 1)
110 #define RADIOLIB_LORAWAN_UPLINK_FCTRL_POS (RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS + 5)
111 #define RADIOLIB_LORAWAN_UPLINK_FCNT_POS (RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS + 6)
112 #define RADIOLIB_LORAWAN_UPLINK_FOPTS_POS (RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS + 8)
113 #define RADIOLIB_LORAWAN_UPLINK_FOPTS_MAX_LEN (RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS + 16)
114 #define RADIOLIB_LORAWAN_UPLINK_FPORT_POS(FOPTS) (RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS + 8 + (FOPTS))
115 #define RADIOLIB_LORAWAN_UPLINK_PAYLOAD_POS(FOPTS) (RADIOLIB_LORAWAN_UPLINK_LEN_START_OFFS + 9 + (FOPTS))
116 
117 // payload encryption/MIC blocks common layout
118 #define RADIOLIB_LORAWAN_BLOCK_MAGIC_POS (0)
119 #define RADIOLIB_LORAWAN_BLOCK_DIR_POS (5)
120 #define RADIOLIB_LORAWAN_BLOCK_DEV_ADDR_POS (6)
121 #define RADIOLIB_LORAWAN_BLOCK_FCNT_POS (10)
122 
123 // payload encryption block layout
124 #define RADIOLIB_LORAWAN_ENC_BLOCK_MAGIC (0x01)
125 #define RADIOLIB_LORAWAN_ENC_BLOCK_COUNTER_POS (15)
126 
127 // payload MIC blocks layout
128 #define RADIOLIB_LORAWAN_MIC_BLOCK_MAGIC (0x49)
129 #define RADIOLIB_LORAWAN_MIC_BLOCK_LEN_POS (15)
130 #define RADIOLIB_LORAWAN_MIC_DATA_RATE_POS (3)
131 #define RADIOLIB_LORAWAN_MIC_CH_INDEX_POS (4)
132 
133 #define RADIOLIB_LORAWAN_MAGIC (0x12AD101B)
134 
142  uint8_t direction;
143 
146 
148  uint8_t numChannels;
149 
151  float freqStart;
152 
154  float freqStep;
155 
157  uint8_t dataRates[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES];
158 };
159 
160 // alias for unused channel span
161 #define RADIOLIB_LORAWAN_CHANNEL_SPAN_NONE { .direction = RADIOLIB_LORAWAN_CHANNEL_DIR_NONE, .joinRequestDataRate = RADIOLIB_LORAWAN_DATA_RATE_UNUSED, .numChannels = 0, .freqStart = 0, .freqStep = 0, .dataRates = { 0 } }
162 
170 
173 
175  uint8_t payloadLenMax[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES];
176 
178  int8_t powerMax;
179 
182 
184  uint8_t cfListType;
185 
188 
191 
194 };
195 
196 // supported bands
197 extern const LoRaWANBand_t EU868;
198 extern const LoRaWANBand_t US915;
199 extern const LoRaWANBand_t CN780;
200 extern const LoRaWANBand_t EU433;
201 extern const LoRaWANBand_t AU915;
202 extern const LoRaWANBand_t CN500;
203 extern const LoRaWANBand_t AS923;
204 extern const LoRaWANBand_t KR920;
205 extern const LoRaWANBand_t IN865;
206 
211 class LoRaWANNode {
212  public:
218  LoRaWANNode(PhysicalLayer* phy, const LoRaWANBand_t* band);
219 
224  void wipe();
225 
230  int16_t begin();
231 
242  int16_t beginOTAA(uint64_t appEUI, uint64_t devEUI, uint8_t* nwkKey, uint8_t* appKey, bool force = false);
243 
253  int16_t beginAPB(uint32_t addr, uint8_t net, uint8_t* nwkSKey, uint8_t* appSKey);
254 
255  #if defined(RADIOLIB_BUILD_ARDUINO)
262  int16_t uplink(String& str, uint8_t port);
263  #endif
264 
271  int16_t uplink(const char* str, uint8_t port);
272 
280  int16_t uplink(uint8_t* data, size_t len, uint8_t port);
281 
288  int16_t configureChannel(uint8_t chan, uint8_t dr);
289 
290 #if !defined(RADIOLIB_GODMODE)
291  private:
292 #endif
293  PhysicalLayer* phyLayer;
294  const LoRaWANBand_t* band;
295 
296  // the following is either provided by the network server (OTAA)
297  // or directly entered by the user (ABP)
298  uint32_t devAddr;
299  uint8_t appSKey[RADIOLIB_AES128_KEY_SIZE];
300  uint8_t fNwkSIntKey[RADIOLIB_AES128_KEY_SIZE];
301  uint8_t sNwkSIntKey[RADIOLIB_AES128_KEY_SIZE];
302  uint8_t nwkSEncKey[RADIOLIB_AES128_KEY_SIZE];
303  uint8_t rxDelay;
304  float availableChannelsFreq[5];
305  uint16_t availableChannelsMask[6];
306 
307  // LoRaWAN revision (1.0 vs 1.1)
308  uint8_t rev;
309 
310  // currently configured data rate DR0 - DR15 (band-dependent!)
311  uint8_t dataRate;
312 
313  // currently configured channel (band-dependent!)
314  uint8_t chIndex;
315 
316  // method to generate message integrity code
317  uint32_t generateMIC(uint8_t* msg, size_t len, uint8_t* key);
318 
319  // method to verify message integrity code
320  // it assumes that the MIC is the last 4 bytes of the message
321  bool verifyMIC(uint8_t* msg, size_t len, uint8_t* key);
322 
323  int16_t setPhyProperties();
324 
325  // network-to-host conversion method - takes data from network packet and converts it to the host endians
326  template<typename T>
327  static T ntoh(uint8_t* buff, size_t size = 0);
328 
329  // host-to-network conversion method - takes data from host variable and and converts it to network packet endians
330  template<typename T>
331  static void hton(uint8_t* buff, T val, size_t size = 0);
332 };
333 
334 #endif
LoRaWAN-compatible node (class A device).
Definition: LoRaWAN.h:211
int16_t beginOTAA(uint64_t appEUI, uint64_t devEUI, uint8_t *nwkKey, uint8_t *appKey, bool force=false)
Join network by performing over-the-air activation. By this procedure, the device will perform an exc...
Definition: LoRaWAN.cpp:49
void wipe()
Wipe internal persistent parameters. This will reset all counters and saved variables,...
Definition: LoRaWAN.cpp:23
int16_t beginAPB(uint32_t addr, uint8_t net, uint8_t *nwkSKey, uint8_t *appSKey)
Join network by performing activation by personalization. In this procedure, all necessary configurat...
Definition: LoRaWAN.cpp:227
LoRaWANNode(PhysicalLayer *phy, const LoRaWANBand_t *band)
Default constructor.
Definition: LoRaWAN.cpp:18
int16_t configureChannel(uint8_t chan, uint8_t dr)
Configure the radio to a given channel frequency and data rate.
Definition: LoRaWAN.cpp:366
int16_t uplink(const char *str, uint8_t port)
Send a message to the server.
Definition: LoRaWAN.cpp:240
int16_t begin()
Join network by loading information from persistent storage.
Definition: LoRaWAN.cpp:28
Provides common interface for protocols that run on LoRa/FSK modules, such as RTTY or LoRaWAN....
Definition: PhysicalLayer.h:33
Structure to save information about LoRaWAN band.
Definition: LoRaWAN.h:167
uint8_t cfListType
Whether the optional channels are defined as list of frequencies or bit mask.
Definition: LoRaWAN.h:184
LoRaWANChannelSpan_t backupChannel
Backup downlink (RX2) channel - just a single channel, but using the same structure for convenience.
Definition: LoRaWAN.h:193
uint8_t downlinkDataRateBase
The base downlink data rate. Used to calculate data rate changes for adaptive data rate.
Definition: LoRaWAN.h:169
int8_t powerMax
Maximum allowed output power in this band in dBm.
Definition: LoRaWAN.h:178
LoRaWANChannelSpan_t defaultChannels[3]
Default uplink (TX/RX1) channels defined by LoRaWAN Regional Parameters.
Definition: LoRaWAN.h:190
uint8_t payloadLenMax[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES]
Array of allowed maximum payload lengths for each data rate.
Definition: LoRaWAN.h:175
int8_t powerNumSteps
Number of power steps in this band.
Definition: LoRaWAN.h:181
uint8_t numChannelSpans
Number of channel spans in the band.
Definition: LoRaWAN.h:187
uint8_t downlinkDataRateMin
The minimum allowed downlink data rate. Used to calculate data rate changes for adaptive data rate.
Definition: LoRaWAN.h:172
Structure to save information about LoRaWAN channels. To save space, adjacent channels are saved in "...
Definition: LoRaWAN.h:140
uint8_t joinRequestDataRate
Allowed data rates for a join request message.
Definition: LoRaWAN.h:145
float freqStart
Center frequency of the first channel in span.
Definition: LoRaWAN.h:151
uint8_t numChannels
Total number of channels in the span.
Definition: LoRaWAN.h:148
uint8_t dataRates[RADIOLIB_LORAWAN_CHANNEL_NUM_DATARATES]
Array of datarates supported by all channels in the span.
Definition: LoRaWAN.h:157
float freqStep
Frequency step between adjacent channels.
Definition: LoRaWAN.h:154
uint8_t direction
Whether this channel span is for uplink, downlink, or both directions.
Definition: LoRaWAN.h:142