/* $NetBSD: sys_generic.c,v 1.99 2007/02/17 22:31:44 pavel Exp $ */ /* * Copyright (c) 1982, 1986, 1989, 1993 * The Regents of the University of California. All rights reserved. * (c) UNIX System Laboratories, Inc. * All or some portions of this file are derived from material licensed * to the University of California by American Telephone and Telegraph * Co. or Unix System Laboratories, Inc. and are reproduced herein with * the permission of UNIX System Laboratories, Inc. * * 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 University 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 REGENTS 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 REGENTS 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. * * @(#)sys_generic.c 8.9 (Berkeley) 2/14/95 */ #include __KERNEL_RCSID(0, "$NetBSD: sys_generic.c,v 1.99 2007/02/17 22:31:44 pavel Exp $"); #include "opt_ktrace.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef KTRACE #include #endif #include #include #include int selscan(struct lwp *, fd_mask *, fd_mask *, int, register_t *); int pollscan(struct lwp *, struct pollfd *, int, register_t *); /* * Read system call. */ /* ARGSUSED */ int sys_read(struct lwp *l, void *v, register_t *retval) { struct sys_read_args /* { syscallarg(int) fd; syscallarg(void *) buf; syscallarg(size_t) nbyte; } */ *uap = v; int fd; struct file *fp; struct proc *p; struct filedesc *fdp; fd = SCARG(uap, fd); p = l->l_proc; fdp = p->p_fd; if ((fp = fd_getfile(fdp, fd)) == NULL) return (EBADF); if ((fp->f_flag & FREAD) == 0) { simple_unlock(&fp->f_slock); return (EBADF); } FILE_USE(fp); /* dofileread() will unuse the descriptor for us */ return (dofileread(l, fd, fp, SCARG(uap, buf), SCARG(uap, nbyte), &fp->f_offset, FOF_UPDATE_OFFSET, retval)); } int dofileread(struct lwp *l, int fd, struct file *fp, void *buf, size_t nbyte, off_t *offset, int flags, register_t *retval) { struct iovec aiov; struct uio auio; struct proc *p; struct vmspace *vm; size_t cnt; int error; #ifdef KTRACE struct iovec ktriov = { .iov_base = NULL, }; #endif p = l->l_proc; error = proc_vmspace_getref(p, &vm); if (error) { goto out; } aiov.iov_base = (caddr_t)buf; aiov.iov_len = nbyte; auio.uio_iov = &aiov; auio.uio_iovcnt = 1; auio.uio_resid = nbyte; auio.uio_rw = UIO_READ; auio.uio_vmspace = vm; /* * Reads return ssize_t because -1 is returned on error. Therefore * we must restrict the length to SSIZE_MAX to avoid garbage return * values. */ if (auio.uio_resid > SSIZE_MAX) { error = EINVAL; goto out; } #ifdef KTRACE /* * if tracing, save a copy of iovec */ if (KTRPOINT(p, KTR_GENIO)) ktriov = aiov; #endif cnt = auio.uio_resid; error = (*fp->f_ops->fo_read)(fp, offset, &auio, fp->f_cred, flags); if (error) if (auio.uio_resid != cnt && (error == ERESTART || error == EINTR || error == EWOULDBLOCK)) error = 0; cnt -= auio.uio_resid; #ifdef KTRACE if (KTRPOINT(p, KTR_GENIO) && error == 0) ktrgenio(l, fd, UIO_READ, &ktriov, cnt, error); #endif *retval = cnt; out: FILE_UNUSE(fp, l); uvmspace_free(vm); return (error); } /* * Scatter read system call. */ int sys_readv(struct lwp *l, void *v, register_t *retval) { struct sys_readv_args /* { syscallarg(int) fd; syscallarg(const struct iovec *) iovp; syscallarg(int) iovcnt; } */ *uap = v; struct filedesc *fdp; struct file *fp; struct proc *p; int fd; fd = SCARG(uap, fd); p = l->l_proc; fdp = p->p_fd; if ((fp = fd_getfile(fdp, fd)) == NULL) return (EBADF); if ((fp->f_flag & FREAD) == 0) { simple_unlock(&fp->f_slock); return (EBADF); } FILE_USE(fp); /* dofilereadv() will unuse the descriptor for us */ return (dofilereadv(l, fd, fp, SCARG(uap, iovp), SCARG(uap, iovcnt), &fp->f_offset, FOF_UPDATE_OFFSET, retval)); } int dofilereadv(struct lwp *l, int fd, struct file *fp, const struct iovec *iovp, int iovcnt, off_t *offset, int flags, register_t *retval) { struct proc *p; struct uio auio; struct iovec *iov, *needfree, aiov[UIO_SMALLIOV]; struct vmspace *vm; int i, error; size_t cnt; u_int iovlen; #ifdef KTRACE struct iovec *ktriov; #endif p = l->l_proc; error = proc_vmspace_getref(p, &vm); if (error) { goto out; } #ifdef KTRACE ktriov = NULL; #endif /* note: can't use iovlen until iovcnt is validated */ iovlen = iovcnt * sizeof(struct iovec); if ((u_int)iovcnt > UIO_SMALLIOV) { if ((u_int)iovcnt > IOV_MAX) { error = EINVAL; goto out; } iov = malloc(iovlen, M_IOV, M_WAITOK); needfree = iov; } else if ((u_int)iovcnt > 0) { iov = aiov; needfree = NULL; } else { error = EINVAL; goto out; } auio.uio_iov = iov; auio.uio_iovcnt = iovcnt; auio.uio_rw = UIO_READ; auio.uio_vmspace = vm; error = copyin(iovp, iov, iovlen); if (error) goto done; auio.uio_resid = 0; for (i = 0; i < iovcnt; i++) { auio.uio_resid += iov->iov_len; /* * Reads return ssize_t because -1 is returned on error. * Therefore we must restrict the length to SSIZE_MAX to * avoid garbage return values. */ if (iov->iov_len > SSIZE_MAX || auio.uio_resid > SSIZE_MAX) { error = EINVAL; goto done; } iov++; } #ifdef KTRACE /* * if tracing, save a copy of iovec */ if (KTRPOINT(p, KTR_GENIO)) { ktriov = malloc(iovlen, M_TEMP, M_WAITOK); memcpy((caddr_t)ktriov, (caddr_t)auio.uio_iov, iovlen); } #endif cnt = auio.uio_resid; error = (*fp->f_ops->fo_read)(fp, offset, &auio, fp->f_cred, flags); if (error) if (auio.uio_resid != cnt && (error == ERESTART || error == EINTR || error == EWOULDBLOCK)) error = 0; cnt -= auio.uio_resid; #ifdef KTRACE if (ktriov != NULL) { if (KTRPOINT(p, KTR_GENIO) && (error == 0)) ktrgenio(l, fd, UIO_READ, ktriov, cnt, error); free(ktriov, M_TEMP); } #endif *retval = cnt; done: if (needfree) free(needfree, M_IOV); out: FILE_UNUSE(fp, l); uvmspace_free(vm); return (error); } /* * Write system call */ int sys_write(struct lwp *l, void *v, register_t *retval) { struct sys_write_args /* { syscallarg(int) fd; syscallarg(const void *) buf; syscallarg(size_t) nbyte; } */ *uap = v; int fd; struct file *fp; struct proc *p; struct filedesc *fdp; fd = SCARG(uap, fd); p = l->l_proc; fdp = p->p_fd; if ((fp = fd_getfile(fdp, fd)) == NULL) return (EBADF); if ((fp->f_flag & FWRITE) == 0) { simple_unlock(&fp->f_slock); return (EBADF); } FILE_USE(fp); /* dofilewrite() will unuse the descriptor for us */ return (dofilewrite(l, fd, fp, SCARG(uap, buf), SCARG(uap, nbyte), &fp->f_offset, FOF_UPDATE_OFFSET, retval)); } int dofilewrite(struct lwp *l, int fd, struct file *fp, const void *buf, size_t nbyte, off_t *offset, int flags, register_t *retval) { struct iovec aiov; struct uio auio; struct proc *p; struct vmspace *vm; size_t cnt; int error; #ifdef KTRACE struct iovec ktriov = { .iov_base = NULL, }; #endif p = l->l_proc; error = proc_vmspace_getref(p, &vm); if (error) { goto out; } aiov.iov_base = __UNCONST(buf); /* XXXUNCONST kills const */ aiov.iov_len = nbyte; auio.uio_iov = &aiov; auio.uio_iovcnt = 1; auio.uio_resid = nbyte; auio.uio_rw = UIO_WRITE; auio.uio_vmspace = vm; /* * Writes return ssize_t because -1 is returned on error. Therefore * we must restrict the length to SSIZE_MAX to avoid garbage return * values. */ if (auio.uio_resid > SSIZE_MAX) { error = EINVAL; goto out; } #ifdef KTRACE /* * if tracing, save a copy of iovec */ if (KTRPOINT(p, KTR_GENIO)) ktriov = aiov; #endif cnt = auio.uio_resid; error = (*fp->f_ops->fo_write)(fp, offset, &auio, fp->f_cred, flags); if (error) { if (auio.uio_resid != cnt && (error == ERESTART || error == EINTR || error == EWOULDBLOCK)) error = 0; if (error == EPIPE) { mutex_enter(&proclist_mutex); psignal(p, SIGPIPE); mutex_exit(&proclist_mutex); } } cnt -= auio.uio_resid; #ifdef KTRACE if (KTRPOINT(p, KTR_GENIO) && error == 0) ktrgenio(l, fd, UIO_WRITE, &ktriov, cnt, error); #endif *retval = cnt; out: FILE_UNUSE(fp, l); uvmspace_free(vm); return (error); } /* * Gather write system call */ int sys_writev(struct lwp *l, void *v, register_t *retval) { struct sys_writev_args /* { syscallarg(int) fd; syscallarg(const struct iovec *) iovp; syscallarg(int) iovcnt; } */ *uap = v; int fd; struct file *fp; struct proc *p; struct filedesc *fdp; fd = SCARG(uap, fd); p = l->l_proc; fdp = p->p_fd; if ((fp = fd_getfile(fdp, fd)) == NULL) return (EBADF); if ((fp->f_flag & FWRITE) == 0) { simple_unlock(&fp->f_slock); return (EBADF); } FILE_USE(fp); /* dofilewritev() will unuse the descriptor for us */ return (dofilewritev(l, fd, fp, SCARG(uap, iovp), SCARG(uap, iovcnt), &fp->f_offset, FOF_UPDATE_OFFSET, retval)); } int dofilewritev(struct lwp *l, int fd, struct file *fp, const struct iovec *iovp, int iovcnt, off_t *offset, int flags, register_t *retval) { struct proc *p; struct uio auio; struct iovec *iov, *needfree, aiov[UIO_SMALLIOV]; struct vmspace *vm; int i, error; size_t cnt; u_int iovlen; #ifdef KTRACE struct iovec *ktriov; #endif p = l->l_proc; error = proc_vmspace_getref(p, &vm); if (error) { goto out; } #ifdef KTRACE ktriov = NULL; #endif /* note: can't use iovlen until iovcnt is validated */ iovlen = iovcnt * sizeof(struct iovec); if ((u_int)iovcnt > UIO_SMALLIOV) { if ((u_int)iovcnt > IOV_MAX) { error = EINVAL; goto out; } iov = malloc(iovlen, M_IOV, M_WAITOK); needfree = iov; } else if ((u_int)iovcnt > 0) { iov = aiov; needfree = NULL; } else { error = EINVAL; goto out; } auio.uio_iov = iov; auio.uio_iovcnt = iovcnt; auio.uio_rw = UIO_WRITE; auio.uio_vmspace = vm; error = copyin(iovp, iov, iovlen); if (error) goto done; auio.uio_resid = 0; for (i = 0; i < iovcnt; i++) { auio.uio_resid += iov->iov_len; /* * Writes return ssize_t because -1 is returned on error. * Therefore we must restrict the length to SSIZE_MAX to * avoid garbage return values. */ if (iov->iov_len > SSIZE_MAX || auio.uio_resid > SSIZE_MAX) { error = EINVAL; goto done; } iov++; } #ifdef KTRACE /* * if tracing, save a copy of iovec */ if (KTRPOINT(p, KTR_GENIO)) { ktriov = malloc(iovlen, M_TEMP, M_WAITOK); memcpy((caddr_t)ktriov, (caddr_t)auio.uio_iov, iovlen); } #endif cnt = auio.uio_resid; error = (*fp->f_ops->fo_write)(fp, offset, &auio, fp->f_cred, flags); if (error) { if (auio.uio_resid != cnt && (error == ERESTART || error == EINTR || error == EWOULDBLOCK)) error = 0; if (error == EPIPE) { mutex_enter(&proclist_mutex); psignal(p, SIGPIPE); mutex_exit(&proclist_mutex); } } cnt -= auio.uio_resid; #ifdef KTRACE if (ktriov != NULL) { if (KTRPOINT(p, KTR_GENIO) && (error == 0)) ktrgenio(l, fd, UIO_WRITE, ktriov, cnt, error); free(ktriov, M_TEMP); } #endif *retval = cnt; done: if (needfree) free(needfree, M_IOV); out: FILE_UNUSE(fp, l); uvmspace_free(vm); return (error); } /* * Ioctl system call */ /* ARGSUSED */ int sys_ioctl(struct lwp *l, void *v, register_t *retval) { struct sys_ioctl_args /* { syscallarg(int) fd; syscallarg(u_long) com; syscallarg(caddr_t) data; } */ *uap = v; struct file *fp; struct proc *p; struct filedesc *fdp; u_long com; int error; u_int size; caddr_t data, memp; #define STK_PARAMS 128 u_long stkbuf[STK_PARAMS/sizeof(u_long)]; error = 0; p = l->l_proc; fdp = p->p_fd; if ((fp = fd_getfile(fdp, SCARG(uap, fd))) == NULL) return (EBADF); FILE_USE(fp); if ((fp->f_flag & (FREAD | FWRITE)) == 0) { error = EBADF; com = 0; goto out; } switch (com = SCARG(uap, com)) { case FIONCLEX: fdp->fd_ofileflags[SCARG(uap, fd)] &= ~UF_EXCLOSE; goto out; case FIOCLEX: fdp->fd_ofileflags[SCARG(uap, fd)] |= UF_EXCLOSE; goto out; } /* * Interpret high order word to find amount of data to be * copied to/from the user's address space. */ size = IOCPARM_LEN(com); if (size > IOCPARM_MAX) { error = ENOTTY; goto out; } memp = NULL; if (size > sizeof(stkbuf)) { memp = (caddr_t)malloc((u_long)size, M_IOCTLOPS, M_WAITOK); data = memp; } else data = (caddr_t)stkbuf; if (com&IOC_IN) { if (size) { error = copyin(SCARG(uap, data), data, size); if (error) { if (memp) free(memp, M_IOCTLOPS); goto out; } #ifdef KTRACE if (KTRPOINT(p, KTR_GENIO)) { struct iovec iov; iov.iov_base = SCARG(uap, data); iov.iov_len = size; ktrgenio(l, SCARG(uap, fd), UIO_WRITE, &iov, size, 0); } #endif } else *(caddr_t *)data = SCARG(uap, data); } else if ((com&IOC_OUT) && size) /* * Zero the buffer so the user always * gets back something deterministic. */ memset(data, 0, size); else if (com&IOC_VOID) *(caddr_t *)data = SCARG(uap, data); switch (com) { case FIONBIO: if (*(int *)data != 0) fp->f_flag |= FNONBLOCK; else fp->f_flag &= ~FNONBLOCK; error = (*fp->f_ops->fo_ioctl)(fp, FIONBIO, data, l); break; case FIOASYNC: if (*(int *)data != 0) fp->f_flag |= FASYNC; else fp->f_flag &= ~FASYNC; error = (*fp->f_ops->fo_ioctl)(fp, FIOASYNC, data, l); break; default: error = (*fp->f_ops->fo_ioctl)(fp, com, data, l); /* * Copy any data to user, size was * already set and checked above. */ if (error == 0 && (com&IOC_OUT) && size) { error = copyout(data, SCARG(uap, data), size); #ifdef KTRACE if (KTRPOINT(p, KTR_GENIO)) { struct iovec iov; iov.iov_base = SCARG(uap, data); iov.iov_len = size; ktrgenio(l, SCARG(uap, fd), UIO_READ, &iov, size, error); } #endif } break; } if (memp) free(memp, M_IOCTLOPS); out: FILE_UNUSE(fp, l); switch (error) { case -1: printf("sys_ioctl: _IO%s%s('%c', %lu, %lu) returned -1: " "pid=%d comm=%s\n", (com & IOC_IN) ? "W" : "", (com & IOC_OUT) ? "R" : "", (char)IOCGROUP(com), (com & 0xff), IOCPARM_LEN(com), p->p_pid, p->p_comm); /* FALLTHROUGH */ case EPASSTHROUGH: error = ENOTTY; /* FALLTHROUGH */ default: return (error); } } int selwait, nselcoll; /* * Select system call. */ int sys_pselect(struct lwp *l, void *v, register_t *retval) { struct sys_pselect_args /* { syscallarg(int) nd; syscallarg(fd_set *) in; syscallarg(fd_set *) ou; syscallarg(fd_set *) ex; syscallarg(const struct timespec *) ts; syscallarg(sigset_t *) mask; } */ * const uap = v; struct timespec ats; struct timeval atv, *tv = NULL; sigset_t amask, *mask = NULL; int error; if (SCARG(uap, ts)) { error = copyin(SCARG(uap, ts), &ats, sizeof(ats)); if (error) return error; atv.tv_sec = ats.tv_sec; atv.tv_usec = ats.tv_nsec / 1000; tv = &atv; } if (SCARG(uap, mask) != NULL) { error = copyin(SCARG(uap, mask), &amask, sizeof(amask)); if (error) return error; mask = &amask; } return selcommon(l, retval, SCARG(uap, nd), SCARG(uap, in), SCARG(uap, ou), SCARG(uap, ex), tv, mask); } int inittimeleft(struct timeval *tv, struct timeval *sleeptv) { if (itimerfix(tv)) return -1; getmicrouptime(sleeptv); return 0; } int gettimeleft(struct timeval *tv, struct timeval *sleeptv) { /* * We have to recalculate the timeout on every retry. */ struct timeval slepttv; /* * reduce tv by elapsed time * based on monotonic time scale */ getmicrouptime(&slepttv); timeradd(tv, sleeptv, tv); timersub(tv, &slepttv, tv); *sleeptv = slepttv; return tvtohz(tv); } int sys_select(struct lwp *l, void *v, register_t *retval) { struct sys_select_args /* { syscallarg(int) nd; syscallarg(fd_set *) in; syscallarg(fd_set *) ou; syscallarg(fd_set *) ex; syscallarg(struct timeval *) tv; } */ * const uap = v; struct timeval atv, *tv = NULL; int error; if (SCARG(uap, tv)) { error = copyin(SCARG(uap, tv), (caddr_t)&atv, sizeof(atv)); if (error) return error; tv = &atv; } return selcommon(l, retval, SCARG(uap, nd), SCARG(uap, in), SCARG(uap, ou), SCARG(uap, ex), tv, NULL); } int selcommon(struct lwp *l, register_t *retval, int nd, fd_set *u_in, fd_set *u_ou, fd_set *u_ex, struct timeval *tv, sigset_t *mask) { char smallbits[howmany(FD_SETSIZE, NFDBITS) * sizeof(fd_mask) * 6]; struct proc * const p = l->l_proc; caddr_t bits; int s, ncoll, error, timo; size_t ni; sigset_t oldmask; struct timeval sleeptv; error = 0; if (nd < 0) return (EINVAL); if (nd > p->p_fd->fd_nfiles) { /* forgiving; slightly wrong */ nd = p->p_fd->fd_nfiles; } ni = howmany(nd, NFDBITS) * sizeof(fd_mask); if (ni * 6 > sizeof(smallbits)) bits = malloc(ni * 6, M_TEMP, M_WAITOK); else bits = smallbits; #define getbits(name, x) \ if (u_ ## name) { \ error = copyin(u_ ## name, bits + ni * x, ni); \ if (error) \ goto done; \ } else \ memset(bits + ni * x, 0, ni); getbits(in, 0); getbits(ou, 1); getbits(ex, 2); #undef getbits timo = 0; if (tv && inittimeleft(tv, &sleeptv) == -1) { error = EINVAL; goto done; } if (mask) { sigminusset(&sigcantmask, mask); mutex_enter(&p->p_smutex); oldmask = l->l_sigmask; l->l_sigmask = *mask; mutex_exit(&p->p_smutex); } else oldmask = l->l_sigmask; /* XXXgcc */ retry: ncoll = nselcoll; l->l_flag |= LW_SELECT; error = selscan(l, (fd_mask *)(bits + ni * 0), (fd_mask *)(bits + ni * 3), nd, retval); if (error || *retval) goto donemask; if (tv && (timo = gettimeleft(tv, &sleeptv)) <= 0) goto donemask; s = splsched(); if ((l->l_flag & LW_SELECT) == 0 || nselcoll != ncoll) { splx(s); goto retry; } l->l_flag &= ~LW_SELECT; error = tsleep((caddr_t)&selwait, PSOCK | PCATCH, "select", timo); splx(s); if (error == 0) goto retry; donemask: if (mask) { mutex_enter(&p->p_smutex); l->l_sigmask = oldmask; mutex_exit(&p->p_smutex); } l->l_flag &= ~LW_SELECT; done: /* select is not restarted after signals... */ if (error == ERESTART) error = EINTR; if (error == EWOULDBLOCK) error = 0; if (error == 0) { #define putbits(name, x) \ if (u_ ## name) { \ error = copyout(bits + ni * x, u_ ## name, ni); \ if (error) \ goto out; \ } putbits(in, 3); putbits(ou, 4); putbits(ex, 5); #undef putbits } out: if (ni * 6 > sizeof(smallbits)) free(bits, M_TEMP); return (error); } int selscan(struct lwp *l, fd_mask *ibitp, fd_mask *obitp, int nfd, register_t *retval) { static const int flag[3] = { POLLRDNORM | POLLHUP | POLLERR, POLLWRNORM | POLLHUP | POLLERR, POLLRDBAND }; struct proc *p = l->l_proc; struct filedesc *fdp; int msk, i, j, fd, n; fd_mask ibits, obits; struct file *fp; fdp = p->p_fd; n = 0; for (msk = 0; msk < 3; msk++) { for (i = 0; i < nfd; i += NFDBITS) { ibits = *ibitp++; obits = 0; while ((j = ffs(ibits)) && (fd = i + --j) < nfd) { ibits &= ~(1 << j); if ((fp = fd_getfile(fdp, fd)) == NULL) return (EBADF); FILE_USE(fp); if ((*fp->f_ops->fo_poll)(fp, flag[msk], l)) { obits |= (1 << j); n++; } FILE_UNUSE(fp, l); } *obitp++ = obits; } } *retval = n; return (0); } /* * Poll system call. */ int sys_poll(struct lwp *l, void *v, register_t *retval) { struct sys_poll_args /* { syscallarg(struct pollfd *) fds; syscallarg(u_int) nfds; syscallarg(int) timeout; } */ * const uap = v; struct timeval atv, *tv = NULL; if (SCARG(uap, timeout) != INFTIM) { atv.tv_sec = SCARG(uap, timeout) / 1000; atv.tv_usec = (SCARG(uap, timeout) % 1000) * 1000; tv = &atv; } return pollcommon(l, retval, SCARG(uap, fds), SCARG(uap, nfds), tv, NULL); } /* * Poll system call. */ int sys_pollts(struct lwp *l, void *v, register_t *retval) { struct sys_pollts_args /* { syscallarg(struct pollfd *) fds; syscallarg(u_int) nfds; syscallarg(const struct timespec *) ts; syscallarg(const sigset_t *) mask; } */ * const uap = v; struct timespec ats; struct timeval atv, *tv = NULL; sigset_t amask, *mask = NULL; int error; if (SCARG(uap, ts)) { error = copyin(SCARG(uap, ts), &ats, sizeof(ats)); if (error) return error; atv.tv_sec = ats.tv_sec; atv.tv_usec = ats.tv_nsec / 1000; tv = &atv; } if (SCARG(uap, mask)) { error = copyin(SCARG(uap, mask), &amask, sizeof(amask)); if (error) return error; mask = &amask; } return pollcommon(l, retval, SCARG(uap, fds), SCARG(uap, nfds), tv, mask); } int pollcommon(struct lwp *l, register_t *retval, struct pollfd *u_fds, u_int nfds, struct timeval *tv, sigset_t *mask) { char smallbits[32 * sizeof(struct pollfd)]; struct proc * const p = l->l_proc; caddr_t bits; sigset_t oldmask; int s, ncoll, error, timo; size_t ni; struct timeval sleeptv; if (nfds > p->p_fd->fd_nfiles) { /* forgiving; slightly wrong */ nfds = p->p_fd->fd_nfiles; } ni = nfds * sizeof(struct pollfd); if (ni > sizeof(smallbits)) bits = malloc(ni, M_TEMP, M_WAITOK); else bits = smallbits; error = copyin(u_fds, bits, ni); if (error) goto done; timo = 0; if (tv && inittimeleft(tv, &sleeptv) == -1) { error = EINVAL; goto done; } if (mask) { sigminusset(&sigcantmask, mask); mutex_enter(&p->p_smutex); oldmask = l->l_sigmask; l->l_sigmask = *mask; mutex_exit(&p->p_smutex); } else oldmask = l->l_sigmask; /* XXXgcc */ retry: ncoll = nselcoll; l->l_flag |= LW_SELECT; error = pollscan(l, (struct pollfd *)bits, nfds, retval); if (error || *retval) goto donemask; if (tv && (timo = gettimeleft(tv, &sleeptv)) <= 0) goto donemask; s = splsched(); if ((l->l_flag & LW_SELECT) == 0 || nselcoll != ncoll) { splx(s); goto retry; } l->l_flag &= ~LW_SELECT; error = tsleep((caddr_t)&selwait, PSOCK | PCATCH, "poll", timo); splx(s); if (error == 0) goto retry; donemask: if (mask) { mutex_enter(&p->p_smutex); l->l_sigmask = oldmask; mutex_exit(&p->p_smutex); } l->l_flag &= ~LW_SELECT; done: /* poll is not restarted after signals... */ if (error == ERESTART) error = EINTR; if (error == EWOULDBLOCK) error = 0; if (error == 0) { error = copyout(bits, u_fds, ni); if (error) goto out; } out: if (ni > sizeof(smallbits)) free(bits, M_TEMP); return (error); } int pollscan(struct lwp *l, struct pollfd *fds, int nfd, register_t *retval) { struct proc *p = l->l_proc; struct filedesc *fdp; int i, n; struct file *fp; fdp = p->p_fd; n = 0; for (i = 0; i < nfd; i++, fds++) { if (fds->fd >= fdp->fd_nfiles) { fds->revents = POLLNVAL; n++; } else if (fds->fd < 0) { fds->revents = 0; } else { if ((fp = fd_getfile(fdp, fds->fd)) == NULL) { fds->revents = POLLNVAL; n++; } else { FILE_USE(fp); fds->revents = (*fp->f_ops->fo_poll)(fp, fds->events | POLLERR | POLLHUP, l); if (fds->revents != 0) n++; FILE_UNUSE(fp, l); } } } *retval = n; return (0); } /*ARGSUSED*/ int seltrue(dev_t dev, int events, struct lwp *l) { return (events & (POLLIN | POLLOUT | POLLRDNORM | POLLWRNORM)); } /* * Record a select request. */ void selrecord(struct lwp *selector, struct selinfo *sip) { struct lwp *l; struct proc *p; pid_t mypid; mypid = selector->l_proc->p_pid; if (sip->sel_pid == mypid) return; mutex_enter(&proclist_mutex); if (sip->sel_pid && (p = p_find(sip->sel_pid, PFIND_LOCKED))) { mutex_enter(&p->p_smutex); LIST_FOREACH(l, &p->p_lwps, l_sibling) { lwp_lock(l); if (l->l_wchan == (caddr_t)&selwait && l->l_stat == LSSLEEP) { sip->sel_collision = 1; lwp_unlock(l); break; } lwp_unlock(l); } mutex_exit(&p->p_smutex); } mutex_exit(&proclist_mutex); if (!sip->sel_collision) sip->sel_pid = mypid; } /* * Do a wakeup when a selectable event occurs. */ void selwakeup(sip) struct selinfo *sip; { struct lwp *l; struct proc *p; if (sip->sel_pid == 0) return; if (sip->sel_collision) { sip->sel_pid = 0; nselcoll++; sip->sel_collision = 0; wakeup((caddr_t)&selwait); return; } /* * We must use the proclist_mutex as we can be called from an * interrupt context. */ mutex_enter(&proclist_mutex); p = p_find(sip->sel_pid, PFIND_LOCKED); sip->sel_pid = 0; if (p == NULL) { mutex_exit(&proclist_mutex); return; } mutex_enter(&p->p_smutex); LIST_FOREACH(l, &p->p_lwps, l_sibling) { lwp_lock(l); if (l->l_wchan == (wchan_t)&selwait && l->l_stat == LSSLEEP) { /* setrunnable() will release the lock. */ setrunnable(l); } else { if (l->l_flag & LW_SELECT) l->l_flag &= ~LW_SELECT; lwp_unlock(l); } } mutex_exit(&p->p_smutex); mutex_exit(&proclist_mutex); }