123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668 |
- /*
- * Copyright (c) 2016 Intel Corporation
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #include <errno.h>
- #include <stdio.h>
- #include <string.h>
- #include <shell/shell.h>
- #include <init.h>
- #include <fs/fs.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <inttypes.h>
- #include <limits.h>
- /* FAT */
- #ifdef CONFIG_FAT_FILESYSTEM_ELM
- #include <ff.h>
- #define FATFS_MNTP "/RAM:"
- /* FatFs work area */
- FATFS fat_fs;
- /* mounting info */
- static struct fs_mount_t fatfs_mnt = {
- .type = FS_FATFS,
- .fs_data = &fat_fs,
- };
- #endif
- /* LITTLEFS */
- #ifdef CONFIG_FILE_SYSTEM_LITTLEFS
- #include <fs/littlefs.h>
- #include <storage/flash_map.h>
- FS_LITTLEFS_DECLARE_DEFAULT_CONFIG(lfs_data);
- static struct fs_mount_t littlefs_mnt = {
- .type = FS_LITTLEFS,
- .fs_data = &lfs_data,
- .storage_dev = (void *)FLASH_AREA_ID(storage),
- };
- #endif
- #define BUF_CNT 64
- #define MAX_PATH_LEN 128
- #define MAX_FILENAME_LEN 128
- #define MAX_INPUT_LEN 20
- #define SHELL_FS "fs"
- /* Maintenance guarantees this begins with '/' and is NUL-terminated. */
- static char cwd[MAX_PATH_LEN] = "/";
- static void create_abs_path(const char *name, char *path, size_t len)
- {
- if (name[0] == '/') {
- strncpy(path, name, len);
- path[len - 1] = '\0';
- } else {
- if (cwd[1] == '\0') {
- __ASSERT_NO_MSG(len >= 2);
- *path++ = '/';
- --len;
- strncpy(path, name, len);
- path[len - 1] = '\0';
- } else {
- strncpy(path, cwd, len);
- path[len - 1] = '\0';
- size_t plen = strlen(path);
- if (plen < len) {
- path += plen;
- *path++ = '/';
- len -= plen + 1U;
- strncpy(path, name, len);
- path[len - 1] = '\0';
- }
- }
- }
- }
- static int cmd_cd(const struct shell *shell, size_t argc, char **argv)
- {
- char path[MAX_PATH_LEN];
- struct fs_dirent entry;
- int err;
- if (argc < 2) {
- strcpy(cwd, "/");
- return 0;
- }
- if (strcmp(argv[1], "..") == 0) {
- char *prev = strrchr(cwd, '/');
- if (!prev || prev == cwd) {
- strcpy(cwd, "/");
- } else {
- *prev = '\0';
- }
- /* No need to test that a parent exists */
- return 0;
- }
- create_abs_path(argv[1], path, sizeof(path));
- err = fs_stat(path, &entry);
- if (err) {
- shell_error(shell, "%s doesn't exist", path);
- return -ENOEXEC;
- }
- if (entry.type != FS_DIR_ENTRY_DIR) {
- shell_error(shell, "%s is not a directory", path);
- return -ENOEXEC;
- }
- strncpy(cwd, path, sizeof(cwd));
- cwd[sizeof(cwd) - 1] = '\0';
- return 0;
- }
- static int cmd_ls(const struct shell *shell, size_t argc, char **argv)
- {
- char path[MAX_PATH_LEN];
- struct fs_dir_t dir;
- int err;
- if (argc < 2) {
- strncpy(path, cwd, sizeof(path));
- path[sizeof(path) - 1] = '\0';
- } else {
- create_abs_path(argv[1], path, sizeof(path));
- }
- fs_dir_t_init(&dir);
- err = fs_opendir(&dir, path);
- if (err) {
- shell_error(shell, "Unable to open %s (err %d)", path, err);
- return -ENOEXEC;
- }
- while (1) {
- struct fs_dirent entry;
- err = fs_readdir(&dir, &entry);
- if (err) {
- shell_error(shell, "Unable to read directory");
- break;
- }
- /* Check for end of directory listing */
- if (entry.name[0] == '\0') {
- break;
- }
- shell_print(shell, "%s%s", entry.name,
- (entry.type == FS_DIR_ENTRY_DIR) ? "/" : "");
- }
- fs_closedir(&dir);
- return 0;
- }
- static int cmd_pwd(const struct shell *shell, size_t argc, char **argv)
- {
- shell_print(shell, "%s", cwd);
- return 0;
- }
- static int cmd_trunc(const struct shell *shell, size_t argc, char **argv)
- {
- char path[MAX_PATH_LEN];
- struct fs_file_t file;
- int length;
- int err;
- create_abs_path(argv[1], path, sizeof(path));
- if (argc > 2) {
- length = strtol(argv[2], NULL, 0);
- } else {
- length = 0;
- }
- fs_file_t_init(&file);
- err = fs_open(&file, path, FS_O_WRITE);
- if (err) {
- shell_error(shell, "Failed to open %s (%d)", path, err);
- return -ENOEXEC;;
- }
- err = fs_truncate(&file, length);
- if (err) {
- shell_error(shell, "Failed to truncate %s (%d)", path, err);
- err = -ENOEXEC;
- }
- fs_close(&file);
- return err;
- }
- static int cmd_mkdir(const struct shell *shell, size_t argc, char **argv)
- {
- int err;
- char path[MAX_PATH_LEN];
- create_abs_path(argv[1], path, sizeof(path));
- err = fs_mkdir(path);
- if (err) {
- shell_error(shell, "Error creating dir[%d]", err);
- err = -ENOEXEC;
- }
- return err;
- }
- static int cmd_rm(const struct shell *shell, size_t argc, char **argv)
- {
- int err;
- char path[MAX_PATH_LEN];
- create_abs_path(argv[1], path, sizeof(path));
- err = fs_unlink(path);
- if (err) {
- shell_error(shell, "Failed to remove %s (%d)", path, err);
- err = -ENOEXEC;
- }
- return err;
- }
- static int cmd_read(const struct shell *shell, size_t argc, char **argv)
- {
- char path[MAX_PATH_LEN];
- struct fs_dirent dirent;
- struct fs_file_t file;
- int count;
- off_t offset;
- int err;
- create_abs_path(argv[1], path, sizeof(path));
- if (argc > 2) {
- count = strtol(argv[2], NULL, 0);
- if (count <= 0) {
- count = INT_MAX;
- }
- } else {
- count = INT_MAX;
- }
- if (argc > 3) {
- offset = strtol(argv[3], NULL, 0);
- } else {
- offset = 0;
- }
- err = fs_stat(path, &dirent);
- if (err) {
- shell_error(shell, "Failed to obtain file %s (err: %d)",
- path, err);
- return -ENOEXEC;
- }
- if (dirent.type != FS_DIR_ENTRY_FILE) {
- shell_error(shell, "Note a file %s", path);
- return -ENOEXEC;
- }
- shell_print(shell, "File size: %zd", dirent.size);
- fs_file_t_init(&file);
- err = fs_open(&file, path, FS_O_READ);
- if (err) {
- shell_error(shell, "Failed to open %s (%d)", path, err);
- return -ENOEXEC;
- }
- if (offset > 0) {
- err = fs_seek(&file, offset, FS_SEEK_SET);
- if (err) {
- shell_error(shell, "Failed to seek %s (%d)",
- path, err);
- fs_close(&file);
- return -ENOEXEC;
- }
- }
- while (count > 0) {
- ssize_t read;
- uint8_t buf[16];
- int i;
- read = fs_read(&file, buf, MIN(count, sizeof(buf)));
- if (read <= 0) {
- break;
- }
- shell_fprintf(shell, SHELL_NORMAL, "%08X ", offset);
- for (i = 0; i < read; i++) {
- shell_fprintf(shell, SHELL_NORMAL, "%02X ", buf[i]);
- }
- for (; i < sizeof(buf); i++) {
- shell_fprintf(shell, SHELL_NORMAL, " ");
- }
- i = sizeof(buf) - i;
- shell_fprintf(shell, SHELL_NORMAL, "%*c", i*3, ' ');
- for (i = 0; i < read; i++) {
- shell_fprintf(shell, SHELL_NORMAL, "%c", buf[i] < 32 ||
- buf[i] > 127 ? '.' : buf[i]);
- }
- shell_print(shell, "");
- offset += read;
- count -= read;
- }
- fs_close(&file);
- return 0;
- }
- static int cmd_cat(const struct shell *shell, size_t argc, char **argv)
- {
- char path[MAX_PATH_LEN];
- uint8_t buf[BUF_CNT];
- struct fs_dirent dirent;
- struct fs_file_t file;
- int err;
- ssize_t read;
- fs_file_t_init(&file);
- for (size_t i = 1; i < argc; ++i) {
- create_abs_path(argv[i], path, sizeof(path));
- err = fs_stat(path, &dirent);
- if (err < 0) {
- shell_error(shell, "Failed to obtain file %s (err: %d)",
- path, err);
- continue;
- }
- if (dirent.type != FS_DIR_ENTRY_FILE) {
- shell_error(shell, "Note a file %s", path);
- continue;
- }
- err = fs_open(&file, path, FS_O_READ);
- if (err < 0) {
- shell_error(shell, "Failed to open %s (%d)", path, err);
- continue;
- }
- while (true) {
- read = fs_read(&file, buf, sizeof(buf));
- if (read <= 0) {
- break;
- }
- for (int j = 0; j < read; j++) {
- shell_fprintf(shell, SHELL_NORMAL, "%c", buf[j]);
- }
- }
- if (read < 0) {
- shell_error(shell, "Failed to read from file %s (err: %d)",
- path, read);
- }
- fs_close(&file);
- }
- return 0;
- }
- static int cmd_statvfs(const struct shell *shell, size_t argc, char **argv)
- {
- int err;
- char path[MAX_PATH_LEN];
- struct fs_statvfs stat;
- create_abs_path(argv[1], path, sizeof(path));
- err = fs_statvfs(path, &stat);
- if (err < 0) {
- shell_error(shell, "Failed to statvfs %s (%d)", path, err);
- return -ENOEXEC;
- }
- shell_fprintf(shell, SHELL_NORMAL,
- "bsize %lu, frsize %lu, blocks %lu, bfree %lu\n",
- stat.f_bsize, stat.f_frsize, stat.f_blocks, stat.f_bfree);
- return 0;
- }
- static int cmd_write(const struct shell *shell, size_t argc, char **argv)
- {
- char path[MAX_PATH_LEN];
- uint8_t buf[BUF_CNT];
- uint8_t buf_len;
- int arg_offset;
- struct fs_file_t file;
- off_t offset = -1;
- int err;
- create_abs_path(argv[1], path, sizeof(path));
- if (!strcmp(argv[2], "-o")) {
- if (argc < 4) {
- shell_error(shell, "Missing argument");
- return -ENOEXEC;
- }
- offset = strtol(argv[3], NULL, 0);
- arg_offset = 4;
- } else {
- arg_offset = 2;
- }
- fs_file_t_init(&file);
- err = fs_open(&file, path, FS_O_CREATE | FS_O_WRITE);
- if (err) {
- shell_error(shell, "Failed to open %s (%d)", path, err);
- return -ENOEXEC;
- }
- if (offset < 0) {
- err = fs_seek(&file, 0, FS_SEEK_END);
- } else {
- err = fs_seek(&file, offset, FS_SEEK_SET);
- }
- if (err) {
- shell_error(shell, "Failed to seek %s (%d)", path, err);
- fs_close(&file);
- return -ENOEXEC;
- }
- buf_len = 0U;
- while (arg_offset < argc) {
- buf[buf_len++] = strtol(argv[arg_offset++], NULL, 16);
- if ((buf_len == BUF_CNT) || (arg_offset == argc)) {
- err = fs_write(&file, buf, buf_len);
- if (err < 0) {
- shell_error(shell, "Failed to write %s (%d)",
- path, err);
- fs_close(&file);
- return -ENOEXEC;
- }
- buf_len = 0U;
- }
- }
- fs_close(&file);
- return 0;
- }
- static int cmd_perf(const struct shell *shell, size_t argc, char **argv)
- {
- char path[MAX_PATH_LEN];
- struct fs_file_t file;
- int count, repeat, idx;
- int err;
- u32_t addr, start, cost;
- u64_t speed;
- create_abs_path(argv[1], path, sizeof(path));
- addr = strtol(argv[2], NULL, 16);
- count = strtol(argv[3], NULL, 0);
- if (argc > 4) {
- repeat = strtol(argv[4], NULL, 0);
- } else {
- repeat = 1;
- }
- fs_file_t_init(&file);
- err = fs_open(&file, path, FS_O_CREATE | FS_O_RDWR);
- if (err) {
- shell_error(shell, "Failed to open %s (%d)", path, err);
- return -ENOEXEC;
- }
- start = k_cycle_get_32();
- for (idx = 0; idx < repeat; idx ++) {
- fs_read(&file, (u8_t*)addr, count);
- }
- cost = k_cyc_to_us_floor32(k_cycle_get_32() - start);
- if (cost > 1) {
- speed = (u64_t)count*repeat/1024*1000000/cost;
- shell_fprintf(shell, SHELL_NORMAL, "%uB %uus (%u KB/s)\n", count, cost, (u32_t)speed);
- }
- fs_close(&file);
- return 0;
- }
- #if defined(CONFIG_FAT_FILESYSTEM_ELM) \
- || defined(CONFIG_FILE_SYSTEM_LITTLEFS)
- static char *mntpt_prepare(char *mntpt)
- {
- char *cpy_mntpt;
- cpy_mntpt = k_malloc(strlen(mntpt) + 1);
- if (cpy_mntpt) {
- strcpy(cpy_mntpt, mntpt);
- }
- return cpy_mntpt;
- }
- #endif
- #if defined(CONFIG_FAT_FILESYSTEM_ELM)
- static int cmd_mount_fat(const struct shell *shell, size_t argc, char **argv)
- {
- char *mntpt;
- int res;
- mntpt = mntpt_prepare(argv[1]);
- if (!mntpt) {
- shell_error(shell,
- "Failed to allocate buffer for mount point");
- return -ENOEXEC;
- }
- fatfs_mnt.mnt_point = (const char *)mntpt;
- res = fs_mount(&fatfs_mnt);
- if (res != 0) {
- shell_error(shell,
- "Error mounting fat fs.Error Code [%d]", res);
- return -ENOEXEC;
- }
- shell_print(shell, "Successfully mounted fat fs:%s",
- fatfs_mnt.mnt_point);
- return 0;
- }
- #endif
- #if defined(CONFIG_FILE_SYSTEM_LITTLEFS)
- static int cmd_mount_littlefs(const struct shell *shell, size_t argc, char **argv)
- {
- if (littlefs_mnt.mnt_point != NULL) {
- return -EBUSY;
- }
- char *mntpt = mntpt_prepare(argv[1]);
- if (!mntpt) {
- shell_error(shell, "Failed to allocate mount point");
- return -ENOEXEC; /* ?!? */
- }
- littlefs_mnt.mnt_point = mntpt;
- int rc = fs_mount(&littlefs_mnt);
- if (rc != 0) {
- shell_error(shell, "Error mounting as littlefs: %d", rc);
- return -ENOEXEC;
- }
- return rc;
- }
- #endif
- #if defined(CONFIG_SD_FS)
- static FATFS sdfs_fs;
- /* mounting info */
- static struct fs_mount_t sdfs_mnt = {
- .type = FS_SDFS,
- .fs_data = &sdfs_fs,
- };
- static int cmd_mount_sdfs(const struct shell *shell, size_t argc, char **argv)
- {
- if (sdfs_mnt.mnt_point != NULL) {
- return -EBUSY;
- }
- char *mntpt = mntpt_prepare(argv[1]);
- if (!mntpt) {
- shell_error(shell, "Failed to allocate mount point");
- return -ENOEXEC; /* ?!? */
- }
- sdfs_mnt.mnt_point = mntpt;
- int rc = fs_mount(&sdfs_mnt);
- if (rc != 0) {
- shell_error(shell, "Error mounting %u as sdfs: %d", rc);
- return -ENOEXEC;
- }
- return rc;
- }
- #endif
- #if defined(CONFIG_FAT_FILESYSTEM_ELM) \
- || defined(CONFIG_FILE_SYSTEM_LITTLEFS)
- SHELL_STATIC_SUBCMD_SET_CREATE(sub_fs_mount,
- #if defined(CONFIG_FAT_FILESYSTEM_ELM)
- SHELL_CMD_ARG(fat, NULL,
- "Mount fatfs. fs mount fat <mount-point>",
- cmd_mount_fat, 2, 0),
- #endif
- #if defined(CONFIG_FILE_SYSTEM_LITTLEFS)
- SHELL_CMD_ARG(littlefs, NULL,
- "Mount littlefs. fs mount littlefs <mount-point>",
- cmd_mount_littlefs, 2, 0),
- #endif
- #if defined(CONFIG_SD_FS)
- SHELL_CMD_ARG(sdfs, NULL,
- "Mount sdfs. fs mount sdfs <mount-point>",
- cmd_mount_sdfs, 2, 0),
- #endif
- SHELL_SUBCMD_SET_END
- );
- #endif
- SHELL_STATIC_SUBCMD_SET_CREATE(sub_fs,
- SHELL_CMD(cd, NULL, "Change working directory", cmd_cd),
- SHELL_CMD(ls, NULL, "List files in current directory", cmd_ls),
- SHELL_CMD_ARG(mkdir, NULL, "Create directory", cmd_mkdir, 2, 0),
- #if defined(CONFIG_FAT_FILESYSTEM_ELM) \
- || defined(CONFIG_FILE_SYSTEM_LITTLEFS)
- SHELL_CMD(mount, &sub_fs_mount,
- "<Mount fs, syntax:- fs mount <fs type> <mount-point>", NULL),
- #endif
- SHELL_CMD(pwd, NULL, "Print current working directory", cmd_pwd),
- SHELL_CMD_ARG(read, NULL, "Read from file", cmd_read, 2, 255),
- SHELL_CMD_ARG(cat, NULL,
- "Concatenate files and print on the standard output",
- cmd_cat, 2, 255),
- SHELL_CMD_ARG(rm, NULL, "Remove file", cmd_rm, 2, 0),
- SHELL_CMD_ARG(statvfs, NULL, "Show file system state", cmd_statvfs, 2, 0),
- SHELL_CMD_ARG(trunc, NULL, "Truncate file", cmd_trunc, 2, 255),
- SHELL_CMD_ARG(write, NULL, "Write file", cmd_write, 3, 255),
- SHELL_CMD_ARG(perf, NULL, "Perf test: fs perf file buf count", cmd_perf, 4, 255),
- SHELL_SUBCMD_SET_END
- );
- SHELL_CMD_REGISTER(fs, &sub_fs, "File system commands", NULL);
|