Merge pull request #100 from geeksville/removeradiohead

Removeradiohead
This commit is contained in:
Kevin Hester 2020-04-18 18:42:25 -07:00 committed by GitHub
commit 88c576798b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 66 additions and 38 deletions

View File

@ -336,9 +336,9 @@ void loop()
{ {
uint32_t msecstosleep = 1000 * 30; // How long can we sleep before we again need to service the main loop? uint32_t msecstosleep = 1000 * 30; // How long can we sleep before we again need to service the main loop?
powerFSM.run_machine();
gps.loop(); gps.loop();
router.loop(); router.loop();
powerFSM.run_machine();
service.loop(); service.loop();
ledPeriodic.loop(); ledPeriodic.loop();

View File

@ -20,7 +20,7 @@ bool CustomRF95::canSleep()
bool res = (_mode == RHModeInitialising || _mode == RHModeIdle || _mode == RHModeRx) && !isRx && txQueue.isEmpty(); bool res = (_mode == RHModeInitialising || _mode == RHModeIdle || _mode == RHModeRx) && !isRx && txQueue.isEmpty();
if (!res) // only print debug messages if we are vetoing sleep if (!res) // only print debug messages if we are vetoing sleep
DEBUG_MSG("canSleep, mode=%d, isRx=%d, txEmpty=%d, txGood=%d\n", _mode, isRx, txQueue.isEmpty(), _txGood); DEBUG_MSG("radio wait to sleep, mode=%d, isRx=%d, txEmpty=%d, txGood=%d\n", _mode, isRx, txQueue.isEmpty(), _txGood);
return res; return res;
} }
@ -78,13 +78,12 @@ void CustomRF95::handleInterrupt()
{ {
RH_RF95::handleInterrupt(); RH_RF95::handleInterrupt();
BaseType_t higherPriWoken = false;
if (_mode == RHModeIdle) // We are now done sending or receiving if (_mode == RHModeIdle) // We are now done sending or receiving
{ {
if (sendingPacket) // Were we sending? if (sendingPacket) // Were we sending?
{ {
// We are done sending that packet, release it // We are done sending that packet, release it
packetPool.releaseFromISR(sendingPacket, &higherPriWoken); packetPool.release(sendingPacket);
sendingPacket = NULL; sendingPacket = NULL;
// DEBUG_MSG("Done with send\n"); // DEBUG_MSG("Done with send\n");
} }
@ -123,43 +122,35 @@ void CustomRF95::handleInterrupt()
} }
if (!pb_decode_from_bytes(payload, payloadLen, SubPacket_fields, p)) { if (!pb_decode_from_bytes(payload, payloadLen, SubPacket_fields, p)) {
packetPool.releaseFromISR(mp, &higherPriWoken); packetPool.release(mp);
} else { } else {
// parsing was successful, queue for our recipient // parsing was successful, queue for our recipient
mp->has_payload = true; mp->has_payload = true;
deliverToReceiverISR(mp, &higherPriWoken); deliverToReceiver(mp);
} }
clearRxBuf(); // This message accepted and cleared clearRxBuf(); // This message accepted and cleared
} }
higherPriWoken |= handleIdleISR(); handleIdleISR();
} }
// If we call this _IT WILL NOT RETURN_
if (higherPriWoken)
portYIELD_FROM_ISR();
} }
/** The ISR doesn't have any good work to do, give a new assignment. /** The ISR doesn't have any good work to do, give a new assignment.
* *
* Return true if a higher pri task has woken * Return true if a higher pri task has woken
*/ */
bool CustomRF95::handleIdleISR() void CustomRF95::handleIdleISR()
{ {
BaseType_t higherPriWoken = false;
// First send any outgoing packets we have ready // First send any outgoing packets we have ready
MeshPacket *txp = txQueue.dequeuePtrFromISR(0); MeshPacket *txp = txQueue.dequeuePtr(0);
if (txp) if (txp)
startSend(txp); startSend(txp);
else { else {
// Nothing to send, let's switch back to receive mode // Nothing to send, let's switch back to receive mode
setModeRx(); setModeRx();
} }
return higherPriWoken;
} }
/// This routine might be called either from user space or ISR /// This routine might be called either from user space or ISR
@ -197,6 +188,8 @@ void CustomRF95::startSend(MeshPacket *txp)
void CustomRF95::loop() void CustomRF95::loop()
{ {
RH_RF95::loop();
// It should never take us more than 30 secs to send a packet, if it does, we have a bug, FIXME, move most of this // It should never take us more than 30 secs to send a packet, if it does, we have a bug, FIXME, move most of this
// into CustomRF95 // into CustomRF95
uint32_t now = millis(); uint32_t now = millis();

View File

@ -52,5 +52,5 @@ class CustomRF95 : public RH_RF95, public RadioInterface
void startSend(MeshPacket *txp); void startSend(MeshPacket *txp);
/// Return true if a higher pri task has woken /// Return true if a higher pri task has woken
bool handleIdleISR(); void handleIdleISR();
}; };

