1.环境配置: 参考 Arduino IDE上手教程 完成IDE安装, 并根据实际使用的开发板安装对应的板管理, 与需要的驱动库。
2.使用到的驱动库:
3.使用到的硬件产品:
LoRa P2P 通信模式可让多个 LoRa 设备直接进行通信。此模式直接基于 LoRa 调制技术进行数据包传输,你能够完全自定义数据帧内容,进而搭建属于自己的私有网络。它适用于对通信实时性要求较高、通信范围相对较小且无需复杂网络管理的场景。
根据实际使用的设备,配置对应的通信参数。LoRa配置参数一致的设备才能进行通信。搭配不同的Atom主控时,请参考宏定义填入对应的TX/RX引脚用于初始化。
#include <M5Unified.h>
#include "rak3172_p2p.hpp"
#define LORA_CONFIG_PRLEN 8
#define LORA_CONFIG_PWR 22
#define LORA_FREQ 868E6
#define LORA_CR 0 // (4/5=0, 4/6=1, 4/7=2,4/8=3)
#define LORA_SF 7 // (6,7, 8, 9, 10, 11, 12)
#define LORA_BW 500 // (125, 250, 500)
#define ATOM_LORA_RX 19
#define ATOM_LORA_TX 22
#define ATOMS3_LORA_RX 6
#define ATOMS3_LORA_TX 5
RAK3172P2P lora;
void LoRaLoopTask(void* arg)
{
while (1) {
lora.update();
vTaskDelay(5);
}
}
void setup()
{
M5.begin();
Serial.begin(115200);
Serial.println("LoRa Init...");
while (!lora.init(&Serial2, ATOMS3_LORA_RX, ATOMS3_LORA_TX, RAK3172_BPS_115200)) {
delay(1000);
}
lora.setMode(P2P_RX_MODE, 0);
if (lora.config(LORA_FREQ, LORA_SF, LORA_BW, LORA_CR, LORA_CONFIG_PRLEN, LORA_CONFIG_PWR)) {
Serial.println("LoRa config success");
} else {
Serial.println("LoRa config failed");
}
lora.setMode(P2P_TX_RX_MODE);
xTaskCreate(LoRaLoopTask, "LoRaLoopTask", 1024 * 10, NULL, 2, NULL);
}
void loop()
{
M5.update();
if (M5.BtnA.wasPressed()) {
if (lora.print("Hello!")) {
Serial.println("Send success");
}
}
if (lora.available()) {
std::vector<p2p_frame_t> frames = lora.read();
for (int i = 0; i < frames.size(); i++) {
Serial.print("RSSI: ");
Serial.print(frames[i].rssi);
Serial.print(" SNR: ");
Serial.print(frames[i].snr);
Serial.print(" LEN: ");
Serial.print(frames[i].len);
Serial.print(" Payload: ");
for (uint8_t j = 0; j < frames[i].len; j++) {
Serial.printf("%02X", frames[i].payload[j]);
}
Serial.println();
}
lora.flush();
}
}
烧录案例程序后,通过串口监视器查看输出日志,单击AtomS3R主控的中心按键发送数据。
LoRaWAN 通信模式下,设备需要连接到 LoRaWAN 网关才能实现数据交互。得益于跳频技术的优势,这种接入方式能够同时管理更多的 LoRa 设备节点,并且可以提供一定的数据安全性保障。 在使用前,请您确认当前区域内存在公共 LoRaWAN 网关,若没有公共网关,用户也可自行搭建网关以实现连接 。
1.参考 TTN - 设备创建教程 ,依据当前使用的设备频段创建节点,并获取对应的密钥信息。这些密钥信息包含 DevEUI、AppEUI 和 AppKey 等,不同的入网模式(OTAA / ABP)所使用的密钥内容也存在差异。这些密钥是设备接入 LoRaWAN 网络的必要信息。
2.根据实际使用的设备频段与LoRaWAN网关的监听频段,配置相应的子网掩码与设备类型(ClassA ~ C)。有关 LoRaWAN 介绍与设备配置信息介绍可参考 TTN - LoRaWAN Docs 。
OTAA入网需要用到 DevEUI、AppEUI 和 AppKey 参数。
#include <M5Unified.h>
#include "rak3172_lorawan.hpp"
#define DEVEUI "****************"
#define APPEUI "****************"
#define APPKEY "********************************"
// get or set the channel mask to close or open the channel (only for US915, AU915, CN470)
#define CHANNEL_MASK "0000"
#define ATOM_LORA_RX 19
#define ATOM_LORA_TX 22
#define ATOMS3_LORA_RX 6
#define ATOMS3_LORA_TX 5
RAK3172LoRaWAN lorawan;
bool isJoin = false;
void joinCallback(bool status)
{
isJoin = status;
if (status) {
Serial.println("[LoRaWAN] Join network successful!");
Serial.println("Device EUI: " + String(DEVEUI));
} else {
Serial.println("[LoRaWAN] Join network failed!");
}
}
void sendCallback()
{
Serial.println("[LoRaWAN] Uplink confirmed by server");
}
void errorCallback(char* error)
{
Serial.print("[LoRaWAN] Error: ");
Serial.println(error);
}
void LoRaWANLoopTask(void* arg)
{
while (1) {
lorawan.update();
vTaskDelay(5);
}
}
void setup()
{
M5.begin();
Serial.begin(115200);
Serial.println("[Init] Initializing LoRaWAN module...");
while (!lorawan.init(&Serial2, ATOMS3_LORA_RX, ATOMS3_LORA_TX, RAK3172_BPS_115200)) {
Serial.println("[Init] Failed to initialize module, retrying...");
delay(1000);
}
Serial.println("Device Init OK");
Serial.println("[Config] Setting band to EU868...");
// get or set the channel mask to close or open the channel (only for US915, AU915, CN470)
while (!lorawan.setBAND(EU868, CHANNEL_MASK)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting OTAA parameters...");
while (!lorawan.setOTAA(DEVEUI, APPEUI, APPKEY)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting device mode to CLASS_C...");
while (!lorawan.setMode(CLASS_C)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting data rate to DR4...");
while (!lorawan.setDR(4)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting Link check...");
while (!lorawan.setLinkCheck(ALLWAYS_LINKCHECK)) {
delay(1000);
}
lorawan.onSend(sendCallback);
lorawan.onJoin(joinCallback);
lorawan.onError(errorCallback);
xTaskCreate(LoRaWANLoopTask, "LoRaWANLoopTask", 1024 * 10, NULL, 5, NULL);
Serial.println("[Info] Attempting to join the network...");
if (lorawan.join(true, false, 10, 10)) {
Serial.println("Start Join...");
} else {
Serial.println("Join Fail");
}
}
void loop()
{
M5.update();
if (M5.BtnA.wasReleased()) {
if (isJoin) {
String data = "UPlink LoRaWAN Frame: " + String(millis());
if (lorawan.send(data)) {
Serial.println("Send Successful");
} else {
Serial.println("Send fail");
}
} else {
Serial.println("LoRaWAN not joined");
}
}
if (lorawan.available()) {
std::vector<lorawan_frame_t> frames = lorawan.read();
for (int i = 0; i < frames.size(); i++) {
Serial.print("RSSI: ");
Serial.println(frames[i].rssi);
Serial.print("SNR: ");
Serial.println(frames[i].snr);
Serial.print("LEN: ");
Serial.println(frames[i].len);
Serial.print("PORT: ");
Serial.println(frames[i].port);
Serial.print("UNITCAST: ");
Serial.println(frames[i].unicast);
Serial.print("Payload: ");
for (uint8_t j = 0; j < frames[i].len; j++) {
Serial.printf("%02X", frames[i].payload[j]);
}
Serial.println();
}
lorawan.flush();
}
if (Serial.available()) { // If the serial port reads data.
String ch = Serial.readString(); // Copy the data read from the serial port
lorawan.sendCommand(ch);
}
}
烧录案例程序后,通过串口监视器查看输出日志,等待join成功后单击AtomS3R主控的中心按键发送数据。
在TTN对应的设备页面能够查看到当前的上下行数据日志。
切换至Messaging页面可进行数据下发。
ABP入网需要用到 DevAddr、AppSkey 和 NwkSkey 参数。不需要执行join过程,初始化密钥信息后即可开始发送数据。
#include <M5Unified.h>
#include "rak3172_lorawan.hpp"
#define DEVADDR "***********" // Device Address
#define APPSKEY "**********************" // Application Session Key
#define NWKSKEY "**********************" // Network Session Key
// get or set the channel mask to close or open the channel (only for US915, AU915, CN470)
#define CHANNEL_MASK "0000"
#define ATOM_LORA_RX 19
#define ATOM_LORA_TX 22
#define ATOMS3_LORA_RX 6
#define ATOMS3_LORA_TX 5
RAK3172LoRaWAN lorawan;
void errorCallback(char* error)
{
Serial.print("[LoRaWAN] Error: ");
Serial.println(error);
}
void LoRaWANLoopTask(void* arg)
{
while (1) {
lorawan.update();
vTaskDelay(5);
}
}
void setup()
{
M5.begin();
Serial.begin(115200);
Serial.println("[Init] Initializing LoRaWAN module...");
while (!lorawan.init(&Serial2, ATOMS3_LORA_RX, ATOMS3_LORA_TX, RAK3172_BPS_115200)) {
Serial.println("[Init] Failed to initialize module, retrying...");
delay(1000);
}
Serial.println("Device Init OK");
Serial.println("[Config] Setting band to EU868...");
// get or set the channel mask to close or open the channel (only for US915, AU915, CN470)
while (!lorawan.setBAND(EU868, CHANNEL_MASK)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting ABP parameters...");
while (!lorawan.setABP(DEVADDR, NWKSKEY, APPSKEY)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting device mode to CLASS_C...");
while (!lorawan.setMode(CLASS_C)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting data rate to DR4...");
while (!lorawan.setDR(4)) {
Serial.println(" failed, retrying...");
delay(1000);
}
Serial.println("[Config] Setting Link check...");
while (!lorawan.setLinkCheck(ALLWAYS_LINKCHECK)) {
delay(1000);
}
lorawan.onError(errorCallback);
xTaskCreate(LoRaWANLoopTask, "LoRaWANLoopTask", 1024 * 10, NULL, 5, NULL);
}
void loop()
{
M5.update();
if (M5.BtnA.wasReleased()) {
String data = "UPlink LoRaWAN Frame: " + String(millis());
if (lorawan.send(data)) {
Serial.println("Send Successful");
} else {
Serial.println("Send fail");
}
}
if (lorawan.available()) {
std::vector<lorawan_frame_t> frames = lorawan.read();
for (int i = 0; i < frames.size(); i++) {
Serial.print("RSSI: ");
Serial.println(frames[i].rssi);
Serial.print("SNR: ");
Serial.println(frames[i].snr);
Serial.print("LEN: ");
Serial.println(frames[i].len);
Serial.print("PORT: ");
Serial.println(frames[i].port);
Serial.print("UNITCAST: ");
Serial.println(frames[i].unicast);
Serial.print("Payload: ");
for (uint8_t j = 0; j < frames[i].len; j++) {
Serial.printf("%02X", frames[i].payload[j]);
}
Serial.println();
}
lorawan.flush();
}
if (Serial.available()) { // If the serial port reads data.
String ch = Serial.readString(); // Copy the data read from the serial port
lorawan.sendCommand(ch);
}
}
CayenneLPP 是一种经过优化的传感器数据格式。在使用 TTN 或者 Chirpstack 上传数据时,从控制台页面或 API 接口获取到的默认数据通常采用 Base64 编码,可读性欠佳。CayenneLPP 支持多种基本传感器类型,并且在借助 LoRaWAN 网络进行环境数据采集时,其数据格式易于解析。所以,推荐使用 CayenneLPP 处理相关数据。只需在控制台的节点配置页面选择 CayenneLPP 解析器,就能在上传数据中更直观地查看传感器数据。使用案例及相关内容可参考下方 Github 链接。