forked from gioblu/PJON
-
Notifications
You must be signed in to change notification settings - Fork 0
/
ESPNOW.h
222 lines (165 loc) · 7.02 KB
/
ESPNOW.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
/* ESPNOW is a Strategy for the PJON framework.
It supports delivering PJON packets through the Espressif ESPNOW 802.11
protocol to a registered list of devices on the 802.11 ESPNOW network. Each
device must be registered with its device id and mac address.
___________________________________________________________________________
ESPNOW strategy proposed and developed by xlfe in September 2018
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License. */
#pragma once
#include "../../interfaces/ARDUINO/ESPNOWHelper.h"
// Timeout waiting for an ACK. This can be increased if the latency is high.
#ifndef EN_RESPONSE_TIMEOUT
#define EN_RESPONSE_TIMEOUT 100000ul
#endif
#ifndef EN_MAX_REMOTE_NODES
#define EN_MAX_REMOTE_NODES 10
#endif
// Recommended receive time for this strategy, in microseconds
#ifndef EN_RECEIVE_TIME
#define EN_RECEIVE_TIME 0
#endif
#define EN_MAGIC_HEADER (uint8_t*)"\xEE\xFE\x0E\xEF"
class ESPNOW {
bool _espnow_initialised = false;
bool _auto_registration = true;
uint8_t _channel = 14;
char _espnow_pmk[17] =
"\xdd\xdb\xdd\x44\x34\xd5\x6a\x0b\x7e\x9f\x4e\x27\xd6\x5b\xa2\x81";
// Remote nodes
uint8_t _remote_node_count = 0;
uint8_t _remote_id[EN_MAX_REMOTE_NODES];
uint8_t _remote_mac[EN_MAX_REMOTE_NODES][ESP_NOW_ETH_ALEN];
ENHelper en;
bool check_en() {
if(!_espnow_initialised) {
en.set_magic_header(EN_MAGIC_HEADER);
if(en.begin(_channel,(uint8_t*)_espnow_pmk))
_espnow_initialised = true;
}
return _espnow_initialised;
};
int16_t find_remote_node(uint8_t id, uint8_t* mac) {
for(uint8_t i = 0; i < _remote_node_count; i++)
if((_remote_id[i] == id && id != 255) || memcmp(_remote_mac, mac, ESP_NOW_ETH_ALEN) == 0)
return i;
return -1;
};
void autoregister_sender(const uint8_t *message, uint16_t length) {
// Add the last sender to the node table
if(_auto_registration && length>4) {
// First get PJON sender id from incoming packet
PJON_Packet_Info packet_info;
PJONTools::parse_header(message, packet_info);
uint8_t sender_id = packet_info.tx.id;
if(sender_id == 0) {
ESP_LOGE("ESPNOW", "AutoRegister parsing failed");
return; // If parsing fails, it will be 0
}
// Then get the mac address of the sender
uint8_t sender_mac[ESP_NOW_ETH_ALEN];
en.get_sender(sender_mac);
// See if PJON id is already registered, add if not
int16_t pos = find_remote_node(sender_id, sender_mac);
if(pos == -1) {
ESP_LOGI("ESPNOW", "Autoregister new sender %d",sender_id);
add_node(sender_id, sender_mac);
}
else if(memcmp(_remote_mac[pos], sender_mac, ESP_NOW_ETH_ALEN) == 0) { // == 0 is the same
// Update mac of existing node
ESP_LOGI("ESPNOW","Update sender sender_id(%d) [%02X:%02X:%02X:%02X:%02X:%02X]", sender_id, sender_mac[0],sender_mac[1],sender_mac[2],sender_mac[3],sender_mac[4],sender_mac[5]);
memcpy(_remote_mac[pos], sender_mac, ESP_NOW_ETH_ALEN);
}
}
};
public:
void set_pmk(char *espnow_pmk) {
memcpy(_espnow_pmk, espnow_pmk, 16);
}
void set_channel(uint8_t channel) {
_channel = channel;
}
/* Register each device we want to send to */
int16_t add_node(
uint8_t remote_id,
uint8_t remote_mac[]
) {
if(_remote_node_count == EN_MAX_REMOTE_NODES) return -1;
_remote_id[_remote_node_count] = remote_id;
memcpy(_remote_mac[_remote_node_count], remote_mac, ESP_NOW_ETH_ALEN);
en.add_node_mac(remote_mac);
_remote_node_count++;
return _remote_node_count - 1;
};
/* Set if each packet's sender is automatically added as a node */
void set_autoregistration(bool enabled) {
_auto_registration = enabled;
};
/* Returns the suggested delay related to attempts passed as parameter: */
uint32_t back_off(uint8_t attempts) {
return 10000ul * attempts + PJON_RANDOM(10000);
};
/* Begin method, to be called on initialization:
(returns always true) */
bool begin(uint8_t /*did*/ = 0) { return check_en(); };
/* Check if the channel is free for transmission */
bool can_start() { return check_en(); };
/* Returns the maximum number of attempts for each transmission: */
static uint8_t get_max_attempts() { return 10; };
/* Returns the recommended receive time for this strategy: */
static uint16_t get_receive_time() { return EN_RECEIVE_TIME; };
/* Handle a collision (empty because handled on Ethernet level): */
void handle_collision() { };
/* Receive a frame: */
uint16_t receive_frame(uint8_t *data, uint16_t max_length) {
uint16_t length = en.receive_frame(data, max_length);
if(length != PJON_FAIL) autoregister_sender(data, length);
return length;
}
/* Receive byte response */
uint16_t receive_response() {
/* TODO: Improve robustness by ignoring packets not from the previous
receiver (Perhaps not that important as long as ACK/NAK responses are
directed, not broadcast) */
uint32_t start = PJON_MICROS();
uint8_t result[PJON_PACKET_MAX_LENGTH];
uint16_t reply_length = 0;
do {
reply_length = receive_frame(result, sizeof result);
// We expect 1, if packet is larger it is not our ACK
if(reply_length == 1)
if(result[0] == PJON_ACK)
return result[0];
} while ((uint32_t)(PJON_MICROS() - start) < EN_RESPONSE_TIMEOUT);
return PJON_FAIL;
};
/* Send byte response to package transmitter.
We have the IP so we can reply directly. */
void send_response(uint8_t response) { // Empty, PJON_ACK is always sent
en.send_response(response);
};
/* Send a frame: */
void send_frame(uint8_t *data, uint16_t length) {
if(length > 0) {
uint8_t id = data[0]; // Package always starts with a receiver id
PJON_Packet_Info packet_info; // use parser to get intended recipient MAC
PJONTools::parse_header(data, packet_info);
if(id == 0) { // Broadcast, send to all receivers
en.send_frame(data, length);
} else { // To a specific receiver
int16_t pos = find_remote_node(id, packet_info.rx.mac);
if(pos != -1)
en.send_frame(data, length, _remote_mac[pos]);
else //Broadcast - any replies will get registered
en.send_frame(data, length);
}
}
};
};