fs.h 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650
  1. /*
  2. * Copyright (c) 2016 Intel Corporation.
  3. * Copyright (c) 2020-2021 Nordic Semiconductor ASA
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #ifndef ZEPHYR_INCLUDE_FS_FS_H_
  8. #define ZEPHYR_INCLUDE_FS_FS_H_
  9. #include <sys/types.h>
  10. #include <sys/dlist.h>
  11. #include <fs/fs_interface.h>
  12. #ifdef __cplusplus
  13. extern "C" {
  14. #endif
  15. /**
  16. * @brief File System APIs
  17. * @defgroup file_system_api File System APIs
  18. * @{
  19. */
  20. struct fs_file_system_t;
  21. enum fs_dir_entry_type {
  22. /** Identifier for file entry */
  23. FS_DIR_ENTRY_FILE = 0,
  24. /** Identifier for directory entry */
  25. FS_DIR_ENTRY_DIR
  26. };
  27. /** @brief Enumeration to uniquely identify file system types.
  28. *
  29. * Zephyr supports in-tree file systems and external ones. Each
  30. * requires a unique identifier used to register the file system
  31. * implementation and to associate a mount point with the file system
  32. * type. This anonymous enum defines global identifiers for the
  33. * in-tree file systems.
  34. *
  35. * External file systems should be registered using unique identifiers
  36. * starting at @c FS_TYPE_EXTERNAL_BASE. It is the responsibility of
  37. * applications that use external file systems to ensure that these
  38. * identifiers are unique if multiple file system implementations are
  39. * used by the application.
  40. */
  41. enum {
  42. /** Identifier for in-tree FatFS file system. */
  43. FS_FATFS = 0,
  44. /** Identifier for in-tree LittleFS file system. */
  45. FS_LITTLEFS,
  46. FS_SDFS,
  47. /** Base identifier for external file systems. */
  48. FS_TYPE_EXTERNAL_BASE,
  49. };
  50. /** Flag prevents formatting device if requested file system not found */
  51. #define FS_MOUNT_FLAG_NO_FORMAT BIT(0)
  52. /** Flag makes mounted file system read-only */
  53. #define FS_MOUNT_FLAG_READ_ONLY BIT(1)
  54. /** Flag used in pre-defined mount structures that are to be mounted
  55. * on startup.
  56. *
  57. * This flag has no impact in user-defined mount structures.
  58. */
  59. #define FS_MOUNT_FLAG_AUTOMOUNT BIT(2)
  60. /**
  61. * @brief File system mount info structure
  62. *
  63. * @param node Entry for the fs_mount_list list
  64. * @param type File system type
  65. * @param mnt_point Mount point directory name (ex: "/fatfs")
  66. * @param fs_data Pointer to file system specific data
  67. * @param storage_dev Pointer to backend storage device
  68. * @param mountp_len Length of Mount point string
  69. * @param fs Pointer to File system interface of the mount point
  70. * @param flags Mount flags
  71. */
  72. struct fs_mount_t {
  73. sys_dnode_t node;
  74. int type;
  75. const char *mnt_point;
  76. void *fs_data;
  77. void *storage_dev;
  78. /* fields filled by file system core */
  79. size_t mountp_len;
  80. const struct fs_file_system_t *fs;
  81. uint8_t flags;
  82. };
  83. /**
  84. * @brief Structure to receive file or directory information
  85. *
  86. * Used in functions that reads the directory entries to get
  87. * file or directory information.
  88. *
  89. * @param dir_entry_type Whether file or directory
  90. * - FS_DIR_ENTRY_FILE
  91. * - FS_DIR_ENTRY_DIR
  92. * @param name Name of directory or file
  93. * @param size Size of file. 0 if directory
  94. */
  95. struct fs_dirent {
  96. enum fs_dir_entry_type type;
  97. char name[MAX_FILE_NAME + 1];
  98. size_t size;
  99. };
  100. /**
  101. * @brief Structure to receive volume statistics
  102. *
  103. * Used to retrieve information about total and available space
  104. * in the volume.
  105. *
  106. * @param f_bsize Optimal transfer block size
  107. * @param f_frsize Allocation unit size
  108. * @param f_blocks Size of FS in f_frsize units
  109. * @param f_bfree Number of free blocks
  110. */
  111. struct fs_statvfs {
  112. unsigned long f_bsize;
  113. unsigned long f_frsize;
  114. unsigned long f_blocks;
  115. unsigned long f_bfree;
  116. };
  117. /**
  118. * @name fs_open open and creation mode flags
  119. * @{
  120. */
  121. /** Open for read flag */
  122. #define FS_O_READ 0x01
  123. /** Open for write flag */
  124. #define FS_O_WRITE 0x02
  125. /** Open for read-write flag combination */
  126. #define FS_O_RDWR (FS_O_READ | FS_O_WRITE)
  127. /** Bitmask for read and write flags */
  128. #define FS_O_MODE_MASK 0x03
  129. /** Create file if it does not exist */
  130. #define FS_O_CREATE 0x10
  131. /** Open/create file for append */
  132. #define FS_O_APPEND 0x20
  133. /** Bitmask for open/create flags */
  134. #define FS_O_FLAGS_MASK 0x30
  135. /** Bitmask for open flags */
  136. #define FS_O_MASK (FS_O_MODE_MASK | FS_O_FLAGS_MASK)
  137. /**
  138. * @}
  139. */
  140. /**
  141. * @name fs_seek whence parameter values
  142. * @{
  143. */
  144. #ifndef FS_SEEK_SET
  145. /** Seek from the beginning of file */
  146. #define FS_SEEK_SET 0
  147. #endif
  148. #ifndef FS_SEEK_CUR
  149. /** Seek from a current position */
  150. #define FS_SEEK_CUR 1
  151. #endif
  152. #ifndef FS_SEEK_END
  153. /** Seek from the end of file */
  154. #define FS_SEEK_END 2
  155. #endif
  156. /**
  157. * @}
  158. */
  159. /*
  160. * @brief Get the common mount flags for an fstab entry.
  161. * @param node_id the node identifier for a child entry in a
  162. * zephyr,fstab node.
  163. * @return a value suitable for initializing an fs_mount_t flags
  164. * member.
  165. */
  166. #define FSTAB_ENTRY_DT_MOUNT_FLAGS(node_id) \
  167. ((DT_PROP(node_id, automount) ? FS_MOUNT_FLAG_AUTOMOUNT : 0) \
  168. | (DT_PROP(node_id, read_only) ? FS_MOUNT_FLAG_READ_ONLY : 0) \
  169. | (DT_PROP(node_id, no_format) ? FS_MOUNT_FLAG_NO_FORMAT : 0))
  170. /**
  171. * @brief The name under which a zephyr,fstab entry mount structure is
  172. * defined.
  173. */
  174. #define FS_FSTAB_ENTRY(node_id) _CONCAT(z_fsmp_, node_id)
  175. /**
  176. * @brief Generate a declaration for the externally defined fstab
  177. * entry.
  178. *
  179. * This will evaluate to the name of a struct fs_mount_t object.
  180. */
  181. #define FS_FSTAB_DECLARE_ENTRY(node_id) \
  182. extern struct fs_mount_t FS_FSTAB_ENTRY(node_id)
  183. /**
  184. * @brief Initialize fs_file_t object
  185. *
  186. * Initializes the fs_file_t object; the function needs to be invoked
  187. * on object before first use with fs_open.
  188. *
  189. * @param zfp Pointer to file object
  190. *
  191. */
  192. static inline void fs_file_t_init(struct fs_file_t *zfp)
  193. {
  194. *zfp = (struct fs_file_t){ 0 };
  195. }
  196. /**
  197. * @brief Initialize fs_dir_t object
  198. *
  199. * Initializes the fs_dir_t object; the function needs to be invoked
  200. * on object before first use with fs_opendir.
  201. *
  202. * @param zdp Pointer to file object
  203. *
  204. */
  205. static inline void fs_dir_t_init(struct fs_dir_t *zdp)
  206. {
  207. *zdp = (struct fs_dir_t){ 0 };
  208. }
  209. /**
  210. * @brief Open or create file
  211. *
  212. * Opens or possibly creates a file and associates a stream with it.
  213. *
  214. * @details
  215. * @p flags can be 0 or a binary combination of one or more of the following
  216. * identifiers:
  217. * - @c FS_O_READ open for read
  218. * - @c FS_O_WRITE open for write
  219. * - @c FS_O_RDWR open for read/write (<tt>FS_O_READ | FS_O_WRITE</tt>)
  220. * - @c FS_O_CREATE create file if it does not exist
  221. * - @c FS_O_APPEND move to end of file before each write
  222. *
  223. * If @p flags are set to 0 the function will attempt to open an existing file
  224. * with no read/write access; this may be used to e.g. check if the file exists.
  225. *
  226. * @param zfp Pointer to a file object
  227. * @param file_name The name of a file to open
  228. * @param flags The mode flags
  229. *
  230. * @retval 0 on success;
  231. * @retval -EINVAL when a bad file name is given;
  232. * @retval -EROFS when opening read-only file for write, or attempting to
  233. * create a file on a system that has been mounted with the
  234. * FS_MOUNT_FLAG_READ_ONLY flag;
  235. * @retval -ENOENT when the file path is not possible (bad mount point);
  236. * @retval <0 an other negative errno code, depending on a file system back-end.
  237. */
  238. int fs_open(struct fs_file_t *zfp, const char *file_name, fs_mode_t flags);
  239. /**
  240. * @brief Close file
  241. *
  242. * Flushes the associated stream and closes the file.
  243. *
  244. * @param zfp Pointer to the file object
  245. *
  246. * @retval 0 on success;
  247. * @retval <0 a negative errno code on error.
  248. */
  249. int fs_close(struct fs_file_t *zfp);
  250. /**
  251. * @brief Unlink file
  252. *
  253. * Deletes the specified file or directory
  254. *
  255. * @param path Path to the file or directory to delete
  256. *
  257. * @retval 0 on success;
  258. * @retval -EROFS if file is read-only, or when file system has been mounted
  259. * with the FS_MOUNT_FLAG_READ_ONLY flag;
  260. * @retval -ENOTSUP when not implemented by underlying file system driver;
  261. * @retval <0 an other negative errno code on error.
  262. */
  263. int fs_unlink(const char *path);
  264. /**
  265. * @brief Rename file or directory
  266. *
  267. * Performs a rename and / or move of the specified source path to the
  268. * specified destination. The source path can refer to either a file or a
  269. * directory. All intermediate directories in the destination path must
  270. * already exist. If the source path refers to a file, the destination path
  271. * must contain a full filename path, rather than just the new parent
  272. * directory. If an object already exists at the specified destination path,
  273. * this function causes it to be unlinked prior to the rename (i.e., the
  274. * destination gets clobbered).
  275. * @note Current implementation does not allow moving files between mount
  276. * points.
  277. *
  278. * @param from The source path
  279. * @param to The destination path
  280. *
  281. * @retval 0 on success;
  282. * @retval -ENOTSUP when not implemented by underlying file system driver;
  283. * @retval <0 an other negative errno code on error.
  284. */
  285. int fs_rename(const char *from, const char *to);
  286. /**
  287. * @brief Read file
  288. *
  289. * Reads up to @p size bytes of data to @p ptr pointed buffer, returns number
  290. * of bytes read. A returned value may be lower than @p size if there were
  291. * fewer bytes available than requested.
  292. *
  293. * @param zfp Pointer to the file object
  294. * @param ptr Pointer to the data buffer
  295. * @param size Number of bytes to be read
  296. *
  297. * @retval >=0 a number of bytes read, on success;
  298. * @retval <0 a negative errno code on error.
  299. */
  300. ssize_t fs_read(struct fs_file_t *zfp, void *ptr, size_t size);
  301. /**
  302. * @brief Write file
  303. *
  304. * Attempts to write @p size number of bytes to the specified file.
  305. * If a negative value is returned from the function, the file pointer has not
  306. * been advanced.
  307. * If the function returns a non-negative number that is lower than @p size,
  308. * the global @c errno variable should be checked for an error code,
  309. * as the device may have no free space for data.
  310. *
  311. * @param zfp Pointer to the file object
  312. * @param ptr Pointer to the data buffer
  313. * @param size Number of bytes to be written
  314. *
  315. * @retval >=0 a number of bytes written, on success;
  316. * @retval -ENOTSUP when not implemented by underlying file system driver;
  317. * @retval <0 an other negative errno code on error.
  318. */
  319. ssize_t fs_write(struct fs_file_t *zfp, const void *ptr, size_t size);
  320. /**
  321. * @brief Seek file
  322. *
  323. * Moves the file position to a new location in the file. The @p offset is added
  324. * to file position based on the @p whence parameter.
  325. *
  326. * @param zfp Pointer to the file object
  327. * @param offset Relative location to move the file pointer to
  328. * @param whence Relative location from where offset is to be calculated.
  329. * - @c FS_SEEK_SET for the beginning of the file;
  330. * - @c FS_SEEK_CUR for the current position;
  331. * - @c FS_SEEK_END for the end of the file.
  332. *
  333. * @retval 0 on success;
  334. * @retval -ENOTSUP if not supported by underlying file system driver;
  335. * @retval <0 an other negative errno code on error.
  336. */
  337. int fs_seek(struct fs_file_t *zfp, off_t offset, int whence);
  338. /**
  339. * @brief Get current file position.
  340. *
  341. * Retrieves and returns the current position in the file stream.
  342. *
  343. * @param zfp Pointer to the file object
  344. *
  345. * @retval >= 0 a current position in file;
  346. * @retval -ENOTSUP if not supported by underlying file system driver;
  347. * @retval <0 an other negative errno code on error.
  348. *
  349. * The current revision does not validate the file object.
  350. */
  351. off_t fs_tell(struct fs_file_t *zfp);
  352. /**
  353. * @brief Truncate or extend an open file to a given size
  354. *
  355. * Truncates the file to the new length if it is shorter than the current
  356. * size of the file. Expands the file if the new length is greater than the
  357. * current size of the file. The expanded region would be filled with zeroes.
  358. *
  359. * @note In the case of expansion, if the volume got full during the
  360. * expansion process, the function will expand to the maximum possible length
  361. * and return success. Caller should check if the expanded size matches the
  362. * requested length.
  363. *
  364. * @param zfp Pointer to the file object
  365. * @param length New size of the file in bytes
  366. *
  367. * @retval 0 on success;
  368. * @retval -ENOTSUP when not implemented by underlying file system driver;
  369. * @retval <0 an other negative errno code on error.
  370. */
  371. int fs_truncate(struct fs_file_t *zfp, off_t length);
  372. /**
  373. * @brief Flush cached write data buffers of an open file
  374. *
  375. * The function flushes the cache of an open file; it can be invoked to ensure
  376. * data gets written to the storage media immediately, e.g. to avoid data loss
  377. * in case if power is removed unexpectedly.
  378. * @note Closing a file will cause caches to be flushed correctly so the
  379. * function need not be called when the file is being closed.
  380. *
  381. * @param zfp Pointer to the file object
  382. *
  383. * @retval 0 on success;
  384. * @retval <0 a negative errno code on error.
  385. */
  386. int fs_sync(struct fs_file_t *zfp);
  387. /**
  388. * @brief Directory create
  389. *
  390. * Creates a new directory using specified path.
  391. *
  392. * @param path Path to the directory to create
  393. *
  394. * @retval 0 on success;
  395. * @retval -ENOTSUP when not implemented by underlying file system driver;
  396. * @retval <0 an other negative errno code on error
  397. */
  398. int fs_mkdir(const char *path);
  399. /**
  400. * @brief Directory open
  401. *
  402. * Opens an existing directory specified by the path.
  403. *
  404. * @param zdp Pointer to the directory object
  405. * @param path Path to the directory to open
  406. *
  407. * @retval 0 on success;
  408. * @retval <0 a negative errno code on error.
  409. */
  410. int fs_opendir(struct fs_dir_t *zdp, const char *path);
  411. /**
  412. * @brief Directory read entry
  413. *
  414. * Reads directory entries of an open directory. In end-of-dir condition,
  415. * the function will return 0 and set the <tt>entry->name[0]</tt> to 0.
  416. *
  417. * @note: Most existing underlying file systems do not generate POSIX
  418. * special directory entries "." or "..". For consistency the
  419. * abstraction layer will remove these from lower layer results so
  420. * higher layers see consistent results.
  421. *
  422. * @param zdp Pointer to the directory object
  423. * @param entry Pointer to zfs_dirent structure to read the entry into
  424. *
  425. * @retval 0 on success or end-of-dir;;
  426. * @retval <0 a negative errno code on error.
  427. */
  428. int fs_readdir(struct fs_dir_t *zdp, struct fs_dirent *entry);
  429. /**
  430. * @brief Directory close
  431. *
  432. * Closes an open directory.
  433. *
  434. * @param zdp Pointer to the directory object
  435. *
  436. * @retval 0 on success;
  437. * @retval <0 a negative errno code on error.
  438. */
  439. int fs_closedir(struct fs_dir_t *zdp);
  440. /**
  441. * @brief Mount filesystem
  442. *
  443. * Perform steps needed for mounting a file system like
  444. * calling the file system specific mount function and adding
  445. * the mount point to mounted file system list.
  446. *
  447. * @note Current implementation of ELM FAT driver allows only following mount
  448. * points: "/RAM:","/NAND:","/CF:","/SD:","/SD2:","/USB:","/USB2:","/USB3:"
  449. * or mount points that consist of single digit, e.g: "/0:", "/1:" and so forth.
  450. *
  451. * @param mp Pointer to the fs_mount_t structure. Referenced object
  452. * is not changed if the mount operation failed.
  453. * A reference is captured in the fs infrastructure if the
  454. * mount operation succeeds, and the application must not
  455. * mutate the structure contents until fs_unmount is
  456. * successfully invoked on the same pointer.
  457. *
  458. * @retval 0 on success;
  459. * @retval -ENOENT when file system type has not been registered;
  460. * @retval -ENOTSUP when not supported by underlying file system driver;
  461. * @retval -EROFS if system requires formatting but @c FS_MOUNT_FLAG_READ_ONLY
  462. * has been set;
  463. * @retval <0 an other negative errno code on error.
  464. */
  465. int fs_mount(struct fs_mount_t *mp);
  466. /**
  467. * @brief Unmount filesystem
  468. *
  469. * Perform steps needed to unmount a file system like
  470. * calling the file system specific unmount function and removing
  471. * the mount point from mounted file system list.
  472. *
  473. * @param mp Pointer to the fs_mount_t structure
  474. *
  475. * @retval 0 on success;
  476. * @retval -EINVAL if no system has been mounted at given mount point;
  477. * @retval -ENOTSUP when not supported by underlying file system driver;
  478. * @retval <0 an other negative errno code on error.
  479. */
  480. int fs_unmount(struct fs_mount_t *mp);
  481. /**
  482. * @brief Get path of mount point at index
  483. *
  484. * This function iterates through the list of mount points and returns
  485. * the directory name of the mount point at the given @p index.
  486. * On success @p index is incremented and @p name is set to the mount directory
  487. * name. If a mount point with the given @p index does not exist, @p name will
  488. * be set to @c NULL.
  489. *
  490. * @param index Pointer to mount point index
  491. * @param name Pointer to pointer to path name
  492. *
  493. * @retval 0 on success;
  494. * @retval -ENOENT if there is no mount point with given index.
  495. */
  496. int fs_readmount(int *index, const char **name);
  497. /**
  498. * @brief File or directory status
  499. *
  500. * Checks the status of a file or directory specified by the @p path.
  501. * @note The file on a storage device may not be updated until it is closed.
  502. *
  503. * @param path Path to the file or directory
  504. * @param entry Pointer to the zfs_dirent structure to fill if the file or
  505. * directory exists.
  506. *
  507. * @retval 0 on success;
  508. * @retval <0 negative errno code on error.
  509. */
  510. int fs_stat(const char *path, struct fs_dirent *entry);
  511. /**
  512. * @brief Retrieves statistics of the file system volume
  513. *
  514. * Returns the total and available space in the file system volume.
  515. *
  516. * @param path Path to the mounted directory
  517. * @param stat Pointer to the zfs_statvfs structure to receive the fs
  518. * statistics
  519. *
  520. * @retval 0 on success;
  521. * @retval -ENOTSUP when not implemented by underlying file system driver;
  522. * @retval <0 an other negative errno code on error.
  523. */
  524. int fs_statvfs(const char *path, struct fs_statvfs *stat);
  525. /**
  526. * @brief Register a file system
  527. *
  528. * Register file system with virtual file system.
  529. *
  530. * @param type Type of file system (ex: @c FS_FATFS)
  531. * @param fs Pointer to File system
  532. *
  533. * @retval 0 on success;
  534. * @retval <0 negative errno code on error.
  535. */
  536. int fs_register(int type, const struct fs_file_system_t *fs);
  537. /**
  538. * @brief Unregister a file system
  539. *
  540. * Unregister file system from virtual file system.
  541. *
  542. * @param type Type of file system (ex: @c FS_FATFS)
  543. * @param fs Pointer to File system
  544. *
  545. * @retval 0 on success;
  546. * @retval <0 negative errno code on error.
  547. */
  548. int fs_unregister(int type, const struct fs_file_system_t *fs);
  549. /**
  550. * @brief Directory open
  551. *
  552. * Opens an existing directory specified by the cluster and blk_ofs.
  553. *
  554. * @param zdp Pointer to the directory object
  555. * @param path Path to the root directory to open
  556. * @param cluster is the cluster of directory to be open
  557. * @param blk_ofs is the blk_ofs of directory in cluster
  558. *
  559. * @retval 0 Success
  560. * @retval -ERRNO errno code if error
  561. */
  562. int fs_opendir_cluster(struct fs_dir_t *zdp, const char *path, unsigned int cluster, unsigned int blk_ofs);
  563. /**
  564. * @brief File open
  565. *
  566. * Opens an existing file or create a new one and associates
  567. * a stream with it.
  568. *
  569. * @param zfp Pointer to file object
  570. * @param dir The root dir
  571. * @param cluster is the cluster of directory to be open
  572. * @param blk_ofs is the blk_ofs of directory in cluster
  573. *
  574. * @retval 0 Success
  575. * @retval -ERRNO errno code if error
  576. */
  577. int fs_open_cluster(struct fs_file_t *zfp, char *dir, unsigned int cluster, unsigned int blk_ofs);
  578. /**
  579. * @brief detect disk
  580. *
  581. * detect disk is add or remove
  582. *
  583. * @param str One of _VOLUME_STRS or numeric drive id with colon
  584. * @param state 0 disk ok, 1 STA_NOINIT, 2 STA_NODISK other: unknow error
  585. *
  586. * @retval 0 Success
  587. * @retval -ERRNO errno code if error
  588. */
  589. int fs_disk_detect(const char *str, unsigned char *state);
  590. /**
  591. * @}
  592. */
  593. #ifdef __cplusplus
  594. }
  595. #endif
  596. #endif /* ZEPHYR_INCLUDE_FS_FS_H_ */