2014-03-28 16:52:27 -03:00
/*
This program is free software : you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation , either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
2015-12-23 09:35:01 -04:00
# include "AP_GPS.h"
2014-03-28 16:52:27 -03:00
2015-08-11 03:28:43 -03:00
# include <AP_Common/AP_Common.h>
# include <AP_HAL/AP_HAL.h>
2015-10-26 09:01:52 -03:00
# include <AP_Math/AP_Math.h>
2015-08-11 03:28:43 -03:00
# include <AP_Notify/AP_Notify.h>
2016-02-09 16:59:12 -04:00
# include <GCS_MAVLink/GCS.h>
2017-04-02 11:56:26 -03:00
# include <AP_BoardConfig/AP_BoardConfig.h>
2019-04-04 07:47:33 -03:00
# include <AP_RTC/AP_RTC.h>
2017-05-29 19:02:25 -03:00
# include <climits>
2022-02-25 01:06:28 -04:00
# include <AP_SerialManager/AP_SerialManager.h>
2014-03-28 16:52:27 -03:00
2016-05-16 10:09:06 -03:00
# include "AP_GPS_NOVA.h"
2016-04-13 14:20:05 -03:00
# include "AP_GPS_ERB.h"
# include "AP_GPS_GSOF.h"
# include "AP_GPS_NMEA.h"
# include "AP_GPS_SBF.h"
# include "AP_GPS_SBP.h"
2017-03-27 22:13:31 -03:00
# include "AP_GPS_SBP2.h"
2016-04-13 14:20:05 -03:00
# include "AP_GPS_SIRF.h"
# include "AP_GPS_UBLOX.h"
2016-05-19 20:24:08 -03:00
# include "AP_GPS_MAV.h"
2020-09-03 06:15:35 -03:00
# include "AP_GPS_MSP.h"
2020-12-28 18:42:14 -04:00
# include "AP_GPS_ExternalAHRS.h"
2016-04-13 14:20:05 -03:00
# include "GPS_Backend.h"
2022-01-15 00:26:06 -04:00
# if HAL_SIM_GPS_ENABLED
# include "AP_GPS_SITL.h"
# endif
2016-04-13 14:20:05 -03:00
2020-05-31 09:06:29 -03:00
# if HAL_ENABLE_LIBUAVCAN_DRIVERS
# include <AP_CANManager/AP_CANManager.h>
2017-04-02 11:56:26 -03:00
# include <AP_UAVCAN/AP_UAVCAN.h>
# include "AP_GPS_UAVCAN.h"
# endif
2019-08-27 17:06:38 -03:00
# include <AP_AHRS/AP_AHRS.h>
2019-04-04 07:47:33 -03:00
# include <AP_Logger/AP_Logger.h>
# define GPS_RTK_INJECT_TO_ALL 127
2020-09-06 08:10:02 -03:00
# ifndef GPS_MAX_RATE_MS
2019-04-04 07:47:33 -03:00
# define GPS_MAX_RATE_MS 200 // maximum value of rate_ms (i.e. slowest update rate) is 5hz or 200ms
2020-09-06 08:10:02 -03:00
# endif
2017-01-29 19:02:57 -04:00
# define GPS_BAUD_TIME_MS 1200
2017-07-08 20:06:19 -03:00
# define GPS_TIMEOUT_MS 4000u
2017-01-29 19:02:57 -04:00
// defines used to specify the mask position for use of different accuracy metrics in the blending algorithm
# define BLEND_MASK_USE_HPOS_ACC 1
# define BLEND_MASK_USE_VPOS_ACC 2
# define BLEND_MASK_USE_SPD_ACC 4
2017-03-08 20:47:40 -04:00
# define BLEND_COUNTER_FAILURE_INCREMENT 10
2017-01-29 19:02:57 -04:00
2020-09-23 18:19:52 -03:00
# ifndef HAL_GPS_COM_PORT_DEFAULT
# define HAL_GPS_COM_PORT_DEFAULT 1
# endif
2015-12-23 09:35:01 -04:00
extern const AP_HAL : : HAL & hal ;
2014-03-28 16:52:27 -03:00
2017-03-08 06:11:38 -04:00
// baudrates to try to detect GPSes with
2019-11-16 00:26:28 -04:00
const uint32_t AP_GPS : : _baudrates [ ] = { 9600U , 115200U , 4800U , 19200U , 38400U , 57600U , 230400U , 460800U } ;
2017-03-08 06:11:38 -04:00
// initialisation blobs to send to the GPS to try to get it into the
// right mode
2022-01-13 18:56:19 -04:00
const char AP_GPS : : _initialisation_blob [ ] =
UBLOX_SET_BINARY_230400
# if AP_GPS_SIRF_ENABLED
SIRF_SET_BINARY
# endif
;
2017-03-08 06:11:38 -04:00
2017-10-24 22:30:10 -03:00
AP_GPS * AP_GPS : : _singleton ;
2014-03-28 16:52:27 -03:00
// table of user settable parameters
2015-10-25 14:03:46 -03:00
const AP_Param : : GroupInfo AP_GPS : : var_info [ ] = {
2020-12-30 11:03:15 -04:00
// @Param: _TYPE
2021-03-12 19:09:09 -04:00
// @DisplayName: 1st GPS type
// @Description: GPS type of 1st GPS
2021-12-13 14:10:05 -04:00
// @Values: 0:None,1:AUTO,2:uBlox,5:NMEA,6:SiRF,7:HIL,8:SwiftNav,9:DroneCAN,10:SBF,11:GSOF,13:ERB,14:MAV,15:NOVA,16:HemisphereNMEA,17:uBlox-MovingBaseline-Base,18:uBlox-MovingBaseline-Rover,19:MSP,20:AllyStar,21:ExternalAHRS,22:DroneCAN-MovingBaseline-Base,23:DroneCAN-MovingBaseline-Rover
2015-07-24 02:32:21 -03:00
// @RebootRequired: True
2016-07-26 02:35:15 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _TYPE " , 0 , AP_GPS , _type [ 0 ] , HAL_GPS_TYPE_DEFAULT ) ,
2014-03-28 16:52:27 -03:00
2019-05-26 22:34:13 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _TYPE2
2014-03-28 16:52:27 -03:00
// @DisplayName: 2nd GPS type
// @Description: GPS type of 2nd GPS
2021-12-13 14:10:05 -04:00
// @Values: 0:None,1:AUTO,2:uBlox,5:NMEA,6:SiRF,7:HIL,8:SwiftNav,9:DroneCAN,10:SBF,11:GSOF,13:ERB,14:MAV,15:NOVA,16:HemisphereNMEA,17:uBlox-MovingBaseline-Base,18:uBlox-MovingBaseline-Rover,19:MSP,20:AllyStar,21:ExternalAHRS,22:DroneCAN-MovingBaseline-Base,23:DroneCAN-MovingBaseline-Rover
2015-07-24 02:32:21 -03:00
// @RebootRequired: True
2016-07-26 02:35:15 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _TYPE2 " , 1 , AP_GPS , _type [ 1 ] , 0 ) ,
2019-05-26 22:34:13 -03:00
# endif
2014-06-06 18:58:11 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _NAVFILTER
2014-03-28 16:52:27 -03:00
// @DisplayName: Navigation filter setting
// @Description: Navigation filter engine setting
2014-05-20 10:08:40 -03:00
// @Values: 0:Portable,2:Stationary,3:Pedestrian,4:Automotive,5:Sea,6:Airborne1G,7:Airborne2G,8:Airborne4G
2016-07-26 02:35:15 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _NAVFILTER " , 2 , AP_GPS , _navfilter , GPS_ENGINE_AIRBORNE_4G ) ,
2014-03-28 16:52:27 -03:00
2019-05-26 22:34:13 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _AUTO_SWITCH
2014-06-06 18:58:11 -03:00
// @DisplayName: Automatic Switchover Setting
2021-05-03 22:34:19 -03:00
// @Description: Automatic switchover to GPS reporting best lock, 1:UseBest selects the GPS with highest status, if both are equal the GPS with highest satellite count is used 4:Use primary if 3D fix or better, will revert to 'UseBest' behaviour if 3D fix is lost on primary
2020-10-29 19:56:09 -03:00
// @Values: 0:Use primary, 1:UseBest, 2:Blend, 4:Use primary if 3D fix or better
2014-06-06 18:58:11 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _AUTO_SWITCH " , 3 , AP_GPS , _auto_switch , ( int8_t ) GPSAutoSwitch : : USE_BEST ) ,
2019-05-26 22:34:13 -03:00
# endif
2014-06-06 18:58:11 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _MIN_DGPS
2014-06-06 18:58:11 -03:00
// @DisplayName: Minimum Lock Type Accepted for DGPS
// @Description: Sets the minimum type of differential GPS corrections required before allowing to switch into DGPS mode.
// @Values: 0:Any,50:FloatRTK,100:IntegerRTK
// @User: Advanced
2015-07-24 02:32:21 -03:00
// @RebootRequired: True
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _MIN_DGPS " , 4 , AP_GPS , _min_dgps , 100 ) ,
2014-06-06 18:58:11 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _SBAS_MODE
2014-09-04 01:27:05 -03:00
// @DisplayName: SBAS Mode
// @Description: This sets the SBAS (satellite based augmentation system) mode if available on this GPS. If set to 2 then the SBAS mode is not changed in the GPS. Otherwise the GPS will be reconfigured to enable/disable SBAS. Disabling SBAS may be worthwhile in some parts of the world where an SBAS signal is available but the baseline is too long to be useful.
// @Values: 0:Disabled,1:Enabled,2:NoChange
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _SBAS_MODE " , 5 , AP_GPS , _sbas_mode , 2 ) ,
2014-09-04 01:27:05 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _MIN_ELEV
2014-09-04 01:43:29 -03:00
// @DisplayName: Minimum elevation
// @Description: This sets the minimum elevation of satellites above the horizon for them to be used for navigation. Setting this to -100 leaves the minimum elevation set to the GPS modules default.
// @Range: -100 90
2017-05-02 10:43:10 -03:00
// @Units: deg
2014-09-04 01:43:29 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _MIN_ELEV " , 6 , AP_GPS , _min_elevation , - 100 ) ,
2014-09-04 01:43:29 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _INJECT_TO
2015-04-22 20:10:46 -03:00
// @DisplayName: Destination for GPS_INJECT_DATA MAVLink packets
// @Description: The GGS can send raw serial packets to inject data to multiple GPSes.
2016-01-07 00:43:23 -04:00
// @Values: 0:send to first GPS,1:send to 2nd GPS,127:send to all
2016-07-26 02:35:15 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _INJECT_TO " , 7 , AP_GPS , _inject_to , GPS_RTK_INJECT_TO_ALL ) ,
2015-04-22 20:10:46 -03:00
2022-01-09 19:15:32 -04:00
# if AP_GPS_SBP2_ENABLED || AP_GPS_SBP_ENABLED
2020-12-30 11:03:15 -04:00
// @Param: _SBP_LOGMASK
2015-04-22 20:10:46 -03:00
// @DisplayName: Swift Binary Protocol Logging Mask
// @Description: Masked with the SBP msg_type field to determine whether SBR1/SBR2 data is logged
2017-02-10 20:26:41 -04:00
// @Values: 0:None (0x0000),-1:All (0xFFFF),-256:External only (0xFF00)
2015-04-22 20:10:46 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _SBP_LOGMASK " , 8 , AP_GPS , _sbp_logmask , - 256 ) ,
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_SBP2_ENABLED || AP_GPS_SBP_ENABLED
2015-04-22 20:10:46 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _RAW_DATA
2015-05-04 05:18:34 -03:00
// @DisplayName: Raw data logging
2019-02-11 04:19:08 -04:00
// @Description: Handles logging raw data; on uBlox chips that support raw data this will log RXM messages into logger; on Septentrio this will log on the equipment's SD card and when set to 2, the autopilot will try to stop logging after disarming and restart after arming
2017-10-19 22:15:11 -03:00
// @Values: 0:Ignore,1:Always log,2:Stop logging when disarmed (SBF only),5:Only log every five samples (uBlox only)
2015-07-24 02:32:21 -03:00
// @RebootRequired: True
2016-07-26 02:35:15 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _RAW_DATA " , 9 , AP_GPS , _raw_data , 0 ) ,
2015-05-04 05:18:34 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _GNSS_MODE
2015-07-14 18:12:47 -03:00
// @DisplayName: GNSS system configuration
2016-02-02 03:58:33 -04:00
// @Description: Bitmask for what GNSS system to use on the first GPS (all unchecked or zero to leave GPS as configured)
2021-04-28 06:00:32 -03:00
// @Bitmask: 0:GPS,1:SBAS,2:Galileo,3:Beidou,4:IMES,5:QZSS,6:GLONASS
2015-07-14 18:12:47 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _GNSS_MODE " , 10 , AP_GPS , _gnss_mode [ 0 ] , 0 ) ,
2015-07-14 18:12:47 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _SAVE_CFG
2015-10-28 21:40:50 -03:00
// @DisplayName: Save GPS configuration
2016-02-12 16:32:23 -04:00
// @Description: Determines whether the configuration for this GPS should be written to non-volatile memory on the GPS. Currently working for UBlox 6 series and above.
// @Values: 0:Do not save config,1:Save config,2:Save only when needed
2015-10-28 21:40:50 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _SAVE_CFG " , 11 , AP_GPS , _save_config , 2 ) ,
2015-10-28 21:40:50 -03:00
2019-05-26 22:34:13 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _GNSS_MODE2
2016-02-02 03:58:33 -04:00
// @DisplayName: GNSS system configuration
// @Description: Bitmask for what GNSS system to use on the second GPS (all unchecked or zero to leave GPS as configured)
2021-04-28 06:00:32 -03:00
// @Bitmask: 0:GPS,1:SBAS,2:Galileo,3:Beidou,4:IMES,5:QZSS,6:GLONASS
2016-02-02 03:58:33 -04:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _GNSS_MODE2 " , 12 , AP_GPS , _gnss_mode [ 1 ] , 0 ) ,
2019-05-26 22:34:13 -03:00
# endif
2016-02-02 03:58:33 -04:00
2020-12-30 11:03:15 -04:00
// @Param: _AUTO_CONFIG
2016-02-02 03:58:33 -04:00
// @DisplayName: Automatic GPS configuration
// @Description: Controls if the autopilot should automatically configure the GPS based on the parameters and default settings
2021-12-13 14:10:05 -04:00
// @Values: 0:Disables automatic configuration,1:Enable automatic configuration for Serial GPSes only,2:Enable automatic configuration for DroneCAN as well
2016-02-02 03:58:33 -04:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _AUTO_CONFIG " , 13 , AP_GPS , _auto_config , 1 ) ,
2016-02-02 03:58:33 -04:00
2020-12-30 11:03:15 -04:00
// @Param: _RATE_MS
2016-10-01 04:42:47 -03:00
// @DisplayName: GPS update rate in milliseconds
2020-11-25 19:54:27 -04:00
// @Description: Controls how often the GPS should provide a position update. Lowering below 5Hz(default) is not allowed. Raising the rate above 5Hz usually provides little benefit and for some GPS (eg Ublox M9N) can severely impact performance.
2017-05-02 10:43:10 -03:00
// @Units: ms
2016-10-01 04:42:47 -03:00
// @Values: 100:10Hz,125:8Hz,200:5Hz
2017-03-10 02:46:16 -04:00
// @Range: 50 200
2016-10-01 04:42:47 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _RATE_MS " , 14 , AP_GPS , _rate_ms [ 0 ] , 200 ) ,
2016-10-01 04:42:47 -03:00
2019-05-26 22:34:13 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _RATE_MS2
2016-10-01 04:42:47 -03:00
// @DisplayName: GPS 2 update rate in milliseconds
2020-11-25 19:54:27 -04:00
// @Description: Controls how often the GPS should provide a position update. Lowering below 5Hz(default) is not allowed. Raising the rate above 5Hz usually provides little benefit and for some GPS (eg Ublox M9N) can severely impact performance.
2017-05-02 10:43:10 -03:00
// @Units: ms
2016-10-01 04:42:47 -03:00
// @Values: 100:10Hz,125:8Hz,200:5Hz
2017-03-10 02:46:16 -04:00
// @Range: 50 200
2016-10-01 04:42:47 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _RATE_MS2 " , 15 , AP_GPS , _rate_ms [ 1 ] , 200 ) ,
2019-05-26 22:34:13 -03:00
# endif
2016-10-01 04:42:47 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _POS1_X
2016-10-07 19:04:14 -03:00
// @DisplayName: Antenna X position offset
2016-10-18 19:33:07 -03:00
// @Description: X position of the first GPS antenna in body frame. Positive X is forward of the origin. Use antenna phase centroid location if provided by the manufacturer.
2016-10-07 19:04:14 -03:00
// @Units: m
2020-01-30 00:30:35 -04:00
// @Range: -5 5
// @Increment: 0.01
2016-10-07 19:04:14 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
// @Param: _POS1_Y
2016-10-07 19:04:14 -03:00
// @DisplayName: Antenna Y position offset
2016-10-18 19:33:07 -03:00
// @Description: Y position of the first GPS antenna in body frame. Positive Y is to the right of the origin. Use antenna phase centroid location if provided by the manufacturer.
2016-10-07 19:04:14 -03:00
// @Units: m
2020-01-30 00:30:35 -04:00
// @Range: -5 5
// @Increment: 0.01
2016-10-07 19:04:14 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
// @Param: _POS1_Z
2016-10-07 19:04:14 -03:00
// @DisplayName: Antenna Z position offset
2016-10-18 19:33:07 -03:00
// @Description: Z position of the first GPS antenna in body frame. Positive Z is down from the origin. Use antenna phase centroid location if provided by the manufacturer.
2016-10-07 19:04:14 -03:00
// @Units: m
2020-01-30 00:30:35 -04:00
// @Range: -5 5
// @Increment: 0.01
2016-10-07 19:04:14 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _POS1 " , 16 , AP_GPS , _antenna_offset [ 0 ] , 0.0f ) ,
2016-10-07 19:04:14 -03:00
2020-01-30 00:30:35 -04:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _POS2_X
2016-10-07 19:04:14 -03:00
// @DisplayName: Antenna X position offset
2016-10-18 19:33:07 -03:00
// @Description: X position of the second GPS antenna in body frame. Positive X is forward of the origin. Use antenna phase centroid location if provided by the manufacturer.
2016-10-07 19:04:14 -03:00
// @Units: m
2020-01-30 00:30:35 -04:00
// @Range: -5 5
// @Increment: 0.01
2016-10-07 19:04:14 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
// @Param: _POS2_Y
2016-10-07 19:04:14 -03:00
// @DisplayName: Antenna Y position offset
2016-10-18 19:33:07 -03:00
// @Description: Y position of the second GPS antenna in body frame. Positive Y is to the right of the origin. Use antenna phase centroid location if provided by the manufacturer.
2016-10-07 19:04:14 -03:00
// @Units: m
2020-01-30 00:30:35 -04:00
// @Range: -5 5
// @Increment: 0.01
2016-10-07 19:04:14 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
// @Param: _POS2_Z
2016-10-07 19:04:14 -03:00
// @DisplayName: Antenna Z position offset
2016-10-18 19:33:07 -03:00
// @Description: Z position of the second GPS antenna in body frame. Positive Z is down from the origin. Use antenna phase centroid location if provided by the manufacturer.
2016-10-07 19:04:14 -03:00
// @Units: m
2020-01-30 00:30:35 -04:00
// @Range: -5 5
// @Increment: 0.01
2016-10-07 19:04:14 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _POS2 " , 17 , AP_GPS , _antenna_offset [ 1 ] , 0.0f ) ,
2019-05-26 22:34:13 -03:00
# endif
2016-10-07 19:04:14 -03:00
2020-12-30 11:03:15 -04:00
// @Param: _DELAY_MS
2016-12-28 18:20:06 -04:00
// @DisplayName: GPS delay in milliseconds
// @Description: Controls the amount of GPS measurement delay that the autopilot compensates for. Set to zero to use the default delay for the detected GPS type.
2017-05-02 10:43:10 -03:00
// @Units: ms
2016-12-28 18:20:06 -04:00
// @Range: 0 250
// @User: Advanced
2017-05-24 14:42:03 -03:00
// @RebootRequired: True
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _DELAY_MS " , 18 , AP_GPS , _delay_ms [ 0 ] , 0 ) ,
2016-12-28 18:20:06 -04:00
2019-05-26 22:34:13 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _DELAY_MS2
2016-12-28 18:20:06 -04:00
// @DisplayName: GPS 2 delay in milliseconds
// @Description: Controls the amount of GPS measurement delay that the autopilot compensates for. Set to zero to use the default delay for the detected GPS type.
2017-05-02 10:43:10 -03:00
// @Units: ms
2016-12-28 18:20:06 -04:00
// @Range: 0 250
// @User: Advanced
2017-05-24 14:42:03 -03:00
// @RebootRequired: True
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _DELAY_MS2 " , 19 , AP_GPS , _delay_ms [ 1 ] , 0 ) ,
2019-05-26 22:34:13 -03:00
# endif
2017-01-29 19:02:57 -04:00
2019-05-26 22:34:13 -03:00
# if defined(GPS_BLENDED_INSTANCE)
2020-12-30 11:03:15 -04:00
// @Param: _BLEND_MASK
2017-01-29 19:02:57 -04:00
// @DisplayName: Multi GPS Blending Mask
2020-06-23 23:07:35 -03:00
// @Description: Determines which of the accuracy measures Horizontal position, Vertical Position and Speed are used to calculate the weighting on each GPS receiver when soft switching has been selected by setting GPS_AUTO_SWITCH to 2(Blend)
2017-01-29 19:02:57 -04:00
// @Bitmask: 0:Horiz Pos,1:Vert Pos,2:Speed
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _BLEND_MASK " , 20 , AP_GPS , _blend_mask , 5 ) ,
2017-01-29 19:02:57 -04:00
2020-12-30 11:03:15 -04:00
// @Param: _BLEND_TC
2017-01-29 19:02:57 -04:00
// @DisplayName: Blending time constant
// @Description: Controls the slowest time constant applied to the calculation of GPS position and height offsets used to adjust different GPS receivers for steady state position differences.
2017-05-02 10:43:10 -03:00
// @Units: s
2017-01-29 19:02:57 -04:00
// @Range: 5.0 30.0
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _BLEND_TC " , 21 , AP_GPS , _blend_tc , 10.0f ) ,
2019-05-26 22:34:13 -03:00
# endif
2017-01-29 19:02:57 -04:00
2020-09-29 16:47:37 -03:00
# if GPS_MOVING_BASELINE
2020-12-30 11:03:15 -04:00
// @Param: _DRV_OPTIONS
2020-04-03 00:54:15 -03:00
// @DisplayName: driver options
// @Description: Additional backend specific options
2021-07-26 07:47:06 -03:00
// @Bitmask: 0:Use UART2 for moving baseline on ublox,1:Use base station for GPS yaw on SBF,2:Use baudrate 115200,3:Use dedicated CAN port b/w GPSes for moving baseline
2020-04-03 00:54:15 -03:00
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _DRV_OPTIONS " , 22 , AP_GPS , _driver_options , 0 ) ,
2020-04-03 00:54:15 -03:00
# endif
2022-01-09 19:15:32 -04:00
# if AP_GPS_SBF_ENABLED
2020-12-30 11:03:15 -04:00
// @Param: _COM_PORT
2020-06-13 17:23:44 -03:00
// @DisplayName: GPS physical COM port
// @Description: The physical COM port on the connected device, currently only applies to SBF GPS
// @Range: 0 10
// @Increment: 1
// @User: Advanced
// @RebootRequired: True
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _COM_PORT " , 23 , AP_GPS , _com_port [ 0 ] , HAL_GPS_COM_PORT_DEFAULT ) ,
2020-06-13 17:23:44 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _COM_PORT2
2020-06-13 17:23:44 -03:00
// @DisplayName: GPS physical COM port
// @Description: The physical COM port on the connected device, currently only applies to SBF GPS
// @Range: 0 10
// @Increment: 1
// @User: Advanced
// @RebootRequired: True
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _COM_PORT2 " , 24 , AP_GPS , _com_port [ 1 ] , HAL_GPS_COM_PORT_DEFAULT ) ,
2020-06-13 17:23:44 -03:00
# endif
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_SBF_ENABLED
2020-06-13 17:23:44 -03:00
2020-09-29 16:47:37 -03:00
# if GPS_MOVING_BASELINE
2020-12-30 11:03:15 -04:00
// @Group: _MB1_
2020-09-29 16:47:37 -03:00
// @Path: MovingBase.cpp
2020-12-30 11:03:15 -04:00
AP_SUBGROUPINFO ( mb_params [ 0 ] , " _MB1_ " , 25 , AP_GPS , MovingBase ) ,
2020-09-29 16:47:37 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Group: _MB2_
2020-09-29 16:47:37 -03:00
// @Path: MovingBase.cpp
2020-12-30 11:03:15 -04:00
AP_SUBGROUPINFO ( mb_params [ 1 ] , " _MB2_ " , 26 , AP_GPS , MovingBase ) ,
2020-09-29 16:47:37 -03:00
# endif // GPS_MAX_RECEIVERS > 1
# endif // GPS_MOVING_BASELINE
2020-10-29 19:54:05 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-30 11:03:15 -04:00
// @Param: _PRIMARY
2020-10-29 19:54:05 -03:00
// @DisplayName: Primary GPS
// @Description: This GPS will be used when GPS_AUTO_SWITCH is 0 and used preferentially with GPS_AUTO_SWITCH = 4.
// @Increment: 1
// @Values: 0:FirstGPS, 1:SecondGPS
// @User: Advanced
2020-12-30 11:03:15 -04:00
AP_GROUPINFO ( " _PRIMARY " , 27 , AP_GPS , _primary , 0 ) ,
2020-10-29 19:54:05 -03:00
# endif
2021-09-30 02:16:55 -03:00
# if HAL_ENABLE_LIBUAVCAN_DRIVERS
2020-12-28 10:30:41 -04:00
// @Param: _CAN_NODEID1
// @DisplayName: GPS Node ID 1
// @Description: GPS Node id for discovered first.
// @ReadOnly: True
// @User: Advanced
AP_GROUPINFO ( " _CAN_NODEID1 " , 28 , AP_GPS , _node_id [ 0 ] , 0 ) ,
2021-09-30 02:16:55 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-28 10:30:41 -04:00
// @Param: _CAN_NODEID2
// @DisplayName: GPS Node ID 2
// @Description: GPS Node id for discovered second.
// @ReadOnly: True
// @User: Advanced
AP_GROUPINFO ( " _CAN_NODEID2 " , 29 , AP_GPS , _node_id [ 1 ] , 0 ) ,
2021-09-30 02:16:55 -03:00
# endif // GPS_MAX_RECEIVERS > 1
2020-12-28 10:30:41 -04:00
// @Param: 1_CAN_OVRIDE
2021-12-13 14:10:05 -04:00
// @DisplayName: First DroneCAN GPS NODE ID
2020-12-28 10:30:41 -04:00
// @Description: GPS Node id for first GPS. If 0 the gps will be automatically selected on first come basis.
// @User: Advanced
AP_GROUPINFO ( " 1_CAN_OVRIDE " , 30 , AP_GPS , _override_node_id [ 0 ] , 0 ) ,
2021-09-30 02:16:55 -03:00
# if GPS_MAX_RECEIVERS > 1
2020-12-28 10:30:41 -04:00
// @Param: 2_CAN_OVRIDE
2021-12-13 14:10:05 -04:00
// @DisplayName: Second DroneCAN GPS NODE ID
2020-12-28 10:30:41 -04:00
// @Description: GPS Node id for second GPS. If 0 the gps will be automatically selected on first come basis.
// @User: Advanced
AP_GROUPINFO ( " 2_CAN_OVRIDE " , 31 , AP_GPS , _override_node_id [ 1 ] , 0 ) ,
2021-09-30 02:16:55 -03:00
# endif // GPS_MAX_RECEIVERS > 1
# endif // HAL_ENABLE_LIBUAVCAN_DRIVERS
2020-12-28 10:30:41 -04:00
2014-03-28 16:52:27 -03:00
AP_GROUPEND
} ;
2017-03-08 05:56:52 -04:00
// constructor
AP_GPS : : AP_GPS ( )
{
2017-05-29 19:02:25 -03:00
static_assert ( ( sizeof ( _initialisation_blob ) * ( CHAR_BIT + 2 ) ) < ( 4800 * GPS_BAUD_TIME_MS * 1e-3 ) ,
2019-01-14 16:20:18 -04:00
" GPS initilisation blob is too large to be completely sent before the baud rate changes " ) ;
2017-05-29 19:02:25 -03:00
2017-03-08 05:56:52 -04:00
AP_Param : : setup_object_defaults ( this , var_info ) ;
2017-10-24 22:30:10 -03:00
if ( _singleton ! = nullptr ) {
AP_HAL : : panic ( " AP_GPS must be singleton " ) ;
}
_singleton = this ;
2017-03-08 05:56:52 -04:00
}
2019-10-23 22:46:26 -03:00
// return true if a specific type of GPS uses a UART
bool AP_GPS : : needs_uart ( GPS_Type type ) const
{
switch ( type ) {
case GPS_TYPE_NONE :
case GPS_TYPE_HIL :
case GPS_TYPE_UAVCAN :
2021-08-11 07:13:55 -03:00
case GPS_TYPE_UAVCAN_RTK_BASE :
case GPS_TYPE_UAVCAN_RTK_ROVER :
2019-10-23 22:46:26 -03:00
case GPS_TYPE_MAV :
2020-09-03 06:15:35 -03:00
case GPS_TYPE_MSP :
2020-12-28 18:42:14 -04:00
case GPS_TYPE_EXTERNAL_AHRS :
2019-10-23 22:46:26 -03:00
return false ;
default :
break ;
}
return true ;
}
2014-03-28 16:52:27 -03:00
/// Startup initialisation.
2017-06-27 05:12:45 -03:00
void AP_GPS : : init ( const AP_SerialManager & serial_manager )
2014-03-28 16:52:27 -03:00
{
2020-10-29 19:54:05 -03:00
// Set new primary param based on old auto_switch use second option
if ( ( _auto_switch . get ( ) = = 3 ) & & ! _primary . configured ( ) ) {
_primary . set_and_save ( 1 ) ;
_auto_switch . set_and_save ( 0 ) ;
}
2015-01-19 09:36:35 -04:00
// search for serial ports with gps protocol
2019-10-23 22:46:26 -03:00
uint8_t uart_idx = 0 ;
2019-05-26 22:34:13 -03:00
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2019-10-23 22:46:26 -03:00
if ( needs_uart ( ( GPS_Type ) _type [ i ] . get ( ) ) ) {
_port [ i ] = serial_manager . find_serial ( AP_SerialManager : : SerialProtocol_GPS , uart_idx ) ;
uart_idx + + ;
}
2019-05-26 22:34:13 -03:00
}
2015-05-21 18:07:59 -03:00
_last_instance_swap_ms = 0 ;
2017-01-29 19:02:57 -04:00
// Initialise class variables used to do GPS blending
_omega_lpf = 1.0f / constrain_float ( _blend_tc , 5.0f , 30.0f ) ;
2017-07-18 15:07:41 -03:00
// prep the state instance fields
for ( uint8_t i = 0 ; i < GPS_MAX_INSTANCES ; i + + ) {
state [ i ] . instance = i ;
}
2017-03-07 23:33:31 -04:00
// sanity check update rate
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( _rate_ms [ i ] < = 0 | | _rate_ms [ i ] > GPS_MAX_RATE_MS ) {
_rate_ms [ i ] = GPS_MAX_RATE_MS ;
}
}
2014-03-28 16:52:27 -03:00
}
2017-03-08 05:56:52 -04:00
// return number of active GPS sensors. Note that if the first GPS
// is not present but the 2nd is then we return 2. Note that a blended
// GPS solution is treated as an additional sensor.
uint8_t AP_GPS : : num_sensors ( void ) const
{
if ( ! _output_is_blended ) {
return num_instances ;
} else {
return num_instances + 1 ;
}
}
2014-03-28 16:52:27 -03:00
2017-03-08 05:56:52 -04:00
bool AP_GPS : : speed_accuracy ( uint8_t instance , float & sacc ) const
{
if ( state [ instance ] . have_speed_accuracy ) {
sacc = state [ instance ] . speed_accuracy ;
return true ;
}
return false ;
}
bool AP_GPS : : horizontal_accuracy ( uint8_t instance , float & hacc ) const
{
if ( state [ instance ] . have_horizontal_accuracy ) {
hacc = state [ instance ] . horizontal_accuracy ;
return true ;
}
return false ;
}
bool AP_GPS : : vertical_accuracy ( uint8_t instance , float & vacc ) const
{
if ( state [ instance ] . have_vertical_accuracy ) {
vacc = state [ instance ] . vertical_accuracy ;
return true ;
}
return false ;
}
2014-03-28 16:52:27 -03:00
2017-03-08 05:59:27 -04:00
/**
convert GPS week and milliseconds to unix epoch in milliseconds
*/
2022-02-02 03:24:03 -04:00
uint64_t AP_GPS : : istate_time_to_epoch_ms ( uint16_t gps_week , uint32_t gps_ms )
2017-03-08 05:59:27 -04:00
{
2017-05-05 07:09:29 -03:00
uint64_t fix_time_ms = UNIX_OFFSET_MSEC + gps_week * AP_MSEC_PER_WEEK + gps_ms ;
2017-03-08 05:59:27 -04:00
return fix_time_ms ;
}
/**
calculate current time since the unix epoch in microseconds
*/
2017-07-25 23:47:00 -03:00
uint64_t AP_GPS : : time_epoch_usec ( uint8_t instance ) const
2017-03-08 05:59:27 -04:00
{
const GPS_State & istate = state [ instance ] ;
2022-01-16 08:15:59 -04:00
if ( ( istate . last_gps_time_ms = = 0 & & istate . last_corrected_gps_time_us = = 0 ) | | istate . time_week = = 0 ) {
2017-03-08 05:59:27 -04:00
return 0 ;
}
2022-01-16 08:15:59 -04:00
uint64_t fix_time_ms ;
// add in the time since the last fix message
if ( istate . last_corrected_gps_time_us ! = 0 ) {
2022-02-02 03:24:03 -04:00
fix_time_ms = istate_time_to_epoch_ms ( istate . time_week , drivers [ instance ] - > get_last_itow_ms ( ) ) ;
2022-01-16 08:15:59 -04:00
return ( fix_time_ms * 1000ULL ) + ( AP_HAL : : micros64 ( ) - istate . last_corrected_gps_time_us ) ;
} else {
2022-02-02 03:24:03 -04:00
fix_time_ms = istate_time_to_epoch_ms ( istate . time_week , istate . time_week_ms ) ;
2022-01-16 08:15:59 -04:00
return ( fix_time_ms + ( AP_HAL : : millis ( ) - istate . last_gps_time_ms ) ) * 1000ULL ;
}
}
/**
calculate last message time since the unix epoch in microseconds
*/
uint64_t AP_GPS : : last_message_epoch_usec ( uint8_t instance ) const
{
const GPS_State & istate = state [ instance ] ;
if ( istate . time_week = = 0 ) {
return 0 ;
}
2022-02-02 03:24:03 -04:00
return istate_time_to_epoch_ms ( istate . time_week , drivers [ instance ] - > get_last_itow_ms ( ) ) * 1000ULL ;
2017-03-08 05:59:27 -04:00
}
2014-03-31 06:48:22 -03:00
/*
send some more initialisation string bytes if there is room in the
UART transmit buffer
*/
2015-10-26 08:25:44 -03:00
void AP_GPS : : send_blob_start ( uint8_t instance , const char * _blob , uint16_t size )
2014-03-31 06:48:22 -03:00
{
initblob_state [ instance ] . blob = _blob ;
initblob_state [ instance ] . remaining = size ;
}
/*
send some more initialisation string bytes if there is room in the
UART transmit buffer
*/
void AP_GPS : : send_blob_update ( uint8_t instance )
{
2015-01-19 09:36:35 -04:00
// exit immediately if no uart for this instance
2016-10-30 02:24:21 -03:00
if ( _port [ instance ] = = nullptr ) {
2015-01-19 09:36:35 -04:00
return ;
}
2019-09-10 19:25:54 -03:00
if ( initblob_state [ instance ] . remaining = = 0 ) {
return ;
2014-03-31 06:48:22 -03:00
}
2019-09-10 19:25:54 -03:00
// see if we can write some more of the initialisation blob
const uint16_t n = MIN ( _port [ instance ] - > txspace ( ) ,
initblob_state [ instance ] . remaining ) ;
const size_t written = _port [ instance ] - > write ( ( const uint8_t * ) initblob_state [ instance ] . blob , n ) ;
initblob_state [ instance ] . blob + = written ;
initblob_state [ instance ] . remaining - = written ;
2014-03-31 06:48:22 -03:00
}
2014-03-28 16:52:27 -03:00
/*
run detection step for one GPS instance . If this finds a GPS then it
will fill in drivers [ instance ] and change state [ instance ] . status
from NO_GPS to NO_FIX .
*/
2017-04-02 11:56:26 -03:00
void AP_GPS : : detect_instance ( uint8_t instance )
2014-03-28 16:52:27 -03:00
{
2016-10-30 02:24:21 -03:00
AP_GPS_Backend * new_gps = nullptr ;
2014-03-28 16:52:27 -03:00
struct detect_state * dstate = & detect_state [ instance ] ;
2017-09-23 20:53:40 -03:00
const uint32_t now = AP_HAL : : millis ( ) ;
2014-03-28 16:52:27 -03:00
2017-07-17 21:57:36 -03:00
state [ instance ] . status = NO_GPS ;
state [ instance ] . hdop = GPS_UNKNOWN_DOP ;
state [ instance ] . vdop = GPS_UNKNOWN_DOP ;
2018-07-18 06:27:15 -03:00
2016-12-15 09:12:54 -04:00
switch ( _type [ instance ] ) {
2016-10-29 04:45:00 -03:00
// user has to explicitly set the MAV type, do not use AUTO
// do not try to detect the MAV type, assume it's there
2018-07-18 06:27:15 -03:00
case GPS_TYPE_MAV :
2019-05-26 22:34:13 -03:00
# ifndef HAL_BUILD_AP_PERIPH
2022-01-09 19:15:32 -04:00
# if AP_GPS_MAV_ENABLED
2017-05-30 03:47:50 -03:00
dstate - > auto_detected_baud = false ; // specified, not detected
2016-10-30 02:24:21 -03:00
new_gps = new AP_GPS_MAV ( * this , state [ instance ] , nullptr ) ;
2016-10-29 04:45:00 -03:00
goto found_gps ;
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_MAV_ENABLED
2019-05-26 22:34:13 -03:00
# endif
2016-12-15 09:12:54 -04:00
break ;
2017-04-02 11:56:26 -03:00
// user has to explicitly set the UAVCAN type, do not use AUTO
2018-07-18 06:27:15 -03:00
case GPS_TYPE_UAVCAN :
2021-08-11 07:13:55 -03:00
case GPS_TYPE_UAVCAN_RTK_BASE :
case GPS_TYPE_UAVCAN_RTK_ROVER :
2020-05-31 09:06:29 -03:00
# if HAL_ENABLE_LIBUAVCAN_DRIVERS
2017-05-30 03:47:50 -03:00
dstate - > auto_detected_baud = false ; // specified, not detected
2018-07-18 06:27:15 -03:00
new_gps = AP_GPS_UAVCAN : : probe ( * this , state [ instance ] ) ;
goto found_gps ;
2017-04-02 11:56:26 -03:00
# endif
2018-07-18 06:27:15 -03:00
return ; // We don't do anything here if UAVCAN is not supported
2020-09-03 06:15:35 -03:00
# if HAL_MSP_GPS_ENABLED
case GPS_TYPE_MSP :
dstate - > auto_detected_baud = false ; // specified, not detected
new_gps = new AP_GPS_MSP ( * this , state [ instance ] , nullptr ) ;
goto found_gps ;
2020-12-28 18:42:14 -04:00
# endif
# if HAL_EXTERNAL_AHRS_ENABLED
case GPS_TYPE_EXTERNAL_AHRS :
dstate - > auto_detected_baud = false ; // specified, not detected
new_gps = new AP_GPS_ExternalAHRS ( * this , state [ instance ] , nullptr ) ;
goto found_gps ;
2020-09-03 06:15:35 -03:00
# endif
2016-12-15 09:12:54 -04:00
default :
break ;
2016-10-29 04:45:00 -03:00
}
2016-10-30 02:24:21 -03:00
if ( _port [ instance ] = = nullptr ) {
2014-03-28 16:52:27 -03:00
// UART not available
return ;
}
2017-05-30 03:47:50 -03:00
// all remaining drivers automatically cycle through baud rates to detect
// the correct baud rate, and should have the selected baud broadcast
dstate - > auto_detected_baud = true ;
2020-05-20 19:56:29 -03:00
// don't build the less common GPS drivers on F1 AP_Periph
# if !defined(HAL_BUILD_AP_PERIPH) || !defined(STM32F1)
2016-12-15 09:12:54 -04:00
switch ( _type [ instance ] ) {
2022-01-09 19:15:32 -04:00
# if AP_GPS_SBF_ENABLED
2016-12-15 09:12:54 -04:00
// by default the sbf/trimble gps outputs no data on its port, until configured.
case GPS_TYPE_SBF :
new_gps = new AP_GPS_SBF ( * this , state [ instance ] , _port [ instance ] ) ;
break ;
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_SBF_ENABLED
# if AP_GPS_GSOF_ENABLED
2016-12-15 09:12:54 -04:00
case GPS_TYPE_GSOF :
new_gps = new AP_GPS_GSOF ( * this , state [ instance ] , _port [ instance ] ) ;
break ;
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_GSOF_ENABLED
# if AP_GPS_NOVA_ENABLED
2016-12-15 09:12:54 -04:00
case GPS_TYPE_NOVA :
new_gps = new AP_GPS_NOVA ( * this , state [ instance ] , _port [ instance ] ) ;
break ;
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_NOVA_ENABLED
2022-01-15 00:26:06 -04:00
# if HAL_SIM_GPS_ENABLED
case GPS_TYPE_SITL :
new_gps = new AP_GPS_SITL ( * this , state [ instance ] , _port [ instance ] ) ;
break ;
# endif // HAL_SIM_GPS_ENABLED
2016-12-15 09:12:54 -04:00
default :
break ;
}
2019-05-26 22:34:13 -03:00
# endif // HAL_BUILD_AP_PERIPH
2015-11-03 09:46:46 -04:00
2015-07-14 04:07:29 -03:00
if ( now - dstate - > last_baud_change_ms > GPS_BAUD_TIME_MS ) {
2014-03-28 16:52:27 -03:00
// try the next baud rate
2016-09-24 03:16:40 -03:00
// incrementing like this will skip the first element in array of bauds
// this is okay, and relied upon
dstate - > current_baud + + ;
if ( dstate - > current_baud = = ARRAY_SIZE ( _baudrates ) ) {
dstate - > current_baud = 0 ;
}
uint32_t baudrate = _baudrates [ dstate - > current_baud ] ;
_port [ instance ] - > begin ( baudrate ) ;
_port [ instance ] - > set_flow_control ( AP_HAL : : UARTDriver : : FLOW_CONTROL_DISABLE ) ;
dstate - > last_baud_change_ms = now ;
2016-09-06 04:16:01 -03:00
2021-07-16 13:21:01 -03:00
if ( _auto_config > = GPS_AUTO_CONFIG_ENABLE_SERIAL_ONLY & & new_gps = = nullptr ) {
2022-01-09 19:15:32 -04:00
if ( _type [ instance ] = = GPS_TYPE_UBLOX & & ( _driver_options & AP_GPS_Backend : : DriverOptions : : UBX_Use115200 ) ) {
static const char blob [ ] = UBLOX_SET_BINARY_115200 ;
send_blob_start ( instance , blob , sizeof ( blob ) ) ;
2021-02-07 20:51:19 -04:00
} else if ( ( _type [ instance ] = = GPS_TYPE_UBLOX_RTK_BASE | |
_type [ instance ] = = GPS_TYPE_UBLOX_RTK_ROVER ) & &
( ( _driver_options . get ( ) & AP_GPS_Backend : : DriverOptions : : UBX_MBUseUart2 ) = = 0 ) ) {
// we use 460800 when doing moving baseline as we need
// more bandwidth. We don't do this if using UART2, as
// in that case the RTCMv3 data doesn't go over the
// link to the flight controller
2019-11-16 00:26:28 -04:00
static const char blob [ ] = UBLOX_SET_BINARY_460800 ;
send_blob_start ( instance , blob , sizeof ( blob ) ) ;
2022-01-09 19:15:32 -04:00
# if AP_GPS_NMEA_ENABLED
} else if ( _type [ instance ] = = GPS_TYPE_HEMI ) {
send_blob_start ( instance , AP_GPS_NMEA_HEMISPHERE_INIT_STRING , strlen ( AP_GPS_NMEA_HEMISPHERE_INIT_STRING ) ) ;
# endif // AP_GPS_NMEA_ENABLED
2017-05-28 01:11:31 -03:00
} else {
send_blob_start ( instance , _initialisation_blob , sizeof ( _initialisation_blob ) ) ;
}
2016-08-08 16:23:17 -03:00
}
2014-03-28 16:52:27 -03:00
}
2021-07-16 13:21:01 -03:00
if ( _auto_config > = GPS_AUTO_CONFIG_ENABLE_SERIAL_ONLY & & new_gps = = nullptr ) {
2016-08-08 16:23:17 -03:00
send_blob_update ( instance ) ;
}
2014-03-28 16:52:27 -03:00
2015-07-14 04:07:29 -03:00
while ( initblob_state [ instance ] . remaining = = 0 & & _port [ instance ] - > available ( ) > 0
2017-04-02 11:56:26 -03:00
& & new_gps = = nullptr ) {
2015-01-19 09:36:35 -04:00
uint8_t data = _port [ instance ] - > read ( ) ;
2014-03-28 16:52:27 -03:00
/*
running a uBlox at less than 38400 will lead to packet
corruption , as we can ' t receive the packets in the 200 ms
window for 5 Hz fixes . The NMEA startup message should force
2020-05-21 23:13:08 -03:00
the uBlox into 230400 no matter what rate it is configured
2014-03-28 16:52:27 -03:00
for .
*/
2021-02-07 20:51:19 -04:00
2019-11-16 00:26:28 -04:00
if ( ( _type [ instance ] = = GPS_TYPE_AUTO | |
_type [ instance ] = = GPS_TYPE_UBLOX ) & &
2017-01-25 15:26:55 -04:00
( ( ! _auto_config & & _baudrates [ dstate - > current_baud ] > = 38400 ) | |
2020-12-07 18:19:24 -04:00
( _baudrates [ dstate - > current_baud ] > = 115200 & & ( _driver_options & AP_GPS_Backend : : DriverOptions : : UBX_Use115200 ) ) | |
2020-05-21 23:13:08 -03:00
_baudrates [ dstate - > current_baud ] = = 230400 ) & &
2014-03-28 16:52:27 -03:00
AP_GPS_UBLOX : : _detect ( dstate - > ublox_detect_state , data ) ) {
2019-11-16 00:26:28 -04:00
new_gps = new AP_GPS_UBLOX ( * this , state [ instance ] , _port [ instance ] , GPS_ROLE_NORMAL ) ;
}
2021-02-07 20:51:19 -04:00
const uint32_t ublox_mb_required_baud = ( _driver_options . get ( ) & AP_GPS_Backend : : DriverOptions : : UBX_MBUseUart2 ) ? 230400 : 460800 ;
2019-11-16 00:26:28 -04:00
if ( ( _type [ instance ] = = GPS_TYPE_UBLOX_RTK_BASE | |
_type [ instance ] = = GPS_TYPE_UBLOX_RTK_ROVER ) & &
2021-02-07 20:51:19 -04:00
_baudrates [ dstate - > current_baud ] = = ublox_mb_required_baud & &
2019-11-16 00:26:28 -04:00
AP_GPS_UBLOX : : _detect ( dstate - > ublox_detect_state , data ) ) {
GPS_Role role ;
if ( _type [ instance ] = = GPS_TYPE_UBLOX_RTK_BASE ) {
role = GPS_ROLE_MB_BASE ;
} else {
role = GPS_ROLE_MB_ROVER ;
}
new_gps = new AP_GPS_UBLOX ( * this , state [ instance ] , _port [ instance ] , role ) ;
2017-04-02 11:56:26 -03:00
}
2019-05-26 22:34:13 -03:00
# ifndef HAL_BUILD_AP_PERIPH
2022-01-09 19:15:32 -04:00
# if AP_GPS_SBP2_ENABLED
2017-03-27 22:13:31 -03:00
else if ( ( _type [ instance ] = = GPS_TYPE_AUTO | | _type [ instance ] = = GPS_TYPE_SBP ) & &
AP_GPS_SBP2 : : _detect ( dstate - > sbp2_detect_state , data ) ) {
new_gps = new AP_GPS_SBP2 ( * this , state [ instance ] , _port [ instance ] ) ;
}
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_SBP2_ENABLED
# if AP_GPS_SBP_ENABLED
2014-04-04 20:05:54 -03:00
else if ( ( _type [ instance ] = = GPS_TYPE_AUTO | | _type [ instance ] = = GPS_TYPE_SBP ) & &
AP_GPS_SBP : : _detect ( dstate - > sbp_detect_state , data ) ) {
2015-01-19 09:36:35 -04:00
new_gps = new AP_GPS_SBP ( * this , state [ instance ] , _port [ instance ] ) ;
2014-04-04 20:05:54 -03:00
}
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_SBP_ENABLED
# if !HAL_MINIMIZE_FEATURES && AP_GPS_SIRF_ENABLED
2016-12-15 09:12:54 -04:00
else if ( ( _type [ instance ] = = GPS_TYPE_AUTO | | _type [ instance ] = = GPS_TYPE_SIRF ) & &
2014-03-28 16:52:27 -03:00
AP_GPS_SIRF : : _detect ( dstate - > sirf_detect_state , data ) ) {
2016-12-15 09:12:54 -04:00
new_gps = new AP_GPS_SIRF ( * this , state [ instance ] , _port [ instance ] ) ;
}
2017-01-27 00:57:34 -04:00
# endif
2022-01-09 19:15:32 -04:00
# if AP_GPS_ERB_ENABLED
2016-01-18 17:54:40 -04:00
else if ( ( _type [ instance ] = = GPS_TYPE_AUTO | | _type [ instance ] = = GPS_TYPE_ERB ) & &
AP_GPS_ERB : : _detect ( dstate - > erb_detect_state , data ) ) {
new_gps = new AP_GPS_ERB ( * this , state [ instance ] , _port [ instance ] ) ;
2022-01-09 19:15:32 -04:00
}
# endif // AP_GPS_ERB_ENABLED
# if AP_GPS_NMEA_ENABLED
else if ( ( _type [ instance ] = = GPS_TYPE_NMEA | |
2020-11-11 19:04:14 -04:00
_type [ instance ] = = GPS_TYPE_HEMI | |
_type [ instance ] = = GPS_TYPE_ALLYSTAR ) & &
2017-05-30 03:33:39 -03:00
AP_GPS_NMEA : : _detect ( dstate - > nmea_detect_state , data ) ) {
new_gps = new AP_GPS_NMEA ( * this , state [ instance ] , _port [ instance ] ) ;
2016-12-15 09:12:54 -04:00
}
2022-01-09 19:15:32 -04:00
# endif //AP_GPS_NMEA_ENABLED
2019-05-26 22:34:13 -03:00
# endif // HAL_BUILD_AP_PERIPH
if ( new_gps ) {
goto found_gps ;
}
2016-12-15 09:12:54 -04:00
}
2014-03-28 16:52:27 -03:00
2014-11-13 23:44:15 -04:00
found_gps :
2016-12-15 09:12:54 -04:00
if ( new_gps ! = nullptr ) {
2014-03-28 16:52:27 -03:00
state [ instance ] . status = NO_FIX ;
2014-03-31 08:13:55 -03:00
drivers [ instance ] = new_gps ;
timing [ instance ] . last_message_time_ms = now ;
2017-07-08 20:06:19 -03:00
timing [ instance ] . delta_time_ms = GPS_TIMEOUT_MS ;
2016-08-01 21:30:12 -03:00
new_gps - > broadcast_gps_type ( ) ;
2016-12-15 09:12:54 -04:00
}
2014-03-28 16:52:27 -03:00
}
2017-04-02 11:56:26 -03:00
AP_GPS : : GPS_Status AP_GPS : : highest_supported_status ( uint8_t instance ) const
2014-06-06 18:58:11 -03:00
{
2017-03-08 05:49:58 -04:00
if ( instance < GPS_MAX_RECEIVERS & & drivers [ instance ] ! = nullptr ) {
2014-06-06 18:58:11 -03:00
return drivers [ instance ] - > highest_supported_status ( ) ;
2016-12-15 09:12:54 -04:00
}
2014-06-06 18:58:11 -03:00
return AP_GPS : : GPS_OK_FIX_3D ;
}
2019-02-11 04:19:08 -04:00
bool AP_GPS : : should_log ( ) const
2018-04-11 09:16:23 -03:00
{
2021-05-19 23:34:13 -03:00
# if HAL_LOGGING_ENABLED
2019-02-11 04:19:08 -04:00
AP_Logger * logger = AP_Logger : : get_singleton ( ) ;
if ( logger = = nullptr ) {
2018-04-11 09:16:23 -03:00
return false ;
}
if ( _log_gps_bit = = ( uint32_t ) - 1 ) {
return false ;
}
2019-02-11 04:19:08 -04:00
if ( ! logger - > should_log ( _log_gps_bit ) ) {
2018-04-11 09:16:23 -03:00
return false ;
}
return true ;
2019-05-26 22:34:13 -03:00
# else
return false ;
# endif
2018-04-11 09:16:23 -03:00
}
2014-03-28 16:52:27 -03:00
/*
2014-03-31 06:48:22 -03:00
update one GPS instance . This should be called at 10 Hz or greater
2014-03-28 16:52:27 -03:00
*/
2017-04-02 11:56:26 -03:00
void AP_GPS : : update_instance ( uint8_t instance )
2014-03-28 16:52:27 -03:00
{
2014-04-01 03:25:15 -03:00
if ( _type [ instance ] = = GPS_TYPE_HIL ) {
// in HIL, leave info alone
return ;
}
2014-03-28 16:52:27 -03:00
if ( _type [ instance ] = = GPS_TYPE_NONE ) {
// not enabled
state [ instance ] . status = NO_GPS ;
2017-05-24 16:19:10 -03:00
state [ instance ] . hdop = GPS_UNKNOWN_DOP ;
state [ instance ] . vdop = GPS_UNKNOWN_DOP ;
2014-03-28 16:52:27 -03:00
return ;
}
2014-04-04 17:33:34 -03:00
if ( locked_ports & ( 1U < < instance ) ) {
// the port is locked by another driver
return ;
}
2019-10-09 01:30:39 -03:00
if ( drivers [ instance ] = = nullptr ) {
2014-03-28 16:52:27 -03:00
// we don't yet know the GPS type of this one, or it has timed
// out and needs to be re-initialised
detect_instance ( instance ) ;
return ;
}
2021-07-16 13:21:01 -03:00
if ( _auto_config > = GPS_AUTO_CONFIG_ENABLE_SERIAL_ONLY ) {
2016-08-08 16:23:17 -03:00
send_blob_update ( instance ) ;
}
2014-03-31 06:48:22 -03:00
2014-03-28 16:52:27 -03:00
// we have an active driver for this instance
bool result = drivers [ instance ] - > read ( ) ;
2018-05-15 22:16:01 -03:00
uint32_t tnow = AP_HAL : : millis ( ) ;
2014-03-28 16:52:27 -03:00
2016-02-02 03:58:33 -04:00
// if we did not get a message, and the idle timer of 2 seconds
2014-03-28 16:52:27 -03:00
// has expired, re-initialise the GPS. This will cause GPS
// detection to run again
2018-04-11 09:16:23 -03:00
bool data_should_be_logged = false ;
2014-03-28 16:52:27 -03:00
if ( ! result ) {
2017-07-08 20:06:19 -03:00
if ( tnow - timing [ instance ] . last_message_time_ms > GPS_TIMEOUT_MS ) {
2018-10-30 14:25:35 -03:00
memset ( ( void * ) & state [ instance ] , 0 , sizeof ( state [ instance ] ) ) ;
2017-09-23 20:55:02 -03:00
state [ instance ] . instance = instance ;
2017-05-24 16:19:10 -03:00
state [ instance ] . hdop = GPS_UNKNOWN_DOP ;
state [ instance ] . vdop = GPS_UNKNOWN_DOP ;
2014-04-01 17:48:36 -03:00
timing [ instance ] . last_message_time_ms = tnow ;
2017-07-08 20:06:19 -03:00
timing [ instance ] . delta_time_ms = GPS_TIMEOUT_MS ;
2020-07-27 09:53:56 -03:00
// do not try to detect again if type is MAV or UAVCAN
2021-08-11 07:13:55 -03:00
if ( _type [ instance ] = = GPS_TYPE_MAV | |
_type [ instance ] = = GPS_TYPE_UAVCAN | |
_type [ instance ] = = GPS_TYPE_UAVCAN_RTK_BASE | |
_type [ instance ] = = GPS_TYPE_UAVCAN_RTK_ROVER ) {
2018-02-14 20:59:14 -04:00
state [ instance ] . status = NO_FIX ;
} else {
// free the driver before we run the next detection, so we
// don't end up with two allocated at any time
delete drivers [ instance ] ;
drivers [ instance ] = nullptr ;
state [ instance ] . status = NO_GPS ;
}
2018-04-11 09:16:23 -03:00
// log this data as a "flag" that the GPS is no longer
// valid (see PR#8144)
data_should_be_logged = true ;
2014-03-28 16:52:27 -03:00
}
} else {
2022-01-18 06:42:57 -04:00
if ( state [ instance ] . corrected_timestamp_updated ) {
2018-05-15 22:16:01 -03:00
// set the timestamp for this messages based on
2022-01-18 06:42:57 -04:00
// set_uart_timestamp() or per specific transport in backend
// , if available
tnow = state [ instance ] . last_corrected_gps_time_us / 1000U ;
state [ instance ] . corrected_timestamp_updated = false ;
2018-05-15 22:16:01 -03:00
}
2017-07-08 20:06:19 -03:00
// delta will only be correct after parsing two messages
timing [ instance ] . delta_time_ms = tnow - timing [ instance ] . last_message_time_ms ;
2014-03-28 16:52:27 -03:00
timing [ instance ] . last_message_time_ms = tnow ;
2019-09-15 03:19:19 -03:00
// if GPS disabled for flight testing then don't update fix timing value
if ( state [ instance ] . status > = GPS_OK_FIX_2D & & ! _force_disable_gps ) {
2014-03-28 16:52:27 -03:00
timing [ instance ] . last_fix_time_ms = tnow ;
}
2018-04-11 09:16:23 -03:00
data_should_be_logged = true ;
}
2020-03-29 19:17:39 -03:00
# if GPS_MAX_RECEIVERS > 1
2019-11-16 00:26:28 -04:00
if ( drivers [ instance ] & & _type [ instance ] = = GPS_TYPE_UBLOX_RTK_BASE ) {
// see if a moving baseline base has some RTCMv3 data
// which we need to pass along to the rover
const uint8_t * rtcm_data ;
uint16_t rtcm_len ;
if ( drivers [ instance ] - > get_RTCMV3 ( rtcm_data , rtcm_len ) ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( i ! = instance & & _type [ i ] = = GPS_TYPE_UBLOX_RTK_ROVER ) {
// pass the data to the rover
inject_data ( i , rtcm_data , rtcm_len ) ;
drivers [ instance ] - > clear_RTCMV3 ( ) ;
break ;
}
}
}
}
2020-03-29 19:17:39 -03:00
# endif
2019-11-16 00:26:28 -04:00
2020-04-04 19:53:27 -03:00
if ( data_should_be_logged ) {
// keep count of delayed frames and average frame delay for health reporting
const uint16_t gps_max_delta_ms = 245 ; // 200 ms (5Hz) + 45 ms buffer
GPS_timing & t = timing [ instance ] ;
if ( t . delta_time_ms > gps_max_delta_ms ) {
t . delayed_count + + ;
} else {
t . delayed_count = 0 ;
}
if ( t . delta_time_ms < 2000 ) {
if ( t . average_delta_ms < = 0 ) {
t . average_delta_ms = t . delta_time_ms ;
} else {
2020-04-16 23:34:29 -03:00
t . average_delta_ms = 0.98f * t . average_delta_ms + 0.02f * t . delta_time_ms ;
2020-04-04 19:53:27 -03:00
}
}
}
2021-05-19 23:34:13 -03:00
# if HAL_LOGGING_ENABLED
2020-11-07 01:58:05 -04:00
if ( data_should_be_logged & & should_log ( ) ) {
2021-02-09 19:08:37 -04:00
Write_GPS ( instance ) ;
2014-03-28 16:52:27 -03:00
}
2021-05-19 23:34:13 -03:00
# endif
2018-04-10 01:48:30 -03:00
2021-05-19 23:34:13 -03:00
# ifndef HAL_BUILD_AP_PERIPH
2018-04-10 01:48:30 -03:00
if ( state [ instance ] . status > = GPS_OK_FIX_3D ) {
const uint64_t now = time_epoch_usec ( instance ) ;
2019-07-31 00:17:06 -03:00
if ( now ! = 0 ) {
AP : : rtc ( ) . set_utc_usec ( now , AP_RTC : : SOURCE_GPS ) ;
}
2018-04-10 01:48:30 -03:00
}
2019-05-26 22:34:13 -03:00
# else
( void ) data_should_be_logged ;
# endif
2014-03-28 16:52:27 -03:00
}
2021-08-11 07:13:55 -03:00
# if GPS_MOVING_BASELINE
void AP_GPS : : get_RelPosHeading ( uint32_t & timestamp , float & relPosHeading , float & relPosLength , float & relPosD , float & accHeading )
{
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( drivers [ i ] & & _type [ i ] = = GPS_TYPE_UBLOX_RTK_ROVER ) {
relPosHeading = state [ i ] . relPosHeading ;
relPosLength = state [ i ] . relPosLength ;
relPosD = state [ i ] . relPosD ;
accHeading = state [ i ] . accHeading ;
timestamp = state [ i ] . relposheading_ts ;
}
}
}
bool AP_GPS : : get_RTCMV3 ( const uint8_t * & bytes , uint16_t & len )
{
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( drivers [ i ] & & _type [ i ] = = GPS_TYPE_UBLOX_RTK_BASE ) {
return drivers [ i ] - > get_RTCMV3 ( bytes , len ) ;
}
}
return false ;
}
void AP_GPS : : clear_RTCMV3 ( )
{
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( drivers [ i ] & & _type [ i ] = = GPS_TYPE_UBLOX_RTK_BASE ) {
drivers [ i ] - > clear_RTCMV3 ( ) ;
}
}
}
/*
inject Moving Baseline Data messages .
*/
void AP_GPS : : inject_MBL_data ( uint8_t * data , uint16_t length )
{
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( _type [ i ] = = GPS_TYPE_UBLOX_RTK_ROVER ) {
// pass the data to the rover
inject_data ( i , data , length ) ;
break ;
}
}
}
# endif //#if GPS_MOVING_BASELINE
2014-03-28 16:52:27 -03:00
/*
update all GPS instances
*/
2017-04-02 11:56:26 -03:00
void AP_GPS : : update ( void )
2014-03-28 16:52:27 -03:00
{
2020-01-03 20:06:05 -04:00
WITH_SEMAPHORE ( rsem ) ;
2017-01-29 19:02:57 -04:00
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2014-03-28 16:52:27 -03:00
update_instance ( i ) ;
}
2014-03-31 16:14:19 -03:00
2017-01-29 19:02:57 -04:00
// calculate number of instances
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2019-10-09 01:30:39 -03:00
if ( drivers [ i ] ! = nullptr ) {
2014-04-03 18:32:34 -03:00
num_instances = i + 1 ;
}
2017-01-29 19:02:57 -04:00
}
2015-05-21 18:07:59 -03:00
2021-08-05 23:25:12 -03:00
# if GPS_MAX_RECEIVERS > 1
2021-07-30 00:18:05 -03:00
# if HAL_LOGGING_ENABLED
const uint8_t old_primary = primary_instance ;
# endif
2019-11-17 06:04:41 -04:00
update_primary ( ) ;
2021-07-30 00:18:05 -03:00
# if HAL_LOGGING_ENABLED
if ( primary_instance ! = old_primary ) {
AP : : logger ( ) . Write_Event ( LogEvent : : GPS_PRIMARY_CHANGED ) ;
}
2021-08-05 23:25:12 -03:00
# endif // HAL_LOGING_ENABLED
# endif // GPS_MAX_RECEIVERS > 1
2019-11-17 06:04:41 -04:00
# ifndef HAL_BUILD_AP_PERIPH
// update notify with gps status. We always base this on the primary_instance
AP_Notify : : flags . gps_status = state [ primary_instance ] . status ;
AP_Notify : : flags . gps_num_sats = state [ primary_instance ] . num_sats ;
# endif
}
/*
update primary GPS instance
*/
2021-08-05 23:25:12 -03:00
# if GPS_MAX_RECEIVERS > 1
2019-11-17 06:04:41 -04:00
void AP_GPS : : update_primary ( void )
{
2019-05-26 22:34:13 -03:00
# if defined(GPS_BLENDED_INSTANCE)
2017-01-29 19:02:57 -04:00
// if blending is requested, attempt to calculate weighting for each GPS
2020-06-23 23:07:35 -03:00
if ( ( GPSAutoSwitch ) _auto_switch . get ( ) = = GPSAutoSwitch : : BLEND ) {
2017-01-29 19:02:57 -04:00
_output_is_blended = calc_blend_weights ( ) ;
2017-03-08 20:47:40 -04:00
// adjust blend health counter
if ( ! _output_is_blended ) {
_blend_health_counter = MIN ( _blend_health_counter + BLEND_COUNTER_FAILURE_INCREMENT , 100 ) ;
} else if ( _blend_health_counter > 0 ) {
_blend_health_counter - - ;
}
2017-03-10 03:09:23 -04:00
// stop blending if unhealthy
if ( _blend_health_counter > = 50 ) {
_output_is_blended = false ;
}
2017-01-29 19:02:57 -04:00
} else {
_output_is_blended = false ;
2017-03-10 02:46:16 -04:00
_blend_health_counter = 0 ;
2017-01-29 19:02:57 -04:00
}
2015-05-21 18:07:59 -03:00
2017-01-29 19:02:57 -04:00
if ( _output_is_blended ) {
// Use the weighting to calculate blended GPS states
calc_blended_state ( ) ;
// set primary to the virtual instance
2017-03-10 02:46:16 -04:00
primary_instance = GPS_BLENDED_INSTANCE ;
2019-11-17 06:04:41 -04:00
return ;
}
2021-08-05 23:25:12 -03:00
# endif // defined (GPS_BLENDED_INSTANCE)
2019-11-17 06:04:41 -04:00
2020-10-29 19:54:05 -03:00
// check the primary param is set to possible GPS
int8_t primary_param = _primary . get ( ) ;
if ( ( primary_param < 0 ) | | ( primary_param > = GPS_MAX_RECEIVERS ) ) {
primary_param = 0 ;
}
// if primary is not enabled try first instance
if ( get_type ( primary_param ) = = GPS_TYPE_NONE ) {
primary_param = 0 ;
2019-11-17 06:04:41 -04:00
}
2020-10-29 19:54:05 -03:00
if ( ( GPSAutoSwitch ) _auto_switch . get ( ) = = GPSAutoSwitch : : NONE ) {
// No switching of GPSs, always use the primary instance
primary_instance = primary_param ;
2019-11-17 06:04:41 -04:00
return ;
}
2019-11-17 06:11:19 -04:00
2019-11-17 06:04:41 -04:00
uint32_t now = AP_HAL : : millis ( ) ;
2019-11-17 06:11:19 -04:00
2021-07-20 04:36:45 -03:00
// special handling of RTK moving baseline pair. Always use the
// base as the rover position is derived from the base, which
// means the rover always has worse position and velocity than the
// base. This overrides the normal logic which would select the
// rover as it typically is in fix type 6 (RTK) whereas base is
// usually fix type 3
2019-11-17 06:11:19 -04:00
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2021-08-11 07:13:55 -03:00
if ( ( ( _type [ i ] = = GPS_TYPE_UBLOX_RTK_BASE ) | | ( _type [ i ] = = GPS_TYPE_UAVCAN_RTK_BASE ) ) & &
( ( _type [ i ^ 1 ] = = GPS_TYPE_UBLOX_RTK_ROVER ) | | ( _type [ i ^ 1 ] = = GPS_TYPE_UAVCAN_RTK_ROVER ) ) & &
2021-07-20 04:36:45 -03:00
( ( state [ i ] . status > = GPS_OK_FIX_3D ) | | ( state [ i ] . status > = state [ i ^ 1 ] . status ) ) ) {
2020-05-20 23:09:45 -03:00
if ( primary_instance ! = i ) {
_last_instance_swap_ms = now ;
primary_instance = i ;
}
2021-07-20 04:36:45 -03:00
// don't do any more switching logic. We don't want the
// RTK status of the rover to cause a switch
2020-05-20 23:09:45 -03:00
return ;
}
2019-11-17 06:11:19 -04:00
}
2021-07-20 04:36:45 -03:00
2021-08-05 23:25:12 -03:00
# if defined(GPS_BLENDED_INSTANCE)
2019-11-17 06:04:41 -04:00
// handling switching away from blended GPS
if ( primary_instance = = GPS_BLENDED_INSTANCE ) {
primary_instance = 0 ;
for ( uint8_t i = 1 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2022-02-01 05:34:22 -04:00
// choose GPS with highest state or higher number of
// satellites. Reject a GPS with an old update time, as it
// may be the old timestamp that triggered the loss of
// blending
const uint32_t delay_threshold = 400 ;
const bool higher_status = state [ i ] . status > state [ primary_instance ] . status ;
const bool old_data_primary = ( now - state [ primary_instance ] . last_gps_time_ms ) > delay_threshold ;
const bool old_data = ( now - state [ i ] . last_gps_time_ms ) > delay_threshold ;
const bool equal_status = state [ i ] . status = = state [ primary_instance ] . status ;
const bool more_sats = state [ i ] . num_sats > state [ primary_instance ] . num_sats ;
if ( old_data & & ! old_data_primary ) {
// don't switch to a GPS that has not updated in 400ms
continue ;
}
if ( state [ i ] . status < GPS_OK_FIX_3D ) {
// don't use a GPS without 3D fix
continue ;
}
// select the new GPS if the primary has old data, or new
// GPS either has higher status, or has the same status
// and more satellites
if ( ( old_data_primary & & ! old_data ) | |
higher_status | |
( equal_status & & more_sats ) ) {
2019-11-17 06:04:41 -04:00
primary_instance = i ;
2014-06-06 18:58:11 -03:00
}
2014-04-03 18:32:34 -03:00
}
2022-02-01 05:34:22 -04:00
_last_instance_swap_ms = now ;
2019-11-17 06:04:41 -04:00
return ;
}
2021-08-05 23:25:12 -03:00
# endif // defined(GPS_BLENDED_INSTANCE)
2020-10-29 19:56:09 -03:00
// Use primary if 3D fix or better
if ( ( ( GPSAutoSwitch ) _auto_switch . get ( ) = = GPSAutoSwitch : : USE_PRIMARY_IF_3D_FIX ) & & ( state [ primary_param ] . status > = GPS_OK_FIX_3D ) ) {
// Primary GPS has a least a 3D fix, switch to it if necessary
if ( primary_instance ! = primary_param ) {
primary_instance = primary_param ;
_last_instance_swap_ms = now ;
}
return ;
}
2019-11-17 06:04:41 -04:00
// handle switch between real GPSs
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( i = = primary_instance ) {
continue ;
}
if ( state [ i ] . status > state [ primary_instance ] . status ) {
// we have a higher status lock, or primary is set to the blended GPS, change GPS
primary_instance = i ;
_last_instance_swap_ms = now ;
continue ;
}
bool another_gps_has_1_or_more_sats = ( state [ i ] . num_sats > = state [ primary_instance ] . num_sats + 1 ) ;
2017-01-29 19:02:57 -04:00
2019-11-17 06:04:41 -04:00
if ( state [ i ] . status = = state [ primary_instance ] . status & & another_gps_has_1_or_more_sats ) {
bool another_gps_has_2_or_more_sats = ( state [ i ] . num_sats > = state [ primary_instance ] . num_sats + 2 ) ;
if ( ( another_gps_has_1_or_more_sats & & ( now - _last_instance_swap_ms ) > = 20000 ) | |
( another_gps_has_2_or_more_sats & & ( now - _last_instance_swap_ms ) > = 5000 ) ) {
// this GPS has more satellites than the
// current primary, switch primary. Once we switch we will
// then tend to stick to the new GPS as primary. We don't
// want to switch too often as it will look like a
// position shift to the controllers.
primary_instance = i ;
_last_instance_swap_ms = now ;
}
}
2014-04-03 18:32:34 -03:00
}
2014-03-28 16:52:27 -03:00
}
2021-08-05 23:25:12 -03:00
# endif // GPS_MAX_RECEIVERS > 1
2014-03-28 16:52:27 -03:00
2019-04-30 07:22:48 -03:00
void AP_GPS : : handle_gps_inject ( const mavlink_message_t & msg )
2017-07-25 04:59:14 -03:00
{
mavlink_gps_inject_data_t packet ;
2019-04-30 07:22:48 -03:00
mavlink_msg_gps_inject_data_decode ( & msg , & packet ) ;
2017-07-25 04:59:14 -03:00
2019-10-21 08:11:26 -03:00
if ( packet . len > sizeof ( packet . data ) ) {
// invalid packet
return ;
}
handle_gps_rtcm_fragment ( 0 , packet . data , packet . len ) ;
2017-07-25 04:59:14 -03:00
}
2016-05-19 20:24:08 -03:00
/*
pass along a mavlink message ( for MAV type )
*/
2019-04-30 07:22:48 -03:00
void AP_GPS : : handle_msg ( const mavlink_message_t & msg )
2016-05-19 20:24:08 -03:00
{
2019-04-30 07:22:48 -03:00
switch ( msg . msgid ) {
2017-07-25 21:19:37 -03:00
case MAVLINK_MSG_ID_GPS_RTCM_DATA :
2016-10-04 04:37:01 -03:00
// pass data to de-fragmenter
handle_gps_rtcm_data ( msg ) ;
2017-07-25 21:19:37 -03:00
break ;
case MAVLINK_MSG_ID_GPS_INJECT_DATA :
2017-07-25 04:59:14 -03:00
handle_gps_inject ( msg ) ;
2017-07-25 21:19:37 -03:00
break ;
default : {
uint8_t i ;
for ( i = 0 ; i < num_instances ; i + + ) {
if ( ( drivers [ i ] ! = nullptr ) & & ( _type [ i ] ! = GPS_TYPE_NONE ) ) {
drivers [ i ] - > handle_msg ( msg ) ;
}
2016-05-19 20:24:08 -03:00
}
2017-07-25 21:19:37 -03:00
break ;
}
2016-05-19 20:24:08 -03:00
}
}
2020-09-03 06:15:35 -03:00
# if HAL_MSP_GPS_ENABLED
void AP_GPS : : handle_msp ( const MSP : : msp_gps_data_message_t & pkt )
{
for ( uint8_t i = 0 ; i < num_instances ; i + + ) {
if ( drivers [ i ] ! = nullptr & & _type [ i ] = = GPS_TYPE_MSP ) {
drivers [ i ] - > handle_msp ( pkt ) ;
}
}
}
# endif // HAL_MSP_GPS_ENABLED
2020-12-28 18:42:14 -04:00
# if HAL_EXTERNAL_AHRS_ENABLED
void AP_GPS : : handle_external ( const AP_ExternalAHRS : : gps_data_message_t & pkt )
{
for ( uint8_t i = 0 ; i < num_instances ; i + + ) {
if ( drivers [ i ] ! = nullptr & & _type [ i ] = = GPS_TYPE_EXTERNAL_AHRS ) {
drivers [ i ] - > handle_external ( pkt ) ;
}
}
}
# endif // HAL_EXTERNAL_AHRS_ENABLED
2014-04-04 17:33:34 -03:00
/**
2017-05-05 09:57:22 -03:00
Lock a GPS port , preventing the GPS driver from using it . This can
2014-04-04 17:33:34 -03:00
be used to allow a user to control a GPS port via the
SERIAL_CONTROL protocol
*/
2017-04-02 11:56:26 -03:00
void AP_GPS : : lock_port ( uint8_t instance , bool lock )
2014-04-04 17:33:34 -03:00
{
2015-04-22 20:10:46 -03:00
2017-01-29 19:02:57 -04:00
if ( instance > = GPS_MAX_RECEIVERS ) {
2014-04-04 17:33:34 -03:00
return ;
}
if ( lock ) {
locked_ports | = ( 1U < < instance ) ;
} else {
locked_ports & = ~ ( 1U < < instance ) ;
}
}
2014-06-06 18:58:11 -03:00
2017-04-02 11:56:26 -03:00
// Inject a packet of raw binary to a GPS
2019-10-21 08:11:26 -03:00
void AP_GPS : : inject_data ( const uint8_t * data , uint16_t len )
2015-04-22 20:10:46 -03:00
{
//Support broadcasting to all GPSes.
2015-06-27 04:26:25 -03:00
if ( _inject_to = = GPS_RTK_INJECT_TO_ALL ) {
2017-01-29 19:02:57 -04:00
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2021-08-11 07:13:55 -03:00
if ( ( _type [ i ] = = GPS_TYPE_UBLOX_RTK_ROVER ) | | ( _type [ i ] = = GPS_TYPE_UAVCAN_RTK_ROVER ) ) {
2019-11-16 00:26:28 -04:00
// we don't externally inject to moving baseline rover
continue ;
}
2015-04-22 20:10:46 -03:00
inject_data ( i , data , len ) ;
}
} else {
inject_data ( _inject_to , data , len ) ;
}
}
2019-10-21 08:11:26 -03:00
void AP_GPS : : inject_data ( uint8_t instance , const uint8_t * data , uint16_t len )
2015-04-22 20:10:46 -03:00
{
2017-01-29 19:02:57 -04:00
if ( instance < GPS_MAX_RECEIVERS & & drivers [ instance ] ! = nullptr ) {
2015-04-22 20:10:46 -03:00
drivers [ instance ] - > inject_data ( data , len ) ;
2016-12-15 09:12:54 -04:00
}
2017-04-02 11:56:26 -03:00
}
2015-04-22 20:10:46 -03:00
2019-12-22 19:03:58 -04:00
/*
get GPS yaw following mavlink GPS_RAW_INT and GPS2_RAW
convention . We return 0 if the GPS is not configured to provide
yaw . We return 65535 for a GPS configured to provide yaw that can ' t
currently provide it . We return from 1 to 36000 for yaw otherwise
*/
uint16_t AP_GPS : : gps_yaw_cdeg ( uint8_t instance ) const
{
if ( ! have_gps_yaw_configured ( instance ) ) {
return 0 ;
}
float yaw_deg , accuracy_deg ;
2021-07-20 04:36:45 -03:00
uint32_t time_ms ;
if ( ! gps_yaw_deg ( instance , yaw_deg , accuracy_deg , time_ms ) ) {
2019-12-22 19:03:58 -04:00
return 65535 ;
}
int yaw_cd = wrap_360_cd ( yaw_deg * 100 ) ;
if ( yaw_cd = = 0 ) {
return 36000 ;
}
return yaw_cd ;
}
2017-04-02 11:56:26 -03:00
void AP_GPS : : send_mavlink_gps_raw ( mavlink_channel_t chan )
2014-06-06 18:58:11 -03:00
{
2015-01-02 02:08:11 -04:00
const Location & loc = location ( 0 ) ;
2017-09-22 05:16:12 -03:00
float hacc = 0.0f ;
float vacc = 0.0f ;
float sacc = 0.0f ;
horizontal_accuracy ( 0 , hacc ) ;
vertical_accuracy ( 0 , vacc ) ;
speed_accuracy ( 0 , sacc ) ;
2015-01-02 02:08:11 -04:00
mavlink_msg_gps_raw_int_send (
chan ,
last_fix_time_ms ( 0 ) * ( uint64_t ) 1000 ,
status ( 0 ) ,
loc . lat , // in 1E7 degrees
loc . lng , // in 1E7 degrees
loc . alt * 10UL , // in mm
get_hdop ( 0 ) ,
2015-09-07 19:37:13 -03:00
get_vdop ( 0 ) ,
2015-01-02 02:08:11 -04:00
ground_speed ( 0 ) * 100 , // cm/s
2016-05-04 22:28:35 -03:00
ground_course ( 0 ) * 100 , // 1/100 degrees,
2017-09-22 05:16:12 -03:00
num_sats ( 0 ) ,
0 , // TODO: Elipsoid height in mm
hacc * 1000 , // one-sigma standard deviation in mm
vacc * 1000 , // one-sigma standard deviation in mm
sacc * 1000 , // one-sigma standard deviation in mm/s
2019-12-22 19:03:58 -04:00
0 , // TODO one-sigma heading accuracy standard deviation
gps_yaw_cdeg ( 0 ) ) ;
2014-06-06 18:58:11 -03:00
}
2019-05-26 22:34:13 -03:00
# if GPS_MAX_RECEIVERS > 1
2017-04-02 11:56:26 -03:00
void AP_GPS : : send_mavlink_gps2_raw ( mavlink_channel_t chan )
2014-06-06 18:58:11 -03:00
{
2020-10-04 18:18:12 -03:00
// always send the message if 2nd GPS is configured
if ( _type [ 1 ] = = GPS_TYPE_NONE ) {
2015-01-02 02:08:11 -04:00
return ;
}
const Location & loc = location ( 1 ) ;
2021-04-17 23:51:22 -03:00
float hacc = 0.0f ;
float vacc = 0.0f ;
float sacc = 0.0f ;
horizontal_accuracy ( 1 , hacc ) ;
vertical_accuracy ( 1 , vacc ) ;
speed_accuracy ( 1 , sacc ) ;
2015-01-02 02:08:11 -04:00
mavlink_msg_gps2_raw_send (
chan ,
last_fix_time_ms ( 1 ) * ( uint64_t ) 1000 ,
status ( 1 ) ,
loc . lat ,
loc . lng ,
loc . alt * 10UL ,
get_hdop ( 1 ) ,
2015-09-07 19:37:13 -03:00
get_vdop ( 1 ) ,
2015-01-02 02:08:11 -04:00
ground_speed ( 1 ) * 100 , // cm/s
2016-05-04 22:28:35 -03:00
ground_course ( 1 ) * 100 , // 1/100 degrees,
2015-01-02 02:08:11 -04:00
num_sats ( 1 ) ,
2019-04-20 18:54:21 -03:00
state [ 1 ] . rtk_num_sats ,
2019-12-22 19:03:58 -04:00
state [ 1 ] . rtk_age_ms ,
2021-04-17 23:51:22 -03:00
gps_yaw_cdeg ( 1 ) ,
0 , // TODO: Elipsoid height in mm
hacc * 1000 , // one-sigma standard deviation in mm
vacc * 1000 , // one-sigma standard deviation in mm
sacc * 1000 , // one-sigma standard deviation in mm/s
0 ) ; // TODO one-sigma heading accuracy standard deviation
2014-06-06 18:58:11 -03:00
}
2019-05-26 22:34:13 -03:00
# endif // GPS_MAX_RECEIVERS
2014-06-06 18:58:11 -03:00
2017-06-30 10:45:24 -03:00
void AP_GPS : : send_mavlink_gps_rtk ( mavlink_channel_t chan , uint8_t inst )
2014-06-06 18:58:11 -03:00
{
2017-06-30 10:45:24 -03:00
if ( inst > = GPS_MAX_RECEIVERS ) {
return ;
2014-06-06 18:58:11 -03:00
}
2017-09-24 18:48:52 -03:00
if ( drivers [ inst ] ! = nullptr & & drivers [ inst ] - > supports_mavlink_gps_rtk_message ( ) ) {
2017-06-30 10:45:24 -03:00
drivers [ inst ] - > send_mavlink_gps_rtk ( chan ) ;
2014-06-06 18:58:11 -03:00
}
}
2016-02-02 03:58:33 -04:00
2019-07-23 03:20:47 -03:00
bool AP_GPS : : first_unconfigured_gps ( uint8_t & instance ) const
2016-02-09 16:59:12 -04:00
{
2017-04-02 11:56:26 -03:00
for ( int i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( _type [ i ] ! = GPS_TYPE_NONE & & ( drivers [ i ] = = nullptr | | ! drivers [ i ] - > is_configured ( ) ) ) {
2019-07-23 03:20:47 -03:00
instance = i ;
return true ;
2016-02-02 03:58:33 -04:00
}
}
2019-07-23 03:20:47 -03:00
return false ;
2016-02-02 03:58:33 -04:00
}
2016-02-09 16:59:12 -04:00
2017-04-02 11:56:26 -03:00
void AP_GPS : : broadcast_first_configuration_failure_reason ( void ) const
{
2019-07-23 03:20:47 -03:00
uint8_t unconfigured ;
if ( first_unconfigured_gps ( unconfigured ) ) {
if ( drivers [ unconfigured ] = = nullptr ) {
2020-03-29 19:49:53 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " GPS %d: was not found " , unconfigured + 1 ) ;
2019-07-23 03:20:47 -03:00
} else {
drivers [ unconfigured ] - > broadcast_configuration_failure_reason ( ) ;
}
2016-04-12 00:16:20 -03:00
}
}
2017-03-08 20:47:40 -04:00
// pre-arm check that all GPSs are close to each other. farthest distance between GPSs (in meters) is returned
bool AP_GPS : : all_consistent ( float & distance ) const
{
// return true immediately if only one valid receiver
if ( num_instances < = 1 | |
2017-03-10 02:46:16 -04:00
drivers [ 0 ] = = nullptr | | _type [ 0 ] = = GPS_TYPE_NONE ) {
2017-03-08 20:47:40 -04:00
distance = 0 ;
return true ;
}
// calculate distance
2019-04-08 10:29:11 -03:00
distance = state [ 0 ] . location . get_distance_NED ( state [ 1 ] . location ) . length ( ) ;
2017-03-08 20:47:40 -04:00
// success if distance is within 50m
return ( distance < 50 ) ;
}
// pre-arm check of GPS blending. True means healthy or that blending is not being used
2017-03-10 04:12:20 -04:00
bool AP_GPS : : blend_health_check ( ) const
2017-03-08 20:47:40 -04:00
{
return ( _blend_health_counter < 50 ) ;
}
2017-04-02 11:56:26 -03:00
/*
2019-10-21 08:11:26 -03:00
re - assemble fragmented RTCM data
2016-10-04 04:37:01 -03:00
*/
2019-10-21 08:11:26 -03:00
void AP_GPS : : handle_gps_rtcm_fragment ( uint8_t flags , const uint8_t * data , uint8_t len )
2016-10-04 04:37:01 -03:00
{
2019-10-21 08:11:26 -03:00
if ( ( flags & 1 ) = = 0 ) {
2016-10-04 04:37:01 -03:00
// it is not fragmented, pass direct
2019-10-21 08:11:26 -03:00
inject_data ( data , len ) ;
2016-10-04 04:37:01 -03:00
return ;
}
// see if we need to allocate re-assembly buffer
if ( rtcm_buffer = = nullptr ) {
rtcm_buffer = ( struct rtcm_buffer * ) calloc ( 1 , sizeof ( * rtcm_buffer ) ) ;
if ( rtcm_buffer = = nullptr ) {
// nothing to do but discard the data
return ;
}
}
2020-10-05 15:55:23 -03:00
const uint8_t fragment = ( flags > > 1U ) & 0x03 ;
const uint8_t sequence = ( flags > > 3U ) & 0x1F ;
2016-10-04 04:37:01 -03:00
// see if this fragment is consistent with existing fragments
if ( rtcm_buffer - > fragments_received & &
( rtcm_buffer - > sequence ! = sequence | |
2020-10-05 15:55:23 -03:00
( rtcm_buffer - > fragments_received & ( 1U < < fragment ) ) ) ) {
2016-10-04 04:37:01 -03:00
// we have one or more partial fragments already received
// which conflict with the new fragment, discard previous fragments
2020-10-05 15:55:23 -03:00
rtcm_buffer - > fragment_count = 0 ;
rtcm_buffer - > fragments_received = 0 ;
2016-10-04 04:37:01 -03:00
}
// add this fragment
rtcm_buffer - > sequence = sequence ;
rtcm_buffer - > fragments_received | = ( 1U < < fragment ) ;
// copy the data
2019-10-21 08:11:26 -03:00
memcpy ( & rtcm_buffer - > buffer [ MAVLINK_MSG_GPS_RTCM_DATA_FIELD_DATA_LEN * ( uint16_t ) fragment ] , data , len ) ;
2016-10-04 04:37:01 -03:00
// when we get a fragment of less than max size then we know the
// number of fragments. Note that this means if you want to send a
// block of RTCM data of an exact multiple of the buffer size you
// need to send a final packet of zero length
2019-10-21 08:11:26 -03:00
if ( len < MAVLINK_MSG_GPS_RTCM_DATA_FIELD_DATA_LEN ) {
2016-10-04 04:37:01 -03:00
rtcm_buffer - > fragment_count = fragment + 1 ;
2019-10-21 08:11:26 -03:00
rtcm_buffer - > total_length = ( MAVLINK_MSG_GPS_RTCM_DATA_FIELD_DATA_LEN * fragment ) + len ;
2016-10-04 04:37:01 -03:00
} else if ( rtcm_buffer - > fragments_received = = 0x0F ) {
// special case of 4 full fragments
rtcm_buffer - > fragment_count = 4 ;
rtcm_buffer - > total_length = MAVLINK_MSG_GPS_RTCM_DATA_FIELD_DATA_LEN * 4 ;
}
// see if we have all fragments
if ( rtcm_buffer - > fragment_count ! = 0 & &
rtcm_buffer - > fragments_received = = ( 1U < < rtcm_buffer - > fragment_count ) - 1 ) {
// we have them all, inject
2016-12-08 04:33:39 -04:00
inject_data ( rtcm_buffer - > buffer , rtcm_buffer - > total_length ) ;
2020-10-05 15:55:23 -03:00
rtcm_buffer - > fragment_count = 0 ;
rtcm_buffer - > fragments_received = 0 ;
2016-10-04 04:37:01 -03:00
}
}
2019-10-21 08:11:26 -03:00
/*
re - assemble GPS_RTCM_DATA message
*/
void AP_GPS : : handle_gps_rtcm_data ( const mavlink_message_t & msg )
{
mavlink_gps_rtcm_data_t packet ;
mavlink_msg_gps_rtcm_data_decode ( & msg , & packet ) ;
if ( packet . len > sizeof ( packet . data ) ) {
// invalid packet
return ;
}
handle_gps_rtcm_fragment ( packet . flags , packet . data , packet . len ) ;
}
2019-01-18 00:23:42 -04:00
void AP_GPS : : Write_AP_Logger_Log_Startup_messages ( )
2016-08-01 08:58:23 -03:00
{
for ( uint8_t instance = 0 ; instance < num_instances ; instance + + ) {
if ( drivers [ instance ] = = nullptr | | state [ instance ] . status = = NO_GPS ) {
continue ;
}
2019-01-18 00:23:42 -04:00
drivers [ instance ] - > Write_AP_Logger_Log_Startup_messages ( ) ;
2016-08-01 08:58:23 -03:00
}
}
2016-12-18 19:31:28 -04:00
/*
2017-01-29 19:02:57 -04:00
return the expected lag ( in seconds ) in the position and velocity readings from the gps
2017-05-21 22:07:26 -03:00
return true if the GPS hardware configuration is known or the delay parameter has been set
2016-12-18 19:31:28 -04:00
*/
2017-05-21 22:07:26 -03:00
bool AP_GPS : : get_lag ( uint8_t instance , float & lag_sec ) const
2016-12-18 19:31:28 -04:00
{
2018-06-28 19:36:35 -03:00
// always enusre a lag is provided
2022-01-03 00:30:34 -04:00
lag_sec = 0.1f ;
2018-06-28 19:36:35 -03:00
2019-03-05 19:24:02 -04:00
if ( instance > = GPS_MAX_INSTANCES ) {
return false ;
}
2019-05-26 22:34:13 -03:00
# if defined(GPS_BLENDED_INSTANCE)
2017-01-29 19:02:57 -04:00
// return lag of blended GPS
2017-05-21 22:07:26 -03:00
if ( instance = = GPS_BLENDED_INSTANCE ) {
lag_sec = _blended_lag_sec ;
// auto switching uses all GPS receivers, so all must be configured
2019-07-23 03:20:47 -03:00
uint8_t inst ; // we don't actually care what the number is, but must pass it
return first_unconfigured_gps ( inst ) ;
2017-01-29 19:02:57 -04:00
}
2019-05-26 22:34:13 -03:00
# endif
2017-01-29 19:02:57 -04:00
2016-12-28 18:20:06 -04:00
if ( _delay_ms [ instance ] > 0 ) {
// if the user has specified a non zero time delay, always return that value
2017-05-21 22:07:26 -03:00
lag_sec = 0.001f * ( float ) _delay_ms [ instance ] ;
// the user is always right !!
return true ;
2016-12-28 18:20:06 -04:00
} else if ( drivers [ instance ] = = nullptr | | state [ instance ] . status = = NO_GPS ) {
2017-05-24 14:42:03 -03:00
// no GPS was detected in this instance so return the worst possible lag term
if ( _type [ instance ] = = GPS_TYPE_NONE ) {
lag_sec = 0.0f ;
return true ;
}
return _type [ instance ] = = GPS_TYPE_AUTO ;
2016-12-28 18:20:06 -04:00
} else {
// the user has not specified a delay so we determine it from the GPS type
2017-05-24 14:42:03 -03:00
return drivers [ instance ] - > get_lag ( lag_sec ) ;
2016-12-18 19:31:28 -04:00
}
}
2017-01-29 19:02:57 -04:00
2017-03-08 05:56:52 -04:00
// return a 3D vector defining the offset of the GPS antenna in meters relative to the body frame origin
const Vector3f & AP_GPS : : get_antenna_offset ( uint8_t instance ) const
{
2019-03-05 19:24:02 -04:00
if ( instance > = GPS_MAX_INSTANCES ) {
// we have to return a reference so use instance 0
return _antenna_offset [ 0 ] ;
}
2019-05-26 22:34:13 -03:00
# if defined(GPS_BLENDED_INSTANCE)
2019-03-05 19:24:02 -04:00
if ( instance = = GPS_BLENDED_INSTANCE ) {
2017-03-08 05:56:52 -04:00
// return an offset for the blended GPS solution
return _blended_antenna_offset ;
}
2019-05-26 22:34:13 -03:00
# endif
2019-03-05 19:24:02 -04:00
return _antenna_offset [ instance ] ;
2017-03-08 05:56:52 -04:00
}
2017-03-07 23:33:31 -04:00
/*
2017-06-11 04:38:11 -03:00
returns the desired gps update rate in milliseconds
2017-09-23 20:51:47 -03:00
this does not provide any guarantee that the GPS is updating at the requested
2017-06-11 04:38:11 -03:00
rate it is simply a helper for use in the backends for determining what rate
they should be configuring the GPS to run at
2017-03-07 23:33:31 -04:00
*/
uint16_t AP_GPS : : get_rate_ms ( uint8_t instance ) const
{
// sanity check
if ( instance > = num_instances | | _rate_ms [ instance ] < = 0 ) {
return GPS_MAX_RATE_MS ;
}
return MIN ( _rate_ms [ instance ] , GPS_MAX_RATE_MS ) ;
}
2019-05-26 22:34:13 -03:00
# if defined(GPS_BLENDED_INSTANCE)
2017-01-29 19:02:57 -04:00
/*
2017-09-23 20:51:47 -03:00
calculate the weightings used to blend GPSs location and velocity data
2017-01-29 19:02:57 -04:00
*/
bool AP_GPS : : calc_blend_weights ( void )
{
// zero the blend weights
memset ( & _blend_weights , 0 , sizeof ( _blend_weights ) ) ;
2021-01-27 16:16:08 -04:00
static_assert ( GPS_MAX_RECEIVERS = = 2 , " GPS blending only currently works with 2 receivers " ) ;
// Note that the early quit below relies upon exactly 2 instances
// The time delta calculations below also rely upon every instance being currently detected and being parsed
2017-01-29 19:02:57 -04:00
// exit immediately if not enough receivers to do blending
2019-03-07 17:26:11 -04:00
if ( state [ 0 ] . status < = NO_FIX | | state [ 1 ] . status < = NO_FIX ) {
2017-01-29 19:02:57 -04:00
return false ;
}
// Use the oldest non-zero time, but if time difference is excessive, use newest to prevent a disconnected receiver from blocking updates
uint32_t max_ms = 0 ; // newest non-zero system time of arrival of a GPS message
uint32_t min_ms = - 1 ; // oldest non-zero system time of arrival of a GPS message
2021-01-27 16:16:08 -04:00
uint32_t max_rate_ms = 0 ; // largest update interval of a GPS receiver
2017-01-29 19:02:57 -04:00
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
// Find largest and smallest times
if ( state [ i ] . last_gps_time_ms > max_ms ) {
max_ms = state [ i ] . last_gps_time_ms ;
}
if ( ( state [ i ] . last_gps_time_ms < min_ms ) & & ( state [ i ] . last_gps_time_ms > 0 ) ) {
min_ms = state [ i ] . last_gps_time_ms ;
}
2021-01-27 16:16:08 -04:00
max_rate_ms = MAX ( get_rate_ms ( i ) , max_rate_ms ) ;
2020-11-22 18:33:20 -04:00
if ( isinf ( state [ i ] . speed_accuracy ) | |
isinf ( state [ i ] . horizontal_accuracy ) | |
isinf ( state [ i ] . vertical_accuracy ) ) {
return false ;
}
2017-01-29 19:02:57 -04:00
}
2021-01-27 16:16:08 -04:00
if ( ( max_ms - min_ms ) < ( 2 * max_rate_ms ) ) {
2017-01-29 19:02:57 -04:00
// data is not too delayed so use the oldest time_stamp to give a chance for data from that receiver to be updated
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . last_gps_time_ms = min_ms ;
2017-01-29 19:02:57 -04:00
} else {
// receiver data has timed out so fail out of blending
return false ;
}
// calculate the sum squared speed accuracy across all GPS sensors
float speed_accuracy_sum_sq = 0.0f ;
if ( _blend_mask & BLEND_MASK_USE_SPD_ACC ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( state [ i ] . status > = GPS_OK_FIX_3D ) {
if ( state [ i ] . have_speed_accuracy & & state [ i ] . speed_accuracy > 0.0f ) {
speed_accuracy_sum_sq + = state [ i ] . speed_accuracy * state [ i ] . speed_accuracy ;
} else {
// not all receivers support this metric so set it to zero and don't use it
speed_accuracy_sum_sq = 0.0f ;
break ;
}
}
}
}
// calculate the sum squared horizontal position accuracy across all GPS sensors
float horizontal_accuracy_sum_sq = 0.0f ;
if ( _blend_mask & BLEND_MASK_USE_HPOS_ACC ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( state [ i ] . status > = GPS_OK_FIX_2D ) {
if ( state [ i ] . have_horizontal_accuracy & & state [ i ] . horizontal_accuracy > 0.0f ) {
horizontal_accuracy_sum_sq + = state [ i ] . horizontal_accuracy * state [ i ] . horizontal_accuracy ;
} else {
// not all receivers support this metric so set it to zero and don't use it
horizontal_accuracy_sum_sq = 0.0f ;
break ;
}
}
}
}
// calculate the sum squared vertical position accuracy across all GPS sensors
float vertical_accuracy_sum_sq = 0.0f ;
if ( _blend_mask & BLEND_MASK_USE_VPOS_ACC ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( state [ i ] . status > = GPS_OK_FIX_3D ) {
if ( state [ i ] . have_vertical_accuracy & & state [ i ] . vertical_accuracy > 0.0f ) {
vertical_accuracy_sum_sq + = state [ i ] . vertical_accuracy * state [ i ] . vertical_accuracy ;
} else {
// not all receivers support this metric so set it to zero and don't use it
vertical_accuracy_sum_sq = 0.0f ;
break ;
}
}
}
}
// Check if we can do blending using reported accuracy
bool can_do_blending = ( horizontal_accuracy_sum_sq > 0.0f | | vertical_accuracy_sum_sq > 0.0f | | speed_accuracy_sum_sq > 0.0f ) ;
2017-05-05 09:57:22 -03:00
// if we can't do blending using reported accuracy, return false and hard switch logic will be used instead
2017-01-29 19:02:57 -04:00
if ( ! can_do_blending ) {
return false ;
}
float sum_of_all_weights = 0.0f ;
// calculate a weighting using the reported horizontal position
float hpos_blend_weights [ GPS_MAX_RECEIVERS ] = { } ;
if ( horizontal_accuracy_sum_sq > 0.0f & & ( _blend_mask & BLEND_MASK_USE_HPOS_ACC ) ) {
// calculate the weights using the inverse of the variances
float sum_of_hpos_weights = 0.0f ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2017-03-10 22:25:46 -04:00
if ( state [ i ] . status > = GPS_OK_FIX_2D & & state [ i ] . horizontal_accuracy > = 0.001f ) {
2017-01-29 19:02:57 -04:00
hpos_blend_weights [ i ] = horizontal_accuracy_sum_sq / ( state [ i ] . horizontal_accuracy * state [ i ] . horizontal_accuracy ) ;
sum_of_hpos_weights + = hpos_blend_weights [ i ] ;
}
}
// normalise the weights
if ( sum_of_hpos_weights > 0.0f ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
hpos_blend_weights [ i ] = hpos_blend_weights [ i ] / sum_of_hpos_weights ;
}
sum_of_all_weights + = 1.0f ;
}
}
// calculate a weighting using the reported vertical position accuracy
2017-03-10 02:48:07 -04:00
float vpos_blend_weights [ GPS_MAX_RECEIVERS ] = { } ;
2017-01-29 19:02:57 -04:00
if ( vertical_accuracy_sum_sq > 0.0f & & ( _blend_mask & BLEND_MASK_USE_VPOS_ACC ) ) {
// calculate the weights using the inverse of the variances
float sum_of_vpos_weights = 0.0f ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2017-03-10 22:25:46 -04:00
if ( state [ i ] . status > = GPS_OK_FIX_3D & & state [ i ] . vertical_accuracy > = 0.001f ) {
2017-01-29 19:02:57 -04:00
vpos_blend_weights [ i ] = vertical_accuracy_sum_sq / ( state [ i ] . vertical_accuracy * state [ i ] . vertical_accuracy ) ;
sum_of_vpos_weights + = vpos_blend_weights [ i ] ;
}
}
// normalise the weights
if ( sum_of_vpos_weights > 0.0f ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
vpos_blend_weights [ i ] = vpos_blend_weights [ i ] / sum_of_vpos_weights ;
}
sum_of_all_weights + = 1.0f ;
} ;
}
// calculate a weighting using the reported speed accuracy
2017-03-10 02:48:07 -04:00
float spd_blend_weights [ GPS_MAX_RECEIVERS ] = { } ;
2017-01-29 19:02:57 -04:00
if ( speed_accuracy_sum_sq > 0.0f & & ( _blend_mask & BLEND_MASK_USE_SPD_ACC ) ) {
// calculate the weights using the inverse of the variances
float sum_of_spd_weights = 0.0f ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2017-03-10 22:25:46 -04:00
if ( state [ i ] . status > = GPS_OK_FIX_3D & & state [ i ] . speed_accuracy > = 0.001f ) {
2017-01-29 19:02:57 -04:00
spd_blend_weights [ i ] = speed_accuracy_sum_sq / ( state [ i ] . speed_accuracy * state [ i ] . speed_accuracy ) ;
sum_of_spd_weights + = spd_blend_weights [ i ] ;
}
}
// normalise the weights
if ( sum_of_spd_weights > 0.0f ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
spd_blend_weights [ i ] = spd_blend_weights [ i ] / sum_of_spd_weights ;
}
sum_of_all_weights + = 1.0f ;
}
}
2020-11-21 23:24:01 -04:00
if ( ! is_positive ( sum_of_all_weights ) ) {
return false ;
}
2017-01-29 19:02:57 -04:00
// calculate an overall weight
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
_blend_weights [ i ] = ( hpos_blend_weights [ i ] + vpos_blend_weights [ i ] + spd_blend_weights [ i ] ) / sum_of_all_weights ;
}
return true ;
}
/*
calculate a blended GPS state
*/
void AP_GPS : : calc_blended_state ( void )
{
// initialise the blended states so we can accumulate the results using the weightings for each GPS receiver
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . instance = GPS_BLENDED_INSTANCE ;
state [ GPS_BLENDED_INSTANCE ] . status = NO_FIX ;
state [ GPS_BLENDED_INSTANCE ] . time_week_ms = 0 ;
state [ GPS_BLENDED_INSTANCE ] . time_week = 0 ;
state [ GPS_BLENDED_INSTANCE ] . ground_speed = 0.0f ;
state [ GPS_BLENDED_INSTANCE ] . ground_course = 0.0f ;
2017-05-24 16:19:10 -03:00
state [ GPS_BLENDED_INSTANCE ] . hdop = GPS_UNKNOWN_DOP ;
state [ GPS_BLENDED_INSTANCE ] . vdop = GPS_UNKNOWN_DOP ;
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . num_sats = 0 ;
state [ GPS_BLENDED_INSTANCE ] . velocity . zero ( ) ;
state [ GPS_BLENDED_INSTANCE ] . speed_accuracy = 1e6 f ;
state [ GPS_BLENDED_INSTANCE ] . horizontal_accuracy = 1e6 f ;
state [ GPS_BLENDED_INSTANCE ] . vertical_accuracy = 1e6 f ;
state [ GPS_BLENDED_INSTANCE ] . have_vertical_velocity = false ;
state [ GPS_BLENDED_INSTANCE ] . have_speed_accuracy = false ;
state [ GPS_BLENDED_INSTANCE ] . have_horizontal_accuracy = false ;
state [ GPS_BLENDED_INSTANCE ] . have_vertical_accuracy = false ;
2019-01-01 23:11:38 -04:00
state [ GPS_BLENDED_INSTANCE ] . location = { } ;
2017-01-29 19:02:57 -04:00
_blended_antenna_offset . zero ( ) ;
_blended_lag_sec = 0 ;
2021-05-19 23:34:13 -03:00
# if HAL_LOGGING_ENABLED
2020-11-18 21:53:42 -04:00
const uint32_t last_blended_message_time_ms = timing [ GPS_BLENDED_INSTANCE ] . last_message_time_ms ;
# endif
2017-03-10 02:46:16 -04:00
timing [ GPS_BLENDED_INSTANCE ] . last_fix_time_ms = 0 ;
timing [ GPS_BLENDED_INSTANCE ] . last_message_time_ms = 0 ;
2017-01-29 19:02:57 -04:00
// combine the states into a blended solution
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
// use the highest status
2017-03-10 02:46:16 -04:00
if ( state [ i ] . status > state [ GPS_BLENDED_INSTANCE ] . status ) {
state [ GPS_BLENDED_INSTANCE ] . status = state [ i ] . status ;
2017-01-29 19:02:57 -04:00
}
// calculate a blended average velocity
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . velocity + = state [ i ] . velocity * _blend_weights [ i ] ;
2017-01-29 19:02:57 -04:00
// report the best valid accuracies and DOP metrics
2017-03-10 02:46:16 -04:00
if ( state [ i ] . have_horizontal_accuracy & & state [ i ] . horizontal_accuracy > 0.0f & & state [ i ] . horizontal_accuracy < state [ GPS_BLENDED_INSTANCE ] . horizontal_accuracy ) {
state [ GPS_BLENDED_INSTANCE ] . have_horizontal_accuracy = true ;
state [ GPS_BLENDED_INSTANCE ] . horizontal_accuracy = state [ i ] . horizontal_accuracy ;
2017-01-29 19:02:57 -04:00
}
2017-03-10 02:46:16 -04:00
if ( state [ i ] . have_vertical_accuracy & & state [ i ] . vertical_accuracy > 0.0f & & state [ i ] . vertical_accuracy < state [ GPS_BLENDED_INSTANCE ] . vertical_accuracy ) {
state [ GPS_BLENDED_INSTANCE ] . have_vertical_accuracy = true ;
state [ GPS_BLENDED_INSTANCE ] . vertical_accuracy = state [ i ] . vertical_accuracy ;
2017-01-29 19:02:57 -04:00
}
2017-04-02 11:56:26 -03:00
if ( state [ i ] . have_vertical_velocity ) {
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . have_vertical_velocity = true ;
2017-01-29 19:02:57 -04:00
}
2017-03-10 02:46:16 -04:00
if ( state [ i ] . have_speed_accuracy & & state [ i ] . speed_accuracy > 0.0f & & state [ i ] . speed_accuracy < state [ GPS_BLENDED_INSTANCE ] . speed_accuracy ) {
state [ GPS_BLENDED_INSTANCE ] . have_speed_accuracy = true ;
state [ GPS_BLENDED_INSTANCE ] . speed_accuracy = state [ i ] . speed_accuracy ;
2017-01-29 19:02:57 -04:00
}
2017-03-10 02:46:16 -04:00
if ( state [ i ] . hdop > 0 & & state [ i ] . hdop < state [ GPS_BLENDED_INSTANCE ] . hdop ) {
state [ GPS_BLENDED_INSTANCE ] . hdop = state [ i ] . hdop ;
2017-01-29 19:02:57 -04:00
}
2017-03-10 02:46:16 -04:00
if ( state [ i ] . vdop > 0 & & state [ i ] . vdop < state [ GPS_BLENDED_INSTANCE ] . vdop ) {
state [ GPS_BLENDED_INSTANCE ] . vdop = state [ i ] . vdop ;
2017-01-29 19:02:57 -04:00
}
2017-03-10 02:46:16 -04:00
if ( state [ i ] . num_sats > 0 & & state [ i ] . num_sats > state [ GPS_BLENDED_INSTANCE ] . num_sats ) {
state [ GPS_BLENDED_INSTANCE ] . num_sats = state [ i ] . num_sats ;
2017-01-29 19:02:57 -04:00
}
// report a blended average GPS antenna position
Vector3f temp_antenna_offset = _antenna_offset [ i ] ;
temp_antenna_offset * = _blend_weights [ i ] ;
_blended_antenna_offset + = temp_antenna_offset ;
// blend the timing data
2017-03-10 02:46:16 -04:00
if ( timing [ i ] . last_fix_time_ms > timing [ GPS_BLENDED_INSTANCE ] . last_fix_time_ms ) {
timing [ GPS_BLENDED_INSTANCE ] . last_fix_time_ms = timing [ i ] . last_fix_time_ms ;
2017-01-29 19:02:57 -04:00
}
2017-03-10 02:46:16 -04:00
if ( timing [ i ] . last_message_time_ms > timing [ GPS_BLENDED_INSTANCE ] . last_message_time_ms ) {
timing [ GPS_BLENDED_INSTANCE ] . last_message_time_ms = timing [ i ] . last_message_time_ms ;
2017-01-29 19:02:57 -04:00
}
}
/*
* Calculate an instantaneous weighted / blended average location from the available GPS instances and store in the _output_state .
2017-09-04 13:58:22 -03:00
* This will be statistically the most likely location , but will be not stable enough for direct use by the autopilot .
2017-01-29 19:02:57 -04:00
*/
// Use the GPS with the highest weighting as the reference position
float best_weight = 0.0f ;
uint8_t best_index = 0 ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( _blend_weights [ i ] > best_weight ) {
best_weight = _blend_weights [ i ] ;
best_index = i ;
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . location = state [ i ] . location ;
2017-01-29 19:02:57 -04:00
}
}
// Calculate the weighted sum of horizontal and vertical position offsets relative to the reference position
Vector2f blended_NE_offset_m ;
float blended_alt_offset_cm = 0.0f ;
blended_NE_offset_m . zero ( ) ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( _blend_weights [ i ] > 0.0f & & i ! = best_index ) {
2019-04-08 10:16:19 -03:00
blended_NE_offset_m + = state [ GPS_BLENDED_INSTANCE ] . location . get_distance_NE ( state [ i ] . location ) * _blend_weights [ i ] ;
2017-03-10 02:46:16 -04:00
blended_alt_offset_cm + = ( float ) ( state [ i ] . location . alt - state [ GPS_BLENDED_INSTANCE ] . location . alt ) * _blend_weights [ i ] ;
2017-01-29 19:02:57 -04:00
}
}
// Add the sum of weighted offsets to the reference location to obtain the blended location
2019-02-24 20:16:21 -04:00
state [ GPS_BLENDED_INSTANCE ] . location . offset ( blended_NE_offset_m . x , blended_NE_offset_m . y ) ;
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . location . alt + = ( int ) blended_alt_offset_cm ;
2017-01-29 19:02:57 -04:00
// Calculate ground speed and course from blended velocity vector
2021-09-11 07:31:17 -03:00
state [ GPS_BLENDED_INSTANCE ] . ground_speed = state [ GPS_BLENDED_INSTANCE ] . velocity . xy ( ) . length ( ) ;
2017-03-12 23:21:47 -03:00
state [ GPS_BLENDED_INSTANCE ] . ground_course = wrap_360 ( degrees ( atan2f ( state [ GPS_BLENDED_INSTANCE ] . velocity . y , state [ GPS_BLENDED_INSTANCE ] . velocity . x ) ) ) ;
2017-01-29 19:02:57 -04:00
// If the GPS week is the same then use a blended time_week_ms
// If week is different, then use time stamp from GPS with largest weighting
// detect inconsistent week data
uint8_t last_week_instance = 0 ;
bool weeks_consistent = true ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( last_week_instance = = 0 & & _blend_weights [ i ] > 0 ) {
// this is our first valid sensor week data
last_week_instance = state [ i ] . time_week ;
} else if ( last_week_instance ! = 0 & & _blend_weights [ i ] > 0 & & last_week_instance ! = state [ i ] . time_week ) {
// there is valid sensor week data that is inconsistent
weeks_consistent = false ;
}
}
// calculate output
if ( ! weeks_consistent ) {
// use data from highest weighted sensor
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . time_week = state [ best_index ] . time_week ;
state [ GPS_BLENDED_INSTANCE ] . time_week_ms = state [ best_index ] . time_week_ms ;
2017-01-29 19:02:57 -04:00
} else {
// use week number from highest weighting GPS (they should all have the same week number)
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . time_week = state [ best_index ] . time_week ;
2017-01-29 19:02:57 -04:00
// calculate a blended value for the number of ms lapsed in the week
double temp_time_0 = 0.0 ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( _blend_weights [ i ] > 0.0f ) {
2017-03-17 07:27:23 -03:00
temp_time_0 + = ( double ) state [ i ] . time_week_ms * ( double ) _blend_weights [ i ] ;
2017-01-29 19:02:57 -04:00
}
}
2017-03-10 02:46:16 -04:00
state [ GPS_BLENDED_INSTANCE ] . time_week_ms = ( uint32_t ) temp_time_0 ;
2017-01-29 19:02:57 -04:00
}
// calculate a blended value for the timing data and lag
double temp_time_1 = 0.0 ;
double temp_time_2 = 0.0 ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( _blend_weights [ i ] > 0.0f ) {
2017-03-17 07:27:23 -03:00
temp_time_1 + = ( double ) timing [ i ] . last_fix_time_ms * ( double ) _blend_weights [ i ] ;
temp_time_2 + = ( double ) timing [ i ] . last_message_time_ms * ( double ) _blend_weights [ i ] ;
2017-05-21 22:07:26 -03:00
float gps_lag_sec = 0 ;
get_lag ( i , gps_lag_sec ) ;
_blended_lag_sec + = gps_lag_sec * _blend_weights [ i ] ;
2017-01-29 19:02:57 -04:00
}
}
2017-03-10 02:46:16 -04:00
timing [ GPS_BLENDED_INSTANCE ] . last_fix_time_ms = ( uint32_t ) temp_time_1 ;
timing [ GPS_BLENDED_INSTANCE ] . last_message_time_ms = ( uint32_t ) temp_time_2 ;
2020-08-24 12:26:49 -03:00
2021-05-19 23:34:13 -03:00
# if HAL_LOGGING_ENABLED
2020-11-18 21:53:42 -04:00
if ( timing [ GPS_BLENDED_INSTANCE ] . last_message_time_ms > last_blended_message_time_ms & &
2020-08-24 12:26:49 -03:00
should_log ( ) ) {
2021-02-09 19:08:37 -04:00
Write_GPS ( GPS_BLENDED_INSTANCE ) ;
2020-08-24 12:26:49 -03:00
}
# endif
2017-01-29 19:02:57 -04:00
}
2019-05-26 22:34:13 -03:00
# endif // GPS_BLENDED_INSTANCE
2017-09-21 22:22:22 -03:00
2019-03-05 19:24:02 -04:00
bool AP_GPS : : is_healthy ( uint8_t instance ) const
{
if ( instance > = GPS_MAX_INSTANCES ) {
return false ;
}
2020-10-29 19:54:05 -03:00
if ( get_type ( _primary . get ( ) ) = = GPS_TYPE_NONE ) {
return false ;
}
2020-04-04 19:53:27 -03:00
/*
allow two lost frames before declaring the GPS unhealthy , but
2020-04-16 23:34:29 -03:00
require the average frame rate to be close to 5 Hz . We allow for
a bit higher average for a rover due to the packet loss that
happens with the RTCMv3 data
2020-04-04 19:53:27 -03:00
*/
const uint8_t delay_threshold = 2 ;
2021-07-16 13:21:01 -03:00
const float delay_avg_max = ( ( _type [ instance ] = = GPS_TYPE_UBLOX_RTK_ROVER ) | | ( _type [ instance ] = = GPS_TYPE_UAVCAN_RTK_ROVER ) ) ? 245 : 215 ;
2020-04-04 19:53:27 -03:00
const GPS_timing & t = timing [ instance ] ;
2020-08-21 02:05:14 -03:00
bool delay_ok = ( t . delayed_count < delay_threshold ) & &
t . average_delta_ms < delay_avg_max & &
state [ instance ] . lagged_sample_count < 5 ;
2019-03-05 19:24:02 -04:00
2019-05-26 22:34:13 -03:00
# if defined(GPS_BLENDED_INSTANCE)
2019-03-05 19:24:02 -04:00
if ( instance = = GPS_BLENDED_INSTANCE ) {
2020-04-04 19:53:27 -03:00
return delay_ok & & blend_health_check ( ) ;
2019-03-05 19:24:02 -04:00
}
2019-05-26 22:34:13 -03:00
# endif
2019-03-05 19:24:02 -04:00
2020-04-04 19:53:27 -03:00
return delay_ok & & drivers [ instance ] ! = nullptr & &
2017-09-21 22:22:22 -03:00
drivers [ instance ] - > is_healthy ( ) ;
}
2017-10-19 22:15:11 -03:00
bool AP_GPS : : prepare_for_arming ( void ) {
bool all_passed = true ;
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( drivers [ i ] ! = nullptr ) {
all_passed & = drivers [ i ] - > prepare_for_arming ( ) ;
}
}
return all_passed ;
}
2017-10-25 00:58:57 -03:00
2020-12-28 10:30:41 -04:00
bool AP_GPS : : backends_healthy ( char failure_msg [ ] , uint16_t failure_msg_len ) {
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
2021-07-29 03:16:08 -03:00
# if HAL_ENABLE_LIBUAVCAN_DRIVERS
2021-08-11 07:13:55 -03:00
if ( _type [ i ] = = GPS_TYPE_UAVCAN | |
_type [ i ] = = GPS_TYPE_UAVCAN_RTK_BASE | |
_type [ i ] = = GPS_TYPE_UAVCAN_RTK_ROVER ) {
2020-12-28 10:30:41 -04:00
if ( ! AP_GPS_UAVCAN : : backends_healthy ( failure_msg , failure_msg_len ) ) {
return false ;
}
}
# endif
2021-08-11 07:13:55 -03:00
if ( _type [ i ] = = GPS_TYPE_UBLOX_RTK_ROVER | |
_type [ i ] = = GPS_TYPE_UAVCAN_RTK_ROVER ) {
2021-07-29 03:16:08 -03:00
if ( AP_HAL : : millis ( ) - state [ i ] . gps_yaw_time_ms > 15000 ) {
hal . util - > snprintf ( failure_msg , failure_msg_len , " GPS[%u] yaw not available " , unsigned ( i + 1 ) ) ;
return false ;
}
}
}
2020-12-28 10:30:41 -04:00
return true ;
}
2019-11-12 02:16:26 -04:00
bool AP_GPS : : logging_failed ( void ) const {
if ( ! logging_enabled ( ) ) {
return false ;
}
for ( uint8_t i = 0 ; i < GPS_MAX_RECEIVERS ; i + + ) {
if ( ( drivers [ i ] ! = nullptr ) & & ! ( drivers [ i ] - > logging_healthy ( ) ) ) {
return true ;
}
}
return false ;
}
2020-09-03 08:48:47 -03:00
// get iTOW, if supported, zero otherwie
uint32_t AP_GPS : : get_itow ( uint8_t instance ) const
{
if ( instance > = GPS_MAX_RECEIVERS | | drivers [ instance ] = = nullptr ) {
return 0 ;
}
2022-02-02 03:21:03 -04:00
return drivers [ instance ] - > get_last_itow_ms ( ) ;
2020-09-03 08:48:47 -03:00
}
2021-02-23 16:18:17 -04:00
bool AP_GPS : : get_error_codes ( uint8_t instance , uint32_t & error_codes ) const
{
if ( instance > = GPS_MAX_RECEIVERS | | drivers [ instance ] = = nullptr ) {
return false ;
}
return drivers [ instance ] - > get_error_codes ( error_codes ) ;
}
2021-02-09 19:08:37 -04:00
// Logging support:
// Write an GPS packet
void AP_GPS : : Write_GPS ( uint8_t i )
{
const uint64_t time_us = AP_HAL : : micros64 ( ) ;
const struct Location & loc = location ( i ) ;
float yaw_deg = 0 , yaw_accuracy_deg = 0 ;
2021-07-20 04:36:45 -03:00
uint32_t yaw_time_ms ;
gps_yaw_deg ( i , yaw_deg , yaw_accuracy_deg , yaw_time_ms ) ;
2021-02-09 19:08:37 -04:00
const struct log_GPS pkt {
LOG_PACKET_HEADER_INIT ( LOG_GPS_MSG ) ,
time_us : time_us ,
instance : i ,
status : ( uint8_t ) status ( i ) ,
gps_week_ms : time_week_ms ( i ) ,
gps_week : time_week ( i ) ,
num_sats : num_sats ( i ) ,
hdop : get_hdop ( i ) ,
latitude : loc . lat ,
longitude : loc . lng ,
altitude : loc . alt ,
ground_speed : ground_speed ( i ) ,
ground_course : ground_course ( i ) ,
vel_z : velocity ( i ) . z ,
yaw : yaw_deg ,
used : ( uint8_t ) ( AP : : gps ( ) . primary_sensor ( ) = = i )
} ;
AP : : logger ( ) . WriteBlock ( & pkt , sizeof ( pkt ) ) ;
/* write auxiliary accuracy information as well */
float hacc = 0 , vacc = 0 , sacc = 0 ;
horizontal_accuracy ( i , hacc ) ;
vertical_accuracy ( i , vacc ) ;
speed_accuracy ( i , sacc ) ;
struct log_GPA pkt2 {
LOG_PACKET_HEADER_INIT ( LOG_GPA_MSG ) ,
time_us : time_us ,
instance : i ,
vdop : get_vdop ( i ) ,
hacc : ( uint16_t ) MIN ( ( hacc * 100 ) , UINT16_MAX ) ,
vacc : ( uint16_t ) MIN ( ( vacc * 100 ) , UINT16_MAX ) ,
sacc : ( uint16_t ) MIN ( ( sacc * 100 ) , UINT16_MAX ) ,
yaw_accuracy : yaw_accuracy_deg ,
have_vv : ( uint8_t ) have_vertical_velocity ( i ) ,
sample_ms : last_message_time_ms ( i ) ,
delta_ms : last_message_delta_time_ms ( i )
} ;
AP : : logger ( ) . WriteBlock ( & pkt2 , sizeof ( pkt2 ) ) ;
}
2021-07-20 04:36:45 -03:00
/*
get GPS based yaw
*/
bool AP_GPS : : gps_yaw_deg ( uint8_t instance , float & yaw_deg , float & accuracy_deg , uint32_t & time_ms ) const
{
# if GPS_MAX_RECEIVERS > 1
if ( instance < GPS_MAX_RECEIVERS & &
2021-08-11 07:13:55 -03:00
( ( _type [ instance ] = = GPS_TYPE_UBLOX_RTK_BASE ) | | ( _type [ instance ] = = GPS_TYPE_UAVCAN_RTK_BASE ) ) & &
( ( _type [ instance ^ 1 ] = = GPS_TYPE_UBLOX_RTK_ROVER ) | | ( _type [ instance ^ 1 ] = = GPS_TYPE_UAVCAN_RTK_ROVER ) ) ) {
2021-07-20 04:36:45 -03:00
// return the yaw from the rover
instance ^ = 1 ;
}
# endif
if ( ! have_gps_yaw ( instance ) ) {
return false ;
}
yaw_deg = state [ instance ] . gps_yaw ;
// get lagged timestamp
time_ms = state [ instance ] . gps_yaw_time_ms ;
float lag_s ;
if ( get_lag ( instance , lag_s ) ) {
uint32_t lag_ms = lag_s * 1000 ;
time_ms - = lag_ms ;
}
if ( state [ instance ] . have_gps_yaw_accuracy ) {
accuracy_deg = state [ instance ] . gps_yaw_accuracy ;
} else {
// fall back to 10 degrees as a generic default
accuracy_deg = 10 ;
}
return true ;
}
2017-10-25 00:58:57 -03:00
namespace AP {
AP_GPS & gps ( )
{
2019-04-05 03:21:03 -03:00
return * AP_GPS : : get_singleton ( ) ;
2017-10-25 00:58:57 -03:00
}
} ;