/* * Copyright (C) 2016 Eistec AB * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup sys_shell_commands * @{ * * @file * @brief Shell commands for the VFS module * * @author Joakim Nohlgård * * @} */ #if MODULE_VFS #include #include #include #include #include #include #include #include #include #include "vfs.h" #if MODULE_VFS_UTIL #include "vfs_util.h" #endif #define SHELL_VFS_BUFSIZE 256 static uint8_t _shell_vfs_data_buffer[SHELL_VFS_BUFSIZE]; /** * @brief Auto-Mount array */ XFA_USE_CONST(vfs_mount_t, vfs_mountpoints_xfa); /** * @brief Number of automatic mountpoints */ #define MOUNTPOINTS_NUMOF XFA_LEN(vfs_mount_t, vfs_mountpoints_xfa) static void _ls_usage(char **argv) { printf("%s \n", argv[0]); puts("list files in "); } static void _vfs_usage(char **argv) { printf("%s r [bytes] [offset]\n", argv[0]); printf("%s w \n", argv[0]); printf("%s ls \n", argv[0]); printf("%s cp \n", argv[0]); printf("%s mv \n", argv[0]); printf("%s rm \n", argv[0]); printf("%s df [path]\n", argv[0]); if (MOUNTPOINTS_NUMOF > 0) { printf("%s mount [path]\n", argv[0]); } puts("r: Read [bytes] bytes at [offset] in file "); puts("w: Write (: append, overwrite) or string in file "); puts("ls: list files in "); puts("mv: Move file to "); puts("cp: Copy file to "); puts("rm: Unlink (delete) "); puts("df: show file system space utilization stats"); } /* Macro used by _errno_string to expand errno labels to string and print it */ #define _case_snprintf_errno_name(x) \ case x: \ res = snprintf(buf, buflen, #x); \ break static int _errno_string(int err, char *buf, size_t buflen) { int len = 0; int res; if (err < 0) { res = snprintf(buf, buflen, "-"); if (res < 0) { return res; } if ((size_t)res <= buflen) { buf += res; buflen -= res; } len += res; err = -err; } switch (err) { _case_snprintf_errno_name(EACCES); _case_snprintf_errno_name(ENOENT); _case_snprintf_errno_name(EINVAL); _case_snprintf_errno_name(EFAULT); _case_snprintf_errno_name(EROFS); _case_snprintf_errno_name(EIO); _case_snprintf_errno_name(ENAMETOOLONG); _case_snprintf_errno_name(EPERM); default: res = snprintf(buf, buflen, "%d", err); break; } if (res < 0) { return res; } len += res; return len; } #undef _case_snprintf_errno_name static void _print_df(vfs_DIR *dir) { struct statvfs buf; int res = vfs_dstatvfs(dir, &buf); printf("%-16s ", dir->mp->mount_point); if (res < 0) { char err[16]; _errno_string(res, err, sizeof(err)); printf("statvfs failed: %s\n", err); return; } printf("%12lu %12lu %12lu %7lu%%\n", (unsigned long)buf.f_blocks, (unsigned long)(buf.f_blocks - buf.f_bfree), (unsigned long)buf.f_bavail, (unsigned long)(((buf.f_blocks - buf.f_bfree) * 100) / buf.f_blocks)); } static int _df_handler(int argc, char **argv) { puts("Mountpoint Total Used Available Capacity"); if (argc > 1) { const char *path = argv[1]; /* Opening a directory just to statfs is somewhat odd, but it is the * easiest to support with a single _print_df function */ vfs_DIR dir; int res = vfs_opendir(&dir, path); if (res == 0) { _print_df(&dir); vfs_closedir(&dir); } else { char err[16]; _errno_string(res, err, sizeof(err)); printf("Failed to open `%s`: %s\n", path, err); } } else { /* Iterate through all mount points */ vfs_DIR it = { 0 }; while (vfs_iterate_mount_dirs(&it)) { _print_df(&it); } } return 0; } static int _mount_handler(int argc, char **argv) { if (argc < 2) { printf("usage: %s [path]\n", argv[0]); puts("mount pre-configured mount point"); return -1; } int res = vfs_mount_by_path(argv[1]); puts(strerror(res)); return res; } static int _read_handler(int argc, char **argv) { uint8_t buf[16]; size_t nbytes = sizeof(buf); off_t offset = 0; char *path = argv[1]; if (argc < 2) { puts("vfs read: missing file name"); return 1; } if (argc > 2) { nbytes = atoi(argv[2]); } if (argc > 3) { offset = atoi(argv[3]); } int res; res = vfs_normalize_path(path, path, strlen(path) + 1); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("Invalid path \"%s\": %s\n", path, buf); return 5; } int fd = vfs_open(path, O_RDONLY, 0); if (fd < 0) { _errno_string(fd, (char *)buf, sizeof(buf)); printf("Error opening file \"%s\": %s\n", path, buf); return 3; } res = vfs_lseek(fd, offset, SEEK_SET); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("Seek error: %s\n", buf); vfs_close(fd); return 4; } while (nbytes > 0) { memset(buf, 0, sizeof(buf)); size_t line_len = (nbytes < sizeof(buf) ? nbytes : sizeof(buf)); res = vfs_read(fd, buf, line_len); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("Read error: %s\n", buf); vfs_close(fd); return 5; } else if ((size_t)res > line_len) { printf("BUFFER OVERRUN! %d > %lu\n", res, (unsigned long)line_len); vfs_close(fd); return 6; } else if (res == 0) { /* EOF */ printf("-- EOF --\n"); break; } printf("%08lx:", (unsigned long)offset); for (int k = 0; k < res; ++k) { if ((k % 2) == 0) { putchar(' '); } printf("%02x", buf[k]); } for (unsigned k = res; k < sizeof(buf); ++k) { if ((k % 2) == 0) { putchar(' '); } putchar(' '); putchar(' '); } putchar(' '); putchar(' '); for (int k = 0; k < res; ++k) { if (isprint(buf[k])) { putchar(buf[k]); } else { putchar('.'); } } puts(""); offset += res; nbytes -= res; } vfs_close(fd); return 0; } static inline int _dehex(char c) { if ('0' <= c && c <= '9') { return c - '0'; } else if ('A' <= c && c <= 'F') { return c - 'A' + 10; } else if ('a' <= c && c <= 'f') { return c - 'a' + 10; } else { return 0; } } static int _write_handler(int argc, char **argv) { char *w_buf; char buf[16]; size_t nbytes = 0; size_t nb_str = 0; char *path = argv[1]; int ascii = 0; int flag = O_CREAT; if (argc < 2) { puts("vfs write: missing file name"); return 1; } if (argc < 3) { puts("vfs write: missing format"); return 1; } if (strcmp(argv[2], "ascii") == 0) { ascii = 1; } else if (strcmp(argv[2], "hex") == 0) { ascii = 0; } else { printf("vfs write: unknown format: %s\n", argv[2]); return 1; } if (argc < 4) { puts("vfs write: missing flag"); return 1; } if (strcmp(argv[3], "a") == 0) { flag |= O_WRONLY | O_APPEND; } else if (strcmp(argv[3], "o") == 0) { flag |= O_WRONLY; } else { printf("vfs write: invalid flag %s\n", argv[3]); return 1; } if (argc < 5) { puts("vfs write: missing data"); return 1; } w_buf = argv[4]; nbytes = strlen(w_buf); /* in hex string mode, bytes may be separated by spaces */ /* in ascii mode, there could be spaces */ /* we need the total number of strings to go through */ nb_str = argc - 4; if (!ascii) { /* sanity check: only hex digit and hex strings length must be even */ for (size_t i = 0; i < nb_str; i++) { char c; size_t j = 0; do { c = argv[argc - nb_str + i][j]; j++; if (c != '\0' && !isxdigit((int)c)) { printf("Non-hex character: %c\n", c); return 6; } } while (c != '\0'); j--; if (j % 2 != 0) { puts("Invalid string length"); return 6; } } } int res; res = vfs_normalize_path(path, path, strlen(path) + 1); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("Invalid path \"%s\": %s\n", path, buf); return 5; } int fd = vfs_open(path, flag, 0); if (fd < 0) { _errno_string(fd, (char *)buf, sizeof(buf)); printf("Error opening file \"%s\": %s\n", path, buf); return 3; } if (ascii) { while (nb_str > 0) { res = vfs_write(fd, w_buf, nbytes); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("Write error: %s\n", buf); vfs_close(fd); return 4; } nb_str--; if (nb_str) { vfs_write(fd, " ", 1); w_buf = argv[argc - nb_str]; } } } else { while (nb_str > 0) { w_buf = argv[argc - nb_str]; nbytes = strlen(w_buf); while (nbytes > 0) { uint8_t byte = _dehex(*w_buf) << 4 | _dehex(*(w_buf + 1)); res = vfs_write(fd, &byte, 1); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("Write error: %s\n", buf); vfs_close(fd); return 4; } w_buf += 2; nbytes -= 2; } nb_str--; } } vfs_close(fd); return 0; } static int _cp_handler(int argc, char **argv) { char errbuf[16]; if (argc < 3) { _vfs_usage(argv); return 1; } char *src_name = argv[1]; char *dest_name = argv[2]; printf("%s: copy src: %s dest: %s\n", argv[0], src_name, dest_name); int fd_in = vfs_open(src_name, O_RDONLY, 0); if (fd_in < 0) { _errno_string(fd_in, (char *)errbuf, sizeof(errbuf)); printf("Error opening file for reading \"%s\": %s\n", src_name, errbuf); return 2; } int fd_out = vfs_open(dest_name, O_WRONLY | O_TRUNC | O_CREAT, S_IRWXU | S_IRWXG | S_IRWXO); if (fd_out < 0) { _errno_string(fd_out, (char *)errbuf, sizeof(errbuf)); printf("Error opening file for writing \"%s\": %s\n", dest_name, errbuf); return 2; } int eof = 0; while (eof == 0) { size_t bufspace = sizeof(_shell_vfs_data_buffer); size_t pos = 0; while (bufspace > 0) { int res = vfs_read(fd_in, &_shell_vfs_data_buffer[pos], bufspace); if (res < 0) { _errno_string(res, (char *)errbuf, sizeof(errbuf)); printf("Error reading %lu bytes @ 0x%lx in \"%s\" (%d): %s\n", (unsigned long)bufspace, (unsigned long)pos, src_name, fd_in, errbuf); vfs_close(fd_in); vfs_close(fd_out); return 2; } if (res == 0) { /* EOF */ eof = 1; break; } if (((unsigned)res) > bufspace) { printf("READ BUFFER OVERRUN! %d > %lu\n", res, (unsigned long)bufspace); vfs_close(fd_in); vfs_close(fd_out); return 3; } pos += res; bufspace -= res; } bufspace = pos; pos = 0; while (bufspace > 0) { int res = vfs_write(fd_out, &_shell_vfs_data_buffer[pos], bufspace); if (res <= 0) { _errno_string(res, (char *)errbuf, sizeof(errbuf)); printf("Error writing %lu bytes @ 0x%lx in \"%s\" (%d): %s\n", (unsigned long)bufspace, (unsigned long)pos, dest_name, fd_out, errbuf); vfs_close(fd_in); vfs_close(fd_out); return 4; } if (((unsigned)res) > bufspace) { printf("WRITE BUFFER OVERRUN! %d > %lu\n", res, (unsigned long)bufspace); vfs_close(fd_in); vfs_close(fd_out); return 5; } bufspace -= res; } } printf("Copied: %s -> %s\n", src_name, dest_name); vfs_close(fd_in); vfs_close(fd_out); return 0; } static int _mv_handler(int argc, char **argv) { if (argc < 3) { _vfs_usage(argv); return 1; } char *src_name = argv[1]; char *dest_name = argv[2]; printf("%s: move src: %s dest: %s\n", argv[0], src_name, dest_name); int res = vfs_rename(src_name, dest_name); if (res < 0) { char errbuf[16]; _errno_string(res, (char *)errbuf, sizeof(errbuf)); printf("mv ERR: %s\n", errbuf); return 2; } return 0; } static int _rm_handler(int argc, char **argv) { if (argc < 2) { _vfs_usage(argv); return 1; } char *rm_name = argv[1]; printf("%s: unlink: %s\n", argv[0], rm_name); int res = vfs_unlink(rm_name); if (res < 0) { char errbuf[16]; _errno_string(res, (char *)errbuf, sizeof(errbuf)); printf("rm ERR: %s\n", errbuf); return 2; } return 0; } int _ls_handler(int argc, char **argv) { if (argc < 2) { _ls_usage(argv); return 1; } char *path = argv[1]; uint8_t buf[16]; int res; int ret = 0; res = vfs_normalize_path(path, path, strlen(path) + 1); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("Invalid path \"%s\": %s\n", path, buf); return 5; } vfs_DIR dir; res = vfs_opendir(&dir, path); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("vfs_opendir error: %s\n", buf); return 1; } unsigned int nfiles = 0; while (1) { char path_name[2 * (VFS_NAME_MAX + 1)]; vfs_dirent_t entry; struct stat stat; res = vfs_readdir(&dir, &entry); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("vfs_readdir error: %s\n", buf); if (res == -EAGAIN) { /* try again */ continue; } ret = 2; break; } if (res == 0) { /* end of stream */ break; } snprintf(path_name, sizeof(path_name), "%s/%s", path, entry.d_name); vfs_stat(path_name, &stat); if (stat.st_mode & S_IFDIR) { printf("%s/\n", entry.d_name); } else if (stat.st_mode & S_IFREG) { printf("%s\t%lu B\n", entry.d_name, stat.st_size); ++nfiles; } else { printf("%s\n", entry.d_name); } } if (ret == 0) { printf("total %u files\n", nfiles); } res = vfs_closedir(&dir); if (res < 0) { _errno_string(res, (char *)buf, sizeof(buf)); printf("vfs_closedir error: %s\n", buf); return 2; } return ret; } int _vfs_handler(int argc, char **argv) { if (argc < 2) { _vfs_usage(argv); return 1; } if (strcmp(argv[1], "r") == 0) { /* pass on to read handler, shifting the arguments by one */ return _read_handler(argc - 1, &argv[1]); } else if (strcmp(argv[1], "w") == 0) { return _write_handler(argc - 1, &argv[1]); } else if (strcmp(argv[1], "ls") == 0) { return _ls_handler(argc - 1, &argv[1]); } else if (strcmp(argv[1], "cp") == 0) { return _cp_handler(argc - 1, &argv[1]); } else if (strcmp(argv[1], "mv") == 0) { return _mv_handler(argc - 1, &argv[1]); } else if (strcmp(argv[1], "rm") == 0) { return _rm_handler(argc - 1, &argv[1]); } else if (strcmp(argv[1], "df") == 0) { return _df_handler(argc - 1, &argv[1]); } else if (MOUNTPOINTS_NUMOF > 0 && strcmp(argv[1], "mount") == 0) { return _mount_handler(argc - 1, &argv[1]); } else { printf("vfs: unsupported sub-command \"%s\"\n", argv[1]); return 1; } } static inline void _print_digest(const uint8_t *digest, size_t len, const char *file) { for (unsigned i = 0; i < len; ++i) { printf("%02x", digest[i]); } printf(" %s\n", file); } #if MODULE_MD5SUM #include "hashes/md5.h" int _vfs_md5sum_cmd(int argc, char **argv) { int res; uint8_t digest[MD5_DIGEST_LENGTH]; if (argc < 2) { printf("usage: %s [file] …\n", argv[0]); return -1; } for (int i = 1; i < argc; ++i) { const char *file = argv[i]; res = vfs_file_md5(file, digest, _shell_vfs_data_buffer, sizeof(_shell_vfs_data_buffer)); if (res < 0) { printf("%s: error %d\n", file, res); } else { _print_digest(digest, sizeof(digest), file); } } return 0; } #endif #if MODULE_SHA1SUM #include "hashes/sha1.h" int _vfs_sha1sum_cmd(int argc, char **argv) { int res; uint8_t digest[SHA1_DIGEST_LENGTH]; if (argc < 2) { printf("usage: %s [file] …\n", argv[0]); return -1; } for (int i = 1; i < argc; ++i) { const char *file = argv[i]; res = vfs_file_sha1(file, digest, _shell_vfs_data_buffer, sizeof(_shell_vfs_data_buffer)); if (res < 0) { printf("%s: error %d\n", file, res); } else { _print_digest(digest, sizeof(digest), file); } } return 0; } #endif #endif