2017-10-08 12:47:17 +02:00
|
|
|
/**
|
|
|
|
* @defgroup pkg_semtech-loramac Semtech LoRaMAC implementation
|
|
|
|
* @ingroup pkg
|
|
|
|
* @ingroup net
|
|
|
|
* @brief Provides a RIOT adaption of Semtech LoRaMAC implementation
|
|
|
|
*
|
|
|
|
* # Introduction
|
|
|
|
*
|
|
|
|
* This package provides an API built on top of the
|
2018-02-28 23:46:43 +01:00
|
|
|
* [Semtech LoRaMAC-node](https://github.com/Lora-net/LoRaMac-node) reference
|
2017-10-08 12:47:17 +02:00
|
|
|
* implementation of a LoRa network.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* # Importing this package in an application
|
|
|
|
*
|
|
|
|
* This package only works with Semtech SX1272 and SX1276 radio devices. Thus,
|
|
|
|
* in order to use it properly, the application `Makefile` must import the
|
|
|
|
* corresponding device driver:
|
|
|
|
* ```
|
|
|
|
* USEMODULE += sx1272 # for a SX1272 radio device
|
|
|
|
* USEMODULE += sx1276 # for a SX1276 radio device
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* In order to use this package in an application, add the following in
|
|
|
|
* the application `Makefile`:
|
|
|
|
* ```
|
|
|
|
* USEPKG += semtech-loramac
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Since the LoRa radio depends on regional parameters regarding the access
|
|
|
|
* to the physical support, the region where the device is used needs to be
|
|
|
|
* set at compile time. Example for EU868:
|
|
|
|
* ```
|
2019-05-25 22:03:52 +02:00
|
|
|
* LORA_REGION = EU868
|
2017-10-08 12:47:17 +02:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* # Using the package API
|
|
|
|
*
|
|
|
|
* The package provides a simple API for initializing the MAC, setting/getting
|
|
|
|
* parameters, joining a network and sending/receiving packets to/from a LoRa
|
|
|
|
* Network.
|
|
|
|
*
|
|
|
|
* In your `main.c`, some header files must be first included:
|
|
|
|
* ```c
|
|
|
|
* #include "net/loramac.h" /* core loramac definitions */
|
2018-03-19 08:28:13 +01:00
|
|
|
* #include "semtech_loramac.h" /* package API */
|
2017-10-08 12:47:17 +02:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Then define global variables:
|
|
|
|
* ```c
|
2018-03-19 08:28:13 +01:00
|
|
|
* semtech_loramac_t loramac; /* The loramac stack device descriptor */
|
2017-10-08 12:47:17 +02:00
|
|
|
* /* define the required keys for OTAA, e.g over-the-air activation (the
|
|
|
|
* null arrays need to be updated with valid LoRa values) */
|
2018-02-25 14:22:30 +01:00
|
|
|
* static const uint8_t deveui[LORAMAC_DEVEUI_LEN] = { 0x00, 0x00, 0x00, 0x00, \
|
|
|
|
* 0x00, 0x00, 0x00, 0x00 };
|
|
|
|
* static const uint8_t appeui[LORAMAC_APPEUI_LEN] = { 0x00, 0x00, 0x00, 0x00, \
|
|
|
|
* 0x00, 0x00, 0x00, 0x00 };
|
|
|
|
* static const uint8_t appkey[LORAMAC_APPKEY_LEN] = { 0x00, 0x00, 0x00, 0x00, \
|
|
|
|
* 0x00, 0x00, 0x00, 0x00, \
|
|
|
|
* 0x00, 0x00, 0x00, 0x00, \
|
|
|
|
* 0x00, 0x00, 0x00, 0x00 };
|
2017-10-08 12:47:17 +02:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Now in the `main` function:
|
2018-03-19 08:28:13 +01:00
|
|
|
* 1. initialize the LoRaMAC MAC layer
|
|
|
|
* 2. set the LoRa keys
|
|
|
|
* 3. join the network
|
|
|
|
* 4. send some data to the network
|
2017-10-08 12:47:17 +02:00
|
|
|
*
|
|
|
|
* ```c
|
|
|
|
* int main(void)
|
|
|
|
* {
|
2018-03-19 08:28:13 +01:00
|
|
|
* /* 1. initialize the LoRaMAC MAC layer */
|
|
|
|
* semtech_loramac_init(&loramac);
|
2017-10-08 12:47:17 +02:00
|
|
|
*
|
2018-03-19 08:28:13 +01:00
|
|
|
* /* 2. set the keys identifying the device */
|
|
|
|
* semtech_loramac_set_deveui(&loramac, deveui);
|
|
|
|
* semtech_loramac_set_appeui(&loramac, appeui);
|
|
|
|
* semtech_loramac_set_appkey(&loramac, appkey);
|
2017-10-08 12:47:17 +02:00
|
|
|
*
|
2018-03-19 08:28:13 +01:00
|
|
|
* /* 3. join the network */
|
|
|
|
* if (semtech_loramac_join(&loramac, LORAMAC_JOIN_OTAA) != SEMTECH_LORAMAC_JOIN_SUCCEEDED) {
|
2017-10-08 12:47:17 +02:00
|
|
|
* puts("Join procedure failed");
|
2018-02-28 23:46:43 +01:00
|
|
|
* return 1;
|
2017-10-08 12:47:17 +02:00
|
|
|
* }
|
|
|
|
* puts("Join procedure succeeded");
|
|
|
|
*
|
2018-03-19 08:28:13 +01:00
|
|
|
* /* 4. send some data */
|
2018-02-25 14:22:30 +01:00
|
|
|
* char *message = "This is RIOT";
|
2019-04-10 08:20:40 +02:00
|
|
|
* if (semtech_loramac_send(&loramac,
|
2019-07-19 12:17:47 +02:00
|
|
|
* (uint8_t *)message, strlen(message)) != SEMTECH_LORAMAC_TX_DONE) {
|
2019-04-10 08:20:40 +02:00
|
|
|
printf("Cannot send message '%s'\n", message);
|
|
|
|
* return 1;
|
|
|
|
* }
|
2018-03-19 08:28:13 +01:00
|
|
|
*
|
2019-05-17 11:35:57 +02:00
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
2019-05-29 10:10:16 +02:00
|
|
|
* To receive downlink messages, enable the `semtech_loramac_rx` and use a
|
|
|
|
* dedicated receiving thread.
|
|
|
|
* - In the application Makefile, add
|
|
|
|
* ```mk
|
|
|
|
* USEMODULE += semtech_loramac_rx
|
|
|
|
* ```
|
2019-05-17 11:35:57 +02:00
|
|
|
* - At the beginning of the application source file, add the necessary
|
|
|
|
* includes and declare the message queue and stack arrays:
|
|
|
|
* ```c
|
|
|
|
* #include "thread.h"
|
|
|
|
*
|
|
|
|
* #define RECV_MSG_QUEUE (4U)
|
|
|
|
* static msg_t _recv_queue[RECV_MSG_QUEUE];
|
|
|
|
*
|
|
|
|
* static char _recv_stack[THREAD_STACKSIZE_DEFAULT];
|
|
|
|
* ```
|
|
|
|
* - Implement the thread: it initializes its message queue and then
|
|
|
|
* simply calls the blocking `semtech_loramac_recv` function in a loop:
|
|
|
|
* ```c
|
|
|
|
* static void *_recv(void *arg)
|
|
|
|
* {
|
|
|
|
* msg_init_queue(_recv_queue, RECV_MSG_QUEUE);
|
|
|
|
*
|
|
|
|
* (void)arg;
|
|
|
|
* while (1) {
|
|
|
|
* /* blocks until some data is received */
|
|
|
|
* semtech_loramac_recv(&loramac);
|
2018-03-19 08:28:13 +01:00
|
|
|
* loramac.rx_data.payload[loramac.rx_data.payload_len] = 0;
|
|
|
|
* printf("Data received: %s, port: %d\n",
|
|
|
|
* (char *)loramac.rx_data.payload, loramac.rx_data.port);
|
2019-05-17 11:35:57 +02:00
|
|
|
* }
|
|
|
|
*
|
2018-03-19 08:28:13 +01:00
|
|
|
* }
|
2019-05-17 11:35:57 +02:00
|
|
|
* return NULL;
|
2017-10-08 12:47:17 +02:00
|
|
|
* }
|
|
|
|
* ```
|
2019-05-17 11:35:57 +02:00
|
|
|
* - Finally, this thread can be started after the join procedure
|
|
|
|
* succeeds:
|
|
|
|
* ```c
|
|
|
|
* thread_create(_recv_stack, sizeof(_recv_stack),
|
|
|
|
* THREAD_PRIORITY_MAIN - 1, 0, _recv, NULL, "recv thread");
|
|
|
|
* ```
|
2017-10-08 12:47:17 +02:00
|
|
|
*
|
2019-05-29 17:33:17 +02:00
|
|
|
* # Persistence
|
|
|
|
*
|
|
|
|
* If the board CPU provides an internal EEPROM, this package provides a
|
2019-10-23 21:18:08 +02:00
|
|
|
* mechanism for storing EUIs, keys and some MAC parameters (frame counter,
|
2019-05-29 17:33:17 +02:00
|
|
|
* join status).
|
|
|
|
* After a successful join procedure, use `semtech_loramac_save` function to
|
|
|
|
* persist this information and it will be loaded automatically at the next
|
|
|
|
* reboot.
|
|
|
|
* If the device is already joined to a network, to avoid another OTAA join
|
|
|
|
* procedure use `semtech_loramac_is_mac_joined` function to check the join
|
|
|
|
* status of the device.
|
|
|
|
*
|
2019-10-23 21:18:08 +02:00
|
|
|
* This mechanism is especially useful when using deep sleep power modes that
|
2019-05-29 17:33:17 +02:00
|
|
|
* don't preserve RAM.
|
|
|
|
*
|
2017-10-08 12:47:17 +02:00
|
|
|
* @warning It is not possible to directly call the original LoRaMAC-node API
|
|
|
|
* using this package. This package should only be considered as a
|
|
|
|
* wrapper around the original LoRaMAC-node API and only the API
|
|
|
|
* provided by this package should be used.
|
|
|
|
*
|
|
|
|
* # License
|
|
|
|
*
|
|
|
|
* The library is using the BSD 3-clause license.
|
|
|
|
*
|
2019-04-10 08:21:41 +02:00
|
|
|
* @see https://github.com/Lora-net/LoRaMac-node
|
|
|
|
*/
|