/**
|
******************************************************************************
|
* @file : KWireLess.c
|
* @brief : K-WireLess Protocol program body
|
******************************************************************************
|
*/
|
#include "KWireLess.h"
|
#include "functions.h"
|
#include "string.h"
|
#include "PLCFunctions.h"
|
#include "stm32f0xx_hal.h"
|
|
|
/**************************************************************************************************************************************
|
Demo 程序流程 RadioEnableMaster=true 为主机端,主机端发送一个"PING"数据后切换到接收,等待从机返回的应答"PONG"数据LED闪烁
|
|
RadioEnableMaster=false 为从机端,从机端接收到主机端发过来的"PING"数据后LED闪烁并发送一个"PONG"数据作为应答
|
***************************************************************************************************************************************/
|
|
#define USE_MODEM_LORA
|
//#define USE_MODEM_FSK
|
|
#define REGION_CN779
|
|
#if defined( REGION_AS923 )
|
|
#define RF_FREQUENCY 923000000 // Hz
|
|
#elif defined( REGION_AU915 )
|
|
#define RF_FREQUENCY 915000000 // Hz
|
|
#elif defined( REGION_CN779 )
|
|
#define RF_FREQUENCY 432000000 // Hz
|
|
#elif defined( REGION_EU868 )
|
|
#define RF_FREQUENCY 868000000 // Hz
|
|
#elif defined( REGION_KR920 )
|
|
#define RF_FREQUENCY 920000000 // Hz
|
|
#elif defined( REGION_IN865 )
|
|
#define RF_FREQUENCY 865000000 // Hz
|
|
#elif defined( REGION_US915 )
|
|
#define RF_FREQUENCY 915000000 // Hz
|
|
#elif defined( REGION_US915_HYBRID )
|
|
#define RF_FREQUENCY 915000000 // Hz
|
|
#else
|
|
#error "Please define a frequency band in the compiler options."
|
|
#endif
|
|
#define nChannelBandwidth 200000 // Hz
|
#define TX_OUTPUT_POWER 22 // 22 dBm
|
|
extern bool IrqFired;
|
|
bool RadioEnableMaster=true;//主从选择
|
uint32_t nRadioChannel=0;
|
uint16_t crc_value;
|
/*!
|
* Radio events function pointer
|
*/
|
|
static RadioEvents_t RadioEvents;
|
|
stWLRunStat KwRunStat;
|
|
#if defined( USE_MODEM_LORA )
|
|
#define LORA_BANDWIDTH 0 // [0: 125 kHz,
|
// 1: 250 kHz,
|
// 2: 500 kHz,
|
// 3: Reserved]
|
#define LORA_SPREADING_FACTOR 6 // [SF7..SF12]
|
#define LORA_CODINGRATE 1 // [1: 4/5,
|
// 2: 4/6,
|
// 3: 4/7,
|
// 4: 4/8]
|
#define LORA_PREAMBLE_LENGTH 8 // Same for Tx and Rx
|
#define LORA_SYMBOL_TIMEOUT 0 // Symbols
|
#define LORA_FIX_LENGTH_PAYLOAD_ON false
|
#define LORA_IQ_INVERSION_ON false
|
|
#elif defined( USE_MODEM_FSK )
|
|
#define FSK_FDEV 20e3 // Hz
|
#define FSK_DATARATE 19.2e3 // bps
|
#define FSK_BANDWIDTH 60e3 // Hz >> DSB in sx126x
|
#define FSK_AFC_BANDWIDTH 200e3 // Hz
|
#define FSK_PREAMBLE_LENGTH 5 // Same for Tx and Rx
|
#define FSK_FIX_LENGTH_PAYLOAD_ON false
|
|
#else
|
#error "Please define a modem in the compiler options."
|
#endif
|
|
typedef enum
|
{
|
LOWPOWER,
|
RX,
|
RX_TIMEOUT,
|
RX_ERROR,
|
TX,
|
TX_TIMEOUT,
|
}States_t;
|
|
#define MASTER_RX_TIMEOUT_VALUE 60
|
#define SLAVE_RX_TIMEOUT_VALUE 200
|
|
#define BUFFER_SIZE 32 // Define the payload size here
|
|
const uint8_t PingMsg[] = "PING";
|
const uint8_t PongMsg[] = "PONG";
|
|
uint16_t BufferSize = BUFFER_SIZE;
|
uint8_t TX_Buffer[BUFFER_SIZE];
|
uint8_t RX_Buffer[BUFFER_SIZE];
|
|
|
States_t State = LOWPOWER;
|
|
int8_t RssiValue = 0;
|
int8_t SnrValue = 0;
|
|
KWStates KW_State=KW_PON;
|
|
void LedToggle(void)
|
{
|
// GPIO_WriteBit( LED1_PORT, LED1_PIN,Bit_RESET);//LED闪烁
|
// HAL_Delay_nMS(10);
|
// GPIO_WriteBit( LED1_PORT, LED1_PIN,Bit_SET);
|
|
LL_GPIO_TogglePin(GPIOC,LL_GPIO_PIN_13);
|
}
|
|
int KWireLessInit(bool bRadioEnableMaster, uint32_t nChn)
|
{
|
|
RadioEnableMaster = bRadioEnableMaster;
|
// Radio initialization
|
RadioEvents.TxDone = OnTxDone;
|
RadioEvents.RxDone = OnRxDone;
|
RadioEvents.TxTimeout = OnTxTimeout;
|
RadioEvents.RxTimeout = OnRxTimeout;
|
RadioEvents.RxError = OnRxError;
|
RadioEvents.CadDone = OnCadDone;
|
|
Radio.Init( &RadioEvents );
|
nRadioChannel = nChn;
|
Radio.SetChannel( RF_FREQUENCY + nRadioChannel * nChannelBandwidth );
|
|
// Radio.WriteBuffer(0x06C0,data,2);
|
// Radio.ReadBuffer(0x06C0,test,2);
|
|
#if defined( USE_MODEM_LORA )
|
|
Radio.SetTxConfig( MODEM_LORA, TX_OUTPUT_POWER, 0, LORA_BANDWIDTH,
|
LORA_SPREADING_FACTOR, LORA_CODINGRATE,
|
LORA_PREAMBLE_LENGTH, LORA_FIX_LENGTH_PAYLOAD_ON,
|
true, 0, 0, LORA_IQ_INVERSION_ON, 3000 );
|
|
Radio.SetRxConfig( MODEM_LORA, LORA_BANDWIDTH, LORA_SPREADING_FACTOR,
|
LORA_CODINGRATE, 0, LORA_PREAMBLE_LENGTH,
|
LORA_SYMBOL_TIMEOUT, LORA_FIX_LENGTH_PAYLOAD_ON,
|
0, true, 0, 0, LORA_IQ_INVERSION_ON, false );
|
|
#elif defined( USE_MODEM_FSK )
|
|
Radio.SetTxConfig( MODEM_FSK, TX_OUTPUT_POWER, FSK_FDEV, 0,
|
FSK_DATARATE, 0,
|
FSK_PREAMBLE_LENGTH, FSK_FIX_LENGTH_PAYLOAD_ON,
|
true, 0, 0, 0, 3000 );
|
|
Radio.SetRxConfig( MODEM_FSK, FSK_BANDWIDTH, FSK_DATARATE,
|
0, FSK_AFC_BANDWIDTH, FSK_PREAMBLE_LENGTH,
|
0, FSK_FIX_LENGTH_PAYLOAD_ON, 0, true,
|
0, 0,false, false );
|
#else
|
#error "Please define a frequency band in the compiler options."
|
#endif
|
SX126xSetRxTxFallbackMode(0x40); // 0x40-> FS 0x30 -> SD_XOSC 0x20 -> SD_RC
|
KwRunStat.runStep=RS_IDEL;
|
return 0;
|
}
|
int KWireLessStart(void)
|
{
|
if(RadioEnableMaster)
|
{
|
KWLMasterSendReqPkt(1);
|
}
|
else
|
{
|
Radio.Rx( SLAVE_RX_TIMEOUT_VALUE );
|
KwRunStat.runStep=RS_RECVING;
|
|
KwRunStat.lastActTime = GetuS();
|
KwRunStat.lastRecvtime = GetuS();
|
}
|
|
// while( 1 )
|
// {
|
// Radio.IrqProcess( ); // Process Radio IRQ
|
// }
|
return 0;
|
}
|
|
void OnTxDone( void )
|
{
|
KwRunStat.runStep=RS_IDEL;
|
KMem.WDT[42]++;
|
KwRunStat.lastSenttime = GetuS();
|
KwRunStat.lastAckTime = GetuS();
|
Radio.Standby();
|
if (RadioEnableMaster) {
|
Radio.Rx( MASTER_RX_TIMEOUT_VALUE ); //进入接收
|
} else {
|
Radio.Rx( SLAVE_RX_TIMEOUT_VALUE ); //进入接收
|
}
|
KwRunStat.runStep=RS_RECVING;
|
|
KwRunStat.lastActTime = GetuS();
|
KwRunStat.lastRecvtime = GetuS();
|
LL_GPIO_TogglePin(GPIOC,LL_GPIO_PIN_14);
|
int us2=GetuS();
|
KMem.ScanTimeuS=us2-KMem.LastScanTime;
|
KMem.LastScanTime = us2;
|
if (KMem.ScanTimeuS < KMem.MinScanTimeuS) {KMem.MinScanTimeuS = KMem.ScanTimeuS;}
|
if (KMem.ScanTimeuS > KMem.MaxScanTimeuS) {KMem.MaxScanTimeuS = KMem.ScanTimeuS;}
|
|
}
|
|
void OnTxTimeout( void )
|
{
|
KwRunStat.runStep=RS_IDEL;
|
KwRunStat.lastAckTime = GetuS();
|
KwRunStat.TXErr++;
|
Radio.Standby();
|
KMem.WDT[44]++;
|
}
|
bool bThisRxError=0;
|
void OnRxDone( uint8_t *payload, uint16_t size, int16_t rssi, int8_t snr )
|
{
|
Radio.Standby();
|
if (bThisRxError) {bThisRxError=0;return;}
|
KwRunStat.lastRecvdtime=GetuS();
|
KwRunStat.lastAckTime = GetuS();
|
|
KwRunStat.curStat = 0;
|
KwRunStat.latancy = GetuS() - KwRunStat.lastSendtime;
|
BufferSize = size;
|
memcpy( RX_Buffer, payload, BufferSize );
|
RssiValue = rssi;
|
SnrValue = snr;
|
|
KwRunStat.RSSI = RssiValue;
|
KwRunStat.SNR = SnrValue;
|
|
KMem.WDT[40]=-RssiValue; //Radio.Rssi(MODEM_FSK);;
|
KMem.WDT[40]=-RssiValue + ((-SnrValue) <<8);
|
|
KMem.WDT[43]++;
|
KwRunStat.recvCount++;
|
KwRunStat.CtnLstPkts=0;
|
bool CRC_OK =1;
|
/*
|
crc_value=RadioComputeCRC(RX_Buffer,size-2,CRC_TYPE_IBM);//计算得出要发送数据包CRC值
|
if (RX_Buffer[size-2] != (crc_value&0xff) && RX_Buffer[size-1] != (crc_value >> 8))
|
{
|
KwRunStat.CRCErr++;
|
KMem.ErrStat=500;
|
CRC_OK = 0;
|
}else {
|
CRC_OK = 1;
|
}
|
*/
|
unsigned char indicator1 = 128+RssiValue;
|
unsigned char indicator2;
|
const unsigned char maxrssi = 100;
|
const unsigned char minrssi = 30;
|
if (indicator1 > maxrssi) indicator2=8;
|
else if (indicator1 < minrssi) indicator2 = 0;
|
|
else indicator2 = (indicator1 - minrssi)*7/(maxrssi-minrssi) + 1;
|
|
unsigned char indicator3 = (1 << indicator2) -1;
|
if(RadioEnableMaster)
|
{
|
if(memcmp(RX_Buffer,PongMsg,4)==0 && CRC_OK)
|
{
|
KWLMasterParsePkt(1);
|
KwRunStat.tRSSI = RX_Buffer[10];
|
KwRunStat.tSNR = RX_Buffer[11];
|
KMem.WDT[41]=(-RX_Buffer[10]) + ((-RX_Buffer[11])<<8);
|
|
LedToggle();//LED闪烁
|
KMem.WFX[1]=(RX_Buffer[4]) + (RX_Buffer[5]<<8);
|
// KMem.WFX[1]=(RX_Buffer[4]) + (indicator3<<8);
|
|
KMem.WFX[2]=(RX_Buffer[6]) + (RX_Buffer[7]<<8);
|
}
|
//KWLMasterSendReqPkt(1);
|
}
|
else //slave
|
{
|
KWLSlaveParsePkt(1);
|
if(memcmp(RX_Buffer,PingMsg,4)==0 && CRC_OK)
|
{
|
LedToggle();//LED闪烁
|
KMem.WFX[1]=(RX_Buffer[4]) + (RX_Buffer[5]<<8);
|
// KMem.WFX[1]=(RX_Buffer[4]) + (indicator3<<8);
|
|
KMem.WFX[2]=(RX_Buffer[6]) + (RX_Buffer[7]<<8);
|
|
KWLSlaveSendRplyPkt(1);
|
|
// KMem.WFY[0]=(RX_Buffer[4]<<8) + RX_Buffer[5];
|
// KMem.WFY[1]=(RX_Buffer[6]<<8) + RX_Buffer[7];
|
}
|
else
|
{
|
Radio.Rx( SLAVE_RX_TIMEOUT_VALUE );
|
KwRunStat.runStep=RS_RECVING;
|
KwRunStat.lastActTime = GetuS();
|
KwRunStat.lastRecvtime = GetuS();
|
|
}
|
}
|
}
|
|
void OnRxTimeout( void )
|
{
|
KwRunStat.runStep=RS_IDEL;
|
|
Radio.Standby();
|
KwRunStat.lastAckTime = GetuS();
|
|
KMem.WDT[45]++;
|
KwRunStat.LostPackets++;
|
KwRunStat.CtnLstPkts++;
|
if (KwRunStat.CtnLstPkts > KwRunStat.MaxCtnLstPkts) {KwRunStat.MaxCtnLstPkts = KwRunStat.CtnLstPkts;}
|
KMem.ErrStat=500;
|
if (KwRunStat.CtnLstPkts > 3) { KMem.ErrStat=5000; }
|
if (KwRunStat.CtnLstPkts > 10) { KMem.ErrStat=5000; KMem.WFX[1]=0; }
|
|
|
if(RadioEnableMaster)
|
{
|
//KWLMasterSendReqPkt(1);
|
}
|
else
|
{
|
Radio.Rx( SLAVE_RX_TIMEOUT_VALUE );
|
KwRunStat.runStep=RS_RECVING;
|
KwRunStat.lastActTime = GetuS();
|
KwRunStat.lastRecvtime = GetuS();
|
}
|
}
|
|
|
void OnRxError( void )
|
{
|
|
// Radio.Standby();
|
KwRunStat.RXErr++;
|
KMem.WDT[46]++;
|
bThisRxError=1;
|
|
if(RadioEnableMaster)
|
{
|
// SendPingMsg();
|
}
|
else
|
{
|
// Radio.Rx( RX_TIMEOUT_VALUE );
|
//KwRunStat.lastActTime = GetuS();
|
KwRunStat.LostPackets++;
|
KwRunStat.CtnLstPkts++;
|
if (KwRunStat.CtnLstPkts > KwRunStat.MaxCtnLstPkts) {KwRunStat.MaxCtnLstPkts = KwRunStat.CtnLstPkts;}
|
KMem.ErrStat=500;
|
if (KwRunStat.CtnLstPkts > 3) { KMem.ErrStat=5000; }
|
if (KwRunStat.CtnLstPkts > 10) { KMem.ErrStat=5000; KMem.WFX[1]=0; }
|
|
}
|
|
}
|
|
void OnCadDone( bool channelActivityDetected)
|
{
|
return;
|
}
|
|
int KWLMasterSendReqPkt(int nChn)
|
{
|
int len1=12;
|
TX_Buffer[0] = 'P';
|
TX_Buffer[1] = 'I';
|
TX_Buffer[2] = 'N';
|
TX_Buffer[3] = 'G';
|
TX_Buffer[4] = KMem.WFY[1];
|
TX_Buffer[5] = KMem.WFY[1]>>8;
|
TX_Buffer[6] = KMem.WFY[2];
|
TX_Buffer[7] = KMem.WFY[2]>>8;
|
|
crc_value=RadioComputeCRC(TX_Buffer,len1,CRC_TYPE_IBM);//计算得出要发送数据包CRC值
|
TX_Buffer[len1]=crc_value;
|
TX_Buffer[len1+1]=crc_value>>8;
|
|
KMem.WDT[56]=crc_value;
|
|
KwRunStat.sentCount++;
|
KwRunStat.cycleTime = GetuS()- KwRunStat.lastSendtime ;
|
KwRunStat.lastSendtime = GetuS();
|
Radio.Send( TX_Buffer, len1+2);
|
KwRunStat.runStep=RS_SENDING;
|
KwRunStat.lastActTime = GetuS();
|
|
return 0;
|
}
|
|
int KWLSlaveSendRplyPkt(int nChn)
|
{
|
int len1=12;
|
TX_Buffer[0] = 'P';
|
TX_Buffer[1] = 'O';
|
TX_Buffer[2] = 'N';
|
TX_Buffer[3] = 'G';
|
TX_Buffer[4] = KMem.WFY[1];
|
TX_Buffer[5] = KMem.WFY[1]>>8;
|
TX_Buffer[6] = KMem.WFY[2];
|
TX_Buffer[7] = KMem.WFY[2]>>8;
|
|
TX_Buffer[10] = RssiValue;
|
TX_Buffer[11] = SnrValue;
|
|
crc_value=RadioComputeCRC(TX_Buffer,len1,CRC_TYPE_IBM);//计算得出要发送数据包CRC值
|
TX_Buffer[len1]=crc_value;
|
TX_Buffer[len1+1]=crc_value>>8;
|
Radio.Send( TX_Buffer, len1+2);
|
KwRunStat.runStep=RS_SENDING;
|
|
KwRunStat.lastActTime = GetuS();
|
KwRunStat.sentCount++;
|
KwRunStat.lastSendtime = GetuS();
|
return 0;
|
}
|
|
|
int KWLMasterProc(int nChn)
|
{
|
return 0;
|
}
|
|
int KWLSlaveProc(int nChn)
|
{
|
return 0;
|
}
|
|
int KWL_Process(int nChn)
|
{
|
RadioState_t stat = Radio.GetStatus();
|
KMem.WDT[32]=stat;
|
// KMem.WDT[38]=Radio.Rssi(MODEM_FSK);
|
if (RadioEnableMaster){
|
if (GetuS() - KwRunStat.lastSendtime >100000 && KwRunStat.curStat == 0 ) //&& stat == RF_IDLE )
|
{
|
KWireLessStart();
|
}
|
}else
|
{
|
if (stat == RF_IDLE){
|
KMem.WDT[48]++;
|
Radio.Standby();
|
KWireLessStart();
|
}
|
if (stat == RF_RX_RUNNING ){
|
if (GetuS() - KwRunStat.lastRecvtime > 300000){
|
KMem.WDT[49]++;
|
Radio.Standby();
|
KWireLessStart();
|
}
|
}
|
if (stat == RF_CAD && GetuS() - KwRunStat.lastRecvtime > 400000){
|
KMem.WDT[50]++;
|
Radio.Standby();
|
KWireLessStart();
|
}
|
|
}
|
return 0;
|
}
|
|
int KWLMasterParsePkt(int nChn)
|
{
|
return 0;
|
}
|
int KWLSlaveParsePkt(int nChn)
|
{
|
return 0;
|
}
|