mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-19 23:28:32 -04:00
273 lines
7.0 KiB
C++
273 lines
7.0 KiB
C++
/*
|
|
* Location.cpp
|
|
*/
|
|
|
|
#include "Location.h"
|
|
|
|
#include <AP_AHRS/AP_AHRS.h>
|
|
#include <AP_Terrain/AP_Terrain.h>
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
AP_Terrain *Location::_terrain = nullptr;
|
|
|
|
/// constructors
|
|
Location::Location()
|
|
{
|
|
zero();
|
|
}
|
|
|
|
const Location definitely_zero{};
|
|
bool Location::is_zero(void) const
|
|
{
|
|
return !memcmp(this, &definitely_zero, sizeof(*this));
|
|
}
|
|
|
|
void Location::zero(void)
|
|
{
|
|
memset(this, 0, sizeof(*this));
|
|
}
|
|
|
|
Location::Location(int32_t latitude, int32_t longitude, int32_t alt_in_cm, AltFrame frame)
|
|
{
|
|
zero();
|
|
lat = latitude;
|
|
lng = longitude;
|
|
set_alt_cm(alt_in_cm, frame);
|
|
}
|
|
|
|
Location::Location(const Vector3f &ekf_offset_neu)
|
|
{
|
|
// store alt and alt frame
|
|
set_alt_cm(ekf_offset_neu.z, AltFrame::ABOVE_ORIGIN);
|
|
|
|
// calculate lat, lon
|
|
Location ekf_origin;
|
|
if (AP::ahrs().get_origin(ekf_origin)) {
|
|
lat = ekf_origin.lat;
|
|
lng = ekf_origin.lng;
|
|
offset(ekf_offset_neu.x / 100.0f, ekf_offset_neu.y / 100.0f);
|
|
}
|
|
}
|
|
|
|
void Location::set_alt_cm(int32_t alt_cm, AltFrame frame)
|
|
{
|
|
alt = alt_cm;
|
|
relative_alt = false;
|
|
terrain_alt = false;
|
|
origin_alt = false;
|
|
switch (frame) {
|
|
case AltFrame::ABSOLUTE:
|
|
// do nothing
|
|
break;
|
|
case AltFrame::ABOVE_HOME:
|
|
relative_alt = true;
|
|
break;
|
|
case AltFrame::ABOVE_ORIGIN:
|
|
origin_alt = true;
|
|
break;
|
|
case AltFrame::ABOVE_TERRAIN:
|
|
// we mark it as a relative altitude, as it doesn't have
|
|
// home alt added
|
|
relative_alt = true;
|
|
terrain_alt = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// converts altitude to new frame
|
|
bool Location::change_alt_frame(AltFrame desired_frame)
|
|
{
|
|
int32_t new_alt_cm;
|
|
if (!get_alt_cm(desired_frame, new_alt_cm)) {
|
|
return false;
|
|
}
|
|
set_alt_cm(new_alt_cm, desired_frame);
|
|
return true;
|
|
}
|
|
|
|
// get altitude frame
|
|
Location::AltFrame Location::get_alt_frame() const
|
|
{
|
|
if (terrain_alt) {
|
|
return AltFrame::ABOVE_TERRAIN;
|
|
}
|
|
if (origin_alt) {
|
|
return AltFrame::ABOVE_ORIGIN;
|
|
}
|
|
if (relative_alt) {
|
|
return AltFrame::ABOVE_HOME;
|
|
}
|
|
return AltFrame::ABSOLUTE;
|
|
}
|
|
|
|
/// get altitude in desired frame
|
|
bool Location::get_alt_cm(AltFrame desired_frame, int32_t &ret_alt_cm) const
|
|
{
|
|
Location::AltFrame frame = get_alt_frame();
|
|
|
|
// shortcut if desired and underlying frame are the same
|
|
if (desired_frame == frame) {
|
|
ret_alt_cm = alt;
|
|
return true;
|
|
}
|
|
|
|
// check for terrain altitude
|
|
float alt_terr_cm = 0;
|
|
if (frame == AltFrame::ABOVE_TERRAIN || desired_frame == AltFrame::ABOVE_TERRAIN) {
|
|
#if AP_TERRAIN_AVAILABLE
|
|
if (_terrain == nullptr || !_terrain->height_amsl(*(Location *)this, alt_terr_cm, true)) {
|
|
return false;
|
|
}
|
|
// convert terrain alt to cm
|
|
alt_terr_cm *= 100.0f;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
// convert alt to absolute
|
|
int32_t alt_abs = 0;
|
|
switch (frame) {
|
|
case AltFrame::ABSOLUTE:
|
|
alt_abs = alt;
|
|
break;
|
|
case AltFrame::ABOVE_HOME:
|
|
if (!AP::ahrs().home_is_set()) {
|
|
return false;
|
|
}
|
|
alt_abs = alt + AP::ahrs().get_home().alt;
|
|
break;
|
|
case AltFrame::ABOVE_ORIGIN:
|
|
{
|
|
// fail if we cannot get ekf origin
|
|
Location ekf_origin;
|
|
if (!AP::ahrs().get_origin(ekf_origin)) {
|
|
return false;
|
|
}
|
|
alt_abs = alt + ekf_origin.alt;
|
|
}
|
|
break;
|
|
case AltFrame::ABOVE_TERRAIN:
|
|
alt_abs = alt + alt_terr_cm;
|
|
break;
|
|
}
|
|
|
|
// convert absolute to desired frame
|
|
switch (desired_frame) {
|
|
case AltFrame::ABSOLUTE:
|
|
ret_alt_cm = alt_abs;
|
|
return true;
|
|
case AltFrame::ABOVE_HOME:
|
|
if (!AP::ahrs().home_is_set()) {
|
|
return false;
|
|
}
|
|
ret_alt_cm = alt_abs - AP::ahrs().get_home().alt;
|
|
return true;
|
|
case AltFrame::ABOVE_ORIGIN:
|
|
{
|
|
// fail if we cannot get ekf origin
|
|
Location ekf_origin;
|
|
if (!AP::ahrs().get_origin(ekf_origin)) {
|
|
return false;
|
|
}
|
|
ret_alt_cm = alt_abs - ekf_origin.alt;
|
|
return true;
|
|
}
|
|
case AltFrame::ABOVE_TERRAIN:
|
|
ret_alt_cm = alt_abs - alt_terr_cm;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool Location::get_vector_xy_from_origin_NE(Vector2f &vec_ne) const
|
|
{
|
|
Location ekf_origin;
|
|
if (!AP::ahrs().get_origin(ekf_origin)) {
|
|
return false;
|
|
}
|
|
vec_ne.x = (lat-ekf_origin.lat) * LATLON_TO_CM;
|
|
vec_ne.y = (lng-ekf_origin.lng) * LATLON_TO_CM * ekf_origin.longitude_scale();
|
|
return true;
|
|
}
|
|
|
|
bool Location::get_vector_from_origin_NEU(Vector3f &vec_neu) const
|
|
{
|
|
// convert lat, lon
|
|
Vector2f vec_ne;
|
|
if (!get_vector_xy_from_origin_NE(vec_ne)) {
|
|
return false;
|
|
}
|
|
vec_neu.x = vec_ne.x;
|
|
vec_neu.y = vec_ne.y;
|
|
|
|
// convert altitude
|
|
int32_t alt_above_origin_cm = 0;
|
|
if (!get_alt_cm(AltFrame::ABOVE_ORIGIN, alt_above_origin_cm)) {
|
|
return false;
|
|
}
|
|
vec_neu.z = alt_above_origin_cm;
|
|
|
|
return true;
|
|
}
|
|
|
|
// return distance in meters between two locations
|
|
float Location::get_distance(const struct Location &loc2) const
|
|
{
|
|
float dlat = (float)(loc2.lat - lat);
|
|
float dlng = ((float)(loc2.lng - lng)) * loc2.longitude_scale();
|
|
return norm(dlat, dlng) * LOCATION_SCALING_FACTOR;
|
|
}
|
|
|
|
// extrapolate latitude/longitude given distances (in meters) north and east
|
|
void Location::offset(float ofs_north, float ofs_east)
|
|
{
|
|
// use is_equal() because is_zero() is a local class conflict and is_zero() in AP_Math does not belong to a class
|
|
if (!is_equal(ofs_north, 0.0f) || !is_equal(ofs_east, 0.0f)) {
|
|
int32_t dlat = ofs_north * LOCATION_SCALING_FACTOR_INV;
|
|
int32_t dlng = (ofs_east * LOCATION_SCALING_FACTOR_INV) / longitude_scale();
|
|
lat += dlat;
|
|
lng += dlng;
|
|
}
|
|
}
|
|
|
|
float Location::longitude_scale() const
|
|
{
|
|
float scale = cosf(lat * (1.0e-7f * DEG_TO_RAD));
|
|
return constrain_float(scale, 0.01f, 1.0f);
|
|
}
|
|
|
|
/*
|
|
* convert invalid waypoint with useful data. return true if location changed
|
|
*/
|
|
bool Location::sanitize(const Location &defaultLoc)
|
|
{
|
|
bool has_changed = false;
|
|
// convert lat/lng=0 to mean current point
|
|
if (lat == 0 && lng == 0) {
|
|
lat = defaultLoc.lat;
|
|
lng = defaultLoc.lng;
|
|
has_changed = true;
|
|
}
|
|
|
|
// convert relative alt=0 to mean current alt
|
|
if (alt == 0 && relative_alt) {
|
|
relative_alt = false;
|
|
alt = defaultLoc.alt;
|
|
has_changed = true;
|
|
}
|
|
|
|
// limit lat/lng to appropriate ranges
|
|
if (!check_latlng(lat, lng)) {
|
|
lat = defaultLoc.lat;
|
|
lng = defaultLoc.lng;
|
|
has_changed = true;
|
|
}
|
|
|
|
return has_changed;
|
|
}
|
|
|
|
// make sure we know what size the Location object is:
|
|
assert_storage_size<Location, 16> _assert_storage_size_Location;
|