2013-09-22 01:45:24 -03:00
|
|
|
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
/*
|
|
|
|
* rally point support
|
|
|
|
* Michael Day, September 2013
|
|
|
|
*/
|
|
|
|
|
|
|
|
//get a rally point
|
2013-10-02 23:20:54 -03:00
|
|
|
static bool get_rally_point_with_index(unsigned i, RallyLocation &ret)
|
2013-09-22 01:45:24 -03:00
|
|
|
{
|
2013-10-02 23:20:54 -03:00
|
|
|
if (i >= (unsigned)g.rally_total) {
|
|
|
|
return false;
|
2013-09-22 01:45:24 -03:00
|
|
|
}
|
2013-10-02 23:20:54 -03:00
|
|
|
hal.storage->read_block(&ret, RALLY_START_BYTE + (i * sizeof(RallyLocation)), sizeof(RallyLocation));
|
|
|
|
if (ret.lat == 0 && ret.lng == 0) {
|
|
|
|
// sanity check ...
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2013-09-22 01:45:24 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
//save a rally point
|
2013-10-02 23:20:54 -03:00
|
|
|
static bool set_rally_point_with_index(unsigned i, const RallyLocation &rallyLoc)
|
2013-09-22 01:45:24 -03:00
|
|
|
{
|
2013-10-02 23:20:54 -03:00
|
|
|
if (i >= (unsigned) g.rally_total) {
|
2013-09-22 01:45:24 -03:00
|
|
|
//not allowed
|
2013-10-02 23:20:54 -03:00
|
|
|
return false;
|
2013-09-22 01:45:24 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (i >= MAX_RALLYPOINTS) {
|
|
|
|
//also not allowed
|
2013-10-02 23:20:54 -03:00
|
|
|
return false;
|
2013-09-22 01:45:24 -03:00
|
|
|
}
|
2013-10-02 23:20:54 -03:00
|
|
|
hal.storage->write_block(RALLY_START_BYTE + (i * sizeof(RallyLocation)), &rallyLoc, sizeof(RallyLocation));
|
|
|
|
return true;
|
2013-09-22 01:45:24 -03:00
|
|
|
}
|
|
|
|
|
2013-10-02 23:20:54 -03:00
|
|
|
// 'best' means 'closest to Location loc' for now.
|
|
|
|
static bool find_best_rally_point(const Location &myloc, const Location &homeloc, RallyLocation &ret)
|
|
|
|
{
|
|
|
|
float min_dis = -1;
|
2013-09-22 01:45:24 -03:00
|
|
|
|
2013-10-02 23:20:54 -03:00
|
|
|
for (unsigned i = 0; i < (unsigned) g.rally_total; i++) {
|
|
|
|
RallyLocation next_rally;
|
|
|
|
if (!get_rally_point_with_index(i, next_rally)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
Location rally_loc = rally_location_to_location(next_rally, homeloc);
|
|
|
|
float dis = get_distance(myloc, rally_loc);
|
2013-09-22 01:45:24 -03:00
|
|
|
|
2013-10-02 23:20:54 -03:00
|
|
|
if (dis < min_dis || min_dis < 0) {
|
2013-09-22 01:45:24 -03:00
|
|
|
min_dis = dis;
|
|
|
|
ret = next_rally;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-02 23:20:54 -03:00
|
|
|
return min_dis >= 0;
|
2013-09-22 01:45:24 -03:00
|
|
|
}
|
|
|
|
|
2013-10-02 23:20:54 -03:00
|
|
|
// translate a RallyLocation to a Location
|
|
|
|
static Location rally_location_to_location(const RallyLocation &r_loc, const Location &homeloc)
|
|
|
|
{
|
2013-09-22 01:45:24 -03:00
|
|
|
Location ret;
|
|
|
|
|
|
|
|
ret.id = MAV_CMD_NAV_LOITER_UNLIM;
|
|
|
|
ret.options = MASK_OPTIONS_RELATIVE_ALT;
|
|
|
|
|
|
|
|
//Currently can't do true AGL on the APM. Relative altitudes are
|
|
|
|
//relative to HOME point's altitude. Terrain on the board is inbound
|
|
|
|
//for the PX4, though. This line will need to be updated when that happens:
|
2013-10-02 23:20:54 -03:00
|
|
|
ret.alt = r_loc.alt + homeloc.alt;
|
2013-09-22 01:45:24 -03:00
|
|
|
|
|
|
|
ret.lat = r_loc.lat;
|
|
|
|
ret.lng = r_loc.lng;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|