luat_fs_lfs2.c 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. #include "luat_base.h"
  2. #include "luat_fs.h"
  3. #include "luat_spi.h"
  4. #include "luat_mem.h"
  5. #define LUAT_LOG_TAG "vfs.lfs2"
  6. #include "luat_log.h"
  7. #ifdef LUAT_USE_FS_VFS
  8. // #ifdef LUAT_VFS_USE_LFS2
  9. #include "lfs.h"
  10. FILE* luat_vfs_lfs2_fopen(void* userdata, const char *filename, const char *mode) {
  11. lfs_t* fs = (lfs_t*)userdata;
  12. lfs_file_t *file = (lfs_file_t*)luat_heap_malloc(sizeof(lfs_file_t));
  13. if (file == NULL) {
  14. LLOGD("out of memory when open file %s", filename);
  15. return NULL;
  16. }
  17. memset(file, 0, sizeof(lfs_file_t));
  18. int flag = 0;
  19. /*
  20. "r": 读模式(默认);
  21. "w": 写模式;
  22. "a": 追加模式;
  23. "r+": 更新模式,所有之前的数据都保留;
  24. "w+": 更新模式,所有之前的数据都删除;
  25. "a+": 追加更新模式,所有之前的数据都保留,只允许在文件尾部做写入。
  26. */
  27. if (!strcmp("r+", mode) || !strcmp("r+b", mode) || !strcmp("rb+", mode)) {
  28. flag = LFS_O_RDWR | LFS_O_CREAT;
  29. }
  30. else if(!strcmp("w+", mode) || !strcmp("w+b", mode) || !strcmp("wb+", mode)) {
  31. flag = LFS_O_RDWR | LFS_O_CREAT | LFS_O_TRUNC;
  32. }
  33. else if(!strcmp("a+", mode) || !strcmp("a+b", mode) || !strcmp("ab+", mode)) {
  34. flag = LFS_O_APPEND | LFS_O_CREAT | LFS_O_WRONLY;
  35. }
  36. else if(!strcmp("w", mode) || !strcmp("wb", mode)) {
  37. flag = LFS_O_RDWR | LFS_O_CREAT | LFS_O_TRUNC;
  38. }
  39. else if(!strcmp("r", mode) || !strcmp("rb", mode)) {
  40. flag = LFS_O_RDONLY;
  41. }
  42. else if(!strcmp("a", mode) || !strcmp("ab", mode)) {
  43. flag = LFS_O_APPEND | LFS_O_CREAT | LFS_O_WRONLY;
  44. }
  45. else {
  46. LLOGW("bad file open mode %s, fallback to 'r'", mode);
  47. flag = LFS_O_RDONLY;
  48. }
  49. int ret = lfs_file_open(fs, file, filename, flag);
  50. if (ret < 0) {
  51. luat_heap_free(file);
  52. return 0;
  53. }
  54. return (FILE*)file;
  55. }
  56. int luat_vfs_lfs2_getc(void* userdata, FILE* stream) {
  57. //LLOGD("posix_getc %p", stream);
  58. lfs_t* fs = (lfs_t*)userdata;
  59. lfs_file_t* file = (lfs_file_t*)stream;
  60. char buff = 0;
  61. int ret = lfs_file_read(fs, file, &buff, 1);
  62. if (ret != 1)
  63. return -1;
  64. return (int)buff;
  65. }
  66. int luat_vfs_lfs2_fseek(void* userdata, FILE* stream, long int offset, int origin) {
  67. lfs_t* fs = (lfs_t*)userdata;
  68. lfs_file_t* file = (lfs_file_t*)stream;
  69. int ret = lfs_file_seek(fs, file, offset, origin);
  70. return ret < 0 ? -1 : 0;
  71. }
  72. int luat_vfs_lfs2_ftell(void* userdata, FILE* stream) {
  73. lfs_t* fs = (lfs_t*)userdata;
  74. lfs_file_t* file = (lfs_file_t*)stream;
  75. int ret = lfs_file_tell(fs, file);
  76. return ret < 0 ? -1 : ret;
  77. }
  78. int luat_vfs_lfs2_fclose(void* userdata, FILE* stream) {
  79. lfs_t* fs = (lfs_t*)userdata;
  80. lfs_file_t* file = (lfs_file_t*)stream;
  81. lfs_file_close(fs, file);
  82. if (file != NULL)
  83. luat_heap_free(file);
  84. return 0;
  85. }
  86. int luat_vfs_lfs2_feof(void* userdata, FILE* stream) {
  87. lfs_t* fs = (lfs_t*)userdata;
  88. lfs_file_t* file = (lfs_file_t*)stream;
  89. if (lfs_file_size(fs, file) <= lfs_file_tell(fs, file))
  90. return 1;
  91. return 0;
  92. }
  93. int luat_vfs_lfs2_ferror(void* userdata, FILE *stream) {
  94. return 0;
  95. }
  96. size_t luat_vfs_lfs2_fread(void* userdata, void *ptr, size_t size, size_t nmemb, FILE *stream) {
  97. lfs_t* fs = (lfs_t*)userdata;
  98. lfs_file_t* file = (lfs_file_t*)stream;
  99. int ret = lfs_file_read(fs, file, ptr, size*nmemb);
  100. return ret < 0 ? 0 : ret;
  101. }
  102. size_t luat_vfs_lfs2_fwrite(void* userdata, const void *ptr, size_t size, size_t nmemb, FILE *stream) {
  103. lfs_t* fs = (lfs_t*)userdata;
  104. lfs_file_t* file = (lfs_file_t*)stream;
  105. int ret = lfs_file_write(fs, file, ptr, size*nmemb);
  106. return ret < 0 ? 0 : ret;
  107. }
  108. int luat_vfs_lfs2_fflush(void* userdata, FILE *stream) {
  109. lfs_t* fs = (lfs_t*)userdata;
  110. lfs_file_t* file = (lfs_file_t*)stream;
  111. int ret = lfs_file_sync(fs, file);
  112. return ret < 0 ? 0 : ret;
  113. }
  114. int luat_vfs_lfs2_remove(void* userdata, const char *filename) {
  115. lfs_t* fs = (lfs_t*)userdata;
  116. return lfs_remove(fs, filename);
  117. }
  118. int luat_vfs_lfs2_rename(void* userdata, const char *old_filename, const char *new_filename) {
  119. lfs_t* fs = (lfs_t*)userdata;
  120. return lfs_rename(fs, old_filename, new_filename);
  121. }
  122. int luat_vfs_lfs2_fexist(void* userdata, const char *filename) {
  123. FILE* fd = luat_vfs_lfs2_fopen(userdata, filename, "rb");
  124. if (fd) {
  125. luat_vfs_lfs2_fclose(userdata, fd);
  126. return 1;
  127. }
  128. return 0;
  129. }
  130. size_t luat_vfs_lfs2_fsize(void* userdata, const char *filename) {
  131. FILE *fd;
  132. size_t size = 0;
  133. fd = luat_vfs_lfs2_fopen(userdata, filename, "rb");
  134. if (fd) {
  135. size = lfs_file_size((lfs_t*)userdata, (lfs_file_t*)fd);
  136. luat_vfs_lfs2_fclose(userdata, fd);
  137. }
  138. return size;
  139. }
  140. int luat_vfs_lfs2_mkfs(void* userdata, luat_fs_conf_t *conf) {
  141. int ret = 0;
  142. lfs_t* fs = (lfs_t*)userdata;
  143. if (fs != NULL && fs->cfg != NULL) {
  144. ret = lfs_format(fs, fs->cfg);
  145. // LLOGD("lfs2 format ret %d", ret);
  146. if (ret < 0)
  147. return ret;
  148. ret = lfs_mount(fs, fs->cfg);
  149. // LLOGD("lfs2 mount ret %d", ret);
  150. return ret;
  151. }
  152. return -1;
  153. }
  154. int luat_vfs_lfs2_mount(void** userdata, luat_fs_conf_t *conf) {
  155. *userdata = (void*)conf->busname;
  156. return 0;
  157. }
  158. int luat_vfs_lfs2_umount(void* userdata, luat_fs_conf_t *conf) {
  159. LLOGE("not support yet : umount");
  160. return 0;
  161. }
  162. static int dir2name(char* buff, char const* _DirName) {
  163. size_t dirlen = strlen(_DirName);
  164. if (dirlen > 63) {
  165. LLOGE("dir too long!! %s", _DirName);
  166. return -1;
  167. }
  168. else if (dirlen < 1) {
  169. LLOGE("dir too short!! %s", _DirName);
  170. return -1;
  171. }
  172. memcpy(buff, _DirName, dirlen);
  173. if (buff[dirlen -1] == '/') {
  174. buff[dirlen -1] = 0;
  175. }
  176. return 0;
  177. }
  178. int luat_vfs_lfs2_mkdir(void* userdata, char const* _DirName) {
  179. lfs_t* fs = (lfs_t*)userdata;
  180. char buff[64] = {0};
  181. if (dir2name(buff, _DirName)) {
  182. return -1;
  183. }
  184. int ret = lfs_mkdir(fs, buff);
  185. return ret == LFS_ERR_OK ? 0 : -1;
  186. }
  187. int luat_vfs_lfs2_rmdir(void* userdata, char const* _DirName) {
  188. lfs_t* fs = (lfs_t*)userdata;
  189. char buff[64] = {0};
  190. if (dir2name(buff, _DirName)) {
  191. return -1;
  192. }
  193. int ret = lfs_remove(fs, buff);
  194. return ret == LFS_ERR_OK ? 0 : -1;
  195. }
  196. int luat_vfs_lfs2_lsdir(void* userdata, char const* _DirName, luat_fs_dirent_t* ents, size_t offset, size_t len) {
  197. lfs_t* fs = (lfs_t*)userdata;
  198. int ret , num = 0;
  199. lfs_dir_t *dir;
  200. struct lfs_info info;
  201. char buff[64] = {0};
  202. if (strlen(_DirName) == 0) {
  203. // OK的, 根目录嘛
  204. }
  205. else if (dir2name(buff, _DirName)) {
  206. return -1;
  207. }
  208. // if (fs->filecount > offset) {
  209. // if (offset + len > fs->filecount)
  210. // len = fs->filecount - offset;
  211. dir = luat_heap_malloc(sizeof(lfs_dir_t));
  212. if (dir == NULL) {
  213. // LLOGE("out of memory when lsdir");
  214. return 0;
  215. }
  216. ret = lfs_dir_open(fs, dir, buff);
  217. if (ret < 0) {
  218. luat_heap_free(dir);
  219. // LLOGE("no such dir %s _DirName");
  220. return 0;
  221. }
  222. // TODO 使用seek/tell组合更快更省
  223. for (size_t i = 0; i < offset; i++)
  224. {
  225. ret = lfs_dir_read(fs, dir, &info);
  226. if (ret <= 0) {
  227. lfs_dir_close(fs, dir);
  228. luat_heap_free(dir);
  229. return 0;
  230. }
  231. }
  232. while (num < len)
  233. {
  234. ret = lfs_dir_read(fs, dir, &info);
  235. if (ret < 0) {
  236. lfs_dir_close(fs, dir);
  237. luat_heap_free(dir);
  238. return 0;
  239. }
  240. if (ret == 0) {
  241. break;
  242. }
  243. if (info.type == 2 && (memcmp(info.name, ".", 2) ==0 ||memcmp(info.name, "..", 3)==0))
  244. continue;
  245. ents[num].d_type = info.type - 1; // lfs file =1, dir=2
  246. strcpy(ents[num].d_name, info.name);
  247. num++;
  248. }
  249. lfs_dir_close(fs, dir);
  250. luat_heap_free(dir);
  251. return num;
  252. // }
  253. return 0;
  254. }
  255. int luat_vfs_lfs2_info(void* userdata, const char* path, luat_fs_info_t *conf) {
  256. //LLOGD("why ? luat_vfs_lfs2_info %p", userdata);
  257. lfs_t* fs = (lfs_t*)userdata;
  258. memcpy(conf->filesystem, "lfs", strlen("lfs")+1);
  259. conf->type = 0;
  260. conf->total_block = fs->cfg->block_count;
  261. conf->block_used = lfs_fs_size(fs);
  262. conf->block_size = fs->cfg->block_size;
  263. //LLOGD("total %d used %d size %d", conf->total_block, conf->block_used, conf->block_size);
  264. return 0;
  265. }
  266. int luat_vfs_lfs2_truncate(void* userdata, const char *filename, size_t len) {
  267. FILE *fd;
  268. int ret = -1;
  269. fd = luat_vfs_lfs2_fopen(userdata, filename, "wb");
  270. if (fd) {
  271. ret = lfs_file_truncate((lfs_t*)userdata, (lfs_file_t*)fd ,(lfs_off_t)len);
  272. luat_vfs_lfs2_fclose(userdata, fd);
  273. }
  274. return ret;
  275. }
  276. #define T(name) .name = luat_vfs_lfs2_##name
  277. const struct luat_vfs_filesystem vfs_fs_lfs2 = {
  278. .name = "lfs2",
  279. .opts = {
  280. T(mkfs),
  281. T(mount),
  282. T(umount),
  283. T(mkdir),
  284. T(rmdir),
  285. T(lsdir),
  286. T(remove),
  287. T(rename),
  288. T(fsize),
  289. T(fexist),
  290. T(info),
  291. T(truncate)
  292. },
  293. .fopts = {
  294. T(fopen),
  295. T(getc),
  296. T(fseek),
  297. T(ftell),
  298. T(fclose),
  299. T(feof),
  300. T(ferror),
  301. T(fread),
  302. T(fwrite),
  303. T(fflush)
  304. }
  305. };
  306. // #endif
  307. #endif