-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathbistatic_interference_radar.cpp
858 lines (637 loc) · 30.1 KB
/
bistatic_interference_radar.cpp
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
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
#include "bistatic_interference_radar.h"
#include <Arduino.h> // only for diagnostic messages via Serial
#include <WiFi.h> // THIS WILL MAKE THE LIBRARY WORK ONLY IN STA MODE OR AP_STA MODE AS LONG AS YOU'RE CONNECTED AS A STATION, TO ANOTHER ACCESS POINT
#include "esp_wifi.h" // IF YOU HAVE AN ESP32, THIS WILL MAKE THE LIBRARY WORK IN MULTISTATIC MODE AND SOFT AP MODE AS LONG AS YOU'VE GOT AT LEAST ONE STATION CONNECTED TO YOUR SoftAP
int debugRadarMsg = 3;
#define ENABLE_ALARM_THRESHOLD 0 // if 0, the raw variance signal is returned, if 1, only variance signals above threshold are returned, signals under threshold instead return zero or invalid (-1)
// internal variables and arrays
int enableThreshold = ENABLE_ALARM_THRESHOLD;
bool enableAutoRegressive = false;
#define MAX_SAMPLEBUFFERSIZE 256
int * sampleBuffer;
int sampleBufferSize = MAX_SAMPLEBUFFERSIZE; // default size, and maximum size // the instant mobile average is calculated from here on mobileAverageFilterSize = n samples
int sampleBufferIndex = 0;
int sampleBufferValid = 0;
#define MAX_AVERAGEBUFFERSIZE 64
int mobileAverageFilterSize = MAX_SAMPLEBUFFERSIZE;
int mobileAverageBufferSize = MAX_AVERAGEBUFFERSIZE;
int mobileAverage = 0;
int mobileAverageTemp = 0;
int * mobileAverageBuffer; // the instant average value fills this circular buffer of size mobileAverageFilterSize with the mobileAverage values
int mobileAverageBufferIndex = 0;
int mobileAverageBufferValid = 0;
int bufferIndex = 0; // index used for all buffers.
//// PLEASE NOTE: the init value is -1, meaning the variance is invalid, it will stay invalid until it can be effectively calculated
// DO NOT CHANGE THE -1 INITIALIZATION VALUE
int variance = RADAR_BOOTING; // this value is calculated from the current sample and the average calculated from the mobileAverageBuffer values
int variancePrev = 0;
int varianceSample = 0; // deviation of the current sample from the average
int varianceAR = 0; // autoregressive version
int varianceIntegral = 0;
#define MAX_VARIANCE 65535
int varianceThreshold = 3; // in variance arbitrary units
int varianceIntegratorLimitMax = MAX_SAMPLEBUFFERSIZE;
int varianceIntegratorLimit = 3;
int varianceBufferSize = MAX_SAMPLEBUFFERSIZE;
int *varianceBuffer; // holds the variance values
int detectionLevel = 0; // holds the detected level integrated from the varianceBuffer
int varianceBufferIndex = 0;
int varianceBufferValid = 0;
int enableCSVout = 0;
int minimumRSSI = MINIMUM_RSSI;
// functions
int bistatic_interference_radar_init() { // initializes the storage arrays in internal RAM
if (sampleBuffer == NULL) {
sampleBuffer = (int*)malloc(sizeof(int) * sampleBufferSize);
for (int sbtempindex=0; sbtempindex<sampleBufferSize; sbtempindex++) { // initialization to zero
sampleBuffer[sbtempindex] = 0x00;
}
}
if (mobileAverageBuffer == NULL) {
mobileAverageBuffer = (int*)malloc(sizeof(int) * mobileAverageBufferSize);
for (int mabtempindex=0; mabtempindex<mobileAverageBufferSize; mabtempindex++) { // initialization to zero
mobileAverageBuffer[mabtempindex] = 0x00;
}
}
if (varianceBuffer == NULL) {
varianceBuffer = (int*)malloc(sizeof(int) * varianceBufferSize);
for (int vartempindex=0; vartempindex<varianceBufferSize; vartempindex++) { // initialization to zero
varianceBuffer[vartempindex] = 0x00;
}
}
variance = -1; // to make it clear that the initial value MUST be -1
return 1;
}
int bistatic_interference_radar_init_PSRAM() { // initializes the storage arrays in PSRAM (only for ESP32-CAM modules and ESP32-WROVER modules with the onboard SPI RAM chip)
if (sampleBuffer == NULL) {
sampleBuffer = (int*)ps_malloc(sizeof(int) * sampleBufferSize);
for (int sbtempindex=0; sbtempindex<sampleBufferSize; sbtempindex++) { // initialization to zero
sampleBuffer[sbtempindex] = 0x00;
}
}
if (mobileAverageBuffer == NULL) {
mobileAverageBuffer = (int*)ps_malloc(sizeof(int) * mobileAverageBufferSize);
for (int mabtempindex=0; mabtempindex<mobileAverageBufferSize; mabtempindex++) { // initialization to zero
mobileAverageBuffer[mabtempindex] = 0x00;
}
}
if (varianceBuffer == NULL) {
varianceBuffer = (int*)ps_malloc(sizeof(int) * varianceBufferSize);
for (int vartempindex=0; vartempindex<varianceBufferSize; vartempindex++) { // initialization to zero
varianceBuffer[vartempindex] = 0x00;
}
}
variance = -1; // to make it clear that the initial value MUST be -1
return 1;
}
int bistatic_interference_radar_deinit() { // frees the storage arrays
if (sampleBuffer != NULL) {
free(sampleBuffer);
}
if (mobileAverageBuffer != NULL) {
free(mobileAverageBuffer);
}
if (varianceBuffer != NULL) {
free(varianceBuffer);
}
return 1;
}
int bistatic_interference_radar_config(int sampleBufSize = 256, int mobileAvgSize = 64, int varThreshold = 3, int varIntegratorLimit = 3, bool enableAR = false) {
if (sampleBufSize >= MAX_SAMPLEBUFFERSIZE) {
sampleBufSize = MAX_SAMPLEBUFFERSIZE;
}
sampleBufferSize = sampleBufSize;
varianceBufferSize = sampleBufferSize;
if (mobileAvgSize >= MAX_SAMPLEBUFFERSIZE) {
mobileAvgSize = MAX_SAMPLEBUFFERSIZE;
}
mobileAverageFilterSize = mobileAvgSize;
if (varThreshold >= MAX_VARIANCE) {
varThreshold = MAX_VARIANCE;
}
varianceThreshold = varThreshold;
if (varIntegratorLimit >= varianceIntegratorLimitMax) {
varIntegratorLimit = varianceIntegratorLimitMax;
}
varianceIntegratorLimit = varIntegratorLimit;
enableAutoRegressive = enableAR;
return 1;
}
int bistatic_interference_radar_process(int sample = 0) { // send the RSSI signal, returns the detection level ( < 0 -> error, == 0 -> no detection, > 0 -> detection level in variance arbitrary units)
if ((sampleBuffer == NULL) || (mobileAverageBuffer == NULL) || (varianceBuffer == NULL)) {
if (debugRadarMsg >= 1) {
Serial.println("bistatic_interference_radar_process(): alarm: detected unallocated buffers: did you call bistatic_interference_radar_init() to allocate the buffers?");
}
return RADAR_UNINITIALIZED; // unallocated buffers
}
if (sample < minimumRSSI) {
if (debugRadarMsg >= 1) {
Serial.print("bistatic_interference_radar_process(): alarm: signal too low: ");
Serial.print(sample);
Serial.print(" while minimum RSSI limit is: ");
Serial.println(minimumRSSI);
}
return RADAR_RSSI_TOO_LOW;
}
sampleBuffer[sampleBufferIndex] = sample;
sampleBufferIndex++;
if ( sampleBufferIndex >= sampleBufferSize ) { // circular buffer, rewinding the index, if the buffer has been filled at least once, then we may start processing valid data
sampleBufferIndex = 0;
sampleBufferValid = 1;
}
if (sampleBufferValid >= 1) {
// filling in the mobile average data buffer
// the mobile average can be re-calculated even on a full sampleBufferSize set of valid samples, I see no problem in terms of computational load
// calculating the current mobile average now. the sampleBufferIndex points now to the oldest sample
mobileAverageTemp = 0;
int mobilePointer = 0;
for (int mobileAverageSampleIndex = 0; mobileAverageSampleIndex < mobileAverageFilterSize; mobileAverageSampleIndex++) {
mobilePointer = sampleBufferIndex - mobileAverageSampleIndex;
if (mobilePointer <= 0) {
mobilePointer = mobilePointer + (sampleBufferSize -1);
}
mobileAverageTemp = mobileAverageTemp + sampleBuffer[mobilePointer];
}
mobileAverage = mobileAverageTemp / mobileAverageFilterSize;
// filling in the mobile average buffer with the fresh new value
mobileAverageBuffer[mobileAverageBufferIndex] = mobileAverage; // to be fair, this buffer is filled but still ...really unused.
// truth be said, I'm filling the mobileAverageBuffer for future logging purposes. (TBD)
// since we have the current mobile average data, we can also extract the current variance data.
// the variable named "variance" at this point still contains the *previous* value of the variance
variancePrev = variance;
// deviation of the current sample
varianceSample = (sample - mobileAverageBuffer[mobileAverageBufferIndex])*(sample - mobileAverageBuffer[mobileAverageBufferIndex]);
// filling in the variance buffer
varianceBuffer[varianceBufferIndex] = varianceSample;
// the following is a mobile integrator filter that parses the circular buffer called varianceBuffer
varianceIntegral = 0;
int variancePointer = 0;
for (int varianceBufferIndexTemp = 0; varianceBufferIndexTemp < varianceIntegratorLimit; varianceBufferIndexTemp++) {
variancePointer = varianceBufferIndex - varianceBufferIndexTemp;
if (variancePointer <=0) {
variancePointer = variancePointer + (varianceBufferSize -1);
}
varianceIntegral = varianceIntegral + varianceBuffer[variancePointer]; // the full effect of this operation is to make the system more sensitive to continued variations of the RSSI, possibly meaning there's a moving object around the area.
}
// increasing and checking the variance buffer index
varianceBufferIndex++;
if ( varianceBufferIndex >= varianceBufferSize ) { // circular buffer, rewinding the index, if the buffer has been filled at least once, then we may start processing valid data
varianceBufferIndex = 0;
varianceBufferValid = 1; //please note we DO NOT need to have a fully validated buffer to work with the current M.A. data
}
// applying the autoregressive part
varianceAR = (varianceIntegral + varianceAR) / 2; // the effect of this filter is to "smooth" down the signal over time, so it's a simple IIR (infinite impulse response) low pass filter. It makes the system less sensitive to noisy signals, especially those with a deviation of less than 1.
// diagnostics section
if (debugRadarMsg >= 2) {
Serial.print("RSSI: ");
Serial.print(sample);
Serial.print(", mobileAverage: ");
Serial.print(mobileAverage);
Serial.print(", deviation: ");
Serial.print(sample - mobileAverage);
Serial.print(", variance: ");
Serial.print(varianceSample);
Serial.print(", varianceIntegral: ");
Serial.print(varianceIntegral);
Serial.print(", varianceAR: ");
Serial.println(varianceAR);
}
// assigning the values according to the settings
variance = varianceSample;
if (enableAutoRegressive) {
variance = varianceAR;
}
if (! enableAutoRegressive) {
variance = varianceIntegral;
}
// note: we needed to point to the current mobile average data for future operations, so we increase the MA buffer index only as the last step
mobileAverageBufferIndex++;
if ( mobileAverageBufferIndex >= mobileAverageBufferSize ) { // circular buffer, rewinding the index, if the buffer has been filled at least once, then we may start processing valid data
mobileAverageBufferIndex = 0;
mobileAverageBufferValid = 1; //please note we DO NOT need to have a fully validated buffer to work with the current M.A. data
}
}
// final check to determine if the detected variance signal is above the detection threshold, this is only done if enableThreshold > 0
if ((variance >= varianceThreshold) && (enableThreshold > 0)) {
detectionLevel = variance;
if (debugRadarMsg >= 1) {
Serial.print("bistatic_interference_radar_process(): detected variance signal above threshold: ");
Serial.print(detectionLevel);
}
return detectionLevel;
}
// variance signal under threshold, but otherwise valid?
if ((variance < varianceThreshold) && (variance >= 0) && (enableThreshold > 0) ) {
detectionLevel = 0;
if (debugRadarMsg >= 2) {
Serial.print("bistatic_interference_radar_process(): variance signal under threshold: ");
Serial.print(variance);
}
return detectionLevel;
}
return variance; // if the sample buffer is still invalid at startup, an invalid value is returned: -1, else the raw variance signal is returned
}
int bistatic_interference_radar() { // request the RSSI level internally, then process the signal and return the detection level in variance arbitrary units
int RSSIlevel = 0;
int res = 0;
/* // see, this gets commented because I suspect this only works with ESP32 or ESP8266 devices. ALSO, the wifi is supposed to be already initialized correctly and connected to an external AP
if (WiFi.getMode() & WIFI_MODE_NULL) {
if (debugRadarMsg >= 1) {
Serial.println("bistatic_interference_radar(): detected WIFI_MODE_NULL: WIFI not configured: you should configure WIFI in advance and have a working connection before using the bistatic_interference_radar library");
}
return WIFI_MODEINVALID;
}
*/
RSSIlevel = (int)WiFi.RSSI(); // I know, this is lame, but will make this library work with most Arduino-supported devices
// if the RSSI is zero, then we are most probably not connected to an upstream AP.
if (RSSIlevel == 0) {
if (debugRadarMsg >= 1) {
Serial.println("bistatic_interference_radar(): WIFI_MODE_STA: RSSI equal to zero detected: wifi connection lost: without a working STA connection the radar is inoperable");
}
return RADAR_INOPERABLE; // radar inoperable
}
res = bistatic_interference_radar_process(RSSIlevel);
return res;
}
// bistatic version ESP32 ONLY
#define SCANMODE_STA 0
#define SCANMODE_SOFTAP 1
#define SCANMODE_WIFIPROBE 2
int scanMode = SCANMODE_STA; // 0 = STA RSSI (SCANMODE_STA); 1 = SoftAP client scan (SCANMODE_SOFTAP); 2 = active wifi scan (SCANMODE_WIFIPROBE);
uint8_t strongestClientBSSID[6] = {0};
int strongestClientRSSI = -100;
int strongestClientfound = 0; // if found set to 1
int modeRes; // wifi mode reporting variable, initialized when radar operations are requested, and shared with other subfunctions
uint8_t BSSIDinUse[6] = {0}; // this array identifies the currently used BSSID
int bistatic_get_rssi_SoftAP_strongestClient() {
int rssi = 0;
//int scanRes = 0;
uint8_t * currentBSSID;
int currentRSSI = 0;
wifi_sta_list_t stationList;
esp_err_t scanRes = esp_wifi_ap_get_sta_list(&stationList);
if (scanRes != ESP_OK) {
if (debugRadarMsg >= 1) {
Serial.print("bistatic_get_rssi_SoftAP_strongestClient(): scan failed: returning zero RSSI: see file esp_err.h for this error code: ");
Serial.println(scanRes);
/*
*
Possible error codes:
ESP_OK: succeed; 0
ESP_ERR_WIFI_NOT_INIT: WiFi is not initialized by esp_wifi_init; 0x3001
ESP_ERR_INVALID_ARG: invalid argument; 0x102
ESP_ERR_WIFI_MODE: WiFi mode is wrong; 0x3005
ESP_ERR_WIFI_CONN: WiFi internal error, the station/soft-AP control block is invalid; 0x3007
https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/error-codes.html
*/
}
return 0;
}
if (debugRadarMsg >= 2) {
Serial.print("bistatic_get_rssi_SoftAP_strongestClient(): found clients: ");
Serial.println(stationList.num);
}
// extracting the strongest signal (we need to mark it for subsequent scans, in order to build reliable stats) // this is executed one time when strongestClientfound == 0
if (strongestClientfound == 0) {
strongestClientRSSI = -100;
for (int netItem = 0; netItem < stationList.num; netItem++) {
wifi_sta_info_t station = stationList.sta[netItem];
currentBSSID = station.mac;
currentRSSI = station.rssi;
if (debugRadarMsg >= 3) {
Serial.print("bistatic_get_rssi_SoftAP_strongestClient(): processing client: ");
Serial.println(WiFi.SSID(netItem));
}
if ((currentRSSI > -100) || (currentRSSI < 0)) {
if (currentRSSI > strongestClientRSSI) {
strongestClientfound = 1;
strongestClientRSSI = currentRSSI;
memcpy ( strongestClientBSSID, currentBSSID, (sizeof(uint8_t) * 6));
if (debugRadarMsg >= 3) {
Serial.print("bistatic_get_rssi_SoftAP_strongestClient(): strongest client found with RSSI: ");
Serial.println(strongestClientRSSI);
}
}
}
}
}
int bssidScanOK = 0;
if (strongestClientfound == 1) { // looking for the specific BSSID, if not found then reset strongestAPfound to zero and return an invalid RSSI set to 0
for (int netItem = 0; netItem < stationList.num; netItem++) { // parse all the network items found, look for the known strongest BSSID
wifi_sta_info_t station = stationList.sta[netItem];
currentBSSID = station.mac;
for (int bssidIndex = 0; bssidIndex < 6; bssidIndex++) { // compare with the strongest BSSID, byte by byte
if (currentBSSID[bssidIndex] == strongestClientBSSID[bssidIndex]) {
bssidScanOK = 1;
} else {
bssidScanOK = 0; // match failed, resetting the flag
break; // no matter wich single byte failed to match, we break out of the compare loop
}
}
if (bssidScanOK == 1) {
currentRSSI = station.rssi;
if (debugRadarMsg >= 3) {
Serial.print("bistatic_get_rssi_SoftAP_strongestClient(): strongest client found with RSSI: ");
Serial.println(currentRSSI);
}
rssi = currentRSSI; // preparing the final result
break; // exit the scan loop
}
}
if (bssidScanOK == 0) {
strongestClientfound = 0;
return 0; // lost the strongest BSSID, returning an invalid value
}
}
if (strongestClientfound == 0) { // on no clients connected, we change scan mode (if possible)
if (modeRes & WIFI_MODE_APSTA) {
if (WiFi.status() == WL_CONNECTED) {
scanMode = SCANMODE_STA;
} else {
scanMode = SCANMODE_WIFIPROBE; // let's change scan mode
}
}
rssi = 0; // in the meanwhile we return zero rssi
}
// returning the strongest signal
memcpy ( BSSIDinUse, strongestClientBSSID, (sizeof(uint8_t) * 6)); // saving the current BSSID in use
return rssi;
}
uint8_t strongestBSSID[6] = {0};
int strongestRSSI = -100;
int strongestChannel = 0;
int strongestAPfound = 0; // if found set to 1
int bistatic_get_rssi_ScanStrongestAP() {
int rssi = 0;
int scanRes = 0;
uint8_t * currentBSSID;
int currentRSSI = 0;
int currentChannel = 0;
if (strongestAPfound == 0) { // don't have a strongest AP on record yet? Do a slow full scan
scanRes = (int) WiFi.scanNetworks(false, false, false, 300, 0); //scanNetworks(bool async = false, bool show_hidden = false, bool passive = false, uint32_t max_ms_per_chan = 300, uint8_t channel = 0);
} else { // do a single channel active scan, this should be much faster
scanRes = (int) WiFi.scanNetworks(false, false, false, 200, strongestChannel); //scanNetworks(bool async = false, bool show_hidden = false, bool passive = false, uint32_t max_ms_per_chan = 300, uint8_t channel = 0);
}
if (debugRadarMsg >= 3) {
Serial.print("bistatic_get_rssi_ScanStrongestAP(): detected n. AP: ");
Serial.println(scanRes);
}
// extracting the strongest signal (we need to mark it for subsequent scans, in order to build reliable stats) // this is executed one time when strongestAPfound == 0
if (strongestAPfound == 0) {
strongestRSSI = -100;
for (int netItem = 0; netItem < scanRes; netItem++) {
currentBSSID = WiFi.BSSID(netItem);
currentRSSI = WiFi.RSSI(netItem);
currentChannel = WiFi.channel(netItem);
if (debugRadarMsg >= 3) {
Serial.print("bistatic_get_rssi_ScanStrongestAP(): processing AP: ");
Serial.print(WiFi.SSID(netItem));
Serial.print(" on channel: ");
Serial.print(currentChannel);
Serial.print(" with RSSI: ");
Serial.println(currentRSSI);
}
if ((currentRSSI > -100) || (currentRSSI < 0)) {
if (currentRSSI > strongestRSSI) {
strongestAPfound = 1;
strongestRSSI = currentRSSI;
strongestChannel = currentChannel;
memcpy ( strongestBSSID, currentBSSID, (sizeof(uint8_t) * 6));
if (debugRadarMsg >= 3) {
Serial.print("bistatic_get_rssi_ScanStrongestAP(): strongest AP found on channel: ");
Serial.print(strongestChannel);
Serial.print(" with RSSI: ");
Serial.println(strongestRSSI);
}
}
}
}
}
int bssidScanOK = 0;
if (strongestAPfound == 1) { // looking for the specific BSSID, if not found then reset strongestAPfound to zero and return an invalid RSSI set to 0
for (int netItem = 0; netItem < scanRes; netItem++) { // parse all the network items found, look for the known strongest BSSID
currentBSSID = WiFi.BSSID(netItem);
for (int bssidIndex = 0; bssidIndex < 6; bssidIndex++) { // compare with the strongest BSSID, byte by byte
if (currentBSSID[bssidIndex] == strongestBSSID[bssidIndex]) {
bssidScanOK = 1;
} else {
bssidScanOK = 0; // match failed, resetting the flag
break; // no matter wich single byte failed to match, we break out of the compare loop
}
}
if (bssidScanOK == 1) {
currentRSSI = WiFi.RSSI(netItem);
currentChannel = WiFi.channel(netItem);
if (debugRadarMsg >= 3) {
Serial.print("bistatic_get_rssi_ScanStrongestAP(): strongest AP found on channel: ");
Serial.print(currentChannel);
Serial.print(" with SSID: ");
Serial.println(WiFi.SSID(netItem));
}
rssi = currentRSSI; // preparing the final result
break; // exit the scan loop
}
}
if (bssidScanOK == 0) {
strongestAPfound = 0;
strongestChannel = 0;
WiFi.scanDelete();
return 0; // lost the strongest BSSID, returning an invalid value
}
}
if (strongestAPfound == 0) {
strongestChannel = 0;
rssi = 0;
if (modeRes & WIFI_MODE_APSTA) {
scanMode = SCANMODE_SOFTAP; // let's change scan mode
}
}
memcpy ( BSSIDinUse, strongestBSSID, (sizeof(uint8_t) * 6)); // saving the current BSSID in use
// returning the strongest signal
WiFi.scanDelete();
return rssi;
}
void serialPrintBSSID(uint8_t * localBSSID) {
for (int bssidIndex = 0; bssidIndex < 6; bssidIndex++) {
if (localBSSID == NULL) {
Serial.print("NULL");
return;
}
Serial.print(localBSSID[bssidIndex], HEX);
}
}
int bistatic_interference_radar_esp() { // request the RSSI level internally, then process the signal and return the detection level in variance arbitrary units, self-detects faults and seeks for alternate solutions
int RSSIlevel = 0;
int res = 0;
int scanRes = 0;
modeRes = (int) WiFi.getMode();
if (modeRes & WIFI_MODE_NULL) {
if (debugRadarMsg >= 1) {
Serial.println("bistatic_interference_radar_esp(): detected WIFI_MODE_NULL: WIFI not configured: you should configure WIFI in advance and have a working connection before using the bistatic_interference_radar library");
}
return WIFI_MODEINVALID;
}
if ((modeRes & WIFI_MODE_APSTA) || (modeRes & WIFI_MODE_STA)) {
RSSIlevel = (int)WiFi.RSSI(); // I know, this is lame, but will make this library work with most Arduino-supported devices
}
// if the RSSI is zero, then we are most probably not connected to an upstream AP.
if (RSSIlevel == 0) {
if (debugRadarMsg >= 4) {
Serial.println("bistatic_interference_radar_esp(): failed to get an RSSI with SCANMODE_STA: attempting to use other modes");
}
//if ((modeRes & WIFI_MODE_STA) && (!(modeRes & WIFI_MODE_APSTA))) { // STA only mode
if ((modeRes & WIFI_MODE_APSTA) || (modeRes & WIFI_MODE_AP)) { // also SoftAP available
// we first check for any connected clients, then scan if zero clients have been found
if (debugRadarMsg >= 4) {
Serial.println("bistatic_interference_radar_esp(): WIFI_MODE_AP or WIFI_MODE_APSTA detected: attempting to scan");
}
if (debugRadarMsg >= 6) {
Serial.print("bistatic_interference_radar_esp(): WIFI_MODE: ");
Serial.print(modeRes);
Serial.print(", SCANMODE: ");
Serial.println(scanMode);
}
if (scanMode == SCANMODE_SOFTAP) {
if (debugRadarMsg >= 6) {
Serial.println("Calling bistatic_get_rssi_SoftAP_strongestClient()");
}
RSSIlevel = bistatic_get_rssi_SoftAP_strongestClient();
}
if (scanMode == SCANMODE_WIFIPROBE) {
if (debugRadarMsg >= 6) {
Serial.println("Calling bistatic_get_rssi_ScanStrongestAP()");
}
RSSIlevel = bistatic_get_rssi_ScanStrongestAP();
}
if ((RSSIlevel == 0) && (scanMode == SCANMODE_SOFTAP)){ // SoftAP scan for connected clients failed, switching scan mode
if (debugRadarMsg >= 6) {
Serial.println("scan failed: setting scanMode = SCANMODE_WIFIPROBE");
Serial.println("Calling bistatic_get_rssi_ScanStrongestAP()");
}
scanMode = SCANMODE_WIFIPROBE;
RSSIlevel = bistatic_get_rssi_ScanStrongestAP();
}
if ((RSSIlevel == 0) && (scanMode == SCANMODE_WIFIPROBE)){ // WiFi probe scan for APs failed, switching scan mode
if (debugRadarMsg >= 6) {
Serial.println("interim check level alpha: scan failed: setting scanMode = SCANMODE_SOFTAP");
Serial.println("Calling bistatic_get_rssi_SoftAP_strongestClient()");
}
scanMode = SCANMODE_SOFTAP;
RSSIlevel = bistatic_get_rssi_SoftAP_strongestClient();
}
if (RSSIlevel == 0) { // also no APs around to be scanned
if (debugRadarMsg >= 6) {
Serial.println("interim check level beta: scan failed: setting scanMode = SCANMODE_SOFTAP");
}
scanMode == SCANMODE_SOFTAP; // it is still worth reverting to the most efficient scan mode.
if (debugRadarMsg >= 1) {
Serial.println("bistatic_interference_radar_esp(): all available scan methods failed: radar inoperable due to lack of connected clients and/or nearby access points: bistatic radars need at least one available external transmitter to operate correctly");
}
//return RADAR_INOPERABLE; // radar inoperable
}
}
if (modeRes & WIFI_MODE_STA) { // STA only mode
if (debugRadarMsg >= 4) {
Serial.println("bistatic_interference_radar_esp(): STA-only wifi mode detected: attempting to use SCANMODE_WIFIPROBE");
}
scanMode = SCANMODE_WIFIPROBE;
if (scanMode == SCANMODE_WIFIPROBE) {
RSSIlevel = bistatic_get_rssi_ScanStrongestAP();
}
if (RSSIlevel == 0) {
if (debugRadarMsg >= 1) {
Serial.println("bistatic_interference_radar_esp(): WIFI_MODE_STA: RSSI equal to zero detected even with SCANMODE_WIFIPROBE: the radar is inoperable");
}
//return RADAR_INOPERABLE; // radar inoperable
}
}
if (RSSIlevel == 0) { // also no APs around to be scanned
if (debugRadarMsg >= 6) {
Serial.println("final check: scan failed: setting scanMode = SCANMODE_SOFTAP");
}
scanMode == SCANMODE_SOFTAP; // it is still worth reverting to the most efficient scan mode.
if (debugRadarMsg >= 1) {
Serial.println("bistatic_interference_radar_esp(): all available scan methods failed: radar inoperable due to lack of connected clients and/or nearby access points: bistatic radars need at least one available external transmitter to operate correctly");
}
return RADAR_INOPERABLE; // radar inoperable
}
}
res = bistatic_interference_radar_process(RSSIlevel); // the core operation won't change.
if (enableCSVout) {
//Serial.println("VarianceLevel");
serialPrintBSSID(BSSIDinUse);
Serial.print("_");
Serial.println(RSSIlevel);
Serial.println(res);
}
return res;
}
int bistatic_interference_radar_debug_via_serial(int debugLevel) {
int debugSave = debugRadarMsg;
debugRadarMsg = debugLevel;
if (debugRadarMsg >= 1) {
Serial.print("bistatic_interference_radar_debug_via_serial(): debugging functions (if the current wifi mode allows it):");
Serial.println(bistatic_interference_radar_esp());
int modeRes = (int) WiFi.getMode();
if (modeRes & WIFI_MODE_NULL) {
Serial.print("bistatic_interference_radar_debug_via_serial(): WIFI_MODE_NULL detected: can do nothing useful");
return 0;
}
Serial.print("bistatic_interference_radar_esp() output:");
Serial.println(bistatic_interference_radar_esp());
if ((modeRes & WIFI_MODE_APSTA) || (modeRes & WIFI_MODE_STA)) {
Serial.print("bistatic_get_rssi_ScanStrongestAP() output:");
Serial.println(bistatic_get_rssi_ScanStrongestAP());
}
if ((modeRes & WIFI_MODE_APSTA) || (modeRes & WIFI_MODE_AP)) {
Serial.print("bistatic_get_rssi_SoftAP_strongestClient() output:");
Serial.println(bistatic_get_rssi_SoftAP_strongestClient());
}
}
debugRadarMsg = debugSave; // restore the normal debug level
return debugRadarMsg;
}
int bistatic_interference_radar_set_debug_level(int debugLevel) {
debugRadarMsg = debugLevel;
return debugRadarMsg;
}
int bistatic_interference_radar_enable_serial_CSV_graph_data(int serialCSVen = 0) {
if (serialCSVen >= 0) {
enableCSVout = serialCSVen;
}
if (serialCSVen > 0) {
bistatic_interference_radar_set_debug_level(0);
}
return serialCSVen;
}
int bistatic_interference_radar_set_minimum_RSSI(int rssiMin) {
if ((rssiMin > 0) || (rssiMin < ABSOLUTE_RSSI_LIMIT)) {
rssiMin == ABSOLUTE_RSSI_LIMIT; // which results in disabling the minimum RSSI check
}
if (debugRadarMsg >= 2) {
Serial.print("bistatic_interference_radar_set_minimum_RSSI(): minimumRSSI: ");
Serial.println(minimumRSSI);
}
minimumRSSI = rssiMin;
if (debugRadarMsg >= 1) {
Serial.print("bistatic_interference_radar_set_minimum_RSSI(): set minimumRSSI to: ");
Serial.println(minimumRSSI);
}
return rssiMin;
}
// current status: IMPLEMENTED
int bistatic_interference_radar_enable_alarm(int thresholdEnable) {
if (thresholdEnable < 0) {
thresholdEnable = 0; // which results in disabling the alarm
}
enableThreshold = thresholdEnable; // Lol!
return enableThreshold;
}
// current status: IMPLEMENTED
int bistatic_interference_radar_set_alarm_threshold(int alarmThreshold) {
if (alarmThreshold < 0) {
alarmThreshold = 0; // which results in always enabling the alarm
}
varianceThreshold = alarmThreshold;
return alarmThreshold;
}