mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-20 07:38:28 -04:00
AP_Logger: allow for retry of log open with LOG_DISARMED=1
if booting without a microSD and with LOG_DISARMED=1 then this allows for a retry of opening the log every 5s
This commit is contained in:
parent
ce839cfef9
commit
66596dcf25
@ -36,6 +36,9 @@ extern const AP_HAL::HAL& hal;
|
|||||||
#define MB_to_B 1000000
|
#define MB_to_B 1000000
|
||||||
#define B_to_MB 0.000001
|
#define B_to_MB 0.000001
|
||||||
|
|
||||||
|
// time between tries to open log
|
||||||
|
#define LOGGER_FILE_REOPEN_MS 5000
|
||||||
|
|
||||||
/*
|
/*
|
||||||
constructor
|
constructor
|
||||||
*/
|
*/
|
||||||
@ -134,6 +137,17 @@ void AP_Logger_File::periodic_1Hz()
|
|||||||
{
|
{
|
||||||
AP_Logger_Backend::periodic_1Hz();
|
AP_Logger_Backend::periodic_1Hz();
|
||||||
|
|
||||||
|
if (_initialised &&
|
||||||
|
_write_fd == -1 && _read_fd == -1 &&
|
||||||
|
logging_enabled() &&
|
||||||
|
!recent_open_error() &&
|
||||||
|
!hal.util->get_soft_armed()) {
|
||||||
|
// retry logging open. This allows for booting with
|
||||||
|
// LOG_DISARMED=1 with a bad microSD or no microSD. Once a
|
||||||
|
// card is inserted then logging starts
|
||||||
|
start_new_log();
|
||||||
|
}
|
||||||
|
|
||||||
if (!io_thread_alive()) {
|
if (!io_thread_alive()) {
|
||||||
if (io_thread_warning_decimation_counter == 0 && _initialised) {
|
if (io_thread_warning_decimation_counter == 0 && _initialised) {
|
||||||
// we don't print this error unless we did initialise. When _initialised is set to true
|
// we don't print this error unless we did initialise. When _initialised is set to true
|
||||||
@ -167,10 +181,18 @@ uint32_t AP_Logger_File::bufferspace_available()
|
|||||||
return (space > crit) ? space - crit : 0;
|
return (space > crit) ? space - crit : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool AP_Logger_File::recent_open_error(void) const
|
||||||
|
{
|
||||||
|
if (_open_error_ms == 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return AP_HAL::millis() - _open_error_ms < LOGGER_FILE_REOPEN_MS;
|
||||||
|
}
|
||||||
|
|
||||||
// return true for CardInserted() if we successfully initialized
|
// return true for CardInserted() if we successfully initialized
|
||||||
bool AP_Logger_File::CardInserted(void) const
|
bool AP_Logger_File::CardInserted(void) const
|
||||||
{
|
{
|
||||||
return _initialised && !_open_error;
|
return _initialised && !recent_open_error();
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns the amount of disk space available in _log_directory (in bytes)
|
// returns the amount of disk space available in _log_directory (in bytes)
|
||||||
@ -450,7 +472,7 @@ bool AP_Logger_File::WritesOK() const
|
|||||||
if (_write_fd == -1) {
|
if (_write_fd == -1) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (_open_error) {
|
if (recent_open_error()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
@ -459,7 +481,7 @@ bool AP_Logger_File::WritesOK() const
|
|||||||
|
|
||||||
bool AP_Logger_File::StartNewLogOK() const
|
bool AP_Logger_File::StartNewLogOK() const
|
||||||
{
|
{
|
||||||
if (_open_error) {
|
if (recent_open_error()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return AP_Logger_Backend::StartNewLogOK();
|
return AP_Logger_Backend::StartNewLogOK();
|
||||||
@ -559,7 +581,9 @@ uint32_t AP_Logger_File::_get_log_size(const uint16_t log_num)
|
|||||||
struct stat st;
|
struct stat st;
|
||||||
EXPECT_DELAY_MS(3000);
|
EXPECT_DELAY_MS(3000);
|
||||||
if (AP::FS().stat(fname, &st) != 0) {
|
if (AP::FS().stat(fname, &st) != 0) {
|
||||||
printf("Unable to fetch Log File Size: %s\n", strerror(errno));
|
if (_open_error_ms == 0) {
|
||||||
|
printf("Unable to fetch Log File Size (%s): %s\n", fname, strerror(errno));
|
||||||
|
}
|
||||||
free(fname);
|
free(fname);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -618,7 +642,7 @@ void AP_Logger_File::get_log_boundaries(const uint16_t list_entry, uint32_t & st
|
|||||||
*/
|
*/
|
||||||
int16_t AP_Logger_File::get_log_data(const uint16_t list_entry, const uint16_t page, const uint32_t offset, const uint16_t len, uint8_t *data)
|
int16_t AP_Logger_File::get_log_data(const uint16_t list_entry, const uint16_t page, const uint32_t offset, const uint16_t len, uint8_t *data)
|
||||||
{
|
{
|
||||||
if (!_initialised || _open_error) {
|
if (!_initialised || recent_open_error()) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -641,7 +665,7 @@ int16_t AP_Logger_File::get_log_data(const uint16_t list_entry, const uint16_t p
|
|||||||
EXPECT_DELAY_MS(3000);
|
EXPECT_DELAY_MS(3000);
|
||||||
_read_fd = AP::FS().open(fname, O_RDONLY);
|
_read_fd = AP::FS().open(fname, O_RDONLY);
|
||||||
if (_read_fd == -1) {
|
if (_read_fd == -1) {
|
||||||
_open_error = true;
|
_open_error_ms = AP_HAL::millis();
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
::printf("Log read open fail for %s - %s\n",
|
::printf("Log read open fail for %s - %s\n",
|
||||||
fname, strerror(saved_errno));
|
fname, strerror(saved_errno));
|
||||||
@ -737,19 +761,21 @@ void AP_Logger_File::stop_logging(void)
|
|||||||
*/
|
*/
|
||||||
void AP_Logger_File::start_new_log(void)
|
void AP_Logger_File::start_new_log(void)
|
||||||
{
|
{
|
||||||
if (_open_error) {
|
if (recent_open_error()) {
|
||||||
// we have previously failed to open a file - don't try again
|
// we have previously failed to open a file - don't try again
|
||||||
// to prevent us trying to open files while in flight
|
// to prevent us trying to open files while in flight
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const bool open_error_ms_was_zero = (_open_error_ms == 0);
|
||||||
|
|
||||||
// set _open_error here to avoid infinite recursion. Simply
|
// set _open_error here to avoid infinite recursion. Simply
|
||||||
// writing a prioritised block may try to open a log - which means
|
// writing a prioritised block may try to open a log - which means
|
||||||
// if anything in the start_new_log path does a gcs().send_text()
|
// if anything in the start_new_log path does a gcs().send_text()
|
||||||
// (for example), you will end up recursing if we don't take
|
// (for example), you will end up recursing if we don't take
|
||||||
// precautions. We will reset _open_error if we actually manage
|
// precautions. We will reset _open_error if we actually manage
|
||||||
// to open the log...
|
// to open the log...
|
||||||
_open_error = true;
|
_open_error_ms = AP_HAL::millis();
|
||||||
|
|
||||||
stop_logging();
|
stop_logging();
|
||||||
|
|
||||||
@ -800,16 +826,18 @@ void AP_Logger_File::start_new_log(void)
|
|||||||
_cached_oldest_log = 0;
|
_cached_oldest_log = 0;
|
||||||
|
|
||||||
if (_write_fd == -1) {
|
if (_write_fd == -1) {
|
||||||
_initialised = false;
|
|
||||||
write_fd_semaphore.give();
|
write_fd_semaphore.give();
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (open_error_ms_was_zero) {
|
||||||
::printf("Log open fail for %s - %s\n",
|
::printf("Log open fail for %s - %s\n",
|
||||||
_write_filename, strerror(saved_errno));
|
_write_filename, strerror(saved_errno));
|
||||||
hal.console->printf("Log open fail for %s - %s\n",
|
hal.console->printf("Log open fail for %s - %s\n",
|
||||||
_write_filename, strerror(saved_errno));
|
_write_filename, strerror(saved_errno));
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_last_write_ms = AP_HAL::millis();
|
_last_write_ms = AP_HAL::millis();
|
||||||
|
_open_error_ms = 0;
|
||||||
_write_offset = 0;
|
_write_offset = 0;
|
||||||
_writebuf.clear();
|
_writebuf.clear();
|
||||||
write_fd_semaphore.give();
|
write_fd_semaphore.give();
|
||||||
@ -821,6 +849,7 @@ void AP_Logger_File::start_new_log(void)
|
|||||||
int fd = AP::FS().open(fname, O_WRONLY|O_CREAT);
|
int fd = AP::FS().open(fname, O_WRONLY|O_CREAT);
|
||||||
free(fname);
|
free(fname);
|
||||||
if (fd == -1) {
|
if (fd == -1) {
|
||||||
|
_open_error_ms = AP_HAL::millis();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -831,9 +860,9 @@ void AP_Logger_File::start_new_log(void)
|
|||||||
AP::FS().close(fd);
|
AP::FS().close(fd);
|
||||||
|
|
||||||
if (written < to_write) {
|
if (written < to_write) {
|
||||||
|
_open_error_ms = AP_HAL::millis();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_open_error = false;
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -844,7 +873,7 @@ void AP_Logger_File::flush(void)
|
|||||||
#if APM_BUILD_TYPE(APM_BUILD_Replay) || APM_BUILD_TYPE(APM_BUILD_UNKNOWN)
|
#if APM_BUILD_TYPE(APM_BUILD_Replay) || APM_BUILD_TYPE(APM_BUILD_UNKNOWN)
|
||||||
{
|
{
|
||||||
uint32_t tnow = AP_HAL::millis();
|
uint32_t tnow = AP_HAL::millis();
|
||||||
while (_write_fd != -1 && _initialised && !_open_error && _writebuf.available()) {
|
while (_write_fd != -1 && _initialised && !recent_open_error() && _writebuf.available()) {
|
||||||
// convince the IO timer that it really is OK to write out
|
// convince the IO timer that it really is OK to write out
|
||||||
// less than _writebuf_chunk bytes:
|
// less than _writebuf_chunk bytes:
|
||||||
if (tnow > 2001) { // avoid resetting _last_write_time to 0
|
if (tnow > 2001) { // avoid resetting _last_write_time to 0
|
||||||
@ -872,7 +901,7 @@ void AP_Logger_File::_io_timer(void)
|
|||||||
{
|
{
|
||||||
uint32_t tnow = AP_HAL::millis();
|
uint32_t tnow = AP_HAL::millis();
|
||||||
_io_timer_heartbeat = tnow;
|
_io_timer_heartbeat = tnow;
|
||||||
if (_write_fd == -1 || !_initialised || _open_error) {
|
if (_write_fd == -1 || !_initialised || recent_open_error()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -892,7 +921,7 @@ void AP_Logger_File::_io_timer(void)
|
|||||||
if (disk_space_avail() < _free_space_min_avail && disk_space() > 0) {
|
if (disk_space_avail() < _free_space_min_avail && disk_space() > 0) {
|
||||||
hal.console->printf("Out of space for logging\n");
|
hal.console->printf("Out of space for logging\n");
|
||||||
stop_logging();
|
stop_logging();
|
||||||
_open_error = true; // prevent logging starting again
|
_open_error_ms = AP_HAL::millis(); // prevent logging starting again for 5s
|
||||||
last_io_operation = "";
|
last_io_operation = "";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -939,7 +968,6 @@ void AP_Logger_File::_io_timer(void)
|
|||||||
AP::FS().close(_write_fd);
|
AP::FS().close(_write_fd);
|
||||||
last_io_operation = "";
|
last_io_operation = "";
|
||||||
_write_fd = -1;
|
_write_fd = -1;
|
||||||
_initialised = false;
|
|
||||||
printf("Failed to write to File: %s\n", strerror(errno));
|
printf("Failed to write to File: %s\n", strerror(errno));
|
||||||
}
|
}
|
||||||
_last_write_failed = true;
|
_last_write_failed = true;
|
||||||
@ -989,7 +1017,7 @@ bool AP_Logger_File::logging_failed() const
|
|||||||
if (!_initialised) {
|
if (!_initialised) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (_open_error) {
|
if (recent_open_error()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (!io_thread_alive()) {
|
if (!io_thread_alive()) {
|
||||||
|
@ -72,7 +72,7 @@ private:
|
|||||||
uint16_t _read_fd_log_num;
|
uint16_t _read_fd_log_num;
|
||||||
uint32_t _read_offset;
|
uint32_t _read_offset;
|
||||||
uint32_t _write_offset;
|
uint32_t _write_offset;
|
||||||
volatile bool _open_error;
|
volatile uint32_t _open_error_ms;
|
||||||
const char *_log_directory;
|
const char *_log_directory;
|
||||||
bool _last_write_failed;
|
bool _last_write_failed;
|
||||||
|
|
||||||
@ -80,6 +80,9 @@ private:
|
|||||||
bool io_thread_alive() const;
|
bool io_thread_alive() const;
|
||||||
uint8_t io_thread_warning_decimation_counter;
|
uint8_t io_thread_warning_decimation_counter;
|
||||||
|
|
||||||
|
// do we have a recent open error?
|
||||||
|
bool recent_open_error(void) const;
|
||||||
|
|
||||||
// possibly time-consuming preparations handling
|
// possibly time-consuming preparations handling
|
||||||
void Prep_MinSpace();
|
void Prep_MinSpace();
|
||||||
int64_t disk_space_avail();
|
int64_t disk_space_avail();
|
||||||
|
Loading…
Reference in New Issue
Block a user