2013-02-18 15:32:11 +00:00
|
|
|
/* 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 "mbed.h"
|
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
#include "diskio.h"
|
2013-02-18 15:32:11 +00:00
|
|
|
#include "ffconf.h"
|
|
|
|
#include "mbed_debug.h"
|
|
|
|
|
|
|
|
#include "FATFileSystem.h"
|
|
|
|
#include "FATFileHandle.h"
|
|
|
|
#include "FATDirHandle.h"
|
2017-01-20 11:53:40 +00:00
|
|
|
//<<<<<<< HEAD
|
2017-01-20 15:23:39 +00:00
|
|
|
//<<<<<<< HEAD
|
2017-01-20 20:09:38 +00:00
|
|
|
//<<<<<<< HEAD
|
2017-02-03 14:06:27 +00:00
|
|
|
//<<<<<<< HEAD
|
2017-01-27 11:10:28 +00:00
|
|
|
#include "mbed_critical.h"
|
2017-01-20 11:53:40 +00:00
|
|
|
//=======
|
|
|
|
//#include "critical.h"
|
2017-01-20 15:23:39 +00:00
|
|
|
//=======
|
|
|
|
//#include "critical.h"
|
|
|
|
#include "ff.h"
|
|
|
|
//>>>>>>> Added errno codes to retarget, mkdir() and ftell() tests.
|
2017-01-20 20:09:38 +00:00
|
|
|
//=======
|
|
|
|
//#include "critical.h"
|
|
|
|
//>>>>>>> bd: Adopted the block storage api in the FATFileSystem
|
2017-01-20 11:53:40 +00:00
|
|
|
#include <errno.h>
|
2017-02-03 11:01:39 +00:00
|
|
|
//<<<<<<< HEAD
|
2017-01-20 11:53:40 +00:00
|
|
|
//>>>>>>> Filesystem: Added EEXIST reporting to mkdir through errno
|
2017-02-03 11:01:39 +00:00
|
|
|
//=======
|
|
|
|
/* toolchain_support.h is included after errno.h so symbols are mapped to
|
|
|
|
* consistent values for all toolchains */
|
|
|
|
#include "toolchain_support.h"
|
|
|
|
//>>>>>>> STORAGE: test case fixes to support ARMCC and IAR toolchains.
|
2017-02-03 14:06:27 +00:00
|
|
|
=======
|
|
|
|
//#include "critical.h"
|
|
|
|
#include "FATMisc.h"
|
|
|
|
//>>>>>>> STORAGE: fixing error handling for fopen() and other upper edge filesystem API methods.
|
2013-02-18 15:32:11 +00:00
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
|
|
|
|
// Global access to block device from FAT driver
|
|
|
|
static BlockDevice *_ffs[_VOLUMES] = {0};
|
|
|
|
static SingletonPtr<PlatformMutex> _ffs_mutex;
|
|
|
|
|
|
|
|
|
|
|
|
// FAT driver functions
|
2013-02-18 15:32:11 +00:00
|
|
|
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 );
|
|
|
|
}
|
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
// Implementation of diskio functions (see ChaN/diskio.h)
|
|
|
|
DSTATUS disk_status(BYTE pdrv) {
|
|
|
|
debug_if(FFS_DBG, "disk_status on pdrv [%d]\n", pdrv);
|
|
|
|
return RES_OK;
|
|
|
|
}
|
2013-02-18 15:32:11 +00:00
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
DSTATUS disk_initialize(BYTE pdrv) {
|
|
|
|
debug_if(FFS_DBG, "disk_initialize on pdrv [%d]\n", pdrv);
|
|
|
|
return (DSTATUS)_ffs[pdrv]->init();
|
|
|
|
}
|
2016-06-08 16:17:14 +00:00
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
DRESULT disk_read(BYTE pdrv, BYTE *buff, DWORD sector, UINT count) {
|
|
|
|
debug_if(FFS_DBG, "disk_read(sector %d, count %d) on pdrv [%d]\n", sector, count, pdrv);
|
|
|
|
bd_size_t ssize = _ffs[pdrv]->get_write_size();
|
|
|
|
int err = _ffs[pdrv]->read(buff, sector*ssize, count*ssize);
|
|
|
|
return err ? RES_PARERR : RES_OK;
|
|
|
|
}
|
2016-06-08 16:17:14 +00:00
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
DRESULT disk_write(BYTE pdrv, const BYTE *buff, DWORD sector, UINT count) {
|
|
|
|
debug_if(FFS_DBG, "disk_write(sector %d, count %d) on pdrv [%d]\n", sector, count, pdrv);
|
|
|
|
bd_size_t ssize = _ffs[pdrv]->get_write_size();
|
|
|
|
int err = _ffs[pdrv]->write(buff, sector*ssize, count*ssize);
|
|
|
|
return err ? RES_PARERR : RES_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
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 count = _ffs[pdrv]->size() / _ffs[pdrv]->get_write_size();
|
|
|
|
*((DWORD*)buff) = count;
|
|
|
|
return RES_OK;
|
|
|
|
}
|
|
|
|
case GET_SECTOR_SIZE:
|
|
|
|
if (_ffs[pdrv] == NULL) {
|
|
|
|
return RES_NOTRDY;
|
|
|
|
} else {
|
|
|
|
DWORD size = _ffs[pdrv]->get_write_size();
|
|
|
|
*((DWORD*)buff) = size;
|
|
|
|
return RES_OK;
|
|
|
|
}
|
|
|
|
case GET_BLOCK_SIZE:
|
|
|
|
*((DWORD*)buff) = 1; // default when not known
|
|
|
|
return RES_OK;
|
2016-06-08 16:17:14 +00:00
|
|
|
}
|
2017-01-20 20:09:38 +00:00
|
|
|
|
|
|
|
return RES_PARERR;
|
2016-06-08 16:17:14 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
// Filesystem implementation (See FATFilySystem.h)
|
|
|
|
FATFileSystem::FATFileSystem(const char *n, BlockDevice *bd)
|
|
|
|
: FileSystemLike(n), _id(-1) {
|
|
|
|
if (bd) {
|
|
|
|
mount(bd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FATFileSystem::~FATFileSystem() {
|
|
|
|
// nop if unmounted
|
|
|
|
unmount();
|
|
|
|
}
|
|
|
|
|
|
|
|
int FATFileSystem::mount(BlockDevice *bd, bool force) {
|
2016-06-08 16:17:14 +00:00
|
|
|
lock();
|
2017-01-20 20:09:38 +00:00
|
|
|
if (_id != -1) {
|
|
|
|
unlock();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < _VOLUMES; i++) {
|
|
|
|
if (!_ffs[i]) {
|
|
|
|
_id = i;
|
|
|
|
_ffs[_id] = bd;
|
|
|
|
_fsid[0] = '0' + _id;
|
2015-07-07 20:41:05 +00:00
|
|
|
_fsid[1] = '\0';
|
2016-06-08 16:17:14 +00:00
|
|
|
debug_if(FFS_DBG, "Mounting [%s] on ffs drive [%s]\n", getName(), _fsid);
|
2017-01-20 20:09:38 +00:00
|
|
|
FRESULT res = f_mount(&_fs, _fsid, force);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2017-01-20 20:09:38 +00:00
|
|
|
return res == 0 ? 0 : -1;
|
2013-02-18 15:32:11 +00:00
|
|
|
}
|
|
|
|
}
|
2017-01-20 20:09:38 +00:00
|
|
|
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2017-01-20 20:09:38 +00:00
|
|
|
return -1;
|
2013-02-18 15:32:11 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
int FATFileSystem::unmount() {
|
2016-06-08 16:17:14 +00:00
|
|
|
lock();
|
2017-01-20 20:09:38 +00:00
|
|
|
if (_id == -1) {
|
|
|
|
unlock();
|
|
|
|
return -1;
|
2013-02-18 15:32:11 +00:00
|
|
|
}
|
2017-01-20 20:09:38 +00:00
|
|
|
|
|
|
|
FRESULT res = f_mount(NULL, _fsid, 0);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2017-01-20 20:09:38 +00:00
|
|
|
_ffs[_id] = NULL;
|
|
|
|
_id = -1;
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2017-01-20 20:09:38 +00:00
|
|
|
return res == 0 ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int FATFileSystem::sync() {
|
|
|
|
lock();
|
|
|
|
if (_id == -1) {
|
|
|
|
unlock();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Always synchronized
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(FR_OK);
|
2017-01-20 20:09:38 +00:00
|
|
|
unlock();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-06 14:30:24 +00:00
|
|
|
/* See http://elm-chan.org/fsw/ff/en/mkfs.html for details of f_mkfs() and
|
|
|
|
* associated arguments. */
|
|
|
|
int FATFileSystem::format(BlockDevice *bd, int allocation_unit) {
|
2017-02-07 11:59:20 +00:00
|
|
|
FATFileSystem fs("");
|
|
|
|
int err = fs.mount(bd, false);
|
|
|
|
if (err) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-01-20 20:09:38 +00:00
|
|
|
// Logical drive number, Partitioning rule, Allocation unit size (bytes per cluster)
|
2017-02-07 11:59:20 +00:00
|
|
|
fs.lock();
|
|
|
|
FRESULT res = f_mkfs(fs._fsid, 0, allocation_unit);
|
|
|
|
fat_filesystem_set_errno(res);
|
|
|
|
fs.unlock();
|
|
|
|
|
|
|
|
err = fs.unmount();
|
|
|
|
if (err) {
|
|
|
|
return -1;
|
|
|
|
}
|
2017-02-06 14:30:24 +00:00
|
|
|
return res == 0 ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
FileHandle *FATFileSystem::open(const char* name, int flags) {
|
2016-06-08 16:17:14 +00:00
|
|
|
lock();
|
|
|
|
debug_if(FFS_DBG, "open(%s) on filesystem [%s], drv [%s]\n", name, getName(), _fsid);
|
2013-02-18 15:32:11 +00:00
|
|
|
char n[64];
|
2015-07-07 20:41:05 +00:00
|
|
|
sprintf(n, "%s:/%s", _fsid, name);
|
2014-05-29 13:39:05 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
}
|
2014-05-29 13:39:05 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
FIL fh;
|
|
|
|
FRESULT res = f_open(&fh, n, openmode);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2014-05-29 13:39:05 +00:00
|
|
|
if (res) {
|
2013-02-18 15:32:11 +00:00
|
|
|
debug_if(FFS_DBG, "f_open('w') failed: %d\n", res);
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2013-02-18 15:32:11 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (flags & O_APPEND) {
|
|
|
|
f_lseek(&fh, fh.fsize);
|
|
|
|
}
|
2017-01-20 20:09:38 +00:00
|
|
|
FATFileHandle *handle = new FATFileHandle(fh, _ffs_mutex.get());
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
|
|
|
return handle;
|
2013-02-18 15:32:11 +00:00
|
|
|
}
|
2014-05-29 13:39:05 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
int FATFileSystem::remove(const char *filename) {
|
2016-06-08 16:17:14 +00:00
|
|
|
lock();
|
2013-02-18 15:32:11 +00:00
|
|
|
FRESULT res = f_unlink(filename);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2014-05-29 13:39:05 +00:00
|
|
|
if (res) {
|
2013-02-18 15:32:11 +00:00
|
|
|
debug_if(FFS_DBG, "f_unlink() failed: %d\n", res);
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2013-02-18 15:32:11 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2013-02-18 15:32:11 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-08-13 09:19:03 +00:00
|
|
|
int FATFileSystem::rename(const char *oldname, const char *newname) {
|
2016-06-08 16:17:14 +00:00
|
|
|
lock();
|
2014-08-13 09:19:03 +00:00
|
|
|
FRESULT res = f_rename(oldname, newname);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2014-08-13 09:19:03 +00:00
|
|
|
if (res) {
|
|
|
|
debug_if(FFS_DBG, "f_rename() failed: %d\n", res);
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2014-08-13 09:19:03 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2014-08-13 09:19:03 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
DirHandle *FATFileSystem::opendir(const char *name) {
|
2016-06-08 16:17:14 +00:00
|
|
|
lock();
|
2013-02-18 15:32:11 +00:00
|
|
|
FATFS_DIR dir;
|
|
|
|
FRESULT res = f_opendir(&dir, name);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2013-02-18 15:32:11 +00:00
|
|
|
if (res != 0) {
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2013-02-18 15:32:11 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2017-01-20 20:09:38 +00:00
|
|
|
FATDirHandle *handle = new FATDirHandle(dir, _ffs_mutex.get());
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
|
|
|
return handle;
|
2013-02-18 15:32:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int FATFileSystem::mkdir(const char *name, mode_t mode) {
|
2016-06-08 16:17:14 +00:00
|
|
|
lock();
|
2013-02-18 15:32:11 +00:00
|
|
|
FRESULT res = f_mkdir(name);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2013-02-18 15:32:11 +00:00
|
|
|
return res == 0 ? 0 : -1;
|
|
|
|
}
|
2014-07-30 16:22:23 +00:00
|
|
|
|
2017-01-19 15:30:06 +00:00
|
|
|
int FATFileSystem::stat(const char *name, struct stat *st) {
|
|
|
|
lock();
|
|
|
|
FILINFO f;
|
|
|
|
memset(&f, 0, sizeof(f));
|
|
|
|
|
|
|
|
FRESULT res = f_stat(name, &f);
|
2017-02-07 11:59:20 +00:00
|
|
|
fat_filesystem_set_errno(res);
|
2017-01-19 15:30:06 +00:00
|
|
|
if (res != 0) {
|
|
|
|
unlock();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-02-03 11:01:39 +00:00
|
|
|
/* ARMCC doesnt support stat(), and these symbols are not defined by the toolchain. */
|
|
|
|
#ifdef TOOLCHAIN_GCC
|
2017-01-19 15:30:06 +00:00
|
|
|
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);
|
2017-02-03 11:01:39 +00:00
|
|
|
#endif /* TOOLCHAIN_GCC */
|
2016-06-08 16:17:14 +00:00
|
|
|
unlock();
|
2014-07-30 16:22:23 +00:00
|
|
|
return res == 0 ? 0 : -1;
|
|
|
|
}
|
2016-06-08 16:17:14 +00:00
|
|
|
|
|
|
|
void FATFileSystem::lock() {
|
2017-01-20 20:09:38 +00:00
|
|
|
_ffs_mutex->lock();
|
2016-06-08 16:17:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FATFileSystem::unlock() {
|
2017-01-20 20:09:38 +00:00
|
|
|
_ffs_mutex->unlock();
|
2016-06-08 16:17:14 +00:00
|
|
|
}
|
2017-02-07 11:59:20 +00:00
|
|
|
|