/* ArduPilot filesystem interface for systems using the FATFS filesystem */ #include "AP_Filesystem_config.h" #if AP_FILESYSTEM_FATFS_ENABLED #include "AP_Filesystem.h" #include <AP_HAL/AP_HAL.h> #include <AP_Math/AP_Math.h> #include <stdio.h> #include <AP_RTC/AP_RTC.h> #include <ff.h> #include <AP_HAL_ChibiOS/sdcard.h> #include <GCS_MAVLink/GCS.h> #if 0 #define debug(fmt, args ...) do {printf("%s:%d: " fmt "\n", __FUNCTION__, __LINE__, ## args); } while(0) #else #define debug(fmt, args ...) #endif extern const AP_HAL::HAL& hal; static bool remount_needed; #define FATFS_R (S_IRUSR | S_IRGRP | S_IROTH) /*< FatFs Read perms */ #define FATFS_W (S_IWUSR | S_IWGRP | S_IWOTH) /*< FatFs Write perms */ #define FATFS_X (S_IXUSR | S_IXGRP | S_IXOTH) /*< FatFs Execute perms */ // don't write more than 4k at a time to prevent needing too much // DMAable memory #define MAX_IO_SIZE 4096 // use a semaphore to ensure that only one filesystem operation is // happening at a time. A recursive semaphore is used to cope with the // mkdir() inside sdcard_retry() HAL_Semaphore AP_Filesystem_FATFS::sem; typedef struct { FIL *fh; char *name; } FAT_FILE; #define MAX_FILES 16 static FAT_FILE *file_table[MAX_FILES]; static int isatty_(int fileno) { if (fileno >= 0 && fileno <= 2) { return true; } return false; } /* allocate a file descriptor */ static int new_file_descriptor(const char *pathname) { int i; FAT_FILE *stream; FIL *fh; for (i=0; i<MAX_FILES; ++i) { if (isatty_(i)) { continue; } if (file_table[i] == NULL) { stream = (FAT_FILE *) calloc(sizeof(FAT_FILE),1); if (stream == NULL) { errno = ENOMEM; return -1; } fh = (FIL *) calloc(sizeof(FIL),1); if (fh == NULL) { free(stream); errno = ENOMEM; return -1; } char *fname = (char *)malloc(strlen(pathname)+1); if (fname == NULL) { free(fh); free(stream); errno = ENOMEM; return -1; } strcpy(fname, pathname); stream->name = fname; file_table[i] = stream; stream->fh = fh; return i; } } errno = ENFILE; return -1; } static FAT_FILE *fileno_to_stream(int fileno) { FAT_FILE *stream; if (fileno < 0 || fileno >= MAX_FILES) { errno = EBADF; return nullptr; } stream = file_table[fileno]; if (stream == NULL) { errno = EBADF; return nullptr; } return stream; } static int free_file_descriptor(int fileno) { FAT_FILE *stream; FIL *fh; if (isatty_( fileno )) { errno = EBADF; return -1; } // checks if fileno out of bounds stream = fileno_to_stream(fileno); if (stream == nullptr) { return -1; } fh = stream->fh; if (fh != NULL) { free(fh); } free(stream->name); stream->name = NULL; file_table[fileno] = NULL; free(stream); return fileno; } static FIL *fileno_to_fatfs(int fileno) { FAT_FILE *stream; FIL *fh; if (isatty_(fileno)) { errno = EBADF; return nullptr; } // checks if fileno out of bounds stream = fileno_to_stream(fileno); if (stream == nullptr) { return nullptr; } fh = stream->fh; if (fh == NULL) { errno = EBADF; return nullptr; } return fh; } static int fatfs_to_errno(FRESULT Result) { switch (Result) { case FR_OK: /* FatFS (0) Succeeded */ return 0; /* POSIX OK */ case FR_DISK_ERR: /* FatFS (1) A hard error occurred in the low level disk I/O layer */ return EIO; /* POSIX Input/output error (POSIX.1) */ case FR_INT_ERR: /* FatFS (2) Assertion failed */ return EPERM; /* POSIX Operation not permitted (POSIX.1) */ case FR_NOT_READY: /* FatFS (3) The physical drive cannot work */ return EBUSY; /* POSIX Device or resource busy (POSIX.1) */ case FR_NO_FILE: /* FatFS (4) Could not find the file */ return ENOENT; /* POSIX No such file or directory (POSIX.1) */ case FR_NO_PATH: /* FatFS (5) Could not find the path */ return ENOENT; /* POSIX No such file or directory (POSIX.1) */ case FR_INVALID_NAME: /* FatFS (6) The path name format is invalid */ return EINVAL; /* POSIX Invalid argument (POSIX.1) */ case FR_DENIED: /* FatFS (7) Access denied due to prohibited access or directory full */ return EACCES; /* POSIX Permission denied (POSIX.1) */ case FR_EXIST: /* file exists */ return EEXIST; /* file exists */ case FR_INVALID_OBJECT: /* FatFS (9) The file/directory object is invalid */ return EINVAL; /* POSIX Invalid argument (POSIX.1) */ case FR_WRITE_PROTECTED: /* FatFS (10) The physical drive is write protected */ return EROFS; /* POSIX Read-only filesystem (POSIX.1) */ case FR_INVALID_DRIVE: /* FatFS (11) The logical drive number is invalid */ return ENXIO; /* POSIX No such device or address (POSIX.1) */ case FR_NOT_ENABLED: /* FatFS (12) The volume has no work area */ return ENOSPC; /* POSIX No space left on device (POSIX.1) */ case FR_NO_FILESYSTEM: /* FatFS (13) There is no valid FAT volume */ return ENXIO; /* POSIX No such device or address (POSIX.1) */ case FR_MKFS_ABORTED: /* FatFS (14) The f_mkfs() aborted due to any parameter error */ return EINVAL; /* POSIX Invalid argument (POSIX.1) */ case FR_TIMEOUT: /* FatFS (15) Could not get a grant to access the volume within defined period */ return EBUSY; /* POSIX Device or resource busy (POSIX.1) */ case FR_LOCKED: /* FatFS (16) The operation is rejected according to the file sharing policy */ return EBUSY; /* POSIX Device or resource busy (POSIX.1) */ case FR_NOT_ENOUGH_CORE: /* FatFS (17) LFN working buffer could not be allocated */ return ENOMEM; /* POSIX Not enough space (POSIX.1) */ case FR_TOO_MANY_OPEN_FILES:/* FatFS (18) Number of open files > _FS_SHARE */ return EMFILE; /* POSIX Too many open files (POSIX.1) */ case FR_INVALID_PARAMETER:/* FatFS (19) Given parameter is invalid */ return EINVAL; /* POSIX Invalid argument (POSIX.1) */ } return EBADMSG; /* POSIX Bad message (POSIX.1) */ } // check for a remount and return -1 if it fails #define CHECK_REMOUNT() do { if (remount_needed && !remount_file_system()) { errno = EIO; return -1; }} while (0) #define CHECK_REMOUNT_NULL() do { if (remount_needed && !remount_file_system()) { errno = EIO; return NULL; }} while (0) // we allow for IO retries if either not armed or not in main thread #define RETRY_ALLOWED() (!hal.scheduler->in_main_thread() || !hal.util->get_soft_armed()) /* try to remount the file system on disk error */ static bool remount_file_system(void) { EXPECT_DELAY_MS(3000); if (!remount_needed) { sdcard_stop(); } if (!sdcard_retry()) { remount_needed = true; EXPECT_DELAY_MS(0); return false; } remount_needed = false; for (uint16_t i=0; i<MAX_FILES; i++) { FAT_FILE *f = file_table[i]; if (!f) { continue; } FIL *fh = f->fh; FSIZE_t offset = fh->fptr; uint8_t flags = fh->flag & (FA_READ | FA_WRITE); memset(fh, 0, sizeof(*fh)); if (flags & FA_WRITE) { // the file may not have been created yet on the sdcard flags |= FA_OPEN_ALWAYS; } FRESULT res = f_open(fh, f->name, flags); debug("reopen %s flags=0x%x ofs=%u -> %d\n", f->name, unsigned(flags), unsigned(offset), int(res)); if (res == FR_OK) { f_lseek(fh, offset); } } EXPECT_DELAY_MS(0); return true; } int AP_Filesystem_FATFS::open(const char *pathname, int flags, bool allow_absolute_path) { int fileno; int fatfs_modes; FAT_FILE *stream; FIL *fh; int res; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); CHECK_REMOUNT(); errno = 0; debug("Open %s 0x%x", pathname, flags); if ((flags & O_ACCMODE) == O_RDWR) { fatfs_modes = FA_READ | FA_WRITE; } else if ((flags & O_ACCMODE) == O_RDONLY) { fatfs_modes = FA_READ; } else { fatfs_modes = FA_WRITE; } if (flags & O_CREAT) { if (flags & O_TRUNC) { fatfs_modes |= FA_CREATE_ALWAYS; } else { fatfs_modes |= FA_OPEN_ALWAYS; } } fileno = new_file_descriptor(pathname); // checks if fileno out of bounds stream = fileno_to_stream(fileno); if (stream == nullptr) { free_file_descriptor(fileno); return -1; } // fileno_to_fatfs checks for fileno out of bounds fh = fileno_to_fatfs(fileno); if (fh == nullptr) { free_file_descriptor(fileno); errno = EBADF; return -1; } res = f_open(fh, pathname, (BYTE) (fatfs_modes & 0xff)); if (res == FR_DISK_ERR && RETRY_ALLOWED()) { // one retry on disk error hal.scheduler->delay(100); if (remount_file_system()) { res = f_open(fh, pathname, (BYTE) (fatfs_modes & 0xff)); } } if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); free_file_descriptor(fileno); return -1; } if (flags & O_APPEND) { /// Seek to end of the file res = f_lseek(fh, f_size(fh)); if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); f_close(fh); free_file_descriptor(fileno); return -1; } } debug("Open %s -> %d", pathname, fileno); return fileno; } int AP_Filesystem_FATFS::close(int fileno) { FAT_FILE *stream; FIL *fh; int res; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); errno = 0; // checks if fileno out of bounds stream = fileno_to_stream(fileno); if (stream == nullptr) { return -1; } // fileno_to_fatfs checks for fileno out of bounds fh = fileno_to_fatfs(fileno); if (fh == nullptr) { return -1; } res = f_close(fh); free_file_descriptor(fileno); if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } return 0; } int32_t AP_Filesystem_FATFS::read(int fd, void *buf, uint32_t count) { UINT bytes = count; int res; FIL *fh; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); CHECK_REMOUNT(); if (count > 0) { *(char *) buf = 0; } errno = 0; // fileno_to_fatfs checks for fd out of bounds fh = fileno_to_fatfs(fd); if (fh == nullptr) { errno = EBADF; return -1; } UINT total = 0; do { UINT size = 0; UINT n = MIN(bytes, MAX_IO_SIZE); res = f_read(fh, (void *)buf, n, &size); if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } if (size == 0) { break; } if (size > n) { errno = EIO; return -1; } total += size; buf = (void *)(((uint8_t *)buf)+size); bytes -= size; if (size < n) { break; } } while (bytes > 0); return (ssize_t)total; } int32_t AP_Filesystem_FATFS::write(int fd, const void *buf, uint32_t count) { UINT bytes = count; FRESULT res; FIL *fh; errno = 0; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); CHECK_REMOUNT(); // fileno_to_fatfs checks for fd out of bounds fh = fileno_to_fatfs(fd); if (fh == nullptr) { errno = EBADF; return -1; } UINT total = 0; do { UINT n = MIN(bytes, MAX_IO_SIZE); UINT size = 0; res = f_write(fh, buf, n, &size); if (res == FR_DISK_ERR && RETRY_ALLOWED()) { // one retry on disk error hal.scheduler->delay(100); if (remount_file_system()) { res = f_write(fh, buf, n, &size); } } if (size > n || size == 0) { errno = EIO; return -1; } if (res != FR_OK || size > n) { errno = fatfs_to_errno(res); return -1; } total += size; buf = (void *)(((uint8_t *)buf)+size); bytes -= size; if (size < n) { break; } } while (bytes > 0); return (ssize_t)total; } int AP_Filesystem_FATFS::fsync(int fileno) { FAT_FILE *stream; FIL *fh; int res; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); errno = 0; // checks if fileno out of bounds stream = fileno_to_stream(fileno); if (stream == nullptr) { return -1; } // fileno_to_fatfs checks for fileno out of bounds fh = fileno_to_fatfs(fileno); if (fh == nullptr) { return -1; } res = f_sync(fh); if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } return 0; } off_t AP_Filesystem_FATFS::lseek(int fileno, off_t position, int whence) { FRESULT res; FIL *fh; errno = 0; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); // fileno_to_fatfs checks for fd out of bounds fh = fileno_to_fatfs(fileno); if (fh == nullptr) { errno = EMFILE; return -1; } if (isatty_(fileno)) { return -1; } if (whence == SEEK_END) { position += f_size(fh); } else if (whence==SEEK_CUR) { position += fh->fptr; } res = f_lseek(fh, position); if (res) { errno = fatfs_to_errno(res); return -1; } return fh->fptr; } static time_t fat_time_to_unix(uint16_t date, uint16_t time) { struct tm tp; time_t unix; memset(&tp, 0, sizeof(struct tm)); tp.tm_sec = (time << 1) & 0x3e; // 2 second resolution tp.tm_min = ((time >> 5) & 0x3f); tp.tm_hour = ((time >> 11) & 0x1f); tp.tm_mday = (date & 0x1f); tp.tm_mon = ((date >> 5) & 0x0f) - 1; tp.tm_year = ((date >> 9) & 0x7f) + 80; unix = AP::rtc().mktime(&tp); return unix; } int AP_Filesystem_FATFS::stat(const char *name, struct stat *buf) { FILINFO info; int res; time_t epoch; uint16_t mode; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); CHECK_REMOUNT(); errno = 0; // f_stat does not handle / or . as root directory if (strcmp(name,"/") == 0 || strcmp(name,".") == 0) { buf->st_atime = 0; buf->st_mtime = 0; buf->st_ctime = 0; buf->st_uid= 0; buf->st_gid= 0; buf->st_size = 0; buf->st_mode = S_IFDIR; return 0; } res = f_stat(name, &info); if (res == FR_DISK_ERR && RETRY_ALLOWED()) { // one retry on disk error if (remount_file_system()) { res = f_stat(name, &info); } } if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } buf->st_size = info.fsize; epoch = fat_time_to_unix(info.fdate, info.ftime); buf->st_atime = epoch; // Access time buf->st_mtime = epoch; // Modification time buf->st_ctime = epoch; // Creation time // We only handle read only case mode = (FATFS_R | FATFS_X); if (!(info.fattrib & AM_RDO)) { mode |= (FATFS_W); // enable write if NOT read only } if (info.fattrib & AM_SYS) { buf->st_uid= 0; buf->st_gid= 0; } { buf->st_uid=1000; buf->st_gid=1000; } if (info.fattrib & AM_DIR) { mode |= S_IFDIR; } else { mode |= S_IFREG; } buf->st_mode = mode; return 0; } int AP_Filesystem_FATFS::unlink(const char *pathname) { FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); errno = 0; int res = f_unlink(pathname); if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } return 0; } int AP_Filesystem_FATFS::mkdir(const char *pathname) { FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); errno = 0; int res = f_mkdir(pathname); if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } return 0; } int AP_Filesystem_FATFS::rename(const char *oldpath, const char *newpath) { FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); errno = 0; int res = f_rename(oldpath, newpath); if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } return 0; } /* wrapper structure to associate a dirent with a DIR */ struct DIR_Wrapper { DIR d; // must be first structure element struct dirent de; }; void *AP_Filesystem_FATFS::opendir(const char *pathdir) { FS_CHECK_ALLOWED(nullptr); WITH_SEMAPHORE(sem); CHECK_REMOUNT_NULL(); debug("Opendir %s", pathdir); struct DIR_Wrapper *ret = new DIR_Wrapper; if (!ret) { return nullptr; } int res = f_opendir(&ret->d, pathdir); if (res == FR_DISK_ERR && RETRY_ALLOWED()) { // one retry on disk error if (remount_file_system()) { res = f_opendir(&ret->d, pathdir); } } if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); delete ret; return nullptr; } debug("Opendir %s -> %p", pathdir, ret); return &ret->d; } struct dirent *AP_Filesystem_FATFS::readdir(void *dirp_void) { FS_CHECK_ALLOWED(nullptr); WITH_SEMAPHORE(sem); DIR *dirp = (DIR *)dirp_void; struct DIR_Wrapper *d = (struct DIR_Wrapper *)dirp; if (!d) { errno = EINVAL; return nullptr; } FILINFO fno; int len; int res; d->de.d_name[0] = 0; res = f_readdir(dirp, &fno); if (res != FR_OK || fno.fname[0] == 0) { errno = fatfs_to_errno((FRESULT)res); return nullptr; } len = strlen(fno.fname); strncpy_noterm(d->de.d_name,fno.fname,len); d->de.d_name[len] = 0; if (fno.fattrib & AM_DIR) { d->de.d_type = DT_DIR; } else { d->de.d_type = DT_REG; } return &d->de; } int AP_Filesystem_FATFS::closedir(void *dirp_void) { DIR *dirp = (DIR *)dirp_void; FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); struct DIR_Wrapper *d = (struct DIR_Wrapper *)dirp; if (!d) { errno = EINVAL; return -1; } int res = f_closedir (dirp); delete d; if (res != FR_OK) { errno = fatfs_to_errno((FRESULT)res); return -1; } debug("closedir"); return 0; } // return free disk space in bytes int64_t AP_Filesystem_FATFS::disk_free(const char *path) { FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); FATFS *fs; DWORD fre_clust, fre_sect; CHECK_REMOUNT(); /* Get volume information and free clusters of drive 1 */ FRESULT res = f_getfree("/", &fre_clust, &fs); if (res) { return res; } /* Get total sectors and free sectors */ fre_sect = fre_clust * fs->csize; return (int64_t)(fre_sect)*512; } // return total disk space in bytes int64_t AP_Filesystem_FATFS::disk_space(const char *path) { FS_CHECK_ALLOWED(-1); WITH_SEMAPHORE(sem); CHECK_REMOUNT(); FATFS *fs; DWORD fre_clust, tot_sect; /* Get volume information and free clusters of drive 1 */ FRESULT res = f_getfree("/", &fre_clust, &fs); if (res) { return -1; } /* Get total sectors and free sectors */ tot_sect = (fs->n_fatent - 2) * fs->csize; return (int64_t)(tot_sect)*512; } /* convert unix time_t to FATFS timestamp */ static void unix_time_to_fat(time_t epoch, uint16_t &date, uint16_t &time) { struct tm *t = gmtime((time_t *)&epoch); /* Pack date and time into a uint32_t variable */ date = ((uint16_t)(t->tm_year - 80) << 9) | (((uint16_t)t->tm_mon+1) << 5) | (((uint16_t)t->tm_mday)); time = ((uint16_t)t->tm_hour << 11) | ((uint16_t)t->tm_min << 5) | ((uint16_t)t->tm_sec >> 1); } /* set mtime on a file */ bool AP_Filesystem_FATFS::set_mtime(const char *filename, const uint32_t mtime_sec) { FILINFO fno; uint16_t fdate, ftime; unix_time_to_fat(mtime_sec, fdate, ftime); fno.fdate = fdate; fno.ftime = ftime; FS_CHECK_ALLOWED(false); WITH_SEMAPHORE(sem); return f_utime(filename, (FILINFO *)&fno) == FR_OK; } /* retry mount of filesystem if needed */ bool AP_Filesystem_FATFS::retry_mount(void) { FS_CHECK_ALLOWED(false); WITH_SEMAPHORE(sem); return sdcard_retry(); } /* unmount filesystem for reboot */ void AP_Filesystem_FATFS::unmount(void) { WITH_SEMAPHORE(sem); return sdcard_stop(); } /* format sdcard */ bool AP_Filesystem_FATFS::format(void) { #if FF_USE_MKFS WITH_SEMAPHORE(sem); hal.scheduler->register_io_process(FUNCTOR_BIND_MEMBER(&AP_Filesystem_FATFS::format_handler, void)); // the format is handled asyncronously, we inform user of success // via a text message. format_status can be polled for progress format_status = FormatStatus::PENDING; return true; #else return false; #endif } /* format sdcard */ void AP_Filesystem_FATFS::format_handler(void) { #if FF_USE_MKFS if (format_status != FormatStatus::PENDING) { return; } WITH_SEMAPHORE(sem); format_status = FormatStatus::IN_PROGRESS; GCS_SEND_TEXT(MAV_SEVERITY_NOTICE, "Formatting SDCard"); uint8_t *buf = (uint8_t *)hal.util->malloc_type(FF_MAX_SS, AP_HAL::Util::MEM_DMA_SAFE); if (buf == nullptr) { return; } // format first disk auto ret = f_mkfs("0:", 0, buf, FF_MAX_SS); hal.util->free_type(buf, FF_MAX_SS, AP_HAL::Util::MEM_DMA_SAFE); if (ret == FR_OK) { format_status = FormatStatus::SUCCESS; GCS_SEND_TEXT(MAV_SEVERITY_NOTICE, "Format: OK"); } else { format_status = FormatStatus::FAILURE; GCS_SEND_TEXT(MAV_SEVERITY_NOTICE, "Format: Failed (%d)", int(ret)); } sdcard_stop(); sdcard_retry(); #endif } // returns true if we are currently formatting the SD card: AP_Filesystem_Backend::FormatStatus AP_Filesystem_FATFS::get_format_status(void) const { // note that format_handler holds sem, so we can't take it here. return format_status; } /* convert POSIX errno to text with user message. */ char *strerror(int errnum) { #define SWITCH_ERROR(errno) case errno: return const_cast<char *>(#errno); break switch (errnum) { SWITCH_ERROR(EPERM); SWITCH_ERROR(ENOENT); SWITCH_ERROR(ESRCH); SWITCH_ERROR(EINTR); SWITCH_ERROR(EIO); SWITCH_ERROR(ENXIO); SWITCH_ERROR(E2BIG); SWITCH_ERROR(ENOEXEC); SWITCH_ERROR(EBADF); SWITCH_ERROR(ECHILD); SWITCH_ERROR(EAGAIN); SWITCH_ERROR(ENOMEM); SWITCH_ERROR(EACCES); SWITCH_ERROR(EFAULT); #ifdef ENOTBLK SWITCH_ERROR(ENOTBLK); #endif // ENOTBLK SWITCH_ERROR(EBUSY); SWITCH_ERROR(EEXIST); SWITCH_ERROR(EXDEV); SWITCH_ERROR(ENODEV); SWITCH_ERROR(ENOTDIR); SWITCH_ERROR(EISDIR); SWITCH_ERROR(EINVAL); SWITCH_ERROR(ENFILE); SWITCH_ERROR(EMFILE); SWITCH_ERROR(ENOTTY); SWITCH_ERROR(ETXTBSY); SWITCH_ERROR(EFBIG); SWITCH_ERROR(ENOSPC); SWITCH_ERROR(ESPIPE); SWITCH_ERROR(EROFS); SWITCH_ERROR(EMLINK); SWITCH_ERROR(EPIPE); SWITCH_ERROR(EDOM); SWITCH_ERROR(ERANGE); SWITCH_ERROR(EBADMSG); } #undef SWITCH_ERROR return NULL; } #endif // AP_FILESYSTEM_FATFS_ENABLED