mirror of
https://github.com/lumapu/ahoy.git
synced 2025-05-21 12:56:11 +02:00
Update hmRadio.h
This commit is contained in:
parent
155d602e6e
commit
6159c07559
1 changed files with 131 additions and 198 deletions
329
src/hm/hmRadio.h
329
src/hm/hmRadio.h
|
@ -9,28 +9,10 @@
|
||||||
#include "../utils/dbg.h"
|
#include "../utils/dbg.h"
|
||||||
#include <RF24.h>
|
#include <RF24.h>
|
||||||
#include "../utils/crc.h"
|
#include "../utils/crc.h"
|
||||||
#ifndef DISABLE_IRQ
|
|
||||||
#if defined(ESP8266) || defined(ESP32)
|
|
||||||
#define DISABLE_IRQ noInterrupts()
|
|
||||||
#define RESTORE_IRQ interrupts()
|
|
||||||
#else
|
|
||||||
#define DISABLE_IRQ \
|
|
||||||
uint8_t sreg = SREG; \
|
|
||||||
cli();
|
|
||||||
|
|
||||||
#define RESTORE_IRQ \
|
#define SPI_SPEED 1000000
|
||||||
SREG = sreg;
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
//#define CHANNEL_HOP // switch between channels or use static channel to send
|
|
||||||
|
|
||||||
#define DEFAULT_RECV_CHANNEL 3
|
#define RF_CHANNELS 5
|
||||||
#define SPI_SPEED 1000000
|
|
||||||
|
|
||||||
#define DUMMY_RADIO_ID ((uint64_t)0xDEADBEEF01ULL)
|
|
||||||
|
|
||||||
#define RF_CHANNELS 5
|
|
||||||
#define RF_LOOP_CNT 300
|
|
||||||
|
|
||||||
#define TX_REQ_INFO 0x15
|
#define TX_REQ_INFO 0x15
|
||||||
#define TX_REQ_DEVCONTROL 0x51
|
#define TX_REQ_DEVCONTROL 0x51
|
||||||
|
@ -61,11 +43,12 @@ const char* const rf24AmpPowerNames[] = {"MIN", "LOW", "HIGH", "MAX"};
|
||||||
|
|
||||||
#define BIT_CNT(x) ((x)<<3)
|
#define BIT_CNT(x) ((x)<<3)
|
||||||
|
|
||||||
|
static volatile bool mIrqRcvd;
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// HM Radio class
|
// HM Radio class
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
template <class BUFFER, uint8_t IRQ_PIN = DEF_IRQ_PIN, uint8_t CE_PIN = DEF_CE_PIN, uint8_t CS_PIN = DEF_CS_PIN, uint8_t AMP_PWR = RF24_PA_LOW>
|
template <uint8_t IRQ_PIN = DEF_IRQ_PIN, uint8_t CE_PIN = DEF_CE_PIN, uint8_t CS_PIN = DEF_CS_PIN, uint8_t AMP_PWR = RF24_PA_LOW>
|
||||||
class HmRadio {
|
class HmRadio {
|
||||||
public:
|
public:
|
||||||
HmRadio() : mNrf24(CE_PIN, CS_PIN, SPI_SPEED) {
|
HmRadio() : mNrf24(CE_PIN, CS_PIN, SPI_SPEED) {
|
||||||
|
@ -84,23 +67,22 @@ class HmRadio {
|
||||||
mRfChLst[3] = 61;
|
mRfChLst[3] = 61;
|
||||||
mRfChLst[4] = 75;
|
mRfChLst[4] = 75;
|
||||||
|
|
||||||
|
// default channels
|
||||||
mTxChIdx = 2; // Start TX with 40
|
mTxChIdx = 2; // Start TX with 40
|
||||||
mRxChIdx = 0; // Start RX with 03
|
mRxChIdx = 0; // Start RX with 03
|
||||||
mRxLoopCnt = RF_LOOP_CNT;
|
|
||||||
|
|
||||||
mSendCnt = 0;
|
mSendCnt = 0;
|
||||||
mRetransmits = 0;
|
mRetransmits = 0;
|
||||||
|
|
||||||
mSerialDebug = false;
|
mSerialDebug = false;
|
||||||
mIrqRcvd = false;
|
mIrqRcvd = false;
|
||||||
}
|
}
|
||||||
~HmRadio() {}
|
~HmRadio() {}
|
||||||
|
|
||||||
void setup(BUFFER *ctrl, uint8_t ampPwr = RF24_PA_LOW, uint8_t irq = IRQ_PIN, uint8_t ce = CE_PIN, uint8_t cs = CS_PIN) {
|
void setup(uint8_t ampPwr = RF24_PA_LOW, uint8_t irq = IRQ_PIN, uint8_t ce = CE_PIN, uint8_t cs = CS_PIN) {
|
||||||
DPRINTLN(DBG_VERBOSE, F("hmRadio.h:setup"));
|
DPRINTLN(DBG_VERBOSE, F("hmRadio.h:setup"));
|
||||||
pinMode(irq, INPUT_PULLUP);
|
pinMode(irq, INPUT_PULLUP);
|
||||||
mBufCtrl = ctrl;
|
attachInterrupt(digitalPinToInterrupt(irq), []()IRAM_ATTR{ mIrqRcvd = true; }, FALLING);
|
||||||
|
|
||||||
|
|
||||||
uint32_t dtuSn = 0x87654321;
|
uint32_t dtuSn = 0x87654321;
|
||||||
uint32_t chipID = 0; // will be filled with last 3 bytes of MAC
|
uint32_t chipID = 0; // will be filled with last 3 bytes of MAC
|
||||||
|
@ -121,27 +103,22 @@ class HmRadio {
|
||||||
DTU_RADIO_ID = ((uint64_t)(((dtuSn >> 24) & 0xFF) | ((dtuSn >> 8) & 0xFF00) | ((dtuSn << 8) & 0xFF0000) | ((dtuSn << 24) & 0xFF000000)) << 8) | 0x01;
|
DTU_RADIO_ID = ((uint64_t)(((dtuSn >> 24) & 0xFF) | ((dtuSn >> 8) & 0xFF00) | ((dtuSn << 8) & 0xFF0000) | ((dtuSn << 24) & 0xFF000000)) << 8) | 0x01;
|
||||||
|
|
||||||
mNrf24.begin(ce, cs);
|
mNrf24.begin(ce, cs);
|
||||||
mNrf24.setRetries(0, 0);
|
mNrf24.setRetries(3, 15); // 3*250us + 250us and 15 loops -> 15ms
|
||||||
|
|
||||||
mNrf24.setChannel(DEFAULT_RECV_CHANNEL);
|
mNrf24.setChannel(mRfChLst[mRxChIdx]);
|
||||||
mNrf24.setDataRate(RF24_250KBPS);
|
mNrf24.setDataRate(RF24_250KBPS);
|
||||||
|
mNrf24.setAutoAck(true);
|
||||||
|
mNrf24.enableDynamicPayloads();
|
||||||
mNrf24.setCRCLength(RF24_CRC_16);
|
mNrf24.setCRCLength(RF24_CRC_16);
|
||||||
mNrf24.setAutoAck(false);
|
|
||||||
mNrf24.setPayloadSize(MAX_RF_PAYLOAD_SIZE);
|
|
||||||
mNrf24.setAddressWidth(5);
|
mNrf24.setAddressWidth(5);
|
||||||
mNrf24.openReadingPipe(1, DTU_RADIO_ID);
|
mNrf24.openReadingPipe(1, DTU_RADIO_ID);
|
||||||
mNrf24.enableDynamicPayloads();
|
|
||||||
|
|
||||||
// enable only receiving interrupts
|
// enable all receiving interrupts
|
||||||
mNrf24.maskIRQ(true, true, false);
|
mNrf24.maskIRQ(false, false, false);
|
||||||
|
|
||||||
DPRINT(DBG_INFO, F("RF24 Amp Pwr: RF24_PA_"));
|
DPRINT(DBG_INFO, F("RF24 Amp Pwr: RF24_PA_"));
|
||||||
DPRINTLN(DBG_INFO, String(rf24AmpPowerNames[ampPwr]));
|
DPRINTLN(DBG_INFO, String(rf24AmpPowerNames[ampPwr]));
|
||||||
mNrf24.setPALevel(ampPwr & 0x03);
|
mNrf24.setPALevel(ampPwr & 0x03);
|
||||||
mNrf24.startListening();
|
|
||||||
|
|
||||||
|
|
||||||
mTxCh = setDefaultChannels();
|
|
||||||
|
|
||||||
if(mNrf24.isChipConnected()) {
|
if(mNrf24.isChipConnected()) {
|
||||||
DPRINTLN(DBG_INFO, F("Radio Config:"));
|
DPRINTLN(DBG_INFO, F("Radio Config:"));
|
||||||
|
@ -151,77 +128,70 @@ class HmRadio {
|
||||||
DPRINTLN(DBG_WARN, F("WARNING! your NRF24 module can't be reached, check the wiring"));
|
DPRINTLN(DBG_WARN, F("WARNING! your NRF24 module can't be reached, check the wiring"));
|
||||||
}
|
}
|
||||||
|
|
||||||
void loop(void) {
|
bool loop(void) {
|
||||||
if(mIrqRcvd) {
|
if (!mIrqRcvd)
|
||||||
DISABLE_IRQ;
|
return false; // nothing to do
|
||||||
mIrqRcvd = false;
|
mIrqRcvd = false;
|
||||||
bool tx_ok, tx_fail, rx_ready;
|
bool tx_ok, tx_fail, rx_ready;
|
||||||
mNrf24.whatHappened(tx_ok, tx_fail, rx_ready); // resets the IRQ pin to HIGH
|
mNrf24.whatHappened(tx_ok, tx_fail, rx_ready); // resets the IRQ pin to HIGH
|
||||||
RESTORE_IRQ;
|
mNrf24.flush_tx(); // empty TX FIFO
|
||||||
uint8_t pipe, len;
|
//DBGPRINTLN("TX whatHappened Ch" + String(mRfChLst[mTxChIdx]) + " " + String(tx_ok) + String(tx_fail) + String(rx_ready));
|
||||||
packet_t *p;
|
|
||||||
while(mNrf24.available(&pipe)) {
|
|
||||||
if(!mBufCtrl->full()) {
|
|
||||||
p = mBufCtrl->getFront();
|
|
||||||
p->rxCh = mRfChLst[mRxChIdx];
|
|
||||||
len = mNrf24.getPayloadSize();
|
|
||||||
if(len > MAX_RF_PAYLOAD_SIZE)
|
|
||||||
len = MAX_RF_PAYLOAD_SIZE;
|
|
||||||
|
|
||||||
mNrf24.read(p->packet, len);
|
// start listening on the default RX channel
|
||||||
mBufCtrl->pushFront(p);
|
mRxChIdx = 0;
|
||||||
yield();
|
mNrf24.setChannel(mRfChLst[mRxChIdx]);
|
||||||
|
mNrf24.startListening();
|
||||||
|
|
||||||
|
//uint32_t debug_ms = millis();
|
||||||
|
uint16_t cnt = 300; // that is 60 times 5 channels
|
||||||
|
while (0 < cnt--) {
|
||||||
|
uint32_t startMillis = millis();
|
||||||
|
while (millis()-startMillis < 4) { // listen 4ms to each channel
|
||||||
|
if (mIrqRcvd) {
|
||||||
|
mIrqRcvd = false;
|
||||||
|
if (getReceived()) { // everything received
|
||||||
|
//DBGPRINTLN("RX finished Cnt: " + String(300-cnt) + " time used: " + String(millis()-debug_ms)+ " ms");
|
||||||
|
mNrf24.stopListening();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
yield();
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
mNrf24.flush_rx(); // drop the packet
|
switchRxCh(); // switch to next RX channel
|
||||||
RESTORE_IRQ;
|
yield();
|
||||||
}
|
}
|
||||||
|
// not finished but time is over
|
||||||
|
//DBGPRINTLN("RX not finished: 300 time used: " + String(millis()-debug_ms)+ " ms");
|
||||||
|
mNrf24.stopListening();
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool isChipConnected(void) {
|
||||||
|
//DPRINTLN(DBG_VERBOSE, F("hmRadio.h:isChipConnected"));
|
||||||
|
return mNrf24.isChipConnected();
|
||||||
|
}
|
||||||
void enableDebug() {
|
void enableDebug() {
|
||||||
mSerialDebug = true;
|
mSerialDebug = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void handleIntr(void) {
|
|
||||||
mIrqRcvd = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t setDefaultChannels(void) {
|
|
||||||
//DPRINTLN(DBG_VERBOSE, F("hmRadio.h:setDefaultChannels"));
|
|
||||||
mTxChIdx = 2; // Start TX with 40
|
|
||||||
mRxChIdx = 0; // Start RX with 03
|
|
||||||
return mRfChLst[mTxChIdx];
|
|
||||||
}
|
|
||||||
|
|
||||||
void sendControlPacket(uint64_t invId, uint8_t cmd, uint16_t *data, bool isRetransmit) {
|
void sendControlPacket(uint64_t invId, uint8_t cmd, uint16_t *data, bool isRetransmit) {
|
||||||
DPRINTLN(DBG_INFO, F("sendControlPacket cmd: 0x") + String(cmd, HEX));
|
DPRINTLN(DBG_INFO, F("sendControlPacket cmd: 0x") + String(cmd, HEX));
|
||||||
sendCmdPacket(invId, TX_REQ_DEVCONTROL, SINGLE_FRAME, false);
|
initPacket(invId, TX_REQ_DEVCONTROL, SINGLE_FRAME);
|
||||||
uint8_t cnt = 0;
|
uint8_t cnt = 10;
|
||||||
mTxBuf[10 + cnt++] = cmd; // cmd -> 0 on, 1 off, 2 restart, 11 active power, 12 reactive power, 13 power factor
|
mTxBuf[cnt++] = cmd; // cmd -> 0 on, 1 off, 2 restart, 11 active power, 12 reactive power, 13 power factor
|
||||||
mTxBuf[10 + cnt++] = 0x00;
|
mTxBuf[cnt++] = 0x00;
|
||||||
if(cmd >= ActivePowerContr && cmd <= PFSet) { // ActivePowerContr, ReactivePowerContr, PFSet
|
if(cmd >= ActivePowerContr && cmd <= PFSet) { // ActivePowerContr, ReactivePowerContr, PFSet
|
||||||
mTxBuf[10 + cnt++] = ((data[0] * 10) >> 8) & 0xff; // power limit
|
mTxBuf[cnt++] = ((data[0] * 10) >> 8) & 0xff; // power limit
|
||||||
mTxBuf[10 + cnt++] = ((data[0] * 10) ) & 0xff; // power limit
|
mTxBuf[cnt++] = ((data[0] * 10) ) & 0xff; // power limit
|
||||||
mTxBuf[10 + cnt++] = ((data[1] ) >> 8) & 0xff; // setting for persistens handlings
|
mTxBuf[cnt++] = ((data[1] ) >> 8) & 0xff; // setting for persistens handlings
|
||||||
mTxBuf[10 + cnt++] = ((data[1] ) ) & 0xff; // setting for persistens handling
|
mTxBuf[cnt++] = ((data[1] ) ) & 0xff; // setting for persistens handling
|
||||||
}
|
}
|
||||||
|
sendPacket(invId, cnt, isRetransmit, true);
|
||||||
// crc control data
|
|
||||||
uint16_t crc = ah::crc16(&mTxBuf[10], cnt);
|
|
||||||
mTxBuf[10 + cnt++] = (crc >> 8) & 0xff;
|
|
||||||
mTxBuf[10 + cnt++] = (crc ) & 0xff;
|
|
||||||
|
|
||||||
// crc over all
|
|
||||||
mTxBuf[10 + cnt] = ah::crc8(mTxBuf, 10 + cnt);
|
|
||||||
|
|
||||||
sendPacket(invId, mTxBuf, 10 + cnt + 1, isRetransmit, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void sendTimePacket(uint64_t invId, uint8_t cmd, uint32_t ts, uint16_t alarmMesId, bool isRetransmit) {
|
void sendTimePacket(uint64_t invId, uint8_t cmd, uint32_t ts, uint16_t alarmMesId, bool isRetransmit) {
|
||||||
DPRINTLN(DBG_DEBUG, F("sendTimePacket 0x") + String(cmd, HEX));
|
DPRINTLN(DBG_DEBUG, F("sendTimePacket 0x") + String(cmd, HEX));
|
||||||
sendCmdPacket(invId, TX_REQ_INFO, ALL_FRAMES, isRetransmit, false);
|
initPacket(invId, TX_REQ_INFO, ALL_FRAMES);
|
||||||
mTxBuf[10] = cmd; // cid
|
mTxBuf[10] = cmd; // cid
|
||||||
mTxBuf[11] = 0x00;
|
mTxBuf[11] = 0x00;
|
||||||
CP_U32_LittleEndian(&mTxBuf[12], ts);
|
CP_U32_LittleEndian(&mTxBuf[12], ts);
|
||||||
|
@ -229,61 +199,16 @@ class HmRadio {
|
||||||
mTxBuf[18] = (alarmMesId >> 8) & 0xff;
|
mTxBuf[18] = (alarmMesId >> 8) & 0xff;
|
||||||
mTxBuf[19] = (alarmMesId ) & 0xff;
|
mTxBuf[19] = (alarmMesId ) & 0xff;
|
||||||
}
|
}
|
||||||
uint16_t crc = ah::crc16(&mTxBuf[10], 14);
|
sendPacket(invId, 24, isRetransmit, true);
|
||||||
mTxBuf[24] = (crc >> 8) & 0xff;
|
|
||||||
mTxBuf[25] = (crc ) & 0xff;
|
|
||||||
mTxBuf[26] = ah::crc8(mTxBuf, 26);
|
|
||||||
|
|
||||||
sendPacket(invId, mTxBuf, 27, isRetransmit, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void sendCmdPacket(uint64_t invId, uint8_t mid, uint8_t pid, bool isRetransmit, bool calcCrc = true) {
|
void sendCmdPacket(uint64_t invId, uint8_t mid, uint8_t pid, bool isRetransmit) {
|
||||||
DPRINTLN(DBG_VERBOSE, F("sendCmdPacket, mid: ") + String(mid, HEX) + F(" pid: ") + String(pid, HEX));
|
initPacket(invId, mid, pid);
|
||||||
memset(mTxBuf, 0, MAX_RF_PAYLOAD_SIZE);
|
sendPacket(invId, 10, isRetransmit, false);
|
||||||
mTxBuf[0] = mid; // message id
|
|
||||||
CP_U32_BigEndian(&mTxBuf[1], (invId >> 8));
|
|
||||||
CP_U32_BigEndian(&mTxBuf[5], (DTU_RADIO_ID >> 8));
|
|
||||||
mTxBuf[9] = pid;
|
|
||||||
if(calcCrc) {
|
|
||||||
mTxBuf[10] = ah::crc8(mTxBuf, 10);
|
|
||||||
sendPacket(invId, mTxBuf, 11, isRetransmit, false);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool checkPaketCrc(uint8_t buf[], uint8_t *len, uint8_t rxCh) {
|
void dumpBuf(uint8_t buf[], uint8_t len) {
|
||||||
//DPRINTLN(DBG_INFO, F("hmRadio.h:checkPaketCrc"));
|
|
||||||
*len = (buf[0] >> 2);
|
|
||||||
if(*len > (MAX_RF_PAYLOAD_SIZE - 2))
|
|
||||||
*len = MAX_RF_PAYLOAD_SIZE - 2;
|
|
||||||
for(uint8_t i = 1; i < (*len + 1); i++) {
|
|
||||||
buf[i-1] = (buf[i] << 1) | (buf[i+1] >> 7);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t crc = ah::crc8(buf, *len-1);
|
|
||||||
bool valid = (crc == buf[*len-1]);
|
|
||||||
|
|
||||||
return valid;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool switchRxCh(uint16_t addLoop = 0) {
|
|
||||||
if(!mNrf24.isChipConnected())
|
|
||||||
return true;
|
|
||||||
mRxLoopCnt += addLoop;
|
|
||||||
if(mRxLoopCnt != 0) {
|
|
||||||
mRxLoopCnt--;
|
|
||||||
DISABLE_IRQ;
|
|
||||||
mNrf24.stopListening();
|
|
||||||
mNrf24.setChannel(getRxNxtChannel());
|
|
||||||
mNrf24.startListening();
|
|
||||||
RESTORE_IRQ;
|
|
||||||
}
|
|
||||||
return (0 == mRxLoopCnt); // receive finished
|
|
||||||
}
|
|
||||||
|
|
||||||
void dumpBuf(const char *info, uint8_t buf[], uint8_t len) {
|
|
||||||
//DPRINTLN(DBG_VERBOSE, F("hmRadio.h:dumpBuf"));
|
//DPRINTLN(DBG_VERBOSE, F("hmRadio.h:dumpBuf"));
|
||||||
if(NULL != info)
|
|
||||||
DBGPRINT(String(info));
|
|
||||||
for(uint8_t i = 0; i < len; i++) {
|
for(uint8_t i = 0; i < len; i++) {
|
||||||
DHEX(buf[i]);
|
DHEX(buf[i]);
|
||||||
DBGPRINT(" ");
|
DBGPRINT(" ");
|
||||||
|
@ -291,11 +216,6 @@ class HmRadio {
|
||||||
DBGPRINTLN("");
|
DBGPRINTLN("");
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isChipConnected(void) {
|
|
||||||
//DPRINTLN(DBG_VERBOSE, F("hmRadio.h:isChipConnected"));
|
|
||||||
return mNrf24.isChipConnected();
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t getDataRate(void) {
|
uint8_t getDataRate(void) {
|
||||||
if(!mNrf24.isChipConnected())
|
if(!mNrf24.isChipConnected())
|
||||||
return 3; // unkown
|
return 3; // unkown
|
||||||
|
@ -306,7 +226,7 @@ class HmRadio {
|
||||||
return mNrf24.isPVariant();
|
return mNrf24.isPVariant();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::queue<packet_t> mBufCtrl;
|
||||||
|
|
||||||
uint32_t mSendCnt;
|
uint32_t mSendCnt;
|
||||||
uint32_t mRetransmits;
|
uint32_t mRetransmits;
|
||||||
|
@ -314,78 +234,91 @@ class HmRadio {
|
||||||
bool mSerialDebug;
|
bool mSerialDebug;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void sendPacket(uint64_t invId, uint8_t buf[], uint8_t len, bool isRetransmit, bool clear=false) {
|
bool getReceived(void) {
|
||||||
|
bool tx_ok, tx_fail, rx_ready;
|
||||||
|
mNrf24.whatHappened(tx_ok, tx_fail, rx_ready); // resets the IRQ pin to HIGH
|
||||||
|
DBGPRINTLN("RX whatHappened Ch" + String(mRfChLst[mRxChIdx]) + " " + String(tx_ok) + String(tx_fail) + String(rx_ready));
|
||||||
|
|
||||||
|
bool isLastPackage = false;
|
||||||
|
while(mNrf24.available()) {
|
||||||
|
uint8_t len;
|
||||||
|
len = mNrf24.getDynamicPayloadSize(); // if payload size > 32, corrupt payload has been flushed
|
||||||
|
if (len > 0) {
|
||||||
|
packet_t p;
|
||||||
|
p.ch = mRfChLst[mRxChIdx];
|
||||||
|
p.len = len;
|
||||||
|
mNrf24.read(p.packet, len);
|
||||||
|
mBufCtrl.push(p);
|
||||||
|
if (p.packet[0] == (TX_REQ_INFO + ALL_FRAMES)) // response from get information command
|
||||||
|
isLastPackage = (p.packet[9] > 0x81); // > 0x81 indicates last packet received
|
||||||
|
else if (p.packet[0] != 0x00) // ignore fragment number zero
|
||||||
|
isLastPackage = true; // response from dev control command
|
||||||
|
yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return isLastPackage;
|
||||||
|
}
|
||||||
|
|
||||||
|
void switchRxCh() {
|
||||||
|
mNrf24.stopListening();
|
||||||
|
// get next channel index
|
||||||
|
if(++mRxChIdx >= RF_CHANNELS)
|
||||||
|
mRxChIdx = 0;
|
||||||
|
mNrf24.setChannel(mRfChLst[mRxChIdx]);
|
||||||
|
mNrf24.startListening();
|
||||||
|
}
|
||||||
|
|
||||||
|
void initPacket(uint64_t invId, uint8_t mid, uint8_t pid) {
|
||||||
|
DPRINTLN(DBG_VERBOSE, F("initPacket, mid: ") + String(mid, HEX) + F(" pid: ") + String(pid, HEX));
|
||||||
|
memset(mTxBuf, 0, MAX_RF_PAYLOAD_SIZE);
|
||||||
|
mTxBuf[0] = mid; // message id
|
||||||
|
CP_U32_BigEndian(&mTxBuf[1], (invId >> 8));
|
||||||
|
CP_U32_BigEndian(&mTxBuf[5], (DTU_RADIO_ID >> 8));
|
||||||
|
mTxBuf[9] = pid;
|
||||||
|
}
|
||||||
|
|
||||||
|
void sendPacket(uint64_t invId, uint8_t len, bool isRetransmit, bool clear=false) {
|
||||||
//DPRINTLN(DBG_VERBOSE, F("hmRadio.h:sendPacket"));
|
//DPRINTLN(DBG_VERBOSE, F("hmRadio.h:sendPacket"));
|
||||||
//DPRINTLN(DBG_VERBOSE, "sent packet: #" + String(mSendCnt));
|
//DPRINTLN(DBG_VERBOSE, "sent packet: #" + String(mSendCnt));
|
||||||
//dumpBuf("SEN ", buf, len);
|
|
||||||
|
// append crc's
|
||||||
|
if (len > 10) {
|
||||||
|
// crc control data
|
||||||
|
uint16_t crc = ah::crc16(&mTxBuf[10], len - 10);
|
||||||
|
mTxBuf[len++] = (crc >> 8) & 0xff;
|
||||||
|
mTxBuf[len++] = (crc ) & 0xff;
|
||||||
|
}
|
||||||
|
// crc over all
|
||||||
|
mTxBuf[len++] = ah::crc8(mTxBuf, len);
|
||||||
|
|
||||||
if(mSerialDebug) {
|
if(mSerialDebug) {
|
||||||
DPRINT(DBG_INFO, "TX " + String(len) + "B Ch" + String(mRfChLst[mTxChIdx]) + " | ");
|
DPRINT(DBG_INFO, "TX " + String(len) + "B Ch" + String(mRfChLst[mTxChIdx]) + " | ");
|
||||||
dumpBuf(NULL, buf, len);
|
dumpBuf(mTxBuf, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
DISABLE_IRQ;
|
|
||||||
mNrf24.stopListening();
|
|
||||||
|
|
||||||
if(clear)
|
|
||||||
mRxLoopCnt = RF_LOOP_CNT;
|
|
||||||
|
|
||||||
mNrf24.setChannel(mRfChLst[mTxChIdx]);
|
mNrf24.setChannel(mRfChLst[mTxChIdx]);
|
||||||
mTxCh = getTxNxtChannel(); // switch channel for next packet
|
mNrf24.openWritingPipe(reinterpret_cast<uint8_t*>(&invId));
|
||||||
mNrf24.openWritingPipe(invId); // TODO: deprecated
|
mNrf24.startWrite(mTxBuf, len, false); // false = request ACK response
|
||||||
mNrf24.setCRCLength(RF24_CRC_16);
|
|
||||||
mNrf24.enableDynamicPayloads();
|
|
||||||
mNrf24.setAutoAck(true);
|
|
||||||
mNrf24.setRetries(3, 15); // 3*250us and 15 loops -> 11.25ms
|
|
||||||
mNrf24.write(buf, len);
|
|
||||||
|
|
||||||
// Try to avoid zero payload acks (has no effect)
|
// switch TX channel for next packet
|
||||||
mNrf24.openWritingPipe(DUMMY_RADIO_ID); // TODO: why dummy radio id?, deprecated
|
if(++mTxChIdx >= RF_CHANNELS)
|
||||||
mRxChIdx = 0;
|
mTxChIdx = 0;
|
||||||
mNrf24.setChannel(mRfChLst[mRxChIdx]);
|
|
||||||
mNrf24.setAutoAck(false);
|
|
||||||
mNrf24.setRetries(0, 0);
|
|
||||||
mNrf24.disableDynamicPayloads();
|
|
||||||
mNrf24.setCRCLength(RF24_CRC_DISABLED);
|
|
||||||
mNrf24.startListening();
|
|
||||||
|
|
||||||
RESTORE_IRQ;
|
|
||||||
if(isRetransmit)
|
if(isRetransmit)
|
||||||
mRetransmits++;
|
mRetransmits++;
|
||||||
else
|
else
|
||||||
mSendCnt++;
|
mSendCnt++;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t getTxNxtChannel(void) {
|
|
||||||
|
|
||||||
if(++mTxChIdx >= RF_CHANNELS)
|
|
||||||
mTxChIdx = 0;
|
|
||||||
return mRfChLst[mTxChIdx];
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t getRxNxtChannel(void) {
|
|
||||||
|
|
||||||
if(++mRxChIdx >= RF_CHANNELS)
|
|
||||||
mRxChIdx = 0;
|
|
||||||
return mRfChLst[mRxChIdx];
|
|
||||||
}
|
|
||||||
|
|
||||||
uint64_t DTU_RADIO_ID;
|
uint64_t DTU_RADIO_ID;
|
||||||
|
|
||||||
uint8_t mTxCh;
|
|
||||||
uint8_t mTxChIdx;
|
|
||||||
|
|
||||||
uint8_t mRfChLst[RF_CHANNELS];
|
uint8_t mRfChLst[RF_CHANNELS];
|
||||||
|
|
||||||
|
uint8_t mTxChIdx;
|
||||||
uint8_t mRxChIdx;
|
uint8_t mRxChIdx;
|
||||||
uint16_t mRxLoopCnt;
|
|
||||||
|
|
||||||
RF24 mNrf24;
|
RF24 mNrf24;
|
||||||
BUFFER *mBufCtrl;
|
|
||||||
uint8_t mTxBuf[MAX_RF_PAYLOAD_SIZE];
|
uint8_t mTxBuf[MAX_RF_PAYLOAD_SIZE];
|
||||||
|
|
||||||
DevControlCmdType DevControlCmd;
|
|
||||||
|
|
||||||
volatile bool mIrqRcvd;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /*__RADIO_H__*/
|
#endif /*__RADIO_H__*/
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue