2004-04-21 06:37:20 +04:00
|
|
|
/* $NetBSD: linux_misc.c,v 1.125 2004/04/21 02:37:20 christos Exp $ */
|
1998-10-01 08:05:54 +04:00
|
|
|
|
|
|
|
/*-
|
1999-05-14 03:42:34 +04:00
|
|
|
* Copyright (c) 1995, 1998, 1999 The NetBSD Foundation, Inc.
|
1998-10-01 08:05:54 +04:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
1999-05-14 03:42:34 +04:00
|
|
|
* by Frank van der Linden and Eric Haszlakiewicz; by Jason R. Thorpe
|
|
|
|
* of the Numerical Aerospace Simulation Facility, NASA Ames Research Center.
|
1998-10-01 08:05:54 +04:00
|
|
|
*
|
|
|
|
* 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. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by the NetBSD
|
|
|
|
* Foundation, Inc. and its contributors.
|
|
|
|
* 4. Neither the name of The NetBSD Foundation 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 NETBSD FOUNDATION, INC. 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 FOUNDATION 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.
|
|
|
|
*/
|
1995-03-01 02:24:35 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Linux compatibility module. Try to deal with various Linux system calls.
|
|
|
|
*/
|
|
|
|
|
1998-10-01 08:05:54 +04:00
|
|
|
/*
|
|
|
|
* These functions have been moved to multiarch to allow
|
|
|
|
* selection of which machines include them to be
|
|
|
|
* determined by the individual files.linux_<arch> files.
|
|
|
|
*
|
|
|
|
* Function in multiarch:
|
|
|
|
* linux_sys_break : linux_break.c
|
|
|
|
* linux_sys_alarm : linux_misc_notalpha.c
|
1999-05-14 22:44:50 +04:00
|
|
|
* linux_sys_getresgid : linux_misc_notalpha.c
|
1998-10-01 08:05:54 +04:00
|
|
|
* linux_sys_nice : linux_misc_notalpha.c
|
|
|
|
* linux_sys_readdir : linux_misc_notalpha.c
|
1999-05-14 22:44:50 +04:00
|
|
|
* linux_sys_setresgid : linux_misc_notalpha.c
|
1998-10-01 08:05:54 +04:00
|
|
|
* linux_sys_time : linux_misc_notalpha.c
|
|
|
|
* linux_sys_utime : linux_misc_notalpha.c
|
|
|
|
* linux_sys_waitpid : linux_misc_notalpha.c
|
|
|
|
* linux_sys_old_mmap : linux_oldmmap.c
|
|
|
|
* linux_sys_oldolduname : linux_oldolduname.c
|
|
|
|
* linux_sys_oldselect : linux_oldselect.c
|
|
|
|
* linux_sys_olduname : linux_olduname.c
|
|
|
|
* linux_sys_pipe : linux_pipe.c
|
|
|
|
*/
|
|
|
|
|
2001-11-13 05:07:52 +03:00
|
|
|
#include <sys/cdefs.h>
|
2004-04-21 06:37:20 +04:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: linux_misc.c,v 1.125 2004/04/21 02:37:20 christos Exp $");
|
2001-11-13 05:07:52 +03:00
|
|
|
|
1995-03-01 02:24:35 +03:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/namei.h>
|
|
|
|
#include <sys/proc.h>
|
1996-08-10 13:08:26 +04:00
|
|
|
#include <sys/dirent.h>
|
1995-03-01 02:24:35 +03:00
|
|
|
#include <sys/file.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/filedesc.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/malloc.h>
|
|
|
|
#include <sys/mbuf.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <sys/mount.h>
|
2000-03-19 01:23:13 +03:00
|
|
|
#include <sys/reboot.h>
|
1995-03-01 02:24:35 +03:00
|
|
|
#include <sys/resource.h>
|
|
|
|
#include <sys/resourcevar.h>
|
|
|
|
#include <sys/signal.h>
|
|
|
|
#include <sys/signalvar.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/times.h>
|
|
|
|
#include <sys/vnode.h>
|
|
|
|
#include <sys/uio.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <sys/utsname.h>
|
|
|
|
#include <sys/unistd.h>
|
2000-12-02 00:14:42 +03:00
|
|
|
#include <sys/swap.h> /* for SWAP_ON */
|
|
|
|
#include <sys/sysctl.h> /* for KERN_DOMAINNAME */
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2000-11-01 23:56:30 +03:00
|
|
|
#include <sys/ptrace.h>
|
|
|
|
#include <machine/ptrace.h>
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
#include <sys/sa.h>
|
1995-03-01 02:24:35 +03:00
|
|
|
#include <sys/syscallargs.h>
|
|
|
|
|
1998-10-04 00:17:36 +04:00
|
|
|
#include <compat/linux/common/linux_types.h>
|
|
|
|
#include <compat/linux/common/linux_signal.h>
|
|
|
|
|
1995-03-01 02:24:35 +03:00
|
|
|
#include <compat/linux/linux_syscallargs.h>
|
1998-10-04 00:17:36 +04:00
|
|
|
|
|
|
|
#include <compat/linux/common/linux_fcntl.h>
|
|
|
|
#include <compat/linux/common/linux_mmap.h>
|
|
|
|
#include <compat/linux/common/linux_dirent.h>
|
|
|
|
#include <compat/linux/common/linux_util.h>
|
|
|
|
#include <compat/linux/common/linux_misc.h>
|
1999-12-16 18:09:49 +03:00
|
|
|
#include <compat/linux/common/linux_ptrace.h>
|
2000-03-19 01:23:13 +03:00
|
|
|
#include <compat/linux/common/linux_reboot.h>
|
2001-03-15 22:18:20 +03:00
|
|
|
#include <compat/linux/common/linux_emuldata.h>
|
1999-12-16 18:09:49 +03:00
|
|
|
|
2000-11-01 23:56:30 +03:00
|
|
|
const int linux_ptrace_request_map[] = {
|
1999-12-16 18:09:49 +03:00
|
|
|
LINUX_PTRACE_TRACEME, PT_TRACE_ME,
|
|
|
|
LINUX_PTRACE_PEEKTEXT, PT_READ_I,
|
|
|
|
LINUX_PTRACE_PEEKDATA, PT_READ_D,
|
|
|
|
LINUX_PTRACE_POKETEXT, PT_WRITE_I,
|
|
|
|
LINUX_PTRACE_POKEDATA, PT_WRITE_D,
|
|
|
|
LINUX_PTRACE_CONT, PT_CONTINUE,
|
|
|
|
LINUX_PTRACE_KILL, PT_KILL,
|
|
|
|
LINUX_PTRACE_ATTACH, PT_ATTACH,
|
|
|
|
LINUX_PTRACE_DETACH, PT_DETACH,
|
2000-11-01 23:56:30 +03:00
|
|
|
#ifdef PT_STEP
|
|
|
|
LINUX_PTRACE_SINGLESTEP, PT_STEP,
|
|
|
|
#endif
|
1999-12-16 18:09:49 +03:00
|
|
|
-1
|
|
|
|
};
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2002-09-23 09:03:03 +04:00
|
|
|
static const struct mnttypes {
|
2002-02-20 20:03:03 +03:00
|
|
|
char *bsd;
|
|
|
|
int linux;
|
|
|
|
} fstypes[] = {
|
|
|
|
{ MOUNT_FFS, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_NFS, LINUX_NFS_SUPER_MAGIC },
|
|
|
|
{ MOUNT_MFS, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_MSDOS, LINUX_MSDOS_SUPER_MAGIC },
|
|
|
|
{ MOUNT_LFS, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_FDESC, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_PORTAL, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_NULL, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_OVERLAY, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_UMAP, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_KERNFS, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_PROCFS, LINUX_PROC_SUPER_MAGIC },
|
|
|
|
{ MOUNT_AFS, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_CD9660, LINUX_ISOFS_SUPER_MAGIC },
|
|
|
|
{ MOUNT_UNION, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_ADOSFS, LINUX_ADFS_SUPER_MAGIC },
|
|
|
|
{ MOUNT_EXT2FS, LINUX_EXT2_SUPER_MAGIC },
|
|
|
|
{ MOUNT_CFS, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_CODA, LINUX_CODA_SUPER_MAGIC },
|
|
|
|
{ MOUNT_FILECORE, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_NTFS, LINUX_DEFAULT_SUPER_MAGIC },
|
|
|
|
{ MOUNT_SMBFS, LINUX_SMB_SUPER_MAGIC }
|
|
|
|
};
|
|
|
|
#define FSTYPESSIZE (sizeof(fstypes) / sizeof(fstypes[0]))
|
|
|
|
|
2002-03-22 20:14:18 +03:00
|
|
|
#ifdef DEBUG_LINUX
|
|
|
|
#define DPRINTF(a) uprintf a
|
|
|
|
#else
|
|
|
|
#define DPRINTF(a)
|
|
|
|
#endif
|
|
|
|
|
1998-10-01 08:05:54 +04:00
|
|
|
/* Local linux_misc.c functions: */
|
2004-04-21 05:05:31 +04:00
|
|
|
static void bsd_to_linux_statfs __P((const struct statvfs *,
|
|
|
|
struct linux_statfs *));
|
2002-02-15 19:47:58 +03:00
|
|
|
static int linux_to_bsd_limit __P((int));
|
2003-06-24 01:25:56 +04:00
|
|
|
static void linux_to_bsd_mmap_args __P((struct sys_mmap_args *,
|
|
|
|
const struct linux_sys_mmap_args *));
|
1996-04-05 03:56:01 +04:00
|
|
|
|
1995-03-01 02:24:35 +03:00
|
|
|
/*
|
|
|
|
* The information on a terminated (or stopped) process needs
|
|
|
|
* to be converted in order for Linux binaries to get a valid signal
|
|
|
|
* number out of it.
|
|
|
|
*/
|
1998-10-01 08:05:54 +04:00
|
|
|
void
|
1998-12-16 13:21:50 +03:00
|
|
|
bsd_to_linux_wstat(st)
|
|
|
|
int *st;
|
1995-03-01 02:24:35 +03:00
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
|
1998-12-16 13:21:50 +03:00
|
|
|
int sig;
|
|
|
|
|
|
|
|
if (WIFSIGNALED(*st)) {
|
|
|
|
sig = WTERMSIG(*st);
|
|
|
|
if (sig >= 0 && sig < NSIG)
|
2002-04-01 02:22:43 +04:00
|
|
|
*st= (*st& ~0177) | native_to_linux_signo[sig];
|
1998-12-16 13:21:50 +03:00
|
|
|
} else if (WIFSTOPPED(*st)) {
|
|
|
|
sig = WSTOPSIG(*st);
|
|
|
|
if (sig >= 0 && sig < NSIG)
|
2002-04-01 02:22:43 +04:00
|
|
|
*st = (*st & ~0xff00) |
|
|
|
|
(native_to_linux_signo[sig] << 8);
|
1998-12-16 13:21:50 +03:00
|
|
|
}
|
1995-03-01 02:24:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is very much the same as waitpid()
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_wait4(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_wait4_args /* {
|
1995-03-01 02:24:35 +03:00
|
|
|
syscallarg(int) pid;
|
|
|
|
syscallarg(int *) status;
|
|
|
|
syscallarg(int) options;
|
|
|
|
syscallarg(struct rusage *) rusage;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
1995-10-07 09:25:19 +03:00
|
|
|
struct sys_wait4_args w4a;
|
1999-05-13 05:00:50 +04:00
|
|
|
int error, *status, tstat, options, linux_options;
|
1995-03-01 02:24:35 +03:00
|
|
|
caddr_t sg;
|
|
|
|
|
1995-08-22 03:15:51 +04:00
|
|
|
if (SCARG(uap, status) != NULL) {
|
2002-03-16 23:43:48 +03:00
|
|
|
sg = stackgap_init(p, 0);
|
|
|
|
status = (int *) stackgap_alloc(p, &sg, sizeof *status);
|
1995-08-22 03:15:51 +04:00
|
|
|
} else
|
|
|
|
status = NULL;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
1999-05-13 05:00:50 +04:00
|
|
|
linux_options = SCARG(uap, options);
|
|
|
|
options = 0;
|
|
|
|
if (linux_options &
|
2001-07-18 23:11:14 +04:00
|
|
|
~(LINUX_WAIT4_WNOHANG|LINUX_WAIT4_WUNTRACED|LINUX_WAIT4_WALL|
|
|
|
|
LINUX_WAIT4_WCLONE))
|
1999-05-13 05:00:50 +04:00
|
|
|
return (EINVAL);
|
|
|
|
|
|
|
|
if (linux_options & LINUX_WAIT4_WNOHANG)
|
|
|
|
options |= WNOHANG;
|
|
|
|
if (linux_options & LINUX_WAIT4_WUNTRACED)
|
|
|
|
options |= WUNTRACED;
|
2001-07-18 23:11:14 +04:00
|
|
|
if (linux_options & LINUX_WAIT4_WALL)
|
|
|
|
options |= WALLSIG;
|
1999-05-13 05:00:50 +04:00
|
|
|
if (linux_options & LINUX_WAIT4_WCLONE)
|
|
|
|
options |= WALTSIG;
|
|
|
|
|
1995-03-01 02:24:35 +03:00
|
|
|
SCARG(&w4a, pid) = SCARG(uap, pid);
|
|
|
|
SCARG(&w4a, status) = status;
|
1999-05-13 05:00:50 +04:00
|
|
|
SCARG(&w4a, options) = options;
|
1995-03-01 02:24:35 +03:00
|
|
|
SCARG(&w4a, rusage) = SCARG(uap, rusage);
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
if ((error = sys_wait4(l, &w4a, retval)))
|
1995-03-01 02:24:35 +03:00
|
|
|
return error;
|
|
|
|
|
2000-12-23 01:58:52 +03:00
|
|
|
sigdelset(&p->p_sigctx.ps_siglist, SIGCHLD);
|
1995-09-08 01:48:59 +04:00
|
|
|
|
1995-08-22 03:15:51 +04:00
|
|
|
if (status != NULL) {
|
|
|
|
if ((error = copyin(status, &tstat, sizeof tstat)))
|
|
|
|
return error;
|
|
|
|
|
|
|
|
bsd_to_linux_wstat(&tstat);
|
|
|
|
return copyout(&tstat, SCARG(uap, status), sizeof tstat);
|
|
|
|
}
|
1995-03-01 02:24:35 +03:00
|
|
|
|
1995-08-22 03:15:51 +04:00
|
|
|
return 0;
|
1995-03-01 02:24:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Linux brk(2). The check if the new address is >= the old one is
|
|
|
|
* done in the kernel in Linux. NetBSD does it in the library.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_brk(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
1995-03-01 02:24:35 +03:00
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_brk_args /* {
|
1995-09-20 02:37:27 +04:00
|
|
|
syscallarg(char *) nsize;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
1995-03-01 02:24:35 +03:00
|
|
|
char *nbrk = SCARG(uap, nsize);
|
1995-10-07 09:25:19 +03:00
|
|
|
struct sys_obreak_args oba;
|
1995-03-01 02:24:35 +03:00
|
|
|
struct vmspace *vm = p->p_vmspace;
|
2001-03-16 02:23:26 +03:00
|
|
|
struct linux_emuldata *ed = (struct linux_emuldata*)p->p_emuldata;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
|
|
|
SCARG(&oba, nsize) = nbrk;
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
if ((caddr_t) nbrk > vm->vm_daddr && sys_obreak(l, &oba, retval) == 0)
|
2001-03-16 02:23:26 +03:00
|
|
|
ed->p_break = (char*)nbrk;
|
|
|
|
else
|
|
|
|
nbrk = ed->p_break;
|
|
|
|
|
|
|
|
retval[0] = (register_t)nbrk;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
1995-03-06 02:23:37 +03:00
|
|
|
* Convert BSD statfs structure to Linux statfs structure.
|
|
|
|
* The Linux structure has less fields, and it also wants
|
|
|
|
* the length of a name in a dir entry in a field, which
|
|
|
|
* we fake (probably the wrong way).
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
bsd_to_linux_statfs(bsp, lsp)
|
2004-04-21 05:05:31 +04:00
|
|
|
const struct statvfs *bsp;
|
1995-03-06 02:23:37 +03:00
|
|
|
struct linux_statfs *lsp;
|
|
|
|
{
|
2002-02-20 20:03:03 +03:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < FSTYPESSIZE; i++)
|
|
|
|
if (strcmp(bsp->f_fstypename, fstypes[i].bsd) == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (i == FSTYPESSIZE) {
|
2002-03-22 20:14:18 +03:00
|
|
|
DPRINTF(("unhandled fstype in linux emulation: %s\n",
|
|
|
|
bsp->f_fstypename));
|
2002-02-20 20:03:03 +03:00
|
|
|
lsp->l_ftype = LINUX_DEFAULT_SUPER_MAGIC;
|
|
|
|
} else {
|
|
|
|
lsp->l_ftype = fstypes[i].linux;
|
|
|
|
}
|
1995-10-07 09:25:19 +03:00
|
|
|
|
1995-03-06 02:23:37 +03:00
|
|
|
lsp->l_fbsize = bsp->f_bsize;
|
|
|
|
lsp->l_fblocks = bsp->f_blocks;
|
|
|
|
lsp->l_fbfree = bsp->f_bfree;
|
|
|
|
lsp->l_fbavail = bsp->f_bavail;
|
|
|
|
lsp->l_ffiles = bsp->f_files;
|
|
|
|
lsp->l_fffree = bsp->f_ffree;
|
2002-02-20 20:03:03 +03:00
|
|
|
/* Linux sets the fsid to 0..., we don't */
|
2004-04-21 05:05:31 +04:00
|
|
|
lsp->l_ffsid.val[0] = bsp->f_fsidx.__fsid_val[0];
|
|
|
|
lsp->l_ffsid.val[1] = bsp->f_fsidx.__fsid_val[1];
|
1995-03-06 02:23:37 +03:00
|
|
|
lsp->l_fnamelen = MAXNAMLEN; /* XXX */
|
2002-02-20 20:03:03 +03:00
|
|
|
(void)memset(lsp->l_fspare, 0, sizeof(lsp->l_fspare));
|
1995-03-06 02:23:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Implement the fs stat functions. Straightforward.
|
1995-03-01 02:24:35 +03:00
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_statfs(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_statfs_args /* {
|
1999-02-09 23:37:19 +03:00
|
|
|
syscallarg(const char *) path;
|
1995-03-01 02:24:35 +03:00
|
|
|
syscallarg(struct linux_statfs *) sp;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2004-04-21 05:05:31 +04:00
|
|
|
struct statvfs btmp, *bsp;
|
1995-03-06 02:23:37 +03:00
|
|
|
struct linux_statfs ltmp;
|
2004-04-21 06:27:34 +04:00
|
|
|
struct sys_statvfs1_args bsa;
|
1995-03-06 02:23:37 +03:00
|
|
|
caddr_t sg;
|
|
|
|
int error;
|
|
|
|
|
2002-03-16 23:43:48 +03:00
|
|
|
sg = stackgap_init(p, 0);
|
2004-04-21 05:05:31 +04:00
|
|
|
bsp = (struct statvfs *) stackgap_alloc(p, &sg, sizeof (struct statvfs));
|
1995-03-06 02:23:37 +03:00
|
|
|
|
2003-06-30 02:28:00 +04:00
|
|
|
CHECK_ALT_EXIST(p, &sg, SCARG(uap, path));
|
1995-03-06 02:23:37 +03:00
|
|
|
|
|
|
|
SCARG(&bsa, path) = SCARG(uap, path);
|
|
|
|
SCARG(&bsa, buf) = bsp;
|
2004-04-21 06:37:20 +04:00
|
|
|
SCARG(&bsa, flags) = ST_WAIT;
|
1995-03-06 02:23:37 +03:00
|
|
|
|
2004-04-21 06:37:20 +04:00
|
|
|
if ((error = sys_statvfs1(l, &bsa, retval)))
|
1995-03-06 02:23:37 +03:00
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin((caddr_t) bsp, (caddr_t) &btmp, sizeof btmp)))
|
|
|
|
return error;
|
|
|
|
|
|
|
|
bsd_to_linux_statfs(&btmp, <mp);
|
|
|
|
|
|
|
|
return copyout((caddr_t) <mp, (caddr_t) SCARG(uap, sp), sizeof ltmp);
|
1995-03-01 02:24:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_fstatfs(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_fstatfs_args /* {
|
1995-03-06 02:23:37 +03:00
|
|
|
syscallarg(int) fd;
|
1995-03-01 02:24:35 +03:00
|
|
|
syscallarg(struct linux_statfs *) sp;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2004-04-21 05:05:31 +04:00
|
|
|
struct statvfs btmp, *bsp;
|
1995-03-06 02:23:37 +03:00
|
|
|
struct linux_statfs ltmp;
|
2004-04-21 06:37:20 +04:00
|
|
|
struct sys_fstatvfs1_args bsa;
|
1995-03-06 02:23:37 +03:00
|
|
|
caddr_t sg;
|
|
|
|
int error;
|
|
|
|
|
2002-03-16 23:43:48 +03:00
|
|
|
sg = stackgap_init(p, 0);
|
2004-04-21 05:05:31 +04:00
|
|
|
bsp = (struct statvfs *) stackgap_alloc(p, &sg, sizeof (struct statvfs));
|
1995-03-06 02:23:37 +03:00
|
|
|
|
|
|
|
SCARG(&bsa, fd) = SCARG(uap, fd);
|
|
|
|
SCARG(&bsa, buf) = bsp;
|
2004-04-21 06:37:20 +04:00
|
|
|
SCARG(&bsa, flags) = ST_WAIT;
|
1995-03-06 02:23:37 +03:00
|
|
|
|
2004-04-21 06:37:20 +04:00
|
|
|
if ((error = sys_fstatvfs1(l, &bsa, retval)))
|
1995-03-06 02:23:37 +03:00
|
|
|
return error;
|
|
|
|
|
|
|
|
if ((error = copyin((caddr_t) bsp, (caddr_t) &btmp, sizeof btmp)))
|
|
|
|
return error;
|
|
|
|
|
|
|
|
bsd_to_linux_statfs(&btmp, <mp);
|
|
|
|
|
|
|
|
return copyout((caddr_t) <mp, (caddr_t) SCARG(uap, sp), sizeof ltmp);
|
1995-03-01 02:24:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* uname(). Just copy the info from the various strings stored in the
|
|
|
|
* kernel, and put it in the Linux utsname structure. That structure
|
|
|
|
* is almost the same as the NetBSD one, only it has fields 65 characters
|
|
|
|
* long, and an extra domainname field.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_uname(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
1995-03-01 02:24:35 +03:00
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_uname_args /* {
|
1995-09-20 02:37:27 +04:00
|
|
|
syscallarg(struct linux_utsname *) up;
|
|
|
|
} */ *uap = v;
|
1995-08-21 07:42:09 +04:00
|
|
|
struct linux_utsname luts;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2000-12-30 00:07:16 +03:00
|
|
|
strncpy(luts.l_sysname, linux_sysname, sizeof(luts.l_sysname));
|
1995-08-21 07:42:09 +04:00
|
|
|
strncpy(luts.l_nodename, hostname, sizeof(luts.l_nodename));
|
2000-12-30 00:07:16 +03:00
|
|
|
strncpy(luts.l_release, linux_release, sizeof(luts.l_release));
|
|
|
|
strncpy(luts.l_version, linux_version, sizeof(luts.l_version));
|
1995-08-21 07:42:09 +04:00
|
|
|
strncpy(luts.l_machine, machine, sizeof(luts.l_machine));
|
|
|
|
strncpy(luts.l_domainname, domainname, sizeof(luts.l_domainname));
|
1995-03-01 02:24:35 +03:00
|
|
|
|
1995-08-21 07:42:09 +04:00
|
|
|
return copyout(&luts, SCARG(uap, up), sizeof(luts));
|
|
|
|
}
|
|
|
|
|
1998-10-01 08:05:54 +04:00
|
|
|
/* Used directly on: alpha, mips, ppc, sparc, sparc64 */
|
|
|
|
/* Used indirectly on: arm, i386, m68k */
|
1995-03-01 02:24:35 +03:00
|
|
|
|
|
|
|
/*
|
1998-10-01 08:05:54 +04:00
|
|
|
* New type Linux mmap call.
|
|
|
|
* Only called directly on machines with >= 6 free regs.
|
1995-03-01 02:24:35 +03:00
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_mmap(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
1995-03-01 02:24:35 +03:00
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_mmap_args /* {
|
1998-10-01 08:05:54 +04:00
|
|
|
syscallarg(unsigned long) addr;
|
|
|
|
syscallarg(size_t) len;
|
|
|
|
syscallarg(int) prot;
|
|
|
|
syscallarg(int) flags;
|
|
|
|
syscallarg(int) fd;
|
2001-09-08 11:09:43 +04:00
|
|
|
syscallarg(linux_off_t) offset;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
1995-10-07 09:25:19 +03:00
|
|
|
struct sys_mmap_args cma;
|
2003-06-24 01:25:56 +04:00
|
|
|
|
2002-12-16 22:37:03 +03:00
|
|
|
if (SCARG(uap, offset) & PAGE_MASK)
|
|
|
|
return EINVAL;
|
|
|
|
|
2003-06-24 01:25:56 +04:00
|
|
|
linux_to_bsd_mmap_args(&cma, uap);
|
|
|
|
SCARG(&cma, pos) = (off_t)SCARG(uap, offset);
|
|
|
|
|
|
|
|
return sys_mmap(l, &cma, retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Guts of most architectures' mmap64() implementations. This shares
|
|
|
|
* its list of arguments with linux_sys_mmap().
|
|
|
|
*
|
|
|
|
* The difference in linux_sys_mmap2() is that "offset" is actually
|
|
|
|
* (offset / pagesize), not an absolute byte count. This translation
|
|
|
|
* to pagesize offsets is done inside glibc between the mmap64() call
|
|
|
|
* point, and the actual syscall.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
linux_sys_mmap2(l, v, retval)
|
|
|
|
struct lwp *l;
|
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_mmap2_args /* {
|
|
|
|
syscallarg(unsigned long) addr;
|
|
|
|
syscallarg(size_t) len;
|
|
|
|
syscallarg(int) prot;
|
|
|
|
syscallarg(int) flags;
|
|
|
|
syscallarg(int) fd;
|
|
|
|
syscallarg(linux_off_t) offset;
|
|
|
|
} */ *uap = v;
|
|
|
|
struct sys_mmap_args cma;
|
|
|
|
|
|
|
|
linux_to_bsd_mmap_args(&cma, uap);
|
|
|
|
SCARG(&cma, pos) = ((off_t)SCARG(uap, offset)) << PAGE_SHIFT;
|
|
|
|
|
|
|
|
return sys_mmap(l, &cma, retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
linux_to_bsd_mmap_args(cma, uap)
|
|
|
|
struct sys_mmap_args *cma;
|
|
|
|
const struct linux_sys_mmap_args *uap;
|
|
|
|
{
|
2003-06-24 01:32:33 +04:00
|
|
|
int flags = MAP_TRYFIXED, fl = SCARG(uap, flags);
|
2003-06-24 01:25:56 +04:00
|
|
|
|
2002-03-22 18:21:28 +03:00
|
|
|
flags |= cvtto_bsd_mask(fl, LINUX_MAP_SHARED, MAP_SHARED);
|
|
|
|
flags |= cvtto_bsd_mask(fl, LINUX_MAP_PRIVATE, MAP_PRIVATE);
|
|
|
|
flags |= cvtto_bsd_mask(fl, LINUX_MAP_FIXED, MAP_FIXED);
|
|
|
|
flags |= cvtto_bsd_mask(fl, LINUX_MAP_ANON, MAP_ANON);
|
1998-10-01 08:05:54 +04:00
|
|
|
/* XXX XAX ERH: Any other flags here? There are more defined... */
|
|
|
|
|
2003-06-24 01:25:56 +04:00
|
|
|
SCARG(cma, addr) = (void *)SCARG(uap, addr);
|
|
|
|
SCARG(cma, len) = SCARG(uap, len);
|
|
|
|
SCARG(cma, prot) = SCARG(uap, prot);
|
|
|
|
if (SCARG(cma, prot) & VM_PROT_WRITE) /* XXX */
|
|
|
|
SCARG(cma, prot) |= VM_PROT_READ;
|
|
|
|
SCARG(cma, flags) = flags;
|
|
|
|
SCARG(cma, fd) = flags & MAP_ANON ? -1 : SCARG(uap, fd);
|
|
|
|
SCARG(cma, pad) = 0;
|
2002-02-15 19:47:58 +03:00
|
|
|
}
|
|
|
|
|
1997-10-10 10:25:34 +04:00
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_mremap(l, v, retval)
|
|
|
|
struct lwp *l;
|
1997-10-10 10:25:34 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_mremap_args /* {
|
|
|
|
syscallarg(void *) old_address;
|
|
|
|
syscallarg(size_t) old_size;
|
|
|
|
syscallarg(size_t) new_size;
|
|
|
|
syscallarg(u_long) flags;
|
|
|
|
} */ *uap = v;
|
1998-07-03 03:26:58 +04:00
|
|
|
struct sys_munmap_args mua;
|
|
|
|
size_t old_size, new_size;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
old_size = round_page(SCARG(uap, old_size));
|
|
|
|
new_size = round_page(SCARG(uap, new_size));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Growing mapped region.
|
|
|
|
*/
|
|
|
|
if (new_size > old_size) {
|
|
|
|
/*
|
|
|
|
* XXX Implement me. What we probably want to do is
|
|
|
|
* XXX dig out the guts of the old mapping, mmap that
|
|
|
|
* XXX object again with the new size, then munmap
|
|
|
|
* XXX the old mapping.
|
|
|
|
*/
|
|
|
|
*retval = 0;
|
|
|
|
return (ENOMEM);
|
|
|
|
}
|
1997-10-10 10:25:34 +04:00
|
|
|
|
1998-07-03 03:26:58 +04:00
|
|
|
/*
|
|
|
|
* Shrinking mapped region.
|
|
|
|
*/
|
|
|
|
if (new_size < old_size) {
|
|
|
|
SCARG(&mua, addr) = (caddr_t)SCARG(uap, old_address) +
|
1998-07-07 04:40:27 +04:00
|
|
|
new_size;
|
1998-07-03 03:26:58 +04:00
|
|
|
SCARG(&mua, len) = old_size - new_size;
|
2003-01-18 11:02:46 +03:00
|
|
|
error = sys_munmap(l, &mua, retval);
|
1998-07-03 03:26:58 +04:00
|
|
|
*retval = error ? 0 : (register_t)SCARG(uap, old_address);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* No change.
|
|
|
|
*/
|
|
|
|
*retval = (register_t)SCARG(uap, old_address);
|
|
|
|
return (0);
|
1997-10-10 10:25:34 +04:00
|
|
|
}
|
|
|
|
|
1995-12-18 17:35:08 +03:00
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_msync(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-12-18 17:35:08 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_msync_args /* {
|
|
|
|
syscallarg(caddr_t) addr;
|
|
|
|
syscallarg(int) len;
|
|
|
|
syscallarg(int) fl;
|
|
|
|
} */ *uap = v;
|
|
|
|
|
1997-10-21 04:57:38 +04:00
|
|
|
struct sys___msync13_args bma;
|
1995-12-18 17:35:08 +03:00
|
|
|
|
|
|
|
/* flags are ignored */
|
|
|
|
SCARG(&bma, addr) = SCARG(uap, addr);
|
|
|
|
SCARG(&bma, len) = SCARG(uap, len);
|
1997-10-21 04:57:38 +04:00
|
|
|
SCARG(&bma, flags) = SCARG(uap, fl);
|
1995-12-18 17:35:08 +03:00
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return sys___msync13(l, &bma, retval);
|
1995-12-18 17:35:08 +03:00
|
|
|
}
|
|
|
|
|
2002-03-22 18:21:28 +03:00
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_mprotect(l, v, retval)
|
|
|
|
struct lwp *l;
|
2002-03-22 18:21:28 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_mprotect_args /* {
|
|
|
|
syscallarg(const void *) start;
|
|
|
|
syscallarg(unsigned long) len;
|
|
|
|
syscallarg(int) prot;
|
|
|
|
} */ *uap = v;
|
|
|
|
unsigned long end, start = (unsigned long)SCARG(uap, start), len;
|
|
|
|
int prot = SCARG(uap, prot);
|
|
|
|
struct vm_map_entry *entry;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct vm_map *map = &l->l_proc->p_vmspace->vm_map;
|
2002-03-22 18:21:28 +03:00
|
|
|
|
|
|
|
if (start & PAGE_MASK)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
len = round_page(SCARG(uap, len));
|
|
|
|
end = start + len;
|
|
|
|
|
|
|
|
if (end < start)
|
|
|
|
return EINVAL;
|
|
|
|
else if (end == start)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (SCARG(uap, prot) & ~(PROT_READ | PROT_WRITE | PROT_EXEC))
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
vm_map_lock(map);
|
|
|
|
#ifdef notdef
|
|
|
|
VM_MAP_RANGE_CHECK(map, start, end);
|
|
|
|
#endif
|
|
|
|
if (!uvm_map_lookup_entry(map, start, &entry) || entry->start > start) {
|
|
|
|
vm_map_unlock(map);
|
|
|
|
return EFAULT;
|
|
|
|
}
|
|
|
|
vm_map_unlock(map);
|
|
|
|
return uvm_map_protect(map, start, end, prot, FALSE);
|
|
|
|
}
|
|
|
|
|
1995-03-01 02:24:35 +03:00
|
|
|
/*
|
|
|
|
* This code is partly stolen from src/lib/libc/compat-43/times.c
|
|
|
|
*/
|
|
|
|
|
2002-11-13 18:20:04 +03:00
|
|
|
#define CONVTCK(r) (r.tv_sec * hz + r.tv_usec / (1000000 / hz))
|
1995-03-01 02:24:35 +03:00
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_times(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
1995-03-01 02:24:35 +03:00
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_times_args /* {
|
1995-09-20 02:37:27 +04:00
|
|
|
syscallarg(struct times *) tms;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
1995-03-01 02:24:35 +03:00
|
|
|
struct timeval t;
|
1995-03-22 08:24:47 +03:00
|
|
|
int error, s;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2002-11-13 11:27:10 +03:00
|
|
|
if (SCARG(uap, tms)) {
|
|
|
|
struct linux_tms ltms;
|
|
|
|
struct rusage ru;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2002-11-13 11:27:10 +03:00
|
|
|
calcru(p, &ru.ru_utime, &ru.ru_stime, NULL);
|
|
|
|
ltms.ltms_utime = CONVTCK(ru.ru_utime);
|
|
|
|
ltms.ltms_stime = CONVTCK(ru.ru_stime);
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2002-11-13 11:27:10 +03:00
|
|
|
ltms.ltms_cutime = CONVTCK(p->p_stats->p_cru.ru_utime);
|
|
|
|
ltms.ltms_cstime = CONVTCK(p->p_stats->p_cru.ru_stime);
|
|
|
|
|
|
|
|
if ((error = copyout(<ms, SCARG(uap, tms), sizeof ltms)))
|
|
|
|
return error;
|
|
|
|
}
|
1995-03-01 02:24:35 +03:00
|
|
|
|
1995-03-22 08:24:47 +03:00
|
|
|
s = splclock();
|
|
|
|
timersub(&time, &boottime, &t);
|
|
|
|
splx(s);
|
1995-03-01 02:24:35 +03:00
|
|
|
|
|
|
|
retval[0] = ((linux_clock_t)(CONVTCK(t)));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-11-13 18:20:04 +03:00
|
|
|
#undef CONVTCK
|
|
|
|
|
1995-03-01 02:24:35 +03:00
|
|
|
/*
|
|
|
|
* Linux 'readdir' call. This code is mostly taken from the
|
|
|
|
* SunOS getdents call (see compat/sunos/sunos_misc.c), though
|
|
|
|
* an attempt has been made to keep it a little cleaner (failing
|
|
|
|
* miserably, because of the cruft needed if count 1 is passed).
|
|
|
|
*
|
1995-08-24 00:17:28 +04:00
|
|
|
* The d_off field should contain the offset of the next valid entry,
|
|
|
|
* but in Linux it has the offset of the entry itself. We emulate
|
|
|
|
* that bug here.
|
|
|
|
*
|
1995-03-01 02:24:35 +03:00
|
|
|
* Read in BSD-style entries, convert them, and copy them out.
|
|
|
|
*
|
|
|
|
* Note that this doesn't handle union-mounted filesystems.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_getdents(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
1998-10-01 08:05:54 +04:00
|
|
|
struct linux_sys_getdents_args /* {
|
1995-03-01 02:24:35 +03:00
|
|
|
syscallarg(int) fd;
|
1998-10-01 08:05:54 +04:00
|
|
|
syscallarg(struct linux_dirent *) dent;
|
1995-03-01 02:24:35 +03:00
|
|
|
syscallarg(unsigned int) count;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2000-03-30 15:24:16 +04:00
|
|
|
struct dirent *bdp;
|
1995-03-01 02:24:35 +03:00
|
|
|
struct vnode *vp;
|
1996-04-05 03:56:01 +04:00
|
|
|
caddr_t inp, buf; /* BSD-format */
|
|
|
|
int len, reclen; /* BSD-format */
|
1998-10-04 14:17:54 +04:00
|
|
|
caddr_t outp; /* Linux-format */
|
1996-04-05 03:56:01 +04:00
|
|
|
int resid, linux_reclen = 0; /* Linux-format */
|
1995-03-01 02:24:35 +03:00
|
|
|
struct file *fp;
|
|
|
|
struct uio auio;
|
|
|
|
struct iovec aiov;
|
|
|
|
struct linux_dirent idb;
|
|
|
|
off_t off; /* true file offset */
|
1995-08-24 00:17:28 +04:00
|
|
|
int buflen, error, eofflag, nbytes, oldcall;
|
1995-03-01 02:24:35 +03:00
|
|
|
struct vattr va;
|
1998-03-03 16:44:48 +03:00
|
|
|
off_t *cookiebuf = NULL, *cookie;
|
1995-10-09 14:23:57 +03:00
|
|
|
int ncookies;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
1999-05-06 00:01:01 +04:00
|
|
|
/* getvnode() will use the descriptor for us */
|
1995-03-01 02:24:35 +03:00
|
|
|
if ((error = getvnode(p->p_fd, SCARG(uap, fd), &fp)) != 0)
|
|
|
|
return (error);
|
|
|
|
|
1999-05-06 00:01:01 +04:00
|
|
|
if ((fp->f_flag & FREAD) == 0) {
|
|
|
|
error = EBADF;
|
|
|
|
goto out1;
|
|
|
|
}
|
1995-03-01 02:24:35 +03:00
|
|
|
|
1995-06-11 02:19:06 +04:00
|
|
|
vp = (struct vnode *)fp->f_data;
|
1999-05-06 00:01:01 +04:00
|
|
|
if (vp->v_type != VDIR) {
|
|
|
|
error = EINVAL;
|
|
|
|
goto out1;
|
|
|
|
}
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2003-06-30 02:28:00 +04:00
|
|
|
if ((error = VOP_GETATTR(vp, &va, p->p_ucred, p)))
|
1999-05-06 00:01:01 +04:00
|
|
|
goto out1;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
|
|
|
nbytes = SCARG(uap, count);
|
1995-08-24 00:17:28 +04:00
|
|
|
if (nbytes == 1) { /* emulating old, broken behaviour */
|
2002-04-10 22:19:34 +04:00
|
|
|
nbytes = sizeof (idb);
|
1995-06-11 02:19:06 +04:00
|
|
|
buflen = max(va.va_blocksize, nbytes);
|
1995-08-24 00:17:28 +04:00
|
|
|
oldcall = 1;
|
1995-06-11 02:19:06 +04:00
|
|
|
} else {
|
|
|
|
buflen = min(MAXBSIZE, nbytes);
|
1997-10-10 05:42:21 +04:00
|
|
|
if (buflen < va.va_blocksize)
|
|
|
|
buflen = va.va_blocksize;
|
1995-08-24 00:17:28 +04:00
|
|
|
oldcall = 0;
|
1995-06-11 02:19:06 +04:00
|
|
|
}
|
1995-03-01 02:24:35 +03:00
|
|
|
buf = malloc(buflen, M_TEMP, M_WAITOK);
|
1997-10-10 05:42:21 +04:00
|
|
|
|
1998-03-01 05:20:01 +03:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY);
|
1995-03-01 02:24:35 +03: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;
|
2003-06-30 02:28:00 +04:00
|
|
|
auio.uio_procp = p;
|
1995-03-01 02:24:35 +03:00
|
|
|
auio.uio_resid = buflen;
|
|
|
|
auio.uio_offset = off;
|
|
|
|
/*
|
|
|
|
* First we read into the malloc'ed buffer, then
|
|
|
|
* we massage it into user space, one record at a time.
|
|
|
|
*/
|
1998-03-01 05:20:01 +03:00
|
|
|
error = VOP_READDIR(vp, &auio, fp->f_cred, &eofflag, &cookiebuf,
|
|
|
|
&ncookies);
|
1995-03-01 02:24:35 +03:00
|
|
|
if (error)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
inp = buf;
|
1998-10-04 14:17:54 +04:00
|
|
|
outp = (caddr_t)SCARG(uap, dent);
|
1995-03-01 02:24:35 +03:00
|
|
|
resid = nbytes;
|
1997-10-11 02:16:04 +04:00
|
|
|
if ((len = buflen - auio.uio_resid) == 0)
|
1995-03-01 02:24:35 +03:00
|
|
|
goto eof;
|
|
|
|
|
1995-10-09 14:23:57 +03:00
|
|
|
for (cookie = cookiebuf; len > 0; len -= reclen) {
|
1995-06-11 02:19:06 +04:00
|
|
|
bdp = (struct dirent *)inp;
|
|
|
|
reclen = bdp->d_reclen;
|
1995-03-01 02:24:35 +03:00
|
|
|
if (reclen & 3)
|
|
|
|
panic("linux_readdir");
|
|
|
|
if (bdp->d_fileno == 0) {
|
|
|
|
inp += reclen; /* it is a hole; squish it out */
|
1995-10-09 14:23:57 +03:00
|
|
|
off = *cookie++;
|
1995-03-01 02:24:35 +03:00
|
|
|
continue;
|
|
|
|
}
|
1995-10-07 09:25:19 +03:00
|
|
|
linux_reclen = LINUX_RECLEN(&idb, bdp->d_namlen);
|
|
|
|
if (reclen > len || resid < linux_reclen) {
|
1995-03-01 02:24:35 +03:00
|
|
|
/* entry too big for buffer, so just stop */
|
|
|
|
outp++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Massage in place to make a Linux-shaped dirent (otherwise
|
|
|
|
* we have to worry about touching user memory outside of
|
|
|
|
* the copyout() call).
|
|
|
|
*/
|
2002-04-10 22:19:34 +04:00
|
|
|
idb.d_ino = bdp->d_fileno;
|
1995-08-24 00:17:28 +04:00
|
|
|
/*
|
1995-10-07 09:25:19 +03:00
|
|
|
* The old readdir() call misuses the offset and reclen fields.
|
1995-08-24 00:17:28 +04:00
|
|
|
*/
|
1995-10-09 14:23:57 +03:00
|
|
|
if (oldcall) {
|
|
|
|
idb.d_off = (linux_off_t)linux_reclen;
|
|
|
|
idb.d_reclen = (u_short)bdp->d_namlen;
|
|
|
|
} else {
|
2002-05-10 18:49:38 +04:00
|
|
|
if (sizeof (idb.d_off) <= 4 && (off >> 32) != 0) {
|
1997-10-10 05:42:21 +04:00
|
|
|
compat_offseterr(vp, "linux_getdents");
|
|
|
|
error = EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
1995-10-09 14:23:57 +03:00
|
|
|
idb.d_off = (linux_off_t)off;
|
|
|
|
idb.d_reclen = (u_short)linux_reclen;
|
|
|
|
}
|
1995-06-11 02:19:06 +04:00
|
|
|
strcpy(idb.d_name, bdp->d_name);
|
1995-10-07 09:25:19 +03:00
|
|
|
if ((error = copyout((caddr_t)&idb, outp, linux_reclen)))
|
1995-03-01 02:24:35 +03:00
|
|
|
goto out;
|
|
|
|
/* advance past this real entry */
|
|
|
|
inp += reclen;
|
1995-10-09 14:23:57 +03:00
|
|
|
off = *cookie++; /* each entry points to itself */
|
1995-03-01 02:24:35 +03:00
|
|
|
/* advance output past Linux-shaped entry */
|
1995-10-07 09:25:19 +03:00
|
|
|
outp += linux_reclen;
|
|
|
|
resid -= linux_reclen;
|
1995-08-24 00:17:28 +04:00
|
|
|
if (oldcall)
|
1995-03-01 02:24:35 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if we squished out the whole block, try again */
|
1998-10-04 14:17:54 +04:00
|
|
|
if (outp == (caddr_t)SCARG(uap, dent))
|
1995-03-01 02:24:35 +03:00
|
|
|
goto again;
|
|
|
|
fp->f_offset = off; /* update the vnode offset */
|
|
|
|
|
1995-08-24 00:17:28 +04:00
|
|
|
if (oldcall)
|
1995-10-07 09:25:19 +03:00
|
|
|
nbytes = resid + linux_reclen;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
2002-04-10 22:19:34 +04:00
|
|
|
eof:
|
|
|
|
*retval = nbytes - resid;
|
|
|
|
out:
|
|
|
|
VOP_UNLOCK(vp, 0);
|
|
|
|
if (cookiebuf)
|
|
|
|
free(cookiebuf, M_TEMP);
|
|
|
|
free(buf, M_TEMP);
|
|
|
|
out1:
|
2003-06-30 02:28:00 +04:00
|
|
|
FILE_UNUSE(fp, p);
|
1995-03-01 02:24:35 +03:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
1995-08-24 00:17:28 +04:00
|
|
|
/*
|
|
|
|
* Even when just using registers to pass arguments to syscalls you can
|
|
|
|
* have 5 of them on the i386. So this newer version of select() does
|
|
|
|
* this.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_select(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_select_args /* {
|
1995-08-24 00:17:28 +04:00
|
|
|
syscallarg(int) nfds;
|
|
|
|
syscallarg(fd_set *) readfds;
|
|
|
|
syscallarg(fd_set *) writefds;
|
|
|
|
syscallarg(fd_set *) exceptfds;
|
|
|
|
syscallarg(struct timeval *) timeout;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return linux_select1(l, retval, SCARG(uap, nfds), SCARG(uap, readfds),
|
1995-08-24 00:17:28 +04:00
|
|
|
SCARG(uap, writefds), SCARG(uap, exceptfds), SCARG(uap, timeout));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Common code for the old and new versions of select(). A couple of
|
|
|
|
* things are important:
|
|
|
|
* 1) return the amount of time left in the 'timeout' parameter
|
|
|
|
* 2) select never returns ERESTART on Linux, always return EINTR
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_select1(l, retval, nfds, readfds, writefds, exceptfds, timeout)
|
|
|
|
struct lwp *l;
|
1995-08-24 00:17:28 +04:00
|
|
|
register_t *retval;
|
|
|
|
int nfds;
|
|
|
|
fd_set *readfds, *writefds, *exceptfds;
|
|
|
|
struct timeval *timeout;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct sys_select_args bsa;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
1995-08-24 00:17:28 +04:00
|
|
|
struct timeval tv0, tv1, utv, *tvp;
|
|
|
|
caddr_t sg;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
SCARG(&bsa, nd) = nfds;
|
|
|
|
SCARG(&bsa, in) = readfds;
|
|
|
|
SCARG(&bsa, ou) = writefds;
|
|
|
|
SCARG(&bsa, ex) = exceptfds;
|
|
|
|
SCARG(&bsa, tv) = timeout;
|
1995-03-01 02:24:35 +03:00
|
|
|
|
1995-06-12 01:51:38 +04:00
|
|
|
/*
|
|
|
|
* Store current time for computation of the amount of
|
|
|
|
* time left.
|
|
|
|
*/
|
1995-08-24 00:17:28 +04:00
|
|
|
if (timeout) {
|
|
|
|
if ((error = copyin(timeout, &utv, sizeof(utv))))
|
1995-08-16 08:29:49 +04:00
|
|
|
return error;
|
|
|
|
if (itimerfix(&utv)) {
|
|
|
|
/*
|
|
|
|
* The timeval was invalid. Convert it to something
|
|
|
|
* valid that will act as it does under Linux.
|
|
|
|
*/
|
2002-03-16 23:43:48 +03:00
|
|
|
sg = stackgap_init(p, 0);
|
|
|
|
tvp = stackgap_alloc(p, &sg, sizeof(utv));
|
1995-08-16 08:29:49 +04:00
|
|
|
utv.tv_sec += utv.tv_usec / 1000000;
|
|
|
|
utv.tv_usec %= 1000000;
|
|
|
|
if (utv.tv_usec < 0) {
|
|
|
|
utv.tv_sec -= 1;
|
|
|
|
utv.tv_usec += 1000000;
|
|
|
|
}
|
|
|
|
if (utv.tv_sec < 0)
|
|
|
|
timerclear(&utv);
|
|
|
|
if ((error = copyout(&utv, tvp, sizeof(utv))))
|
|
|
|
return error;
|
|
|
|
SCARG(&bsa, tv) = tvp;
|
|
|
|
}
|
1995-06-12 01:51:38 +04:00
|
|
|
microtime(&tv0);
|
1995-08-16 08:29:49 +04:00
|
|
|
}
|
1995-06-12 01:51:38 +04:00
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
error = sys_select(l, &bsa, retval);
|
1995-08-13 21:51:40 +04:00
|
|
|
if (error) {
|
|
|
|
/*
|
|
|
|
* See fs/select.c in the Linux kernel. Without this,
|
|
|
|
* Maelstrom doesn't work.
|
|
|
|
*/
|
|
|
|
if (error == ERESTART)
|
|
|
|
error = EINTR;
|
1995-06-12 01:51:38 +04:00
|
|
|
return error;
|
1995-08-13 21:51:40 +04:00
|
|
|
}
|
1995-06-12 01:51:38 +04:00
|
|
|
|
1995-08-24 00:17:28 +04:00
|
|
|
if (timeout) {
|
1995-08-16 08:50:17 +04:00
|
|
|
if (*retval) {
|
1995-06-12 01:51:38 +04:00
|
|
|
/*
|
1995-08-16 08:29:49 +04:00
|
|
|
* Compute how much time was left of the timeout,
|
1995-06-12 01:51:38 +04:00
|
|
|
* by subtracting the current time and the time
|
|
|
|
* before we started the call, and subtracting
|
|
|
|
* that result from the user-supplied value.
|
|
|
|
*/
|
|
|
|
microtime(&tv1);
|
|
|
|
timersub(&tv1, &tv0, &tv1);
|
|
|
|
timersub(&utv, &tv1, &utv);
|
1995-08-16 08:50:17 +04:00
|
|
|
if (utv.tv_sec < 0)
|
|
|
|
timerclear(&utv);
|
|
|
|
} else
|
|
|
|
timerclear(&utv);
|
1995-08-24 00:17:28 +04:00
|
|
|
if ((error = copyout(&utv, timeout, sizeof(utv))))
|
1995-06-12 01:51:38 +04:00
|
|
|
return error;
|
|
|
|
}
|
1995-08-16 08:29:49 +04:00
|
|
|
|
1995-06-12 01:51:38 +04:00
|
|
|
return 0;
|
1995-03-01 02:24:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get the process group of a certain process. Look it up
|
|
|
|
* and return the value.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_getpgid(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
1995-03-01 02:24:35 +03:00
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_getpgid_args /* {
|
1995-09-20 02:37:27 +04:00
|
|
|
syscallarg(int) pid;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
1995-03-01 02:24:35 +03:00
|
|
|
struct proc *targp;
|
|
|
|
|
1996-04-05 03:56:01 +04:00
|
|
|
if (SCARG(uap, pid) != 0 && SCARG(uap, pid) != p->p_pid) {
|
1995-03-01 02:24:35 +03:00
|
|
|
if ((targp = pfind(SCARG(uap, pid))) == 0)
|
|
|
|
return ESRCH;
|
1996-04-05 03:56:01 +04:00
|
|
|
}
|
1995-03-01 02:24:35 +03:00
|
|
|
else
|
|
|
|
targp = p;
|
|
|
|
|
|
|
|
retval[0] = targp->p_pgid;
|
|
|
|
return 0;
|
|
|
|
}
|
1995-06-11 18:56:47 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the 'personality' (emulation mode) for the current process. Only
|
|
|
|
* accept the Linux personality here (0). This call is needed because
|
|
|
|
* the Linux ELF crt0 issues it in an ugly kludge to make sure that
|
|
|
|
* ELF binaries run in Linux mode, not SVR4 mode.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_personality(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
1995-06-11 18:56:47 +04:00
|
|
|
register_t *retval;
|
|
|
|
{
|
1995-10-07 09:25:19 +03:00
|
|
|
struct linux_sys_personality_args /* {
|
1995-09-20 02:37:27 +04:00
|
|
|
syscallarg(int) per;
|
|
|
|
} */ *uap = v;
|
|
|
|
|
1995-06-11 18:56:47 +04:00
|
|
|
if (SCARG(uap, per) != 0)
|
|
|
|
return EINVAL;
|
|
|
|
retval[0] = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
1995-09-08 01:48:59 +04:00
|
|
|
|
2000-12-21 23:19:22 +03:00
|
|
|
#if defined(__i386__) || defined(__m68k__)
|
1995-09-08 01:48:59 +04:00
|
|
|
/*
|
|
|
|
* The calls are here because of type conversions.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setreuid16(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
2000-12-18 17:40:02 +03:00
|
|
|
struct linux_sys_setreuid16_args /* {
|
1995-09-08 01:48:59 +04:00
|
|
|
syscallarg(int) ruid;
|
|
|
|
syscallarg(int) euid;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
1996-06-23 15:13:32 +04:00
|
|
|
struct sys_setreuid_args bsa;
|
1995-09-08 01:48:59 +04:00
|
|
|
|
|
|
|
SCARG(&bsa, ruid) = ((linux_uid_t)SCARG(uap, ruid) == (linux_uid_t)-1) ?
|
|
|
|
(uid_t)-1 : SCARG(uap, ruid);
|
|
|
|
SCARG(&bsa, euid) = ((linux_uid_t)SCARG(uap, euid) == (linux_uid_t)-1) ?
|
|
|
|
(uid_t)-1 : SCARG(uap, euid);
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return sys_setreuid(l, &bsa, retval);
|
1995-09-08 01:48:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setregid16(l, v, retval)
|
|
|
|
struct lwp *l;
|
1995-09-20 02:37:27 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
2000-12-18 17:40:02 +03:00
|
|
|
struct linux_sys_setregid16_args /* {
|
1995-09-08 01:48:59 +04:00
|
|
|
syscallarg(int) rgid;
|
|
|
|
syscallarg(int) egid;
|
1995-09-20 02:37:27 +04:00
|
|
|
} */ *uap = v;
|
1996-06-23 15:13:32 +04:00
|
|
|
struct sys_setregid_args bsa;
|
1995-09-08 01:48:59 +04:00
|
|
|
|
|
|
|
SCARG(&bsa, rgid) = ((linux_gid_t)SCARG(uap, rgid) == (linux_gid_t)-1) ?
|
|
|
|
(uid_t)-1 : SCARG(uap, rgid);
|
|
|
|
SCARG(&bsa, egid) = ((linux_gid_t)SCARG(uap, egid) == (linux_gid_t)-1) ?
|
|
|
|
(uid_t)-1 : SCARG(uap, egid);
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return sys_setregid(l, &bsa, retval);
|
1995-09-08 01:48:59 +04:00
|
|
|
}
|
2000-12-28 01:01:42 +03:00
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setresuid16(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-28 01:01:42 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_setresuid16_args /* {
|
|
|
|
syscallarg(uid_t) ruid;
|
|
|
|
syscallarg(uid_t) euid;
|
|
|
|
syscallarg(uid_t) suid;
|
|
|
|
} */ *uap = v;
|
|
|
|
struct linux_sys_setresuid16_args lsa;
|
|
|
|
|
|
|
|
SCARG(&lsa, ruid) = ((linux_uid_t)SCARG(uap, ruid) == (linux_uid_t)-1) ?
|
|
|
|
(uid_t)-1 : SCARG(uap, ruid);
|
|
|
|
SCARG(&lsa, euid) = ((linux_uid_t)SCARG(uap, euid) == (linux_uid_t)-1) ?
|
|
|
|
(uid_t)-1 : SCARG(uap, euid);
|
|
|
|
SCARG(&lsa, suid) = ((linux_uid_t)SCARG(uap, suid) == (linux_uid_t)-1) ?
|
|
|
|
(uid_t)-1 : SCARG(uap, suid);
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return linux_sys_setresuid(l, &lsa, retval);
|
2000-12-28 01:01:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setresgid16(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-28 01:01:42 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_setresgid16_args /* {
|
|
|
|
syscallarg(gid_t) rgid;
|
|
|
|
syscallarg(gid_t) egid;
|
|
|
|
syscallarg(gid_t) sgid;
|
|
|
|
} */ *uap = v;
|
|
|
|
struct linux_sys_setresgid16_args lsa;
|
|
|
|
|
|
|
|
SCARG(&lsa, rgid) = ((linux_gid_t)SCARG(uap, rgid) == (linux_gid_t)-1) ?
|
|
|
|
(gid_t)-1 : SCARG(uap, rgid);
|
|
|
|
SCARG(&lsa, egid) = ((linux_gid_t)SCARG(uap, egid) == (linux_gid_t)-1) ?
|
|
|
|
(gid_t)-1 : SCARG(uap, egid);
|
|
|
|
SCARG(&lsa, sgid) = ((linux_gid_t)SCARG(uap, sgid) == (linux_gid_t)-1) ?
|
|
|
|
(gid_t)-1 : SCARG(uap, sgid);
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return linux_sys_setresgid(l, &lsa, retval);
|
2000-12-28 01:01:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_getgroups16(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-28 01:01:42 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_getgroups16_args /* {
|
|
|
|
syscallarg(int) gidsetsize;
|
|
|
|
syscallarg(linux_gid_t *) gidset;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2000-12-28 01:01:42 +03:00
|
|
|
caddr_t sg;
|
|
|
|
int n, error, i;
|
|
|
|
struct sys_getgroups_args bsa;
|
|
|
|
gid_t *bset, *kbset;
|
|
|
|
linux_gid_t *lset;
|
|
|
|
struct pcred *pc = p->p_cred;
|
|
|
|
|
|
|
|
n = SCARG(uap, gidsetsize);
|
|
|
|
if (n < 0)
|
|
|
|
return EINVAL;
|
|
|
|
error = 0;
|
|
|
|
bset = kbset = NULL;
|
|
|
|
lset = NULL;
|
|
|
|
if (n > 0) {
|
|
|
|
n = min(pc->pc_ucred->cr_ngroups, n);
|
2002-03-16 23:43:48 +03:00
|
|
|
sg = stackgap_init(p, 0);
|
|
|
|
bset = stackgap_alloc(p, &sg, n * sizeof (gid_t));
|
2000-12-28 01:01:42 +03:00
|
|
|
kbset = malloc(n * sizeof (gid_t), M_TEMP, M_WAITOK);
|
|
|
|
lset = malloc(n * sizeof (linux_gid_t), M_TEMP, M_WAITOK);
|
|
|
|
if (bset == NULL || kbset == NULL || lset == NULL)
|
|
|
|
return ENOMEM;
|
|
|
|
SCARG(&bsa, gidsetsize) = n;
|
|
|
|
SCARG(&bsa, gidset) = bset;
|
2003-01-18 11:02:46 +03:00
|
|
|
error = sys_getgroups(l, &bsa, retval);
|
2000-12-28 01:01:42 +03:00
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
error = copyin(bset, kbset, n * sizeof (gid_t));
|
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
lset[i] = (linux_gid_t)kbset[i];
|
|
|
|
error = copyout(lset, SCARG(uap, gidset),
|
|
|
|
n * sizeof (linux_gid_t));
|
|
|
|
} else
|
|
|
|
*retval = pc->pc_ucred->cr_ngroups;
|
|
|
|
out:
|
|
|
|
if (kbset != NULL)
|
|
|
|
free(kbset, M_TEMP);
|
|
|
|
if (lset != NULL)
|
|
|
|
free(lset, M_TEMP);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setgroups16(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-28 01:01:42 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_setgroups16_args /* {
|
|
|
|
syscallarg(int) gidsetsize;
|
|
|
|
syscallarg(linux_gid_t *) gidset;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2000-12-28 01:01:42 +03:00
|
|
|
caddr_t sg;
|
|
|
|
int n;
|
|
|
|
int error, i;
|
|
|
|
struct sys_setgroups_args bsa;
|
|
|
|
gid_t *bset, *kbset;
|
|
|
|
linux_gid_t *lset;
|
|
|
|
|
|
|
|
n = SCARG(uap, gidsetsize);
|
|
|
|
if (n < 0 || n > NGROUPS)
|
|
|
|
return EINVAL;
|
2002-03-16 23:43:48 +03:00
|
|
|
sg = stackgap_init(p, 0);
|
|
|
|
bset = stackgap_alloc(p, &sg, n * sizeof (gid_t));
|
2000-12-28 01:01:42 +03:00
|
|
|
lset = malloc(n * sizeof (linux_gid_t), M_TEMP, M_WAITOK);
|
2002-04-03 14:17:01 +04:00
|
|
|
kbset = malloc(n * sizeof (gid_t), M_TEMP, M_WAITOK);
|
2000-12-28 01:01:42 +03:00
|
|
|
if (lset == NULL || bset == NULL)
|
|
|
|
return ENOMEM;
|
|
|
|
error = copyin(SCARG(uap, gidset), lset, n * sizeof (linux_gid_t));
|
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
kbset[i] = (gid_t)lset[i];
|
|
|
|
error = copyout(kbset, bset, n * sizeof (gid_t));
|
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
SCARG(&bsa, gidsetsize) = n;
|
|
|
|
SCARG(&bsa, gidset) = bset;
|
2003-01-18 11:02:46 +03:00
|
|
|
error = sys_setgroups(l, &bsa, retval);
|
2000-12-28 01:01:42 +03:00
|
|
|
|
|
|
|
out:
|
|
|
|
if (lset != NULL)
|
|
|
|
free(lset, M_TEMP);
|
|
|
|
if (kbset != NULL)
|
|
|
|
free(kbset, M_TEMP);
|
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2000-12-21 23:19:22 +03:00
|
|
|
#endif /* __i386__ || __m68k__ */
|
1996-05-20 05:59:09 +04:00
|
|
|
|
2000-02-03 13:02:59 +03:00
|
|
|
/*
|
2000-02-17 15:34:26 +03:00
|
|
|
* We have nonexistent fsuid equal to uid.
|
|
|
|
* If modification is requested, refuse.
|
2000-02-03 13:02:59 +03:00
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setfsuid(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-02-03 13:02:59 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_setfsuid_args /* {
|
|
|
|
syscallarg(uid_t) uid;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2000-02-03 13:02:59 +03:00
|
|
|
uid_t uid;
|
|
|
|
|
|
|
|
uid = SCARG(uap, uid);
|
|
|
|
if (p->p_cred->p_ruid != uid)
|
2003-01-18 11:02:46 +03:00
|
|
|
return sys_nosys(l, v, retval);
|
2000-02-03 13:02:59 +03:00
|
|
|
else
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2000-03-18 23:42:14 +03:00
|
|
|
/* XXX XXX XXX */
|
|
|
|
#ifndef alpha
|
2000-02-03 13:02:59 +03:00
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_getfsuid(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-03-18 23:42:14 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
2000-02-03 13:02:59 +03:00
|
|
|
{
|
2003-01-18 11:02:46 +03:00
|
|
|
return sys_getuid(l, v, retval);
|
2000-02-03 13:02:59 +03:00
|
|
|
}
|
2000-03-18 23:42:14 +03:00
|
|
|
#endif
|
2000-02-03 13:02:59 +03:00
|
|
|
|
1999-05-14 22:44:50 +04:00
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setresuid(l, v, retval)
|
|
|
|
struct lwp *l;
|
1999-05-14 22:44:50 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_setresuid_args /* {
|
|
|
|
syscallarg(uid_t) ruid;
|
|
|
|
syscallarg(uid_t) euid;
|
|
|
|
syscallarg(uid_t) suid;
|
|
|
|
} */ *uap = v;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note: These checks are a little different than the NetBSD
|
|
|
|
* setreuid(2) call performs. This precisely follows the
|
|
|
|
* behavior of the Linux kernel.
|
|
|
|
*/
|
|
|
|
|
2003-03-05 21:44:46 +03:00
|
|
|
return do_setresuid(l, SCARG(uap, ruid), SCARG(uap, euid),
|
|
|
|
SCARG(uap, suid),
|
|
|
|
ID_R_EQ_R | ID_R_EQ_E | ID_R_EQ_S |
|
|
|
|
ID_E_EQ_R | ID_E_EQ_E | ID_E_EQ_S |
|
|
|
|
ID_S_EQ_R | ID_S_EQ_E | ID_S_EQ_S );
|
1999-05-14 22:44:50 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_getresuid(l, v, retval)
|
|
|
|
struct lwp *l;
|
1999-05-14 22:44:50 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_getresuid_args /* {
|
|
|
|
syscallarg(uid_t *) ruid;
|
|
|
|
syscallarg(uid_t *) euid;
|
|
|
|
syscallarg(uid_t *) suid;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
1999-05-14 22:44:50 +04:00
|
|
|
struct pcred *pc = p->p_cred;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Linux copies these values out to userspace like so:
|
|
|
|
*
|
|
|
|
* 1. Copy out ruid.
|
|
|
|
* 2. If that succeeds, copy out euid.
|
|
|
|
* 3. If both of those succeed, copy out suid.
|
|
|
|
*/
|
|
|
|
if ((error = copyout(&pc->p_ruid, SCARG(uap, ruid),
|
|
|
|
sizeof(uid_t))) != 0)
|
|
|
|
return (error);
|
|
|
|
|
|
|
|
if ((error = copyout(&pc->pc_ucred->cr_uid, SCARG(uap, euid),
|
|
|
|
sizeof(uid_t))) != 0)
|
|
|
|
return (error);
|
|
|
|
|
|
|
|
return (copyout(&pc->p_svuid, SCARG(uap, suid), sizeof(uid_t)));
|
|
|
|
}
|
1999-12-16 18:09:49 +03:00
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_ptrace(l, v, retval)
|
|
|
|
struct lwp *l;
|
1999-12-16 18:09:49 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_ptrace_args /* {
|
2001-05-20 13:29:10 +04:00
|
|
|
i386, m68k, powerpc: T=int
|
2000-03-18 23:42:14 +03:00
|
|
|
alpha: T=long
|
|
|
|
syscallarg(T) request;
|
|
|
|
syscallarg(T) pid;
|
|
|
|
syscallarg(T) addr;
|
|
|
|
syscallarg(T) data;
|
1999-12-16 18:09:49 +03:00
|
|
|
} */ *uap = v;
|
2000-11-01 23:56:30 +03:00
|
|
|
const int *ptr;
|
|
|
|
int request;
|
2001-05-28 01:17:16 +04:00
|
|
|
int error;
|
1999-12-16 18:09:49 +03:00
|
|
|
|
|
|
|
ptr = linux_ptrace_request_map;
|
|
|
|
request = SCARG(uap, request);
|
|
|
|
while (*ptr != -1)
|
|
|
|
if (*ptr++ == request) {
|
|
|
|
struct sys_ptrace_args pta;
|
|
|
|
|
|
|
|
SCARG(&pta, req) = *ptr;
|
|
|
|
SCARG(&pta, pid) = SCARG(uap, pid);
|
|
|
|
SCARG(&pta, addr) = (caddr_t)SCARG(uap, addr);
|
|
|
|
SCARG(&pta, data) = SCARG(uap, data);
|
|
|
|
|
2000-11-01 23:56:30 +03:00
|
|
|
/*
|
|
|
|
* Linux ptrace(PTRACE_CONT, pid, 0, 0) means actually
|
2001-06-02 15:46:09 +04:00
|
|
|
* to continue where the process left off previously.
|
|
|
|
* The same thing is achieved by addr == (caddr_t) 1
|
|
|
|
* on NetBSD, so rewrite 'addr' appropriately.
|
2000-11-01 23:56:30 +03:00
|
|
|
*/
|
|
|
|
if (request == LINUX_PTRACE_CONT && SCARG(uap, addr)==0)
|
|
|
|
SCARG(&pta, addr) = (caddr_t) 1;
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
error = sys_ptrace(l, &pta, retval);
|
2001-06-04 11:44:39 +04:00
|
|
|
if (error)
|
|
|
|
return error;
|
|
|
|
switch (request) {
|
|
|
|
case LINUX_PTRACE_PEEKTEXT:
|
|
|
|
case LINUX_PTRACE_PEEKDATA:
|
|
|
|
error = copyout (retval,
|
|
|
|
(caddr_t)SCARG(uap, data), sizeof *retval);
|
|
|
|
*retval = SCARG(uap, data);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2001-05-28 01:17:16 +04:00
|
|
|
return error;
|
1999-12-16 18:09:49 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
ptr++;
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return LINUX_SYS_PTRACE_ARCH(l, uap, retval);
|
1999-12-16 18:09:49 +03:00
|
|
|
}
|
2000-03-19 01:23:13 +03:00
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_reboot(struct lwp *l, void *v, register_t *retval)
|
2000-03-19 01:23:13 +03:00
|
|
|
{
|
|
|
|
struct linux_sys_reboot_args /* {
|
|
|
|
syscallarg(int) magic1;
|
|
|
|
syscallarg(int) magic2;
|
|
|
|
syscallarg(int) cmd;
|
|
|
|
syscallarg(void *) arg;
|
|
|
|
} */ *uap = v;
|
|
|
|
struct sys_reboot_args /* {
|
|
|
|
syscallarg(int) opt;
|
|
|
|
syscallarg(char *) bootstr;
|
|
|
|
} */ sra;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2000-03-19 01:23:13 +03:00
|
|
|
int error;
|
|
|
|
|
|
|
|
if ((error = suser(p->p_ucred, &p->p_acflag)) != 0)
|
|
|
|
return(error);
|
|
|
|
|
|
|
|
if (SCARG(uap, magic1) != LINUX_REBOOT_MAGIC1)
|
|
|
|
return(EINVAL);
|
|
|
|
if (SCARG(uap, magic2) != LINUX_REBOOT_MAGIC2 &&
|
|
|
|
SCARG(uap, magic2) != LINUX_REBOOT_MAGIC2A &&
|
|
|
|
SCARG(uap, magic2) != LINUX_REBOOT_MAGIC2B)
|
|
|
|
return(EINVAL);
|
|
|
|
|
|
|
|
switch (SCARG(uap, cmd)) {
|
|
|
|
case LINUX_REBOOT_CMD_RESTART:
|
|
|
|
SCARG(&sra, opt) = RB_AUTOBOOT;
|
|
|
|
break;
|
|
|
|
case LINUX_REBOOT_CMD_HALT:
|
|
|
|
SCARG(&sra, opt) = RB_HALT;
|
|
|
|
break;
|
|
|
|
case LINUX_REBOOT_CMD_POWER_OFF:
|
|
|
|
SCARG(&sra, opt) = RB_HALT|RB_POWERDOWN;
|
|
|
|
break;
|
|
|
|
case LINUX_REBOOT_CMD_RESTART2:
|
|
|
|
/* Reboot with an argument. */
|
|
|
|
SCARG(&sra, opt) = RB_AUTOBOOT|RB_STRING;
|
|
|
|
SCARG(&sra, bootstr) = SCARG(uap, arg);
|
|
|
|
break;
|
|
|
|
case LINUX_REBOOT_CMD_CAD_ON:
|
|
|
|
return(EINVAL); /* We don't implement ctrl-alt-delete */
|
|
|
|
case LINUX_REBOOT_CMD_CAD_OFF:
|
|
|
|
return(0);
|
|
|
|
default:
|
|
|
|
return(EINVAL);
|
|
|
|
}
|
|
|
|
|
2003-01-18 11:02:46 +03:00
|
|
|
return(sys_reboot(l, &sra, retval));
|
2000-03-19 01:23:13 +03:00
|
|
|
}
|
2000-12-02 00:14:42 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copy of compat_12_sys_swapon().
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_swapon(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-02 00:14:42 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct sys_swapctl_args ua;
|
|
|
|
struct linux_sys_swapon_args /* {
|
|
|
|
syscallarg(const char *) name;
|
|
|
|
} */ *uap = v;
|
|
|
|
|
|
|
|
SCARG(&ua, cmd) = SWAP_ON;
|
|
|
|
SCARG(&ua, arg) = (void *)SCARG(uap, name);
|
|
|
|
SCARG(&ua, misc) = 0; /* priority */
|
2003-01-18 11:02:46 +03:00
|
|
|
return (sys_swapctl(l, &ua, retval));
|
2000-12-02 00:14:42 +03:00
|
|
|
}
|
|
|
|
|
2000-12-02 19:43:50 +03:00
|
|
|
/*
|
|
|
|
* Stop swapping to the file or block device specified by path.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_swapoff(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-02 19:43:50 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct sys_swapctl_args ua;
|
|
|
|
struct linux_sys_swapoff_args /* {
|
|
|
|
syscallarg(const char *) path;
|
|
|
|
} */ *uap = v;
|
|
|
|
|
|
|
|
SCARG(&ua, cmd) = SWAP_OFF;
|
|
|
|
SCARG(&ua, arg) = (void *)SCARG(uap, path);
|
2003-01-18 11:02:46 +03:00
|
|
|
return (sys_swapctl(l, &ua, retval));
|
2000-12-02 19:43:50 +03:00
|
|
|
}
|
|
|
|
|
2000-12-02 00:14:42 +03:00
|
|
|
/*
|
|
|
|
* Copy of compat_09_sys_setdomainname()
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setdomainname(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-02 00:14:42 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_setdomainname_args /* {
|
|
|
|
syscallarg(char *) domainname;
|
|
|
|
syscallarg(int) len;
|
|
|
|
} */ *uap = v;
|
Dynamic sysctl.
Gone are the old kern_sysctl(), cpu_sysctl(), hw_sysctl(),
vfs_sysctl(), etc, routines, along with sysctl_int() et al. Now all
nodes are registered with the tree, and nodes can be added (or
removed) easily, and I/O to and from the tree is handled generically.
Since the nodes are registered with the tree, the mapping from name to
number (and back again) can now be discovered, instead of having to be
hard coded. Adding new nodes to the tree is likewise much simpler --
the new infrastructure handles almost all the work for simple types,
and just about anything else can be done with a small helper function.
All existing nodes are where they were before (numerically speaking),
so all existing consumers of sysctl information should notice no
difference.
PS - I'm sorry, but there's a distinct lack of documentation at the
moment. I'm working on sysctl(3/8/9) right now, and I promise to
watch out for buses.
2003-12-04 22:38:21 +03:00
|
|
|
int name[2];
|
2000-12-02 00:14:42 +03:00
|
|
|
|
Dynamic sysctl.
Gone are the old kern_sysctl(), cpu_sysctl(), hw_sysctl(),
vfs_sysctl(), etc, routines, along with sysctl_int() et al. Now all
nodes are registered with the tree, and nodes can be added (or
removed) easily, and I/O to and from the tree is handled generically.
Since the nodes are registered with the tree, the mapping from name to
number (and back again) can now be discovered, instead of having to be
hard coded. Adding new nodes to the tree is likewise much simpler --
the new infrastructure handles almost all the work for simple types,
and just about anything else can be done with a small helper function.
All existing nodes are where they were before (numerically speaking),
so all existing consumers of sysctl information should notice no
difference.
PS - I'm sorry, but there's a distinct lack of documentation at the
moment. I'm working on sysctl(3/8/9) right now, and I promise to
watch out for buses.
2003-12-04 22:38:21 +03:00
|
|
|
name[0] = CTL_KERN;
|
|
|
|
name[1] = KERN_DOMAINNAME;
|
|
|
|
return (old_sysctl(&name[0], 2, 0, 0, SCARG(uap, domainname),
|
|
|
|
SCARG(uap, len), l));
|
2000-12-02 00:14:42 +03:00
|
|
|
}
|
2000-12-14 00:41:23 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* sysinfo()
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_sysinfo(l, v, retval)
|
|
|
|
struct lwp *l;
|
2000-12-14 00:41:23 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_sysinfo_args /* {
|
|
|
|
syscallarg(struct linux_sysinfo *) arg;
|
|
|
|
} */ *uap = v;
|
|
|
|
struct linux_sysinfo si;
|
|
|
|
struct loadavg *la;
|
|
|
|
|
|
|
|
si.uptime = time.tv_sec - boottime.tv_sec;
|
|
|
|
la = &averunnable;
|
|
|
|
si.loads[0] = la->ldavg[0] * LINUX_SYSINFO_LOADS_SCALE / la->fscale;
|
|
|
|
si.loads[1] = la->ldavg[1] * LINUX_SYSINFO_LOADS_SCALE / la->fscale;
|
|
|
|
si.loads[2] = la->ldavg[2] * LINUX_SYSINFO_LOADS_SCALE / la->fscale;
|
|
|
|
si.totalram = ctob(physmem);
|
|
|
|
si.freeram = uvmexp.free * uvmexp.pagesize;
|
|
|
|
si.sharedram = 0; /* XXX */
|
2001-12-09 06:07:43 +03:00
|
|
|
si.bufferram = uvmexp.filepages * uvmexp.pagesize;
|
2000-12-14 00:41:23 +03:00
|
|
|
si.totalswap = uvmexp.swpages * uvmexp.pagesize;
|
|
|
|
si.freeswap = (uvmexp.swpages - uvmexp.swpginuse) * uvmexp.pagesize;
|
|
|
|
si.procs = nprocs;
|
|
|
|
|
|
|
|
/* The following are only present in newer Linux kernels. */
|
|
|
|
si.totalbig = 0;
|
|
|
|
si.freebig = 0;
|
|
|
|
si.mem_unit = 1;
|
|
|
|
|
|
|
|
return (copyout(&si, SCARG(uap, arg), sizeof si));
|
|
|
|
}
|
2001-03-30 21:16:33 +04:00
|
|
|
|
2002-02-15 19:47:58 +03:00
|
|
|
#define bsd_to_linux_rlimit1(l, b, f) \
|
2002-11-25 05:11:23 +03:00
|
|
|
(l)->f = ((b)->f == RLIM_INFINITY || \
|
|
|
|
((b)->f & 0xffffffff00000000ULL) != 0) ? \
|
2002-02-15 19:47:58 +03:00
|
|
|
LINUX_RLIM_INFINITY : (int32_t)(b)->f
|
|
|
|
#define bsd_to_linux_rlimit(l, b) \
|
|
|
|
bsd_to_linux_rlimit1(l, b, rlim_cur); \
|
|
|
|
bsd_to_linux_rlimit1(l, b, rlim_max)
|
|
|
|
|
|
|
|
#define linux_to_bsd_rlimit1(b, l, f) \
|
2002-06-20 08:01:43 +04:00
|
|
|
(b)->f = (l)->f == LINUX_RLIM_INFINITY ? RLIM_INFINITY : (l)->f
|
2002-02-15 19:47:58 +03:00
|
|
|
#define linux_to_bsd_rlimit(b, l) \
|
|
|
|
linux_to_bsd_rlimit1(b, l, rlim_cur); \
|
|
|
|
linux_to_bsd_rlimit1(b, l, rlim_max)
|
|
|
|
|
|
|
|
static int
|
|
|
|
linux_to_bsd_limit(lim)
|
|
|
|
int lim;
|
|
|
|
{
|
|
|
|
switch (lim) {
|
|
|
|
case LINUX_RLIMIT_CPU:
|
|
|
|
return RLIMIT_CPU;
|
|
|
|
case LINUX_RLIMIT_FSIZE:
|
|
|
|
return RLIMIT_FSIZE;
|
|
|
|
case LINUX_RLIMIT_DATA:
|
|
|
|
return RLIMIT_DATA;
|
|
|
|
case LINUX_RLIMIT_STACK:
|
|
|
|
return RLIMIT_STACK;
|
|
|
|
case LINUX_RLIMIT_CORE:
|
|
|
|
return RLIMIT_CORE;
|
|
|
|
case LINUX_RLIMIT_RSS:
|
|
|
|
return RLIMIT_RSS;
|
|
|
|
case LINUX_RLIMIT_NPROC:
|
|
|
|
return RLIMIT_NPROC;
|
|
|
|
case LINUX_RLIMIT_NOFILE:
|
|
|
|
return RLIMIT_NOFILE;
|
|
|
|
case LINUX_RLIMIT_MEMLOCK:
|
|
|
|
return RLIMIT_MEMLOCK;
|
|
|
|
case LINUX_RLIMIT_AS:
|
|
|
|
case LINUX_RLIMIT_LOCKS:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_getrlimit(l, v, retval)
|
|
|
|
struct lwp *l;
|
2002-02-15 19:47:58 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_getrlimit_args /* {
|
|
|
|
syscallarg(int) which;
|
|
|
|
syscallarg(struct orlimit *) rlp;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2002-03-16 23:43:48 +03:00
|
|
|
caddr_t sg = stackgap_init(p, 0);
|
2002-02-15 19:47:58 +03:00
|
|
|
struct sys_getrlimit_args ap;
|
|
|
|
struct rlimit rl;
|
|
|
|
struct orlimit orl;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
SCARG(&ap, which) = linux_to_bsd_limit(SCARG(uap, which));
|
|
|
|
if ((error = SCARG(&ap, which)) < 0)
|
|
|
|
return -error;
|
2002-03-16 23:43:48 +03:00
|
|
|
SCARG(&ap, rlp) = stackgap_alloc(p, &sg, sizeof rl);
|
2003-01-18 11:02:46 +03:00
|
|
|
if ((error = sys_getrlimit(l, &ap, retval)) != 0)
|
2002-02-15 19:47:58 +03:00
|
|
|
return error;
|
|
|
|
if ((error = copyin(SCARG(&ap, rlp), &rl, sizeof(rl))) != 0)
|
|
|
|
return error;
|
|
|
|
bsd_to_linux_rlimit(&orl, &rl);
|
|
|
|
return copyout(&orl, SCARG(uap, rlp), sizeof(orl));
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_setrlimit(l, v, retval)
|
|
|
|
struct lwp *l;
|
2002-02-15 19:47:58 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
struct linux_sys_setrlimit_args /* {
|
|
|
|
syscallarg(int) which;
|
|
|
|
syscallarg(struct orlimit *) rlp;
|
|
|
|
} */ *uap = v;
|
2003-01-18 11:02:46 +03:00
|
|
|
struct proc *p = l->l_proc;
|
2002-03-16 23:43:48 +03:00
|
|
|
caddr_t sg = stackgap_init(p, 0);
|
2002-02-15 19:47:58 +03:00
|
|
|
struct sys_setrlimit_args ap;
|
|
|
|
struct rlimit rl;
|
|
|
|
struct orlimit orl;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
SCARG(&ap, which) = linux_to_bsd_limit(SCARG(uap, which));
|
2002-03-16 23:43:48 +03:00
|
|
|
SCARG(&ap, rlp) = stackgap_alloc(p, &sg, sizeof rl);
|
2002-02-15 19:47:58 +03:00
|
|
|
if ((error = SCARG(&ap, which)) < 0)
|
|
|
|
return -error;
|
|
|
|
if ((error = copyin(SCARG(uap, rlp), &orl, sizeof(orl))) != 0)
|
|
|
|
return error;
|
|
|
|
linux_to_bsd_rlimit(&rl, &orl);
|
2002-02-19 01:24:17 +03:00
|
|
|
/* XXX: alpha complains about this */
|
|
|
|
if ((error = copyout(&rl, (void *)SCARG(&ap, rlp), sizeof(rl))) != 0)
|
2002-02-15 19:47:58 +03:00
|
|
|
return error;
|
2003-01-18 11:02:46 +03:00
|
|
|
return sys_setrlimit(l, &ap, retval);
|
2002-02-15 19:47:58 +03:00
|
|
|
}
|
|
|
|
|
2002-02-18 02:56:35 +03:00
|
|
|
#ifndef __mips__
|
|
|
|
/* XXX: this doesn't look 100% common, at least mips doesn't have it */
|
2002-02-15 19:47:58 +03:00
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_ugetrlimit(l, v, retval)
|
|
|
|
struct lwp *l;
|
2002-02-15 19:47:58 +03:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
2003-01-18 11:02:46 +03:00
|
|
|
return linux_sys_getrlimit(l, v, retval);
|
2002-02-15 19:47:58 +03:00
|
|
|
}
|
2002-02-18 02:56:35 +03:00
|
|
|
#endif
|
2002-02-15 19:47:58 +03:00
|
|
|
|
2001-03-30 21:16:33 +04:00
|
|
|
/*
|
|
|
|
* This gets called for unsupported syscalls. The difference to sys_nosys()
|
|
|
|
* is that process does not get SIGSYS, the call just returns with ENOSYS.
|
|
|
|
* This is the way Linux does it and glibc depends on this behaviour.
|
|
|
|
*/
|
|
|
|
int
|
2003-01-18 11:02:46 +03:00
|
|
|
linux_sys_nosys(l, v, retval)
|
|
|
|
struct lwp *l;
|
2001-03-30 21:16:33 +04:00
|
|
|
void *v;
|
|
|
|
register_t *retval;
|
|
|
|
{
|
|
|
|
return (ENOSYS);
|
|
|
|
}
|