2018-03-16 09:34:06 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Inria
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup examples
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Example demonstrating the use of LoRaWAN with RIOT
|
|
|
|
*
|
|
|
|
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
#include "msg.h"
|
|
|
|
#include "thread.h"
|
|
|
|
#include "fmt.h"
|
|
|
|
|
2021-11-11 10:36:33 +01:00
|
|
|
#if IS_USED(MODULE_PERIPH_RTC)
|
2018-03-16 09:34:06 +01:00
|
|
|
#include "periph/rtc.h"
|
2021-11-11 10:36:33 +01:00
|
|
|
#else
|
|
|
|
#include "timex.h"
|
|
|
|
#include "ztimer.h"
|
|
|
|
#endif
|
2018-03-16 09:34:06 +01:00
|
|
|
|
|
|
|
#include "net/loramac.h"
|
|
|
|
#include "semtech_loramac.h"
|
|
|
|
|
2021-03-12 16:14:46 +01:00
|
|
|
#if IS_USED(MODULE_SX127X)
|
2021-03-12 10:33:29 +01:00
|
|
|
#include "sx127x.h"
|
|
|
|
#include "sx127x_netdev.h"
|
|
|
|
#include "sx127x_params.h"
|
2021-03-12 16:14:46 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_SX126X)
|
|
|
|
#include "sx126x.h"
|
|
|
|
#include "sx126x_netdev.h"
|
|
|
|
#include "sx126x_params.h"
|
|
|
|
#endif
|
2021-03-12 10:33:29 +01:00
|
|
|
|
2018-03-16 09:34:06 +01:00
|
|
|
/* Messages are sent every 20s to respect the duty cycle on each channel */
|
2021-11-11 10:36:33 +01:00
|
|
|
#define PERIOD_S (20U)
|
2018-03-16 09:34:06 +01:00
|
|
|
|
|
|
|
#define SENDER_PRIO (THREAD_PRIORITY_MAIN - 1)
|
|
|
|
static kernel_pid_t sender_pid;
|
|
|
|
static char sender_stack[THREAD_STACKSIZE_MAIN / 2];
|
|
|
|
|
2021-03-12 10:33:29 +01:00
|
|
|
static semtech_loramac_t loramac;
|
2021-06-28 14:07:28 +02:00
|
|
|
#if IS_USED(MODULE_SX127X)
|
2021-03-12 10:33:29 +01:00
|
|
|
static sx127x_t sx127x;
|
2021-06-28 14:07:28 +02:00
|
|
|
#endif
|
|
|
|
#if IS_USED(MODULE_SX126X)
|
|
|
|
static sx126x_t sx126x;
|
|
|
|
#endif
|
2021-11-11 10:36:33 +01:00
|
|
|
#if !IS_USED(MODULE_PERIPH_RTC)
|
|
|
|
static ztimer_t timer;
|
|
|
|
#endif
|
2018-03-16 09:34:06 +01:00
|
|
|
|
|
|
|
static const char *message = "This is RIOT!";
|
|
|
|
|
|
|
|
static uint8_t deveui[LORAMAC_DEVEUI_LEN];
|
|
|
|
static uint8_t appeui[LORAMAC_APPEUI_LEN];
|
|
|
|
static uint8_t appkey[LORAMAC_APPKEY_LEN];
|
|
|
|
|
2021-11-11 10:36:33 +01:00
|
|
|
static void _alarm_cb(void *arg)
|
2018-03-16 09:34:06 +01:00
|
|
|
{
|
|
|
|
(void) arg;
|
|
|
|
msg_t msg;
|
|
|
|
msg_send(&msg, sender_pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _prepare_next_alarm(void)
|
|
|
|
{
|
2021-11-11 10:36:33 +01:00
|
|
|
#if IS_USED(MODULE_PERIPH_RTC)
|
2018-03-16 09:34:06 +01:00
|
|
|
struct tm time;
|
|
|
|
rtc_get_time(&time);
|
|
|
|
/* set initial alarm */
|
2021-11-11 10:36:33 +01:00
|
|
|
time.tm_sec += PERIOD_S;
|
2018-03-16 09:34:06 +01:00
|
|
|
mktime(&time);
|
2021-11-11 10:36:33 +01:00
|
|
|
rtc_set_alarm(&time, _alarm_cb, NULL);
|
|
|
|
#else
|
|
|
|
timer.callback = _alarm_cb;
|
|
|
|
ztimer_set(ZTIMER_MSEC, &timer, PERIOD_S * MS_PER_SEC);
|
|
|
|
#endif
|
2018-03-16 09:34:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void _send_message(void)
|
|
|
|
{
|
|
|
|
printf("Sending: %s\n", message);
|
2019-01-30 10:05:37 +01:00
|
|
|
/* Try to send the message */
|
|
|
|
uint8_t ret = semtech_loramac_send(&loramac,
|
|
|
|
(uint8_t *)message, strlen(message));
|
2019-07-19 12:17:47 +02:00
|
|
|
if (ret != SEMTECH_LORAMAC_TX_DONE) {
|
2019-01-30 10:05:37 +01:00
|
|
|
printf("Cannot send message '%s', ret code: %d\n", message, ret);
|
|
|
|
return;
|
|
|
|
}
|
2018-03-16 09:34:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void *sender(void *arg)
|
|
|
|
{
|
|
|
|
(void)arg;
|
|
|
|
|
|
|
|
msg_t msg;
|
|
|
|
msg_t msg_queue[8];
|
|
|
|
msg_init_queue(msg_queue, 8);
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
msg_receive(&msg);
|
|
|
|
|
|
|
|
/* Trigger the message send */
|
|
|
|
_send_message();
|
|
|
|
|
|
|
|
/* Schedule the next wake-up alarm */
|
|
|
|
_prepare_next_alarm();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* this should never be reached */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
puts("LoRaWAN Class A low-power application");
|
|
|
|
puts("=====================================");
|
|
|
|
|
|
|
|
/* Convert identifiers and application key */
|
2020-11-25 22:13:37 +01:00
|
|
|
fmt_hex_bytes(deveui, CONFIG_LORAMAC_DEV_EUI_DEFAULT);
|
|
|
|
fmt_hex_bytes(appeui, CONFIG_LORAMAC_APP_EUI_DEFAULT);
|
|
|
|
fmt_hex_bytes(appkey, CONFIG_LORAMAC_APP_KEY_DEFAULT);
|
2018-03-16 09:34:06 +01:00
|
|
|
|
2021-03-12 10:33:29 +01:00
|
|
|
/* Initialize the radio driver */
|
2021-03-12 16:14:46 +01:00
|
|
|
#if IS_USED(MODULE_SX127X)
|
2021-03-12 10:33:29 +01:00
|
|
|
sx127x_setup(&sx127x, &sx127x_params[0], 0);
|
2021-06-22 11:01:01 +02:00
|
|
|
loramac.netdev = &sx127x.netdev;
|
2021-03-12 10:33:29 +01:00
|
|
|
loramac.netdev->driver = &sx127x_driver;
|
2021-03-12 16:14:46 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_SX126X)
|
|
|
|
sx126x_setup(&sx126x, &sx126x_params[0], 0);
|
2021-06-22 11:01:01 +02:00
|
|
|
loramac.netdev = &sx126x.netdev;
|
2021-03-12 16:14:46 +01:00
|
|
|
loramac.netdev->driver = &sx126x_driver;
|
|
|
|
#endif
|
2021-03-12 10:33:29 +01:00
|
|
|
|
2018-03-16 09:34:06 +01:00
|
|
|
/* Initialize the loramac stack */
|
|
|
|
semtech_loramac_init(&loramac);
|
|
|
|
semtech_loramac_set_deveui(&loramac, deveui);
|
|
|
|
semtech_loramac_set_appeui(&loramac, appeui);
|
|
|
|
semtech_loramac_set_appkey(&loramac, appkey);
|
|
|
|
|
|
|
|
/* Use a fast datarate, e.g. BW125/SF7 in EU868 */
|
|
|
|
semtech_loramac_set_dr(&loramac, LORAMAC_DR_5);
|
|
|
|
|
|
|
|
/* Start the Over-The-Air Activation (OTAA) procedure to retrieve the
|
|
|
|
* generated device address and to get the network and application session
|
|
|
|
* keys.
|
|
|
|
*/
|
|
|
|
puts("Starting join procedure");
|
|
|
|
if (semtech_loramac_join(&loramac, LORAMAC_JOIN_OTAA) != SEMTECH_LORAMAC_JOIN_SUCCEEDED) {
|
|
|
|
puts("Join procedure failed");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
puts("Join procedure succeeded");
|
|
|
|
|
|
|
|
/* start the sender thread */
|
|
|
|
sender_pid = thread_create(sender_stack, sizeof(sender_stack),
|
|
|
|
SENDER_PRIO, 0, sender, NULL, "sender");
|
|
|
|
|
|
|
|
/* trigger the first send */
|
|
|
|
msg_t msg;
|
|
|
|
msg_send(&msg, sender_pid);
|
|
|
|
return 0;
|
|
|
|
}
|