littlefs_fs.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849
  1. /*
  2. * Copyright (c) 2019 Bolt Innovation Management, LLC
  3. * Copyright (c) 2019 Peter Bigot Consulting, LLC
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #include <stdio.h>
  8. #include <string.h>
  9. #include <kernel.h>
  10. #include <errno.h>
  11. #include <init.h>
  12. #include <fs/fs.h>
  13. #include <fs/fs_sys.h>
  14. #define LFS_LOG_REGISTER
  15. #include <lfs_util.h>
  16. #include <lfs.h>
  17. #include <fs/littlefs.h>
  18. #include <drivers/flash.h>
  19. #include <storage/flash_map.h>
  20. #include "fs_impl.h"
  21. struct lfs_file_data {
  22. struct lfs_file file;
  23. struct lfs_file_config config;
  24. void *cache_block;
  25. };
  26. #define LFS_FILEP(fp) (&((struct lfs_file_data *)(fp->filep))->file)
  27. /* Global memory pool for open files and dirs */
  28. static K_MEM_SLAB_DEFINE(file_data_pool, sizeof(struct lfs_file_data),
  29. CONFIG_FS_LITTLEFS_NUM_FILES, 4);
  30. static K_MEM_SLAB_DEFINE(lfs_dir_pool, sizeof(struct lfs_dir),
  31. CONFIG_FS_LITTLEFS_NUM_DIRS, 4);
  32. /* Inferred overhead, in bytes, for each k_heap_aligned allocation for
  33. * the filecache heap. This relates to the CHUNK_UNIT parameter in
  34. * the heap implementation, but that value is not visible outside the
  35. * kernel.
  36. */
  37. #define FC_HEAP_PER_ALLOC_OVERHEAD 24U
  38. #if (CONFIG_FS_LITTLEFS_FC_HEAP_SIZE - 0) <= 0
  39. /* Auto-generate heap size from cache size and number of files */
  40. #undef CONFIG_FS_LITTLEFS_FC_HEAP_SIZE
  41. #define CONFIG_FS_LITTLEFS_FC_HEAP_SIZE \
  42. ((CONFIG_FS_LITTLEFS_CACHE_SIZE + FC_HEAP_PER_ALLOC_OVERHEAD) * \
  43. CONFIG_FS_LITTLEFS_NUM_FILES)
  44. #endif /* CONFIG_FS_LITTLEFS_FC_HEAP_SIZE */
  45. static K_HEAP_DEFINE(file_cache_heap, CONFIG_FS_LITTLEFS_FC_HEAP_SIZE);
  46. static inline void *fc_allocate(size_t size)
  47. {
  48. void *ret = NULL;
  49. ret = k_heap_alloc(&file_cache_heap, size, K_NO_WAIT);
  50. return ret;
  51. }
  52. static inline void fc_release(void *buf)
  53. {
  54. k_heap_free(&file_cache_heap, buf);
  55. }
  56. static inline void fs_lock(struct fs_littlefs *fs)
  57. {
  58. k_mutex_lock(&fs->mutex, K_FOREVER);
  59. }
  60. static inline void fs_unlock(struct fs_littlefs *fs)
  61. {
  62. k_mutex_unlock(&fs->mutex);
  63. }
  64. static int lfs_to_errno(int error)
  65. {
  66. if (error >= 0) {
  67. return error;
  68. }
  69. switch (error) {
  70. default:
  71. case LFS_ERR_IO: /* Error during device operation */
  72. return -EIO;
  73. case LFS_ERR_CORRUPT: /* Corrupted */
  74. return -EFAULT;
  75. case LFS_ERR_NOENT: /* No directory entry */
  76. return -ENOENT;
  77. case LFS_ERR_EXIST: /* Entry already exists */
  78. return -EEXIST;
  79. case LFS_ERR_NOTDIR: /* Entry is not a dir */
  80. return -ENOTDIR;
  81. case LFS_ERR_ISDIR: /* Entry is a dir */
  82. return -EISDIR;
  83. case LFS_ERR_NOTEMPTY: /* Dir is not empty */
  84. return -ENOTEMPTY;
  85. case LFS_ERR_BADF: /* Bad file number */
  86. return -EBADF;
  87. case LFS_ERR_FBIG: /* File too large */
  88. return -EFBIG;
  89. case LFS_ERR_INVAL: /* Invalid parameter */
  90. return -EINVAL;
  91. case LFS_ERR_NOSPC: /* No space left on device */
  92. return -ENOSPC;
  93. case LFS_ERR_NOMEM: /* No more memory available */
  94. return -ENOMEM;
  95. }
  96. }
  97. static int errno_to_lfs(int error)
  98. {
  99. if (error >= 0) {
  100. return LFS_ERR_OK;
  101. }
  102. switch (error) {
  103. default:
  104. case -EIO: /* Error during device operation */
  105. return LFS_ERR_IO;
  106. case -EFAULT: /* Corrupted */
  107. return LFS_ERR_CORRUPT;
  108. case -ENOENT: /* No directory entry */
  109. return LFS_ERR_NOENT;
  110. case -EEXIST: /* Entry already exists */
  111. return LFS_ERR_EXIST;
  112. case -ENOTDIR: /* Entry is not a dir */
  113. return LFS_ERR_NOTDIR;
  114. case -EISDIR: /* Entry is a dir */
  115. return LFS_ERR_ISDIR;
  116. case -ENOTEMPTY: /* Dir is not empty */
  117. return LFS_ERR_NOTEMPTY;
  118. case -EBADF: /* Bad file number */
  119. return LFS_ERR_BADF;
  120. case -EFBIG: /* File too large */
  121. return LFS_ERR_FBIG;
  122. case -EINVAL: /* Invalid parameter */
  123. return LFS_ERR_INVAL;
  124. case -ENOSPC: /* No space left on device */
  125. return LFS_ERR_NOSPC;
  126. case -ENOMEM: /* No more memory available */
  127. return LFS_ERR_NOMEM;
  128. }
  129. }
  130. static int lfs_api_read(const struct lfs_config *c, lfs_block_t block,
  131. lfs_off_t off, void *buffer, lfs_size_t size)
  132. {
  133. const struct flash_area *fa = c->context;
  134. size_t offset = block * c->block_size + off;
  135. int rc = flash_area_read(fa, offset, buffer, size);
  136. return errno_to_lfs(rc);
  137. }
  138. static int lfs_api_prog(const struct lfs_config *c, lfs_block_t block,
  139. lfs_off_t off, const void *buffer, lfs_size_t size)
  140. {
  141. const struct flash_area *fa = c->context;
  142. size_t offset = block * c->block_size + off;
  143. int rc = flash_area_write(fa, offset, buffer, size);
  144. return errno_to_lfs(rc);
  145. }
  146. static int lfs_api_erase(const struct lfs_config *c, lfs_block_t block)
  147. {
  148. const struct flash_area *fa = c->context;
  149. size_t offset = block * c->block_size;
  150. int rc = flash_area_erase(fa, offset, c->block_size);
  151. return errno_to_lfs(rc);
  152. }
  153. static int lfs_api_sync(const struct lfs_config *c)
  154. {
  155. return LFS_ERR_OK;
  156. }
  157. static void release_file_data(struct fs_file_t *fp)
  158. {
  159. struct lfs_file_data *fdp = fp->filep;
  160. if (fdp->config.buffer) {
  161. fc_release(fdp->cache_block);
  162. }
  163. k_mem_slab_free(&file_data_pool, &fp->filep);
  164. fp->filep = NULL;
  165. }
  166. static int lfs_flags_from_zephyr(unsigned int zflags)
  167. {
  168. int flags = (zflags & FS_O_CREATE) ? LFS_O_CREAT : 0;
  169. /* LFS_O_READONLY and LFS_O_WRONLY can be selected at the same time,
  170. * this is not a mistake, together they create RDWR access.
  171. */
  172. flags |= (zflags & FS_O_READ) ? LFS_O_RDONLY : 0;
  173. flags |= (zflags & FS_O_WRITE) ? LFS_O_WRONLY : 0;
  174. flags |= (zflags & FS_O_APPEND) ? LFS_O_APPEND : 0;
  175. return flags;
  176. }
  177. static int littlefs_open(struct fs_file_t *fp, const char *path,
  178. fs_mode_t zflags)
  179. {
  180. struct fs_littlefs *fs = fp->mp->fs_data;
  181. struct lfs *lfs = &fs->lfs;
  182. int flags = lfs_flags_from_zephyr(zflags);
  183. int ret = k_mem_slab_alloc(&file_data_pool, &fp->filep, K_NO_WAIT);
  184. if (ret != 0) {
  185. return ret;
  186. }
  187. struct lfs_file_data *fdp = fp->filep;
  188. memset(fdp, 0, sizeof(*fdp));
  189. fdp->cache_block = fc_allocate(lfs->cfg->cache_size);
  190. if (fdp->cache_block == NULL) {
  191. ret = -ENOMEM;
  192. goto out;
  193. }
  194. fdp->config.buffer = fdp->cache_block;
  195. path = fs_impl_strip_prefix(path, fp->mp);
  196. fs_lock(fs);
  197. ret = lfs_file_opencfg(&fs->lfs, &fdp->file,
  198. path, flags, &fdp->config);
  199. fs_unlock(fs);
  200. out:
  201. if (ret < 0) {
  202. release_file_data(fp);
  203. }
  204. return lfs_to_errno(ret);
  205. }
  206. static int littlefs_close(struct fs_file_t *fp)
  207. {
  208. struct fs_littlefs *fs = fp->mp->fs_data;
  209. fs_lock(fs);
  210. int ret = lfs_file_close(&fs->lfs, LFS_FILEP(fp));
  211. fs_unlock(fs);
  212. release_file_data(fp);
  213. return lfs_to_errno(ret);
  214. }
  215. static int littlefs_unlink(struct fs_mount_t *mountp, const char *path)
  216. {
  217. struct fs_littlefs *fs = mountp->fs_data;
  218. path = fs_impl_strip_prefix(path, mountp);
  219. fs_lock(fs);
  220. int ret = lfs_remove(&fs->lfs, path);
  221. fs_unlock(fs);
  222. return lfs_to_errno(ret);
  223. }
  224. static int littlefs_rename(struct fs_mount_t *mountp, const char *from,
  225. const char *to)
  226. {
  227. struct fs_littlefs *fs = mountp->fs_data;
  228. from = fs_impl_strip_prefix(from, mountp);
  229. to = fs_impl_strip_prefix(to, mountp);
  230. fs_lock(fs);
  231. int ret = lfs_rename(&fs->lfs, from, to);
  232. fs_unlock(fs);
  233. return lfs_to_errno(ret);
  234. }
  235. static ssize_t littlefs_read(struct fs_file_t *fp, void *ptr, size_t len)
  236. {
  237. struct fs_littlefs *fs = fp->mp->fs_data;
  238. fs_lock(fs);
  239. ssize_t ret = lfs_file_read(&fs->lfs, LFS_FILEP(fp), ptr, len);
  240. fs_unlock(fs);
  241. return lfs_to_errno(ret);
  242. }
  243. static ssize_t littlefs_write(struct fs_file_t *fp, const void *ptr, size_t len)
  244. {
  245. struct fs_littlefs *fs = fp->mp->fs_data;
  246. fs_lock(fs);
  247. ssize_t ret = lfs_file_write(&fs->lfs, LFS_FILEP(fp), ptr, len);
  248. fs_unlock(fs);
  249. return lfs_to_errno(ret);
  250. }
  251. BUILD_ASSERT((FS_SEEK_SET == LFS_SEEK_SET)
  252. && (FS_SEEK_CUR == LFS_SEEK_CUR)
  253. && (FS_SEEK_END == LFS_SEEK_END));
  254. static int littlefs_seek(struct fs_file_t *fp, off_t off, int whence)
  255. {
  256. struct fs_littlefs *fs = fp->mp->fs_data;
  257. fs_lock(fs);
  258. off_t ret = lfs_file_seek(&fs->lfs, LFS_FILEP(fp), off, whence);
  259. fs_unlock(fs);
  260. if (ret >= 0) {
  261. ret = 0;
  262. }
  263. return lfs_to_errno(ret);
  264. }
  265. static off_t littlefs_tell(struct fs_file_t *fp)
  266. {
  267. struct fs_littlefs *fs = fp->mp->fs_data;
  268. fs_lock(fs);
  269. off_t ret = lfs_file_tell(&fs->lfs, LFS_FILEP(fp));
  270. fs_unlock(fs);
  271. return ret;
  272. }
  273. static int littlefs_truncate(struct fs_file_t *fp, off_t length)
  274. {
  275. struct fs_littlefs *fs = fp->mp->fs_data;
  276. fs_lock(fs);
  277. int ret = lfs_file_truncate(&fs->lfs, LFS_FILEP(fp), length);
  278. fs_unlock(fs);
  279. return lfs_to_errno(ret);
  280. }
  281. static int littlefs_sync(struct fs_file_t *fp)
  282. {
  283. struct fs_littlefs *fs = fp->mp->fs_data;
  284. fs_lock(fs);
  285. int ret = lfs_file_sync(&fs->lfs, LFS_FILEP(fp));
  286. fs_unlock(fs);
  287. return lfs_to_errno(ret);
  288. }
  289. static int littlefs_mkdir(struct fs_mount_t *mountp, const char *path)
  290. {
  291. struct fs_littlefs *fs = mountp->fs_data;
  292. path = fs_impl_strip_prefix(path, mountp);
  293. fs_lock(fs);
  294. int ret = lfs_mkdir(&fs->lfs, path);
  295. fs_unlock(fs);
  296. return lfs_to_errno(ret);
  297. }
  298. static int littlefs_opendir(struct fs_dir_t *dp, const char *path)
  299. {
  300. struct fs_littlefs *fs = dp->mp->fs_data;
  301. if (k_mem_slab_alloc(&lfs_dir_pool, &dp->dirp, K_NO_WAIT) != 0) {
  302. return -ENOMEM;
  303. }
  304. memset(dp->dirp, 0, sizeof(struct lfs_dir));
  305. path = fs_impl_strip_prefix(path, dp->mp);
  306. fs_lock(fs);
  307. int ret = lfs_dir_open(&fs->lfs, dp->dirp, path);
  308. fs_unlock(fs);
  309. if (ret < 0) {
  310. k_mem_slab_free(&lfs_dir_pool, &dp->dirp);
  311. }
  312. return lfs_to_errno(ret);
  313. }
  314. static void info_to_dirent(const struct lfs_info *info, struct fs_dirent *entry)
  315. {
  316. entry->type = ((info->type == LFS_TYPE_DIR) ?
  317. FS_DIR_ENTRY_DIR : FS_DIR_ENTRY_FILE);
  318. entry->size = info->size;
  319. strncpy(entry->name, info->name, sizeof(entry->name));
  320. entry->name[sizeof(entry->name) - 1] = '\0';
  321. }
  322. static int littlefs_readdir(struct fs_dir_t *dp, struct fs_dirent *entry)
  323. {
  324. struct fs_littlefs *fs = dp->mp->fs_data;
  325. fs_lock(fs);
  326. struct lfs_info info;
  327. int ret = lfs_dir_read(&fs->lfs, dp->dirp, &info);
  328. fs_unlock(fs);
  329. if (ret > 0) {
  330. info_to_dirent(&info, entry);
  331. ret = 0;
  332. } else if (ret == 0) {
  333. entry->name[0] = 0;
  334. }
  335. return lfs_to_errno(ret);
  336. }
  337. static int littlefs_closedir(struct fs_dir_t *dp)
  338. {
  339. struct fs_littlefs *fs = dp->mp->fs_data;
  340. fs_lock(fs);
  341. int ret = lfs_dir_close(&fs->lfs, dp->dirp);
  342. fs_unlock(fs);
  343. k_mem_slab_free(&lfs_dir_pool, &dp->dirp);
  344. return lfs_to_errno(ret);
  345. }
  346. static int littlefs_stat(struct fs_mount_t *mountp,
  347. const char *path, struct fs_dirent *entry)
  348. {
  349. struct fs_littlefs *fs = mountp->fs_data;
  350. path = fs_impl_strip_prefix(path, mountp);
  351. fs_lock(fs);
  352. struct lfs_info info;
  353. int ret = lfs_stat(&fs->lfs, path, &info);
  354. fs_unlock(fs);
  355. if (ret >= 0) {
  356. info_to_dirent(&info, entry);
  357. ret = 0;
  358. }
  359. return lfs_to_errno(ret);
  360. }
  361. static int littlefs_statvfs(struct fs_mount_t *mountp,
  362. const char *path, struct fs_statvfs *stat)
  363. {
  364. struct fs_littlefs *fs = mountp->fs_data;
  365. struct lfs *lfs = &fs->lfs;
  366. stat->f_bsize = lfs->cfg->prog_size;
  367. stat->f_frsize = lfs->cfg->block_size;
  368. stat->f_blocks = lfs->cfg->block_count;
  369. path = fs_impl_strip_prefix(path, mountp);
  370. fs_lock(fs);
  371. ssize_t ret = lfs_fs_size(lfs);
  372. fs_unlock(fs);
  373. if (ret >= 0) {
  374. stat->f_bfree = stat->f_blocks - ret;
  375. ret = 0;
  376. }
  377. return lfs_to_errno(ret);
  378. }
  379. /* Return maximum page size in a flash area. There's no flash_area
  380. * API to implement this, so we have to make one here.
  381. */
  382. struct get_page_ctx {
  383. const struct flash_area *area;
  384. lfs_size_t max_size;
  385. };
  386. static bool get_page_cb(const struct flash_pages_info *info, void *ctxp)
  387. {
  388. struct get_page_ctx *ctx = ctxp;
  389. size_t info_start = info->start_offset;
  390. size_t info_end = info_start + info->size - 1U;
  391. size_t area_start = ctx->area->fa_off;
  392. size_t area_end = area_start + ctx->area->fa_size - 1U;
  393. /* Ignore pages outside the area */
  394. if (info_end < area_start) {
  395. return true;
  396. }
  397. if (info_start > area_end) {
  398. return false;
  399. }
  400. if (info->size > ctx->max_size) {
  401. ctx->max_size = info->size;
  402. }
  403. return true;
  404. }
  405. /* Iterate over all page groups in the flash area and return the
  406. * largest page size we see. This works as long as the partition is
  407. * aligned so that erasing with this size is supported throughout the
  408. * partition.
  409. */
  410. static lfs_size_t get_block_size(const struct flash_area *fa)
  411. {
  412. struct get_page_ctx ctx = {
  413. .area = fa,
  414. .max_size = 0,
  415. };
  416. const struct device *dev = flash_area_get_device(fa);
  417. flash_page_foreach(dev, get_page_cb, &ctx);
  418. return ctx.max_size;
  419. }
  420. static int littlefs_mount(struct fs_mount_t *mountp)
  421. {
  422. int ret;
  423. struct fs_littlefs *fs = mountp->fs_data;
  424. unsigned int area_id = (uintptr_t)mountp->storage_dev;
  425. const struct device *dev;
  426. LOG_INF("LittleFS version %u.%u, disk version %u.%u",
  427. LFS_VERSION_MAJOR, LFS_VERSION_MINOR,
  428. LFS_DISK_VERSION_MAJOR, LFS_DISK_VERSION_MINOR);
  429. if (fs->area) {
  430. return -EBUSY;
  431. }
  432. /* Create and take mutex. */
  433. k_mutex_init(&fs->mutex);
  434. fs_lock(fs);
  435. /* Open flash area */
  436. ret = flash_area_open(area_id, &fs->area);
  437. if ((ret < 0) || (fs->area == NULL)) {
  438. LOG_ERR("can't open flash area %d", area_id);
  439. ret = -ENODEV;
  440. goto out;
  441. }
  442. LOG_DBG("FS area %u at 0x%x for %u bytes",
  443. area_id, (uint32_t)fs->area->fa_off,
  444. (uint32_t)fs->area->fa_size);
  445. dev = flash_area_get_device(fs->area);
  446. if (dev == NULL) {
  447. LOG_ERR("can't get flash device: %s", log_strdup(fs->area->fa_dev_name));
  448. ret = -ENODEV;
  449. goto out;
  450. }
  451. BUILD_ASSERT(CONFIG_FS_LITTLEFS_READ_SIZE > 0);
  452. BUILD_ASSERT(CONFIG_FS_LITTLEFS_PROG_SIZE > 0);
  453. BUILD_ASSERT(CONFIG_FS_LITTLEFS_CACHE_SIZE > 0);
  454. BUILD_ASSERT(CONFIG_FS_LITTLEFS_LOOKAHEAD_SIZE > 0);
  455. BUILD_ASSERT((CONFIG_FS_LITTLEFS_LOOKAHEAD_SIZE % 8) == 0);
  456. BUILD_ASSERT((CONFIG_FS_LITTLEFS_CACHE_SIZE
  457. % CONFIG_FS_LITTLEFS_READ_SIZE) == 0);
  458. BUILD_ASSERT((CONFIG_FS_LITTLEFS_CACHE_SIZE
  459. % CONFIG_FS_LITTLEFS_PROG_SIZE) == 0);
  460. struct lfs_config *lcp = &fs->cfg;
  461. lfs_size_t read_size = lcp->read_size;
  462. if (read_size == 0) {
  463. read_size = CONFIG_FS_LITTLEFS_READ_SIZE;
  464. }
  465. lfs_size_t prog_size = lcp->prog_size;
  466. if (prog_size == 0) {
  467. prog_size = CONFIG_FS_LITTLEFS_PROG_SIZE;
  468. }
  469. /* Yes, you can override block size. */
  470. lfs_size_t block_size = lcp->block_size;
  471. if (block_size == 0) {
  472. block_size = get_block_size(fs->area);
  473. }
  474. if (block_size == 0) {
  475. __ASSERT_NO_MSG(block_size != 0);
  476. ret = -EINVAL;
  477. goto out;
  478. }
  479. int32_t block_cycles = lcp->block_cycles;
  480. if (block_cycles == 0) {
  481. block_cycles = CONFIG_FS_LITTLEFS_BLOCK_CYCLES;
  482. }
  483. if (block_cycles <= 0) {
  484. /* Disable leveling (littlefs v2.1+ semantics) */
  485. block_cycles = -1;
  486. }
  487. lfs_size_t cache_size = lcp->cache_size;
  488. if (cache_size == 0) {
  489. cache_size = CONFIG_FS_LITTLEFS_CACHE_SIZE;
  490. }
  491. lfs_size_t lookahead_size = lcp->lookahead_size;
  492. if (lookahead_size == 0) {
  493. lookahead_size = CONFIG_FS_LITTLEFS_LOOKAHEAD_SIZE;
  494. }
  495. /* No, you don't get to override this. */
  496. lfs_size_t block_count = fs->area->fa_size / block_size;
  497. LOG_INF("FS at %s:0x%x is %u 0x%x-byte blocks with %u cycle",
  498. log_strdup(dev->name), (uint32_t)fs->area->fa_off,
  499. block_count, block_size, block_cycles);
  500. LOG_INF("sizes: rd %u ; pr %u ; ca %u ; la %u",
  501. read_size, prog_size, cache_size, lookahead_size);
  502. __ASSERT_NO_MSG(prog_size != 0);
  503. __ASSERT_NO_MSG(read_size != 0);
  504. __ASSERT_NO_MSG(cache_size != 0);
  505. __ASSERT_NO_MSG(block_size != 0);
  506. __ASSERT((fs->area->fa_size % block_size) == 0,
  507. "partition size must be multiple of block size");
  508. __ASSERT((block_size % prog_size) == 0,
  509. "erase size must be multiple of write size");
  510. __ASSERT((block_size % cache_size) == 0,
  511. "cache size incompatible with block size");
  512. /* Set the validated/defaulted values. */
  513. lcp->context = (void *)fs->area;
  514. lcp->read = lfs_api_read;
  515. lcp->prog = lfs_api_prog;
  516. lcp->erase = lfs_api_erase;
  517. lcp->sync = lfs_api_sync;
  518. lcp->read_size = read_size;
  519. lcp->prog_size = prog_size;
  520. lcp->block_size = block_size;
  521. lcp->block_count = block_count;
  522. lcp->block_cycles = block_cycles;
  523. lcp->cache_size = cache_size;
  524. lcp->lookahead_size = lookahead_size;
  525. /* Mount it, formatting if needed. */
  526. ret = lfs_mount(&fs->lfs, &fs->cfg);
  527. if (ret < 0 &&
  528. (mountp->flags & FS_MOUNT_FLAG_NO_FORMAT) == 0) {
  529. LOG_WRN("can't mount (LFS %d); formatting", ret);
  530. if ((mountp->flags & FS_MOUNT_FLAG_READ_ONLY) == 0) {
  531. ret = lfs_format(&fs->lfs, &fs->cfg);
  532. if (ret < 0) {
  533. LOG_ERR("format failed (LFS %d)", ret);
  534. ret = lfs_to_errno(ret);
  535. goto out;
  536. }
  537. } else {
  538. LOG_ERR("can not format read-only system");
  539. ret = -EROFS;
  540. goto out;
  541. }
  542. ret = lfs_mount(&fs->lfs, &fs->cfg);
  543. if (ret < 0) {
  544. LOG_ERR("remount after format failed (LFS %d)", ret);
  545. ret = lfs_to_errno(ret);
  546. goto out;
  547. }
  548. }
  549. LOG_INF("%s mounted", log_strdup(mountp->mnt_point));
  550. out:
  551. if (ret < 0) {
  552. fs->area = NULL;
  553. }
  554. fs_unlock(fs);
  555. return ret;
  556. }
  557. static int littlefs_unmount(struct fs_mount_t *mountp)
  558. {
  559. struct fs_littlefs *fs = mountp->fs_data;
  560. fs_lock(fs);
  561. lfs_unmount(&fs->lfs);
  562. flash_area_close(fs->area);
  563. fs->area = NULL;
  564. fs_unlock(fs);
  565. LOG_INF("%s unmounted", log_strdup(mountp->mnt_point));
  566. return 0;
  567. }
  568. /* File system interface */
  569. static const struct fs_file_system_t littlefs_fs = {
  570. .open = littlefs_open,
  571. .close = littlefs_close,
  572. .read = littlefs_read,
  573. .write = littlefs_write,
  574. .lseek = littlefs_seek,
  575. .tell = littlefs_tell,
  576. .truncate = littlefs_truncate,
  577. .sync = littlefs_sync,
  578. .opendir = littlefs_opendir,
  579. .readdir = littlefs_readdir,
  580. .closedir = littlefs_closedir,
  581. .mount = littlefs_mount,
  582. .unmount = littlefs_unmount,
  583. .unlink = littlefs_unlink,
  584. .rename = littlefs_rename,
  585. .mkdir = littlefs_mkdir,
  586. .stat = littlefs_stat,
  587. .statvfs = littlefs_statvfs,
  588. };
  589. #define DT_DRV_COMPAT zephyr_fstab_littlefs
  590. #define FS_PARTITION(inst) DT_PHANDLE_BY_IDX(DT_DRV_INST(inst), partition, 0)
  591. #define DEFINE_FS(inst) \
  592. static uint8_t __aligned(4) \
  593. read_buffer_##inst[DT_INST_PROP(inst, cache_size)]; \
  594. static uint8_t __aligned(4) \
  595. prog_buffer_##inst[DT_INST_PROP(inst, cache_size)]; \
  596. static uint32_t lookahead_buffer_##inst[DT_INST_PROP(inst, lookahead_size) \
  597. / sizeof(uint32_t)]; \
  598. BUILD_ASSERT(DT_INST_PROP(inst, read_size) > 0); \
  599. BUILD_ASSERT(DT_INST_PROP(inst, prog_size) > 0); \
  600. BUILD_ASSERT(DT_INST_PROP(inst, cache_size) > 0); \
  601. BUILD_ASSERT(DT_INST_PROP(inst, lookahead_size) > 0); \
  602. BUILD_ASSERT((DT_INST_PROP(inst, lookahead_size) % 8) == 0); \
  603. BUILD_ASSERT((DT_INST_PROP(inst, cache_size) \
  604. % DT_INST_PROP(inst, read_size)) == 0); \
  605. BUILD_ASSERT((DT_INST_PROP(inst, cache_size) \
  606. % DT_INST_PROP(inst, prog_size)) == 0); \
  607. static struct fs_littlefs fs_data_##inst = { \
  608. .cfg = { \
  609. .read_size = DT_INST_PROP(inst, read_size), \
  610. .prog_size = DT_INST_PROP(inst, prog_size), \
  611. .cache_size = DT_INST_PROP(inst, cache_size), \
  612. .lookahead_size = DT_INST_PROP(inst, lookahead_size), \
  613. .read_buffer = read_buffer_##inst, \
  614. .prog_buffer = prog_buffer_##inst, \
  615. .lookahead_buffer = lookahead_buffer_##inst, \
  616. }, \
  617. }; \
  618. struct fs_mount_t FS_FSTAB_ENTRY(DT_DRV_INST(inst)) = { \
  619. .type = FS_LITTLEFS, \
  620. .mnt_point = DT_INST_PROP(inst, mount_point), \
  621. .fs_data = &fs_data_##inst, \
  622. .storage_dev = (void *)DT_FIXED_PARTITION_ID(FS_PARTITION(inst)), \
  623. .flags = FSTAB_ENTRY_DT_MOUNT_FLAGS(DT_DRV_INST(inst)), \
  624. };
  625. DT_INST_FOREACH_STATUS_OKAY(DEFINE_FS)
  626. #define REFERENCE_MOUNT(inst) (&FS_FSTAB_ENTRY(DT_DRV_INST(inst))),
  627. static void mount_init(struct fs_mount_t *mp)
  628. {
  629. LOG_INF("littlefs partition at %s", mp->mnt_point);
  630. if ((mp->flags & FS_MOUNT_FLAG_AUTOMOUNT) != 0) {
  631. int rc = fs_mount(mp);
  632. if (rc < 0) {
  633. LOG_ERR("Automount %s failed: %d",
  634. mp->mnt_point, rc);
  635. } else {
  636. LOG_INF("Automount %s succeeded",
  637. mp->mnt_point);
  638. }
  639. }
  640. }
  641. static int littlefs_init(const struct device *dev)
  642. {
  643. ARG_UNUSED(dev);
  644. static struct fs_mount_t *partitions[] = {
  645. DT_INST_FOREACH_STATUS_OKAY(REFERENCE_MOUNT)
  646. };
  647. int rc = fs_register(FS_LITTLEFS, &littlefs_fs);
  648. if (rc == 0) {
  649. struct fs_mount_t **mpi = partitions;
  650. while (mpi < (partitions + ARRAY_SIZE(partitions))) {
  651. mount_init(*mpi++);
  652. }
  653. }
  654. return rc;
  655. }
  656. SYS_INIT(littlefs_init, POST_KERNEL, 99);