Add initial implementation of LoRaWAN communication for HT-CT62 devices with basic and advanced sketches
This commit is contained in:
commit
07ecde511d
BIN
HT-CT62(Rev1.1).pdf
Normal file
BIN
HT-CT62(Rev1.1).pdf
Normal file
Binary file not shown.
153
LoRaWan/LoRaWan.ino
Normal file
153
LoRaWan/LoRaWan.ino
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
/* Heltec Automation LoRaWAN communication example
|
||||||
|
*
|
||||||
|
* Function:
|
||||||
|
* 1. Upload node data to the server using the standard LoRaWAN protocol.
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* 1. Communicate using LoRaWAN protocol.
|
||||||
|
*
|
||||||
|
* HelTec AutoMation, Chengdu, China
|
||||||
|
* 成都惠利特自动化科技有限公司
|
||||||
|
* www.heltec.org
|
||||||
|
*
|
||||||
|
* */
|
||||||
|
|
||||||
|
#include "LoRaWan_APP.h"
|
||||||
|
|
||||||
|
/* OTAA para*/
|
||||||
|
uint8_t devEui[] = {0x70, 0xB3, 0xD5, 0x7E, 0xD0, 0x06, 0x53, 0xC8};
|
||||||
|
uint8_t appEui[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
||||||
|
uint8_t appKey[] = {0x74, 0xD6, 0x6E, 0x63, 0x45, 0x82, 0x48, 0x27, 0xFE, 0xC5, 0xB7, 0x70, 0xBA, 0x2B, 0x50, 0x45};
|
||||||
|
|
||||||
|
/* ABP para*/
|
||||||
|
uint8_t nwkSKey[] = {0x15, 0xb1, 0xd0, 0xef, 0xa4, 0x63, 0xdf, 0xbe, 0x3d, 0x11, 0x18, 0x1e, 0x1e, 0xc7, 0xda, 0x85};
|
||||||
|
uint8_t appSKey[] = {0xd7, 0x2c, 0x78, 0x75, 0x8c, 0xdc, 0xca, 0xbf, 0x55, 0xee, 0x4a, 0x77, 0x8d, 0x16, 0xef, 0x67};
|
||||||
|
uint32_t devAddr = (uint32_t)0x007e6ae1;
|
||||||
|
|
||||||
|
/*LoraWan channelsmask, default channels 0-7*/
|
||||||
|
uint16_t userChannelsMask[6] = {0x00FF, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000};
|
||||||
|
|
||||||
|
/*LoraWan region, select in arduino IDE tools*/
|
||||||
|
LoRaMacRegion_t loraWanRegion = ACTIVE_REGION;
|
||||||
|
|
||||||
|
/*LoraWan Class, Class A and Class C are supported*/
|
||||||
|
DeviceClass_t loraWanClass = CLASS_A;
|
||||||
|
|
||||||
|
/*the application data transmission duty cycle. value in [ms].*/
|
||||||
|
uint32_t appTxDutyCycle = 15000;
|
||||||
|
|
||||||
|
/*OTAA or ABP*/
|
||||||
|
bool overTheAirActivation = true;
|
||||||
|
|
||||||
|
/*ADR enable*/
|
||||||
|
bool loraWanAdr = true;
|
||||||
|
|
||||||
|
/* Indicates if the node is sending confirmed or unconfirmed messages */
|
||||||
|
bool isTxConfirmed = true;
|
||||||
|
|
||||||
|
/* Application port */
|
||||||
|
uint8_t appPort = 2;
|
||||||
|
/*!
|
||||||
|
* Number of trials to transmit the frame, if the LoRaMAC layer did not
|
||||||
|
* receive an acknowledgment. The MAC performs a datarate adaptation,
|
||||||
|
* according to the LoRaWAN Specification V1.0.2, chapter 18.4, according
|
||||||
|
* to the following table:
|
||||||
|
*
|
||||||
|
* Transmission nb | Data Rate
|
||||||
|
* ----------------|-----------
|
||||||
|
* 1 (first) | DR
|
||||||
|
* 2 | DR
|
||||||
|
* 3 | max(DR-1,0)
|
||||||
|
* 4 | max(DR-1,0)
|
||||||
|
* 5 | max(DR-2,0)
|
||||||
|
* 6 | max(DR-2,0)
|
||||||
|
* 7 | max(DR-3,0)
|
||||||
|
* 8 | max(DR-3,0)
|
||||||
|
*
|
||||||
|
* Note, that if NbTrials is set to 1 or 2, the MAC will not decrease
|
||||||
|
* the datarate, in case the LoRaMAC layer did not receive an acknowledgment
|
||||||
|
*/
|
||||||
|
uint8_t confirmedNbTrials = 4;
|
||||||
|
|
||||||
|
/* Prepares the payload of the frame */
|
||||||
|
static void prepareTxFrame(uint8_t port)
|
||||||
|
{
|
||||||
|
/*appData size is LORAWAN_APP_DATA_MAX_SIZE which is defined in "commissioning.h".
|
||||||
|
*appDataSize max value is LORAWAN_APP_DATA_MAX_SIZE.
|
||||||
|
*if enabled AT, don't modify LORAWAN_APP_DATA_MAX_SIZE, it may cause system hanging or failure.
|
||||||
|
*if disabled AT, LORAWAN_APP_DATA_MAX_SIZE can be modified, the max value is reference to lorawan region and SF.
|
||||||
|
*for example, if use REGION_CN470,
|
||||||
|
*the max value for different DR can be found in MaxPayloadOfDatarateCN470 refer to DataratesCN470 and BandwidthsCN470 in "RegionCN470.h".
|
||||||
|
*/
|
||||||
|
appDataSize = 32;
|
||||||
|
// appData[0] = 0x00;
|
||||||
|
// appData[1] = 0x01;
|
||||||
|
// appData[2] = 0x02;
|
||||||
|
// appData[3] = 0x03;
|
||||||
|
|
||||||
|
uint8_t mockData[32] = {0x68, 0X65, 0X6C, 0X6C, 0X6F, 0X20, 0X77, 0X6F, 0X72, 0X6C, 0X64, 0X21};
|
||||||
|
memcpy(appData, mockData, appDataSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
// if true, next uplink will add MOTE_MAC_DEVICE_TIME_REQ
|
||||||
|
|
||||||
|
void setup()
|
||||||
|
{
|
||||||
|
Serial.begin(115200);
|
||||||
|
while (!Serial)
|
||||||
|
;
|
||||||
|
Mcu.begin(HELTEC_BOARD, SLOW_CLK_TPYE);
|
||||||
|
Serial.println("HELTEC CT-R2 ready");
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop()
|
||||||
|
{
|
||||||
|
switch (deviceState)
|
||||||
|
{
|
||||||
|
case DEVICE_STATE_INIT:
|
||||||
|
{
|
||||||
|
Serial.println("Device INIT");
|
||||||
|
#if (LORAWAN_DEVEUI_AUTO)
|
||||||
|
LoRaWAN.generateDeveuiByChipID();
|
||||||
|
#endif
|
||||||
|
LoRaWAN.init(loraWanClass, loraWanRegion);
|
||||||
|
// both set join DR and DR when ADR off
|
||||||
|
LoRaWAN.setDefaultDR(3);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_JOIN:
|
||||||
|
{
|
||||||
|
Serial.println("Join Network");
|
||||||
|
LoRaWAN.join();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_SEND:
|
||||||
|
{
|
||||||
|
Serial.println("Send Data");
|
||||||
|
prepareTxFrame(appPort);
|
||||||
|
LoRaWAN.send();
|
||||||
|
deviceState = DEVICE_STATE_CYCLE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_CYCLE:
|
||||||
|
{
|
||||||
|
Serial.println("Schedule packet transmission");
|
||||||
|
// Schedule next packet transmission
|
||||||
|
txDutyCycleTime = appTxDutyCycle + randr(-APP_TX_DUTYCYCLE_RND, APP_TX_DUTYCYCLE_RND);
|
||||||
|
LoRaWAN.cycle(txDutyCycleTime);
|
||||||
|
deviceState = DEVICE_STATE_SLEEP;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_SLEEP:
|
||||||
|
{
|
||||||
|
// Serial.println("Device Sleep");
|
||||||
|
LoRaWAN.sleep(loraWanClass);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
deviceState = DEVICE_STATE_INIT;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
306
LoRaWanNode/LoRaWanNode.ino
Normal file
306
LoRaWanNode/LoRaWanNode.ino
Normal file
@ -0,0 +1,306 @@
|
|||||||
|
/* Heltec Automation LoRaWAN communication example
|
||||||
|
*
|
||||||
|
* Function:
|
||||||
|
* 1. Upload node data to the server using the standard LoRaWAN protocol.
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* 1. Communicate using LoRaWAN protocol.
|
||||||
|
*
|
||||||
|
* HelTec AutoMation, Chengdu, China
|
||||||
|
* 成都惠利特自动化科技有限公司
|
||||||
|
* www.heltec.org
|
||||||
|
*
|
||||||
|
* */
|
||||||
|
|
||||||
|
#define DEBUG_MODE 0
|
||||||
|
|
||||||
|
#if DEBUG_MODE
|
||||||
|
#define DEBUG_PRINT(x) Serial.print(x)
|
||||||
|
#define DEBUG_PRINTLN(x) Serial.println(x)
|
||||||
|
#else
|
||||||
|
#define DEBUG_PRINT(x)
|
||||||
|
#define DEBUG_PRINTLN(x)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "LoRaWan_APP.h"
|
||||||
|
#include <EEPROM.h>
|
||||||
|
|
||||||
|
#define APP_EUI_ADDRESS 0
|
||||||
|
#define APP_KEY_ADDRESS 8
|
||||||
|
|
||||||
|
/* OTAA para*/
|
||||||
|
uint8_t devEui[] = {0x70, 0xB3, 0xD5, 0x7E, 0xD0, 0x06, 0x53, 0xC8};
|
||||||
|
uint8_t defaultAppEui[8] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01};
|
||||||
|
uint8_t defaultAppKey[16] = {0x74, 0xD6, 0x6E, 0x63, 0x45, 0x82, 0x48, 0x27, 0xFE, 0xC5, 0xB7, 0x70, 0xBA, 0x2B, 0x50, 0x45};
|
||||||
|
// uint8_t appEui[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01};
|
||||||
|
// uint8_t appKey[] = {0x74, 0xD6, 0x6E, 0x63, 0x45, 0x82, 0x48, 0x27, 0xFE, 0xC5, 0xB7, 0x70, 0xBA, 0x2B, 0x50, 0x45};
|
||||||
|
uint8_t appEui[8];
|
||||||
|
uint8_t appKey[16];
|
||||||
|
|
||||||
|
/* ABP para*/
|
||||||
|
uint8_t nwkSKey[] = {0x15, 0xb1, 0xd0, 0xef, 0xa4, 0x63, 0xdf, 0xbe, 0x3d, 0x11, 0x18, 0x1e, 0x1e, 0xc7, 0xda, 0x85};
|
||||||
|
uint8_t appSKey[] = {0xd7, 0x2c, 0x78, 0x75, 0x8c, 0xdc, 0xca, 0xbf, 0x55, 0xee, 0x4a, 0x77, 0x8d, 0x16, 0xef, 0x67};
|
||||||
|
uint32_t devAddr = (uint32_t)0x007e6ae1;
|
||||||
|
|
||||||
|
/*LoraWan channelsmask, default channels 0-7*/
|
||||||
|
uint16_t userChannelsMask[6] = {0x00FF, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000};
|
||||||
|
|
||||||
|
/*LoraWan region, select in arduino IDE tools*/
|
||||||
|
LoRaMacRegion_t loraWanRegion = ACTIVE_REGION;
|
||||||
|
|
||||||
|
/*LoraWan Class, Class A and Class C are supported*/
|
||||||
|
DeviceClass_t loraWanClass = CLASS_C;
|
||||||
|
|
||||||
|
/*the application data transmission duty cycle. value in [ms].*/
|
||||||
|
uint32_t appTxDutyCycle = 15000;
|
||||||
|
|
||||||
|
/*OTAA or ABP*/
|
||||||
|
bool overTheAirActivation = true;
|
||||||
|
|
||||||
|
/*ADR enable*/
|
||||||
|
bool loraWanAdr = true;
|
||||||
|
|
||||||
|
/* Indicates if the node is sending confirmed or unconfirmed messages */
|
||||||
|
bool isTxConfirmed = true;
|
||||||
|
|
||||||
|
/* Application port */
|
||||||
|
uint8_t appPort = 2;
|
||||||
|
/*!
|
||||||
|
* Number of trials to transmit the frame, if the LoRaMAC layer did not
|
||||||
|
* receive an acknowledgment. The MAC performs a datarate adaptation,
|
||||||
|
* according to the LoRaWAN Specification V1.0.2, chapter 18.4, according
|
||||||
|
* to the following table:
|
||||||
|
*
|
||||||
|
* Transmission nb | Data Rate
|
||||||
|
* ----------------|-----------
|
||||||
|
* 1 (first) | DR
|
||||||
|
* 2 | DR
|
||||||
|
* 3 | max(DR-1,0)
|
||||||
|
* 4 | max(DR-1,0)
|
||||||
|
* 5 | max(DR-2,0)
|
||||||
|
* 6 | max(DR-2,0)
|
||||||
|
* 7 | max(DR-3,0)
|
||||||
|
* 8 | max(DR-3,0)
|
||||||
|
*
|
||||||
|
* Note, that if NbTrials is set to 1 or 2, the MAC will not decrease
|
||||||
|
* the datarate, in case the LoRaMAC layer did not receive an acknowledgment
|
||||||
|
*/
|
||||||
|
uint8_t confirmedNbTrials = 4;
|
||||||
|
|
||||||
|
void device_restart()
|
||||||
|
{
|
||||||
|
DEBUG_PRINTLN("Restarting device...");
|
||||||
|
delay(1000);
|
||||||
|
ESP.restart();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Prepares the payload of the frame */
|
||||||
|
static void prepareTxFrame(uint8_t port)
|
||||||
|
{
|
||||||
|
/*appData size is LORAWAN_APP_DATA_MAX_SIZE which is defined in "commissioning.h".
|
||||||
|
*appDataSize max value is LORAWAN_APP_DATA_MAX_SIZE.
|
||||||
|
*if enabled AT, don't modify LORAWAN_APP_DATA_MAX_SIZE, it may cause system hanging or failure.
|
||||||
|
*if disabled AT, LORAWAN_APP_DATA_MAX_SIZE can be modified, the max value is reference to lorawan region and SF.
|
||||||
|
*for example, if use REGION_CN470,
|
||||||
|
*the max value for different DR can be found in MaxPayloadOfDatarateCN470 refer to DataratesCN470 and BandwidthsCN470 in "RegionCN470.h".
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (Serial.available() > 0)
|
||||||
|
{
|
||||||
|
String incomingData = Serial.readStringUntil('\n'); // Read until newline
|
||||||
|
DEBUG_PRINT("Received message: ");
|
||||||
|
DEBUG_PRINTLN(incomingData);
|
||||||
|
|
||||||
|
unsigned char payload[32];
|
||||||
|
incomingData.getBytes(payload, incomingData.length() + 1); // Convert String to byte array
|
||||||
|
|
||||||
|
uint8_t command = payload[0];
|
||||||
|
String data = incomingData.substring(1); // Extract data after the command byte
|
||||||
|
|
||||||
|
DEBUG_PRINTLN("Command byte: " + String(command));
|
||||||
|
DEBUG_PRINTLN("Data: " + data);
|
||||||
|
// The payload structure is defined as follows:
|
||||||
|
// Byte 0 [command byte] -> '1' for sending data, '2' for restart device, '3' for change app EUI, '4' for change app key
|
||||||
|
|
||||||
|
switch (command)
|
||||||
|
{
|
||||||
|
// Note: The command byte is expected to be a character representing a number, so we compare it with the ASCII values of '1', '2', '3', and '4'.
|
||||||
|
case 49:
|
||||||
|
DEBUG_PRINTLN("Command: Send Data");
|
||||||
|
appDataSize = data.length();
|
||||||
|
|
||||||
|
memcpy(appData, payload + 1, appDataSize);
|
||||||
|
DEBUG_PRINTLN("Send Data");
|
||||||
|
LoRaWAN.send();
|
||||||
|
break;
|
||||||
|
case 50:
|
||||||
|
DEBUG_PRINTLN("Command: Restart Device");
|
||||||
|
device_restart();
|
||||||
|
break;
|
||||||
|
case 51:
|
||||||
|
DEBUG_PRINTLN("Command: Change App EUI");
|
||||||
|
if (data.length() != 8)
|
||||||
|
{
|
||||||
|
DEBUG_PRINTLN("Invalid App EUI length. Expected 8 characters.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// Replace appEui in EEPROM
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
{
|
||||||
|
EEPROM.write(APP_EUI_ADDRESS + i, data[i]);
|
||||||
|
}
|
||||||
|
EEPROM.commit();
|
||||||
|
DEBUG_PRINTLN("App EUI updated in EEPROM");
|
||||||
|
// Restart device to apply new appEui
|
||||||
|
device_restart();
|
||||||
|
break;
|
||||||
|
case 52:
|
||||||
|
DEBUG_PRINTLN("Command: Change App Key");
|
||||||
|
if (data.length() != 16)
|
||||||
|
{
|
||||||
|
DEBUG_PRINTLN("Invalid App Key length. Expected 16 characters.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// Replace appKey in EEPROM
|
||||||
|
for (int i = 0; i < 16; i++)
|
||||||
|
{
|
||||||
|
EEPROM.write(APP_KEY_ADDRESS + i, data[i]);
|
||||||
|
}
|
||||||
|
EEPROM.commit();
|
||||||
|
DEBUG_PRINTLN("App Key updated in EEPROM");
|
||||||
|
// Restart device to apply new appKey
|
||||||
|
device_restart();
|
||||||
|
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
DEBUG_PRINTLN("Unknown command");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
memset(appData, 0, sizeof(appData));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// if true, next uplink will add MOTE_MAC_DEVICE_TIME_REQ
|
||||||
|
|
||||||
|
void setup()
|
||||||
|
{
|
||||||
|
EEPROM.begin(32); // Initialize EEPROM with a size of 32 bytes
|
||||||
|
// Read appEui and appKey from EEPROM
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
{
|
||||||
|
appEui[i] = EEPROM.read(APP_EUI_ADDRESS + i);
|
||||||
|
}
|
||||||
|
for (int i = 0; i < 16; i++)
|
||||||
|
{
|
||||||
|
appKey[i] = EEPROM.read(APP_KEY_ADDRESS + i);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set default appEui and appKey if EEPROM is empty (all bytes are 0xFF)
|
||||||
|
if (appEui[0] == 0xFF && appEui[1] == 0xFF && appEui[2] == 0xFF && appEui[3] == 0xFF &&
|
||||||
|
appEui[4] == 0xFF && appEui[5] == 0xFF && appEui[6] == 0xFF && appEui[7] == 0xFF)
|
||||||
|
{
|
||||||
|
|
||||||
|
memcpy(appEui, defaultAppEui, sizeof(defaultAppEui));
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
{
|
||||||
|
EEPROM.write(APP_EUI_ADDRESS + i, appEui[i]);
|
||||||
|
}
|
||||||
|
EEPROM.commit();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appKey[0] == 0xFF && appKey[1] == 0xFF && appKey[2] == 0xFF && appKey[3] == 0xFF &&
|
||||||
|
appKey[4] == 0xFF && appKey[5] == 0xFF && appKey[6] == 0xFF && appKey[7] == 0xFF &&
|
||||||
|
appKey[8] == 0xFF && appKey[9] == 0xFF && appKey[10] == 0xFF && appKey[11] == 0xFF &&
|
||||||
|
appKey[12] == 0xFF && appKey[13] == 0xFF && appKey[14] == 0xFF && appKey[15] == 0xFF)
|
||||||
|
{
|
||||||
|
|
||||||
|
memcpy(appKey, defaultAppKey, sizeof(defaultAppKey));
|
||||||
|
for (int i = 0; i < 16; i++)
|
||||||
|
{
|
||||||
|
EEPROM.write(APP_KEY_ADDRESS + i, appKey[i]);
|
||||||
|
}
|
||||||
|
EEPROM.commit();
|
||||||
|
}
|
||||||
|
|
||||||
|
Serial.begin(115200);
|
||||||
|
while (!Serial)
|
||||||
|
;
|
||||||
|
|
||||||
|
DEBUG_PRINTLN("APP EUI: ");
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
{
|
||||||
|
DEBUG_PRINT(appEui[i]);
|
||||||
|
if (i < 7)
|
||||||
|
DEBUG_PRINT(":");
|
||||||
|
}
|
||||||
|
DEBUG_PRINTLN();
|
||||||
|
|
||||||
|
DEBUG_PRINTLN("APP Key: ");
|
||||||
|
for (int i = 0; i < 16; i++)
|
||||||
|
{
|
||||||
|
DEBUG_PRINT(appKey[i]);
|
||||||
|
if (i < 15)
|
||||||
|
DEBUG_PRINT(":");
|
||||||
|
}
|
||||||
|
DEBUG_PRINTLN();
|
||||||
|
|
||||||
|
Mcu.begin(HELTEC_BOARD, SLOW_CLK_TPYE);
|
||||||
|
DEBUG_PRINTLN("HELTEC CT-R2 ready");
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop()
|
||||||
|
{
|
||||||
|
switch (deviceState)
|
||||||
|
{
|
||||||
|
case DEVICE_STATE_INIT:
|
||||||
|
{
|
||||||
|
DEBUG_PRINTLN("Device INIT");
|
||||||
|
#if (LORAWAN_DEVEUI_AUTO)
|
||||||
|
LoRaWAN.generateDeveuiByChipID();
|
||||||
|
#endif
|
||||||
|
LoRaWAN.init(loraWanClass, loraWanRegion);
|
||||||
|
// both set join DR and DR when ADR off
|
||||||
|
LoRaWAN.setDefaultDR(3);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_JOIN:
|
||||||
|
{
|
||||||
|
DEBUG_PRINTLN("Join Network");
|
||||||
|
LoRaWAN.join();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_SEND:
|
||||||
|
{
|
||||||
|
|
||||||
|
prepareTxFrame(appPort);
|
||||||
|
|
||||||
|
deviceState = DEVICE_STATE_CYCLE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_CYCLE:
|
||||||
|
{
|
||||||
|
DEBUG_PRINTLN("Schedule packet transmission");
|
||||||
|
// Schedule next packet transmission
|
||||||
|
// txDutyCycleTime = appTxDutyCycle + randr(-APP_TX_DUTYCYCLE_RND, APP_TX_DUTYCYCLE_RND);
|
||||||
|
txDutyCycleTime = appTxDutyCycle;
|
||||||
|
|
||||||
|
LoRaWAN.cycle(txDutyCycleTime);
|
||||||
|
deviceState = DEVICE_STATE_SLEEP;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DEVICE_STATE_SLEEP:
|
||||||
|
{
|
||||||
|
// DEBUG_PRINTLN("Device Sleep");
|
||||||
|
LoRaWAN.sleep(loraWanClass);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
deviceState = DEVICE_STATE_INIT;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
163
README.md
Normal file
163
README.md
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
# LoRaWAN Sensor HT-CT62
|
||||||
|
|
||||||
|
A LoRaWAN communication project for Heltec HT-CT62 devices, featuring two Arduino sketch implementations for wireless sensor data transmission.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
This project provides LoRaWAN connectivity solutions for the Heltec HT-CT62 device, supporting both basic and advanced communication modes. The system enables remote sensor monitoring through LoRaWAN networks with configurable parameters and multiple authentication methods.
|
||||||
|
|
||||||
|
## Project Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
├── LoRaWan/ # Basic LoRaWAN implementation
|
||||||
|
│ └── LoRaWan.ino # Simple LoRaWAN communication sketch
|
||||||
|
├── LoRaWanNode/ # Advanced LoRaWAN node implementation
|
||||||
|
│ └── LoRaWanNode.ino # Feature-rich node with serial commands
|
||||||
|
├── HT-CT62(Rev1.1).pdf # Device datasheet and technical documentation
|
||||||
|
└── README.md # This file
|
||||||
|
```
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
### Basic Implementation (LoRaWan/)
|
||||||
|
- Standard LoRaWAN Class A communication
|
||||||
|
- OTAA (Over-The-Air Activation) support
|
||||||
|
- Mock data transmission for testing
|
||||||
|
- 15-second transmission intervals
|
||||||
|
- Confirmed message transmission
|
||||||
|
|
||||||
|
### Advanced Implementation (LoRaWanNode/)
|
||||||
|
- Enhanced LoRaWAN Class C communication
|
||||||
|
- Serial command interface for real-time control
|
||||||
|
- EEPROM storage for persistent configuration
|
||||||
|
- Debug mode with detailed logging
|
||||||
|
- Dynamic payload management
|
||||||
|
- Remote device restart capability
|
||||||
|
- Runtime App EUI and App Key modification
|
||||||
|
|
||||||
|
## Hardware Requirements
|
||||||
|
|
||||||
|
- **Device**: Heltec HT-CT62 (Rev 1.1 or later)
|
||||||
|
- **Connectivity**: LoRaWAN gateway within range
|
||||||
|
- **Development**: Arduino IDE with Heltec board support
|
||||||
|
|
||||||
|
## Software Requirements
|
||||||
|
|
||||||
|
- Arduino IDE
|
||||||
|
- Heltec ESP32 board package
|
||||||
|
- LoRaWAN library for Heltec devices
|
||||||
|
|
||||||
|
## Configuration
|
||||||
|
|
||||||
|
### Network Parameters
|
||||||
|
|
||||||
|
The project supports both OTAA and ABP activation methods:
|
||||||
|
|
||||||
|
#### OTAA Configuration
|
||||||
|
```cpp
|
||||||
|
uint8_t devEui[] = {0x70, 0xB3, 0xD5, 0x7E, 0xD0, 0x06, 0x53, 0xC8};
|
||||||
|
uint8_t appEui[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
||||||
|
uint8_t appKey[] = {0x74, 0xD6, 0x6E, 0x63, 0x45, 0x82, 0x48, 0x27, 0xFE, 0xC5, 0xB7, 0x70, 0xBA, 0x2B, 0x50, 0x45};
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Regional Settings
|
||||||
|
- **Default Region**: Configurable via Arduino IDE tools
|
||||||
|
- **Channel Mask**: Channels 0-7 enabled by default
|
||||||
|
- **Data Rate**: DR3 (configurable)
|
||||||
|
|
||||||
|
### Advanced Node Serial Commands
|
||||||
|
|
||||||
|
The advanced implementation supports real-time commands via serial interface:
|
||||||
|
|
||||||
|
| Command | Format | Description |
|
||||||
|
|---------|--------|-------------|
|
||||||
|
| `1` + data | `1Hello World` | Send custom data payload |
|
||||||
|
| `2` | `2` | Restart the device |
|
||||||
|
| `3` + 8 bytes | `3XXXXXXXX` | Change App EUI |
|
||||||
|
| `4` + 16 bytes | `4XXXXXXXXXXXXXXXX` | Change App Key |
|
||||||
|
|
||||||
|
## Installation and Setup
|
||||||
|
|
||||||
|
1. **Install Arduino IDE** with Heltec ESP32 support
|
||||||
|
2. **Clone/Download** this repository
|
||||||
|
3. **Open** desired sketch (`LoRaWan.ino` or `LoRaWanNode.ino`)
|
||||||
|
4. **Configure** LoRaWAN parameters for your network
|
||||||
|
5. **Select** correct board and region in Arduino IDE
|
||||||
|
6. **Upload** the sketch to your HT-CT62 device
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
### Basic Implementation
|
||||||
|
1. Power on the device
|
||||||
|
2. Monitor serial output for connection status
|
||||||
|
3. Device will automatically join the network and transmit data every 15 seconds
|
||||||
|
|
||||||
|
### Advanced Implementation
|
||||||
|
1. Power on and monitor serial output
|
||||||
|
2. Send commands via serial interface for real-time control:
|
||||||
|
```
|
||||||
|
1sensor_data_123 # Send sensor data
|
||||||
|
2 # Restart device
|
||||||
|
3APPEUI01 # Set new App EUI
|
||||||
|
4APPKEY1234567890AB # Set new App Key
|
||||||
|
```
|
||||||
|
|
||||||
|
## Debug Mode
|
||||||
|
|
||||||
|
Enable debug output in the advanced implementation:
|
||||||
|
```cpp
|
||||||
|
#define DEBUG_MODE 1 // Set to 1 to enable debug messages
|
||||||
|
```
|
||||||
|
|
||||||
|
## Transmission Settings
|
||||||
|
|
||||||
|
- **Duty Cycle**: 15 seconds (configurable)
|
||||||
|
- **Confirmed Messages**: Enabled
|
||||||
|
- **Retry Attempts**: 4 attempts with adaptive data rate
|
||||||
|
- **Application Port**: 2
|
||||||
|
- **Max Payload**: 32 bytes
|
||||||
|
|
||||||
|
## Technical Specifications
|
||||||
|
|
||||||
|
- **Device Class**: A (basic) / C (advanced)
|
||||||
|
- **Activation**: OTAA preferred, ABP supported
|
||||||
|
- **ADR**: Adaptive Data Rate enabled
|
||||||
|
- **Regional Support**: Multiple regions via Arduino IDE configuration
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Common Issues
|
||||||
|
1. **Join Failed**: Check devEui, appEui, and appKey configuration
|
||||||
|
2. **No Gateway**: Ensure LoRaWAN gateway is within range
|
||||||
|
3. **Region Mismatch**: Verify region settings match your location
|
||||||
|
4. **Payload Too Large**: Keep data payload ≤ 32 bytes
|
||||||
|
|
||||||
|
### Debug Steps
|
||||||
|
1. Enable debug mode in advanced implementation
|
||||||
|
2. Monitor serial output for detailed logs
|
||||||
|
3. Verify network credentials with your LoRaWAN provider
|
||||||
|
4. Check regional frequency plan compatibility
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
This project is based on Heltec Automation examples and follows their licensing terms.
|
||||||
|
|
||||||
|
## Contributing
|
||||||
|
|
||||||
|
1. Fork the repository
|
||||||
|
2. Create a feature branch
|
||||||
|
3. Commit your changes
|
||||||
|
4. Push to the branch
|
||||||
|
5. Create a Pull Request
|
||||||
|
|
||||||
|
## Support
|
||||||
|
|
||||||
|
- **Heltec Documentation**: https://www.heltec.org
|
||||||
|
- **LoRaWAN Specification**: LoRaWAN 1.0.2+
|
||||||
|
- **Hardware Manual**: See `HT-CT62(Rev1.1).pdf` for detailed specifications
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Manufacturer**: Heltec Automation
|
||||||
|
**Website**: www.heltec.org
|
||||||
|
**Device**: HT-CT62 LoRaWAN Sensor Node
|
||||||
Loading…
Reference in New Issue
Block a user