RF24Mesh - Automated Networking for nrf24L01 & nrf52x radios v2.0.1
2024 - A user friendly mesh overlay for sensor neworks using RF24Network
Loading...
Searching...
No Matches
RF24Mesh.cpp
Go to the documentation of this file.
1
7#include "RF24Mesh.h"
8#include "RF24Mesh_config.h"
9#if defined(__linux) && !defined(__ARDUINO_X86__)
10 #include <fstream>
11#endif
12
13template<class network_t, class radio_t>
14ESBMesh<network_t, radio_t>::ESBMesh(radio_t& _radio, network_t& _network) : radio(_radio), network(_network)
15{
16 setCallback(NULL);
17 meshStarted = false;
18#if !defined(MESH_NOMASTER)
19 addrMemAllocated = false;
20#endif
21}
22
23/*****************************************************/
24
25template<class network_t, class radio_t>
26bool ESBMesh<network_t, radio_t>::begin(uint8_t channel, rf24_datarate_e data_rate, uint32_t timeout)
27{
28 //delay(1); // Found problems w/SPIDEV & ncurses. Without this, getch() returns a stream of garbage
29 if (meshStarted) {
30 radio.stopListening();
31 }
32 meshStarted = true;
33 if (!radio.begin())
34 return 0;
35 radio.setChannel(channel);
36 radio.setDataRate(data_rate);
37 network.returnSysMsgs = true;
38
39 if (getNodeID() > 0) { //Not master node
40 if (renewAddress(timeout) == MESH_DEFAULT_ADDRESS) {
41 return false;
42 }
43 }
44 else {
45#if !defined(MESH_NOMASTER)
46 if (!addrMemAllocated) {
47 addrMemAllocated = true;
48 addrList = (addrListStruct*)malloc((MESH_MEM_ALLOC_SIZE * sizeof(addrListStruct)));
49 addrListTop = 0;
50 loadDHCP();
51 }
52#endif
53 mesh_address = 0;
54 network.begin(mesh_address);
55 }
56
57 return true;
58}
59
60/*****************************************************/
61
62template<class network_t, class radio_t>
64{
65 uint8_t type = network.update();
66 if (mesh_address == MESH_DEFAULT_ADDRESS) return type;
67
68#if !defined(MESH_NOMASTER)
69 if (type == NETWORK_REQ_ADDRESS) {
70 doDHCP = 1;
71 }
72
73 if (!getNodeID()) {
74 if ((type == MESH_ADDR_LOOKUP || type == MESH_ID_LOOKUP)) {
75 RF24NetworkHeader* header = (RF24NetworkHeader*)(&network.frame_buffer);
76 header->to_node = header->from_node;
77
78 int16_t returnAddr = 0;
79 if (type == MESH_ADDR_LOOKUP) {
80 returnAddr = getAddress(network.frame_buffer[sizeof(RF24NetworkHeader)]);
81 network.write(*header, &returnAddr, sizeof(returnAddr));
82 }
83 else {
84 int16_t addr = 0;
85 memcpy(&addr, &network.frame_buffer[sizeof(RF24NetworkHeader)], sizeof(addr));
86 returnAddr = getNodeID(addr);
87 network.write(*header, &returnAddr, sizeof(returnAddr));
88 }
89 }
90 else if (type == MESH_ADDR_RELEASE) {
91 uint16_t* fromAddr = (uint16_t*)network.frame_buffer;
92 releaseAddress(*fromAddr);
93 }
94 }
95#endif
96
97 return type;
98}
99
100/*****************************************************/
101
102template<class network_t, class radio_t>
103bool ESBMesh<network_t, radio_t>::write(uint16_t to_node, const void* data, uint8_t msg_type, size_t size)
104{
105 if (mesh_address == MESH_DEFAULT_ADDRESS) return 0;
106
107 RF24NetworkHeader header(to_node, msg_type);
108 return network.write(header, data, size);
109}
110
111/*****************************************************/
112
113template<class network_t, class radio_t>
114bool ESBMesh<network_t, radio_t>::write(const void* data, uint8_t msg_type, size_t size, uint8_t nodeID)
115{
116 if (mesh_address == MESH_DEFAULT_ADDRESS) return 0;
117
118 int16_t toNode = 0;
119 uint32_t lookupTimeout = millis() + MESH_WRITE_TIMEOUT;
120 uint32_t retryDelay = 5;
121
122 if (nodeID) {
123 while ((toNode = getAddress(nodeID)) < 0) {
124 if (millis() > lookupTimeout || toNode == -2) {
125 return 0;
126 }
127 retryDelay += 10;
128 delay(retryDelay);
129 }
130 }
131 return write(toNode, data, msg_type, size);
132}
133
134/*****************************************************/
135
136template<class network_t, class radio_t>
138{
139 radio.stopListening();
140 radio.setChannel(_channel);
141 radio.startListening();
142}
143
144/*****************************************************/
145
146template<class network_t, class radio_t>
148{
149 network.networkFlags = allow ? network.networkFlags & ~FLAG_NO_POLL : network.networkFlags | FLAG_NO_POLL;
150}
151
152/*****************************************************/
153
154template<class network_t, class radio_t>
156{
157 // getAddress() doesn't use auto-ack; do a double-check to manually retry 1 more time
158 for (uint8_t i = 0; i < MESH_CONNECTION_CHECK_ATTEMPTS; i++) {
159
160 int16_t result = getAddress(_nodeID);
161 switch (result) {
162 case -2: return false; // Address not found in list or is default
163 case -1: continue; // Write failed or timed out
164 case 0: return false; // This is a master node
165 default:
166 if ((uint16_t)result == mesh_address) { // Successful address lookup if result == RF24Network address
167 return true;
168 }
169 break;
170 }
171 }
172 return false;
173}
174
175/*****************************************************/
176
177template<class network_t, class radio_t>
179{ // Master will return and send 00 address for a nodeID with address 0, -1 if not found
180
181 //if (nodeID == _nodeID) return mesh_address;
182 if (!nodeID) return 0;
183 if (mesh_address == MESH_DEFAULT_ADDRESS) return -2;
184
185// Lets say 0 if nodeID 0, -1 if write failed or timed out, -2 if not found in list or address is default,
186#if !defined(MESH_NOMASTER)
187 if (!getNodeID()) { //Master Node
188 for (uint8_t i = 0; i < addrListTop; i++) {
189 if (addrList[i].nodeID == nodeID) {
190 return addrList[i].address;
191 }
192 }
193 return -2;
194 }
195#endif
196
197 RF24NetworkHeader header(00, MESH_ADDR_LOOKUP);
198 if (network.write(header, &nodeID, sizeof(nodeID))) {
199 uint32_t timer = millis();
200 while (network.update() != MESH_ADDR_LOOKUP) {
202 if (millis() - timer > MESH_LOOKUP_TIMEOUT) {
203 return -1;
204 }
205 }
206 int16_t address = 0;
207 memcpy(&address, network.frame_buffer + sizeof(RF24NetworkHeader), sizeof(address));
208 return address;
209 }
210 return -1;
211}
212
213/*****************************************************/
214
215template<class network_t, class radio_t>
217{
218 if (address == MESH_BLANK_ID) return _nodeID;
219 if (address == 0) return 0;
220 if (mesh_address == MESH_DEFAULT_ADDRESS) return -2;
221
222#if !defined(MESH_NOMASTER)
223 if (!mesh_address) { //Master Node
224 for (uint8_t i = 0; i < addrListTop; i++) {
225 if (addrList[i].address == address) {
226 return addrList[i].nodeID;
227 }
228 }
229 return -2;
230 }
231#endif
232
233 RF24NetworkHeader header(00, MESH_ID_LOOKUP);
234 if (network.write(header, &address, sizeof(address))) {
235 uint32_t timer = millis();
236 while (network.update() != MESH_ID_LOOKUP) {
238 if (millis() - timer > MESH_LOOKUP_TIMEOUT) return -1;
239 }
240 int16_t ID = 0;
241 memcpy(&ID, &network.frame_buffer[sizeof(RF24NetworkHeader)], sizeof(ID));
242 return ID;
243 }
244 return -1;
245}
246
247/*****************************************************/
248
249template<class network_t, class radio_t>
250uint8_t ESBMesh<network_t, radio_t>::getLevel(uint16_t address)
251{
252 uint8_t count = 0;
253 while (address) {
254 address >>= 3;
255 count++;
256 }
257 return count;
258}
259
260/*****************************************************/
261
262template<class network_t, class radio_t>
264{
265 radio.stopListening();
267 mesh_address = MESH_DEFAULT_ADDRESS;
268}
269
270/*****************************************************/
271
272template<class network_t, class radio_t>
274{
275 if (mesh_address == MESH_DEFAULT_ADDRESS) return 0;
276
277 RF24NetworkHeader header(00, MESH_ADDR_RELEASE);
278 if (network.write(header, 0, 0)) {
279 beginDefault();
280 return 1;
281 }
282 return 0;
283}
284
285/*****************************************************/
286
287#ifndef MESH_NOMASTER
288template<class network_t, class radio_t>
290{
291 for (uint8_t i = 0; i < addrListTop; ++i) {
292 if (addrList[i].address == address) {
293 addrList[i].address = 0;
294 return true;
295 }
296 }
297 return false;
298}
299#endif
300
301/*****************************************************/
302
303template<class network_t, class radio_t>
305{
306 if (radio.available()) network.update();
307
308 uint8_t reqCounter = 0;
309 uint8_t totalReqs = 0;
310
311 beginDefault();
312
313 uint32_t start = millis();
314 while (!requestAddress(reqCounter)) {
315 if (millis() - start > timeout) break;
316
317 uint32_t timeoutInternal = 50 + ((totalReqs + 1) * (reqCounter + 1)) * 2;
318 uint32_t startInternal = millis();
319 while (millis() - startInternal < timeoutInternal) {
321 delay(1);
322 }
323 reqCounter++;
324 reqCounter = reqCounter % 4;
325 totalReqs++;
326 totalReqs = totalReqs % 10;
327 }
328 return mesh_address;
329}
330
331/*****************************************************/
332
333template<class network_t, class radio_t>
335{
336 RF24NetworkHeader header(MESH_MULTICAST_ADDRESS, NETWORK_POLL);
337 //Find another radio, starting with level 0 multicast
338 IF_RF24MESH_DEBUG(printf_P(PSTR("MSH Poll Level %d\n"), level));
339 network.multicast(header, 0, 0, level);
340
341 uint32_t timeout = millis() + 55;
342#define MESH_MAXPOLLS 4
343 uint16_t contactNode[MESH_MAXPOLLS];
344 uint8_t pollCount = 0;
345
346 while (millis() < timeout && pollCount < MESH_MAXPOLLS) {
347#if defined(RF24MESH_DEBUG)
348 bool goodSignal = radio.testRPD();
349#endif
350 if (network.update() == NETWORK_POLL) {
351 uint16_t contact = 0;
352 memcpy(&contact, &network.frame_buffer[0], sizeof(contact));
353
354 // Drop duplicate polls to help prevent duplicate requests
355 bool isDupe = false;
356 for (uint8_t i = 0; i < pollCount; ++i) {
357 if (contact == contactNode[i]) {
358 isDupe = true;
359 break;
360 }
361 }
362 if (!isDupe) {
363 contactNode[pollCount] = contact;
364 ++pollCount;
365 IF_RF24MESH_DEBUG(printf_P(PSTR("MSH Poll %c -64dbm from 0%o \n"), (goodSignal ? '>' : '<'), contact));
366 }
367
368 } // end if
369
371 } // end while
372
373 IF_RF24MESH_DEBUG(printf_P(PSTR("MSH Polls from level %d: %d\n"), level, pollCount));
374
375 if (!pollCount) return 0;
376
377 for (uint8_t i = 0; i < pollCount; i++) {
378
379 bool gotResponse = 0;
380
381 // Request an address via the contact node
382 header.type = NETWORK_REQ_ADDRESS;
383 header.reserved = _nodeID;
384 header.to_node = contactNode[i];
385
386 // Do a direct write (no ack) to the contact node. Include the nodeId and address.
387 network.write(header, 0, 0, contactNode[i]);
388
389 IF_RF24MESH_DEBUG(printf_P(PSTR("MSH Request address from: 0%o\n"), contactNode[i]));
390
391 timeout = millis() + 225;
392
393 while (millis() < timeout) {
394 if (network.update() == NETWORK_ADDR_RESPONSE) {
395 if (network.frame_buffer[7] == _nodeID) {
396 uint16_t newAddy = 0;
397 memcpy(&newAddy, &network.frame_buffer[sizeof(RF24NetworkHeader)], sizeof(newAddy));
398 uint16_t mask = 0xFFFF;
399 newAddy &= ~(mask << (3 * getLevel(contactNode[i]))); // Get the level of contact node. Multiply by 3 to get the number of bits to shift (3 per digit)
400 if (newAddy == contactNode[i]) { // Then shift the mask by this much, and invert it bitwise. Apply the mask to the newly received
401 gotResponse = 1; // address to evalute whether 'subnet' of the assigned address matches the contact node address.
402 break;
403 }
404 }
405 }
407 }
408
409 if (!gotResponse) {
410 continue;
411 }
412
413 uint16_t newAddress = 0;
414 memcpy(&newAddress, network.frame_buffer + sizeof(RF24NetworkHeader), sizeof(newAddress));
415
416 IF_RF24MESH_DEBUG(printf_P(PSTR("Set address: Current: 0%o New: 0%o\n"), mesh_address, newAddress));
417 mesh_address = newAddress;
418
419 radio.stopListening();
420 network.begin(mesh_address);
421
422 // getNodeID() doesn't use auto-ack; do a double-check to manually retry 1 more time
423 if (getNodeID(mesh_address) != _nodeID) {
424 if (getNodeID(mesh_address) != _nodeID) {
425 beginDefault();
426 continue;
427 }
428 }
429 return 1;
430 } // end for
431
432 return 0;
433}
434
435/*****************************************************/
436
437template<class network_t, class radio_t>
439{
440 _nodeID = nodeID;
441}
442
443/*****************************************************/
444#if !defined(MESH_NOMASTER)
445
446template<class network_t, class radio_t>
447void ESBMesh<network_t, radio_t>::setStaticAddress(uint8_t nodeID, uint16_t address)
448{
449 setAddress(nodeID, address);
450}
451
452/*****************************************************/
453
454template<class network_t, class radio_t>
455void ESBMesh<network_t, radio_t>::setAddress(uint8_t nodeID, uint16_t address, bool searchBy)
456{
457 //Look for the node in the list
458 for (uint8_t i = 0; i < addrListTop; i++) {
459 if (searchBy == false) {
460 if (addrList[i].nodeID == nodeID) {
461 addrList[i].address = address;
462 #if defined(__linux) && !defined(__ARDUINO_X86__)
463 saveDHCP();
464 #endif
465 return; //Found & set, complete
466 }
467 }
468 else { // Search by address, set the nodeID
469 if (addrList[i].address == address) {
470 //printf("*** Addr 0%o Found, reassign fr ID %d to ID %d ***\n", addrList[i].address, addrList[i].nodeID, nodeID);
471 addrList[i].nodeID = nodeID;
472 #if defined(__linux) && !defined(__ARDUINO_X86__)
473 saveDHCP();
474 #endif
475 return;
476 }
477 }
478 }
479
480 if (addrListTop > 0 && addrListTop % MESH_MEM_ALLOC_SIZE == 0) {
481 addrList = (addrListStruct*)realloc(addrList, (addrListTop + MESH_MEM_ALLOC_SIZE) * sizeof(addrListStruct));
482 }
483 addrList[addrListTop].address = address;
484 addrList[addrListTop++].nodeID = nodeID; //Set the value AND increment Top without another line of code
485 #if defined(__linux) && !defined(__ARDUINO_X86__)
486 saveDHCP();
487 #endif
488}
489
490/*****************************************************/
491
492template<class network_t, class radio_t>
494{
495
496 #if defined(__linux) && !defined(__ARDUINO_X86__)
497 std::ifstream infile("dhcplist.txt", std::ifstream::binary);
498 if (!infile) return;
499
500 infile.seekg(0, infile.end);
501 int length = infile.tellg();
502 infile.seekg(0, infile.beg);
503
504 addrListStruct tmpNode;
505
506 for (uint8_t i = 0; i < (length / sizeof(addrListStruct)); i++) {
507 infile.read((char*)&tmpNode, sizeof(addrListStruct));
508 setAddress(tmpNode.nodeID, tmpNode.address);
509 }
510 infile.close();
511 #endif
512}
513
514/*****************************************************/
515
516template<class network_t, class radio_t>
518{
519 #if defined(__linux) && !defined(__ARDUINO_X86__)
520 std::ofstream outfile("dhcplist.txt", std::ofstream::binary | std::ofstream::trunc);
521
522 for (int i = 0; i < addrListTop; i++) {
523 outfile.write((char*)&addrList[i], sizeof(addrListStruct));
524 }
525 outfile.close();
526 #endif // __linux & not X86
527}
528
529/*****************************************************/
530
531template<class network_t, class radio_t>
533{
534 if (doDHCP)
535 doDHCP = false;
536 else
537 return;
538
539 RF24NetworkHeader header;
540 memcpy(&header, network.frame_buffer, sizeof(RF24NetworkHeader));
541
542 uint16_t newAddress;
543
544 // Get the unique id of the requester (ID is in header.reserved)
545 if (!header.reserved || header.type != NETWORK_REQ_ADDRESS) {
546 IF_RF24MESH_DEBUG(printf_P(PSTR("MSH Invalid id or type rcvd\n")));
547 return;
548 }
549
550 uint16_t fwd_by = 0;
551 uint8_t shiftVal = 0;
552 bool extraChild = false;
553
554 if (header.from_node != MESH_DEFAULT_ADDRESS) {
555 fwd_by = header.from_node;
556 uint16_t m = fwd_by;
557 uint8_t count = 0;
558
559 while (m) { //Octal addresses convert nicely to binary in threes. Address 03 = B011 Address 033 = B011011
560 m >>= 3; //Find out how many digits are in the octal address
561 count += 3;
562 }
563 shiftVal = count; //Now we know how many bits to shift when adding a child node 1-5 (B001 to B101) to any address
564 }
565 else {
566 //If request is coming from level 1, add an extra child to the master
567 extraChild = 1;
568 }
569
570 // IF_RF24MESH_DEBUG(printf_P(PSTR("%u: MSH Rcv addr req from_id %d\n"), millis(), header.reserved));
571
572 for (int i = MESH_MAX_CHILDREN + extraChild; i > 0; i--) { // For each of the possible addresses (5 max)
573
574 bool found = false;
575 newAddress = fwd_by | (i << shiftVal);
576
577 if (newAddress == MESH_DEFAULT_ADDRESS) continue;
578
579 for (uint8_t i = 0; i < addrListTop; i++) {
580 IF_RF24MESH_DEBUG_MINIMAL(printf_P(PSTR("ID: %d ADDR: 0%o\n"), addrList[i].nodeID, addrList[i].address));
581 if (addrList[i].address == newAddress && addrList[i].nodeID != header.reserved) {
582 found = true;
583 break;
584 }
585 } // 3 conditions: 1. address in list = assigned to somebody else (bad); 2. address in list = assigned to this nodeID (ok); 3. address not in list (ok)
586
587 if (!found) {
588 header.type = NETWORK_ADDR_RESPONSE;
589 header.to_node = header.from_node;
590 //This is a routed request to 00
591
592 setAddress(header.reserved, newAddress);
593 // without this delay, address renewal fails for children with slower execution speed
594 #if defined(SLOW_ADDR_POLL_RESPONSE)
595 delay(SLOW_ADDR_POLL_RESPONSE);
596 #endif // defined (SLOW_ADDR_POLL_RESPONSE)
597
598 if (header.from_node != MESH_DEFAULT_ADDRESS) { //Is NOT node 01 to 05
599 //delay(2);
600 if (!network.write(header, &newAddress, sizeof(newAddress))) {
601 network.write(header, &newAddress, sizeof(newAddress));
602 }
603 }
604 else {
605 //delay(2);
606 network.write(header, &newAddress, sizeof(newAddress), header.to_node);
607 }
608
609 IF_RF24MESH_DEBUG(printf_P(PSTR("Sent to 0%o phys: 0%o new: 0%o id: %d\n"), header.to_node, MESH_DEFAULT_ADDRESS, newAddress, header.reserved));
610 break;
611 }
612 else {
613 IF_RF24MESH_DEBUG(printf_P(PSTR("not allocated\n")));
614 }
615 } // end for
616}
617
618/*****************************************************/
619
620#endif // !MESH_NOMASTER
621
622template<class network_t, class radio_t>
623void ESBMesh<network_t, radio_t>::setCallback(void (*meshCallback)(void))
624{
625
626 this->meshCallback = meshCallback;
627}
628
629/*****************************************************/
630
631// ensure the compiler is aware of the possible datatype for the template class
632template class ESBMesh<ESBNetwork<RF24>, RF24>;
633#if defined(ARDUINO_ARCH_NRF52) || defined(ARDUINO_ARCH_NRF52840)
634template class ESBMesh<ESBNetwork<nrf_to_nrf>, nrf_to_nrf>;
635#endif
#define MESH_MAXPOLLS
#define MESH_ADDR_LOOKUP
Definition RF24Mesh.h:25
#define MESH_ID_LOOKUP
Definition RF24Mesh.h:27
#define MESH_CALLBACK
Definition RF24Mesh.h:251
#define MESH_ADDR_RELEASE
Definition RF24Mesh.h:26
#define MESH_BLANK_ID
Definition RF24Mesh.h:29
#define MESH_MULTICAST_ADDRESS
#define IF_RF24MESH_DEBUG(x)
#define MESH_MAX_CHILDREN
Set 1 to 4 (Default: 4) Restricts the maximum children per node.
#define MESH_LOOKUP_TIMEOUT
How long to wait in ms for a response during individual address lookups.
#define IF_RF24MESH_DEBUG_MINIMAL(x)
#define MESH_DEFAULT_ADDRESS
#define MESH_CONNECTION_CHECK_ATTEMPTS
Number of attempts to verify a connection.
#define MESH_MEM_ALLOC_SIZE
master node memory allocation
#define MESH_WRITE_TIMEOUT
How long RF24Mesh::write() retries address lookups before timing out. Allows multiple attempts.
int16_t getNodeID(uint16_t address=MESH_BLANK_ID)
Definition RF24Mesh.cpp:216
void setStaticAddress(uint8_t nodeID, uint16_t address)
Definition RF24Mesh.cpp:447
void loadDHCP()
Definition RF24Mesh.cpp:493
uint16_t renewAddress(uint32_t timeout=MESH_RENEWAL_TIMEOUT)
Reconnect to the mesh and renew the current RF24Network address.
Definition RF24Mesh.cpp:304
void setChild(bool allow)
Definition RF24Mesh.cpp:147
bool checkConnection()
Definition RF24Mesh.cpp:155
bool begin(uint8_t channel=MESH_DEFAULT_CHANNEL, rf24_datarate_e data_rate=RF24_1MBPS, uint32_t timeout=MESH_RENEWAL_TIMEOUT)
Definition RF24Mesh.cpp:26
void DHCP()
Definition RF24Mesh.cpp:532
bool releaseAddress()
Definition RF24Mesh.cpp:273
void saveDHCP()
Definition RF24Mesh.cpp:517
void setCallback(void(*meshCallback)(void))
Definition RF24Mesh.cpp:623
uint8_t update()
Definition RF24Mesh.cpp:63
ESBMesh(radio_t &_radio, network_t &_network)
Definition RF24Mesh.cpp:14
void setNodeID(uint8_t nodeID)
Definition RF24Mesh.cpp:438
bool write(const void *data, uint8_t msg_type, size_t size, uint8_t nodeID=0)
Definition RF24Mesh.cpp:114
void setChannel(uint8_t _channel)
Definition RF24Mesh.cpp:137
void setAddress(uint8_t nodeID, uint16_t address, bool searchBy=false)
Definition RF24Mesh.cpp:455
int16_t getAddress(uint8_t nodeID)
Convert a nodeID into an RF24Network address.
Definition RF24Mesh.cpp:178
A struct for storing a nodeID and an address in a single element of the ESBMesh::addrList array.
Definition RF24Mesh.h:322
uint8_t nodeID
The nodeID of an network node (child)
Definition RF24Mesh.h:324
uint16_t address
The logical address of an network node (child)
Definition RF24Mesh.h:326