1
+ // ///////////////////////////////////////////////////////////////
2
+ /*
3
+ UWB Trilateration with ESP32, DW3000 #LocalPositioning
4
+ For More Information: https://youtu.be/NjHMXEOxNRE
5
+ Created by Eric N. (ThatProject)
6
+ */
7
+ // ///////////////////////////////////////////////////////////////
8
+ // Version Info
9
+ // ESP32 v3.0.2
10
+ // Hardware Info
11
+ // https://github.com/Makerfabs/MaUWB_DW3000-with-STM32-AT-Command/
12
+
13
+ #include " config.h"
14
+ #include " Trilateration.hpp"
15
+ #include < Wire.h>
16
+ #include < Adafruit_GFX.h>
17
+ #include < Adafruit_SSD1306.h>
18
+
19
+ HardwareSerial SERIAL_AT (2 );
20
+ Adafruit_SSD1306 display (128 , 64 , &Wire, -1 );
21
+
22
+ String response = " " ;
23
+
24
+ Anchor anchor0 (-340 , 240 );
25
+ Anchor anchor1 (340 , 240 );
26
+ Anchor anchor2 (0 , -240 );
27
+
28
+ Trilateration trilateration (&anchor0, &anchor1, &anchor2);
29
+
30
+ void setup () {
31
+ pinMode (RESET, OUTPUT);
32
+ digitalWrite (RESET, HIGH);
33
+
34
+ SERIAL_LOG.begin (115200 );
35
+
36
+ xTaskCreatePinnedToCore (Task_UWB,
37
+ " Task_UWB" ,
38
+ 1024 * 20 ,
39
+ NULL ,
40
+ tskIDLE_PRIORITY,
41
+ NULL ,
42
+ 0 );
43
+ }
44
+
45
+ void loop () {}
46
+
47
+ void drawScreen () {
48
+ display.clearDisplay ();
49
+ display.setTextSize (1 ); // Normal 1:1 pixel scale
50
+ display.setTextColor (SSD1306_WHITE); // Draw white text
51
+ display.setCursor (0 , 0 ); // Start at top-left corner
52
+ display.println (F (" UWB Trilateration" ));
53
+ display.setCursor (0 , 10 );
54
+
55
+ String freqText = " Freq: " ;
56
+
57
+ #ifdef FREQ_850K
58
+ freqText = freqText + " 850k" ;
59
+ #else
60
+ freqText = freqText + " 6.8M" ;
61
+ #endif
62
+ display.println (freqText);
63
+ display.setTextSize (2 );
64
+ display.setCursor (0 , 20 );
65
+
66
+ String infoText = " " ;
67
+
68
+ #ifdef TAG
69
+ infoText = infoText + " Tag: " + UWB_INDEX;
70
+ #else
71
+ infoText = infoText + " Anchor: " + UWB_INDEX;
72
+ #endif
73
+ display.println (infoText);
74
+
75
+ #ifdef TAG
76
+ display.setTextSize (1 );
77
+ display.setCursor (0 , 40 );
78
+ display.println (" User Location: " );
79
+
80
+ #else
81
+ display.setCursor (0 , 50 );
82
+ display.setTextSize (1 );
83
+ display.println (" ......Updating......" );
84
+ #endif
85
+ display.display ();
86
+
87
+ delay (1000 );
88
+ }
89
+
90
+ String sendData (String command, const int timeout, boolean debug) {
91
+ String response = " " ;
92
+
93
+ SERIAL_LOG.println (command);
94
+ SERIAL_AT.println (command); // send the read character to the SERIAL_LOG
95
+
96
+ long int time = millis ();
97
+
98
+ while ((time + timeout) > millis ()) {
99
+ while (SERIAL_AT.available ()) {
100
+
101
+ // The esp has data so display its output to the serial window
102
+ char c = SERIAL_AT.read (); // read the next character.
103
+ response += c;
104
+ }
105
+ }
106
+ return response;
107
+ }
108
+
109
+ String config_cmd () {
110
+ String temp = " AT+SETCFG=" ;
111
+
112
+ // Set device id
113
+ temp = temp + UWB_INDEX;
114
+
115
+ // Set device role
116
+ #ifdef TAG
117
+ temp = temp + " ,0" ;
118
+ #endif
119
+ #ifdef ANCHOR
120
+ temp = temp + " ,1" ;
121
+ #endif
122
+
123
+ // Set frequence 850k or 6.8M
124
+ #ifdef FREQ_850K
125
+ temp = temp + " ,0" ;
126
+ #endif
127
+ #ifdef FREQ_6800K
128
+ temp = temp + " ,1" ;
129
+ #endif
130
+
131
+ // Set range filter
132
+ temp = temp + " ,1" ;
133
+
134
+ return temp;
135
+ }
136
+
137
+ String cap_cmd () {
138
+ String temp = " AT+SETCAP=" ;
139
+
140
+ // Set Tag capacity
141
+ temp = temp + UWB_TAG_COUNT;
142
+
143
+ // Time of a single time slot
144
+ #ifdef FREQ_850K
145
+ temp = temp + " ,15" ;
146
+ #endif
147
+ #ifdef FREQ_6800K
148
+ temp = temp + " ,10" ;
149
+ #endif
150
+
151
+ return temp;
152
+ }
153
+
154
+
155
+
156
+ void Task_UWB (void *pvParameters) {
157
+
158
+ SERIAL_AT.begin (115200 , SERIAL_8N1, IO_RXD2, IO_TXD2);
159
+
160
+ SERIAL_AT.println (" AT" );
161
+ Wire.begin (I2C_SDA, I2C_SCL);
162
+ delay (1000 );
163
+ // SSD1306_SWITCHCAPVCC = generate display voltage from 3.3V internally
164
+ if (!display.begin (SSD1306_SWITCHCAPVCC, 0x3C )) { // Address 0x3C for 128x32
165
+ SERIAL_LOG.println (F (" SSD1306 allocation failed" ));
166
+ for (;;)
167
+ ; // Don't proceed, loop forever
168
+ }
169
+ display.clearDisplay ();
170
+
171
+ drawScreen ();
172
+
173
+ sendData (" AT?" , 2000 , 1 );
174
+ // sendData("AT+RESTORE", 5000, 1);
175
+
176
+ sendData (config_cmd (), 2000 , 1 );
177
+ sendData (cap_cmd (), 2000 , 1 );
178
+
179
+ sendData (" AT+SETRPT=1" , 2000 , 1 );
180
+ // sendData("AT+SAVE", 2000, 1);
181
+ sendData (" AT+RESTART" , 2000 , 1 );
182
+
183
+
184
+ while (1 ) {
185
+
186
+ #ifdef TAG
187
+ tag_action ();
188
+ #else
189
+ vTaskDelay (10 );
190
+ #endif
191
+ }
192
+ }
193
+
194
+ void tag_action () {
195
+ while (SERIAL_LOG.available () > 0 ) {
196
+ SERIAL_AT.write (SERIAL_LOG.read ());
197
+ yield ();
198
+ }
199
+ while (SERIAL_AT.available () > 0 ) {
200
+ char c = SERIAL_AT.read ();
201
+
202
+ if (c == ' \r ' )
203
+ continue ;
204
+ else if (c == ' \n ' || c == ' \r ' ) {
205
+ SERIAL_LOG.println (response);
206
+
207
+ float rangeValues[3 ];
208
+ // Parse data, update distances, and calculate user location
209
+ if (parseData (response, rangeValues)) {
210
+ updateAnchorDistances (rangeValues);
211
+ calculateUserLocation ();
212
+ }
213
+
214
+ response = " " ;
215
+ vTaskDelay (10 );
216
+ } else
217
+ response += c;
218
+ }
219
+ }
220
+
221
+ bool parseData (String dataString, float rangeValues[3 ]) {
222
+ // Find the position of "range:("
223
+ int rangeStart = dataString.indexOf (" range:(" );
224
+ if (rangeStart == -1 ) return false ; // Return false if "range:(" not found
225
+
226
+ // Extract the substring containing the range values
227
+ int rangeEnd = dataString.indexOf (" )" , rangeStart);
228
+ String rangePart = dataString.substring (rangeStart + 7 , rangeEnd);
229
+
230
+ // Parse the range values
231
+ char charArray[rangePart.length () + 1 ];
232
+ rangePart.toCharArray (charArray, rangePart.length () + 1 );
233
+ char *token = strtok (charArray, " ," );
234
+ int valueIndex = 0 ;
235
+ while (token != NULL && valueIndex < 3 ) {
236
+ rangeValues[valueIndex] = atoi (token);
237
+ valueIndex++;
238
+ token = strtok (NULL , " ," );
239
+ }
240
+
241
+ return true ;
242
+ }
243
+
244
+ // Function to update the distances for the anchors
245
+ void updateAnchorDistances (float rangeValues[3 ]) {
246
+ anchor0.updateDistance (rangeValues[0 ]);
247
+ anchor1.updateDistance (rangeValues[1 ]);
248
+ anchor2.updateDistance (rangeValues[2 ]);
249
+ }
250
+
251
+ // Function to perform trilateration and calculate user location
252
+ void calculateUserLocation () {
253
+ float user_x, user_y;
254
+ trilateration.calcUserLocation (user_x, user_y);
255
+
256
+ char buffer[100 ];
257
+ snprintf (buffer, sizeof (buffer), " x = %.2f\n y = %.2f" , user_x, user_y);
258
+ Serial.println (buffer);
259
+
260
+ display.fillRect (0 , 48 , 128 , 16 , SSD1306_BLACK);
261
+ display.setCursor (0 , 48 );
262
+ display.print (buffer);
263
+ display.display ();
264
+ }
0 commit comments