/* * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved. * Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved. * * 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 of the copyright holder 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 HOLDER 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. */ #include "syscall_pub.h" #ifdef LOSCFG_FS_VFS #include "errno.h" #include "unistd.h" #include "fs/fd_table.h" #include "fs/file.h" #include "fs/fs.h" #include "fs/fs_operation.h" #include "sys/mount.h" #include "los_task_pri.h" #include "sys/utsname.h" #include "sys/uio.h" #include "poll.h" #include "sys/prctl.h" #include "epoll.h" #ifdef LOSCFG_KERNEL_DYNLOAD #include "los_exec_elf.h" #endif #include "los_syscall.h" #include "dirent.h" #include "user_copy.h" #include "los_vm_map.h" #include "los_memory.h" #include "los_strncpy_from_user.h" #include "capability_type.h" #include "capability_api.h" #include "sys/statfs.h" #define HIGH_SHIFT_BIT 32 #define TIMESPEC_TIMES_NUM 2 static int CheckNewAttrTime(struct IATTR *attr, struct timespec times[TIMESPEC_TIMES_NUM]) { int ret = ENOERR; struct timespec stp = {0}; if (times) { if (times[0].tv_nsec == UTIME_OMIT) { attr->attr_chg_valid &= ~CHG_ATIME; } else if (times[0].tv_nsec == UTIME_NOW) { ret = clock_gettime(CLOCK_REALTIME, &stp); if (ret < 0) { return -get_errno(); } attr->attr_chg_atime = (unsigned int)stp.tv_sec; attr->attr_chg_valid |= CHG_ATIME; } else { attr->attr_chg_atime = (unsigned int)times[0].tv_sec; attr->attr_chg_valid |= CHG_ATIME; } if (times[1].tv_nsec == UTIME_OMIT) { attr->attr_chg_valid &= ~CHG_MTIME; } else if (times[1].tv_nsec == UTIME_NOW) { ret = clock_gettime(CLOCK_REALTIME, &stp); if (ret < 0) { return -get_errno(); } attr->attr_chg_mtime = (unsigned int)stp.tv_sec; attr->attr_chg_valid |= CHG_MTIME; } else { attr->attr_chg_mtime = (unsigned int)times[1].tv_sec; attr->attr_chg_valid |= CHG_MTIME; } } else { ret = clock_gettime(CLOCK_REALTIME, &stp); if (ret < 0) { return -get_errno(); } attr->attr_chg_atime = (unsigned int)stp.tv_sec; attr->attr_chg_mtime = (unsigned int)stp.tv_sec; attr->attr_chg_valid |= CHG_ATIME; attr->attr_chg_valid |= CHG_MTIME; } return ret; } static int GetFullpathNull(int fd, const char *path, char **filePath) { int ret; char *fullPath = NULL; struct file *file = NULL; if ((fd != AT_FDCWD) && (path == NULL)) { fd = GetAssociatedSystemFd(fd); ret = fs_getfilep(fd, &file); if (ret < 0) { return -get_errno(); } fullPath = strdup(file->f_path); if (fullPath == NULL) { ret = -ENOMEM; } } else { ret = GetFullpath(fd, path, &fullPath); if (ret < 0) { return ret; } } *filePath = fullPath; return ret; } static int UserIovItemCheck(const struct iovec *iov, const int iovcnt) { int i; for (i = 0; i < iovcnt; ++i) { if (iov[i].iov_len == 0) { continue; } if (!LOS_IsUserAddressRange((vaddr_t)(UINTPTR)iov[i].iov_base, iov[i].iov_len)) { return i; } } return iovcnt; } static int UserIovCopy(struct iovec **iovBuf, const struct iovec *iov, const int iovcnt, int *valid_iovcnt) { int ret; int bufLen = iovcnt * sizeof(struct iovec); if (bufLen < 0) { return -EINVAL; } *iovBuf = (struct iovec*)LOS_MemAlloc(OS_SYS_MEM_ADDR, bufLen); if (*iovBuf == NULL) { return -ENOMEM; } if (LOS_ArchCopyFromUser(*iovBuf, iov, bufLen) != 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, *iovBuf); return -EFAULT; } ret = UserIovItemCheck(*iovBuf, iovcnt); if (ret == 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, *iovBuf); return -EFAULT; } *valid_iovcnt = ret; return 0; } static int PollfdToSystem(struct pollfd *fds, nfds_t nfds, int **pollFdsBak) { if ((nfds != 0 && fds == NULL) || (pollFdsBak == NULL)) { set_errno(EINVAL); return -1; } if (nfds == 0) { return 0; } int *pollFds = (int *)malloc(sizeof(int) * nfds); if (pollFds == NULL) { set_errno(ENOMEM); return -1; } for (int i = 0; i < nfds; ++i) { struct pollfd *p_fds = &fds[i]; pollFds[i] = p_fds->fd; if (p_fds->fd < 0) { set_errno(EBADF); free(pollFds); return -1; } p_fds->fd = GetAssociatedSystemFd(p_fds->fd); } *pollFdsBak = pollFds; return 0; } static void RestorePollfd(struct pollfd *fds, nfds_t nfds, const int *pollFds) { if ((fds == NULL) || (pollFds == NULL)) { return; } for (int i = 0; i < nfds; ++i) { struct pollfd *p_fds = &fds[i]; p_fds->fd = pollFds[i]; } } static int UserPoll(struct pollfd *fds, nfds_t nfds, int timeout) { int *pollFds = NULL; int ret = PollfdToSystem(fds, nfds, &pollFds); if (ret < 0) { return -1; } ret = poll(fds, nfds, timeout); RestorePollfd(fds, nfds, pollFds); free(pollFds); return ret; } int SysClose(int fd) { int ret; /* Process fd convert to system global fd */ int sysfd = DisassociateProcessFd(fd); ret = close(sysfd); if (ret < 0) { AssociateSystemFd(fd, sysfd); return -get_errno(); } FreeProcessFd(fd); return ret; } ssize_t SysRead(int fd, void *buf, size_t nbytes) { int ret; if (nbytes == 0) { return 0; } if (!LOS_IsUserAddressRange((vaddr_t)(UINTPTR)buf, nbytes)) { return -EFAULT; } /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = read(fd, buf, nbytes); if (ret < 0) { return -get_errno(); } return ret; } ssize_t SysWrite(int fd, const void *buf, size_t nbytes) { int ret; if (nbytes == 0) { return 0; } if (!LOS_IsUserAddressRange((vaddr_t)(UINTPTR)buf, nbytes)) { return -EFAULT; } /* Process fd convert to system global fd */ int sysfd = GetAssociatedSystemFd(fd); ret = write(sysfd, buf, nbytes); if (ret < 0) { return -get_errno(); } return ret; } int SysOpen(const char *path, int oflags, ...) { int ret; int procFd; mode_t mode = DEFAULT_FILE_MODE; /* 0666: File read-write properties. */ char *pathRet = NULL; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { return ret; } } procFd = AllocProcessFd(); if (procFd < 0) { ret = -EMFILE; goto ERROUT; } if (oflags & O_CLOEXEC) { SetCloexecFlag(procFd); } if ((unsigned int)oflags & O_DIRECTORY) { ret = do_opendir(pathRet, oflags); } else { #ifdef LOSCFG_FILE_MODE va_list ap; va_start(ap, oflags); mode = va_arg(ap, int); va_end(ap); #endif ret = do_open(AT_FDCWD, pathRet, oflags, mode); } if (ret < 0) { ret = -get_errno(); goto ERROUT; } AssociateSystemFd(procFd, ret); if (pathRet != NULL) { LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return procFd; ERROUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } if (procFd >= 0) { FreeProcessFd(procFd); } return ret; } int SysCreat(const char *pathname, mode_t mode) { int ret = 0; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } int procFd = AllocProcessFd(); if (procFd < 0) { ret = -EMFILE; goto OUT; } ret = open((pathname ? pathRet : NULL), O_CREAT | O_TRUNC | O_WRONLY, mode); if (ret < 0) { FreeProcessFd(procFd); ret = -get_errno(); } else { AssociateSystemFd(procFd, ret); ret = procFd; } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysLink(const char *oldpath, const char *newpath) { int ret; char *oldpathRet = NULL; char *newpathRet = NULL; if (oldpath != NULL) { ret = UserPathCopy(oldpath, &oldpathRet); if (ret != 0) { goto OUT; } } if (newpath != NULL) { ret = UserPathCopy(newpath, &newpathRet); if (ret != 0) { goto OUT; } } ret = link(oldpathRet, newpathRet); if (ret < 0) { ret = -get_errno(); } OUT: if (oldpathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, oldpathRet); } if (newpathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, newpathRet); } return ret; } ssize_t SysReadlink(const char *pathname, char *buf, size_t bufsize) { ssize_t ret; char *pathRet = NULL; if (bufsize == 0) { return -EINVAL; } if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } if (!LOS_IsUserAddressRange((vaddr_t)(UINTPTR)buf, bufsize)) { ret = -EFAULT; goto OUT; } ret = readlink(pathRet, buf, bufsize); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysSymlink(const char *target, const char *linkpath) { int ret; char *targetRet = NULL; char *pathRet = NULL; if (target != NULL) { ret = UserPathCopy(target, &targetRet); if (ret != 0) { goto OUT; } } if (linkpath != NULL) { ret = UserPathCopy(linkpath, &pathRet); if (ret != 0) { goto OUT; } } ret = symlink(targetRet, pathRet); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } if (targetRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, targetRet); } return ret; } int SysUnlink(const char *pathname) { int ret; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } ret = do_unlink(AT_FDCWD, (pathname ? pathRet : NULL)); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } #ifdef LOSCFG_KERNEL_DYNLOAD int SysExecve(const char *fileName, char *const *argv, char *const *envp) { return LOS_DoExecveFile(fileName, argv, envp); } #endif int SysFchdir(int fd) { int ret; int sysFd; struct file *file = NULL; sysFd = GetAssociatedSystemFd(fd); if (sysFd < 0) { return -EBADF; } ret = fs_getfilep(sysFd, &file); if (ret < 0) { return -get_errno(); } ret = chdir(file->f_path); if (ret < 0) { ret = -get_errno(); } return ret; } int SysChdir(const char *path) { int ret; char *pathRet = NULL; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } ret = chdir(path ? pathRet : NULL); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } off_t SysLseek(int fd, off_t offset, int whence) { /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); return _lseek(fd, offset, whence); } off64_t SysLseek64(int fd, int offsetHigh, int offsetLow, off64_t *result, int whence) { off64_t ret; off64_t res; int retVal; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = _lseek64(fd, offsetHigh, offsetLow, &res, whence); if (ret != 0) { return ret; } retVal = LOS_ArchCopyToUser(result, &res, sizeof(off64_t)); if (retVal != 0) { return -EFAULT; } return 0; } #ifdef LOSCFG_FS_NFS static int NfsMountRef(const char *serverIpAndPath, const char *mountPath, unsigned int uid, unsigned int gid) __attribute__((weakref("nfs_mount"))); static int NfsMount(const char *serverIpAndPath, const char *mountPath, unsigned int uid, unsigned int gid) { int ret; if ((serverIpAndPath == NULL) || (mountPath == NULL)) { return -EINVAL; } ret = NfsMountRef(serverIpAndPath, mountPath, uid, gid); if (ret < 0) { ret = -get_errno(); } return ret; } #endif int SysMount(const char *source, const char *target, const char *filesystemtype, unsigned long mountflags, const void *data) { int ret; char *sourceRet = NULL; char *targetRet = NULL; char *dataRet = NULL; char fstypeRet[FILESYSTEM_TYPE_MAX + 1] = {0}; if (!IsCapPermit(CAP_FS_MOUNT)) { return -EPERM; } if (target != NULL) { ret = UserPathCopy(target, &targetRet); if (ret != 0) { goto OUT; } } if (filesystemtype != NULL) { ret = LOS_StrncpyFromUser(fstypeRet, filesystemtype, FILESYSTEM_TYPE_MAX + 1); if (ret < 0) { goto OUT; } else if (ret > FILESYSTEM_TYPE_MAX) { ret = -ENODEV; goto OUT; } if (strcmp(fstypeRet, "ramfs") && (source != NULL)) { ret = UserPathCopy(source, &sourceRet); if (ret != 0) { goto OUT; } } #ifdef LOSCFG_FS_NFS if (strcmp(fstypeRet, "nfs") == 0) { ret = NfsMount(sourceRet, targetRet, 0, 0); goto OUT; } #endif } if (data != NULL) { ret = UserPathCopy(data, &dataRet); if (ret != 0) { goto OUT; } } ret = mount(sourceRet, targetRet, (filesystemtype ? fstypeRet : NULL), mountflags, dataRet); if (ret < 0) { ret = -get_errno(); } OUT: if (sourceRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, sourceRet); } if (targetRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, targetRet); } if (dataRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, dataRet); } return ret; } int SysUmount(const char *target) { int ret; char *pathRet = NULL; if (!IsCapPermit(CAP_FS_MOUNT)) { return -EPERM; } if (target != NULL) { ret = UserPathCopy(target, &pathRet); if (ret != 0) { goto OUT; } } ret = umount(target ? pathRet : NULL); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysAccess(const char *path, int amode) { int ret; char *pathRet = NULL; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } ret = access(pathRet, amode); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysRename(const char *oldpath, const char *newpath) { int ret; char *pathOldRet = NULL; char *pathNewRet = NULL; if (oldpath != NULL) { ret = UserPathCopy(oldpath, &pathOldRet); if (ret != 0) { goto OUT; } } if (newpath != NULL) { ret = UserPathCopy(newpath, &pathNewRet); if (ret != 0) { goto OUT; } } ret = do_rename(AT_FDCWD, (oldpath ? pathOldRet : NULL), AT_FDCWD, (newpath ? pathNewRet : NULL)); if (ret < 0) { ret = -get_errno(); } OUT: if (pathOldRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathOldRet); } if (pathNewRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathNewRet); } return ret; } int SysMkdir(const char *pathname, mode_t mode) { int ret; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } ret = do_mkdir(AT_FDCWD, (pathname ? pathRet : NULL), mode); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysRmdir(const char *pathname) { int ret; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } ret = do_rmdir(AT_FDCWD, (pathname ? pathRet : NULL)); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysDup(int fd) { int sysfd = GetAssociatedSystemFd(fd); /* Check if the param is valid, note that: socket fd is not support dup2 */ if ((sysfd < 0) || (sysfd >= CONFIG_NFILE_DESCRIPTORS)) { return -EBADF; } int dupfd = AllocProcessFd(); if (dupfd < 0) { return -EMFILE; } files_refer(sysfd); AssociateSystemFd(dupfd, sysfd); return dupfd; } void SysSync(void) { sync(); } int SysUmount2(const char *target, int flags) { if (flags != 0) { return -EINVAL; } return SysUmount(target); } int SysIoctl(int fd, int req, void *arg) { int ret; unsigned int size = _IOC_SIZE((unsigned int)req); unsigned int dir = _IOC_DIR((unsigned int)req); if ((size == 0) && (dir != _IOC_NONE)) { return -EINVAL; } if ((dir != _IOC_NONE) && (((void *)(uintptr_t)arg) == NULL)) { return -EINVAL; } if ((dir & _IOC_READ) || (dir & _IOC_WRITE)) { if (!LOS_IsUserAddressRange((uintptr_t)arg, size)) { return -EFAULT; } } /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = ioctl(fd, req, arg); if (ret < 0) { return -get_errno(); } return ret; } int SysFcntl(int fd, int cmd, void *arg) { /* Process fd convert to system global fd */ int sysfd = GetAssociatedSystemFd(fd); int ret = VfsFcntl(fd, cmd, arg); if (ret == CONTINE_NUTTX_FCNTL) { ret = fcntl(sysfd, cmd, arg); } if (ret < 0) { return -get_errno(); } return ret; } #ifdef LOSCFG_KERNEL_PIPE int SysPipe(int pipefd[2]) /* 2 : pipe fds for read and write */ { int ret; int pipeFdIntr[2] = {0}; /* 2 : pipe fds for read and write */ int procFd0 = AllocProcessFd(); if (procFd0 < 0) { return -EMFILE; } int procFd1 = AllocProcessFd(); if (procFd1 < 0) { FreeProcessFd(procFd0); return -EMFILE; } ret = pipe(pipeFdIntr); if (ret < 0) { FreeProcessFd(procFd0); FreeProcessFd(procFd1); return -get_errno(); } int sysPipeFd0 = pipeFdIntr[0]; int sysPipeFd1 = pipeFdIntr[1]; AssociateSystemFd(procFd0, sysPipeFd0); AssociateSystemFd(procFd1, sysPipeFd1); pipeFdIntr[0] = procFd0; pipeFdIntr[1] = procFd1; ret = LOS_ArchCopyToUser(pipefd, pipeFdIntr, sizeof(pipeFdIntr)); if (ret != 0) { FreeProcessFd(procFd0); FreeProcessFd(procFd1); close(sysPipeFd0); close(sysPipeFd1); return -EFAULT; } return ret; } #endif int SysDup2(int fd1, int fd2) { int ret; int sysfd1 = GetAssociatedSystemFd(fd1); int sysfd2 = GetAssociatedSystemFd(fd2); /* Check if the param is valid, note that: socket fd is not support dup2 */ if ((sysfd1 < 0) || (sysfd1 >= CONFIG_NFILE_DESCRIPTORS) || (CheckProcessFd(fd2) != OK)) { return -EBADF; } /* Handle special circumstances */ if (fd1 == fd2) { return fd2; } ret = AllocSpecifiedProcessFd(fd2); if (ret != OK) { return ret; } /* close the sysfd2 in need */ if (sysfd2 >= 0) { ret = close(sysfd2); if (ret < 0) { AssociateSystemFd(fd2, sysfd2); return -get_errno(); } } files_refer(sysfd1); AssociateSystemFd(fd2, sysfd1); /* if fd1 is not equal to fd2, the FD_CLOEXEC flag associated with fd2 shall be cleared */ ClearCloexecFlag(fd2); return fd2; } static int SelectParamCheckCopy(fd_set *readfds, fd_set *writefds, fd_set *exceptfds, fd_set **fdsBuf) { fd_set *readfdsRet = NULL; fd_set *writefdsRet = NULL; fd_set *exceptfdsRet = NULL; *fdsBuf = (fd_set *)LOS_MemAlloc(OS_SYS_MEM_ADDR, sizeof(fd_set) * 3); /* 3: three param need check and copy */ if (*fdsBuf == NULL) { return -ENOMEM; } readfdsRet = *fdsBuf; /* LOS_MemAlloc 3 sizeof(fd_set) space,first use for readfds */ writefdsRet = *fdsBuf + 1; /* 1: LOS_MemAlloc 3 sizeof(fd_set) space,second use for writefds */ exceptfdsRet = *fdsBuf + 2; /* 2: LOS_MemAlloc 3 sizeof(fd_set) space,thired use for exceptfds */ if (readfds != NULL) { if (LOS_ArchCopyFromUser(readfdsRet, readfds, sizeof(fd_set)) != 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, *fdsBuf); return -EFAULT; } } if (writefds != NULL) { if (LOS_ArchCopyFromUser(writefdsRet, writefds, sizeof(fd_set)) != 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, *fdsBuf); return -EFAULT; } } if (exceptfds != NULL) { if (LOS_ArchCopyFromUser(exceptfdsRet, exceptfds, sizeof(fd_set)) != 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, *fdsBuf); return -EFAULT; } } return 0; } int SysSelect(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout) { int ret; fd_set *fdsRet = NULL; fd_set *readfdsRet = NULL; fd_set *writefdsRet = NULL; fd_set *exceptfdsRet = NULL; struct timeval timeoutRet = {0}; ret = SelectParamCheckCopy(readfds, writefds, exceptfds, &fdsRet); if (ret != 0) { return ret; } readfdsRet = fdsRet; /* LOS_MemAlloc 3 sizeof(fd_set) space,first use for readfds */ writefdsRet = fdsRet + 1; /* 1: LOS_MemAlloc 3 sizeof(fd_set) space,second use for writefds */ exceptfdsRet = fdsRet + 2; /* 2: LOS_MemAlloc 3 sizeof(fd_set) space,thired use for exceptfds */ if (timeout != NULL) { if (LOS_ArchCopyFromUser(&timeoutRet, timeout, sizeof(struct timeval)) != 0) { goto ERROUT; } } ret = do_select(nfds, (readfds ? readfdsRet : NULL), (writefds ? writefdsRet : NULL), (exceptfds ? exceptfdsRet : NULL), (timeout ? (&timeoutRet) : NULL), UserPoll); if (ret < 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, fdsRet); return -get_errno(); } if (readfds != NULL) { if (LOS_ArchCopyToUser(readfds, readfdsRet, sizeof(fd_set)) != 0) { goto ERROUT; } } if (writefds != NULL) { if (LOS_ArchCopyToUser(writefds, writefdsRet, sizeof(fd_set)) != 0) { goto ERROUT; } } if (exceptfds != 0) { if (LOS_ArchCopyToUser(exceptfds, exceptfdsRet, sizeof(fd_set)) != 0) { goto ERROUT; } } (void)LOS_MemFree(OS_SYS_MEM_ADDR, fdsRet); return ret; ERROUT: (void)LOS_MemFree(OS_SYS_MEM_ADDR, fdsRet); return -EFAULT; } int SysTruncate(const char *path, off_t length) { int ret; int fd = -1; char *pathRet = NULL; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } fd = open((path ? pathRet : NULL), O_RDWR); if (fd < 0) { /* The errno value has already been set */ ret = -get_errno(); goto OUT; } ret = ftruncate(fd, length); close(fd); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysTruncate64(const char *path, off64_t length) { int ret; int fd = -1; char *pathRet = NULL; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } fd = open((path ? pathRet : NULL), O_RDWR); if (fd < 0) { /* The errno value has already been set */ ret = -get_errno(); goto OUT; } ret = ftruncate64(fd, length); close(fd); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysFtruncate(int fd, off_t length) { int ret; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = ftruncate(fd, length); if (ret < 0) { return -get_errno(); } return ret; } int SysStatfs(const char *path, struct statfs *buf) { int ret; char *pathRet = NULL; struct statfs bufRet = {0}; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } ret = statfs((path ? pathRet : NULL), (buf ? (&bufRet) : NULL)); if (ret < 0) { ret = -get_errno(); goto OUT; } ret = LOS_ArchCopyToUser(buf, &bufRet, sizeof(struct statfs)); if (ret != 0) { ret = -EFAULT; } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysStatfs64(const char *path, size_t sz, struct statfs *buf) { int ret; char *pathRet = NULL; struct statfs bufRet = {0}; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } if (sz != sizeof(*buf)) { ret = -EINVAL; goto OUT; } ret = statfs((path ? pathRet : NULL), (buf ? (&bufRet) : NULL)); if (ret < 0) { ret = -get_errno(); goto OUT; } ret = LOS_ArchCopyToUser(buf, &bufRet, sizeof(struct statfs)); if (ret != 0) { ret = -EFAULT; } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysStat(const char *path, struct kstat *buf) { int ret; char *pathRet = NULL; struct stat bufRet = {0}; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } ret = stat((path ? pathRet : NULL), (buf ? (&bufRet) : NULL)); if (ret < 0) { ret = -get_errno(); goto OUT; } ret = LOS_ArchCopyToUser(buf, &bufRet, sizeof(struct kstat)); if (ret != 0) { ret = -EFAULT; } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysLstat(const char *path, struct kstat *buffer) { int ret; char *pathRet = NULL; struct stat bufRet = {0}; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } ret = stat((path ? pathRet : NULL), (buffer ? (&bufRet) : NULL)); if (ret < 0) { ret = -get_errno(); goto OUT; } ret = LOS_ArchCopyToUser(buffer, &bufRet, sizeof(struct kstat)); if (ret != 0) { ret = -EFAULT; } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysFstat(int fd, struct kstat *buf) { int ret; struct stat bufRet = {0}; struct file *filep = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = fs_getfilep(fd, &filep); if (ret < 0) { return -get_errno(); } if (filep->f_oflags & O_DIRECTORY) { return -EBADF; } ret = stat(filep->f_path, (buf ? (&bufRet) : NULL)); if (ret < 0) { return -get_errno(); } ret = LOS_ArchCopyToUser(buf, &bufRet, sizeof(struct kstat)); if (ret != 0) { return -EFAULT; } return ret; } int SysStatx(int fd, const char *restrict path, int flag, unsigned mask, struct statx *restrict stx) { return -ENOSYS; } int SysFsync(int fd) { int ret; struct file *filep = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); /* Get the file structure corresponding to the file descriptor. */ ret = fs_getfilep(fd, &filep); if (ret < 0) { /* The errno value has already been set */ return -get_errno(); } if (filep->f_oflags & O_DIRECTORY) { return -EBADF; } /* Perform the fsync operation */ ret = file_fsync(filep); if (ret < 0) { return -get_errno(); } return ret; } ssize_t SysReadv(int fd, const struct iovec *iov, int iovcnt) { int ret; int valid_iovcnt = -1; struct iovec *iovRet = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); if ((iov == NULL) || (iovcnt < 0) || (iovcnt > IOV_MAX)) { return -EINVAL; } if (iovcnt == 0) { return 0; } ret = UserIovCopy(&iovRet, iov, iovcnt, &valid_iovcnt); if (ret != 0) { return ret; } if (valid_iovcnt <= 0) { ret = -EFAULT; goto OUT; } ret = vfs_readv(fd, iovRet, valid_iovcnt, NULL); if (ret < 0) { ret = -get_errno(); } OUT: (void)LOS_MemFree(OS_SYS_MEM_ADDR, iovRet); return ret; } ssize_t SysWritev(int fd, const struct iovec *iov, int iovcnt) { int ret; int valid_iovcnt = -1; struct iovec *iovRet = NULL; /* Process fd convert to system global fd */ int sysfd = GetAssociatedSystemFd(fd); if ((iovcnt < 0) || (iovcnt > IOV_MAX)) { return -EINVAL; } if (iovcnt == 0) { return 0; } if (iov == NULL) { return -EFAULT; } ret = UserIovCopy(&iovRet, iov, iovcnt, &valid_iovcnt); if (ret != 0) { return ret; } if (valid_iovcnt != iovcnt) { ret = -EFAULT; goto OUT_FREE; } ret = writev(sysfd, iovRet, valid_iovcnt); if (ret < 0) { ret = -get_errno(); } OUT_FREE: (void)LOS_MemFree(OS_SYS_MEM_ADDR, iovRet); return ret; } int SysPoll(struct pollfd *fds, nfds_t nfds, int timeout) { int ret; struct pollfd *kfds = NULL; if ((nfds >= MAX_POLL_NFDS) || (nfds == 0) || (fds == NULL)) { return -EINVAL; } kfds = (struct pollfd *)malloc(sizeof(struct pollfd) * nfds); if (kfds != NULL) { if (LOS_ArchCopyFromUser(kfds, fds, sizeof(struct pollfd) * nfds) != 0) { ret = -EFAULT; goto OUT_KFD; } } int *pollFds = NULL; ret = PollfdToSystem(kfds, nfds, &pollFds); if (ret < 0) { ret = -get_errno(); goto OUT_KFD; } ret = poll(kfds, nfds, timeout); if (ret < 0) { ret = -get_errno(); goto OUT; } if (kfds != NULL) { RestorePollfd(kfds, nfds, pollFds); if (LOS_ArchCopyToUser(fds, kfds, sizeof(struct pollfd) * nfds) != 0) { ret = -EFAULT; goto OUT; } } OUT: free(pollFds); OUT_KFD: free(kfds); return ret; } int SysPrctl(int option, ...) { unsigned long name; va_list ap; errno_t err; va_start(ap, option); if (option != PR_SET_NAME) { PRINT_ERR("%s: %d, no support option : 0x%x\n", __FUNCTION__, __LINE__, option); err = EOPNOTSUPP; goto ERROR; } name = va_arg(ap, unsigned long); if (!LOS_IsUserAddress(name)) { err = EFAULT; goto ERROR; } err = OsSetTaskName(OsCurrTaskGet(), (const char *)(uintptr_t)name, TRUE); if (err != LOS_OK) { goto ERROR; } va_end(ap); return ENOERR; ERROR: va_end(ap); return -err; } ssize_t SysPread64(int fd, void *buf, size_t nbytes, off64_t offset) { int ret, retVal; char *bufRet = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); if (nbytes == 0) { ret = pread64(fd, buf, nbytes, offset); if (ret < 0) { return -get_errno(); } else { return ret; } } bufRet = (char *)LOS_MemAlloc(OS_SYS_MEM_ADDR, nbytes); if (bufRet == NULL) { return -ENOMEM; } (void)memset_s(bufRet, nbytes, 0, nbytes); ret = pread64(fd, (buf ? bufRet : NULL), nbytes, offset); if (ret < 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return -get_errno(); } retVal = LOS_ArchCopyToUser(buf, bufRet, ret); if (retVal != 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return -EFAULT; } (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return ret; } ssize_t SysPwrite64(int fd, const void *buf, size_t nbytes, off64_t offset) { int ret; char *bufRet = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); if (nbytes == 0) { ret = pwrite64(fd, buf, nbytes, offset); if (ret < 0) { return -get_errno(); } return ret; } bufRet = (char *)LOS_MemAlloc(OS_SYS_MEM_ADDR, nbytes); if (bufRet == NULL) { return -ENOMEM; } if (buf != NULL) { ret = LOS_ArchCopyFromUser(bufRet, buf, nbytes); if (ret != 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return -EFAULT; } } ret = pwrite64(fd, (buf ? bufRet : NULL), nbytes, offset); if (ret < 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return -get_errno(); } (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return ret; } char *SysGetcwd(char *buf, size_t n) { char *ret = NULL; char *bufRet = NULL; size_t bufLen = n; int retVal; if (bufLen > PATH_MAX) { bufLen = PATH_MAX; } bufRet = (char *)LOS_MemAlloc(OS_SYS_MEM_ADDR, bufLen); if (bufRet == NULL) { return (char *)(intptr_t)-ENOMEM; } (void)memset_s(bufRet, bufLen, 0, bufLen); ret = getcwd((buf ? bufRet : NULL), bufLen); if (ret == NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return (char *)(intptr_t)-get_errno(); } retVal = LOS_ArchCopyToUser(buf, bufRet, bufLen); if (retVal != 0) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return (char *)(intptr_t)-EFAULT; } ret = buf; (void)LOS_MemFree(OS_SYS_MEM_ADDR, bufRet); return ret; } ssize_t SysSendFile(int outfd, int infd, off_t *offset, size_t count) { int ret, retVal; off_t offsetRet; retVal = LOS_ArchCopyFromUser(&offsetRet, offset, sizeof(off_t)); if (retVal != 0) { return -EFAULT; } /* Process fd convert to system global fd */ outfd = GetAssociatedSystemFd(outfd); infd = GetAssociatedSystemFd(infd); ret = sendfile(outfd, infd, (offset ? (&offsetRet) : NULL), count); if (ret < 0) { return -get_errno(); } retVal = LOS_ArchCopyToUser(offset, &offsetRet, sizeof(off_t)); if (retVal != 0) { return -EFAULT; } return ret; } int SysFtruncate64(int fd, off64_t length) { int ret; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = ftruncate64(fd, length); if (ret < 0) { return -get_errno(); } return ret; } int SysOpenat(int dirfd, const char *path, int oflags, ...) { int ret; int procFd; char *pathRet = NULL; mode_t mode; #ifdef LOSCFG_FILE_MODE va_list ap; va_start(ap, oflags); mode = va_arg(ap, int); va_end(ap); #else mode = 0666; /* 0666: File read-write properties. */ #endif if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { return ret; } } procFd = AllocProcessFd(); if (procFd < 0) { ret = -EMFILE; goto ERROUT; } if (oflags & O_CLOEXEC) { SetCloexecFlag(procFd); } if (dirfd != AT_FDCWD) { /* Process fd convert to system global fd */ dirfd = GetAssociatedSystemFd(dirfd); } ret = do_open(dirfd, (path ? pathRet : NULL), oflags, mode); if (ret < 0) { ret = -get_errno(); goto ERROUT; } AssociateSystemFd(procFd, ret); if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return procFd; ERROUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } if (procFd >= 0) { FreeProcessFd(procFd); } return ret; } int SysMkdirat(int dirfd, const char *pathname, mode_t mode) { int ret; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } if (dirfd != AT_FDCWD) { /* Process fd convert to system global fd */ dirfd = GetAssociatedSystemFd(dirfd); } ret = do_mkdir(dirfd, (pathname ? pathRet : NULL), mode); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysLinkat(int olddirfd, const char *oldpath, int newdirfd, const char *newpath, int flags) { int ret; char *oldpathRet = NULL; char *newpathRet = NULL; if (oldpath != NULL) { ret = UserPathCopy(oldpath, &oldpathRet); if (ret != 0) { goto OUT; } } if (newpath != NULL) { ret = UserPathCopy(newpath, &newpathRet); if (ret != 0) { goto OUT; } } if (olddirfd != AT_FDCWD) { /* Process fd convert to system global fd */ olddirfd = GetAssociatedSystemFd(olddirfd); } if (newdirfd != AT_FDCWD) { /* Process fd convert to system global fd */ newdirfd = GetAssociatedSystemFd(newdirfd); } ret = linkat(olddirfd, oldpathRet, newdirfd, newpathRet, flags); if (ret < 0) { ret = -get_errno(); } OUT: if (oldpathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, oldpathRet); } if (newpathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, newpathRet); } return ret; } int SysSymlinkat(const char *target, int dirfd, const char *linkpath) { int ret; char *pathRet = NULL; char *targetRet = NULL; if (target != NULL) { ret = UserPathCopy(target, &targetRet); if (ret != 0) { goto OUT; } } if (linkpath != NULL) { ret = UserPathCopy(linkpath, &pathRet); if (ret != 0) { goto OUT; } } if (dirfd != AT_FDCWD) { /* Process fd convert to system global fd */ dirfd = GetAssociatedSystemFd(dirfd); } ret = symlinkat(targetRet, dirfd, pathRet); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } if (targetRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, targetRet); } return ret; } ssize_t SysReadlinkat(int dirfd, const char *pathname, char *buf, size_t bufsize) { ssize_t ret; char *pathRet = NULL; if (bufsize == 0) { return -EINVAL; } if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } if (dirfd != AT_FDCWD) { /* Process fd convert to system global fd */ dirfd = GetAssociatedSystemFd(dirfd); } if (!LOS_IsUserAddressRange((vaddr_t)(UINTPTR)buf, bufsize)) { ret = -EFAULT; goto OUT; } ret = readlinkat(dirfd, pathRet, buf, bufsize); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysUnlinkat(int dirfd, const char *pathname, int flag) { int ret; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } if (dirfd != AT_FDCWD) { /* Process fd convert to system global fd */ dirfd = GetAssociatedSystemFd(dirfd); } ret = unlinkat(dirfd, (pathname ? pathRet : NULL), flag); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysRenameat(int oldfd, const char *oldpath, int newdfd, const char *newpath) { int ret; char *pathOldRet = NULL; char *pathNewRet = NULL; if (oldpath != NULL) { ret = UserPathCopy(oldpath, &pathOldRet); if (ret != 0) { goto OUT; } } if (newpath != NULL) { ret = UserPathCopy(newpath, &pathNewRet); if (ret != 0) { goto OUT; } } if (oldfd != AT_FDCWD) { /* Process fd convert to system global fd */ oldfd = GetAssociatedSystemFd(oldfd); } if (newdfd != AT_FDCWD) { /* Process fd convert to system global fd */ newdfd = GetAssociatedSystemFd(newdfd); } ret = do_rename(oldfd, (oldpath ? pathOldRet : NULL), newdfd, (newpath ? pathNewRet : NULL)); if (ret < 0) { ret = -get_errno(); } OUT: if (pathOldRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathOldRet); } if (pathNewRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathNewRet); } return ret; } int SysFallocate(int fd, int mode, off_t offset, off_t len) { int ret; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = fallocate(fd, mode, offset, len); if (ret < 0) { return -get_errno(); } return ret; } int SysFallocate64(int fd, int mode, off64_t offset, off64_t len) { int ret; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = fallocate64(fd, mode, offset, len); if (ret < 0) { return -get_errno(); } return ret; } ssize_t SysPreadv(int fd, const struct iovec *iov, int iovcnt, long loffset, long hoffset) { off_t offsetflag; offsetflag = (off_t)((unsigned long long)loffset | (((unsigned long long)hoffset) << HIGH_SHIFT_BIT)); int ret; int valid_iovcnt = -1; struct iovec *iovRet = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); if ((iov == NULL) || (iovcnt < 0) || (iovcnt > IOV_MAX)) { return -EINVAL; } if (iovcnt == 0) { return 0; } ret = UserIovCopy(&iovRet, iov, iovcnt, &valid_iovcnt); if (ret != 0) { return ret; } if (valid_iovcnt <= 0) { ret = -EFAULT; goto OUT_FREE; } ret = preadv(fd, iovRet, valid_iovcnt, offsetflag); if (ret < 0) { ret = -get_errno(); } OUT_FREE: (void)(void)LOS_MemFree(OS_SYS_MEM_ADDR, iovRet); return ret; } ssize_t SysPwritev(int fd, const struct iovec *iov, int iovcnt, long loffset, long hoffset) { off_t offsetflag; offsetflag = (off_t)((unsigned long long)loffset | (((unsigned long long)hoffset) << HIGH_SHIFT_BIT)); int ret; int valid_iovcnt = -1; struct iovec *iovRet = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); if ((iov == NULL) || (iovcnt < 0) || (iovcnt > IOV_MAX)) { return -EINVAL; } if (iovcnt == 0) { return 0; } ret = UserIovCopy(&iovRet, iov, iovcnt, &valid_iovcnt); if (ret != 0) { return ret; } if (valid_iovcnt != iovcnt) { ret = -EFAULT; goto OUT_FREE; } ret = pwritev(fd, iovRet, valid_iovcnt, offsetflag); if (ret < 0) { ret = -get_errno(); } OUT_FREE: (void)LOS_MemFree(OS_SYS_MEM_ADDR, iovRet); return ret; } #ifdef LOSCFG_FS_FAT int SysFormat(const char *dev, int sectors, int option) { int ret; char *devRet = NULL; if (!IsCapPermit(CAP_FS_FORMAT)) { return -EPERM; } if (dev != NULL) { ret = UserPathCopy(dev, &devRet); if (ret != 0) { goto OUT; } } ret = format((dev ? devRet : NULL), sectors, option); if (ret < 0) { ret = -get_errno(); } OUT: if (devRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, devRet); } return ret; } #endif int SysFstat64(int fd, struct kstat *buf) { int ret; struct stat64 bufRet = {0}; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = fstat64(fd, (buf ? (&bufRet) : NULL)); if (ret < 0) { return -get_errno(); } ret = LOS_ArchCopyToUser(buf, &bufRet, sizeof(struct kstat)); if (ret != 0) { return -EFAULT; } return ret; } int SysFcntl64(int fd, int cmd, void *arg) { /* Process fd convert to system global fd */ int sysfd = GetAssociatedSystemFd(fd); int ret = VfsFcntl(fd, cmd, arg); if (ret == CONTINE_NUTTX_FCNTL) { ret = fcntl64(sysfd, cmd, arg); } if (ret < 0) { return -get_errno(); } return ret; } int SysGetdents64(int fd, struct dirent *de_user, unsigned int count) { if (!LOS_IsUserAddressRange((VADDR_T)(UINTPTR)de_user, count)) { return -EFAULT; } struct dirent *de_knl = NULL; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); int ret = do_readdir(fd, &de_knl, count); if (ret < 0) { return ret; } if (de_knl != NULL) { int cpy_ret = LOS_ArchCopyToUser(de_user, de_knl, ret); if (cpy_ret != 0) { return -EFAULT; } } return ret; } char *SysRealpath(const char *path, char *resolved_path) { char *pathRet = NULL; char *resolved_pathRet = NULL; char *result = NULL; int ret; if (resolved_path == NULL) { return (char *)(intptr_t)-EINVAL; } if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { result = (char *)(intptr_t)ret; goto OUT; } } resolved_pathRet = realpath((path ? pathRet : NULL), NULL); if (resolved_pathRet == NULL) { result = (char *)(intptr_t)-get_errno(); goto OUT; } ret = LOS_ArchCopyToUser(resolved_path, resolved_pathRet, strlen(resolved_pathRet) + 1); if (ret != 0) { result = (char *)(intptr_t)-EFAULT; goto OUT; } result = resolved_path; OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } if (resolved_pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, resolved_pathRet); } return result; } int SysUtimensat(int fd, const char *path, struct timespec times[TIMESPEC_TIMES_NUM], int flag) { int ret; int timeLen; struct IATTR attr = {0}; char *filePath = NULL; timeLen = TIMESPEC_TIMES_NUM * sizeof(struct timespec); CHECK_ASPACE(times, timeLen); DUP_FROM_USER(times, timeLen); ret = CheckNewAttrTime(&attr, times); FREE_DUP(times); if (ret < 0) { goto OUT; } ret = GetFullpathNull(fd, path, &filePath); if (ret < 0) { goto OUT; } ret = chattr(filePath, &attr); if (ret < 0) { ret = -get_errno(); } OUT: PointerFree(filePath); return ret; } int SysChmod(const char *pathname, mode_t mode) { int ret; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } ret = chmod(pathRet, mode); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysFchmodat(int fd, const char *path, mode_t mode, int flag) { int ret; char *pathRet = NULL; char *fullpath = NULL; struct IATTR attr = { .attr_chg_mode = mode, .attr_chg_valid = CHG_MODE, }; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } if (fd != AT_FDCWD) { /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); } ret = vfs_normalize_pathat(fd, pathRet, &fullpath); if (ret < 0) { goto OUT; } ret = chattr(fullpath, &attr); if (ret < 0) { ret = -get_errno(); } OUT: PointerFree(pathRet); PointerFree(fullpath); return ret; } int SysFchmod(int fd, mode_t mode) { int ret; int sysFd; struct IATTR attr = { .attr_chg_mode = mode, .attr_chg_valid = CHG_MODE, /* change mode */ }; struct file *file = NULL; sysFd = GetAssociatedSystemFd(fd); if (sysFd < 0) { return -EBADF; } ret = fs_getfilep(sysFd, &file); if (ret < 0) { return -get_errno(); } ret = chattr(file->f_path, &attr); if (ret < 0) { return -get_errno(); } return ret; } int SysFchownat(int fd, const char *path, uid_t owner, gid_t group, int flag) { int ret; char *fullpath = NULL; struct IATTR attr = { .attr_chg_valid = 0, }; ret = GetFullpath(fd, path, &fullpath); if (ret < 0) { goto OUT; } if (owner != (uid_t)-1) { attr.attr_chg_uid = owner; attr.attr_chg_valid |= CHG_UID; } if (group != (gid_t)-1) { attr.attr_chg_gid = group; attr.attr_chg_valid |= CHG_GID; } ret = chattr(fullpath, &attr); if (ret < 0) { ret = -get_errno(); } OUT: PointerFree(fullpath); return ret; } int SysFchown(int fd, uid_t owner, gid_t group) { int ret; int sysFd; struct IATTR attr = {0}; attr.attr_chg_valid = 0; struct file *file = NULL; sysFd = GetAssociatedSystemFd(fd); if (sysFd < 0) { return -EBADF; } ret = fs_getfilep(sysFd, &file); if (ret < 0) { return -get_errno(); } if (owner != (uid_t)-1) { attr.attr_chg_uid = owner; attr.attr_chg_valid |= CHG_UID; } if (group != (gid_t)-1) { attr.attr_chg_gid = group; attr.attr_chg_valid |= CHG_GID; } ret = chattr(file->f_path, &attr); if (ret < 0) { ret = -get_errno(); } return ret; } int SysChown(const char *pathname, uid_t owner, gid_t group) { int ret; char *pathRet = NULL; if (pathname != NULL) { ret = UserPathCopy(pathname, &pathRet); if (ret != 0) { goto OUT; } } ret = chown(pathRet, owner, group); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } int SysFstatat64(int dirfd, const char *restrict path, struct kstat *restrict buf, int flag) { int ret; struct stat bufRet = {0}; char *pathRet = NULL; char *fullpath = NULL; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } if (dirfd != AT_FDCWD) { /* Process fd convert to system global fd */ dirfd = GetAssociatedSystemFd(dirfd); } ret = vfs_normalize_pathat(dirfd, pathRet, &fullpath); if (ret < 0) { goto OUT; } ret = stat(fullpath, &bufRet); if (ret < 0) { ret = -get_errno(); goto OUT; } ret = LOS_ArchCopyToUser(buf, &bufRet, sizeof(struct kstat)); if (ret != 0) { ret = -EFAULT; goto OUT; } OUT: if (pathRet != NULL) { LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } if (fullpath != NULL) { free(fullpath); } return ret; } int SysFaccessat(int fd, const char *filename, int amode, int flag) { int ret; struct stat buf; struct statfs fsBuf; char *fullDirectory = NULL; ret = GetFullpath(fd, filename, &fullDirectory); if (ret < 0) { goto OUT; } ret = statfs(fullDirectory, &fsBuf); if (ret != 0) { ret = -get_errno(); goto OUT; } if ((fsBuf.f_flags & MS_RDONLY) && ((unsigned int)amode & W_OK)) { ret = -EROFS; goto OUT; } ret = stat(fullDirectory, &buf); if (ret != 0) { ret = -get_errno(); goto OUT; } if (VfsPermissionCheck(buf.st_uid, buf.st_gid, buf.st_mode, amode)) { ret = -EACCES; } OUT: PointerFree(fullDirectory); return ret; } int SysFstatfs(int fd, struct statfs *buf) { int ret; struct file *filep = NULL; struct statfs bufRet = {0}; /* Process fd convert to system global fd */ fd = GetAssociatedSystemFd(fd); ret = fs_getfilep(fd, &filep); if (ret < 0) { ret = -get_errno(); return ret; } ret = statfs(filep->f_path, &bufRet); if (ret < 0) { ret = -get_errno(); return ret; } ret = LOS_ArchCopyToUser(buf, &bufRet, sizeof(struct statfs)); if (ret != 0) { ret = -EFAULT; } return ret; } int SysFstatfs64(int fd, size_t sz, struct statfs *buf) { int ret; if (sz != sizeof(struct statfs)) { ret = -EINVAL; return ret; } ret = SysFstatfs(fd, buf); return ret; } int SysPpoll(struct pollfd *fds, nfds_t nfds, const struct timespec *tmo_p, const sigset_t *sigMask, int nsig) { int timeout, retVal; sigset_t_l origMask = {0}; sigset_t_l set = {0}; CHECK_ASPACE(tmo_p, sizeof(struct timespec)); CPY_FROM_USER(tmo_p); if (tmo_p != NULL) { timeout = tmo_p->tv_sec * OS_SYS_US_PER_MS + tmo_p->tv_nsec / OS_SYS_NS_PER_MS; if (timeout < 0) { return -EINVAL; } } else { timeout = -1; } if (sigMask != NULL) { retVal = LOS_ArchCopyFromUser(&set, sigMask, sizeof(sigset_t)); if (retVal != 0) { return -EFAULT; } (VOID)OsSigprocMask(SIG_SETMASK, &set, &origMask); } else { (VOID)OsSigprocMask(SIG_SETMASK, NULL, &origMask); } retVal = SysPoll(fds, nfds, timeout); (VOID)OsSigprocMask(SIG_SETMASK, &origMask, NULL); return retVal; } int SysPselect6(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, const struct timespec *timeout, const long data[2]) { int ret; int retVal; sigset_t_l origMask; sigset_t_l setl; CHECK_ASPACE(readfds, sizeof(fd_set)); CHECK_ASPACE(writefds, sizeof(fd_set)); CHECK_ASPACE(exceptfds, sizeof(fd_set)); CHECK_ASPACE(timeout, sizeof(struct timeval)); CPY_FROM_USER(readfds); CPY_FROM_USER(writefds); CPY_FROM_USER(exceptfds); DUP_FROM_USER(timeout, sizeof(struct timeval)); if (timeout != NULL) { ((struct timeval *)timeout)->tv_usec = timeout->tv_nsec / 1000; /* 1000, convert ns to us */ } if (data != NULL) { retVal = LOS_ArchCopyFromUser(&(setl.sig[0]), (int *)((UINTPTR)data[0]), sizeof(sigset_t)); if (retVal != 0) { ret = -EFAULT; FREE_DUP(timeout); return ret; } } OsSigprocMask(SIG_SETMASK, &setl, &origMask); ret = do_select(nfds, readfds, writefds, exceptfds, (struct timeval *)timeout, UserPoll); if (ret < 0) { /* do not copy parameter back to user mode if do_select failed */ ret = -get_errno(); FREE_DUP(timeout); return ret; } OsSigprocMask(SIG_SETMASK, &origMask, NULL); CPY_TO_USER(readfds); CPY_TO_USER(writefds); CPY_TO_USER(exceptfds); FREE_DUP(timeout); return ret; } static int DoEpollCreate1(int flags) { int ret; int procFd; ret = epoll_create1(flags); if (ret < 0) { ret = -get_errno(); return ret; } procFd = AllocAndAssocProcessFd((INTPTR)(ret), MIN_START_FD); if (procFd == -1) { epoll_close(ret); return -EMFILE; } return procFd; } int SysEpollCreate(int size) { (void)size; return DoEpollCreate1(0); } int SysEpollCreate1(int flags) { return DoEpollCreate1(flags); } int SysEpollCtl(int epfd, int op, int fd, struct epoll_event *ev) { int ret; CHECK_ASPACE(ev, sizeof(struct epoll_event)); CPY_FROM_USER(ev); fd = GetAssociatedSystemFd(fd); epfd = GetAssociatedSystemFd(epfd); if ((fd < 0) || (epfd < 0)) { ret = -EBADF; goto OUT; } ret = epoll_ctl(epfd, op, fd, ev); if (ret < 0) { ret = -EBADF; goto OUT; } CPY_TO_USER(ev); OUT: return (ret == -1) ? -get_errno() : ret; } int SysEpollWait(int epfd, struct epoll_event *evs, int maxevents, int timeout) { int ret = 0; CHECK_ASPACE(evs, sizeof(struct epoll_event)); CPY_FROM_USER(evs); epfd = GetAssociatedSystemFd(epfd); if (epfd < 0) { ret = -EBADF; goto OUT; } ret = epoll_wait(epfd, evs, maxevents, timeout); if (ret < 0) { ret = -get_errno(); } CPY_TO_USER(evs); OUT: return (ret == -1) ? -get_errno() : ret; } int SysEpollPwait(int epfd, struct epoll_event *evs, int maxevents, int timeout, const sigset_t *mask) { sigset_t_l origMask; sigset_t_l setl; int ret = 0; CHECK_ASPACE(mask, sizeof(sigset_t)); if (mask != NULL) { ret = LOS_ArchCopyFromUser(&setl, mask, sizeof(sigset_t)); if (ret != 0) { return -EFAULT; } } CHECK_ASPACE(evs, sizeof(struct epoll_event)); CPY_FROM_USER(evs); epfd = GetAssociatedSystemFd(epfd); if (epfd < 0) { ret = -EBADF; goto OUT; } OsSigprocMask(SIG_SETMASK, &setl, &origMask); ret = epoll_wait(epfd, evs, maxevents, timeout); if (ret < 0) { ret = -get_errno(); } OsSigprocMask(SIG_SETMASK, &origMask, NULL); CPY_TO_USER(evs); OUT: return (ret == -1) ? -get_errno() : ret; } #ifdef LOSCFG_CHROOT int SysChroot(const char *path) { int ret; char *pathRet = NULL; if (path != NULL) { ret = UserPathCopy(path, &pathRet); if (ret != 0) { goto OUT; } } ret = chroot(path ? pathRet : NULL); if (ret < 0) { ret = -get_errno(); } OUT: if (pathRet != NULL) { (void)LOS_MemFree(OS_SYS_MEM_ADDR, pathRet); } return ret; } #endif #endif