/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- /* DataFlash logging - file oriented variant This uses posix file IO to create log files called logs/NN.bin in the given directory */ #include #if HAL_OS_POSIX_IO #include "DataFlash_File.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef __APPLE__ #include #include #else #include #endif extern const AP_HAL::HAL& hal; #define MAX_LOG_FILES 500U #define DATAFLASH_PAGE_SIZE 1024UL /* constructor */ DataFlash_File::DataFlash_File(DataFlash_Class &front, const char *log_directory) : DataFlash_Backend(front), _write_fd(-1), _read_fd(-1), _read_fd_log_num(0), _read_offset(0), _write_offset(0), _initialised(false), _open_error(false), _log_directory(log_directory), _writebuf(NULL), _writebuf_size(16*1024), #if defined(CONFIG_ARCH_BOARD_PX4FMU_V1) // V1 gets IO errors with larger than 512 byte writes _writebuf_chunk(512), #elif defined(CONFIG_ARCH_BOARD_VRBRAIN_V45) _writebuf_chunk(512), #elif defined(CONFIG_ARCH_BOARD_VRBRAIN_V51) _writebuf_chunk(512), #elif defined(CONFIG_ARCH_BOARD_VRBRAIN_V52) _writebuf_chunk(512), #elif defined(CONFIG_ARCH_BOARD_VRUBRAIN_V51) _writebuf_chunk(512), #elif defined(CONFIG_ARCH_BOARD_VRUBRAIN_V52) _writebuf_chunk(512), #elif defined(CONFIG_ARCH_BOARD_VRHERO_V10) _writebuf_chunk(512), #else _writebuf_chunk(4096), #endif _writebuf_head(0), _writebuf_tail(0), _last_write_time(0) #if CONFIG_HAL_BOARD == HAL_BOARD_PX4 || CONFIG_HAL_BOARD == HAL_BOARD_VRBRAIN ,_perf_write(perf_alloc(PC_ELAPSED, "DF_write")), _perf_fsync(perf_alloc(PC_ELAPSED, "DF_fsync")), _perf_errors(perf_alloc(PC_COUNT, "DF_errors")), _perf_overruns(perf_alloc(PC_COUNT, "DF_overruns")) #endif {} // initialisation void DataFlash_File::Init(const struct LogStructure *structure, uint8_t num_types) { DataFlash_Backend::Init(structure, num_types); // create the log directory if need be int ret; struct stat st; #if CONFIG_HAL_BOARD == HAL_BOARD_PX4 || CONFIG_HAL_BOARD == HAL_BOARD_VRBRAIN // try to cope with an existing lowercase log directory // name. NuttX does not handle case insensitive VFAT well DIR *d = opendir("/fs/microsd/APM"); if (d != NULL) { for (struct dirent *de=readdir(d); de; de=readdir(d)) { if (strcmp(de->d_name, "logs") == 0) { rename("/fs/microsd/APM/logs", "/fs/microsd/APM/OLDLOGS"); break; } } closedir(d); } #endif const char* custom_dir = hal.util->get_custom_log_directory(); if (custom_dir != NULL){ _log_directory = custom_dir; } ret = stat(_log_directory, &st); if (ret == -1) { ret = mkdir(_log_directory, 0777); } if (ret == -1) { hal.console->printf("Failed to create log directory %s\n", _log_directory); return; } if (_writebuf != NULL) { free(_writebuf); _writebuf = NULL; } /* if we can't allocate the full writebuf then try reducing it until we can allocate it */ while (_writebuf == NULL && _writebuf_size >= _writebuf_chunk) { _writebuf = (uint8_t *)malloc(_writebuf_size); if (_writebuf == NULL) { _writebuf_size /= 2; } } if (_writebuf == NULL) { hal.console->printf("Out of memory for logging\n"); return; } _writebuf_head = _writebuf_tail = 0; _initialised = true; hal.scheduler->register_io_process(FUNCTOR_BIND_MEMBER(&DataFlash_File::_io_timer, void)); } void DataFlash_File::periodic_fullrate(const uint32_t now) { DataFlash_Backend::push_log_blocks(); } uint16_t DataFlash_File::bufferspace_available() { uint16_t _head; return (BUF_SPACE(_writebuf)) - critical_message_reserved_space(); } // return true for CardInserted() if we successfully initialised bool DataFlash_File::CardInserted(void) { return _initialised && !_open_error; } uint64_t DataFlash_File::disk_space_avail() { struct statfs stats; if (statfs(_log_directory, &stats) < 0) { return -1; } return (((uint64_t)stats.f_bavail) * stats.f_bsize); } uint64_t DataFlash_File::disk_space() { struct statfs stats; if (statfs(_log_directory, &stats) < 0) { return -1; } return (((uint64_t)stats.f_blocks) * stats.f_bsize); } float DataFlash_File::avail_space_percent() { uint64_t avail = disk_space_avail(); uint64_t space = disk_space(); return (avail/(float)space) * 100; } // find_first_log - find lowest-numbered log in _log_directory // returns 0 if no log was found uint16_t DataFlash_File::find_first_log(void) { // iterate through directory entries to find lowest log number. // We could count up to find_last_log(), but if people start // relying on the min_avail_space_percent feature we could end up // doing a *lot* of asprintf()s and stat()s DIR *d = opendir(_log_directory); if (d == NULL) { // internal_error(); return 0; } // we only remove files which look like xxx.BIN uint32_t lowest_number = (uint32_t)-1; // unsigned integer wrap for (struct dirent *de=readdir(d); de; de=readdir(d)) { uint8_t length = strlen(de->d_name); if (length < 5) { // not long enough for \d+[.]BIN continue; } if (strncmp(&de->d_name[length-4], ".BIN", 4)) { // doesn't end in .BIN continue; } uint16_t thisnum = strtoul(de->d_name, NULL, 10); if (thisnum < lowest_number) { lowest_number = thisnum; } } closedir(d); if (lowest_number == (uint32_t)-1) { return 0; } return lowest_number; } void DataFlash_File::Prep_MinSpace() { uint16_t log_to_remove = find_first_log(); if (log_to_remove == 0) { // no files to remove return; } uint16_t count = 0; while (avail_space_percent() < min_avail_space_percent) { if (count++ > 500) { // *way* too many deletions going on here. Possible internal error. // deletion rate seems to be ~50 files/second. // internal_error(); break; } char *filename_to_remove = _log_file_name(log_to_remove); if (filename_to_remove == NULL) { // internal_error(); break; } hal.console->printf("Removing (%s) for minimum-space requirements\n", filename_to_remove); if (unlink(filename_to_remove) == -1) { hal.console->printf("Failed to remove %s: (%d/%d) %s\n", filename_to_remove, errno, ENOENT, strerror(errno)); free(filename_to_remove); if (errno == ENOENT) { log_to_remove = find_first_log(); if (log_to_remove == 0) { // out of files to remove... break; } // corruption - should always have a continuous // sequence of files... however, we now have a file // we can delete, so keep going. } else { break; } } else { free(filename_to_remove); log_to_remove++; } } } void DataFlash_File::Prep() { if (hal.util->get_soft_armed()) { // do not want to do any filesystem operations while we are e.g. flying return; } Prep_MinSpace(); } bool DataFlash_File::NeedPrep() { if (!CardInserted()) { // should not have been called?! return false; } if (avail_space_percent() < min_avail_space_percent) { return true; } return false; } /* construct a log file name given a log number. Note: Caller must free. */ char *DataFlash_File::_log_file_name(uint16_t log_num) { char *buf = NULL; if (asprintf(&buf, "%s/%u.BIN", _log_directory, (unsigned)log_num) == 0) { return NULL; } return buf; } /* return path name of the lastlog.txt marker file Note: Caller must free. */ char *DataFlash_File::_lastlog_file_name(void) { char *buf = NULL; if (asprintf(&buf, "%s/LASTLOG.TXT", _log_directory) == 0) { return NULL; } return buf; } // remove all log files void DataFlash_File::EraseAll() { uint16_t log_num; stop_logging(); for (log_num=0; log_num size) n = size; assert(_writebuf_tail+n <= _writebuf_size); memcpy(&_writebuf[_writebuf_tail], pBuffer, n); BUF_ADVANCETAIL(_writebuf, n); pBuffer = (const void *)(((const uint8_t *)pBuffer) + n); n = size - n; if (n > 0) { assert(_writebuf_tail+n <= _writebuf_size); memcpy(&_writebuf[_writebuf_tail], pBuffer, n); BUF_ADVANCETAIL(_writebuf, n); } } return true; } /* read a packet. The header bytes have already been read. */ bool DataFlash_File::ReadBlock(void *pkt, uint16_t size) { if (_read_fd == -1 || !_initialised || _open_error) { return false; } memset(pkt, 0, size); if (::read(_read_fd, pkt, size) != size) { return false; } _read_offset += size; return true; } /* find the highest log number */ uint16_t DataFlash_File::find_last_log(void) { unsigned ret = 0; char *fname = _lastlog_file_name(); if (fname == NULL) { return ret; } FILE *f = ::fopen(fname, "r"); free(fname); if (f != NULL) { char buf[10]; memset(buf, 0, sizeof(buf)); // PX4 doesn't have fscanf() if (fread(buf, 1, sizeof(buf)-1, f) > 0) { sscanf(buf, "%u", &ret); } fclose(f); } return ret; } bool DataFlash_File::_log_exists(uint16_t log_num) { char *fname = _log_file_name(log_num); if (fname == NULL) { return 0; } struct stat st; // stat return 0 if the file exists: bool ret = ::stat(fname, &st) ? false : true; free(fname); return ret; } uint32_t DataFlash_File::_get_log_size(uint16_t log_num) { char *fname = _log_file_name(log_num); if (fname == NULL) { return 0; } struct stat st; if (::stat(fname, &st) != 0) { free(fname); return 0; } free(fname); return st.st_size; } uint32_t DataFlash_File::_get_log_time(uint16_t log_num) { char *fname = _log_file_name(log_num); if (fname == NULL) { return 0; } struct stat st; if (::stat(fname, &st) != 0) { free(fname); return 0; } free(fname); return st.st_mtime; } /* find the number of pages in a log */ void DataFlash_File::get_log_boundaries(uint16_t log_num, uint16_t & start_page, uint16_t & end_page) { start_page = 0; end_page = _get_log_size(log_num) / DATAFLASH_PAGE_SIZE; } /* find the number of pages in a log */ int16_t DataFlash_File::get_log_data(uint16_t log_num, uint16_t page, uint32_t offset, uint16_t len, uint8_t *data) { if (!_initialised || _open_error) { return -1; } if (_read_fd != -1 && log_num != _read_fd_log_num) { ::close(_read_fd); _read_fd = -1; } if (_read_fd == -1) { char *fname = _log_file_name(log_num); if (fname == NULL) { return -1; } stop_logging(); _read_fd = ::open(fname, O_RDONLY); if (_read_fd == -1) { _open_error = true; int saved_errno = errno; ::printf("Log read open fail for %s - %s\n", fname, strerror(saved_errno)); hal.console->printf("Log read open fail for %s - %s\n", fname, strerror(saved_errno)); free(fname); return -1; } free(fname); _read_offset = 0; _read_fd_log_num = log_num; } uint32_t ofs = page * (uint32_t)DATAFLASH_PAGE_SIZE + offset; /* this rather strange bit of code is here to work around a bug in file offsets in NuttX. Every few hundred blocks of reads (starting at around 350k into a file) NuttX will get the wrong offset for sequential reads. The offset it gets is typically 128k earlier than it should be. It turns out that calling lseek() with 0 offset and SEEK_CUR works around the bug. We can remove this once we find the real bug. */ if (ofs / 4096 != (ofs+len) / 4096) { off_t seek_current = ::lseek(_read_fd, 0, SEEK_CUR); if (seek_current != (off_t)_read_offset) { ::lseek(_read_fd, _read_offset, SEEK_SET); } } if (ofs != _read_offset) { ::lseek(_read_fd, ofs, SEEK_SET); _read_offset = ofs; } int16_t ret = (int16_t)::read(_read_fd, data, len); if (ret > 0) { _read_offset += ret; } return ret; } /* find size and date of a log */ void DataFlash_File::get_log_info(uint16_t log_num, uint32_t &size, uint32_t &time_utc) { size = _get_log_size(log_num); time_utc = _get_log_time(log_num); } /* get the number of logs - note that the log numbers must be consecutive */ uint16_t DataFlash_File::get_num_logs(void) { uint16_t ret; uint16_t high = find_last_log(); for (ret=0; ret 0 || log_num == 0) { log_num++; } if (log_num > MAX_LOG_FILES) { log_num = 1; } char *fname = _log_file_name(log_num); _write_fd = ::open(fname, O_WRONLY|O_CREAT|O_TRUNC, 0666); if (_write_fd == -1) { _initialised = false; _open_error = true; int saved_errno = errno; ::printf("Log open fail for %s - %s\n", fname, strerror(saved_errno)); hal.console->printf("Log open fail for %s - %s\n", fname, strerror(saved_errno)); free(fname); return 0xFFFF; } free(fname); _write_offset = 0; _writebuf_head = 0; _writebuf_tail = 0; log_write_started = true; // now update lastlog.txt with the new log number fname = _lastlog_file_name(); FILE *f = ::fopen(fname, "w"); fprintf(f, "%u\r\n", (unsigned)log_num); fclose(f); free(fname); return log_num; } /* Read the log and print it on port */ void DataFlash_File::LogReadProcess(uint16_t log_num, uint16_t start_page, uint16_t end_page, print_mode_fn print_mode, AP_HAL::BetterStream *port) { uint8_t log_step = 0; if (!_initialised || _open_error) { return; } if (_read_fd != -1) { ::close(_read_fd); _read_fd = -1; } char *fname = _log_file_name(log_num); if (fname == NULL) { return; } _read_fd = ::open(fname, O_RDONLY); free(fname); if (_read_fd == -1) { return; } _read_fd_log_num = log_num; _read_offset = 0; if (start_page != 0) { ::lseek(_read_fd, start_page * DATAFLASH_PAGE_SIZE, SEEK_SET); _read_offset = start_page * DATAFLASH_PAGE_SIZE; } uint8_t log_counter = 0; while (true) { uint8_t data; if (::read(_read_fd, &data, 1) != 1) { // reached end of file break; } _read_offset++; // This is a state machine to read the packets switch(log_step) { case 0: if (data == HEAD_BYTE1) { log_step++; } break; case 1: if (data == HEAD_BYTE2) { log_step++; } else { log_step = 0; } break; case 2: log_step = 0; _print_log_entry(data, print_mode, port); log_counter++; if (log_counter == 10) { log_counter = 0; ::lseek(_read_fd, 0, SEEK_CUR); } break; } if (_read_offset >= (end_page+1) * DATAFLASH_PAGE_SIZE) { break; } } ::close(_read_fd); _read_fd = -1; } /* this is a lot less verbose than the block interface. Dumping 2Gbyte of logs a page at a time isn't so useful. Just pull the SD card out and look at it on your PC */ void DataFlash_File::DumpPageInfo(AP_HAL::BetterStream *port) { port->printf_P(PSTR("DataFlash: num_logs=%u\n"), (unsigned)get_num_logs()); } void DataFlash_File::ShowDeviceInfo(AP_HAL::BetterStream *port) { port->printf_P(PSTR("DataFlash logs stored in %s\n"), _log_directory); } /* list available log numbers */ void DataFlash_File::ListAvailableLogs(AP_HAL::BetterStream *port) { uint16_t num_logs = get_num_logs(); int16_t last_log_num = find_last_log(); if (num_logs == 0) { port->printf_P(PSTR("\nNo logs\n\n")); return; } port->printf_P(PSTR("\n%u logs\n"), (unsigned)num_logs); for (uint16_t i=num_logs; i>=1; i--) { uint16_t log_num = last_log_num - i + 1; off_t size; char *filename = _log_file_name(log_num); if (filename != NULL) { size = _get_log_size(log_num); struct stat st; if (stat(filename, &st) == 0) { struct tm *tm = gmtime(&st.st_mtime); port->printf_P(PSTR("Log %u in %s of size %u %u/%u/%u %u:%u\n"), (unsigned)log_num, filename, (unsigned)size, (unsigned)tm->tm_year+1900, (unsigned)tm->tm_mon+1, (unsigned)tm->tm_mday, (unsigned)tm->tm_hour, (unsigned)tm->tm_min); } free(filename); } } port->println(); } #if CONFIG_HAL_BOARD == HAL_BOARD_SITL || CONFIG_HAL_BOARD == HAL_BOARD_LINUX void DataFlash_File::flush(void) { uint16_t _tail; uint32_t tnow = hal.scheduler->micros(); hal.scheduler->suspend_timer_procs(); while (_write_fd != -1 && _initialised && !_open_error && BUF_AVAILABLE(_writebuf)) { // convince the IO timer that it really is OK to write out // less than _writebuf_chunk bytes: _last_write_time = tnow - 2000000; _io_timer(); } hal.scheduler->resume_timer_procs(); if (_write_fd != -1) { ::fsync(_write_fd); } } #endif void DataFlash_File::_io_timer(void) { uint16_t _tail; if (_write_fd == -1 || !_initialised || _open_error) { return; } uint16_t nbytes = BUF_AVAILABLE(_writebuf); if (nbytes == 0) { return; } uint32_t tnow = hal.scheduler->micros(); if (nbytes < _writebuf_chunk && tnow - _last_write_time < 2000000UL) { // write in _writebuf_chunk-sized chunks, but always write at // least once per 2 seconds if data is available return; } perf_begin(_perf_write); _last_write_time = tnow; if (nbytes > _writebuf_chunk) { // be kind to the FAT PX4 filesystem nbytes = _writebuf_chunk; } if (_writebuf_head > _tail) { // only write to the end of the buffer nbytes = min(nbytes, _writebuf_size - _writebuf_head); } // try to align writes on a 512 byte boundary to avoid filesystem // reads if ((nbytes + _write_offset) % 512 != 0) { uint32_t ofs = (nbytes + _write_offset) % 512; if (ofs < nbytes) { nbytes -= ofs; } } assert(_writebuf_head+nbytes <= _writebuf_size); ssize_t nwritten = ::write(_write_fd, &_writebuf[_writebuf_head], nbytes); if (nwritten <= 0) { perf_count(_perf_errors); close(_write_fd); _write_fd = -1; _initialised = false; } else { _write_offset += nwritten; /* the best strategy for minimising corruption on microSD cards seems to be to write in 4k chunks and fsync the file on each chunk, ensuring the directory entry is updated after each write. */ BUF_ADVANCEHEAD(_writebuf, nwritten); #if CONFIG_HAL_BOARD != HAL_BOARD_SITL && CONFIG_HAL_BOARD_SUBTYPE != HAL_BOARD_SUBTYPE_LINUX_NONE ::fsync(_write_fd); #endif } perf_end(_perf_write); } #endif // HAL_OS_POSIX_IO