mbed-os5 only for TYBLE16

Dependents:   TYBLE16_simple_data_logger TYBLE16_MP3_Air

Revision:
0:5b88d5760320
Child:
1:9db0e321a9f4
diff -r 000000000000 -r 5b88d5760320 features/storage/filesystem/fat/FATFileSystem.cpp
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/features/storage/filesystem/fat/FATFileSystem.cpp	Tue Dec 17 23:23:45 2019 +0000
@@ -0,0 +1,862 @@
+/* mbed Microcontroller Library
+ * Copyright (c) 2006-2012 ARM Limited
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+ * SOFTWARE.
+ */
+#include "diskio.h"
+#include "ffconf.h"
+#include "platform/mbed_debug.h"
+#include "platform/mbed_critical.h"
+#include "filesystem/mbed_filesystem.h"
+#include "FATFileSystem.h"
+
+#include <errno.h>
+#include <stdlib.h>
+
+namespace mbed {
+
+using namespace mbed;
+
+static int fat_error_remap(FRESULT res)
+{
+    switch (res) {
+        case FR_OK:                   // (0) Succeeded
+            return 0;
+        case FR_DISK_ERR:             // (1) A hard error occurred in the low level disk I/O layer
+            return -EIO;
+        case FR_INT_ERR:              // (2) Assertion failed
+            return -1;
+        case FR_NOT_READY:            // (3) The physical drive cannot work
+            return -EIO;
+        case FR_NO_FILE:              // (4) Could not find the file
+            return -ENOENT;
+        case FR_NO_PATH:              // (5) Could not find the path
+            return -ENOTDIR;
+        case FR_INVALID_NAME:         // (6) The path name format is invalid
+            return -EINVAL;
+        case FR_DENIED:               // (7) Access denied due to prohibited access or directory full
+            return -EACCES;
+        case FR_EXIST:                // (8) Access denied due to prohibited access
+            return -EEXIST;
+        case FR_INVALID_OBJECT:       // (9) The file/directory object is invalid
+            return -EBADF;
+        case FR_WRITE_PROTECTED:      // (10) The physical drive is write protected
+            return -EACCES;
+        case FR_INVALID_DRIVE:        // (11) The logical drive number is invalid
+            return -ENODEV;
+        case FR_NOT_ENABLED:          // (12) The volume has no work area
+            return -ENODEV;
+        case FR_NO_FILESYSTEM:        // (13) There is no valid FAT volume
+            return -EINVAL;
+        case FR_MKFS_ABORTED:         // (14) The f_mkfs() aborted due to any problem
+            return -EIO;
+        case FR_TIMEOUT:              // (15) Could not get a grant to access the volume within defined period
+            return -ETIMEDOUT;
+        case FR_LOCKED:               // (16) The operation is rejected according to the file sharing policy
+            return -EBUSY;
+        case FR_NOT_ENOUGH_CORE:      // (17) LFN working buffer could not be allocated
+            return -ENOMEM;
+        case FR_TOO_MANY_OPEN_FILES:  // (18) Number of open files > FF_FS_LOCK
+            return -ENFILE;
+        case FR_INVALID_PARAMETER:    // (19) Given parameter is invalid
+            return -EINVAL;
+        default:
+            return -res;
+    }
+}
+
+// Helper class for deferring operations when variable falls out of scope
+template <typename T>
+class Deferred {
+public:
+    T _t;
+    Callback<void(T)> _ondefer;
+
+    Deferred(const Deferred &);
+    Deferred &operator=(const Deferred &);
+
+public:
+    Deferred(T t, Callback<void(T)> ondefer = NULL)
+        : _t(t), _ondefer(ondefer)
+    {
+    }
+
+    operator T()
+    {
+        return _t;
+    }
+
+    ~Deferred()
+    {
+        if (_ondefer) {
+            _ondefer(_t);
+        }
+    }
+};
+
+static void dodelete(const char *data)
+{
+    delete[] data;
+}
+
+// Adds prefix needed internally by fatfs, this can be avoided for the first fatfs
+// (id 0) otherwise a prefix of "id:/" is inserted in front of the string.
+static Deferred<const char *> fat_path_prefix(int id, const char *path)
+{
+    // We can avoid dynamic allocation when only on fatfs is in use
+    if (id == 0) {
+        return path;
+    }
+
+    // Prefix path with id, will look something like 2:/hi/hello/filehere.txt
+    char *buffer = new char[strlen("0:/") + strlen(path) + 1];
+    if (!buffer) {
+        return NULL;
+    }
+
+    buffer[0] = '0' + id;
+    buffer[1] = ':';
+    buffer[2] = '/';
+    strcpy(buffer + strlen("0:/"), path);
+    return Deferred<const char *>(buffer, dodelete);
+}
+
+////// Disk operations //////
+
+// Global access to block device from FAT driver
+static mbed::BlockDevice *_ffs[FF_VOLUMES] = {0};
+static SingletonPtr<PlatformMutex> _ffs_mutex;
+
+// FAT driver functions
+extern "C" DWORD get_fattime(void)
+{
+    time_t rawtime;
+    time(&rawtime);
+    struct tm *ptm = localtime(&rawtime);
+    return (DWORD)(ptm->tm_year - 80) << 25
+           | (DWORD)(ptm->tm_mon + 1) << 21
+           | (DWORD)(ptm->tm_mday) << 16
+           | (DWORD)(ptm->tm_hour) << 11
+           | (DWORD)(ptm->tm_min) << 5
+           | (DWORD)(ptm->tm_sec / 2);
+}
+
+extern "C" void *ff_memalloc(UINT size)
+{
+    return malloc(size);
+}
+
+extern "C" void ff_memfree(void *p)
+{
+    free(p);
+}
+
+// Implementation of diskio functions (see ChaN/diskio.h)
+static WORD disk_get_sector_size(BYTE pdrv)
+{
+    bd_size_t sector_size = _ffs[pdrv]->get_erase_size();
+    MBED_ASSERT(sector_size <= WORD(-1));
+
+    WORD ssize = sector_size;
+    if (ssize < 512) {
+        ssize = 512;
+    }
+
+    MBED_ASSERT(ssize >= FF_MIN_SS && ssize <= FF_MAX_SS);
+    MBED_ASSERT(_ffs[pdrv]->get_read_size() <= _ffs[pdrv]->get_erase_size());
+    MBED_ASSERT(_ffs[pdrv]->get_program_size() <= _ffs[pdrv]->get_erase_size());
+    return ssize;
+}
+
+static DWORD disk_get_sector_count(BYTE pdrv)
+{
+    DWORD scount = _ffs[pdrv]->size() / disk_get_sector_size(pdrv);
+    MBED_ASSERT(scount >= 64);
+    return scount;
+}
+
+extern "C" DSTATUS disk_status(BYTE pdrv)
+{
+    debug_if(FFS_DBG, "disk_status on pdrv [%d]\n", pdrv);
+    return RES_OK;
+}
+
+extern "C" DSTATUS disk_initialize(BYTE pdrv)
+{
+    debug_if(FFS_DBG, "disk_initialize on pdrv [%d]\n", pdrv);
+    return (DSTATUS)_ffs[pdrv]->init();
+}
+
+extern "C" DRESULT disk_read(BYTE pdrv, BYTE *buff, DWORD sector, UINT count)
+{
+    debug_if(FFS_DBG, "disk_read(sector %lu, count %u) on pdrv [%d]\n", sector, count, pdrv);
+    DWORD ssize = disk_get_sector_size(pdrv);
+    mbed::bd_addr_t addr = (mbed::bd_addr_t)sector * ssize;
+    mbed::bd_size_t size = (mbed::bd_size_t)count * ssize;
+    int err = _ffs[pdrv]->read(buff, addr, size);
+    return err ? RES_PARERR : RES_OK;
+}
+
+extern "C" DRESULT disk_write(BYTE pdrv, const BYTE *buff, DWORD sector, UINT count)
+{
+    debug_if(FFS_DBG, "disk_write(sector %lu, count %u) on pdrv [%d]\n", sector, count, pdrv);
+    DWORD ssize = disk_get_sector_size(pdrv);
+    mbed::bd_addr_t addr = (mbed::bd_addr_t)sector * ssize;
+    mbed::bd_size_t size = (mbed::bd_size_t)count * ssize;
+
+    int err = _ffs[pdrv]->erase(addr, size);
+    if (err) {
+        return RES_PARERR;
+    }
+
+    err = _ffs[pdrv]->program(buff, addr, size);
+    if (err) {
+        return RES_PARERR;
+    }
+
+    return RES_OK;
+}
+
+extern "C" DRESULT disk_ioctl(BYTE pdrv, BYTE cmd, void *buff)
+{
+    debug_if(FFS_DBG, "disk_ioctl(%d)\n", cmd);
+    switch (cmd) {
+        case CTRL_SYNC:
+            if (_ffs[pdrv] == NULL) {
+                return RES_NOTRDY;
+            } else {
+                return RES_OK;
+            }
+        case GET_SECTOR_COUNT:
+            if (_ffs[pdrv] == NULL) {
+                return RES_NOTRDY;
+            } else {
+                *((DWORD *)buff) = disk_get_sector_count(pdrv);
+                return RES_OK;
+            }
+        case GET_SECTOR_SIZE:
+            if (_ffs[pdrv] == NULL) {
+                return RES_NOTRDY;
+            } else {
+                *((WORD *)buff) = disk_get_sector_size(pdrv);
+                return RES_OK;
+            }
+        case GET_BLOCK_SIZE:
+            *((DWORD *)buff) = 1; // default when not known
+            return RES_OK;
+        case CTRL_TRIM:
+            if (_ffs[pdrv] == NULL) {
+                return RES_NOTRDY;
+            } else {
+                DWORD *sectors = (DWORD *)buff;
+                DWORD ssize = disk_get_sector_size(pdrv);
+                mbed::bd_addr_t addr = (mbed::bd_addr_t)sectors[0] * ssize;
+                mbed::bd_size_t size = (mbed::bd_size_t)(sectors[1] - sectors[0] + 1) * ssize;
+                int err = _ffs[pdrv]->trim(addr, size);
+                return err ? RES_PARERR : RES_OK;
+            }
+    }
+
+    return RES_PARERR;
+}
+
+////// Generic filesystem operations //////
+
+// Filesystem implementation (See FATFilySystem.h)
+FATFileSystem::FATFileSystem(const char *name, BlockDevice *bd)
+    : FileSystem(name), _id(-1)
+{
+    if (bd) {
+        mount(bd);
+    }
+}
+
+FATFileSystem::~FATFileSystem()
+{
+    // nop if unmounted
+    unmount();
+}
+
+int FATFileSystem::mount(BlockDevice *bd)
+{
+    // requires duplicate definition to allow virtual overload to work
+    return mount(bd, true);
+}
+
+int FATFileSystem::mount(BlockDevice *bd, bool mount)
+{
+    lock();
+    if (_id != -1) {
+        unlock();
+        return -EINVAL;
+    }
+
+    for (int i = 0; i < FF_VOLUMES; i++) {
+        if (!_ffs[i]) {
+            _id = i;
+            _ffs[_id] = bd;
+            _fsid[0] = '0' + _id;
+            _fsid[1] = ':';
+            _fsid[2] = '\0';
+            debug_if(FFS_DBG, "Mounting [%s] on ffs drive [%s]\n", getName(), _fsid);
+            FRESULT res = f_mount(&_fs, _fsid, mount);
+            unlock();
+            return fat_error_remap(res);
+        }
+    }
+
+    unlock();
+    return -ENOMEM;
+}
+
+int FATFileSystem::unmount()
+{
+    lock();
+    if (_id == -1) {
+        unlock();
+        return -EINVAL;
+    }
+
+    FRESULT res = f_mount(NULL, _fsid, 0);
+    _ffs[_id] = NULL;
+    _id = -1;
+    unlock();
+    return fat_error_remap(res);
+}
+
+/* See http://elm-chan.org/fsw/ff/en/mkfs.html for details of f_mkfs() and
+ * associated arguments. */
+int FATFileSystem::format(BlockDevice *bd, bd_size_t cluster_size)
+{
+    FATFileSystem fs;
+    fs.lock();
+
+    int err = bd->init();
+    if (err) {
+        fs.unlock();
+        return err;
+    }
+
+    // erase first handful of blocks
+    bd_size_t header = 2 * bd->get_erase_size();
+    err = bd->erase(0, header);
+    if (err) {
+        bd->deinit();
+        fs.unlock();
+        return err;
+    }
+
+    if (bd->get_erase_value() < 0) {
+        // erase is unknown, need to write 1s
+        bd_size_t program_size = bd->get_program_size();
+        void *buf = malloc(program_size);
+        if (!buf) {
+            bd->deinit();
+            fs.unlock();
+            return -ENOMEM;
+        }
+
+        memset(buf, 0xff, program_size);
+
+        for (bd_addr_t i = 0; i < header; i += program_size) {
+            err = bd->program(buf, i, program_size);
+            if (err) {
+                free(buf);
+                bd->deinit();
+                fs.unlock();
+                return err;
+            }
+        }
+
+        free(buf);
+    }
+
+    // trim entire device to indicate it is unneeded
+    err = bd->trim(0, bd->size());
+    if (err) {
+        bd->deinit();
+        fs.unlock();
+        return err;
+    }
+
+    err = bd->deinit();
+    if (err) {
+        fs.unlock();
+        return err;
+    }
+
+    err = fs.mount(bd, false);
+    if (err) {
+        fs.unlock();
+        return err;
+    }
+
+    // Logical drive number, Partitioning rule, Allocation unit size (bytes per cluster)
+    FRESULT res = f_mkfs(fs._fsid, FM_ANY | FM_SFD, cluster_size, NULL, 0);
+    if (res != FR_OK) {
+        fs.unmount();
+        fs.unlock();
+        return fat_error_remap(res);
+    }
+
+    err = fs.unmount();
+    if (err) {
+        fs.unlock();
+        return err;
+    }
+
+    fs.unlock();
+    return 0;
+}
+
+int FATFileSystem::reformat(BlockDevice *bd, int allocation_unit)
+{
+    lock();
+    if (_id != -1) {
+        if (!bd) {
+            bd = _ffs[_id];
+        }
+
+        int err = unmount();
+        if (err) {
+            unlock();
+            return err;
+        }
+    }
+
+    if (!bd) {
+        unlock();
+        return -ENODEV;
+    }
+
+    int err = FATFileSystem::format(bd, allocation_unit);
+    if (err) {
+        unlock();
+        return err;
+    }
+
+    err = mount(bd);
+    unlock();
+    return err;
+}
+
+int FATFileSystem::remove(const char *path)
+{
+    Deferred<const char *> fpath = fat_path_prefix(_id, path);
+
+    lock();
+    FRESULT res = f_unlink(fpath);
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "f_unlink() failed: %d\n", res);
+        if (res == FR_DENIED) {
+            return -ENOTEMPTY;
+        }
+    }
+    return fat_error_remap(res);
+}
+
+int FATFileSystem::rename(const char *oldpath, const char *newpath)
+{
+    Deferred<const char *> oldfpath = fat_path_prefix(_id, oldpath);
+    Deferred<const char *> newfpath = fat_path_prefix(_id, newpath);
+
+    lock();
+    FRESULT res = f_rename(oldfpath, newfpath);
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "f_rename() failed: %d\n", res);
+    }
+    return fat_error_remap(res);
+}
+
+int FATFileSystem::mkdir(const char *path, mode_t mode)
+{
+    Deferred<const char *> fpath = fat_path_prefix(_id, path);
+
+    lock();
+    FRESULT res = f_mkdir(fpath);
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "f_mkdir() failed: %d\n", res);
+    }
+    return fat_error_remap(res);
+}
+
+int FATFileSystem::stat(const char *path, struct stat *st)
+{
+    Deferred<const char *> fpath = fat_path_prefix(_id, path);
+
+    lock();
+    FILINFO f;
+    memset(&f, 0, sizeof(f));
+
+    FRESULT res = f_stat(fpath, &f);
+    if (res != FR_OK) {
+        unlock();
+        return fat_error_remap(res);
+    }
+
+    /* ARMCC doesnt support stat(), and these symbols are not defined by the toolchain. */
+#ifdef TOOLCHAIN_GCC
+    st->st_size = f.fsize;
+    st->st_mode = 0;
+    st->st_mode |= (f.fattrib & AM_DIR) ? S_IFDIR : S_IFREG;
+    st->st_mode |= (f.fattrib & AM_RDO) ?
+                   (S_IRUSR | S_IXUSR | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH) :
+                   (S_IRWXU | S_IRWXG | S_IRWXO);
+#endif /* TOOLCHAIN_GCC */
+    unlock();
+
+    return 0;
+}
+
+int FATFileSystem::statvfs(const char *path, struct statvfs *buf)
+{
+
+    memset(buf, 0, sizeof(struct statvfs));
+    FATFS *fs;
+    DWORD fre_clust;
+
+    lock();
+    FRESULT res = f_getfree(_fsid, &fre_clust, &fs);
+    if (res != FR_OK) {
+        unlock();
+        return fat_error_remap(res);
+    }
+
+    buf->f_bsize = fs->ssize;
+    buf->f_frsize = fs->ssize;
+    buf->f_blocks = (fs->n_fatent - 2) * fs->csize;
+    buf->f_bfree = fre_clust * fs->csize;
+    buf->f_bavail = buf->f_bfree;
+#if FF_USE_LFN
+    buf->f_namemax = FF_LFN_BUF;
+#else
+    buf->f_namemax = FF_SFN_BUF;
+#endif
+
+    unlock();
+    return 0;
+}
+
+void FATFileSystem::lock()
+{
+    _ffs_mutex->lock();
+}
+
+void FATFileSystem::unlock()
+{
+    _ffs_mutex->unlock();
+}
+
+
+////// File operations //////
+int FATFileSystem::file_open(fs_file_t *file, const char *path, int flags)
+{
+    debug_if(FFS_DBG, "open(%s) on filesystem [%s], drv [%d]\n", path, getName(), _id);
+
+    FIL *fh = new FIL;
+    Deferred<const char *> fpath = fat_path_prefix(_id, path);
+
+    /* POSIX flags -> FatFS open mode */
+    BYTE openmode;
+    if (flags & O_RDWR) {
+        openmode = FA_READ | FA_WRITE;
+    } else if (flags & O_WRONLY) {
+        openmode = FA_WRITE;
+    } else {
+        openmode = FA_READ;
+    }
+
+    if (flags & O_CREAT) {
+        if (flags & O_TRUNC) {
+            openmode |= FA_CREATE_ALWAYS;
+        } else {
+            openmode |= FA_OPEN_ALWAYS;
+        }
+    }
+
+    if (flags & O_APPEND) {
+        openmode |= FA_OPEN_APPEND;
+    }
+
+    lock();
+    FRESULT res = f_open(fh, fpath, openmode);
+
+    if (res != FR_OK) {
+        unlock();
+        debug_if(FFS_DBG, "f_open('w') failed: %d\n", res);
+        delete fh;
+        return fat_error_remap(res);
+    }
+
+    unlock();
+
+    *file = fh;
+    return 0;
+}
+
+int FATFileSystem::file_close(fs_file_t file)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    FRESULT res = f_close(fh);
+    unlock();
+
+    delete fh;
+    return fat_error_remap(res);
+}
+
+ssize_t FATFileSystem::file_read(fs_file_t file, void *buffer, size_t len)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    UINT n;
+    FRESULT res = f_read(fh, buffer, len, &n);
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "f_read() failed: %d\n", res);
+        return fat_error_remap(res);
+    } else {
+        return n;
+    }
+}
+
+ssize_t FATFileSystem::file_write(fs_file_t file, const void *buffer, size_t len)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    UINT n;
+    FRESULT res = f_write(fh, buffer, len, &n);
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "f_write() failed: %d", res);
+        return fat_error_remap(res);
+    } else {
+        return n;
+    }
+}
+
+int FATFileSystem::file_sync(fs_file_t file)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    FRESULT res = f_sync(fh);
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "f_sync() failed: %d\n", res);
+    }
+    return fat_error_remap(res);
+}
+
+off_t FATFileSystem::file_seek(fs_file_t file, off_t offset, int whence)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    if (whence == SEEK_END) {
+        offset += f_size(fh);
+    } else if (whence == SEEK_CUR) {
+        offset += f_tell(fh);
+    }
+
+    FRESULT res = f_lseek(fh, offset);
+    off_t noffset = fh->fptr;
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "lseek failed: %d\n", res);
+        return fat_error_remap(res);
+    } else {
+        return noffset;
+    }
+}
+
+off_t FATFileSystem::file_tell(fs_file_t file)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    off_t res = f_tell(fh);
+    unlock();
+
+    return res;
+}
+
+off_t FATFileSystem::file_size(fs_file_t file)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    off_t res = f_size(fh);
+    unlock();
+
+    return res;
+}
+
+int FATFileSystem::file_truncate(fs_file_t file, off_t length)
+{
+    FIL *fh = static_cast<FIL *>(file);
+
+    lock();
+    // save current position
+    FSIZE_t oldoff = f_tell(fh);
+
+    // seek to new file size and truncate
+    FRESULT res = f_lseek(fh, length);
+    if (res) {
+        unlock();
+        return fat_error_remap(res);
+    }
+
+    res = f_truncate(fh);
+    if (res) {
+        unlock();
+        return fat_error_remap(res);
+    }
+
+    // restore old position
+    res = f_lseek(fh, oldoff);
+    if (res) {
+        unlock();
+        return fat_error_remap(res);
+    }
+
+    return 0;
+}
+
+
+////// Dir operations //////
+int FATFileSystem::dir_open(fs_dir_t *dir, const char *path)
+{
+    FATFS_DIR *dh = new FATFS_DIR;
+    Deferred<const char *> fpath = fat_path_prefix(_id, path);
+
+    lock();
+    FRESULT res = f_opendir(dh, fpath);
+    unlock();
+
+    if (res != FR_OK) {
+        debug_if(FFS_DBG, "f_opendir() failed: %d\n", res);
+        delete dh;
+        return fat_error_remap(res);
+    }
+
+    *dir = dh;
+    return 0;
+}
+
+int FATFileSystem::dir_close(fs_dir_t dir)
+{
+    FATFS_DIR *dh = static_cast<FATFS_DIR *>(dir);
+
+    lock();
+    FRESULT res = f_closedir(dh);
+    unlock();
+
+    delete dh;
+    return fat_error_remap(res);
+}
+
+ssize_t FATFileSystem::dir_read(fs_dir_t dir, struct dirent *ent)
+{
+    FATFS_DIR *dh = static_cast<FATFS_DIR *>(dir);
+    FILINFO finfo;
+
+    lock();
+    FRESULT res = f_readdir(dh, &finfo);
+    unlock();
+
+    if (res != FR_OK) {
+        return fat_error_remap(res);
+    } else if (finfo.fname[0] == 0) {
+        return 0;
+    }
+
+    ent->d_type = (finfo.fattrib & AM_DIR) ? DT_DIR : DT_REG;
+
+#if FF_USE_LFN
+    if (ent->d_name[0] == 0) {
+        // No long filename so use short filename.
+        strncpy(ent->d_name, finfo.fname, FF_LFN_BUF);
+    }
+#else
+    strncpy(ent->d_name, finfo.fname, FF_SFN_BUF);
+#endif
+
+    return 1;
+}
+
+void FATFileSystem::dir_seek(fs_dir_t dir, off_t offset)
+{
+    FATFS_DIR *dh = static_cast<FATFS_DIR *>(dir);
+    off_t dptr = static_cast<off_t>(dh->dptr);
+
+    lock();
+
+    if (offset < dptr) {
+        f_rewinddir(dh);
+    }
+    while (dptr < offset) {
+        FILINFO finfo;
+        FRESULT res;
+
+        res = f_readdir(dh, &finfo);
+        dptr = dh->dptr;
+        if (res != FR_OK) {
+            break;
+        } else if (finfo.fname[0] == 0) {
+            break;
+        }
+    }
+    unlock();
+}
+
+off_t FATFileSystem::dir_tell(fs_dir_t dir)
+{
+    FATFS_DIR *dh = static_cast<FATFS_DIR *>(dir);
+
+    lock();
+    off_t offset = dh->dptr;
+    unlock();
+
+    return offset;
+}
+
+void FATFileSystem::dir_rewind(fs_dir_t dir)
+{
+    FATFS_DIR *dh = static_cast<FATFS_DIR *>(dir);
+
+    lock();
+    f_rewinddir(dh);
+    unlock();
+}
+
+} // namespace mbed