#include "luat_base.h" #include "luat_fs.h" #include "luat_mem.h" #define LUAT_LOG_TAG "fatfs" #include "luat_log.h" #if defined(LUAT_USE_FS_VFS) && defined(LUAT_USE_FATFS) #include "ff.h" #include "diskio.h" FILE* luat_vfs_fatfs_fopen(void* userdata, const char *filename, const char *mode) { //LLOGD("f_open %s %s", filename, mode); //FATFS *fs = (FATFS*)userdata; FIL* fp = luat_heap_malloc(sizeof(FIL)); int flag = 0; for (size_t i = 0; i < strlen(mode); i++) { char m = *(mode + i); switch (m) { case 'r': flag |= FA_READ; break; case 'w': flag |= FA_WRITE | FA_CREATE_ALWAYS; break; case 'a': flag |= FA_OPEN_APPEND | FA_WRITE; break; case '+': flag |= FA_OPEN_APPEND; break; default: break; } } FRESULT ret = f_open(fp, filename, flag); if (ret != FR_OK) { LLOGD("f_open %s %d", filename, ret); luat_heap_free(fp); return 0; } return (FILE*)fp; } int luat_vfs_fatfs_getc(void* userdata, FILE* stream) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; char buff = 0; UINT result = 0; FRESULT ret = f_read(fp, (void*)&buff, 1, &result); if (ret == FR_OK && result == 1) { return buff; } return -1; } int luat_vfs_fatfs_fseek(void* userdata, FILE* stream, long int offset, int origin) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; int npos = f_tell(fp); if (origin == SEEK_SET) { npos = offset; } else if (origin == SEEK_CUR) { npos += offset; } else if (origin == SEEK_END) { npos = f_size(fp); } FRESULT ret = f_lseek(fp, npos); if (ret == FR_OK) { return 0; } return -1; } int luat_vfs_fatfs_ftell(void* userdata, FILE* stream) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; return f_tell(fp); } int luat_vfs_fatfs_fclose(void* userdata, FILE* stream) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; if (fp != NULL) { f_close(fp); luat_heap_free(fp); } return 0; } int luat_vfs_fatfs_feof(void* userdata, FILE* stream) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; return f_eof(fp); } int luat_vfs_fatfs_ferror(void* userdata, FILE *stream) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; return f_error(fp); } size_t luat_vfs_fatfs_fread(void* userdata, void *ptr, size_t size, size_t nmemb, FILE *stream) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; UINT result = 0; FRESULT ret = f_read(fp, ptr, size*nmemb, &result); if (ret == FR_OK) { return result; } return 0; } size_t luat_vfs_fatfs_fwrite(void* userdata, const void *ptr, size_t size, size_t nmemb, FILE *stream) { //FATFS *fs = (FATFS*)userdata; FIL* fp = (FIL*)stream; UINT result = 0; FRESULT ret = f_write(fp, ptr, size*nmemb, &result); if (ret == FR_OK) { return result; } return 0; } int luat_vfs_fatfs_remove(void* userdata, const char *filename) { return f_unlink(filename); } int luat_vfs_fatfs_rename(void* userdata, const char *old_filename, const char *new_filename) { return f_rename(old_filename + (old_filename[0] == '/' ? 1 : 0), new_filename + (new_filename[0] == '/' ? 1 : 0)); } int luat_vfs_fatfs_fexist(void* userdata, const char *filename) { FILINFO fno = {0}; FRESULT ret = f_stat(filename, &fno); if (ret == FR_OK) { return 1; } return 0; } size_t luat_vfs_fatfs_fsize(void* userdata, const char *filename) { FILINFO fno = {0}; FRESULT ret = f_stat(filename, &fno); if (ret == FR_OK) { return fno.fsize; } return 0; } int luat_vfs_fatfs_mkfs(void* userdata, luat_fs_conf_t *conf) { (void)userdata; (void)conf; MKFS_PARM parm = { .fmt = FM_ANY, // 暂时应付一下ramdisk .au_size = 0, .align = 0, .n_fat = 0, .n_root = 0, }; BYTE work[FF_MAX_SS] = {0}; int re = f_mkfs("/", &parm, work, FF_MAX_SS); LLOGD("format ret %d", re); return re; } int luat_vfs_fatfs_mount(void** userdata, luat_fs_conf_t *conf) { *userdata = (void*)conf->busname; return 0; } int luat_vfs_fatfs_umount(void* userdata, luat_fs_conf_t *conf) { //LLOGE("not support yet : umount"); return 0; } int luat_vfs_fatfs_mkdir(void* userdata, char const* _DirName) { FRESULT ret = f_mkdir(_DirName); if (FR_OK == ret) return 0; LLOGD("mkdir ret %d", ret); return -1; } int luat_vfs_fatfs_rmdir(void* userdata, char const* _DirName) { FRESULT ret = f_rmdir(_DirName); if (FR_OK == ret) return 0; LLOGD("rmdir ret %d", ret); return -1; } int luat_vfs_fatfs_lsdir(void* userdata, char const* _DirName, luat_fs_dirent_t* ents, size_t offset, size_t len) { DIR dir = {0}; FRESULT res = 0; FILINFO fno = {0}; size_t i = 0; size_t count = 0; res = f_opendir(&dir, _DirName); if (res != FR_OK) { LLOGD("opendir %s %d", _DirName, res); return 0; } for (i = 0; i < offset; i++) { res = f_readdir(&dir, &fno); if (res != FR_OK || fno.fname[0] == 0) { return 0; // 读取失败或者是读完了 }; } for (i = 0; i < len; i++) { res = f_readdir(&dir, &fno); if (res != FR_OK || fno.fname[0] == 0) { break; }; count ++; if (fno.fattrib & AM_DIR) { ents[i].d_type = 1; } else { ents[i].d_type = 0; } if (strlen(fno.fname) < 255) memcpy(&ents[i].d_name, fno.fname, strlen(fno.fname) + 1); else { memcpy(ents[i].d_name, fno.fname, 254); ents[i].d_name[254] = 0; } } return count; } int luat_vfs_fatfs_info(void* userdata, const char* path, luat_fs_info_t *conf) { // DWORD fre_clust = 0; // DWORD fre_sect = 0 // DWORD tot_sect = 0; FATFS *fs = (FATFS*)userdata; // tot_sect = (fs->n_fatent - 2) * fs->csize; // fre_sect = (fs->free_clst) * fs->csize; memcpy(conf->filesystem, "fatfs", strlen("fatfs")+1); conf->type = 0; conf->total_block = (fs->n_fatent - 2) * fs->csize; conf->block_used = (fs->n_fatent - 2 - fs->free_clst) * fs->csize; conf->block_size = 512; return 0; } void* luat_vfs_fatfs_opendir(void* userdata, const char *_DirName) { DIR *dp = (DIR*)luat_heap_malloc(sizeof(DIR)); if (dp == NULL) { LLOGD("out of memory when open file %s", dp); return NULL; } memset(dp, 0, sizeof(DIR)); FRESULT ret = f_opendir(dp, _DirName); if (ret != FR_OK) { luat_heap_free(dp); return NULL; } return (void*)dp; } int luat_vfs_fatfs_closedir(void* userdata, void* dir) { DIR* dp = (DIR*)dir; f_closedir(dp); if (dp)luat_heap_free(dp); return 0; } #define T(name) .name = luat_vfs_fatfs_##name const struct luat_vfs_filesystem vfs_fs_fatfs = { .name = "fatfs", .opts = { T(mkfs), T(mount), T(umount), T(mkdir), T(rmdir), T(lsdir), T(remove), T(rename), T(fsize), T(fexist), T(info), T(opendir), T(closedir) }, .fopts = { T(fopen), T(getc), T(fseek), T(ftell), T(fclose), T(feof), T(ferror), T(fread), T(fwrite) } }; #endif