shell.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668
  1. /*
  2. * Copyright (c) 2016 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <errno.h>
  7. #include <stdio.h>
  8. #include <string.h>
  9. #include <shell/shell.h>
  10. #include <init.h>
  11. #include <fs/fs.h>
  12. #include <stdio.h>
  13. #include <stdlib.h>
  14. #include <inttypes.h>
  15. #include <limits.h>
  16. /* FAT */
  17. #ifdef CONFIG_FAT_FILESYSTEM_ELM
  18. #include <ff.h>
  19. #define FATFS_MNTP "/RAM:"
  20. /* FatFs work area */
  21. FATFS fat_fs;
  22. /* mounting info */
  23. static struct fs_mount_t fatfs_mnt = {
  24. .type = FS_FATFS,
  25. .fs_data = &fat_fs,
  26. };
  27. #endif
  28. /* LITTLEFS */
  29. #ifdef CONFIG_FILE_SYSTEM_LITTLEFS
  30. #include <fs/littlefs.h>
  31. #include <storage/flash_map.h>
  32. FS_LITTLEFS_DECLARE_DEFAULT_CONFIG(lfs_data);
  33. static struct fs_mount_t littlefs_mnt = {
  34. .type = FS_LITTLEFS,
  35. .fs_data = &lfs_data,
  36. .storage_dev = (void *)FLASH_AREA_ID(storage),
  37. };
  38. #endif
  39. #define BUF_CNT 64
  40. #define MAX_PATH_LEN 128
  41. #define MAX_FILENAME_LEN 128
  42. #define MAX_INPUT_LEN 20
  43. #define SHELL_FS "fs"
  44. /* Maintenance guarantees this begins with '/' and is NUL-terminated. */
  45. static char cwd[MAX_PATH_LEN] = "/";
  46. static void create_abs_path(const char *name, char *path, size_t len)
  47. {
  48. if (name[0] == '/') {
  49. strncpy(path, name, len);
  50. path[len - 1] = '\0';
  51. } else {
  52. if (cwd[1] == '\0') {
  53. __ASSERT_NO_MSG(len >= 2);
  54. *path++ = '/';
  55. --len;
  56. strncpy(path, name, len);
  57. path[len - 1] = '\0';
  58. } else {
  59. strncpy(path, cwd, len);
  60. path[len - 1] = '\0';
  61. size_t plen = strlen(path);
  62. if (plen < len) {
  63. path += plen;
  64. *path++ = '/';
  65. len -= plen + 1U;
  66. strncpy(path, name, len);
  67. path[len - 1] = '\0';
  68. }
  69. }
  70. }
  71. }
  72. static int cmd_cd(const struct shell *shell, size_t argc, char **argv)
  73. {
  74. char path[MAX_PATH_LEN];
  75. struct fs_dirent entry;
  76. int err;
  77. if (argc < 2) {
  78. strcpy(cwd, "/");
  79. return 0;
  80. }
  81. if (strcmp(argv[1], "..") == 0) {
  82. char *prev = strrchr(cwd, '/');
  83. if (!prev || prev == cwd) {
  84. strcpy(cwd, "/");
  85. } else {
  86. *prev = '\0';
  87. }
  88. /* No need to test that a parent exists */
  89. return 0;
  90. }
  91. create_abs_path(argv[1], path, sizeof(path));
  92. err = fs_stat(path, &entry);
  93. if (err) {
  94. shell_error(shell, "%s doesn't exist", path);
  95. return -ENOEXEC;
  96. }
  97. if (entry.type != FS_DIR_ENTRY_DIR) {
  98. shell_error(shell, "%s is not a directory", path);
  99. return -ENOEXEC;
  100. }
  101. strncpy(cwd, path, sizeof(cwd));
  102. cwd[sizeof(cwd) - 1] = '\0';
  103. return 0;
  104. }
  105. static int cmd_ls(const struct shell *shell, size_t argc, char **argv)
  106. {
  107. char path[MAX_PATH_LEN];
  108. struct fs_dir_t dir;
  109. int err;
  110. if (argc < 2) {
  111. strncpy(path, cwd, sizeof(path));
  112. path[sizeof(path) - 1] = '\0';
  113. } else {
  114. create_abs_path(argv[1], path, sizeof(path));
  115. }
  116. fs_dir_t_init(&dir);
  117. err = fs_opendir(&dir, path);
  118. if (err) {
  119. shell_error(shell, "Unable to open %s (err %d)", path, err);
  120. return -ENOEXEC;
  121. }
  122. while (1) {
  123. struct fs_dirent entry;
  124. err = fs_readdir(&dir, &entry);
  125. if (err) {
  126. shell_error(shell, "Unable to read directory");
  127. break;
  128. }
  129. /* Check for end of directory listing */
  130. if (entry.name[0] == '\0') {
  131. break;
  132. }
  133. shell_print(shell, "%s%s", entry.name,
  134. (entry.type == FS_DIR_ENTRY_DIR) ? "/" : "");
  135. }
  136. fs_closedir(&dir);
  137. return 0;
  138. }
  139. static int cmd_pwd(const struct shell *shell, size_t argc, char **argv)
  140. {
  141. shell_print(shell, "%s", cwd);
  142. return 0;
  143. }
  144. static int cmd_trunc(const struct shell *shell, size_t argc, char **argv)
  145. {
  146. char path[MAX_PATH_LEN];
  147. struct fs_file_t file;
  148. int length;
  149. int err;
  150. create_abs_path(argv[1], path, sizeof(path));
  151. if (argc > 2) {
  152. length = strtol(argv[2], NULL, 0);
  153. } else {
  154. length = 0;
  155. }
  156. fs_file_t_init(&file);
  157. err = fs_open(&file, path, FS_O_WRITE);
  158. if (err) {
  159. shell_error(shell, "Failed to open %s (%d)", path, err);
  160. return -ENOEXEC;;
  161. }
  162. err = fs_truncate(&file, length);
  163. if (err) {
  164. shell_error(shell, "Failed to truncate %s (%d)", path, err);
  165. err = -ENOEXEC;
  166. }
  167. fs_close(&file);
  168. return err;
  169. }
  170. static int cmd_mkdir(const struct shell *shell, size_t argc, char **argv)
  171. {
  172. int err;
  173. char path[MAX_PATH_LEN];
  174. create_abs_path(argv[1], path, sizeof(path));
  175. err = fs_mkdir(path);
  176. if (err) {
  177. shell_error(shell, "Error creating dir[%d]", err);
  178. err = -ENOEXEC;
  179. }
  180. return err;
  181. }
  182. static int cmd_rm(const struct shell *shell, size_t argc, char **argv)
  183. {
  184. int err;
  185. char path[MAX_PATH_LEN];
  186. create_abs_path(argv[1], path, sizeof(path));
  187. err = fs_unlink(path);
  188. if (err) {
  189. shell_error(shell, "Failed to remove %s (%d)", path, err);
  190. err = -ENOEXEC;
  191. }
  192. return err;
  193. }
  194. static int cmd_read(const struct shell *shell, size_t argc, char **argv)
  195. {
  196. char path[MAX_PATH_LEN];
  197. struct fs_dirent dirent;
  198. struct fs_file_t file;
  199. int count;
  200. off_t offset;
  201. int err;
  202. create_abs_path(argv[1], path, sizeof(path));
  203. if (argc > 2) {
  204. count = strtol(argv[2], NULL, 0);
  205. if (count <= 0) {
  206. count = INT_MAX;
  207. }
  208. } else {
  209. count = INT_MAX;
  210. }
  211. if (argc > 3) {
  212. offset = strtol(argv[3], NULL, 0);
  213. } else {
  214. offset = 0;
  215. }
  216. err = fs_stat(path, &dirent);
  217. if (err) {
  218. shell_error(shell, "Failed to obtain file %s (err: %d)",
  219. path, err);
  220. return -ENOEXEC;
  221. }
  222. if (dirent.type != FS_DIR_ENTRY_FILE) {
  223. shell_error(shell, "Note a file %s", path);
  224. return -ENOEXEC;
  225. }
  226. shell_print(shell, "File size: %zd", dirent.size);
  227. fs_file_t_init(&file);
  228. err = fs_open(&file, path, FS_O_READ);
  229. if (err) {
  230. shell_error(shell, "Failed to open %s (%d)", path, err);
  231. return -ENOEXEC;
  232. }
  233. if (offset > 0) {
  234. err = fs_seek(&file, offset, FS_SEEK_SET);
  235. if (err) {
  236. shell_error(shell, "Failed to seek %s (%d)",
  237. path, err);
  238. fs_close(&file);
  239. return -ENOEXEC;
  240. }
  241. }
  242. while (count > 0) {
  243. ssize_t read;
  244. uint8_t buf[16];
  245. int i;
  246. read = fs_read(&file, buf, MIN(count, sizeof(buf)));
  247. if (read <= 0) {
  248. break;
  249. }
  250. shell_fprintf(shell, SHELL_NORMAL, "%08X ", offset);
  251. for (i = 0; i < read; i++) {
  252. shell_fprintf(shell, SHELL_NORMAL, "%02X ", buf[i]);
  253. }
  254. for (; i < sizeof(buf); i++) {
  255. shell_fprintf(shell, SHELL_NORMAL, " ");
  256. }
  257. i = sizeof(buf) - i;
  258. shell_fprintf(shell, SHELL_NORMAL, "%*c", i*3, ' ');
  259. for (i = 0; i < read; i++) {
  260. shell_fprintf(shell, SHELL_NORMAL, "%c", buf[i] < 32 ||
  261. buf[i] > 127 ? '.' : buf[i]);
  262. }
  263. shell_print(shell, "");
  264. offset += read;
  265. count -= read;
  266. }
  267. fs_close(&file);
  268. return 0;
  269. }
  270. static int cmd_cat(const struct shell *shell, size_t argc, char **argv)
  271. {
  272. char path[MAX_PATH_LEN];
  273. uint8_t buf[BUF_CNT];
  274. struct fs_dirent dirent;
  275. struct fs_file_t file;
  276. int err;
  277. ssize_t read;
  278. fs_file_t_init(&file);
  279. for (size_t i = 1; i < argc; ++i) {
  280. create_abs_path(argv[i], path, sizeof(path));
  281. err = fs_stat(path, &dirent);
  282. if (err < 0) {
  283. shell_error(shell, "Failed to obtain file %s (err: %d)",
  284. path, err);
  285. continue;
  286. }
  287. if (dirent.type != FS_DIR_ENTRY_FILE) {
  288. shell_error(shell, "Note a file %s", path);
  289. continue;
  290. }
  291. err = fs_open(&file, path, FS_O_READ);
  292. if (err < 0) {
  293. shell_error(shell, "Failed to open %s (%d)", path, err);
  294. continue;
  295. }
  296. while (true) {
  297. read = fs_read(&file, buf, sizeof(buf));
  298. if (read <= 0) {
  299. break;
  300. }
  301. for (int j = 0; j < read; j++) {
  302. shell_fprintf(shell, SHELL_NORMAL, "%c", buf[j]);
  303. }
  304. }
  305. if (read < 0) {
  306. shell_error(shell, "Failed to read from file %s (err: %d)",
  307. path, read);
  308. }
  309. fs_close(&file);
  310. }
  311. return 0;
  312. }
  313. static int cmd_statvfs(const struct shell *shell, size_t argc, char **argv)
  314. {
  315. int err;
  316. char path[MAX_PATH_LEN];
  317. struct fs_statvfs stat;
  318. create_abs_path(argv[1], path, sizeof(path));
  319. err = fs_statvfs(path, &stat);
  320. if (err < 0) {
  321. shell_error(shell, "Failed to statvfs %s (%d)", path, err);
  322. return -ENOEXEC;
  323. }
  324. shell_fprintf(shell, SHELL_NORMAL,
  325. "bsize %lu, frsize %lu, blocks %lu, bfree %lu\n",
  326. stat.f_bsize, stat.f_frsize, stat.f_blocks, stat.f_bfree);
  327. return 0;
  328. }
  329. static int cmd_write(const struct shell *shell, size_t argc, char **argv)
  330. {
  331. char path[MAX_PATH_LEN];
  332. uint8_t buf[BUF_CNT];
  333. uint8_t buf_len;
  334. int arg_offset;
  335. struct fs_file_t file;
  336. off_t offset = -1;
  337. int err;
  338. create_abs_path(argv[1], path, sizeof(path));
  339. if (!strcmp(argv[2], "-o")) {
  340. if (argc < 4) {
  341. shell_error(shell, "Missing argument");
  342. return -ENOEXEC;
  343. }
  344. offset = strtol(argv[3], NULL, 0);
  345. arg_offset = 4;
  346. } else {
  347. arg_offset = 2;
  348. }
  349. fs_file_t_init(&file);
  350. err = fs_open(&file, path, FS_O_CREATE | FS_O_WRITE);
  351. if (err) {
  352. shell_error(shell, "Failed to open %s (%d)", path, err);
  353. return -ENOEXEC;
  354. }
  355. if (offset < 0) {
  356. err = fs_seek(&file, 0, FS_SEEK_END);
  357. } else {
  358. err = fs_seek(&file, offset, FS_SEEK_SET);
  359. }
  360. if (err) {
  361. shell_error(shell, "Failed to seek %s (%d)", path, err);
  362. fs_close(&file);
  363. return -ENOEXEC;
  364. }
  365. buf_len = 0U;
  366. while (arg_offset < argc) {
  367. buf[buf_len++] = strtol(argv[arg_offset++], NULL, 16);
  368. if ((buf_len == BUF_CNT) || (arg_offset == argc)) {
  369. err = fs_write(&file, buf, buf_len);
  370. if (err < 0) {
  371. shell_error(shell, "Failed to write %s (%d)",
  372. path, err);
  373. fs_close(&file);
  374. return -ENOEXEC;
  375. }
  376. buf_len = 0U;
  377. }
  378. }
  379. fs_close(&file);
  380. return 0;
  381. }
  382. static int cmd_perf(const struct shell *shell, size_t argc, char **argv)
  383. {
  384. char path[MAX_PATH_LEN];
  385. struct fs_file_t file;
  386. int count, repeat, idx;
  387. int err;
  388. u32_t addr, start, cost;
  389. u64_t speed;
  390. create_abs_path(argv[1], path, sizeof(path));
  391. addr = strtol(argv[2], NULL, 16);
  392. count = strtol(argv[3], NULL, 0);
  393. if (argc > 4) {
  394. repeat = strtol(argv[4], NULL, 0);
  395. } else {
  396. repeat = 1;
  397. }
  398. fs_file_t_init(&file);
  399. err = fs_open(&file, path, FS_O_CREATE | FS_O_RDWR);
  400. if (err) {
  401. shell_error(shell, "Failed to open %s (%d)", path, err);
  402. return -ENOEXEC;
  403. }
  404. start = k_cycle_get_32();
  405. for (idx = 0; idx < repeat; idx ++) {
  406. fs_read(&file, (u8_t*)addr, count);
  407. }
  408. cost = k_cyc_to_us_floor32(k_cycle_get_32() - start);
  409. if (cost > 1) {
  410. speed = (u64_t)count*repeat/1024*1000000/cost;
  411. shell_fprintf(shell, SHELL_NORMAL, "%uB %uus (%u KB/s)\n", count, cost, (u32_t)speed);
  412. }
  413. fs_close(&file);
  414. return 0;
  415. }
  416. #if defined(CONFIG_FAT_FILESYSTEM_ELM) \
  417. || defined(CONFIG_FILE_SYSTEM_LITTLEFS)
  418. static char *mntpt_prepare(char *mntpt)
  419. {
  420. char *cpy_mntpt;
  421. cpy_mntpt = k_malloc(strlen(mntpt) + 1);
  422. if (cpy_mntpt) {
  423. strcpy(cpy_mntpt, mntpt);
  424. }
  425. return cpy_mntpt;
  426. }
  427. #endif
  428. #if defined(CONFIG_FAT_FILESYSTEM_ELM)
  429. static int cmd_mount_fat(const struct shell *shell, size_t argc, char **argv)
  430. {
  431. char *mntpt;
  432. int res;
  433. mntpt = mntpt_prepare(argv[1]);
  434. if (!mntpt) {
  435. shell_error(shell,
  436. "Failed to allocate buffer for mount point");
  437. return -ENOEXEC;
  438. }
  439. fatfs_mnt.mnt_point = (const char *)mntpt;
  440. res = fs_mount(&fatfs_mnt);
  441. if (res != 0) {
  442. shell_error(shell,
  443. "Error mounting fat fs.Error Code [%d]", res);
  444. return -ENOEXEC;
  445. }
  446. shell_print(shell, "Successfully mounted fat fs:%s",
  447. fatfs_mnt.mnt_point);
  448. return 0;
  449. }
  450. #endif
  451. #if defined(CONFIG_FILE_SYSTEM_LITTLEFS)
  452. static int cmd_mount_littlefs(const struct shell *shell, size_t argc, char **argv)
  453. {
  454. if (littlefs_mnt.mnt_point != NULL) {
  455. return -EBUSY;
  456. }
  457. char *mntpt = mntpt_prepare(argv[1]);
  458. if (!mntpt) {
  459. shell_error(shell, "Failed to allocate mount point");
  460. return -ENOEXEC; /* ?!? */
  461. }
  462. littlefs_mnt.mnt_point = mntpt;
  463. int rc = fs_mount(&littlefs_mnt);
  464. if (rc != 0) {
  465. shell_error(shell, "Error mounting as littlefs: %d", rc);
  466. return -ENOEXEC;
  467. }
  468. return rc;
  469. }
  470. #endif
  471. #if defined(CONFIG_SD_FS)
  472. static FATFS sdfs_fs;
  473. /* mounting info */
  474. static struct fs_mount_t sdfs_mnt = {
  475. .type = FS_SDFS,
  476. .fs_data = &sdfs_fs,
  477. };
  478. static int cmd_mount_sdfs(const struct shell *shell, size_t argc, char **argv)
  479. {
  480. if (sdfs_mnt.mnt_point != NULL) {
  481. return -EBUSY;
  482. }
  483. char *mntpt = mntpt_prepare(argv[1]);
  484. if (!mntpt) {
  485. shell_error(shell, "Failed to allocate mount point");
  486. return -ENOEXEC; /* ?!? */
  487. }
  488. sdfs_mnt.mnt_point = mntpt;
  489. int rc = fs_mount(&sdfs_mnt);
  490. if (rc != 0) {
  491. shell_error(shell, "Error mounting %u as sdfs: %d", rc);
  492. return -ENOEXEC;
  493. }
  494. return rc;
  495. }
  496. #endif
  497. #if defined(CONFIG_FAT_FILESYSTEM_ELM) \
  498. || defined(CONFIG_FILE_SYSTEM_LITTLEFS)
  499. SHELL_STATIC_SUBCMD_SET_CREATE(sub_fs_mount,
  500. #if defined(CONFIG_FAT_FILESYSTEM_ELM)
  501. SHELL_CMD_ARG(fat, NULL,
  502. "Mount fatfs. fs mount fat <mount-point>",
  503. cmd_mount_fat, 2, 0),
  504. #endif
  505. #if defined(CONFIG_FILE_SYSTEM_LITTLEFS)
  506. SHELL_CMD_ARG(littlefs, NULL,
  507. "Mount littlefs. fs mount littlefs <mount-point>",
  508. cmd_mount_littlefs, 2, 0),
  509. #endif
  510. #if defined(CONFIG_SD_FS)
  511. SHELL_CMD_ARG(sdfs, NULL,
  512. "Mount sdfs. fs mount sdfs <mount-point>",
  513. cmd_mount_sdfs, 2, 0),
  514. #endif
  515. SHELL_SUBCMD_SET_END
  516. );
  517. #endif
  518. SHELL_STATIC_SUBCMD_SET_CREATE(sub_fs,
  519. SHELL_CMD(cd, NULL, "Change working directory", cmd_cd),
  520. SHELL_CMD(ls, NULL, "List files in current directory", cmd_ls),
  521. SHELL_CMD_ARG(mkdir, NULL, "Create directory", cmd_mkdir, 2, 0),
  522. #if defined(CONFIG_FAT_FILESYSTEM_ELM) \
  523. || defined(CONFIG_FILE_SYSTEM_LITTLEFS)
  524. SHELL_CMD(mount, &sub_fs_mount,
  525. "<Mount fs, syntax:- fs mount <fs type> <mount-point>", NULL),
  526. #endif
  527. SHELL_CMD(pwd, NULL, "Print current working directory", cmd_pwd),
  528. SHELL_CMD_ARG(read, NULL, "Read from file", cmd_read, 2, 255),
  529. SHELL_CMD_ARG(cat, NULL,
  530. "Concatenate files and print on the standard output",
  531. cmd_cat, 2, 255),
  532. SHELL_CMD_ARG(rm, NULL, "Remove file", cmd_rm, 2, 0),
  533. SHELL_CMD_ARG(statvfs, NULL, "Show file system state", cmd_statvfs, 2, 0),
  534. SHELL_CMD_ARG(trunc, NULL, "Truncate file", cmd_trunc, 2, 255),
  535. SHELL_CMD_ARG(write, NULL, "Write file", cmd_write, 3, 255),
  536. SHELL_CMD_ARG(perf, NULL, "Perf test: fs perf file buf count", cmd_perf, 4, 255),
  537. SHELL_SUBCMD_SET_END
  538. );
  539. SHELL_CMD_REGISTER(fs, &sub_fs, "File system commands", NULL);