fs.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387
  1. /*
  2. * Copyright (c) 2018 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <errno.h>
  7. #include <kernel.h>
  8. #include <limits.h>
  9. #include <posix/unistd.h>
  10. #include <posix/dirent.h>
  11. #include <string.h>
  12. #include <sys/fdtable.h>
  13. #include <sys/stat.h>
  14. #include <fcntl.h>
  15. #include <fs/fs.h>
  16. BUILD_ASSERT(PATH_MAX >= MAX_FILE_NAME, "PATH_MAX is less than MAX_FILE_NAME");
  17. struct posix_fs_desc {
  18. union {
  19. struct fs_file_t file;
  20. struct fs_dir_t dir;
  21. };
  22. bool is_dir;
  23. bool used;
  24. };
  25. static struct posix_fs_desc desc_array[CONFIG_POSIX_MAX_OPEN_FILES];
  26. static struct fs_dirent fdirent;
  27. static struct dirent pdirent;
  28. static struct fd_op_vtable fs_fd_op_vtable;
  29. static struct posix_fs_desc *posix_fs_alloc_obj(bool is_dir)
  30. {
  31. int i;
  32. struct posix_fs_desc *ptr = NULL;
  33. unsigned int key = irq_lock();
  34. for (i = 0; i < CONFIG_POSIX_MAX_OPEN_FILES; i++) {
  35. if (desc_array[i].used == false) {
  36. ptr = &desc_array[i];
  37. ptr->used = true;
  38. ptr->is_dir = is_dir;
  39. break;
  40. }
  41. }
  42. irq_unlock(key);
  43. return ptr;
  44. }
  45. static inline void posix_fs_free_obj(struct posix_fs_desc *ptr)
  46. {
  47. ptr->used = false;
  48. }
  49. static int posix_mode_to_zephyr(int mf)
  50. {
  51. int mode = (mf & O_CREAT) ? FS_O_CREATE : 0;
  52. mode |= (mf & O_APPEND) ? FS_O_APPEND : 0;
  53. switch (mf & O_ACCMODE) {
  54. case O_RDONLY:
  55. mode |= FS_O_READ;
  56. break;
  57. case O_WRONLY:
  58. mode |= FS_O_WRITE;
  59. break;
  60. case O_RDWR:
  61. mode |= FS_O_RDWR;
  62. break;
  63. default:
  64. break;
  65. }
  66. return mode;
  67. }
  68. /**
  69. * @brief Open a file.
  70. *
  71. * See IEEE 1003.1
  72. */
  73. int open(const char *name, int flags, ...)
  74. {
  75. int rc, fd;
  76. struct posix_fs_desc *ptr = NULL;
  77. int zmode = posix_mode_to_zephyr(flags);
  78. if (zmode < 0) {
  79. return zmode;
  80. }
  81. fd = z_reserve_fd();
  82. if (fd < 0) {
  83. return -1;
  84. }
  85. ptr = posix_fs_alloc_obj(false);
  86. if (ptr == NULL) {
  87. z_free_fd(fd);
  88. errno = EMFILE;
  89. return -1;
  90. }
  91. fs_file_t_init(&ptr->file);
  92. rc = fs_open(&ptr->file, name, zmode);
  93. if (rc < 0) {
  94. posix_fs_free_obj(ptr);
  95. z_free_fd(fd);
  96. errno = -rc;
  97. return -1;
  98. }
  99. z_finalize_fd(fd, ptr, &fs_fd_op_vtable);
  100. return fd;
  101. }
  102. FUNC_ALIAS(open, _open, int);
  103. static int fs_close_vmeth(void *obj)
  104. {
  105. struct posix_fs_desc *ptr = obj;
  106. int rc;
  107. rc = fs_close(&ptr->file);
  108. posix_fs_free_obj(ptr);
  109. return rc;
  110. }
  111. static int fs_ioctl_vmeth(void *obj, unsigned int request, va_list args)
  112. {
  113. int rc = 0;
  114. struct posix_fs_desc *ptr = obj;
  115. switch (request) {
  116. case ZFD_IOCTL_LSEEK: {
  117. off_t offset;
  118. int whence;
  119. offset = va_arg(args, off_t);
  120. whence = va_arg(args, int);
  121. rc = fs_seek(&ptr->file, offset, whence);
  122. if (rc == 0) {
  123. rc = fs_tell(&ptr->file);
  124. }
  125. break;
  126. }
  127. case ZFD_IOCTL_FSYNC: {
  128. rc = fs_sync(&ptr->file);
  129. break;
  130. }
  131. default:
  132. errno = EOPNOTSUPP;
  133. return -1;
  134. }
  135. if (rc < 0) {
  136. errno = -rc;
  137. return -1;
  138. }
  139. return rc;
  140. }
  141. /**
  142. * @brief Write to a file.
  143. *
  144. * See IEEE 1003.1
  145. */
  146. static ssize_t fs_write_vmeth(void *obj, const void *buffer, size_t count)
  147. {
  148. ssize_t rc;
  149. struct posix_fs_desc *ptr = obj;
  150. rc = fs_write(&ptr->file, buffer, count);
  151. if (rc < 0) {
  152. errno = -rc;
  153. return -1;
  154. }
  155. return rc;
  156. }
  157. /**
  158. * @brief Read from a file.
  159. *
  160. * See IEEE 1003.1
  161. */
  162. static ssize_t fs_read_vmeth(void *obj, void *buffer, size_t count)
  163. {
  164. ssize_t rc;
  165. struct posix_fs_desc *ptr = obj;
  166. rc = fs_read(&ptr->file, buffer, count);
  167. if (rc < 0) {
  168. errno = -rc;
  169. return -1;
  170. }
  171. return rc;
  172. }
  173. static struct fd_op_vtable fs_fd_op_vtable = {
  174. .read = fs_read_vmeth,
  175. .write = fs_write_vmeth,
  176. .close = fs_close_vmeth,
  177. .ioctl = fs_ioctl_vmeth,
  178. };
  179. /**
  180. * @brief Open a directory stream.
  181. *
  182. * See IEEE 1003.1
  183. */
  184. DIR *opendir(const char *dirname)
  185. {
  186. int rc;
  187. struct posix_fs_desc *ptr;
  188. ptr = posix_fs_alloc_obj(true);
  189. if (ptr == NULL) {
  190. errno = EMFILE;
  191. return NULL;
  192. }
  193. fs_dir_t_init(&ptr->dir);
  194. rc = fs_opendir(&ptr->dir, dirname);
  195. if (rc < 0) {
  196. posix_fs_free_obj(ptr);
  197. errno = -rc;
  198. return NULL;
  199. }
  200. return ptr;
  201. }
  202. /**
  203. * @brief Close a directory stream.
  204. *
  205. * See IEEE 1003.1
  206. */
  207. int closedir(DIR *dirp)
  208. {
  209. int rc;
  210. struct posix_fs_desc *ptr = dirp;
  211. if (dirp == NULL) {
  212. errno = EBADF;
  213. return -1;
  214. }
  215. rc = fs_closedir(&ptr->dir);
  216. posix_fs_free_obj(ptr);
  217. if (rc < 0) {
  218. errno = -rc;
  219. return -1;
  220. }
  221. return 0;
  222. }
  223. /**
  224. * @brief Read a directory.
  225. *
  226. * See IEEE 1003.1
  227. */
  228. struct dirent *readdir(DIR *dirp)
  229. {
  230. int rc;
  231. struct posix_fs_desc *ptr = dirp;
  232. if (dirp == NULL) {
  233. errno = EBADF;
  234. return NULL;
  235. }
  236. rc = fs_readdir(&ptr->dir, &fdirent);
  237. if (rc < 0) {
  238. errno = -rc;
  239. return NULL;
  240. }
  241. rc = strlen(fdirent.name);
  242. rc = (rc < MAX_FILE_NAME) ? rc : (MAX_FILE_NAME - 1);
  243. (void)memcpy(pdirent.d_name, fdirent.name, rc);
  244. /* Make sure the name is NULL terminated */
  245. pdirent.d_name[rc] = '\0';
  246. pdirent.is_dir = (fdirent.type == FS_DIR_ENTRY_DIR ? 1 : 0);
  247. return &pdirent;
  248. }
  249. /**
  250. * @brief Rename a file.
  251. *
  252. * See IEEE 1003.1
  253. */
  254. int rename(const char *old, const char *new)
  255. {
  256. int rc;
  257. rc = fs_rename(old, new);
  258. if (rc < 0) {
  259. errno = -rc;
  260. return -1;
  261. }
  262. return 0;
  263. }
  264. /**
  265. * @brief Remove a directory entry.
  266. *
  267. * See IEEE 1003.1
  268. */
  269. int unlink(const char *path)
  270. {
  271. int rc;
  272. rc = fs_unlink(path);
  273. if (rc < 0) {
  274. errno = -rc;
  275. return -1;
  276. }
  277. return 0;
  278. }
  279. /**
  280. * @brief Get file status.
  281. *
  282. * See IEEE 1003.1
  283. */
  284. int stat(const char *path, struct stat *buf)
  285. {
  286. int rc;
  287. struct fs_statvfs stat;
  288. if (buf == NULL) {
  289. errno = EBADF;
  290. return -1;
  291. }
  292. rc = fs_statvfs(path, &stat);
  293. if (rc < 0) {
  294. errno = -rc;
  295. return -1;
  296. }
  297. buf->st_size = stat.f_bsize * stat.f_blocks;
  298. buf->st_blksize = stat.f_bsize;
  299. buf->st_blocks = stat.f_blocks;
  300. return 0;
  301. }
  302. /**
  303. * @brief Make a directory.
  304. *
  305. * See IEEE 1003.1
  306. */
  307. int mkdir(const char *path, mode_t mode)
  308. {
  309. int rc;
  310. ARG_UNUSED(mode);
  311. rc = fs_mkdir(path);
  312. if (rc < 0) {
  313. errno = -rc;
  314. return -1;
  315. }
  316. return 0;
  317. }