View File

@ -34,16 +34,12 @@ bool RH_RF95::init()
if (!RHSPIDriver::init()) if (!RHSPIDriver::init())
return false; return false;
// Determine the interrupt number that corresponds to the interruptPin
int interruptNumber = digitalPinToInterrupt(_interruptPin);
if (interruptNumber == NOT_AN_INTERRUPT)
return false;
#ifdef RH_ATTACHINTERRUPT_TAKES_PIN_NUMBER #ifdef RH_ATTACHINTERRUPT_TAKES_PIN_NUMBER
interruptNumber = _interruptPin; interruptNumber = _interruptPin;
#endif #endif
// Tell the low level SPI interface we will use SPI within this interrupt // Tell the low level SPI interface we will use SPI within this interrupt
spiUsingInterrupt(interruptNumber); // spiUsingInterrupt(interruptNumber);
// No way to check the device type :-( // No way to check the device type :-(
@ -114,6 +110,17 @@ bool RH_RF95::init()
return false; // Too many devices, not enough interrupt vectors return false; // Too many devices, not enough interrupt vectors
} }
_deviceForInterrupt[_myInterruptIndex] = this; _deviceForInterrupt[_myInterruptIndex] = this;
return enableInterrupt();
}
bool RH_RF95::enableInterrupt()
{
// Determine the interrupt number that corresponds to the interruptPin
int interruptNumber = digitalPinToInterrupt(_interruptPin);
if (interruptNumber == NOT_AN_INTERRUPT)
return false;
if (_myInterruptIndex == 0) if (_myInterruptIndex == 0)
attachInterrupt(interruptNumber, isr0, ONHIGH); attachInterrupt(interruptNumber, isr0, ONHIGH);
else if (_myInterruptIndex == 1) else if (_myInterruptIndex == 1)
@ -126,6 +133,12 @@ bool RH_RF95::init()
return true; return true;
} }
void RH_INTERRUPT_ATTR RH_RF95::disableInterrupt()
{
int interruptNumber = digitalPinToInterrupt(_interruptPin);
detachInterrupt(interruptNumber);
}
void RH_RF95::prepareDeepSleep() void RH_RF95::prepareDeepSleep()
{ {
// Determine the interrupt number that corresponds to the interruptPin // Determine the interrupt number that corresponds to the interruptPin
@ -143,6 +156,13 @@ bool RH_RF95::isReceiving()
RH_RF95_MODEM_STATUS_HEADER_INFO_VALID)) != 0; RH_RF95_MODEM_STATUS_HEADER_INFO_VALID)) != 0;
} }
void RH_INTERRUPT_ATTR RH_RF95::handleInterruptLevel0()
{
disableInterrupt(); // Disable our interrupt until our helper thread can run (because the IRQ will remain asserted until we
// talk to it via SPI)
pendingInterrupt = true;
}
// C++ level interrupt handler for this instance // C++ level interrupt handler for this instance
// LORA is unusual in that it has several interrupt lines, and not a single, combined one. // LORA is unusual in that it has several interrupt lines, and not a single, combined one.
// On MiniWirelessLoRa, only one of the several interrupt lines (DI0) from the RFM95 is usefuly // On MiniWirelessLoRa, only one of the several interrupt lines (DI0) from the RFM95 is usefuly
@ -153,15 +173,14 @@ void RH_RF95::handleInterrupt()
// Read the interrupt register // Read the interrupt register
uint8_t irq_flags = spiRead(RH_RF95_REG_12_IRQ_FLAGS); uint8_t irq_flags = spiRead(RH_RF95_REG_12_IRQ_FLAGS);
// ack all interrupts, note - we did this already in the RX_DONE case above, and we don't want to do it twice // ack all interrupts
// note from radiohead author wrt old code (with IMO wrong fix)
// Sigh: on some processors, for some unknown reason, doing this only once does not actually // Sigh: on some processors, for some unknown reason, doing this only once does not actually
// clear the radio's interrupt flag. So we do it twice. Why? (kevinh - I think the root cause we want level // clear the radio's interrupt flag. So we do it twice. Why? (kevinh - I think the root cause we want level
// triggered interrupts here - not edge. Because edge allows us to miss handling secondard interrupts that occurred // triggered interrupts here - not edge. Because edge allows us to miss handling secondard interrupts that occurred
// while this ISR was running. Better to instead, configure the interrupts as level triggered and clear pending // while this ISR was running. Better to instead, configure the interrupts as level triggered and clear pending
// at the _beginning_ of the ISR. If any interrupts occur while handling the ISR, the signal will remain asserted and // at the _beginning_ of the ISR. If any interrupts occur while handling the ISR, the signal will remain asserted and
// our ISR will be reinvoked to handle that case) // our ISR will be reinvoked to handle that case)
// kevinh: turn this off until root cause is known, because it can cause missed interrupts!
// spiWrite(RH_RF95_REG_12_IRQ_FLAGS, 0xff); // Clear all IRQ flags
spiWrite(RH_RF95_REG_12_IRQ_FLAGS, 0xff); // Clear all IRQ flags spiWrite(RH_RF95_REG_12_IRQ_FLAGS, 0xff); // Clear all IRQ flags
// Note: there can be substantial latency between ISR assertion and this function being run, therefore // Note: there can be substantial latency between ISR assertion and this function being run, therefore
@ -169,14 +188,10 @@ void RH_RF95::handleInterrupt()
// Note: we are running the chip in continuous receive mode (currently, so RX_TIMEOUT shouldn't ever occur) // Note: we are running the chip in continuous receive mode (currently, so RX_TIMEOUT shouldn't ever occur)
bool haveRxError = irq_flags & (RH_RF95_RX_TIMEOUT | RH_RF95_PAYLOAD_CRC_ERROR); bool haveRxError = irq_flags & (RH_RF95_RX_TIMEOUT | RH_RF95_PAYLOAD_CRC_ERROR);
if (haveRxError) if (haveRxError) {
// if (_mode == RHModeRx && irq_flags & (RH_RF95_RX_TIMEOUT | RH_RF95_PAYLOAD_CRC_ERROR))
{
_rxBad++; _rxBad++;
clearRxBuf(); clearRxBuf();
} } else if (irq_flags & RH_RF95_RX_DONE) {
if ((irq_flags & RH_RF95_RX_DONE) && !haveRxError) {
// Read the RegHopChannel register to check if CRC presence is signalled // Read the RegHopChannel register to check if CRC presence is signalled
// in the header. If not it might be a stray (noise) packet.* // in the header. If not it might be a stray (noise) packet.*
uint8_t crc_present = spiRead(RH_RF95_REG_1C_HOP_CHANNEL) & RH_RF95_RX_PAYLOAD_CRC_IS_ON; uint8_t crc_present = spiRead(RH_RF95_REG_1C_HOP_CHANNEL) & RH_RF95_RX_PAYLOAD_CRC_IS_ON;
@ -227,6 +242,16 @@ void RH_RF95::handleInterrupt()
_cad = irq_flags & RH_RF95_CAD_DETECTED; _cad = irq_flags & RH_RF95_CAD_DETECTED;
setModeIdle(); setModeIdle();
} }
enableInterrupt(); // Let ISR run again
}
void RH_RF95::loop()
{
while (pendingInterrupt) {
pendingInterrupt = false; // If the flag was set, it is _guaranteed_ the ISR won't be running, because it masked itself
handleInterrupt();
}
} }
// These are low level functions that call the interrupt handler for the correct // These are low level functions that call the interrupt handler for the correct
@ -235,17 +260,17 @@ void RH_RF95::handleInterrupt()
void RH_INTERRUPT_ATTR RH_RF95::isr0() void RH_INTERRUPT_ATTR RH_RF95::isr0()
{ {
if (_deviceForInterrupt[0]) if (_deviceForInterrupt[0])
_deviceForInterrupt[0]->handleInterrupt(); _deviceForInterrupt[0]->handleInterruptLevel0();
} }
void RH_INTERRUPT_ATTR RH_RF95::isr1() void RH_INTERRUPT_ATTR RH_RF95::isr1()
{ {
if (_deviceForInterrupt[1]) if (_deviceForInterrupt[1])
_deviceForInterrupt[1]->handleInterrupt(); _deviceForInterrupt[1]->handleInterruptLevel0();
} }
void RH_INTERRUPT_ATTR RH_RF95::isr2() void RH_INTERRUPT_ATTR RH_RF95::isr2()
{ {
if (_deviceForInterrupt[2]) if (_deviceForInterrupt[2])
_deviceForInterrupt[2]->handleInterrupt(); _deviceForInterrupt[2]->handleInterruptLevel0();
} }
// Check whether the latest received message is complete and uncorrupted // Check whether the latest received message is complete and uncorrupted

