Refactored common code from leaf/relay nodes into static functions. Improved comments.
This commit is contained in:
parent
1d28d28a29
commit
7e658242e2
|
@ -9,20 +9,27 @@
|
||||||
/**
|
/**
|
||||||
* Example RF Radio Ping Star Group with Relay
|
* Example RF Radio Ping Star Group with Relay
|
||||||
*
|
*
|
||||||
* This sketch is a more complex example of using the RF24 library for Arduino.
|
* This sketch is a very complex example of using the RF24 library for Arduino.
|
||||||
* Deploy this on up to six nodes. Set one as the 'pong receiver' by tying the
|
* Deploy this on any number of nodes to create a basic mesh network. I have
|
||||||
* role_pin low, and the others will be 'ping transmit' units. The ping units
|
* tested this on 6 nodes, but it should work on many more.
|
||||||
* unit will send out the value of millis() once a second. The pong unit will
|
|
||||||
* respond back with a copy of the value. Each ping unit can get that response
|
|
||||||
* back, and determine how long the whole cycle took.
|
|
||||||
*
|
*
|
||||||
* This example introduces a new role, the 'relay', which can relay pings or
|
* There are three different roles a node can be:
|
||||||
* pongs from one host to another. This is needed in larger meshes because
|
|
||||||
* each radio can only listen to 5-6 others.
|
|
||||||
*
|
*
|
||||||
* This example requires a bit more complexity to determine which unit is which.
|
* @li Leaf. Leaf nodes send a ping to the base unit, and wait for a pong in
|
||||||
* The pong receiver is identified by having its role_pin tied to ground.
|
* return
|
||||||
* The ping senders are further differentiated by a byte in eeprom.
|
*
|
||||||
|
* @li Relay. Relay nodes do the same as a leaf node, AND they relay pings
|
||||||
|
* from leaf nodes toward the base, and relay pongs toward the leaves.
|
||||||
|
*
|
||||||
|
* @li Base. One node is the base station, which receives pings, and sends
|
||||||
|
* a pong back out.
|
||||||
|
*
|
||||||
|
* The address of each node is a number from 1 to n (the # of known nodes).
|
||||||
|
* It is set in EEPROM. To change a nodes address, send the character code
|
||||||
|
* for that address. e.g. send the character '5' to set address 5.
|
||||||
|
*
|
||||||
|
* The role is determined from the topology table. Leafs have no children.
|
||||||
|
* The base node has no parent. Relays have parents and children.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <SPI.h>
|
#include <SPI.h>
|
||||||
|
@ -43,17 +50,11 @@ RF24 radio(8,9);
|
||||||
// Topology
|
// Topology
|
||||||
//
|
//
|
||||||
|
|
||||||
// Radio pipe addresses for the nodes to communicate. Only ping nodes need
|
|
||||||
// dedicated pipes in this topology. Each ping node has a talking pipe
|
|
||||||
// that it will ping into, and a listening pipe that it will listen for
|
|
||||||
// the pong. The pong node listens on all the ping node talking pipes
|
|
||||||
// and sends the pong back on the sending node's specific listening pipe.
|
|
||||||
|
|
||||||
struct node_info
|
struct node_info
|
||||||
{
|
{
|
||||||
uint64_t talking_pipe; // Pipe used to talk to parent node
|
uint64_t talking_pipe; // Pipe used to talk to parent node
|
||||||
uint64_t listening_pipe; // Pipe used to listen to parent node
|
uint64_t listening_pipe; // Pipe used to listen to parent node
|
||||||
uint8_t parent_node; // Number of parent node
|
uint8_t parent_node; // Address of parent node
|
||||||
};
|
};
|
||||||
|
|
||||||
const node_info topology[] =
|
const node_info topology[] =
|
||||||
|
@ -106,10 +107,8 @@ uint8_t find_node( uint8_t current_node, uint8_t target_node )
|
||||||
// Role management
|
// Role management
|
||||||
//
|
//
|
||||||
// Set up role. This sketch uses the same software for all the nodes
|
// Set up role. This sketch uses the same software for all the nodes
|
||||||
// in this system. Doing so greatly simplifies testing. The hardware itself specifies
|
// in this system. Doing so greatly simplifies testing. Role is
|
||||||
// which node it is.
|
// determined by the topology table.
|
||||||
//
|
|
||||||
// This is done through the role_pin
|
|
||||||
//
|
//
|
||||||
|
|
||||||
// The various roles supported by this sketch
|
// The various roles supported by this sketch
|
||||||
|
@ -133,7 +132,7 @@ const uint8_t valid_eeprom_flag = 0xdf;
|
||||||
|
|
||||||
// What is our address (SRAM cache of the address from EEPROM)
|
// What is our address (SRAM cache of the address from EEPROM)
|
||||||
// This is an index into the topology[] table above
|
// This is an index into the topology[] table above
|
||||||
uint8_t node_address = role_invalid;;
|
uint8_t node_address = -1;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Payload
|
// Payload
|
||||||
|
@ -171,15 +170,16 @@ const unsigned long pong_timeout = 250; // ms
|
||||||
const unsigned long ping_phase_shift = 100; // ms
|
const unsigned long ping_phase_shift = 100; // ms
|
||||||
const short timeout_shift_threshold = 3;
|
const short timeout_shift_threshold = 3;
|
||||||
|
|
||||||
|
// Space to track the last packet we received from each node, useful
|
||||||
|
// for tracking lost packets
|
||||||
|
static uint16_t last_id_received[num_nodes];
|
||||||
|
|
||||||
void setup(void)
|
void setup(void)
|
||||||
{
|
{
|
||||||
//
|
//
|
||||||
// Address
|
// Address
|
||||||
//
|
//
|
||||||
|
|
||||||
// Unless we find reasonable values in the EEPROM, these are the defaults
|
|
||||||
node_address = -1;
|
|
||||||
|
|
||||||
// Look for the token in EEPROM to indicate the following value is
|
// Look for the token in EEPROM to indicate the following value is
|
||||||
// a validly set node address
|
// a validly set node address
|
||||||
if ( EEPROM.read(address_at_eeprom_location) == valid_eeprom_flag )
|
if ( EEPROM.read(address_at_eeprom_location) == valid_eeprom_flag )
|
||||||
|
@ -244,7 +244,7 @@ void setup(void)
|
||||||
|
|
||||||
// Each leaf node has a talking pipe that it will ping into, and a listening
|
// Each leaf node has a talking pipe that it will ping into, and a listening
|
||||||
// pipe that it will listen for the pong. Relay nodes also do this.
|
// pipe that it will listen for the pong. Relay nodes also do this.
|
||||||
if ( role == role_leaf || role == role_relay )
|
if ( role == role_leaf )
|
||||||
{
|
{
|
||||||
// Write on our talking pipe
|
// Write on our talking pipe
|
||||||
radio.openWritingPipe(topology[node_address].talking_pipe);
|
radio.openWritingPipe(topology[node_address].talking_pipe);
|
||||||
|
@ -258,11 +258,14 @@ void setup(void)
|
||||||
// Remember to re-open the reading pipe whenever we start to listen again.
|
// Remember to re-open the reading pipe whenever we start to listen again.
|
||||||
if ( role == role_relay )
|
if ( role == role_relay )
|
||||||
{
|
{
|
||||||
|
// Write on our talking pipe
|
||||||
|
radio.openWritingPipe(topology[node_address].talking_pipe);
|
||||||
|
|
||||||
// Listen on our listening pipe
|
// Listen on our listening pipe
|
||||||
radio.openReadingPipe(0,topology[node_address].listening_pipe);
|
radio.openReadingPipe(0,topology[node_address].listening_pipe);
|
||||||
}
|
}
|
||||||
|
|
||||||
// The base and relay nodes listens on all their children node's talking pipes
|
// The base and relay nodes listen on all their children node's talking pipes
|
||||||
// and sends the pong back on the child node's specific listening pipe.
|
// and sends the pong back on the child node's specific listening pipe.
|
||||||
if ( role == role_base || role == role_relay )
|
if ( role == role_base || role == role_relay )
|
||||||
{
|
{
|
||||||
|
@ -300,6 +303,10 @@ void setup(void)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ping_if_ready(void);
|
||||||
|
void handle_pong(const payload_t& payload);
|
||||||
|
void check_pong_timeout(void);
|
||||||
|
|
||||||
void loop(void)
|
void loop(void)
|
||||||
{
|
{
|
||||||
//
|
//
|
||||||
|
@ -308,39 +315,12 @@ void loop(void)
|
||||||
|
|
||||||
if ( role == role_leaf )
|
if ( role == role_leaf )
|
||||||
{
|
{
|
||||||
// Is it time to ping again?
|
ping_if_ready();
|
||||||
unsigned long now = millis();
|
check_pong_timeout();
|
||||||
if ( now - last_ping_sent_at >= ping_delay )
|
|
||||||
{
|
|
||||||
last_ping_sent_at = now;
|
|
||||||
waiting_for_pong = true;
|
|
||||||
|
|
||||||
// First, stop listening so we can talk.
|
|
||||||
radio.stopListening();
|
|
||||||
|
|
||||||
// Take the time, and send it to the base. This will block until complete
|
|
||||||
payload_t ping(node_address,0,millis());
|
|
||||||
|
|
||||||
// Print details.
|
|
||||||
printf("%lu ",millis());
|
|
||||||
payload_printf(">PING",ping);
|
|
||||||
bool ok = radio.write( &ping, sizeof(payload_t) );
|
|
||||||
if (ok)
|
|
||||||
printf(" ok\n\r");
|
|
||||||
else
|
|
||||||
printf(" failed\n\r");
|
|
||||||
|
|
||||||
// Now, continue listening
|
|
||||||
radio.startListening();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Did we get a pong?
|
// Did we get a pong?
|
||||||
if ( radio.available() )
|
if ( radio.available() )
|
||||||
{
|
{
|
||||||
// Not waiting anymore, got one.
|
|
||||||
waiting_for_pong = false;
|
|
||||||
consecutive_timeouts = 0;
|
|
||||||
|
|
||||||
// Dump the payloads until we've gotten everything
|
// Dump the payloads until we've gotten everything
|
||||||
payload_t payload;
|
payload_t payload;
|
||||||
boolean done = false;
|
boolean done = false;
|
||||||
|
@ -349,32 +329,9 @@ void loop(void)
|
||||||
// Fetch the payload, and see if this was the last one.
|
// Fetch the payload, and see if this was the last one.
|
||||||
done = radio.read( &payload, sizeof(payload_t) );
|
done = radio.read( &payload, sizeof(payload_t) );
|
||||||
|
|
||||||
// Print details.
|
handle_pong(payload);
|
||||||
printf("%lu ",millis());
|
|
||||||
payload_printf(">PONG",payload);
|
|
||||||
printf(" Round-trip delay: %lu\n\r",millis()-payload.time);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Have we timed out waiting for our pong?
|
|
||||||
if ( waiting_for_pong && ( millis() - last_ping_sent_at > pong_timeout ) )
|
|
||||||
{
|
|
||||||
// Not waiting anymore, timed out.
|
|
||||||
waiting_for_pong = false;
|
|
||||||
|
|
||||||
// Timeouts usually happen because of collisions with other nodes
|
|
||||||
// getting a pong just as we are trying to get a ping. The best thing
|
|
||||||
// to do right now is offset our ping timing to search for a slot
|
|
||||||
// that's not occupied.
|
|
||||||
//
|
|
||||||
// Only do this after getting a few timeouts, so we aren't always skittishly
|
|
||||||
// moving around the cycle.
|
|
||||||
if ( ++consecutive_timeouts > timeout_shift_threshold )
|
|
||||||
last_ping_sent_at += ping_phase_shift;
|
|
||||||
|
|
||||||
// Print details
|
|
||||||
printf("TIMED OUT.\n\r");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
|
@ -385,35 +342,8 @@ void loop(void)
|
||||||
{
|
{
|
||||||
#if 1
|
#if 1
|
||||||
// Relay role is ALSO a ping sender!!
|
// Relay role is ALSO a ping sender!!
|
||||||
|
ping_if_ready();
|
||||||
// Is it time to ping again?
|
check_pong_timeout();
|
||||||
unsigned long now = millis();
|
|
||||||
if ( now - last_ping_sent_at >= ping_delay )
|
|
||||||
{
|
|
||||||
last_ping_sent_at = now;
|
|
||||||
|
|
||||||
// First, stop listening so we can talk.
|
|
||||||
radio.stopListening();
|
|
||||||
|
|
||||||
// Write on our talking pipe. The relay has to do this every time, because
|
|
||||||
// we ALSO use pipe 0 as a listening pipe.
|
|
||||||
radio.openWritingPipe(topology[node_address].talking_pipe);
|
|
||||||
|
|
||||||
// Take the time, and send it to the base. This will block until complete
|
|
||||||
payload_t ping(node_address,0,millis());
|
|
||||||
|
|
||||||
printf("%lu ",millis());
|
|
||||||
payload_printf(">PING",ping);
|
|
||||||
bool ok = radio.write( &ping, sizeof(payload_t) );
|
|
||||||
if (ok)
|
|
||||||
printf(" ok.\n\r");
|
|
||||||
else
|
|
||||||
printf(" failed.\n\r");
|
|
||||||
|
|
||||||
// Now, continue listening
|
|
||||||
radio.openReadingPipe(0,topology[node_address].listening_pipe);
|
|
||||||
radio.startListening();
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
// if there is data ready
|
// if there is data ready
|
||||||
uint8_t pipe_num;
|
uint8_t pipe_num;
|
||||||
|
@ -430,10 +360,7 @@ void loop(void)
|
||||||
// Is this for us?
|
// Is this for us?
|
||||||
if ( payload.to_node == node_address )
|
if ( payload.to_node == node_address )
|
||||||
{
|
{
|
||||||
// Treat it as a PONG
|
handle_pong(payload);
|
||||||
printf("%lu ",millis());
|
|
||||||
payload_printf(">PONG",payload);
|
|
||||||
printf(" Round-trip delay: %lu\n\r",millis()-payload.time);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -472,12 +399,10 @@ void loop(void)
|
||||||
uint16_t pipe_id = out_pipe & 0xffff;
|
uint16_t pipe_id = out_pipe & 0xffff;
|
||||||
printf("OUT on pipe %04x %s.\n\r",pipe_id,ok?"ok":"failed");
|
printf("OUT on pipe %04x %s.\n\r",pipe_id,ok?"ok":"failed");
|
||||||
|
|
||||||
|
// Now, resume listening so we catch the next packets.
|
||||||
|
radio.startListening();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now, resume listening so we catch the next packets.
|
|
||||||
radio.openReadingPipe(0,topology[node_address].listening_pipe);
|
|
||||||
radio.startListening();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -503,6 +428,17 @@ void loop(void)
|
||||||
printf("%lu ",millis());
|
printf("%lu ",millis());
|
||||||
payload_printf("PING",ping);
|
payload_printf("PING",ping);
|
||||||
printf(" on pipe %u. ",pipe_num);
|
printf(" on pipe %u. ",pipe_num);
|
||||||
|
|
||||||
|
// Track the packets lost since we last heard from this node
|
||||||
|
// Packet loss is generally a sign of poor system health
|
||||||
|
uint16_t* last_id_ptr = &last_id_received[ping.from_node];
|
||||||
|
if ( *last_id_ptr && ping.id > *last_id_ptr )
|
||||||
|
{
|
||||||
|
uint16_t lost = ping.id - *last_id_ptr - 1;
|
||||||
|
if ( lost )
|
||||||
|
printf(" lost %u",lost);
|
||||||
|
}
|
||||||
|
*last_id_ptr = ping.id;
|
||||||
}
|
}
|
||||||
|
|
||||||
// First, stop listening so we can talk
|
// First, stop listening so we can talk
|
||||||
|
@ -552,4 +488,72 @@ void loop(void)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ping_if_ready(void)
|
||||||
|
{
|
||||||
|
// Is it time to ping again?
|
||||||
|
unsigned long now = millis();
|
||||||
|
if ( now - last_ping_sent_at >= ping_delay )
|
||||||
|
{
|
||||||
|
last_ping_sent_at = now;
|
||||||
|
waiting_for_pong = true;
|
||||||
|
|
||||||
|
// First, stop listening so we can talk.
|
||||||
|
radio.stopListening();
|
||||||
|
|
||||||
|
// Write on our talking pipe. The relay has to do this every time, because
|
||||||
|
// we ALSO use pipe 0 as a listening pipe.
|
||||||
|
radio.openWritingPipe(topology[node_address].talking_pipe);
|
||||||
|
|
||||||
|
// Take the time, and send it to the base. This will block until complete
|
||||||
|
payload_t ping(node_address,0,millis());
|
||||||
|
|
||||||
|
// Print details.
|
||||||
|
printf("%lu ",millis());
|
||||||
|
payload_printf(">PING",ping);
|
||||||
|
bool ok = radio.write( &ping, sizeof(payload_t) );
|
||||||
|
if (ok)
|
||||||
|
printf(" ok\n\r");
|
||||||
|
else
|
||||||
|
printf(" failed\n\r");
|
||||||
|
|
||||||
|
// Now, continue listening
|
||||||
|
radio.startListening();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void handle_pong(const payload_t& payload)
|
||||||
|
{
|
||||||
|
// Not waiting anymore, got one.
|
||||||
|
waiting_for_pong = false;
|
||||||
|
consecutive_timeouts = 0;
|
||||||
|
|
||||||
|
// Print details.
|
||||||
|
printf("%lu ",millis());
|
||||||
|
payload_printf(">PONG",payload);
|
||||||
|
printf(" Round-trip delay: %lu\n\r",millis()-payload.time);
|
||||||
|
}
|
||||||
|
|
||||||
|
void check_pong_timeout(void)
|
||||||
|
{
|
||||||
|
// Have we timed out waiting for our pong?
|
||||||
|
if ( waiting_for_pong && ( millis() - last_ping_sent_at > pong_timeout ) )
|
||||||
|
{
|
||||||
|
// Not waiting anymore, timed out.
|
||||||
|
waiting_for_pong = false;
|
||||||
|
|
||||||
|
// Timeouts usually happen because of collisions with other nodes
|
||||||
|
// getting a pong just as we are trying to send a ping. The best thing
|
||||||
|
// to do right now is offset our ping timing to search for a slot
|
||||||
|
// that's not occupied.
|
||||||
|
//
|
||||||
|
// Only do this after getting a few timeouts, so we aren't always skittishly
|
||||||
|
// moving around the cycle.
|
||||||
|
if ( ++consecutive_timeouts > timeout_shift_threshold )
|
||||||
|
last_ping_sent_at += ping_phase_shift;
|
||||||
|
|
||||||
|
// Print details
|
||||||
|
printf("TIMED OUT.\n\r");
|
||||||
|
}
|
||||||
|
}
|
||||||
// vim:ai:cin:sts=2 sw=2 ft=cpp
|
// vim:ai:cin:sts=2 sw=2 ft=cpp
|
||||||
|
|
Loading…
Reference in New Issue