/**************************************************************************** * apps/nshlib/nsh_mntcmds.c * * Copyright (C) 2007-2009, 2011-2013 Gregory Nutt. All rights reserved. * Author: Gregory Nutt * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name NuttX nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "nsh.h" #include "nsh_console.h" /**************************************************************************** * Definitions ****************************************************************************/ /**************************************************************************** * Private Types ****************************************************************************/ /**************************************************************************** * Private Function Prototypes ****************************************************************************/ /**************************************************************************** * Private Data ****************************************************************************/ /**************************************************************************** * Public Data ****************************************************************************/ /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: get_fstype ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_MOUNT) #if !defined(CONFIG_BUILD_PROTECTED) && !defined(CONFIG_BUILD_KERNEL) static const char* get_fstype(FAR struct statfs *statbuf) { FAR const char *fstype; /* Get the file system type */ switch (statbuf->f_type) { #ifdef CONFIG_FS_FAT case MSDOS_SUPER_MAGIC: fstype = "vfat"; break; #endif #ifdef CONFIG_FS_ROMFS case ROMFS_MAGIC: fstype = "romfs"; break; #endif #ifdef CONFIG_FS_BINFS case BINFS_MAGIC: fstype = "binfs"; break; #endif #ifdef CONFIG_FS_NXFFS case NXFFS_MAGIC: fstype = "nxffs"; break; #endif #ifdef CONFIG_NFS case NFS_SUPER_MAGIC: fstype = "nfs"; break; #endif #ifdef CONFIG_FS_SMARTFS case SMARTFS_MAGIC: fstype = "smartfs"; break; #endif #ifdef CONFIG_FS_PROCFS case PROCFS_MAGIC: fstype = "procfs"; break; #endif #ifdef CONFIG_FS_UNIONFS case UNIONFS_MAGIC: fstype = "unionfs"; break; #endif default: fstype = "Unrecognized"; break; } return fstype; } #endif #endif /**************************************************************************** * Name: df_handler ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_DF) static int df_handler(FAR const char *mountpoint, FAR struct statfs *statbuf, FAR void *arg) { FAR struct nsh_vtbl_s *vtbl = (FAR struct nsh_vtbl_s *)arg; DEBUGASSERT(mountpoint && statbuf && vtbl); nsh_output(vtbl, "%6ld %8ld %8ld %8ld %s\n", statbuf->f_bsize, statbuf->f_blocks, statbuf->f_blocks - statbuf->f_bavail, statbuf->f_bavail, mountpoint); return OK; } /**************************************************************************** * Name: df_man_readable_handler ****************************************************************************/ #ifdef CONFIG_NSH_CMDOPT_DF_H static int df_man_readable_handler(FAR const char *mountpoint, FAR struct statfs *statbuf, FAR void *arg) { FAR struct nsh_vtbl_s *vtbl = (FAR struct nsh_vtbl_s *)arg; uint32_t size; uint32_t used; uint32_t free; int which; char sizelabel; char freelabel; char usedlabel; const char labels[5] = { 'B', 'K', 'M', 'G', 'T' }; DEBUGASSERT(mountpoint && statbuf && vtbl); size = statbuf->f_bsize * statbuf->f_blocks; free = statbuf->f_bsize * statbuf->f_bavail; used = size - free; /* Find the label for size */ which = 0; while (size >= 9999 || ((size & 0x3ff) == 0 && size != 0)) { which++; size >>= 10; } sizelabel = labels[which]; /* Find the label for free */ which = 0; while (free >= 9999 || ((free & 0x3ff) == 0 && free != 0)) { which++; free >>= 10; } freelabel = labels[which]; /* Find the label for used */ which = 0; while (used >= 9999 || ((used & 0x3ff) == 0 && used != 0)) { which++; used >>= 10; } usedlabel = labels[which]; #if !defined(CONFIG_BUILD_PROTECTED) && !defined(CONFIG_BUILD_KERNEL) nsh_output(vtbl, "%-10s %6ld%c %8ld%c %8ld%c %s\n", get_fstype(statbuf), size, sizelabel, used, usedlabel, free, freelabel, mountpoint); #else nsh_output(vtbl, "%6ld%c %8ld%c %8ld%c %s\n", size, sizelabel, used, usedlabel, free, freelabel, mountpoint); #endif return OK; } #endif /* CONFIG_NSH_CMDOPT_DF_H */ #endif /* CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_DF) */ /**************************************************************************** * Name: mount_handler ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_MOUNT) #if !defined(CONFIG_BUILD_PROTECTED) && !defined(CONFIG_BUILD_KERNEL) static int mount_handler(FAR const char *mountpoint, FAR struct statfs *statbuf, FAR void *arg) { FAR struct nsh_vtbl_s *vtbl = (FAR struct nsh_vtbl_s *)arg; FAR const char *fstype; DEBUGASSERT(mountpoint && statbuf && vtbl); /* Get the file system type */ fstype = get_fstype(statbuf); nsh_output(vtbl, " %s type %s\n", mountpoint, fstype); return OK; } #endif #endif /**************************************************************************** * Name: mount_show ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_MOUNT) #if !defined(CONFIG_BUILD_PROTECTED) && !defined(CONFIG_BUILD_KERNEL) static inline int mount_show(FAR struct nsh_vtbl_s *vtbl, FAR const char *progname) { return foreach_mountpoint(mount_handler, (FAR void *)vtbl); } #endif #endif /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: cmd_df ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_DF) int cmd_df(FAR struct nsh_vtbl_s *vtbl, int argc, char **argv) { #ifdef CONFIG_NSH_CMDOPT_DF_H if (argc > 1 && strcmp(argv[1], "-h") == 0) { #if !defined(CONFIG_BUILD_PROTECTED) && !defined(CONFIG_BUILD_KERNEL) nsh_output(vtbl, "Filesystem Size Used Available Mounted on\n"); #else nsh_output(vtbl, "Size Used Available Mounted on\n"); #endif return foreach_mountpoint(df_man_readable_handler, (FAR void *)vtbl); } else #endif { nsh_output(vtbl, " Block Number\n"); nsh_output(vtbl, " Size Blocks Used Available Mounted on\n"); return foreach_mountpoint(df_handler, (FAR void *)vtbl); } } #endif /**************************************************************************** * Name: cmd_mount ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_MOUNT) int cmd_mount(FAR struct nsh_vtbl_s *vtbl, int argc, char **argv) { FAR const char *source; FAR char *fullsource; FAR const char *target; FAR char *fulltarget; FAR const char *filesystem = NULL; bool badarg = false; int option; int ret; /* The mount command behaves differently if no parameters are provided */ #if !defined(CONFIG_BUILD_PROTECTED) && !defined(CONFIG_BUILD_KERNEL) if (argc < 2) { return mount_show(vtbl, argv[0]); } #endif /* Get the mount options. NOTE: getopt() is not thread safe nor re-entrant. * To keep its state proper for the next usage, it is necessary to parse to * the end of the line even if an error occurs. If an error occurs, this * logic just sets 'badarg' and continues. */ while ((option = getopt(argc, argv, ":t:")) != ERROR) { switch (option) { case 't': filesystem = optarg; break; case ':': nsh_output(vtbl, g_fmtargrequired, argv[0]); badarg = true; break; case '?': default: nsh_output(vtbl, g_fmtarginvalid, argv[0]); badarg = true; break; } } /* If a bad argument was encountered, then return without processing the * command. */ if (badarg) { return ERROR; } /* There may be one or two required arguments after the options: the source * and target paths. Some file systems do not require the source parameter * so if there is only one parameter left, it must be the target. */ if (optind >= argc) { nsh_output(vtbl, g_fmtargrequired, argv[0]); return ERROR; } source = NULL; target = argv[optind]; optind++; if (optind < argc) { source = target; target = argv[optind]; optind++; if (optind < argc) { nsh_output(vtbl, g_fmttoomanyargs, argv[0]); return ERROR; } } /* While the above parsing for the -t argument looks nice, the -t argument * not really optional. */ if (!filesystem) { nsh_output(vtbl, g_fmtargrequired, argv[0]); return ERROR; } /* The source and target paths might be relative to the current * working directory. */ fullsource = NULL; fulltarget = NULL; if (source) { fullsource = nsh_getfullpath(vtbl, source); if (!fullsource) { return ERROR; } } fulltarget = nsh_getfullpath(vtbl, target); if (!fulltarget) { ret = ERROR; goto errout; } /* Perform the mount */ ret = mount(fullsource, fulltarget, filesystem, 0, NULL); if (ret < 0) { nsh_output(vtbl, g_fmtcmdfailed, argv[0], "mount", NSH_ERRNO); } errout: if (fullsource) { nsh_freefullpath(fullsource); } if (fulltarget) { nsh_freefullpath(fulltarget); } return ret; } #endif /**************************************************************************** * Name: cmd_nfsmount ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_NET) && defined(CONFIG_NFS) && !defined(CONFIG_NSH_DISABLE_NFSMOUNT) int cmd_nfsmount(FAR struct nsh_vtbl_s *vtbl, int argc, char **argv) { struct nfs_args data; FAR char *address; FAR char *lpath; FAR char *rpath; bool badarg = false; #ifdef CONFIG_NET_IPv6 FAR struct sockaddr_in6 *sin; struct in6_addr inaddr; #else FAR struct sockaddr_in *sin; struct in_addr inaddr; #endif int ret; /* If a bad argument was encountered, then return without processing the * command. */ if (badarg) { return ERROR; } /* The fist argument on the command line should be the NFS server IP address * in standard IPv4 (or IPv6) dot format. */ address = argv[1]; if (!address) { return ERROR; } /* The local mount point path (lpath) might be relative to the current working * directory. */ lpath = nsh_getfullpath(vtbl, argv[2]); if (!lpath) { return ERROR; } /* Get the remote mount point path */ rpath = argv[3]; /* Convert the IP address string into its binary form */ #ifdef CONFIG_NET_IPv6 ret = inet_pton(AF_INET6, address, &inaddr); #else ret = inet_pton(AF_INET, address, &inaddr); #endif if (ret != 1) { nsh_freefullpath(lpath); return ERROR; } /* Place all of the NFS arguements into the nfs_args structure */ memset(&data, 0, sizeof(data)); #ifdef CONFIG_NET_IPv6 sin = (FAR struct sockaddr_in6 *)&data.addr; sin->sin6_family = AF_INET6; sin->sin6_port = htons(NFS_PMAPPORT); memcpy(&sin->sin6_addr, &inaddr, sizeof(struct in6_addr)); data.addrlen = sizeof(struct sockaddr_in6); #else sin = (FAR struct sockaddr_in *)&data.addr; sin->sin_family = AF_INET; sin->sin_port = htons(NFS_PMAPPORT); sin->sin_addr = inaddr; data.addrlen = sizeof(struct sockaddr_in); #endif data.sotype = SOCK_DGRAM; data.path = rpath; data.flags = 0; /* 0=Use all defaults */ /* Perform the mount */ ret = mount(NULL, lpath, "nfs", 0, (FAR void *)&data); if (ret < 0) { nsh_output(vtbl, g_fmtcmdfailed, argv[0], "mount", NSH_ERRNO); } /* We no longer need the allocated mount point path */ nsh_freefullpath(lpath); return ret; } #endif /**************************************************************************** * Name: cmd_umount ****************************************************************************/ #if CONFIG_NFILE_DESCRIPTORS > 0 && !defined(CONFIG_DISABLE_MOUNTPOINT) && \ defined(CONFIG_FS_READABLE) && !defined(CONFIG_NSH_DISABLE_UMOUNT) int cmd_umount(FAR struct nsh_vtbl_s *vtbl, int argc, char **argv) { char *fullpath = nsh_getfullpath(vtbl, argv[1]); int ret = ERROR; if (fullpath) { /* Perform the umount */ ret = umount(fullpath); if (ret < 0) { nsh_output(vtbl, g_fmtcmdfailed, argv[0], "umount", NSH_ERRNO); } nsh_freefullpath(fullpath); } return ret; } #endif