1999-01-30 06:29:48 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1998 Mark Newton
|
|
|
|
* Copyright (c) 1994 Christos Zoulas
|
|
|
|
* 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. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* SVR4 compatibility module.
|
|
|
|
*
|
|
|
|
* SVR4 system calls that are implemented differently in BSD are
|
|
|
|
* handled here.
|
|
|
|
*/
|
|
|
|
|
2003-06-10 21:44:29 +00:00
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
2002-08-12 01:42:21 +00:00
|
|
|
#include "opt_mac.h"
|
|
|
|
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/dirent.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/fcntl.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/filedesc.h>
|
2001-01-23 21:39:15 +00:00
|
|
|
#include <sys/jail.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/kernel.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/lock.h>
|
2002-08-12 01:42:21 +00:00
|
|
|
#include <sys/mac.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/malloc.h>
|
|
|
|
#include <sys/file.h> /* Must come after sys/malloc.h */
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <sys/mount.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/msg.h>
|
|
|
|
#include <sys/mutex.h>
|
|
|
|
#include <sys/namei.h>
|
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/ptrace.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/resource.h>
|
|
|
|
#include <sys/resourcevar.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/sem.h>
|
|
|
|
#include <sys/stat.h>
|
2001-03-28 15:04:22 +00:00
|
|
|
#include <sys/sx.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/sysproto.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/times.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/uio.h>
|
2001-01-23 21:39:15 +00:00
|
|
|
#include <sys/user.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/vnode.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/wait.h>
|
|
|
|
|
2000-08-31 22:54:09 +00:00
|
|
|
#include <compat/svr4/svr4.h>
|
|
|
|
#include <compat/svr4/svr4_types.h>
|
|
|
|
#include <compat/svr4/svr4_signal.h>
|
|
|
|
#include <compat/svr4/svr4_proto.h>
|
|
|
|
#include <compat/svr4/svr4_util.h>
|
|
|
|
#include <compat/svr4/svr4_sysconfig.h>
|
|
|
|
#include <compat/svr4/svr4_dirent.h>
|
|
|
|
#include <compat/svr4/svr4_acl.h>
|
|
|
|
#include <compat/svr4/svr4_ulimit.h>
|
|
|
|
#include <compat/svr4/svr4_statvfs.h>
|
|
|
|
#include <compat/svr4/svr4_hrt.h>
|
|
|
|
#include <compat/svr4/svr4_mman.h>
|
|
|
|
#include <compat/svr4/svr4_wait.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
#include <machine/vmparam.h>
|
|
|
|
#include <vm/vm.h>
|
|
|
|
#include <vm/vm_param.h>
|
|
|
|
#include <vm/vm_map.h>
|
2001-01-23 21:30:25 +00:00
|
|
|
#if defined(__FreeBSD__)
|
2002-03-20 10:35:22 +00:00
|
|
|
#include <vm/uma.h>
|
2003-01-14 21:31:31 +00:00
|
|
|
#include <vm/vm_extern.h>
|
2001-01-23 21:30:25 +00:00
|
|
|
#endif
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
#if defined(NetBSD)
|
|
|
|
# if defined(UVM)
|
|
|
|
# include <uvm/uvm_extern.h>
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define BSD_DIRENT(cp) ((struct dirent *)(cp))
|
|
|
|
|
2002-03-20 05:48:58 +00:00
|
|
|
static int svr4_mknod(struct thread *, register_t *, char *,
|
|
|
|
svr4_mode_t, svr4_dev_t);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-03-20 05:48:58 +00:00
|
|
|
static __inline clock_t timeval_to_clock_t(struct timeval *);
|
|
|
|
static int svr4_setinfo (struct proc *, int, svr4_siginfo_t *);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
struct svr4_hrtcntl_args;
|
2002-03-20 05:48:58 +00:00
|
|
|
static int svr4_hrtcntl (struct thread *, struct svr4_hrtcntl_args *,
|
|
|
|
register_t *);
|
|
|
|
static void bsd_statfs_to_svr4_statvfs(const struct statfs *,
|
|
|
|
struct svr4_statvfs *);
|
|
|
|
static void bsd_statfs_to_svr4_statvfs64(const struct statfs *,
|
|
|
|
struct svr4_statvfs64 *);
|
|
|
|
static struct proc *svr4_pfind(pid_t pid);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
/* BOGUS noop */
|
|
|
|
#if defined(BOGUS)
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_setitimer(td, uap)
|
|
|
|
register struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_setitimer_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
td->td_retval[0] = 0;
|
1999-01-30 06:29:48 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_wait(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_wait_args *uap;
|
|
|
|
{
|
|
|
|
struct wait_args w4;
|
2001-09-12 08:38:13 +00:00
|
|
|
int error, *retval = td->td_retval, st, sig;
|
2002-12-14 01:56:26 +00:00
|
|
|
size_t sz = sizeof(*w4.status);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
w4.rusage = NULL;
|
|
|
|
w4.options = 0;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->status == NULL) {
|
1999-01-30 06:29:48 +00:00
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
w4.status = stackgap_alloc(&sg, sz);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
else
|
2002-12-14 01:56:26 +00:00
|
|
|
w4.status = uap->status;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
w4.pid = WAIT_ANY;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((error = wait4(td, &w4)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if ((error = copyin(w4.status, &st, sizeof(st))) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
if (WIFSIGNALED(st)) {
|
|
|
|
sig = WTERMSIG(st);
|
|
|
|
if (sig >= 0 && sig < NSIG)
|
1999-09-29 15:12:18 +00:00
|
|
|
st = (st & ~0177) | SVR4_BSD2SVR4_SIG(sig);
|
1999-01-30 06:29:48 +00:00
|
|
|
} else if (WIFSTOPPED(st)) {
|
|
|
|
sig = WSTOPSIG(st);
|
|
|
|
if (sig >= 0 && sig < NSIG)
|
1999-09-29 15:12:18 +00:00
|
|
|
st = (st & ~0xff00) | (SVR4_BSD2SVR4_SIG(sig) << 8);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* It looks like wait(2) on svr4/solaris/2.4 returns
|
|
|
|
* the status in retval[1], and the pid on retval[0].
|
|
|
|
*/
|
|
|
|
retval[1] = st;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->status)
|
|
|
|
if ((error = copyout(&st, uap->status, sizeof(st))) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_execv(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_execv_args *uap;
|
|
|
|
{
|
|
|
|
struct execve_args ap;
|
|
|
|
caddr_t sg;
|
|
|
|
|
|
|
|
sg = stackgap_init();
|
2002-12-14 01:56:26 +00:00
|
|
|
CHECKALTEXIST(td, &sg, uap->path);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
ap.fname = uap->path;
|
|
|
|
ap.argv = uap->argp;
|
|
|
|
ap.envv = NULL;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
return execve(td, &ap);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_execve(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_execve_args *uap;
|
|
|
|
{
|
|
|
|
struct execve_args ap;
|
|
|
|
caddr_t sg;
|
|
|
|
|
|
|
|
sg = stackgap_init();
|
2001-09-12 08:38:13 +00:00
|
|
|
CHECKALTEXIST(td, &sg, uap->path);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
ap.fname = uap->path;
|
|
|
|
ap.argv = uap->argp;
|
|
|
|
ap.envv = uap->envp;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
return execve(td, &ap);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_time(td, v)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_time_args *v;
|
|
|
|
{
|
|
|
|
struct svr4_sys_time_args *uap = v;
|
|
|
|
int error = 0;
|
|
|
|
struct timeval tv;
|
|
|
|
|
|
|
|
microtime(&tv);
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->t)
|
|
|
|
error = copyout(&tv.tv_sec, uap->t,
|
|
|
|
sizeof(*(uap->t)));
|
2001-09-12 08:38:13 +00:00
|
|
|
td->td_retval[0] = (int) tv.tv_sec;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read SVR4-style directory entries. We suck them into kernel space so
|
1999-12-12 11:25:33 +00:00
|
|
|
* that they can be massaged before being copied out to user code.
|
1999-01-30 06:29:48 +00:00
|
|
|
*
|
1999-12-12 11:25:33 +00:00
|
|
|
* This code is ported from the Linux emulator: Changes to the VFS interface
|
|
|
|
* between FreeBSD and NetBSD have made it simpler to port it from there than
|
|
|
|
* to adapt the NetBSD version.
|
1999-01-30 06:29:48 +00:00
|
|
|
*/
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_getdents64(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_getdents64_args *uap;
|
|
|
|
{
|
1999-12-12 11:25:33 +00:00
|
|
|
register struct dirent *bdp;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct vnode *vp;
|
1999-12-12 11:25:33 +00:00
|
|
|
caddr_t inp, buf; /* BSD-format */
|
|
|
|
int len, reclen; /* BSD-format */
|
|
|
|
caddr_t outp; /* SVR4-format */
|
|
|
|
int resid, svr4reclen=0; /* SVR4-format */
|
1999-01-30 06:29:48 +00:00
|
|
|
struct file *fp;
|
|
|
|
struct uio auio;
|
|
|
|
struct iovec aiov;
|
1999-12-12 11:25:33 +00:00
|
|
|
struct vattr va;
|
|
|
|
off_t off;
|
|
|
|
struct svr4_dirent64 svr4_dirent;
|
|
|
|
int buflen, error, eofflag, nbytes, justone;
|
|
|
|
u_long *cookies = NULL, *cookiep;
|
|
|
|
int ncookies;
|
|
|
|
|
|
|
|
DPRINTF(("svr4_sys_getdents64(%d, *, %d)\n",
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->fd, uap->nbytes));
|
|
|
|
if ((error = getvnode(td->td_proc->p_fd, uap->fd, &fp)) != 0) {
|
1999-01-30 06:29:48 +00:00
|
|
|
return (error);
|
1999-12-12 11:25:33 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
if ((fp->f_flag & FREAD) == 0) {
|
|
|
|
fdrop(fp, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
return (EBADF);
|
2002-01-13 11:58:06 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2003-01-13 00:33:17 +00:00
|
|
|
vp = fp->f_data;
|
1999-12-12 11:25:33 +00:00
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
if (vp->v_type != VDIR) {
|
|
|
|
fdrop(fp, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
return (EINVAL);
|
2002-01-13 11:58:06 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-02-27 18:32:23 +00:00
|
|
|
if ((error = VOP_GETATTR(vp, &va, td->td_ucred, td))) {
|
2002-01-13 11:58:06 +00:00
|
|
|
fdrop(fp, td);
|
1999-12-12 11:25:33 +00:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
nbytes = uap->nbytes;
|
1999-12-12 11:25:33 +00:00
|
|
|
if (nbytes == 1) {
|
|
|
|
nbytes = sizeof (struct svr4_dirent64);
|
|
|
|
justone = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
justone = 0;
|
|
|
|
|
|
|
|
off = fp->f_offset;
|
|
|
|
#define DIRBLKSIZ 512 /* XXX we used to use ufs's DIRBLKSIZ */
|
|
|
|
buflen = max(DIRBLKSIZ, nbytes);
|
|
|
|
buflen = min(buflen, MAXBSIZE);
|
2003-02-19 05:47:46 +00:00
|
|
|
buf = malloc(buflen, M_TEMP, M_WAITOK);
|
2001-09-12 08:38:13 +00:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
again:
|
|
|
|
aiov.iov_base = buf;
|
|
|
|
aiov.iov_len = buflen;
|
|
|
|
auio.uio_iov = &aiov;
|
|
|
|
auio.uio_iovcnt = 1;
|
|
|
|
auio.uio_rw = UIO_READ;
|
|
|
|
auio.uio_segflg = UIO_SYSSPACE;
|
2001-09-12 08:38:13 +00:00
|
|
|
auio.uio_td = td;
|
1999-01-30 06:29:48 +00:00
|
|
|
auio.uio_resid = buflen;
|
|
|
|
auio.uio_offset = off;
|
|
|
|
|
1999-12-12 11:25:33 +00:00
|
|
|
if (cookies) {
|
|
|
|
free(cookies, M_TEMP);
|
|
|
|
cookies = NULL;
|
|
|
|
}
|
|
|
|
|
2002-08-12 01:42:21 +00:00
|
|
|
#ifdef MAC
|
|
|
|
error = mac_check_vnode_readdir(td->td_ucred, vp);
|
|
|
|
if (error)
|
2002-08-15 15:46:10 +00:00
|
|
|
goto out;
|
2002-08-12 01:42:21 +00:00
|
|
|
#endif
|
|
|
|
|
1999-12-12 11:25:33 +00:00
|
|
|
error = VOP_READDIR(vp, &auio, fp->f_cred, &eofflag,
|
|
|
|
&ncookies, &cookies);
|
|
|
|
if (error) {
|
1999-01-30 06:29:48 +00:00
|
|
|
goto out;
|
1999-12-12 11:25:33 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
inp = buf;
|
2002-12-14 01:56:26 +00:00
|
|
|
outp = (caddr_t) uap->dp;
|
1999-12-12 11:25:33 +00:00
|
|
|
resid = nbytes;
|
|
|
|
if ((len = buflen - auio.uio_resid) <= 0) {
|
1999-01-30 06:29:48 +00:00
|
|
|
goto eof;
|
1999-12-12 11:25:33 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
1999-12-12 11:25:33 +00:00
|
|
|
cookiep = cookies;
|
|
|
|
|
|
|
|
if (cookies) {
|
|
|
|
/*
|
|
|
|
* When using cookies, the vfs has the option of reading from
|
|
|
|
* a different offset than that supplied (UFS truncates the
|
|
|
|
* offset to a block boundary to make sure that it never reads
|
|
|
|
* partway through a directory entry, even if the directory
|
|
|
|
* has been compacted).
|
|
|
|
*/
|
|
|
|
while (len > 0 && ncookies > 0 && *cookiep <= off) {
|
|
|
|
bdp = (struct dirent *) inp;
|
|
|
|
len -= bdp->d_reclen;
|
|
|
|
inp += bdp->d_reclen;
|
|
|
|
cookiep++;
|
|
|
|
ncookies--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while (len > 0) {
|
|
|
|
if (cookiep && ncookies == 0)
|
|
|
|
break;
|
|
|
|
bdp = (struct dirent *) inp;
|
1999-01-30 06:29:48 +00:00
|
|
|
reclen = bdp->d_reclen;
|
1999-12-12 11:25:33 +00:00
|
|
|
if (reclen & 3) {
|
|
|
|
DPRINTF(("svr4_readdir: reclen=%d\n", reclen));
|
|
|
|
error = EFAULT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
1999-01-30 06:29:48 +00:00
|
|
|
if (bdp->d_fileno == 0) {
|
1999-12-12 11:25:33 +00:00
|
|
|
inp += reclen;
|
|
|
|
if (cookiep) {
|
|
|
|
off = *cookiep++;
|
|
|
|
ncookies--;
|
|
|
|
} else
|
|
|
|
off += reclen;
|
|
|
|
len -= reclen;
|
1999-01-30 06:29:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
1999-12-12 11:25:33 +00:00
|
|
|
svr4reclen = SVR4_RECLEN(&svr4_dirent, bdp->d_namlen);
|
|
|
|
if (reclen > len || resid < svr4reclen) {
|
1999-01-30 06:29:48 +00:00
|
|
|
outp++;
|
|
|
|
break;
|
|
|
|
}
|
1999-12-12 11:25:33 +00:00
|
|
|
svr4_dirent.d_ino = (long) bdp->d_fileno;
|
|
|
|
if (justone) {
|
|
|
|
/*
|
|
|
|
* old svr4-style readdir usage.
|
|
|
|
*/
|
|
|
|
svr4_dirent.d_off = (svr4_off_t) svr4reclen;
|
|
|
|
svr4_dirent.d_reclen = (u_short) bdp->d_namlen;
|
|
|
|
} else {
|
|
|
|
svr4_dirent.d_off = (svr4_off_t)(off + reclen);
|
|
|
|
svr4_dirent.d_reclen = (u_short) svr4reclen;
|
|
|
|
}
|
|
|
|
strcpy(svr4_dirent.d_name, bdp->d_name);
|
|
|
|
if ((error = copyout((caddr_t)&svr4_dirent, outp, svr4reclen)))
|
1999-01-30 06:29:48 +00:00
|
|
|
goto out;
|
|
|
|
inp += reclen;
|
1999-12-12 11:25:33 +00:00
|
|
|
if (cookiep) {
|
|
|
|
off = *cookiep++;
|
|
|
|
ncookies--;
|
|
|
|
} else
|
|
|
|
off += reclen;
|
|
|
|
outp += svr4reclen;
|
|
|
|
resid -= svr4reclen;
|
|
|
|
len -= reclen;
|
|
|
|
if (justone)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if (outp == (caddr_t) uap->dp)
|
1999-01-30 06:29:48 +00:00
|
|
|
goto again;
|
1999-12-12 11:25:33 +00:00
|
|
|
fp->f_offset = off;
|
|
|
|
|
|
|
|
if (justone)
|
|
|
|
nbytes = resid + svr4reclen;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
eof:
|
2001-09-12 08:38:13 +00:00
|
|
|
td->td_retval[0] = nbytes - resid;
|
1999-01-30 06:29:48 +00:00
|
|
|
out:
|
2002-01-13 11:58:06 +00:00
|
|
|
VOP_UNLOCK(vp, 0, td);
|
|
|
|
fdrop(fp, td);
|
1999-12-12 11:25:33 +00:00
|
|
|
if (cookies)
|
|
|
|
free(cookies, M_TEMP);
|
1999-01-30 06:29:48 +00:00
|
|
|
free(buf, M_TEMP);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_getdents(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_getdents_args *uap;
|
|
|
|
{
|
|
|
|
struct dirent *bdp;
|
|
|
|
struct vnode *vp;
|
|
|
|
caddr_t inp, buf; /* BSD-format */
|
|
|
|
int len, reclen; /* BSD-format */
|
|
|
|
caddr_t outp; /* SVR4-format */
|
|
|
|
int resid, svr4_reclen; /* SVR4-format */
|
|
|
|
struct file *fp;
|
|
|
|
struct uio auio;
|
|
|
|
struct iovec aiov;
|
|
|
|
struct svr4_dirent idb;
|
|
|
|
off_t off; /* true file offset */
|
|
|
|
int buflen, error, eofflag;
|
|
|
|
u_long *cookiebuf = NULL, *cookie;
|
2001-09-12 08:38:13 +00:00
|
|
|
int ncookies = 0, *retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if ((error = getvnode(td->td_proc->p_fd, uap->fd, &fp)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return (error);
|
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
if ((fp->f_flag & FREAD) == 0) {
|
|
|
|
fdrop(fp, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
return (EBADF);
|
2002-01-13 11:58:06 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2003-01-13 00:33:17 +00:00
|
|
|
vp = fp->f_data;
|
2002-01-13 11:58:06 +00:00
|
|
|
if (vp->v_type != VDIR) {
|
|
|
|
fdrop(fp, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
return (EINVAL);
|
2002-01-13 11:58:06 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
buflen = min(MAXBSIZE, uap->nbytes);
|
2003-02-19 05:47:46 +00:00
|
|
|
buf = malloc(buflen, M_TEMP, M_WAITOK);
|
2001-09-12 08:38:13 +00:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
off = fp->f_offset;
|
|
|
|
again:
|
|
|
|
aiov.iov_base = buf;
|
|
|
|
aiov.iov_len = buflen;
|
|
|
|
auio.uio_iov = &aiov;
|
|
|
|
auio.uio_iovcnt = 1;
|
|
|
|
auio.uio_rw = UIO_READ;
|
|
|
|
auio.uio_segflg = UIO_SYSSPACE;
|
2001-09-12 08:38:13 +00:00
|
|
|
auio.uio_td = td;
|
1999-01-30 06:29:48 +00:00
|
|
|
auio.uio_resid = buflen;
|
|
|
|
auio.uio_offset = off;
|
2002-08-12 01:42:21 +00:00
|
|
|
|
|
|
|
#ifdef MAC
|
|
|
|
error = mac_check_vnode_readdir(td->td_ucred, vp);
|
|
|
|
if (error)
|
|
|
|
goto out;
|
|
|
|
#endif
|
|
|
|
|
1999-01-30 06:29:48 +00:00
|
|
|
/*
|
|
|
|
* First we read into the malloc'ed buffer, then
|
|
|
|
* we massage it into user space, one record at a time.
|
|
|
|
*/
|
|
|
|
error = VOP_READDIR(vp, &auio, fp->f_cred, &eofflag, &ncookies,
|
|
|
|
&cookiebuf);
|
2002-01-13 11:58:06 +00:00
|
|
|
if (error) {
|
1999-01-30 06:29:48 +00:00
|
|
|
goto out;
|
2002-01-13 11:58:06 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
inp = buf;
|
2002-12-14 01:56:26 +00:00
|
|
|
outp = uap->buf;
|
|
|
|
resid = uap->nbytes;
|
1999-01-30 06:29:48 +00:00
|
|
|
if ((len = buflen - auio.uio_resid) == 0)
|
|
|
|
goto eof;
|
|
|
|
|
|
|
|
for (cookie = cookiebuf; len > 0; len -= reclen) {
|
|
|
|
bdp = (struct dirent *)inp;
|
|
|
|
reclen = bdp->d_reclen;
|
|
|
|
if (reclen & 3)
|
1999-12-12 11:25:33 +00:00
|
|
|
panic("svr4_sys_getdents64: bad reclen");
|
1999-01-30 06:29:48 +00:00
|
|
|
off = *cookie++; /* each entry points to the next */
|
|
|
|
if ((off >> 32) != 0) {
|
1999-12-12 11:25:33 +00:00
|
|
|
uprintf("svr4_sys_getdents64: dir offset too large for emulated program");
|
1999-01-30 06:29:48 +00:00
|
|
|
error = EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (bdp->d_fileno == 0) {
|
|
|
|
inp += reclen; /* it is a hole; squish it out */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
svr4_reclen = SVR4_RECLEN(&idb, bdp->d_namlen);
|
|
|
|
if (reclen > len || resid < svr4_reclen) {
|
|
|
|
/* entry too big for buffer, so just stop */
|
|
|
|
outp++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Massage in place to make a SVR4-shaped dirent (otherwise
|
|
|
|
* we have to worry about touching user memory outside of
|
|
|
|
* the copyout() call).
|
|
|
|
*/
|
|
|
|
idb.d_ino = (svr4_ino_t)bdp->d_fileno;
|
|
|
|
idb.d_off = (svr4_off_t)off;
|
|
|
|
idb.d_reclen = (u_short)svr4_reclen;
|
|
|
|
strcpy(idb.d_name, bdp->d_name);
|
|
|
|
if ((error = copyout((caddr_t)&idb, outp, svr4_reclen)))
|
|
|
|
goto out;
|
|
|
|
/* advance past this real entry */
|
|
|
|
inp += reclen;
|
|
|
|
/* advance output past SVR4-shaped entry */
|
|
|
|
outp += svr4_reclen;
|
|
|
|
resid -= svr4_reclen;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if we squished out the whole block, try again */
|
2002-12-14 01:56:26 +00:00
|
|
|
if (outp == uap->buf)
|
1999-01-30 06:29:48 +00:00
|
|
|
goto again;
|
|
|
|
fp->f_offset = off; /* update the vnode offset */
|
|
|
|
|
|
|
|
eof:
|
2002-12-14 01:56:26 +00:00
|
|
|
*retval = uap->nbytes - resid;
|
1999-01-30 06:29:48 +00:00
|
|
|
out:
|
2001-09-12 08:38:13 +00:00
|
|
|
VOP_UNLOCK(vp, 0, td);
|
2002-01-13 11:58:06 +00:00
|
|
|
fdrop(fp, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
if (cookiebuf)
|
|
|
|
free(cookiebuf, M_TEMP);
|
|
|
|
free(buf, M_TEMP);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_mmap(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_mmap_args *uap;
|
|
|
|
{
|
|
|
|
struct mmap_args mm;
|
|
|
|
int *retval;
|
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
#define _MAP_NEW 0x80000000
|
|
|
|
/*
|
|
|
|
* Verify the arguments.
|
|
|
|
*/
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->prot & ~(PROT_READ | PROT_WRITE | PROT_EXEC))
|
1999-01-30 06:29:48 +00:00
|
|
|
return EINVAL; /* XXX still needed? */
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->len == 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return EINVAL;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
mm.prot = uap->prot;
|
|
|
|
mm.len = uap->len;
|
|
|
|
mm.flags = uap->flags & ~_MAP_NEW;
|
|
|
|
mm.fd = uap->fd;
|
|
|
|
mm.addr = uap->addr;
|
|
|
|
mm.pos = uap->pos;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
return mmap(td, &mm);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_mmap64(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_mmap64_args *uap;
|
|
|
|
{
|
|
|
|
struct mmap_args mm;
|
|
|
|
void *rp;
|
|
|
|
|
|
|
|
#define _MAP_NEW 0x80000000
|
|
|
|
/*
|
|
|
|
* Verify the arguments.
|
|
|
|
*/
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->prot & ~(PROT_READ | PROT_WRITE | PROT_EXEC))
|
1999-01-30 06:29:48 +00:00
|
|
|
return EINVAL; /* XXX still needed? */
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->len == 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return EINVAL;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
mm.prot = uap->prot;
|
|
|
|
mm.len = uap->len;
|
|
|
|
mm.flags = uap->flags & ~_MAP_NEW;
|
|
|
|
mm.fd = uap->fd;
|
|
|
|
mm.addr = uap->addr;
|
|
|
|
mm.pos = uap->pos;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-10-10 23:06:54 +00:00
|
|
|
rp = (void *) round_page((vm_offset_t)(td->td_proc->p_vmspace->vm_daddr + maxdsiz));
|
2002-12-14 01:56:26 +00:00
|
|
|
if ((mm.flags & MAP_FIXED) == 0 &&
|
|
|
|
mm.addr != 0 && (void *)mm.addr < rp)
|
|
|
|
mm.addr = rp;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
return mmap(td, &mm);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_fchroot(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_fchroot_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
struct filedesc *fdp = td->td_proc->p_fd;
|
2002-01-19 03:48:38 +00:00
|
|
|
struct vnode *vp, *vpold;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct file *fp;
|
|
|
|
int error;
|
|
|
|
|
2002-04-01 21:31:13 +00:00
|
|
|
if ((error = suser(td)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
2002-12-14 01:56:26 +00:00
|
|
|
if ((error = getvnode(fdp, uap->fd, &fp)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
2003-01-13 00:33:17 +00:00
|
|
|
vp = fp->f_data;
|
2001-09-12 08:38:13 +00:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
if (vp->v_type != VDIR)
|
|
|
|
error = ENOTDIR;
|
2001-01-27 00:01:31 +00:00
|
|
|
else
|
2002-02-27 18:32:23 +00:00
|
|
|
error = VOP_ACCESS(vp, VEXEC, td->td_ucred, td);
|
2001-09-12 08:38:13 +00:00
|
|
|
VOP_UNLOCK(vp, 0, td);
|
2002-01-19 03:48:38 +00:00
|
|
|
if (error) {
|
2002-01-13 11:58:06 +00:00
|
|
|
fdrop(fp, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
2002-01-13 11:58:06 +00:00
|
|
|
}
|
1999-01-30 06:29:48 +00:00
|
|
|
VREF(vp);
|
2002-01-13 11:58:06 +00:00
|
|
|
FILEDESC_LOCK(fdp);
|
|
|
|
vpold = fdp->fd_rdir;
|
1999-01-30 06:29:48 +00:00
|
|
|
fdp->fd_rdir = vp;
|
2002-01-13 11:58:06 +00:00
|
|
|
FILEDESC_UNLOCK(fdp);
|
|
|
|
if (vpold != NULL)
|
|
|
|
vrele(vpold);
|
|
|
|
fdrop(fp, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_mknod(td, retval, path, mode, dev)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
register_t *retval;
|
|
|
|
char *path;
|
|
|
|
svr4_mode_t mode;
|
|
|
|
svr4_dev_t dev;
|
|
|
|
{
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
CHECKALTEXIST(td, &sg, path);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
if (S_ISFIFO(mode)) {
|
|
|
|
struct mkfifo_args ap;
|
2002-12-14 01:56:26 +00:00
|
|
|
ap.path = path;
|
|
|
|
ap.mode = mode;
|
2001-09-12 08:38:13 +00:00
|
|
|
return mkfifo(td, &ap);
|
1999-01-30 06:29:48 +00:00
|
|
|
} else {
|
|
|
|
struct mknod_args ap;
|
2002-12-14 01:56:26 +00:00
|
|
|
ap.path = path;
|
|
|
|
ap.mode = mode;
|
|
|
|
ap.dev = dev;
|
2001-09-12 08:38:13 +00:00
|
|
|
return mknod(td, &ap);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_mknod(td, uap)
|
|
|
|
register struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_mknod_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
int *retval = td->td_retval;
|
|
|
|
return svr4_mknod(td, retval,
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->path, uap->mode,
|
|
|
|
(svr4_dev_t)svr4_to_bsd_odev_t(uap->dev));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_xmknod(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_xmknod_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
int *retval = td->td_retval;
|
|
|
|
return svr4_mknod(td, retval,
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->path, uap->mode,
|
|
|
|
(svr4_dev_t)svr4_to_bsd_dev_t(uap->dev));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_vhangup(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_vhangup_args *uap;
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_sysconfig(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_sysconfig_args *uap;
|
|
|
|
{
|
|
|
|
int *retval;
|
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
retval = &(td->td_retval[0]);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->name) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case SVR4_CONFIG_UNUSED:
|
|
|
|
*retval = 0;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_NGROUPS:
|
|
|
|
*retval = NGROUPS_MAX;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_CHILD_MAX:
|
|
|
|
*retval = maxproc;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_OPEN_FILES:
|
|
|
|
*retval = maxfiles;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_POSIX_VER:
|
|
|
|
*retval = 198808;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_PAGESIZE:
|
|
|
|
*retval = PAGE_SIZE;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_CLK_TCK:
|
|
|
|
*retval = 60; /* should this be `hz', ie. 100? */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_XOPEN_VER:
|
|
|
|
*retval = 2; /* XXX: What should that be? */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_PROF_TCK:
|
|
|
|
*retval = 60; /* XXX: What should that be? */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_NPROC_CONF:
|
|
|
|
*retval = 1; /* Only one processor for now */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_NPROC_ONLN:
|
|
|
|
*retval = 1; /* And it better be online */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_AIO_LISTIO_MAX:
|
|
|
|
case SVR4_CONFIG_AIO_MAX:
|
|
|
|
case SVR4_CONFIG_AIO_PRIO_DELTA_MAX:
|
|
|
|
*retval = 0; /* No aio support */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_DELAYTIMER_MAX:
|
|
|
|
*retval = 0; /* No delaytimer support */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_MQ_OPEN_MAX:
|
|
|
|
*retval = msginfo.msgmni;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_MQ_PRIO_MAX:
|
|
|
|
*retval = 0; /* XXX: Don't know */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_RTSIG_MAX:
|
|
|
|
*retval = 0;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_SEM_NSEMS_MAX:
|
|
|
|
*retval = seminfo.semmni;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_SEM_VALUE_MAX:
|
|
|
|
*retval = seminfo.semvmx;
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_SIGQUEUE_MAX:
|
|
|
|
*retval = 0; /* XXX: Don't know */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_SIGRT_MIN:
|
|
|
|
case SVR4_CONFIG_SIGRT_MAX:
|
|
|
|
*retval = 0; /* No real time signals */
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_TIMER_MAX:
|
|
|
|
*retval = 3; /* XXX: real, virtual, profiling */
|
|
|
|
break;
|
|
|
|
#if defined(NOTYET)
|
|
|
|
case SVR4_CONFIG_PHYS_PAGES:
|
|
|
|
#if defined(UVM)
|
|
|
|
*retval = uvmexp.free; /* XXX: free instead of total */
|
|
|
|
#else
|
|
|
|
*retval = cnt.v_free_count; /* XXX: free instead of total */
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
case SVR4_CONFIG_AVPHYS_PAGES:
|
|
|
|
#if defined(UVM)
|
|
|
|
*retval = uvmexp.active; /* XXX: active instead of avg */
|
|
|
|
#else
|
|
|
|
*retval = cnt.v_active_count; /* XXX: active instead of avg */
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
#endif /* NOTYET */
|
|
|
|
|
|
|
|
default:
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern int swap_pager_full;
|
|
|
|
|
|
|
|
/* ARGSUSED */
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_break(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_break_args *uap;
|
|
|
|
{
|
2002-08-31 19:33:59 +00:00
|
|
|
struct proc *p = td->td_proc;
|
|
|
|
struct vmspace *vm = p->p_vmspace;
|
1999-01-30 06:29:48 +00:00
|
|
|
vm_offset_t new, old, base, ns;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
base = round_page((vm_offset_t) vm->vm_daddr);
|
2002-12-14 01:56:26 +00:00
|
|
|
ns = (vm_offset_t)uap->nsize;
|
1999-01-30 06:29:48 +00:00
|
|
|
new = round_page(ns);
|
2001-01-23 21:30:25 +00:00
|
|
|
/* For p_rlimit. */
|
|
|
|
mtx_assert(&Giant, MA_OWNED);
|
1999-01-30 06:29:48 +00:00
|
|
|
if (new > base) {
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((new - base) > (unsigned) td->td_proc->p_rlimit[RLIMIT_DATA].rlim_cur) {
|
1999-01-30 06:29:48 +00:00
|
|
|
return ENOMEM;
|
|
|
|
}
|
|
|
|
if (new >= VM_MAXUSER_ADDRESS) {
|
|
|
|
return (ENOMEM);
|
|
|
|
}
|
|
|
|
} else if (new < base) {
|
|
|
|
/*
|
|
|
|
* This is simply an invalid value. If someone wants to
|
|
|
|
* do fancy address space manipulations, mmap and munmap
|
|
|
|
* can do most of what the user would want.
|
|
|
|
*/
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
old = base + ctob(vm->vm_dsize);
|
|
|
|
|
|
|
|
if (new > old) {
|
|
|
|
vm_size_t diff;
|
|
|
|
diff = new - old;
|
2002-08-30 18:09:46 +00:00
|
|
|
if (vm->vm_map.size + diff > p->p_rlimit[RLIMIT_VMEM].rlim_cur)
|
|
|
|
return(ENOMEM);
|
1999-01-30 06:29:48 +00:00
|
|
|
rv = vm_map_find(&vm->vm_map, NULL, 0, &old, diff, FALSE,
|
|
|
|
VM_PROT_ALL, VM_PROT_ALL, 0);
|
|
|
|
if (rv != KERN_SUCCESS) {
|
|
|
|
return (ENOMEM);
|
|
|
|
}
|
|
|
|
vm->vm_dsize += btoc(diff);
|
|
|
|
} else if (new < old) {
|
|
|
|
rv = vm_map_remove(&vm->vm_map, new, old);
|
|
|
|
if (rv != KERN_SUCCESS) {
|
|
|
|
return (ENOMEM);
|
|
|
|
}
|
|
|
|
vm->vm_dsize -= btoc(old - new);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline clock_t
|
|
|
|
timeval_to_clock_t(tv)
|
|
|
|
struct timeval *tv;
|
|
|
|
{
|
|
|
|
return tv->tv_sec * hz + tv->tv_usec / (1000000 / hz);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_times(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_times_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
int error, *retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct tms tms;
|
|
|
|
struct timeval t;
|
|
|
|
struct rusage *ru;
|
|
|
|
struct rusage r;
|
|
|
|
struct getrusage_args ga;
|
|
|
|
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
ru = stackgap_alloc(&sg, sizeof(struct rusage));
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
ga.who = RUSAGE_SELF;
|
|
|
|
ga.rusage = ru;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
error = getrusage(td, &ga);
|
1999-01-30 06:29:48 +00:00
|
|
|
if (error)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin(ru, &r, sizeof r)) != 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
tms.tms_utime = timeval_to_clock_t(&r.ru_utime);
|
|
|
|
tms.tms_stime = timeval_to_clock_t(&r.ru_stime);
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
ga.who = RUSAGE_CHILDREN;
|
2001-09-12 08:38:13 +00:00
|
|
|
error = getrusage(td, &ga);
|
1999-01-30 06:29:48 +00:00
|
|
|
if (error)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin(ru, &r, sizeof r)) != 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
tms.tms_cutime = timeval_to_clock_t(&r.ru_utime);
|
|
|
|
tms.tms_cstime = timeval_to_clock_t(&r.ru_stime);
|
|
|
|
|
|
|
|
microtime(&t);
|
|
|
|
*retval = timeval_to_clock_t(&t);
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(&tms, uap->tp, sizeof(tms));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_ulimit(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_ulimit_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
int *retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->cmd) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case SVR4_GFILLIM:
|
2001-01-23 21:30:25 +00:00
|
|
|
/* For p_rlimit below. */
|
|
|
|
mtx_assert(&Giant, MA_OWNED);
|
2001-09-12 08:38:13 +00:00
|
|
|
*retval = td->td_proc->p_rlimit[RLIMIT_FSIZE].rlim_cur / 512;
|
1999-01-30 06:29:48 +00:00
|
|
|
if (*retval == -1)
|
|
|
|
*retval = 0x7fffffff;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case SVR4_SFILLIM:
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
struct __setrlimit_args srl;
|
|
|
|
struct rlimit krl;
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
struct rlimit *url = (struct rlimit *)
|
|
|
|
stackgap_alloc(&sg, sizeof *url);
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
krl.rlim_cur = uap->newlimit * 512;
|
2001-01-23 21:30:25 +00:00
|
|
|
mtx_assert(&Giant, MA_OWNED);
|
2001-09-12 08:38:13 +00:00
|
|
|
krl.rlim_max = td->td_proc->p_rlimit[RLIMIT_FSIZE].rlim_max;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
error = copyout(&krl, url, sizeof(*url));
|
|
|
|
if (error)
|
|
|
|
return error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
srl.which = RLIMIT_FSIZE;
|
|
|
|
srl.rlp = url;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
error = setrlimit(td, &srl);
|
1999-01-30 06:29:48 +00:00
|
|
|
if (error)
|
|
|
|
return error;
|
|
|
|
|
2001-01-23 21:30:25 +00:00
|
|
|
mtx_assert(&Giant, MA_OWNED);
|
2001-09-12 08:38:13 +00:00
|
|
|
*retval = td->td_proc->p_rlimit[RLIMIT_FSIZE].rlim_cur;
|
1999-01-30 06:29:48 +00:00
|
|
|
if (*retval == -1)
|
|
|
|
*retval = 0x7fffffff;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SVR4_GMEMLIM:
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
struct vmspace *vm = td->td_proc->p_vmspace;
|
2001-01-23 21:30:25 +00:00
|
|
|
register_t r;
|
|
|
|
|
|
|
|
mtx_assert(&Giant, MA_OWNED);
|
2001-09-12 08:38:13 +00:00
|
|
|
r = td->td_proc->p_rlimit[RLIMIT_DATA].rlim_cur;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
if (r == -1)
|
|
|
|
r = 0x7fffffff;
|
|
|
|
r += (long) vm->vm_daddr;
|
|
|
|
if (r < 0)
|
|
|
|
r = 0x7fffffff;
|
|
|
|
*retval = r;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SVR4_GDESLIM:
|
2001-01-23 21:30:25 +00:00
|
|
|
mtx_assert(&Giant, MA_OWNED);
|
2001-09-12 08:38:13 +00:00
|
|
|
*retval = td->td_proc->p_rlimit[RLIMIT_NOFILE].rlim_cur;
|
1999-01-30 06:29:48 +00:00
|
|
|
if (*retval == -1)
|
|
|
|
*retval = 0x7fffffff;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct proc *
|
|
|
|
svr4_pfind(pid)
|
|
|
|
pid_t pid;
|
|
|
|
{
|
|
|
|
struct proc *p;
|
|
|
|
|
|
|
|
/* look in the live processes */
|
2001-01-23 21:30:25 +00:00
|
|
|
if ((p = pfind(pid)) == NULL)
|
|
|
|
/* look in the zombies */
|
|
|
|
p = zpfind(pid);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2000-12-13 00:17:05 +00:00
|
|
|
return p;
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_pgrpsys(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_pgrpsys_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
int *retval = td->td_retval;
|
|
|
|
struct proc *p = td->td_proc;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->cmd) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case 1: /* setpgrp() */
|
|
|
|
/*
|
|
|
|
* SVR4 setpgrp() (which takes no arguments) has the
|
|
|
|
* semantics that the session ID is also created anew, so
|
|
|
|
* in almost every sense, setpgrp() is identical to
|
|
|
|
* setsid() for SVR4. (Under BSD, the difference is that
|
|
|
|
* a setpgid(0,0) will not create a new session.)
|
|
|
|
*/
|
2001-09-12 08:38:13 +00:00
|
|
|
setsid(td, NULL);
|
1999-01-30 06:29:48 +00:00
|
|
|
/*FALLTHROUGH*/
|
|
|
|
|
|
|
|
case 0: /* getpgrp() */
|
2002-02-23 11:12:57 +00:00
|
|
|
PROC_LOCK(p);
|
1999-01-30 06:29:48 +00:00
|
|
|
*retval = p->p_pgrp->pg_id;
|
2002-02-23 11:12:57 +00:00
|
|
|
PROC_UNLOCK(p);
|
1999-01-30 06:29:48 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
case 2: /* getsid(pid) */
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->pid == 0)
|
2002-01-19 06:34:58 +00:00
|
|
|
PROC_LOCK(p);
|
2002-12-14 01:56:26 +00:00
|
|
|
else if ((p = svr4_pfind(uap->pid)) == NULL)
|
1999-01-30 06:29:48 +00:00
|
|
|
return ESRCH;
|
|
|
|
/*
|
|
|
|
* This has already been initialized to the pid of
|
|
|
|
* the session leader.
|
|
|
|
*/
|
2002-01-19 05:31:51 +00:00
|
|
|
*retval = (register_t) p->p_session->s_sid;
|
2001-04-24 00:51:53 +00:00
|
|
|
PROC_UNLOCK(p);
|
1999-01-30 06:29:48 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
case 3: /* setsid() */
|
2001-09-12 08:38:13 +00:00
|
|
|
return setsid(td, NULL);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
case 4: /* getpgid(pid) */
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->pid == 0)
|
2002-01-19 06:34:58 +00:00
|
|
|
PROC_LOCK(p);
|
2002-12-14 01:56:26 +00:00
|
|
|
else if ((p = svr4_pfind(uap->pid)) == NULL)
|
1999-01-30 06:29:48 +00:00
|
|
|
return ESRCH;
|
|
|
|
|
|
|
|
*retval = (int) p->p_pgrp->pg_id;
|
2001-04-24 00:51:53 +00:00
|
|
|
PROC_UNLOCK(p);
|
1999-01-30 06:29:48 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
case 5: /* setpgid(pid, pgid); */
|
|
|
|
{
|
|
|
|
struct setpgid_args sa;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
sa.pid = uap->pid;
|
|
|
|
sa.pgid = uap->pgid;
|
2001-09-12 08:38:13 +00:00
|
|
|
return setpgid(td, &sa);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define syscallarg(x) union { x datum; register_t pad; }
|
|
|
|
|
|
|
|
struct svr4_hrtcntl_args {
|
|
|
|
int cmd;
|
|
|
|
int fun;
|
|
|
|
int clk;
|
|
|
|
svr4_hrt_interval_t * iv;
|
|
|
|
svr4_hrt_time_t * ti;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_hrtcntl(td, uap, retval)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_hrtcntl_args *uap;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->fun) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case SVR4_HRT_CNTL_RES:
|
|
|
|
DPRINTF(("htrcntl(RES)\n"));
|
|
|
|
*retval = SVR4_HRT_USEC;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case SVR4_HRT_CNTL_TOFD:
|
|
|
|
DPRINTF(("htrcntl(TOFD)\n"));
|
|
|
|
{
|
|
|
|
struct timeval tv;
|
|
|
|
svr4_hrt_time_t t;
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->clk != SVR4_HRT_CLK_STD) {
|
|
|
|
DPRINTF(("clk == %d\n", uap->clk));
|
1999-01-30 06:29:48 +00:00
|
|
|
return EINVAL;
|
|
|
|
}
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->ti == NULL) {
|
1999-01-30 06:29:48 +00:00
|
|
|
DPRINTF(("ti NULL\n"));
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
microtime(&tv);
|
|
|
|
t.h_sec = tv.tv_sec;
|
|
|
|
t.h_rem = tv.tv_usec;
|
|
|
|
t.h_res = SVR4_HRT_USEC;
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(&t, uap->ti, sizeof(t));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
case SVR4_HRT_CNTL_START:
|
|
|
|
DPRINTF(("htrcntl(START)\n"));
|
|
|
|
return ENOSYS;
|
|
|
|
|
|
|
|
case SVR4_HRT_CNTL_GET:
|
|
|
|
DPRINTF(("htrcntl(GET)\n"));
|
|
|
|
return ENOSYS;
|
|
|
|
default:
|
2002-12-14 01:56:26 +00:00
|
|
|
DPRINTF(("Bad htrcntl command %d\n", uap->fun));
|
1999-01-30 06:29:48 +00:00
|
|
|
return ENOSYS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_hrtsys(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_hrtsys_args *uap;
|
|
|
|
{
|
2001-09-12 08:38:13 +00:00
|
|
|
int *retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->cmd) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case SVR4_HRT_CNTL:
|
2001-09-12 08:38:13 +00:00
|
|
|
return svr4_hrtcntl(td, (struct svr4_hrtcntl_args *) uap,
|
1999-01-30 06:29:48 +00:00
|
|
|
retval);
|
|
|
|
|
|
|
|
case SVR4_HRT_ALRM:
|
|
|
|
DPRINTF(("hrtalarm\n"));
|
|
|
|
return ENOSYS;
|
|
|
|
|
|
|
|
case SVR4_HRT_SLP:
|
|
|
|
DPRINTF(("hrtsleep\n"));
|
|
|
|
return ENOSYS;
|
|
|
|
|
|
|
|
case SVR4_HRT_CAN:
|
|
|
|
DPRINTF(("hrtcancel\n"));
|
|
|
|
return ENOSYS;
|
|
|
|
|
|
|
|
default:
|
2002-12-14 01:56:26 +00:00
|
|
|
DPRINTF(("Bad hrtsys command %d\n", uap->cmd));
|
1999-01-30 06:29:48 +00:00
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
svr4_setinfo(p, st, s)
|
|
|
|
struct proc *p;
|
|
|
|
int st;
|
|
|
|
svr4_siginfo_t *s;
|
|
|
|
{
|
|
|
|
svr4_siginfo_t i;
|
|
|
|
int sig;
|
|
|
|
|
|
|
|
memset(&i, 0, sizeof(i));
|
|
|
|
|
|
|
|
i.si_signo = SVR4_SIGCHLD;
|
|
|
|
i.si_errno = 0; /* XXX? */
|
|
|
|
|
|
|
|
if (p) {
|
|
|
|
i.si_pid = p->p_pid;
|
Change and clean the mutex lock interface.
mtx_enter(lock, type) becomes:
mtx_lock(lock) for sleep locks (MTX_DEF-initialized locks)
mtx_lock_spin(lock) for spin locks (MTX_SPIN-initialized)
similarily, for releasing a lock, we now have:
mtx_unlock(lock) for MTX_DEF and mtx_unlock_spin(lock) for MTX_SPIN.
We change the caller interface for the two different types of locks
because the semantics are entirely different for each case, and this
makes it explicitly clear and, at the same time, it rids us of the
extra `type' argument.
The enter->lock and exit->unlock change has been made with the idea
that we're "locking data" and not "entering locked code" in mind.
Further, remove all additional "flags" previously passed to the
lock acquire/release routines with the exception of two:
MTX_QUIET and MTX_NOSWITCH
The functionality of these flags is preserved and they can be passed
to the lock/unlock routines by calling the corresponding wrappers:
mtx_{lock, unlock}_flags(lock, flag(s)) and
mtx_{lock, unlock}_spin_flags(lock, flag(s)) for MTX_DEF and MTX_SPIN
locks, respectively.
Re-inline some lock acq/rel code; in the sleep lock case, we only
inline the _obtain_lock()s in order to ensure that the inlined code
fits into a cache line. In the spin lock case, we inline recursion and
actually only perform a function call if we need to spin. This change
has been made with the idea that we generally tend to avoid spin locks
and that also the spin locks that we do have and are heavily used
(i.e. sched_lock) do recurse, and therefore in an effort to reduce
function call overhead for some architectures (such as alpha), we
inline recursion for this case.
Create a new malloc type for the witness code and retire from using
the M_DEV type. The new type is called M_WITNESS and is only declared
if WITNESS is enabled.
Begin cleaning up some machdep/mutex.h code - specifically updated the
"optimized" inlined code in alpha/mutex.h and wrote MTX_LOCK_SPIN
and MTX_UNLOCK_SPIN asm macros for the i386/mutex.h as we presently
need those.
Finally, caught up to the interface changes in all sys code.
Contributors: jake, jhb, jasone (in no particular order)
2001-02-09 06:11:45 +00:00
|
|
|
mtx_lock_spin(&sched_lock);
|
Part 1 of KSE-III
The ability to schedule multiple threads per process
(one one cpu) by making ALL system calls optionally asynchronous.
to come: ia64 and power-pc patches, patches for gdb, test program (in tools)
Reviewed by: Almost everyone who counts
(at various times, peter, jhb, matt, alfred, mini, bernd,
and a cast of thousands)
NOTE: this is still Beta code, and contains lots of debugging stuff.
expect slight instability in signals..
2002-06-29 17:26:22 +00:00
|
|
|
if (p->p_state == PRS_ZOMBIE) {
|
1999-01-30 06:29:48 +00:00
|
|
|
i.si_stime = p->p_ru->ru_stime.tv_sec;
|
|
|
|
i.si_utime = p->p_ru->ru_utime.tv_sec;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
i.si_stime = p->p_stats->p_ru.ru_stime.tv_sec;
|
|
|
|
i.si_utime = p->p_stats->p_ru.ru_utime.tv_sec;
|
|
|
|
}
|
Change and clean the mutex lock interface.
mtx_enter(lock, type) becomes:
mtx_lock(lock) for sleep locks (MTX_DEF-initialized locks)
mtx_lock_spin(lock) for spin locks (MTX_SPIN-initialized)
similarily, for releasing a lock, we now have:
mtx_unlock(lock) for MTX_DEF and mtx_unlock_spin(lock) for MTX_SPIN.
We change the caller interface for the two different types of locks
because the semantics are entirely different for each case, and this
makes it explicitly clear and, at the same time, it rids us of the
extra `type' argument.
The enter->lock and exit->unlock change has been made with the idea
that we're "locking data" and not "entering locked code" in mind.
Further, remove all additional "flags" previously passed to the
lock acquire/release routines with the exception of two:
MTX_QUIET and MTX_NOSWITCH
The functionality of these flags is preserved and they can be passed
to the lock/unlock routines by calling the corresponding wrappers:
mtx_{lock, unlock}_flags(lock, flag(s)) and
mtx_{lock, unlock}_spin_flags(lock, flag(s)) for MTX_DEF and MTX_SPIN
locks, respectively.
Re-inline some lock acq/rel code; in the sleep lock case, we only
inline the _obtain_lock()s in order to ensure that the inlined code
fits into a cache line. In the spin lock case, we inline recursion and
actually only perform a function call if we need to spin. This change
has been made with the idea that we generally tend to avoid spin locks
and that also the spin locks that we do have and are heavily used
(i.e. sched_lock) do recurse, and therefore in an effort to reduce
function call overhead for some architectures (such as alpha), we
inline recursion for this case.
Create a new malloc type for the witness code and retire from using
the M_DEV type. The new type is called M_WITNESS and is only declared
if WITNESS is enabled.
Begin cleaning up some machdep/mutex.h code - specifically updated the
"optimized" inlined code in alpha/mutex.h and wrote MTX_LOCK_SPIN
and MTX_UNLOCK_SPIN asm macros for the i386/mutex.h as we presently
need those.
Finally, caught up to the interface changes in all sys code.
Contributors: jake, jhb, jasone (in no particular order)
2001-02-09 06:11:45 +00:00
|
|
|
mtx_unlock_spin(&sched_lock);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (WIFEXITED(st)) {
|
|
|
|
i.si_status = WEXITSTATUS(st);
|
|
|
|
i.si_code = SVR4_CLD_EXITED;
|
|
|
|
} else if (WIFSTOPPED(st)) {
|
|
|
|
sig = WSTOPSIG(st);
|
|
|
|
if (sig >= 0 && sig < NSIG)
|
1999-09-29 15:12:18 +00:00
|
|
|
i.si_status = SVR4_BSD2SVR4_SIG(sig);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
if (i.si_status == SVR4_SIGCONT)
|
|
|
|
i.si_code = SVR4_CLD_CONTINUED;
|
|
|
|
else
|
|
|
|
i.si_code = SVR4_CLD_STOPPED;
|
|
|
|
} else {
|
|
|
|
sig = WTERMSIG(st);
|
|
|
|
if (sig >= 0 && sig < NSIG)
|
1999-09-29 15:12:18 +00:00
|
|
|
i.si_status = SVR4_BSD2SVR4_SIG(sig);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
if (WCOREDUMP(st))
|
|
|
|
i.si_code = SVR4_CLD_DUMPED;
|
|
|
|
else
|
|
|
|
i.si_code = SVR4_CLD_KILLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPRINTF(("siginfo [pid %ld signo %d code %d errno %d status %d]\n",
|
|
|
|
i.si_pid, i.si_signo, i.si_code, i.si_errno, i.si_status));
|
|
|
|
|
|
|
|
return copyout(&i, s, sizeof(i));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_waitsys(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_waitsys_args *uap;
|
|
|
|
{
|
|
|
|
int nfound;
|
2001-09-12 08:38:13 +00:00
|
|
|
int error, *retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct proc *q, *t;
|
|
|
|
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->grp) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case SVR4_P_PID:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SVR4_P_PGID:
|
2002-02-23 11:12:57 +00:00
|
|
|
PROC_LOCK(td->td_proc);
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->id = -td->td_proc->p_pgid;
|
2002-02-23 11:12:57 +00:00
|
|
|
PROC_UNLOCK(td->td_proc);
|
1999-01-30 06:29:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SVR4_P_ALL:
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->id = WAIT_ANY;
|
1999-01-30 06:29:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPRINTF(("waitsys(%d, %d, %p, %x)\n",
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->grp, uap->id,
|
|
|
|
uap->info, uap->options));
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
loop:
|
|
|
|
nfound = 0;
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_slock(&proctree_lock);
|
2001-09-12 08:38:13 +00:00
|
|
|
LIST_FOREACH(q, &td->td_proc->p_children, p_sibling) {
|
2002-01-13 11:58:06 +00:00
|
|
|
PROC_LOCK(q);
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->id != WAIT_ANY &&
|
|
|
|
q->p_pid != uap->id &&
|
|
|
|
q->p_pgid != -uap->id) {
|
2002-01-13 11:58:06 +00:00
|
|
|
PROC_UNLOCK(q);
|
1999-01-30 06:29:48 +00:00
|
|
|
DPRINTF(("pid %d pgid %d != %d\n", q->p_pid,
|
2002-12-14 01:56:26 +00:00
|
|
|
q->p_pgid, uap->id));
|
1999-01-30 06:29:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
nfound++;
|
Part 1 of KSE-III
The ability to schedule multiple threads per process
(one one cpu) by making ALL system calls optionally asynchronous.
to come: ia64 and power-pc patches, patches for gdb, test program (in tools)
Reviewed by: Almost everyone who counts
(at various times, peter, jhb, matt, alfred, mini, bernd,
and a cast of thousands)
NOTE: this is still Beta code, and contains lots of debugging stuff.
expect slight instability in signals..
2002-06-29 17:26:22 +00:00
|
|
|
if ((q->p_state == PRS_ZOMBIE) &&
|
2002-12-14 01:56:26 +00:00
|
|
|
((uap->options & (SVR4_WEXITED|SVR4_WTRAPPED)))) {
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_UNLOCK(q);
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_sunlock(&proctree_lock);
|
1999-01-30 06:29:48 +00:00
|
|
|
*retval = 0;
|
|
|
|
DPRINTF(("found %d\n", q->p_pid));
|
2002-12-14 01:56:26 +00:00
|
|
|
error = svr4_setinfo(q, q->p_xstat, uap->info);
|
2001-01-23 21:30:25 +00:00
|
|
|
if (error != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if ((uap->options & SVR4_WNOWAIT)) {
|
1999-01-30 06:29:48 +00:00
|
|
|
DPRINTF(("Don't wait\n"));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we got the child via ptrace(2) or procfs, and
|
|
|
|
* the parent is different (meaning the process was
|
|
|
|
* attached, rather than run as a child), then we need
|
2001-01-23 21:30:25 +00:00
|
|
|
* to give it back to the old parent, and send the
|
1999-01-30 06:29:48 +00:00
|
|
|
* parent a SIGCHLD. The rest of the cleanup will be
|
|
|
|
* done when the old parent waits on the child.
|
|
|
|
*/
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_xlock(&proctree_lock);
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_LOCK(q);
|
2000-12-23 19:43:10 +00:00
|
|
|
if (q->p_flag & P_TRACED) {
|
|
|
|
if (q->p_oppid != q->p_pptr->p_pid) {
|
2001-03-07 02:17:43 +00:00
|
|
|
PROC_UNLOCK(q);
|
2000-12-23 19:43:10 +00:00
|
|
|
t = pfind(q->p_oppid);
|
2002-02-23 11:12:57 +00:00
|
|
|
if (t == NULL) {
|
|
|
|
t = initproc;
|
|
|
|
PROC_LOCK(initproc);
|
|
|
|
}
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_LOCK(q);
|
2002-02-23 11:12:57 +00:00
|
|
|
proc_reparent(q, t);
|
2001-03-07 02:17:43 +00:00
|
|
|
q->p_oppid = 0;
|
2000-12-23 19:43:10 +00:00
|
|
|
q->p_flag &= ~(P_TRACED | P_WAITED);
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_UNLOCK(q);
|
2001-03-07 02:17:43 +00:00
|
|
|
psignal(t, SIGCHLD);
|
2001-04-24 00:51:53 +00:00
|
|
|
wakeup(t);
|
2001-03-07 02:17:43 +00:00
|
|
|
PROC_UNLOCK(t);
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_xunlock(&proctree_lock);
|
2000-12-23 19:43:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2001-03-07 02:17:43 +00:00
|
|
|
}
|
|
|
|
PROC_UNLOCK(q);
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_xunlock(&proctree_lock);
|
1999-01-30 06:29:48 +00:00
|
|
|
q->p_xstat = 0;
|
2001-09-12 08:38:13 +00:00
|
|
|
ruadd(&td->td_proc->p_stats->p_cru, q->p_ru);
|
1999-01-30 06:29:48 +00:00
|
|
|
FREE(q->p_ru, M_ZOMBIE);
|
2001-01-23 21:30:25 +00:00
|
|
|
q->p_ru = 0;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
/*
|
2001-01-23 21:30:25 +00:00
|
|
|
* Decrement the count of procs running with this uid.
|
1999-01-30 06:29:48 +00:00
|
|
|
*/
|
o Merge contents of struct pcred into struct ucred. Specifically, add the
real uid, saved uid, real gid, and saved gid to ucred, as well as the
pcred->pc_uidinfo, which was associated with the real uid, only rename
it to cr_ruidinfo so as not to conflict with cr_uidinfo, which
corresponds to the effective uid.
o Remove p_cred from struct proc; add p_ucred to struct proc, replacing
original macro that pointed.
p->p_ucred to p->p_cred->pc_ucred.
o Universally update code so that it makes use of ucred instead of pcred,
p->p_ucred instead of p->p_pcred, cr_ruidinfo instead of p_uidinfo,
cr_{r,sv}{u,g}id instead of p_*, etc.
o Remove pcred0 and its initialization from init_main.c; initialize
cr_ruidinfo there.
o Restruction many credential modification chunks to always crdup while
we figure out locking and optimizations; generally speaking, this
means moving to a structure like this:
newcred = crdup(oldcred);
...
p->p_ucred = newcred;
crfree(oldcred);
It's not race-free, but better than nothing. There are also races
in sys_process.c, all inter-process authorization, fork, exec, and
exit.
o Remove sigio->sio_ruid since sigio->sio_ucred now contains the ruid;
remove comments indicating that the old arrangement was a problem.
o Restructure exec1() a little to use newcred/oldcred arrangement, and
use improved uid management primitives.
o Clean up exit1() so as to do less work in credential cleanup due to
pcred removal.
o Clean up fork1() so as to do less work in credential cleanup and
allocation.
o Clean up ktrcanset() to take into account changes, and move to using
suser_xxx() instead of performing a direct uid==0 comparision.
o Improve commenting in various kern_prot.c credential modification
calls to better document current behavior. In a couple of places,
current behavior is a little questionable and we need to check
POSIX.1 to make sure it's "right". More commenting work still
remains to be done.
o Update credential management calls, such as crfree(), to take into
account new ruidinfo reference.
o Modify or add the following uid and gid helper routines:
change_euid()
change_egid()
change_ruid()
change_rgid()
change_svuid()
change_svgid()
In each case, the call now acts on a credential not a process, and as
such no longer requires more complicated process locking/etc. They
now assume the caller will do any necessary allocation of an
exclusive credential reference. Each is commented to document its
reference requirements.
o CANSIGIO() is simplified to require only credentials, not processes
and pcreds.
o Remove lots of (p_pcred==NULL) checks.
o Add an XXX to authorization code in nfs_lock.c, since it's
questionable, and needs to be considered carefully.
o Simplify posix4 authorization code to require only credentials, not
processes and pcreds. Note that this authorization, as well as
CANSIGIO(), needs to be updated to use the p_cansignal() and
p_cansched() centralized authorization routines, as they currently
do not take into account some desirable restrictions that are handled
by the centralized routines, as well as being inconsistent with other
similar authorization instances.
o Update libkvm to take these changes into account.
Obtained from: TrustedBSD Project
Reviewed by: green, bde, jhb, freebsd-arch, freebsd-audit
2001-05-25 16:59:11 +00:00
|
|
|
(void)chgproccnt(q->p_ucred->cr_ruidinfo, -1, 0);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
/*
|
2001-01-23 21:30:25 +00:00
|
|
|
* Release reference to text vnode.
|
1999-01-30 06:29:48 +00:00
|
|
|
*/
|
2001-01-23 21:30:25 +00:00
|
|
|
if (q->p_textvp)
|
|
|
|
vrele(q->p_textvp);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Free up credentials.
|
|
|
|
*/
|
o Merge contents of struct pcred into struct ucred. Specifically, add the
real uid, saved uid, real gid, and saved gid to ucred, as well as the
pcred->pc_uidinfo, which was associated with the real uid, only rename
it to cr_ruidinfo so as not to conflict with cr_uidinfo, which
corresponds to the effective uid.
o Remove p_cred from struct proc; add p_ucred to struct proc, replacing
original macro that pointed.
p->p_ucred to p->p_cred->pc_ucred.
o Universally update code so that it makes use of ucred instead of pcred,
p->p_ucred instead of p->p_pcred, cr_ruidinfo instead of p_uidinfo,
cr_{r,sv}{u,g}id instead of p_*, etc.
o Remove pcred0 and its initialization from init_main.c; initialize
cr_ruidinfo there.
o Restruction many credential modification chunks to always crdup while
we figure out locking and optimizations; generally speaking, this
means moving to a structure like this:
newcred = crdup(oldcred);
...
p->p_ucred = newcred;
crfree(oldcred);
It's not race-free, but better than nothing. There are also races
in sys_process.c, all inter-process authorization, fork, exec, and
exit.
o Remove sigio->sio_ruid since sigio->sio_ucred now contains the ruid;
remove comments indicating that the old arrangement was a problem.
o Restructure exec1() a little to use newcred/oldcred arrangement, and
use improved uid management primitives.
o Clean up exit1() so as to do less work in credential cleanup due to
pcred removal.
o Clean up fork1() so as to do less work in credential cleanup and
allocation.
o Clean up ktrcanset() to take into account changes, and move to using
suser_xxx() instead of performing a direct uid==0 comparision.
o Improve commenting in various kern_prot.c credential modification
calls to better document current behavior. In a couple of places,
current behavior is a little questionable and we need to check
POSIX.1 to make sure it's "right". More commenting work still
remains to be done.
o Update credential management calls, such as crfree(), to take into
account new ruidinfo reference.
o Modify or add the following uid and gid helper routines:
change_euid()
change_egid()
change_ruid()
change_rgid()
change_svuid()
change_svgid()
In each case, the call now acts on a credential not a process, and as
such no longer requires more complicated process locking/etc. They
now assume the caller will do any necessary allocation of an
exclusive credential reference. Each is commented to document its
reference requirements.
o CANSIGIO() is simplified to require only credentials, not processes
and pcreds.
o Remove lots of (p_pcred==NULL) checks.
o Add an XXX to authorization code in nfs_lock.c, since it's
questionable, and needs to be considered carefully.
o Simplify posix4 authorization code to require only credentials, not
processes and pcreds. Note that this authorization, as well as
CANSIGIO(), needs to be updated to use the p_cansignal() and
p_cansched() centralized authorization routines, as they currently
do not take into account some desirable restrictions that are handled
by the centralized routines, as well as being inconsistent with other
similar authorization instances.
o Update libkvm to take these changes into account.
Obtained from: TrustedBSD Project
Reviewed by: green, bde, jhb, freebsd-arch, freebsd-audit
2001-05-25 16:59:11 +00:00
|
|
|
crfree(q->p_ucred);
|
|
|
|
q->p_ucred = NULL;
|
2001-01-23 21:30:25 +00:00
|
|
|
|
1999-01-30 06:29:48 +00:00
|
|
|
/*
|
2001-01-23 21:30:25 +00:00
|
|
|
* Remove unused arguments
|
1999-01-30 06:29:48 +00:00
|
|
|
*/
|
2002-03-27 21:36:18 +00:00
|
|
|
pargs_drop(q->p_args);
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_UNLOCK(q);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Finally finished with old proc entry.
|
|
|
|
* Unlink it from its process group and free it.
|
|
|
|
*/
|
2002-04-16 17:04:21 +00:00
|
|
|
sx_xlock(&proctree_lock);
|
2001-01-23 21:30:25 +00:00
|
|
|
leavepgrp(q);
|
|
|
|
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_xlock(&allproc_lock);
|
2001-01-23 21:30:25 +00:00
|
|
|
LIST_REMOVE(q, p_list); /* off zombproc */
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_xunlock(&allproc_lock);
|
2001-01-23 21:30:25 +00:00
|
|
|
|
|
|
|
LIST_REMOVE(q, p_sibling);
|
2001-03-28 11:52:56 +00:00
|
|
|
sx_xunlock(&proctree_lock);
|
2001-01-23 21:30:25 +00:00
|
|
|
|
|
|
|
PROC_LOCK(q);
|
- Merge struct procsig with struct sigacts.
- Move struct sigacts out of the u-area and malloc() it using the
M_SUBPROC malloc bucket.
- Add a small sigacts_*() API for managing sigacts structures: sigacts_alloc(),
sigacts_free(), sigacts_copy(), sigacts_share(), and sigacts_shared().
- Remove the p_sigignore, p_sigacts, and p_sigcatch macros.
- Add a mutex to struct sigacts that protects all the members of the struct.
- Add sigacts locking.
- Remove Giant from nosys(), kill(), killpg(), and kern_sigaction() now
that sigacts is locked.
- Several in-kernel functions such as psignal(), tdsignal(), trapsignal(),
and thread_stopped() are now MP safe.
Reviewed by: arch@
Approved by: re (rwatson)
2003-05-13 20:36:02 +00:00
|
|
|
sigacts_free(q->p_sigacts);
|
|
|
|
q->p_sigacts = NULL;
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_UNLOCK(q);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Give machine-dependent layer a chance
|
|
|
|
* to free anything that cpu_exit couldn't
|
|
|
|
* release while still running in process context.
|
|
|
|
*/
|
2003-01-13 22:44:23 +00:00
|
|
|
vm_waitproc(q);
|
1999-01-30 06:29:48 +00:00
|
|
|
#if defined(__NetBSD__)
|
|
|
|
pool_put(&proc_pool, q);
|
2001-01-23 21:30:25 +00:00
|
|
|
#endif
|
|
|
|
#ifdef __FreeBSD__
|
|
|
|
mtx_destroy(&q->p_mtx);
|
2003-01-13 22:44:23 +00:00
|
|
|
#ifdef MAC
|
|
|
|
mac_destroy_proc(q);
|
|
|
|
#endif
|
2002-03-20 10:35:22 +00:00
|
|
|
uma_zfree(proc_zone, q);
|
1999-01-30 06:29:48 +00:00
|
|
|
#endif
|
|
|
|
nprocs--;
|
|
|
|
return 0;
|
|
|
|
}
|
Part 1 of KSE-III
The ability to schedule multiple threads per process
(one one cpu) by making ALL system calls optionally asynchronous.
to come: ia64 and power-pc patches, patches for gdb, test program (in tools)
Reviewed by: Almost everyone who counts
(at various times, peter, jhb, matt, alfred, mini, bernd,
and a cast of thousands)
NOTE: this is still Beta code, and contains lots of debugging stuff.
expect slight instability in signals..
2002-06-29 17:26:22 +00:00
|
|
|
/* XXXKSE this needs clarification */
|
|
|
|
if (P_SHOULDSTOP(q) && ((q->p_flag & P_WAITED) == 0) &&
|
1999-01-30 06:29:48 +00:00
|
|
|
(q->p_flag & P_TRACED ||
|
2002-12-14 01:56:26 +00:00
|
|
|
(uap->options & (SVR4_WSTOPPED|SVR4_WCONTINUED)))) {
|
1999-01-30 06:29:48 +00:00
|
|
|
DPRINTF(("jobcontrol %d\n", q->p_pid));
|
2002-12-14 01:56:26 +00:00
|
|
|
if (((uap->options & SVR4_WNOWAIT)) == 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
q->p_flag |= P_WAITED;
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_UNLOCK(q);
|
1999-01-30 06:29:48 +00:00
|
|
|
*retval = 0;
|
|
|
|
return svr4_setinfo(q, W_STOPCODE(q->p_xstat),
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->info);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
2001-01-23 21:30:25 +00:00
|
|
|
PROC_UNLOCK(q);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (nfound == 0)
|
|
|
|
return ECHILD;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->options & SVR4_WNOHANG) {
|
1999-01-30 06:29:48 +00:00
|
|
|
*retval = 0;
|
2002-12-14 01:56:26 +00:00
|
|
|
if ((error = svr4_setinfo(NULL, 0, uap->info)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-03-02 16:54:40 +00:00
|
|
|
if ((error = tsleep(td->td_proc, PWAIT | PCATCH, "svr4_wait", 0)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
goto loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
bsd_statfs_to_svr4_statvfs(bfs, sfs)
|
|
|
|
const struct statfs *bfs;
|
|
|
|
struct svr4_statvfs *sfs;
|
|
|
|
{
|
|
|
|
sfs->f_bsize = bfs->f_iosize; /* XXX */
|
|
|
|
sfs->f_frsize = bfs->f_bsize;
|
|
|
|
sfs->f_blocks = bfs->f_blocks;
|
|
|
|
sfs->f_bfree = bfs->f_bfree;
|
|
|
|
sfs->f_bavail = bfs->f_bavail;
|
|
|
|
sfs->f_files = bfs->f_files;
|
|
|
|
sfs->f_ffree = bfs->f_ffree;
|
|
|
|
sfs->f_favail = bfs->f_ffree;
|
|
|
|
sfs->f_fsid = bfs->f_fsid.val[0];
|
|
|
|
memcpy(sfs->f_basetype, bfs->f_fstypename, sizeof(sfs->f_basetype));
|
|
|
|
sfs->f_flag = 0;
|
|
|
|
if (bfs->f_flags & MNT_RDONLY)
|
|
|
|
sfs->f_flag |= SVR4_ST_RDONLY;
|
|
|
|
if (bfs->f_flags & MNT_NOSUID)
|
|
|
|
sfs->f_flag |= SVR4_ST_NOSUID;
|
|
|
|
sfs->f_namemax = MAXNAMLEN;
|
|
|
|
memcpy(sfs->f_fstr, bfs->f_fstypename, sizeof(sfs->f_fstr)); /* XXX */
|
|
|
|
memset(sfs->f_filler, 0, sizeof(sfs->f_filler));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
bsd_statfs_to_svr4_statvfs64(bfs, sfs)
|
|
|
|
const struct statfs *bfs;
|
|
|
|
struct svr4_statvfs64 *sfs;
|
|
|
|
{
|
|
|
|
sfs->f_bsize = bfs->f_iosize; /* XXX */
|
|
|
|
sfs->f_frsize = bfs->f_bsize;
|
|
|
|
sfs->f_blocks = bfs->f_blocks;
|
|
|
|
sfs->f_bfree = bfs->f_bfree;
|
|
|
|
sfs->f_bavail = bfs->f_bavail;
|
|
|
|
sfs->f_files = bfs->f_files;
|
|
|
|
sfs->f_ffree = bfs->f_ffree;
|
|
|
|
sfs->f_favail = bfs->f_ffree;
|
|
|
|
sfs->f_fsid = bfs->f_fsid.val[0];
|
|
|
|
memcpy(sfs->f_basetype, bfs->f_fstypename, sizeof(sfs->f_basetype));
|
|
|
|
sfs->f_flag = 0;
|
|
|
|
if (bfs->f_flags & MNT_RDONLY)
|
|
|
|
sfs->f_flag |= SVR4_ST_RDONLY;
|
|
|
|
if (bfs->f_flags & MNT_NOSUID)
|
|
|
|
sfs->f_flag |= SVR4_ST_NOSUID;
|
|
|
|
sfs->f_namemax = MAXNAMLEN;
|
|
|
|
memcpy(sfs->f_fstr, bfs->f_fstypename, sizeof(sfs->f_fstr)); /* XXX */
|
|
|
|
memset(sfs->f_filler, 0, sizeof(sfs->f_filler));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_statvfs(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_statvfs_args *uap;
|
|
|
|
{
|
|
|
|
struct statfs_args fs_args;
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
struct statfs *fs = stackgap_alloc(&sg, sizeof(struct statfs));
|
|
|
|
struct statfs bfs;
|
|
|
|
struct svr4_statvfs sfs;
|
|
|
|
int error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
CHECKALTEXIST(td, &sg, uap->path);
|
|
|
|
fs_args.path = uap->path;
|
|
|
|
fs_args.buf = fs;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((error = statfs(td, &fs_args)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin(fs, &bfs, sizeof(bfs))) != 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
bsd_statfs_to_svr4_statvfs(&bfs, &sfs);
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(&sfs, uap->fs, sizeof(sfs));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_fstatvfs(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_fstatvfs_args *uap;
|
|
|
|
{
|
|
|
|
struct fstatfs_args fs_args;
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
struct statfs *fs = stackgap_alloc(&sg, sizeof(struct statfs));
|
|
|
|
struct statfs bfs;
|
|
|
|
struct svr4_statvfs sfs;
|
|
|
|
int error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
fs_args.fd = uap->fd;
|
|
|
|
fs_args.buf = fs;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((error = fstatfs(td, &fs_args)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin(fs, &bfs, sizeof(bfs))) != 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
bsd_statfs_to_svr4_statvfs(&bfs, &sfs);
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(&sfs, uap->fs, sizeof(sfs));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_statvfs64(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_statvfs64_args *uap;
|
|
|
|
{
|
|
|
|
struct statfs_args fs_args;
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
struct statfs *fs = stackgap_alloc(&sg, sizeof(struct statfs));
|
|
|
|
struct statfs bfs;
|
|
|
|
struct svr4_statvfs64 sfs;
|
|
|
|
int error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
CHECKALTEXIST(td, &sg, uap->path);
|
|
|
|
fs_args.path = uap->path;
|
|
|
|
fs_args.buf = fs;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((error = statfs(td, &fs_args)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin(fs, &bfs, sizeof(bfs))) != 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
bsd_statfs_to_svr4_statvfs64(&bfs, &sfs);
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(&sfs, uap->fs, sizeof(sfs));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_fstatvfs64(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_fstatvfs64_args *uap;
|
|
|
|
{
|
|
|
|
struct fstatfs_args fs_args;
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
struct statfs *fs = stackgap_alloc(&sg, sizeof(struct statfs));
|
|
|
|
struct statfs bfs;
|
|
|
|
struct svr4_statvfs64 sfs;
|
|
|
|
int error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
fs_args.fd = uap->fd;
|
|
|
|
fs_args.buf = fs;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((error = fstatfs(td, &fs_args)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin(fs, &bfs, sizeof(bfs))) != 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
bsd_statfs_to_svr4_statvfs64(&bfs, &sfs);
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(&sfs, uap->fs, sizeof(sfs));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_alarm(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_alarm_args *uap;
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
struct itimerval *itp, *oitp;
|
|
|
|
struct setitimer_args sa;
|
|
|
|
caddr_t sg = stackgap_init();
|
|
|
|
|
|
|
|
itp = stackgap_alloc(&sg, sizeof(*itp));
|
|
|
|
oitp = stackgap_alloc(&sg, sizeof(*oitp));
|
|
|
|
timevalclear(&itp->it_interval);
|
2002-12-14 01:56:26 +00:00
|
|
|
itp->it_value.tv_sec = uap->sec;
|
1999-01-30 06:29:48 +00:00
|
|
|
itp->it_value.tv_usec = 0;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
sa.which = ITIMER_REAL;
|
|
|
|
sa.itv = itp;
|
|
|
|
sa.oitv = oitp;
|
2001-09-12 08:38:13 +00:00
|
|
|
error = setitimer(td, &sa);
|
1999-01-30 06:29:48 +00:00
|
|
|
if (error)
|
|
|
|
return error;
|
|
|
|
if (oitp->it_value.tv_usec)
|
|
|
|
oitp->it_value.tv_sec++;
|
2001-09-12 08:38:13 +00:00
|
|
|
td->td_retval[0] = oitp->it_value.tv_sec;
|
1999-01-30 06:29:48 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_gettimeofday(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_gettimeofday_args *uap;
|
|
|
|
{
|
2002-12-14 01:56:26 +00:00
|
|
|
if (uap->tp) {
|
1999-01-30 06:29:48 +00:00
|
|
|
struct timeval atv;
|
|
|
|
|
|
|
|
microtime(&atv);
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(&atv, uap->tp, sizeof (atv));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_facl(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_facl_args *uap;
|
|
|
|
{
|
|
|
|
int *retval;
|
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
*retval = 0;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->cmd) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case SVR4_SYS_SETACL:
|
|
|
|
/* We don't support acls on any filesystem */
|
|
|
|
return ENOSYS;
|
|
|
|
|
|
|
|
case SVR4_SYS_GETACL:
|
2002-12-14 01:56:26 +00:00
|
|
|
return copyout(retval, &uap->num,
|
|
|
|
sizeof(uap->num));
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
case SVR4_SYS_GETACLCNT:
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_acl(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_acl_args *uap;
|
|
|
|
{
|
|
|
|
/* XXX: for now the same */
|
2001-09-12 08:38:13 +00:00
|
|
|
return svr4_sys_facl(td, (struct svr4_sys_facl_args *)uap);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_auditsys(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_auditsys_args *uap;
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* XXX: Big brother is *not* watching.
|
|
|
|
*/
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_memcntl(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_memcntl_args *uap;
|
|
|
|
{
|
2002-12-14 01:56:26 +00:00
|
|
|
switch (uap->cmd) {
|
1999-01-30 06:29:48 +00:00
|
|
|
case SVR4_MC_SYNC:
|
|
|
|
{
|
|
|
|
struct msync_args msa;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
msa.addr = uap->addr;
|
|
|
|
msa.len = uap->len;
|
|
|
|
msa.flags = (int)uap->arg;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
return msync(td, &msa);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
case SVR4_MC_ADVISE:
|
|
|
|
{
|
|
|
|
struct madvise_args maa;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
maa.addr = uap->addr;
|
|
|
|
maa.len = uap->len;
|
|
|
|
maa.behav = (int)uap->arg;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
return madvise(td, &maa);
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
case SVR4_MC_LOCK:
|
|
|
|
case SVR4_MC_UNLOCK:
|
|
|
|
case SVR4_MC_LOCKAS:
|
|
|
|
case SVR4_MC_UNLOCKAS:
|
|
|
|
return EOPNOTSUPP;
|
|
|
|
default:
|
|
|
|
return ENOSYS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_nice(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_nice_args *uap;
|
|
|
|
{
|
|
|
|
struct setpriority_args ap;
|
|
|
|
int error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
ap.which = PRIO_PROCESS;
|
|
|
|
ap.who = 0;
|
|
|
|
ap.prio = uap->prio;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((error = setpriority(td, &ap)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
/* the cast is stupid, but the structures are the same */
|
2001-09-12 08:38:13 +00:00
|
|
|
if ((error = getpriority(td, (struct getpriority_args *)&ap)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
return error;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
svr4_sys_resolvepath(td, uap)
|
|
|
|
struct thread *td;
|
1999-01-30 06:29:48 +00:00
|
|
|
struct svr4_sys_resolvepath_args *uap;
|
|
|
|
{
|
|
|
|
struct nameidata nd;
|
2001-09-12 08:38:13 +00:00
|
|
|
int error, *retval = td->td_retval;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
NDINIT(&nd, LOOKUP, NOFOLLOW | SAVENAME, UIO_USERSPACE,
|
2002-12-14 01:56:26 +00:00
|
|
|
uap->path, td);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
if ((error = namei(&nd)) != 0)
|
|
|
|
return error;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
if ((error = copyout(nd.ni_cnd.cn_pnbuf, uap->buf,
|
|
|
|
uap->bufsiz)) != 0)
|
1999-01-30 06:29:48 +00:00
|
|
|
goto bad;
|
|
|
|
|
2002-12-14 01:56:26 +00:00
|
|
|
*retval = strlen(nd.ni_cnd.cn_pnbuf) < uap->bufsiz ?
|
|
|
|
strlen(nd.ni_cnd.cn_pnbuf) + 1 : uap->bufsiz;
|
1999-01-30 06:29:48 +00:00
|
|
|
bad:
|
1999-12-15 23:02:35 +00:00
|
|
|
NDFREE(&nd, NDF_ONLY_PNBUF);
|
1999-01-30 06:29:48 +00:00
|
|
|
vput(nd.ni_vp);
|
|
|
|
return error;
|
|
|
|
}
|