You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
419 lines
13 KiB
419 lines
13 KiB
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- |
|
/* |
|
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/>. |
|
*/ |
|
|
|
#ifndef __AP_GPS_H__ |
|
#define __AP_GPS_H__ |
|
|
|
#include <AP_HAL.h> |
|
#include <inttypes.h> |
|
#include <AP_Progmem.h> |
|
#include <AP_Common.h> |
|
#include <AP_Param.h> |
|
#include <AP_Math.h> |
|
#include <GCS_MAVLink.h> |
|
#include <AP_Vehicle.h> |
|
#include "GPS_detect_state.h" |
|
|
|
/** |
|
maximum number of GPS instances available on this platform. If more |
|
than 1 then redundent sensors may be available |
|
*/ |
|
#if HAL_CPU_CLASS > HAL_CPU_CLASS_16 |
|
#define GPS_MAX_INSTANCES 2 |
|
#else |
|
#define GPS_MAX_INSTANCES 1 |
|
#endif |
|
|
|
#if HAL_CPU_CLASS >= HAL_CPU_CLASS_75 |
|
#define GPS_RTK_AVAILABLE 1 |
|
#else |
|
#define GPS_RTK_AVAILABLE 0 |
|
#endif |
|
|
|
/** |
|
* save flash by skipping NMEA and SIRF support on ArduCopter on APM1/2 or any frame type on AVR1280 CPUs |
|
*/ |
|
#if HAL_CPU_CLASS < HAL_CPU_CLASS_75 && defined(APM_BUILD_DIRECTORY) |
|
#if (APM_BUILD_TYPE(APM_BUILD_ArduCopter) || defined(__AVR_ATmega1280__)) |
|
#define GPS_SKIP_SIRF_NMEA |
|
#endif |
|
#endif |
|
|
|
class DataFlash_Class; |
|
class AP_GPS_Backend; |
|
|
|
/// @class AP_GPS |
|
/// GPS driver main class |
|
class AP_GPS |
|
{ |
|
public: |
|
// constructor |
|
AP_GPS() { |
|
AP_Param::setup_object_defaults(this, var_info); |
|
} |
|
|
|
/// Startup initialisation. |
|
void init(DataFlash_Class *dataflash); |
|
|
|
/// Update GPS state based on possible bytes received from the module. |
|
/// This routine must be called periodically (typically at 10Hz or |
|
/// more) to process incoming data. |
|
void update(void); |
|
|
|
//True if any of the underlying GPS Drivers are ready to enter |
|
//a dgps-based fix beyond 3D lock, such as RTK mode. |
|
bool can_calculate_base_pos(void); |
|
|
|
//Allows the underlying GPS Drivers to enter a differential lock |
|
//Might cause a position jump, thus only do this on the ground. |
|
void calculate_base_pos(void); |
|
|
|
// GPS driver types |
|
enum GPS_Type { |
|
GPS_TYPE_NONE = 0, |
|
GPS_TYPE_AUTO = 1, |
|
GPS_TYPE_UBLOX = 2, |
|
GPS_TYPE_MTK = 3, |
|
GPS_TYPE_MTK19 = 4, |
|
GPS_TYPE_NMEA = 5, |
|
GPS_TYPE_SIRF = 6, |
|
GPS_TYPE_HIL = 7, |
|
GPS_TYPE_SBP = 8, |
|
GPS_TYPE_PX4 = 9 |
|
}; |
|
|
|
/// GPS status codes |
|
enum GPS_Status { |
|
NO_GPS = 0, ///< No GPS connected/detected |
|
NO_FIX = 1, ///< Receiving valid GPS messages but no lock |
|
GPS_OK_FIX_2D = 2, ///< Receiving valid messages and 2D lock |
|
GPS_OK_FIX_3D = 3, ///< Receiving valid messages and 3D lock |
|
GPS_OK_FIX_3D_DGPS = 4, ///< Receiving valid messages and 3D lock with differential improvements |
|
GPS_OK_FIX_3D_RTK = 5, ///< Receiving valid messages and 3D lock, with relative-positioning improvements |
|
}; |
|
|
|
// GPS navigation engine settings. Not all GPS receivers support |
|
// this |
|
enum GPS_Engine_Setting { |
|
GPS_ENGINE_NONE = -1, |
|
GPS_ENGINE_PORTABLE = 0, |
|
GPS_ENGINE_STATIONARY = 2, |
|
GPS_ENGINE_PEDESTRIAN = 3, |
|
GPS_ENGINE_AUTOMOTIVE = 4, |
|
GPS_ENGINE_SEA = 5, |
|
GPS_ENGINE_AIRBORNE_1G = 6, |
|
GPS_ENGINE_AIRBORNE_2G = 7, |
|
GPS_ENGINE_AIRBORNE_4G = 8 |
|
}; |
|
|
|
/* |
|
The GPS_State structure is filled in by the backend driver as it |
|
parses each message from the GPS. |
|
*/ |
|
struct GPS_State { |
|
uint8_t instance; // the instance number of this GPS |
|
|
|
// all the following fields must all be filled by the backend driver |
|
GPS_Status status; ///< driver fix status |
|
uint32_t time_week_ms; ///< GPS time (milliseconds from start of GPS week) |
|
uint16_t time_week; ///< GPS week number |
|
Location location; ///< last fix location |
|
float ground_speed; ///< ground speed in m/sec |
|
int32_t ground_course_cd; ///< ground course in 100ths of a degree |
|
uint16_t hdop; ///< horizontal dilution of precision in cm |
|
uint8_t num_sats; ///< Number of visible satelites |
|
Vector3f velocity; ///< 3D velocitiy in m/s, in NED format |
|
float speed_accuracy; |
|
float horizontal_accuracy; |
|
float vertical_accuracy; |
|
bool have_vertical_velocity:1; ///< does this GPS give vertical velocity? |
|
bool have_speed_accuracy:1; |
|
bool have_horizontal_accuracy:1; |
|
bool have_vertical_accuracy:1; |
|
uint32_t last_gps_time_ms; ///< the system time we got the last GPS timestamp, milliseconds |
|
}; |
|
|
|
// Accessor functions |
|
|
|
// return number of active GPS sensors. Note that if the first GPS |
|
// is not present but the 2nd is then we return 2 |
|
uint8_t num_sensors(void) const { |
|
return num_instances; |
|
} |
|
|
|
uint8_t primary_sensor(void) const { |
|
return primary_instance; |
|
} |
|
|
|
// using these macros saves some code space on APM2 |
|
#if GPS_MAX_INSTANCES == 1 |
|
# define _GPS_STATE(instance) state[0] |
|
# define _GPS_TIMING(instance) timing[0] |
|
#else |
|
# define _GPS_STATE(instance) state[instance] |
|
# define _GPS_TIMING(instance) timing[instance] |
|
#endif |
|
|
|
/// Query GPS status |
|
GPS_Status status(uint8_t instance) const { |
|
return _GPS_STATE(instance).status; |
|
} |
|
GPS_Status status(void) const { |
|
return status(primary_instance); |
|
} |
|
|
|
// Query the highest status this GPS supports |
|
GPS_Status highest_supported_status(uint8_t instance) const; |
|
GPS_Status highest_supported_status(void) const; |
|
|
|
// location of last fix |
|
const Location &location(uint8_t instance) const { |
|
return _GPS_STATE(instance).location; |
|
} |
|
const Location &location() const { |
|
return location(primary_instance); |
|
} |
|
|
|
bool speed_accuracy(uint8_t instance, float &sacc) const { |
|
if(_GPS_STATE(instance).have_speed_accuracy) { |
|
sacc = _GPS_STATE(instance).speed_accuracy; |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
bool speed_accuracy(float &sacc) const { |
|
return speed_accuracy(primary_instance, sacc); |
|
} |
|
|
|
bool horizontal_accuracy(uint8_t instance, float &hacc) const { |
|
if(_GPS_STATE(instance).have_horizontal_accuracy) { |
|
hacc = _GPS_STATE(instance).horizontal_accuracy; |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
bool horizontal_accuracy(float &hacc) const { |
|
return horizontal_accuracy(primary_instance, hacc); |
|
} |
|
|
|
bool vertical_accuracy(uint8_t instance, float &vacc) const { |
|
if(_GPS_STATE(instance).have_vertical_accuracy) { |
|
vacc = _GPS_STATE(instance).vertical_accuracy; |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
bool vertical_accuracy(float &vacc) const { |
|
return vertical_accuracy(primary_instance, vacc); |
|
} |
|
|
|
// 3D velocity in NED format |
|
const Vector3f &velocity(uint8_t instance) const { |
|
return _GPS_STATE(instance).velocity; |
|
} |
|
const Vector3f &velocity() const { |
|
return velocity(primary_instance); |
|
} |
|
|
|
// ground speed in m/s |
|
float ground_speed(uint8_t instance) const { |
|
return _GPS_STATE(instance).ground_speed; |
|
} |
|
float ground_speed() const { |
|
return ground_speed(primary_instance); |
|
} |
|
|
|
// ground speed in cm/s |
|
uint32_t ground_speed_cm(void) { |
|
return ground_speed() * 100; |
|
} |
|
|
|
// ground course in centidegrees |
|
int32_t ground_course_cd(uint8_t instance) const { |
|
return _GPS_STATE(instance).ground_course_cd; |
|
} |
|
int32_t ground_course_cd() const { |
|
return ground_course_cd(primary_instance); |
|
} |
|
|
|
// number of locked satellites |
|
uint8_t num_sats(uint8_t instance) const { |
|
return _GPS_STATE(instance).num_sats; |
|
} |
|
uint8_t num_sats() const { |
|
return num_sats(primary_instance); |
|
} |
|
|
|
// GPS time of week in milliseconds |
|
uint32_t time_week_ms(uint8_t instance) const { |
|
return _GPS_STATE(instance).time_week_ms; |
|
} |
|
uint32_t time_week_ms() const { |
|
return time_week_ms(primary_instance); |
|
} |
|
|
|
// GPS week |
|
uint16_t time_week(uint8_t instance) const { |
|
return _GPS_STATE(instance).time_week; |
|
} |
|
uint16_t time_week() const { |
|
return time_week(primary_instance); |
|
} |
|
|
|
// horizontal dilution of precision |
|
uint16_t get_hdop(uint8_t instance) const { |
|
return _GPS_STATE(instance).hdop; |
|
} |
|
uint16_t get_hdop() const { |
|
return get_hdop(primary_instance); |
|
} |
|
|
|
// the time we got our last fix in system milliseconds. This is |
|
// used when calculating how far we might have moved since that fix |
|
uint32_t last_fix_time_ms(uint8_t instance) const { |
|
return _GPS_TIMING(instance).last_fix_time_ms; |
|
} |
|
uint32_t last_fix_time_ms(void) const { |
|
return last_fix_time_ms(primary_instance); |
|
} |
|
|
|
// the time we last processed a message in milliseconds. This is |
|
// used to indicate that we have new GPS data to process |
|
uint32_t last_message_time_ms(uint8_t instance) const { |
|
return _GPS_TIMING(instance).last_message_time_ms; |
|
} |
|
uint32_t last_message_time_ms(void) const { |
|
return last_message_time_ms(primary_instance); |
|
} |
|
|
|
// return last fix time since the 1/1/1970 in microseconds |
|
uint64_t time_epoch_usec(uint8_t instance); |
|
uint64_t time_epoch_usec(void) { |
|
return time_epoch_usec(primary_instance); |
|
} |
|
|
|
// return true if the GPS supports vertical velocity values |
|
bool have_vertical_velocity(uint8_t instance) const { |
|
return _GPS_STATE(instance).have_vertical_velocity; |
|
} |
|
bool have_vertical_velocity(void) const { |
|
return have_vertical_velocity(primary_instance); |
|
} |
|
|
|
// the expected lag (in seconds) in the position and velocity readings from the gps |
|
float get_lag() const { return 0.2f; } |
|
|
|
// set position for HIL |
|
void setHIL(uint8_t instance, GPS_Status status, uint64_t time_epoch_ms, |
|
const Location &location, const Vector3f &velocity, uint8_t num_sats, |
|
uint16_t hdop, bool _have_vertical_velocity); |
|
|
|
static const struct AP_Param::GroupInfo var_info[]; |
|
|
|
// dataflash for logging, if available |
|
DataFlash_Class *_DataFlash; |
|
|
|
// configuration parameters |
|
AP_Int8 _type[GPS_MAX_INSTANCES]; |
|
AP_Int8 _navfilter; |
|
#if GPS_MAX_INSTANCES > 1 |
|
AP_Int8 _auto_switch; |
|
AP_Int8 _min_dgps; |
|
#endif |
|
AP_Int8 _sbas_mode; |
|
AP_Int8 _min_elevation; |
|
|
|
// handle sending of initialisation strings to the GPS |
|
void send_blob_start(uint8_t instance, const prog_char *_blob, uint16_t size); |
|
void send_blob_update(uint8_t instance); |
|
|
|
// lock out a GPS port, allowing another application to use the port |
|
void lock_port(uint8_t instance, bool locked); |
|
|
|
//MAVLink Status Sending |
|
void send_mavlink_gps_raw(mavlink_channel_t chan); |
|
#if GPS_MAX_INSTANCES > 1 |
|
void send_mavlink_gps2_raw(mavlink_channel_t chan); |
|
#endif |
|
|
|
#if GPS_RTK_AVAILABLE |
|
void send_mavlink_gps_rtk(mavlink_channel_t chan); |
|
#if GPS_MAX_INSTANCES > 1 |
|
void send_mavlink_gps2_rtk(mavlink_channel_t chan); |
|
#endif |
|
#endif |
|
|
|
private: |
|
struct GPS_timing { |
|
// the time we got our last fix in system milliseconds |
|
uint32_t last_fix_time_ms; |
|
|
|
// the time we got our last fix in system milliseconds |
|
uint32_t last_message_time_ms; |
|
}; |
|
GPS_timing timing[GPS_MAX_INSTANCES]; |
|
GPS_State state[GPS_MAX_INSTANCES]; |
|
AP_GPS_Backend *drivers[GPS_MAX_INSTANCES]; |
|
|
|
/// primary GPS instance |
|
uint8_t primary_instance:2; |
|
|
|
/// number of GPS instances present |
|
uint8_t num_instances:2; |
|
|
|
// which ports are locked |
|
uint8_t locked_ports:2; |
|
|
|
// state of auto-detection process, per instance |
|
struct detect_state { |
|
uint32_t detect_started_ms; |
|
uint32_t last_baud_change_ms; |
|
uint8_t last_baud; |
|
struct UBLOX_detect_state ublox_detect_state; |
|
struct MTK_detect_state mtk_detect_state; |
|
struct MTK19_detect_state mtk19_detect_state; |
|
struct SIRF_detect_state sirf_detect_state; |
|
struct NMEA_detect_state nmea_detect_state; |
|
#if GPS_RTK_AVAILABLE |
|
struct SBP_detect_state sbp_detect_state; |
|
#endif |
|
} detect_state[GPS_MAX_INSTANCES]; |
|
|
|
struct { |
|
const prog_char *blob; |
|
uint16_t remaining; |
|
} initblob_state[GPS_MAX_INSTANCES]; |
|
|
|
static const uint32_t _baudrates[]; |
|
static const prog_char _initialisation_blob[]; |
|
|
|
void detect_instance(uint8_t instance); |
|
void update_instance(uint8_t instance); |
|
}; |
|
|
|
#include <GPS_Backend.h> |
|
#include <AP_GPS_UBLOX.h> |
|
#include <AP_GPS_MTK.h> |
|
#include <AP_GPS_MTK19.h> |
|
#include <AP_GPS_NMEA.h> |
|
#include <AP_GPS_SIRF.h> |
|
#include <AP_GPS_SBP.h> |
|
#include <AP_GPS_PX4.h> |
|
|
|
#endif // __AP_GPS_H__
|
|
|