mirror of https://github.com/arendst/Tasmota.git
219 lines
7.5 KiB
C++
219 lines
7.5 KiB
C++
|
|
/*
|
|
* Getting Started example sketch for nRF24L01+ radios
|
|
* This is a very basic example of how to send data from one node to another
|
|
* but modified to include failure handling.
|
|
*
|
|
* The nrf24l01+ radios are fairly reliable devices, but on breadboards etc, with inconsistent wiring, failures may
|
|
* occur randomly after many hours to days or weeks. This sketch demonstrates how to handle the various failures and
|
|
* keep the radio operational.
|
|
*
|
|
* The three main failure modes of the radio include:
|
|
* Writing to radio: Radio unresponsive - Fixed internally by adding a timeout to the internal write functions in RF24 (failure handling)
|
|
* Reading from radio: Available returns true always - Fixed by adding a timeout to available functions by the user. This is implemented internally in RF24Network.
|
|
* Radio configuration settings are lost - Fixed by monitoring a value that is different from the default, and re-configuring the radio if this setting reverts to the default.
|
|
*
|
|
* The printDetails output should appear as follows for radio #0:
|
|
*
|
|
* STATUS = 0x0e RX_DR=0 TX_DS=0 MAX_RT=0 RX_P_NO=7 TX_FULL=0
|
|
* RX_ADDR_P0-1 = 0x65646f4e31 0x65646f4e32
|
|
* RX_ADDR_P2-5 = 0xc3 0xc4 0xc5 0xc6
|
|
* TX_ADDR = 0x65646f4e31
|
|
* RX_PW_P0-6 = 0x20 0x20 0x00 0x00 0x00 0x00
|
|
* EN_AA = 0x3f
|
|
* EN_RXADDR = 0x02
|
|
* RF_CH = 0x4c
|
|
* RF_SETUP = 0x03
|
|
* CONFIG = 0x0f
|
|
* DYNPD/FEATURE = 0x00 0x00
|
|
* Data Rate = 1MBPS
|
|
* Model = nRF24L01+
|
|
* CRC Length = 16 bits
|
|
* PA Power = PA_LOW
|
|
*
|
|
*Users can use this sketch to troubleshoot radio module wiring etc. as it makes the radios hot-swapable
|
|
*
|
|
* Updated: 2019 by TMRh20
|
|
*/
|
|
|
|
#include <SPI.h>
|
|
#include "RF24.h"
|
|
#include "printf.h"
|
|
|
|
/****************** User Config ***************************/
|
|
/*** Set this radio as radio number 0 or 1 ***/
|
|
bool radioNumber = 0;
|
|
|
|
/* Hardware configuration: Set up nRF24L01 radio on SPI bus plus pins 7 & 8 */
|
|
RF24 radio(7,8);
|
|
/**********************************************************/
|
|
|
|
byte addresses[][6] = {"1Node","2Node"};
|
|
|
|
// Used to control whether this node is sending or receiving
|
|
bool role = 0;
|
|
|
|
/**********************************************************/
|
|
//Function to configure the radio
|
|
void configureRadio(){
|
|
|
|
radio.begin();
|
|
|
|
// Set the PA Level low to prevent power supply related issues since this is a
|
|
// getting_started sketch, and the likelihood of close proximity of the devices. RF24_PA_MAX is default.
|
|
radio.setPALevel(RF24_PA_LOW);
|
|
|
|
// Open a writing and reading pipe on each radio, with opposite addresses
|
|
if(radioNumber){
|
|
radio.openWritingPipe(addresses[1]);
|
|
radio.openReadingPipe(1,addresses[0]);
|
|
}else{
|
|
radio.openWritingPipe(addresses[0]);
|
|
radio.openReadingPipe(1,addresses[1]);
|
|
}
|
|
|
|
// Start the radio listening for data
|
|
radio.startListening();
|
|
radio.printDetails();
|
|
}
|
|
|
|
/**********************************************************/
|
|
|
|
void setup() {
|
|
Serial.begin(115200);
|
|
Serial.println(F("RF24/examples/GettingStarted"));
|
|
Serial.println(F("*** PRESS 'T' to begin transmitting to the other node"));
|
|
|
|
printf_begin();
|
|
|
|
configureRadio();
|
|
}
|
|
|
|
uint32_t configTimer = millis();
|
|
|
|
void loop() {
|
|
|
|
if(radio.failureDetected){
|
|
radio.failureDetected = false;
|
|
delay(250);
|
|
Serial.println("Radio failure detected, restarting radio");
|
|
configureRadio();
|
|
}
|
|
//Every 5 seconds, verify the configuration of the radio. This can be done using any
|
|
//setting that is different from the radio defaults.
|
|
if(millis() - configTimer > 5000){
|
|
configTimer = millis();
|
|
if(radio.getDataRate() != RF24_1MBPS){
|
|
radio.failureDetected = true;
|
|
Serial.print("Radio configuration error detected");
|
|
}
|
|
}
|
|
|
|
/****************** Ping Out Role ***************************/
|
|
if (role == 1) {
|
|
|
|
radio.stopListening(); // First, stop listening so we can talk.
|
|
|
|
|
|
Serial.println(F("Now sending"));
|
|
|
|
unsigned long start_time = micros(); // Take the time, and send it. This will block until complete
|
|
if (!radio.write( &start_time, sizeof(unsigned long) )){
|
|
Serial.println(F("failed"));
|
|
}
|
|
|
|
radio.startListening(); // Now, continue listening
|
|
|
|
unsigned long started_waiting_at = micros(); // Set up a timeout period, get the current microseconds
|
|
boolean timeout = false; // Set up a variable to indicate if a response was received or not
|
|
|
|
while ( ! radio.available() ){ // While nothing is received
|
|
if (micros() - started_waiting_at > 200000 ){ // If waited longer than 200ms, indicate timeout and exit while loop
|
|
timeout = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( timeout ){ // Describe the results
|
|
Serial.println(F("Failed, response timed out."));
|
|
}else{
|
|
unsigned long got_time; // Grab the response, compare, and send to debugging spew
|
|
|
|
//Failure Handling:
|
|
uint32_t failTimer = millis();
|
|
while(radio.available()){ //If available always returns true, there is a problem
|
|
if(millis() - failTimer > 250){
|
|
radio.failureDetected = true;
|
|
Serial.println("Radio available failure detected");
|
|
break;
|
|
}
|
|
radio.read( &got_time, sizeof(unsigned long) );
|
|
}
|
|
unsigned long end_time = micros();
|
|
|
|
// Spew it
|
|
Serial.print(F("Sent "));
|
|
Serial.print(start_time);
|
|
Serial.print(F(", Got response "));
|
|
Serial.print(got_time);
|
|
Serial.print(F(", Round-trip delay "));
|
|
Serial.print(end_time-start_time);
|
|
Serial.println(F(" microseconds"));
|
|
}
|
|
|
|
// Try again 1s later
|
|
delay(1000);
|
|
}
|
|
|
|
|
|
|
|
/****************** Pong Back Role ***************************/
|
|
|
|
if ( role == 0 )
|
|
{
|
|
unsigned long got_time;
|
|
|
|
if( radio.available()){
|
|
|
|
uint32_t failTimer = millis(); // Variable for the received timestamp
|
|
while (radio.available()) { // While there is data ready
|
|
if(millis()-failTimer > 500){
|
|
Serial.println("Radio available failure detected");
|
|
radio.failureDetected = true;
|
|
break;
|
|
}
|
|
radio.read( &got_time, sizeof(unsigned long) ); // Get the payload
|
|
}
|
|
|
|
radio.stopListening(); // First, stop listening so we can talk
|
|
radio.write( &got_time, sizeof(unsigned long) ); // Send the final one back.
|
|
radio.startListening(); // Now, resume listening so we catch the next packets.
|
|
Serial.print(F("Sent response "));
|
|
Serial.println(got_time);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************** Change Roles via Serial Commands ***************************/
|
|
|
|
if ( Serial.available() )
|
|
{
|
|
char c = toupper(Serial.read());
|
|
if ( c == 'T' && role == 0 ){
|
|
Serial.println(F("*** CHANGING TO TRANSMIT ROLE -- PRESS 'R' TO SWITCH BACK"));
|
|
role = 1; // Become the primary transmitter (ping out)
|
|
|
|
}else
|
|
if ( c == 'R' && role == 1 ){
|
|
Serial.println(F("*** CHANGING TO RECEIVE ROLE -- PRESS 'T' TO SWITCH BACK"));
|
|
role = 0; // Become the primary receiver (pong back)
|
|
radio.startListening();
|
|
|
|
}
|
|
}
|
|
|
|
|
|
} // Loop
|