View File

@ -806,12 +806,17 @@ class RH_RF95 : public RHSPIDriver
/// Return true if we are currently receiving a packet /// Return true if we are currently receiving a packet
bool isReceiving(); bool isReceiving();
void loop(); // Perform idle processing
protected: protected:
/// This is a low level function to handle the interrupts for one instance of RH_RF95. /// This is a low level function to handle the interrupts for one instance of RH_RF95.
/// Called automatically by isr*() /// Called automatically by isr*()
/// Should not need to be called by user code. /// Should not need to be called by user code.
virtual void handleInterrupt(); virtual void handleInterrupt();
/// This is the only code called in ISR context, it just queues up our helper thread to run handleInterrupt();
void RH_INTERRUPT_ATTR handleInterruptLevel0();
/// Examine the revceive buffer to determine whether the message is for this node /// Examine the revceive buffer to determine whether the message is for this node
void validateRxBuf(); void validateRxBuf();
@ -846,6 +851,11 @@ class RH_RF95 : public RHSPIDriver
/// Index of next interrupt number to use in _deviceForInterrupt /// Index of next interrupt number to use in _deviceForInterrupt
static uint8_t _interruptCount; static uint8_t _interruptCount;
bool enableInterrupt(); // enable our IRQ
void disableInterrupt(); // disable our IRQ
volatile bool pendingInterrupt = false;
/// The configured interrupt pin connected to this instance /// The configured interrupt pin connected to this instance
uint8_t _interruptPin; uint8_t _interruptPin;

View File

@ -15,8 +15,8 @@ ErrorCode SimRadio::send(MeshPacket *p)
return ERRNO_OK; return ERRNO_OK;
} }
void RadioInterface::deliverToReceiverISR(MeshPacket *p, BaseType_t *higherPriWoken) void RadioInterface::deliverToReceiver(MeshPacket *p)
{ {
assert(rxDest); assert(rxDest);
assert(rxDest->enqueueFromISR(p, higherPriWoken)); // NOWAIT - fixme, if queue is full, delete older messages assert(rxDest->enqueue(p, 0)); // NOWAIT - fixme, if queue is full, delete older messages
} }

View File

@ -24,7 +24,7 @@ class RadioInterface
/** /**
* Enqueue a received packet for the registered receiver * Enqueue a received packet for the registered receiver
*/ */
void deliverToReceiverISR(MeshPacket *p, BaseType_t *higherPriWoken); void deliverToReceiver(MeshPacket *p);
public: public:
/** pool is the pool we will alloc our rx packets from /** pool is the pool we will alloc our rx packets from