| |
| #include "ff.h" |
| #include "diskio.h" |
| |
| #include "fatfs_blit_api.hpp" |
| |
| std::vector<void *> open_files; |
| |
| [[gnu::weak]] |
| bool is_filesystem_access_disabled() { |
| return false; |
| } |
| |
| bool get_files_open() { |
| return open_files.size() > 0; |
| } |
| |
| void close_open_files() { |
| while(!open_files.empty()) |
| close_file(open_files.back()); |
| } |
| |
| void *open_file(const std::string &file, int mode) { |
| if(is_filesystem_access_disabled()) |
| return nullptr; |
| |
| FIL *f = new FIL(); |
| |
| BYTE ff_mode = 0; |
| |
| if(mode & blit::OpenMode::read) |
| ff_mode |= FA_READ; |
| |
| if(mode & blit::OpenMode::write) |
| ff_mode |= FA_WRITE; |
| |
| if(mode == blit::OpenMode::write) |
| ff_mode |= FA_CREATE_ALWAYS; |
| |
| FRESULT r = f_open(f, file.c_str(), ff_mode); |
| |
| if(r == FR_OK) { |
| open_files.push_back(f); |
| return f; |
| } |
| |
| delete f; |
| return nullptr; |
| } |
| |
| int32_t read_file(void *fh, uint32_t offset, uint32_t length, char *buffer) { |
| FRESULT r = FR_OK; |
| FIL *f = (FIL *)fh; |
| |
| if(offset != f_tell(f)) |
| r = f_lseek(f, offset); |
| |
| if(r == FR_OK){ |
| unsigned int bytes_read; |
| r = f_read(f, buffer, length, &bytes_read); |
| if(r == FR_OK){ |
| return bytes_read; |
| } |
| } |
| |
| return -1; |
| } |
| |
| int32_t write_file(void *fh, uint32_t offset, uint32_t length, const char *buffer) { |
| FRESULT r = FR_OK; |
| FIL *f = (FIL *)fh; |
| |
| if(offset != f_tell(f)) |
| r = f_lseek(f, offset); |
| |
| if(r == FR_OK) { |
| unsigned int bytes_written; |
| r = f_write(f, buffer, length, &bytes_written); |
| if(r == FR_OK) { |
| return bytes_written; |
| } |
| } |
| |
| return -1; |
| } |
| |
| int32_t close_file(void *fh) { |
| FRESULT r; |
| |
| r = f_close((FIL *)fh); |
| |
| for(auto it = open_files.begin(); it != open_files.end(); ++it) { |
| if(*it == fh) { |
| open_files.erase(it); |
| break; |
| } |
| } |
| |
| delete (FIL *)fh; |
| return r == FR_OK ? 0 : -1; |
| } |
| |
| uint32_t get_file_length(void *fh) { |
| return f_size((FIL *)fh); |
| } |
| |
| void list_files(const std::string &path, std::function<void(blit::FileInfo &)> callback) { |
| if(is_filesystem_access_disabled()) |
| return; |
| |
| DIR dir; |
| |
| if(f_opendir(&dir, path.c_str()) != FR_OK) |
| return; |
| |
| FILINFO ent; |
| |
| while(f_readdir(&dir, &ent) == FR_OK && ent.fname[0]) { |
| blit::FileInfo info; |
| |
| info.name = ent.fname; |
| info.flags = 0; |
| info.size = ent.fsize; |
| |
| if(ent.fattrib & AM_DIR) |
| info.flags |= blit::FileFlags::directory; |
| |
| callback(info); |
| } |
| |
| f_closedir(&dir); |
| } |
| |
| bool file_exists(const std::string &path) { |
| FILINFO info; |
| return f_stat(path.c_str(), &info) == FR_OK && !(info.fattrib & AM_DIR); |
| } |
| |
| bool directory_exists(const std::string &path) { |
| FILINFO info; |
| return f_stat(path.c_str(), &info) == FR_OK && (info.fattrib & AM_DIR); |
| } |
| |
| bool create_directory(const std::string &path) { |
| FRESULT r; |
| |
| // strip trailing slash |
| if(path.back() == '/') |
| r = f_mkdir(path.substr(0, path.length() - 1).c_str()); |
| else |
| r = f_mkdir(path.c_str()); |
| |
| return r == FR_OK || r == FR_EXIST; |
| } |
| |
| bool rename_file(const std::string &old_name, const std::string &new_name) { |
| return f_rename(old_name.c_str(), new_name.c_str()) == FR_OK; |
| } |
| |
| bool remove_file(const std::string &path) { |
| return f_unlink(path.c_str()) == FR_OK; |
| } |