adsb enhanced
[inav.git] / src / main / io / adsb.c
blob260f89fa6af30a89a6638f5a53dee38aeec2c60f
1 /*
2 * This file is part of Cleanflight.
4 * Cleanflight is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
9 * Cleanflight is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with Cleanflight. If not, see <http://www.gnu.org/licenses/>.
19 #include <string.h>
22 #include "adsb.h"
24 #include "navigation/navigation.h"
25 #include "navigation/navigation_private.h"
27 #include "common/maths.h"
28 #pragma GCC diagnostic push
29 #pragma GCC diagnostic ignored "-Wunused-function"
30 #include "common/mavlink.h"
31 #pragma GCC diagnostic pop
34 #include "math.h"
37 #ifdef USE_ADSB
39 adsbVehicle_t adsbVehiclesList[MAX_ADSB_VEHICLES];
40 adsbVehicleStatus_t adsbVehiclesStatus;
42 adsbVehicleValues_t vehicleValues;
45 adsbVehicleValues_t* getVehicleForFill(void){
46 return &vehicleValues;
49 // use bsearch function
50 adsbVehicle_t *findVehicleByIcao(uint32_t avicao) {
51 for (uint8_t i = 0; i < MAX_ADSB_VEHICLES; i++) {
52 if (avicao == adsbVehiclesList[i].vehicleValues.icao) {
53 return &adsbVehiclesList[i];
56 return NULL;
59 adsbVehicle_t *findVehicleFarthest(void) {
60 adsbVehicle_t *adsbLocal = NULL;
61 for (uint8_t i = 0; i < MAX_ADSB_VEHICLES; i++) {
62 if (adsbVehiclesList[i].ttl > 0 && adsbVehiclesList[i].calculatedVehicleValues.valid && (adsbLocal == NULL || adsbLocal->calculatedVehicleValues.dist < adsbVehiclesList[i].calculatedVehicleValues.dist)) {
63 adsbLocal = &adsbVehiclesList[i];
66 return adsbLocal;
69 uint8_t getActiveVehiclesCount(void) {
70 uint8_t total = 0;
71 for (uint8_t i = 0; i < MAX_ADSB_VEHICLES; i++) {
72 if (adsbVehiclesList[i].ttl > 0) {
73 total++;
76 return total;
79 adsbVehicle_t *findVehicleClosest(void) {
80 adsbVehicle_t *adsbLocal = NULL;
81 for (uint8_t i = 0; i < MAX_ADSB_VEHICLES; i++) {
82 if (adsbVehiclesList[i].ttl > 0 && adsbVehiclesList[i].calculatedVehicleValues.valid && (adsbLocal == NULL || adsbLocal->calculatedVehicleValues.dist > adsbVehiclesList[i].calculatedVehicleValues.dist)) {
83 adsbLocal = &adsbVehiclesList[i];
86 return adsbLocal;
89 adsbVehicle_t *findFreeSpaceInList(void) {
90 //find expired first
91 for (uint8_t i = 0; i < MAX_ADSB_VEHICLES; i++) {
92 if (adsbVehiclesList[i].ttl == 0) {
93 return &adsbVehiclesList[i];
97 return NULL;
100 adsbVehicle_t *findVehicleNotCalculated(void) {
101 //find expired first
102 for (uint8_t i = 0; i < MAX_ADSB_VEHICLES; i++) {
103 if (adsbVehiclesList[i].calculatedVehicleValues.valid == false) {
104 return &adsbVehiclesList[i];
108 return NULL;
111 adsbVehicle_t* findVehicle(uint8_t index)
113 if (index < MAX_ADSB_VEHICLES){
114 return &adsbVehiclesList[index];
117 return NULL;
120 adsbVehicleStatus_t* getAdsbStatus(void){
121 return &adsbVehiclesStatus;
124 void gpsDistanceCmBearing(int32_t currentLat1, int32_t currentLon1, int32_t destinationLat2, int32_t destinationLon2, uint32_t *dist, int32_t *bearing) {
125 float GPS_scaleLonDown = cos_approx((fabsf((float) gpsSol.llh.lat) / 10000000.0f) * 0.0174532925f);
126 const float dLat = destinationLat2 - currentLat1; // difference of latitude in 1/10 000 000 degrees
127 const float dLon = (float) (destinationLon2 - currentLon1) * GPS_scaleLonDown;
129 *dist = sqrtf(sq(dLat) + sq(dLon)) * DISTANCE_BETWEEN_TWO_LONGITUDE_POINTS_AT_EQUATOR;
130 *bearing = 9000.0f + RADIANS_TO_CENTIDEGREES(atan2_approx(-dLat, dLon)); // Convert the output radians to 100xdeg
131 *bearing = wrap_36000(*bearing);
134 bool adsbHeartbeat(void){
135 adsbVehiclesStatus.heartbeatMessagesTotal++;
136 return true;
139 void adsbNewVehicle(adsbVehicleValues_t* vehicleValuesLocal) {
141 // no valid lat lon or altitude
142 if((vehicleValuesLocal->flags & (ADSB_FLAGS_VALID_ALTITUDE | ADSB_FLAGS_VALID_COORDS)) != (ADSB_FLAGS_VALID_ALTITUDE | ADSB_FLAGS_VALID_COORDS)){
143 return;
146 adsbVehiclesStatus.vehiclesMessagesTotal++;
148 adsbVehicle_t *vehicle = NULL;
150 vehicle = findVehicleByIcao(vehicleValuesLocal->icao);
151 if(vehicle != NULL && vehicleValuesLocal->tslc > ADSB_MAX_SECONDS_KEEP_INACTIVE_PLANE_IN_LIST){
152 vehicle->ttl = 0;
153 return;
156 // non GPS mode, GPS is not fix, just find free space in list or by icao and save vehicle without calculated values
157 if (!enviromentOkForCalculatingDistaceBearing()) {
159 if(vehicle == NULL){
160 vehicle = findFreeSpaceInList();
163 if (vehicle != NULL) {
164 memcpy(&(vehicle->vehicleValues), vehicleValuesLocal, sizeof(vehicle->vehicleValues));
165 vehicle->ttl = ADSB_MAX_SECONDS_KEEP_INACTIVE_PLANE_IN_LIST;
166 vehicle->calculatedVehicleValues.valid = false;
167 return;
169 } else {
170 // GPS mode, GPS is fixed and has enough sats
173 if(vehicle == NULL){
174 vehicle = findFreeSpaceInList();
177 if(vehicle == NULL){
178 vehicle = findVehicleNotCalculated();
181 if(vehicle == NULL){
182 vehicle = findVehicleFarthest();
185 if (vehicle != NULL) {
186 memcpy(&(vehicle->vehicleValues), vehicleValuesLocal, sizeof(vehicle->vehicleValues));
187 recalculateVehicle(vehicle);
188 vehicle->ttl = ADSB_MAX_SECONDS_KEEP_INACTIVE_PLANE_IN_LIST;
189 return;
194 void recalculateVehicle(adsbVehicle_t* vehicle){
195 gpsDistanceCmBearing(gpsSol.llh.lat, gpsSol.llh.lon, vehicle->vehicleValues.lat, vehicle->vehicleValues.lon, &(vehicle->calculatedVehicleValues.dist), &(vehicle->calculatedVehicleValues.dir));
197 if (vehicle != NULL && vehicle->calculatedVehicleValues.dist > ADSB_LIMIT_CM) {
198 vehicle->ttl = 0;
199 return;
202 vehicle->calculatedVehicleValues.verticalDistance = vehicle->vehicleValues.alt - (int32_t)getEstimatedActualPosition(Z) - GPS_home.alt;
203 vehicle->calculatedVehicleValues.valid = true;
206 void adsbTtlClean(timeUs_t currentTimeUs) {
208 static timeUs_t adsbTtlLastCleanServiced = 0;
209 timeDelta_t adsbTtlSinceLastCleanServiced = cmpTimeUs(currentTimeUs, adsbTtlLastCleanServiced);
212 if (adsbTtlSinceLastCleanServiced > 1000000) // 1s
214 for (uint8_t i = 0; i < MAX_ADSB_VEHICLES; i++) {
215 if (adsbVehiclesList[i].ttl > 0) {
216 adsbVehiclesList[i].ttl--;
220 adsbTtlLastCleanServiced = currentTimeUs;
224 bool enviromentOkForCalculatingDistaceBearing(void){
225 return (STATE(GPS_FIX) && gpsSol.numSat > 4);
228 #endif