/**************************************************************************** * fs/vfs/fs_ioctl.c * * Copyright (C) 2007-2010, 2012-2014, 2016-2017 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 #ifdef CONFIG_NET # include #endif #include "inode/inode.h" /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: file_ioctl and file_vioctl * * Description: * Perform device specific operations. * * Input Parameters: * file File structure instance * req The ioctl command * ap The argument of the ioctl cmd * * Returned Value: * Returns a non-negative number on success; A negated errno value is * returned on any failure (see comments ioctl() for a list of appropriate * errno values). * ****************************************************************************/ int file_vioctl(FAR struct file *filep, int req, va_list ap) { FAR struct inode *inode; DEBUGASSERT(filep != NULL); /* Is a driver opened? */ inode = filep->f_inode; if (!inode) { return -EBADF; } /* Does the driver support the ioctl method? */ if (inode->u.i_ops == NULL || inode->u.i_ops->ioctl == NULL) { return -ENOTTY; } /* Yes on both accounts. Let the driver perform the ioctl command */ return inode->u.i_ops->ioctl(filep, req, va_arg(ap, unsigned long)); } int file_ioctl(FAR struct file *filep, int req, ...) { va_list ap; int ret; /* Let file_vioctl() do the real work. */ va_start(ap, req); ret = file_vioctl(filep, req, ap); va_end(ap); return ret; } /**************************************************************************** * Name: nx_ioctl and nx_vioctl * * Description: * nx_ioctl() is similar to the standard 'ioctl' interface except that is * not a cancellation point and it does not modify the errno variable. * * nx_ioctl() is an internal NuttX interface and should not be called from * applications. * * Returned Value: * Returns a non-negative number on success; A negated errno value is * returned on any failure (see comments ioctl() for a list of appropriate * errno values). * ****************************************************************************/ int nx_vioctl(int fd, int req, va_list ap) { FAR struct file *filep; FAR int *arg; int ret; /* Did we get a valid file descriptor? */ if (fd >= CONFIG_NFILE_DESCRIPTORS) { /* Perform the socket ioctl */ #ifdef CONFIG_NET if (fd < (CONFIG_NFILE_DESCRIPTORS + CONFIG_NSOCKET_DESCRIPTORS)) { ret = netdev_vioctl(fd, req, ap); } else #endif { return -EBADF; } } else { /* Get the file structure corresponding to the file descriptor. */ ret = fs_getfilep(fd, &filep); if (ret < 0) { return ret; } DEBUGASSERT(filep != NULL); /* Perform the file ioctl. */ ret = file_vioctl(filep, req, ap); } /* Check for File system IOCTL commands that can be implemented via * fcntl() */ if (ret == -ENOTTY) { switch (req) { case FIONBIO: arg = va_arg(ap, FAR int *); if (arg && *arg) { ret = nx_fcntl(fd, F_SETFL, nx_fcntl(fd, F_GETFL) | O_NONBLOCK); } else { ret = nx_fcntl(fd, F_SETFL, nx_fcntl(fd, F_GETFL) & ~O_NONBLOCK); } break; case FIOCLEX: ret = nx_fcntl(fd, F_SETFD, nx_fcntl(fd, F_GETFD) | FD_CLOEXEC); break; case FIONCLEX: ret = nx_fcntl(fd, F_SETFD, nx_fcntl(fd, F_GETFD) & ~FD_CLOEXEC); break; } } return ret; } int nx_ioctl(int fd, int req, ...) { va_list ap; int ret; /* Let nx_vioctl() do the real work. */ va_start(ap, req); ret = nx_vioctl(fd, req, ap); va_end(ap); return ret; } /**************************************************************************** * Name: ioctl * * Description: * Perform device specific operations. * * Input Parameters: * fd File/socket descriptor of device * req The ioctl command * * Returned Value: * >=0 on success (positive non-zero values are cmd-specific) * -1 on failure with errno set properly: * * EBADF * 'fd' is not a valid descriptor. * EFAULT * 'arg' references an inaccessible memory area. * EINVAL * 'cmd' or 'arg' is not valid. * ENOTTY * 'fd' is not associated with a character special device. * ENOTTY * The specified request does not apply to the kind of object that the * descriptor 'fd' references. * ****************************************************************************/ int ioctl(int fd, int req, ...) { va_list ap; int ret; /* Let nx_vioctl() do the real work. */ va_start(ap, req); ret = nx_vioctl(fd, req, ap); va_end(ap); if (ret < 0) { set_errno(-ret); ret = ERROR; } return ret; }