fuse_fs_access.c 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  1. /*
  2. * Copyright (c) 2019 Jan Van Winkel <jan.van_winkel@dxplore.eu>
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #define FUSE_USE_VERSION 26
  7. #include <fuse.h>
  8. #include <libgen.h>
  9. #include <linux/limits.h>
  10. #include <unistd.h>
  11. #include <pthread.h>
  12. #include <stdio.h>
  13. #include <stdlib.h>
  14. #include <string.h>
  15. #include <sys/mount.h>
  16. #include <sys/time.h>
  17. #include <sys/types.h>
  18. #include <zephyr.h>
  19. #include <fs/fs.h>
  20. #include "cmdline.h"
  21. #include "soc.h"
  22. #define S_IRWX_DIR (0775)
  23. #define S_IRW_FILE (0664)
  24. #define NUMBER_OF_OPEN_FILES 128
  25. #define INVALID_FILE_HANDLE (NUMBER_OF_OPEN_FILES + 1)
  26. #define DIR_END '\0'
  27. static struct fs_file_t files[NUMBER_OF_OPEN_FILES];
  28. static uint8_t file_handles[NUMBER_OF_OPEN_FILES];
  29. static pthread_t fuse_thread;
  30. static const char default_fuse_mountpoint[] = "flash";
  31. static const char *fuse_mountpoint;
  32. static ssize_t get_new_file_handle(void)
  33. {
  34. size_t idx;
  35. for (idx = 0; idx < ARRAY_SIZE(file_handles); ++idx) {
  36. if (file_handles[idx] == 0) {
  37. ++file_handles[idx];
  38. return idx;
  39. }
  40. }
  41. return -ENOMEM;
  42. }
  43. static void release_file_handle(size_t handle)
  44. {
  45. if (handle < ARRAY_SIZE(file_handles)) {
  46. --file_handles[handle];
  47. }
  48. }
  49. static bool is_mount_point(const char *path)
  50. {
  51. char dir_path[PATH_MAX];
  52. sprintf(dir_path, "%s", path);
  53. return strcmp(dirname(dir_path), "/") == 0;
  54. }
  55. static int fuse_fs_access_getattr(const char *path, struct stat *stat)
  56. {
  57. struct fs_dirent entry;
  58. int err;
  59. stat->st_dev = 0;
  60. stat->st_ino = 0;
  61. stat->st_nlink = 0;
  62. stat->st_uid = getuid();
  63. stat->st_gid = getgid();
  64. stat->st_rdev = 0;
  65. stat->st_blksize = 0;
  66. stat->st_blocks = 0;
  67. stat->st_atime = 0;
  68. stat->st_mtime = 0;
  69. stat->st_ctime = 0;
  70. if ((strcmp(path, "/") == 0) || is_mount_point(path)) {
  71. if (strstr(path, "/.") != NULL) {
  72. return -ENOENT;
  73. }
  74. stat->st_mode = S_IFDIR | S_IRWX_DIR;
  75. stat->st_size = 0;
  76. return 0;
  77. }
  78. err = fs_stat(path, &entry);
  79. if (err != 0) {
  80. return err;
  81. }
  82. if (entry.type == FS_DIR_ENTRY_DIR) {
  83. stat->st_mode = S_IFDIR | S_IRWX_DIR;
  84. stat->st_size = 0;
  85. } else {
  86. stat->st_mode = S_IFREG | S_IRW_FILE;
  87. stat->st_size = entry.size;
  88. }
  89. return 0;
  90. }
  91. static int fuse_fs_access_readmount(void *buf, fuse_fill_dir_t filler)
  92. {
  93. int mnt_nbr = 0;
  94. const char *mnt_name;
  95. struct stat stat;
  96. int err;
  97. stat.st_dev = 0;
  98. stat.st_ino = 0;
  99. stat.st_nlink = 0;
  100. stat.st_uid = getuid();
  101. stat.st_gid = getgid();
  102. stat.st_rdev = 0;
  103. stat.st_atime = 0;
  104. stat.st_mtime = 0;
  105. stat.st_ctime = 0;
  106. stat.st_mode = S_IFDIR | S_IRWX_DIR;
  107. stat.st_size = 0;
  108. stat.st_blksize = 0;
  109. stat.st_blocks = 0;
  110. filler(buf, ".", &stat, 0);
  111. filler(buf, "..", NULL, 0);
  112. do {
  113. err = fs_readmount(&mnt_nbr, &mnt_name);
  114. if (err < 0) {
  115. break;
  116. }
  117. filler(buf, &mnt_name[1], &stat, 0);
  118. } while (true);
  119. if (err == -ENOENT) {
  120. err = 0;
  121. }
  122. return err;
  123. }
  124. static int fuse_fs_access_readdir(const char *path, void *buf,
  125. fuse_fill_dir_t filler, off_t off,
  126. struct fuse_file_info *fi)
  127. {
  128. struct fs_dir_t dir;
  129. struct fs_dirent entry;
  130. int err;
  131. struct stat stat;
  132. ARG_UNUSED(off);
  133. ARG_UNUSED(fi);
  134. if (strcmp(path, "/") == 0) {
  135. return fuse_fs_access_readmount(buf, filler);
  136. }
  137. fs_dir_t_init(&dir);
  138. if (is_mount_point(path)) {
  139. /* File system API expects trailing slash for a mount point
  140. * directory but FUSE strips the trailing slashes from
  141. * directory names so add it back.
  142. */
  143. char mount_path[PATH_MAX];
  144. sprintf(mount_path, "%s/", path);
  145. err = fs_opendir(&dir, mount_path);
  146. } else {
  147. err = fs_opendir(&dir, path);
  148. }
  149. if (err) {
  150. return -ENOEXEC;
  151. }
  152. stat.st_dev = 0;
  153. stat.st_ino = 0;
  154. stat.st_nlink = 0;
  155. stat.st_uid = getuid();
  156. stat.st_gid = getgid();
  157. stat.st_rdev = 0;
  158. stat.st_atime = 0;
  159. stat.st_mtime = 0;
  160. stat.st_ctime = 0;
  161. stat.st_mode = S_IFDIR | S_IRWX_DIR;
  162. stat.st_size = 0;
  163. stat.st_blksize = 0;
  164. stat.st_blocks = 0;
  165. filler(buf, ".", &stat, 0);
  166. filler(buf, "..", &stat, 0);
  167. do {
  168. err = fs_readdir(&dir, &entry);
  169. if (err) {
  170. break;
  171. }
  172. if (entry.name[0] == DIR_END) {
  173. break;
  174. }
  175. if (entry.type == FS_DIR_ENTRY_DIR) {
  176. stat.st_mode = S_IFDIR | S_IRWX_DIR;
  177. stat.st_size = 0;
  178. } else {
  179. stat.st_mode = S_IFREG | S_IRW_FILE;
  180. stat.st_size = entry.size;
  181. }
  182. if (filler(buf, entry.name, &stat, 0)) {
  183. break;
  184. }
  185. } while (1);
  186. fs_closedir(&dir);
  187. return err;
  188. }
  189. static int fuse_fs_access_create(const char *path, mode_t mode,
  190. struct fuse_file_info *fi)
  191. {
  192. int err;
  193. ssize_t handle;
  194. ARG_UNUSED(mode);
  195. if (is_mount_point(path)) {
  196. return -ENOENT;
  197. }
  198. handle = get_new_file_handle();
  199. if (handle < 0) {
  200. return handle;
  201. }
  202. fi->fh = handle;
  203. err = fs_open(&files[handle], path, FS_O_CREATE | FS_O_WRITE);
  204. if (err != 0) {
  205. release_file_handle(handle);
  206. fi->fh = INVALID_FILE_HANDLE;
  207. return err;
  208. }
  209. return 0;
  210. }
  211. static int fuse_fs_access_open(const char *path, struct fuse_file_info *fi)
  212. {
  213. return fuse_fs_access_create(path, 0, fi);
  214. }
  215. static int fuse_fs_access_release(const char *path, struct fuse_file_info *fi)
  216. {
  217. ARG_UNUSED(path);
  218. if (fi->fh == INVALID_FILE_HANDLE) {
  219. return -EINVAL;
  220. }
  221. fs_close(&files[fi->fh]);
  222. release_file_handle(fi->fh);
  223. return 0;
  224. }
  225. static int fuse_fs_access_read(const char *path, char *buf, size_t size,
  226. off_t off, struct fuse_file_info *fi)
  227. {
  228. int err;
  229. ARG_UNUSED(path);
  230. if (fi->fh == INVALID_FILE_HANDLE) {
  231. return -EINVAL;
  232. }
  233. err = fs_seek(&files[fi->fh], off, FS_SEEK_SET);
  234. if (err != 0) {
  235. return err;
  236. }
  237. err = fs_read(&files[fi->fh], buf, size);
  238. return err;
  239. }
  240. static int fuse_fs_access_write(const char *path, const char *buf, size_t size,
  241. off_t off, struct fuse_file_info *fi)
  242. {
  243. int err;
  244. ARG_UNUSED(path);
  245. if (fi->fh == INVALID_FILE_HANDLE) {
  246. return -EINVAL;
  247. }
  248. err = fs_seek(&files[fi->fh], off, FS_SEEK_SET);
  249. if (err != 0) {
  250. return err;
  251. }
  252. err = fs_write(&files[fi->fh], buf, size);
  253. return err;
  254. }
  255. static int fuse_fs_access_ftruncate(const char *path, off_t size,
  256. struct fuse_file_info *fi)
  257. {
  258. int err;
  259. ARG_UNUSED(path);
  260. if (fi->fh == INVALID_FILE_HANDLE) {
  261. return -EINVAL;
  262. }
  263. err = fs_truncate(&files[fi->fh], size);
  264. return err;
  265. }
  266. static int fuse_fs_access_truncate(const char *path, off_t size)
  267. {
  268. int err;
  269. static struct fs_file_t file;
  270. err = fs_open(&file, path, FS_O_CREATE | FS_O_WRITE);
  271. if (err != 0) {
  272. return err;
  273. }
  274. err = fs_truncate(&file, size);
  275. if (err != 0) {
  276. fs_close(&file);
  277. return err;
  278. }
  279. err = fs_close(&file);
  280. return err;
  281. }
  282. static int fuse_fs_access_mkdir(const char *path, mode_t mode)
  283. {
  284. ARG_UNUSED(mode);
  285. return fs_mkdir(path);
  286. }
  287. static int fuse_fs_access_rmdir(const char *path)
  288. {
  289. return fs_unlink(path);
  290. }
  291. static int fuse_fs_access_unlink(const char *path)
  292. {
  293. return fs_unlink(path);
  294. }
  295. static int fuse_fs_access_statfs(const char *path, struct statvfs *buf)
  296. {
  297. ARG_UNUSED(path);
  298. ARG_UNUSED(buf);
  299. return 0;
  300. }
  301. static int fuse_fs_access_utimens(const char *path, const struct timespec tv[2])
  302. {
  303. /* dummy */
  304. ARG_UNUSED(path);
  305. ARG_UNUSED(tv);
  306. return 0;
  307. }
  308. static struct fuse_operations fuse_fs_access_oper = {
  309. .getattr = fuse_fs_access_getattr,
  310. .readlink = NULL,
  311. .getdir = NULL,
  312. .mknod = NULL,
  313. .mkdir = fuse_fs_access_mkdir,
  314. .unlink = fuse_fs_access_unlink,
  315. .rmdir = fuse_fs_access_rmdir,
  316. .symlink = NULL,
  317. .rename = NULL,
  318. .link = NULL,
  319. .chmod = NULL,
  320. .chown = NULL,
  321. .truncate = fuse_fs_access_truncate,
  322. .utime = NULL,
  323. .open = fuse_fs_access_open,
  324. .read = fuse_fs_access_read,
  325. .write = fuse_fs_access_write,
  326. .statfs = fuse_fs_access_statfs,
  327. .flush = NULL,
  328. .release = fuse_fs_access_release,
  329. .fsync = NULL,
  330. .setxattr = NULL,
  331. .getxattr = NULL,
  332. .listxattr = NULL,
  333. .removexattr = NULL,
  334. .opendir = NULL,
  335. .readdir = fuse_fs_access_readdir,
  336. .releasedir = NULL,
  337. .fsyncdir = NULL,
  338. .init = NULL,
  339. .destroy = NULL,
  340. .access = NULL,
  341. .create = fuse_fs_access_create,
  342. .ftruncate = fuse_fs_access_ftruncate,
  343. .fgetattr = NULL,
  344. .lock = NULL,
  345. .utimens = fuse_fs_access_utimens,
  346. .bmap = NULL,
  347. .flag_nullpath_ok = 0,
  348. .flag_nopath = 0,
  349. .flag_utime_omit_ok = 0,
  350. .flag_reserved = 0,
  351. .ioctl = NULL,
  352. .poll = NULL,
  353. .write_buf = NULL,
  354. .read_buf = NULL,
  355. .flock = NULL,
  356. .fallocate = NULL,
  357. };
  358. static void *fuse_fs_access_main(void *arg)
  359. {
  360. ARG_UNUSED(arg);
  361. char *argv[] = {
  362. "",
  363. "-f",
  364. "-s",
  365. (char *) fuse_mountpoint
  366. };
  367. int argc = ARRAY_SIZE(argv);
  368. posix_print_trace("Mounting flash at %s/\n", fuse_mountpoint);
  369. fuse_main(argc, argv, &fuse_fs_access_oper, NULL);
  370. pthread_exit(0);
  371. }
  372. static void fuse_fs_access_init(void)
  373. {
  374. int err;
  375. struct stat st;
  376. size_t i = 0;
  377. while (i < ARRAY_SIZE(files)) {
  378. fs_file_t_init(&files[i]);
  379. ++i;
  380. }
  381. if (fuse_mountpoint == NULL) {
  382. fuse_mountpoint = default_fuse_mountpoint;
  383. }
  384. if (stat(fuse_mountpoint, &st) < 0) {
  385. if (mkdir(fuse_mountpoint, 0700) < 0) {
  386. posix_print_error_and_exit("Failed to create"
  387. " directory for flash mount point (%s): %s\n",
  388. fuse_mountpoint, strerror(errno));
  389. }
  390. } else if (!S_ISDIR(st.st_mode)) {
  391. posix_print_error_and_exit("%s is not a directory\n",
  392. fuse_mountpoint);
  393. }
  394. err = pthread_create(&fuse_thread, NULL, fuse_fs_access_main, NULL);
  395. if (err < 0) {
  396. posix_print_error_and_exit(
  397. "Failed to create thread for "
  398. "fuse_fs_access_main\n");
  399. }
  400. }
  401. static void fuse_fs_access_exit(void)
  402. {
  403. char *full_cmd;
  404. const char cmd[] = "fusermount -uz ";
  405. if (fuse_mountpoint == NULL) {
  406. return;
  407. }
  408. full_cmd = malloc(strlen(cmd) + strlen(fuse_mountpoint) + 1);
  409. sprintf(full_cmd, "%s%s", cmd, fuse_mountpoint);
  410. if (system(full_cmd) < -1) {
  411. printf("Failed to unmount fuse mount point\n");
  412. }
  413. free(full_cmd);
  414. pthread_join(fuse_thread, NULL);
  415. }
  416. static void fuse_fs_access_options(void)
  417. {
  418. static struct args_struct_t fuse_fs_access_options[] = {
  419. { .manual = false,
  420. .is_mandatory = false,
  421. .is_switch = false,
  422. .option = "flash-mount",
  423. .name = "path",
  424. .type = 's',
  425. .dest = (void *)&fuse_mountpoint,
  426. .call_when_found = NULL,
  427. .descript = "Path to the directory where to mount flash" },
  428. ARG_TABLE_ENDMARKER
  429. };
  430. native_add_command_line_opts(fuse_fs_access_options);
  431. }
  432. NATIVE_TASK(fuse_fs_access_options, PRE_BOOT_1, 1);
  433. NATIVE_TASK(fuse_fs_access_init, PRE_BOOT_2, 1);
  434. NATIVE_TASK(fuse_fs_access_exit, ON_EXIT, 1);