mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-03 14:38:30 -04:00
AP_Mission: support storage on microSD using mission.stg
This commit is contained in:
parent
182ab996b1
commit
9cef639a97
@ -53,6 +53,31 @@ HAL_Semaphore AP_Mission::_rsem;
|
|||||||
/// init - initialises this library including checks the version in eeprom matches this library
|
/// init - initialises this library including checks the version in eeprom matches this library
|
||||||
void AP_Mission::init()
|
void AP_Mission::init()
|
||||||
{
|
{
|
||||||
|
#if AP_SDCARD_STORAGE_ENABLED
|
||||||
|
// check for extra storage on microsd
|
||||||
|
const auto *bc = AP::boardConfig();
|
||||||
|
if (bc != nullptr) {
|
||||||
|
const auto size_kb = bc->get_sdcard_mission_kb();
|
||||||
|
if (size_kb > 0) {
|
||||||
|
_failed_sdcard_storage = !_storage.attach_file(AP_MISSION_SDCARD_FILENAME, size_kb);
|
||||||
|
if (_failed_sdcard_storage) {
|
||||||
|
// wipe mission if storage not available, but don't save. This allows sdcard error to be fixed and reboot
|
||||||
|
_cmd_total.set(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// work out maximum index for our storage size
|
||||||
|
if (_storage.size() >= AP_MISSION_EEPROM_COMMAND_SIZE+4) {
|
||||||
|
_commands_max = (_storage.size()-4U) / AP_MISSION_EEPROM_COMMAND_SIZE;
|
||||||
|
}
|
||||||
|
if (_cmd_total.get() > _commands_max) {
|
||||||
|
// wipe mission if storage not available, but don't save. This allows sdcard error to be fixed and reboot
|
||||||
|
_cmd_total.set(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// check_eeprom_version - checks version of missions stored in eeprom matches this library
|
// check_eeprom_version - checks version of missions stored in eeprom matches this library
|
||||||
// command list will be cleared if they do not match
|
// command list will be cleared if they do not match
|
||||||
check_eeprom_version();
|
check_eeprom_version();
|
||||||
@ -600,7 +625,7 @@ bool AP_Mission::restart_current_nav_cmd()
|
|||||||
bool AP_Mission::set_item(uint16_t index, mavlink_mission_item_int_t& src_packet)
|
bool AP_Mission::set_item(uint16_t index, mavlink_mission_item_int_t& src_packet)
|
||||||
{
|
{
|
||||||
// this is the on-storage format
|
// this is the on-storage format
|
||||||
AP_Mission::Mission_Command cmd;
|
AP_Mission::Mission_Command cmd {};
|
||||||
|
|
||||||
// can't handle request for anything bigger than the mission size+1...
|
// can't handle request for anything bigger than the mission size+1...
|
||||||
if (index > num_commands()) {
|
if (index > num_commands()) {
|
||||||
@ -628,7 +653,7 @@ bool AP_Mission::get_item(uint16_t index, mavlink_mission_item_int_t& ret_packet
|
|||||||
// means it contains invalid data after it leaves here.
|
// means it contains invalid data after it leaves here.
|
||||||
|
|
||||||
// this is the on-storage format
|
// this is the on-storage format
|
||||||
AP_Mission::Mission_Command cmd;
|
AP_Mission::Mission_Command cmd {};
|
||||||
|
|
||||||
// can't handle request for anything bigger than the mission size...
|
// can't handle request for anything bigger than the mission size...
|
||||||
if (index >= num_commands()) {
|
if (index >= num_commands()) {
|
||||||
@ -718,8 +743,7 @@ bool AP_Mission::read_cmd_from_storage(uint16_t index, Mission_Command& cmd) con
|
|||||||
cmd.content.location = AP::ahrs().get_home();
|
cmd.content.location = AP::ahrs().get_home();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
if (index >= (unsigned)_cmd_total || index >= _commands_max) {
|
||||||
if (index >= (unsigned)_cmd_total) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2181,15 +2205,6 @@ void AP_Mission::check_eeprom_version()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
return total number of commands that can fit in storage space
|
|
||||||
*/
|
|
||||||
uint16_t AP_Mission::num_commands_max(void) const
|
|
||||||
{
|
|
||||||
// -4 to remove space for eeprom version number
|
|
||||||
return (_storage.size() - 4) / AP_MISSION_EEPROM_COMMAND_SIZE;
|
|
||||||
}
|
|
||||||
|
|
||||||
// find the nearest landing sequence starting point (DO_LAND_START) and
|
// find the nearest landing sequence starting point (DO_LAND_START) and
|
||||||
// return its index. Returns 0 if no appropriate DO_LAND_START point can
|
// return its index. Returns 0 if no appropriate DO_LAND_START point can
|
||||||
// be found.
|
// be found.
|
||||||
@ -2206,7 +2221,11 @@ uint16_t AP_Mission::get_landing_sequence_start()
|
|||||||
float min_distance = -1;
|
float min_distance = -1;
|
||||||
|
|
||||||
// Go through mission looking for nearest landing start command
|
// Go through mission looking for nearest landing start command
|
||||||
for (uint16_t i = 1; i < num_commands(); i++) {
|
const auto count = num_commands();
|
||||||
|
for (uint16_t i = 1; i < count; i++) {
|
||||||
|
if (get_command_id(i) != uint16_t(MAV_CMD_DO_LAND_START)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
Mission_Command tmp;
|
Mission_Command tmp;
|
||||||
if (!read_cmd_from_storage(i, tmp)) {
|
if (!read_cmd_from_storage(i, tmp)) {
|
||||||
continue;
|
continue;
|
||||||
@ -2269,7 +2288,11 @@ bool AP_Mission::jump_to_abort_landing_sequence(void)
|
|||||||
if (AP::ahrs().get_location(current_loc)) {
|
if (AP::ahrs().get_location(current_loc)) {
|
||||||
float min_distance = FLT_MAX;
|
float min_distance = FLT_MAX;
|
||||||
|
|
||||||
for (uint16_t i = 1; i < num_commands(); i++) {
|
const auto count = num_commands();
|
||||||
|
for (uint16_t i = 1; i < count; i++) {
|
||||||
|
if (get_command_id(i) != uint16_t(MAV_CMD_DO_GO_AROUND)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
Mission_Command tmp;
|
Mission_Command tmp;
|
||||||
if (!read_cmd_from_storage(i, tmp)) {
|
if (!read_cmd_from_storage(i, tmp)) {
|
||||||
continue;
|
continue;
|
||||||
@ -2566,9 +2589,36 @@ const char *AP_Mission::Mission_Command::type() const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
get the command ID of a mission index. Caller should have checked the index is in range
|
||||||
|
*/
|
||||||
|
uint16_t AP_Mission::get_command_id(uint16_t index) const
|
||||||
|
{
|
||||||
|
const uint16_t pos_in_storage = 4 + (index * AP_MISSION_EEPROM_COMMAND_SIZE);
|
||||||
|
uint8_t b[3] {};
|
||||||
|
if (!_storage.read_block(b, pos_in_storage, sizeof(b))) {
|
||||||
|
return 0U;
|
||||||
|
}
|
||||||
|
uint16_t id = 0;
|
||||||
|
if (b[0] == 0 || b[0] == 1) {
|
||||||
|
memcpy((void*)&id, (void*)&b[1], 2);
|
||||||
|
} else {
|
||||||
|
id = b[0];
|
||||||
|
}
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
see if the mission contains a particular item
|
||||||
|
*/
|
||||||
bool AP_Mission::contains_item(MAV_CMD command) const
|
bool AP_Mission::contains_item(MAV_CMD command) const
|
||||||
{
|
{
|
||||||
for (uint16_t i = 1; i < num_commands(); i++) {
|
const auto count = num_commands();
|
||||||
|
for (uint16_t i = 1; i < count; i++) {
|
||||||
|
if (get_command_id(i) != uint16_t(command)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// confirm with full read
|
||||||
Mission_Command tmp;
|
Mission_Command tmp;
|
||||||
if (!read_cmd_from_storage(i, tmp)) {
|
if (!read_cmd_from_storage(i, tmp)) {
|
||||||
continue;
|
continue;
|
||||||
@ -2603,7 +2653,11 @@ bool AP_Mission::contains_terrain_alt_items(void)
|
|||||||
|
|
||||||
bool AP_Mission::calculate_contains_terrain_alt_items(void) const
|
bool AP_Mission::calculate_contains_terrain_alt_items(void) const
|
||||||
{
|
{
|
||||||
for (uint16_t i = 1; i < num_commands(); i++) {
|
const auto count = num_commands();
|
||||||
|
for (uint16_t i = 1; i < count; i++) {
|
||||||
|
if (!stored_in_location(get_command_id(i))) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
Mission_Command tmp;
|
Mission_Command tmp;
|
||||||
if (!read_cmd_from_storage(i, tmp)) {
|
if (!read_cmd_from_storage(i, tmp)) {
|
||||||
continue;
|
continue;
|
||||||
|
@ -52,6 +52,12 @@
|
|||||||
#define AP_MISSION_MAX_WP_HISTORY 7 // The maximum number of previous wp commands that will be stored from the active missions history
|
#define AP_MISSION_MAX_WP_HISTORY 7 // The maximum number of previous wp commands that will be stored from the active missions history
|
||||||
#define LAST_WP_PASSED (AP_MISSION_MAX_WP_HISTORY-2)
|
#define LAST_WP_PASSED (AP_MISSION_MAX_WP_HISTORY-2)
|
||||||
|
|
||||||
|
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||||
|
#define AP_MISSION_SDCARD_FILENAME "APM/mission.stg"
|
||||||
|
#else
|
||||||
|
#define AP_MISSION_SDCARD_FILENAME "mission.stg"
|
||||||
|
#endif
|
||||||
|
|
||||||
union PackedContent;
|
union PackedContent;
|
||||||
|
|
||||||
/// @class AP_Mission
|
/// @class AP_Mission
|
||||||
@ -431,7 +437,9 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// num_commands_max - returns maximum number of commands that can be stored
|
/// num_commands_max - returns maximum number of commands that can be stored
|
||||||
uint16_t num_commands_max() const;
|
uint16_t num_commands_max() const {
|
||||||
|
return _commands_max;
|
||||||
|
}
|
||||||
|
|
||||||
/// start - resets current commands to point to the beginning of the mission
|
/// start - resets current commands to point to the beginning of the mission
|
||||||
/// To-Do: should we validate the mission first and return true/false?
|
/// To-Do: should we validate the mission first and return true/false?
|
||||||
@ -674,6 +682,12 @@ public:
|
|||||||
// Returns 0 if no appropriate JUMP_TAG match can be found.
|
// Returns 0 if no appropriate JUMP_TAG match can be found.
|
||||||
uint16_t get_index_of_jump_tag(const uint16_t tag) const;
|
uint16_t get_index_of_jump_tag(const uint16_t tag) const;
|
||||||
|
|
||||||
|
#if AP_SDCARD_STORAGE_ENABLED
|
||||||
|
bool failed_sdcard_storage(void) const {
|
||||||
|
return _failed_sdcard_storage;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static AP_Mission *_singleton;
|
static AP_Mission *_singleton;
|
||||||
|
|
||||||
@ -803,6 +817,16 @@ private:
|
|||||||
uint32_t _last_change_time_ms;
|
uint32_t _last_change_time_ms;
|
||||||
uint32_t _last_change_time_prev_ms;
|
uint32_t _last_change_time_prev_ms;
|
||||||
|
|
||||||
|
// maximum number of commands that will fit in storage
|
||||||
|
uint16_t _commands_max;
|
||||||
|
|
||||||
|
#if AP_SDCARD_STORAGE_ENABLED
|
||||||
|
bool _failed_sdcard_storage;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// fast call to get command ID of a mission index
|
||||||
|
uint16_t get_command_id(uint16_t index) const;
|
||||||
|
|
||||||
// memoisation of contains-relative:
|
// memoisation of contains-relative:
|
||||||
bool _contains_terrain_alt_items; // true if the mission has terrain-relative items
|
bool _contains_terrain_alt_items; // true if the mission has terrain-relative items
|
||||||
uint32_t _last_contains_relative_calculated_ms; // will be equal to _last_change_time_ms if _contains_terrain_alt_items is up-to-date
|
uint32_t _last_contains_relative_calculated_ms; // will be equal to _last_change_time_ms if _contains_terrain_alt_items is up-to-date
|
||||||
|
Loading…
Reference in New Issue
Block a user