AP_Mission: support storage on microSD using mission.stg

This commit is contained in:
Andrew Tridgell 2023-02-14 20:39:36 +11:00 committed by Peter Barker
parent 182ab996b1
commit 9cef639a97
2 changed files with 96 additions and 18 deletions

View File

@ -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;

View File

@ -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