2005-01-05 22:34:37 +00:00
|
|
|
/*-
|
1999-07-30 12:45:21 +00:00
|
|
|
* Copyright (c) 1998 Mark Newton
|
|
|
|
* 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. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by Christos Zoulas.
|
|
|
|
* 4. 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.
|
|
|
|
*/
|
|
|
|
|
2003-06-10 21:44:29 +00:00
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
1999-01-30 06:29:48 +00:00
|
|
|
/* XXX we use functions that might not exist. */
|
|
|
|
#include "opt_compat.h"
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
1999-07-03 04:56:57 +00:00
|
|
|
#include <sys/systm.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/sysent.h>
|
|
|
|
#include <sys/imgact.h>
|
|
|
|
#include <sys/imgact_elf.h>
|
2008-03-31 12:01:21 +00:00
|
|
|
#include <sys/fcntl.h>
|
2002-04-11 21:00:38 +00:00
|
|
|
#include <sys/lock.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/malloc.h>
|
- Implement svr4_emul_find() using kern_alternate_path(). This changes
the semantics in that the returned filename to use is now a kernel
pointer rather than a user space pointer. This required changing the
arguments to the CHECKALT*() macros some and changing the various system
calls that used pathnames to use the kern_foo() functions that can accept
kernel space filename pointers instead of calling the system call
directly.
- Use kern_open(), kern_access(), kern_msgctl(), kern_execve(),
kern_mkfifo(), kern_mknod(), kern_statfs(), kern_fstatfs(),
kern_setitimer(), kern_stat(), kern_lstat(), kern_fstat(), kern_utimes(),
kern_pathconf(), and kern_unlink().
2005-02-07 21:53:42 +00:00
|
|
|
#include <sys/module.h>
|
2002-04-11 21:00:38 +00:00
|
|
|
#include <sys/mutex.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/namei.h>
|
- Implement svr4_emul_find() using kern_alternate_path(). This changes
the semantics in that the returned filename to use is now a kernel
pointer rather than a user space pointer. This required changing the
arguments to the CHECKALT*() macros some and changing the various system
calls that used pathnames to use the kern_foo() functions that can accept
kernel space filename pointers instead of calling the system call
directly.
- Use kern_open(), kern_access(), kern_msgctl(), kern_execve(),
kern_mkfifo(), kern_mknod(), kern_statfs(), kern_fstatfs(),
kern_setitimer(), kern_stat(), kern_lstat(), kern_fstat(), kern_utimes(),
kern_pathconf(), and kern_unlink().
2005-02-07 21:53:42 +00:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/syscallsubr.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
#include <sys/vnode.h>
|
|
|
|
#include <vm/vm.h>
|
|
|
|
#include <sys/exec.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <machine/cpu.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
|
2000-08-31 22:54:09 +00:00
|
|
|
#include <compat/svr4/svr4.h>
|
|
|
|
#include <compat/svr4/svr4_types.h>
|
|
|
|
#include <compat/svr4/svr4_syscall.h>
|
|
|
|
#include <compat/svr4/svr4_signal.h>
|
2006-07-21 20:40:13 +00:00
|
|
|
#include <compat/svr4/svr4_socket.h>
|
2000-08-31 22:54:09 +00:00
|
|
|
#include <compat/svr4/svr4_sockio.h>
|
|
|
|
#include <compat/svr4/svr4_errno.h>
|
|
|
|
#include <compat/svr4/svr4_proto.h>
|
|
|
|
#include <compat/svr4/svr4_siginfo.h>
|
|
|
|
#include <compat/svr4/svr4_util.h>
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
int bsd_to_svr4_errno[ELAST+1] = {
|
|
|
|
0,
|
|
|
|
SVR4_EPERM,
|
|
|
|
SVR4_ENOENT,
|
|
|
|
SVR4_ESRCH,
|
|
|
|
SVR4_EINTR,
|
|
|
|
SVR4_EIO,
|
|
|
|
SVR4_ENXIO,
|
|
|
|
SVR4_E2BIG,
|
|
|
|
SVR4_ENOEXEC,
|
|
|
|
SVR4_EBADF,
|
|
|
|
SVR4_ECHILD,
|
|
|
|
SVR4_EDEADLK,
|
|
|
|
SVR4_ENOMEM,
|
|
|
|
SVR4_EACCES,
|
|
|
|
SVR4_EFAULT,
|
|
|
|
SVR4_ENOTBLK,
|
|
|
|
SVR4_EBUSY,
|
|
|
|
SVR4_EEXIST,
|
|
|
|
SVR4_EXDEV,
|
|
|
|
SVR4_ENODEV,
|
|
|
|
SVR4_ENOTDIR,
|
|
|
|
SVR4_EISDIR,
|
|
|
|
SVR4_EINVAL,
|
|
|
|
SVR4_ENFILE,
|
|
|
|
SVR4_EMFILE,
|
|
|
|
SVR4_ENOTTY,
|
|
|
|
SVR4_ETXTBSY,
|
|
|
|
SVR4_EFBIG,
|
|
|
|
SVR4_ENOSPC,
|
|
|
|
SVR4_ESPIPE,
|
|
|
|
SVR4_EROFS,
|
|
|
|
SVR4_EMLINK,
|
|
|
|
SVR4_EPIPE,
|
|
|
|
SVR4_EDOM,
|
|
|
|
SVR4_ERANGE,
|
|
|
|
SVR4_EAGAIN,
|
|
|
|
SVR4_EINPROGRESS,
|
|
|
|
SVR4_EALREADY,
|
|
|
|
SVR4_ENOTSOCK,
|
|
|
|
SVR4_EDESTADDRREQ,
|
|
|
|
SVR4_EMSGSIZE,
|
|
|
|
SVR4_EPROTOTYPE,
|
|
|
|
SVR4_ENOPROTOOPT,
|
|
|
|
SVR4_EPROTONOSUPPORT,
|
|
|
|
SVR4_ESOCKTNOSUPPORT,
|
|
|
|
SVR4_EOPNOTSUPP,
|
|
|
|
SVR4_EPFNOSUPPORT,
|
|
|
|
SVR4_EAFNOSUPPORT,
|
|
|
|
SVR4_EADDRINUSE,
|
|
|
|
SVR4_EADDRNOTAVAIL,
|
|
|
|
SVR4_ENETDOWN,
|
|
|
|
SVR4_ENETUNREACH,
|
|
|
|
SVR4_ENETRESET,
|
|
|
|
SVR4_ECONNABORTED,
|
|
|
|
SVR4_ECONNRESET,
|
|
|
|
SVR4_ENOBUFS,
|
|
|
|
SVR4_EISCONN,
|
|
|
|
SVR4_ENOTCONN,
|
|
|
|
SVR4_ESHUTDOWN,
|
|
|
|
SVR4_ETOOMANYREFS,
|
|
|
|
SVR4_ETIMEDOUT,
|
|
|
|
SVR4_ECONNREFUSED,
|
|
|
|
SVR4_ELOOP,
|
|
|
|
SVR4_ENAMETOOLONG,
|
|
|
|
SVR4_EHOSTDOWN,
|
|
|
|
SVR4_EHOSTUNREACH,
|
|
|
|
SVR4_ENOTEMPTY,
|
|
|
|
SVR4_EPROCLIM,
|
|
|
|
SVR4_EUSERS,
|
|
|
|
SVR4_EDQUOT,
|
|
|
|
SVR4_ESTALE,
|
|
|
|
SVR4_EREMOTE,
|
|
|
|
SVR4_EBADRPC,
|
|
|
|
SVR4_ERPCMISMATCH,
|
|
|
|
SVR4_EPROGUNAVAIL,
|
|
|
|
SVR4_EPROGMISMATCH,
|
|
|
|
SVR4_EPROCUNAVAIL,
|
|
|
|
SVR4_ENOLCK,
|
|
|
|
SVR4_ENOSYS,
|
|
|
|
SVR4_EFTYPE,
|
|
|
|
SVR4_EAUTH,
|
|
|
|
SVR4_ENEEDAUTH,
|
|
|
|
SVR4_EIDRM,
|
|
|
|
SVR4_ENOMSG,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2000-04-18 02:39:26 +00:00
|
|
|
static int svr4_fixup(register_t **stack_base, struct image_params *imgp);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
|
|
|
extern struct sysent svr4_sysent[];
|
|
|
|
#undef szsigcode
|
|
|
|
#undef sigcode
|
|
|
|
|
|
|
|
extern int svr4_szsigcode;
|
|
|
|
extern char svr4_sigcode[];
|
|
|
|
|
|
|
|
struct sysentvec svr4_sysvec = {
|
2008-09-24 10:14:37 +00:00
|
|
|
.sv_size = SVR4_SYS_MAXSYSCALL,
|
|
|
|
.sv_table = svr4_sysent,
|
|
|
|
.sv_mask = 0xff,
|
|
|
|
.sv_sigsize = SVR4_NSIG-1, /* NB: signal trans table indexed with signno-1 */
|
|
|
|
.sv_sigtbl = bsd_to_svr4_sig+1,
|
|
|
|
.sv_errsize = ELAST, /* ELAST */
|
|
|
|
.sv_errtbl = bsd_to_svr4_errno,
|
|
|
|
.sv_transtrap = NULL,
|
|
|
|
.sv_fixup = svr4_fixup,
|
|
|
|
.sv_sendsig = svr4_sendsig,
|
|
|
|
.sv_sigcode = svr4_sigcode,
|
|
|
|
.sv_szsigcode = &svr4_szsigcode,
|
|
|
|
.sv_prepsyscall = NULL,
|
|
|
|
.sv_name = "SVR4",
|
|
|
|
.sv_coredump = elf32_coredump,
|
|
|
|
.sv_imgact_try = NULL,
|
|
|
|
.sv_minsigstksz = SVR4_MINSIGSTKSZ,
|
|
|
|
.sv_pagesize = PAGE_SIZE,
|
|
|
|
.sv_minuser = VM_MIN_ADDRESS,
|
|
|
|
.sv_maxuser = VM_MAXUSER_ADDRESS,
|
|
|
|
.sv_usrstack = USRSTACK,
|
|
|
|
.sv_psstrings = PS_STRINGS,
|
|
|
|
.sv_stackprot = VM_PROT_ALL,
|
|
|
|
.sv_copyout_strings = exec_copyout_strings,
|
|
|
|
.sv_setregs = exec_setregs,
|
|
|
|
.sv_fixlimit = NULL,
|
2008-11-22 12:36:15 +00:00
|
|
|
.sv_maxssiz = NULL,
|
Reorganize syscall entry and leave handling.
Extend struct sysvec with three new elements:
sv_fetch_syscall_args - the method to fetch syscall arguments from
usermode into struct syscall_args. The structure is machine-depended
(this might be reconsidered after all architectures are converted).
sv_set_syscall_retval - the method to set a return value for usermode
from the syscall. It is a generalization of
cpu_set_syscall_retval(9) to allow ABIs to override the way to set a
return value.
sv_syscallnames - the table of syscall names.
Use sv_set_syscall_retval in kern_sigsuspend() instead of hardcoding
the call to cpu_set_syscall_retval().
The new functions syscallenter(9) and syscallret(9) are provided that
use sv_*syscall* pointers and contain the common repeated code from
the syscall() implementations for the architecture-specific syscall
trap handlers.
Syscallenter() fetches arguments, calls syscall implementation from
ABI sysent table, and set up return frame. The end of syscall
bookkeeping is done by syscallret().
Take advantage of single place for MI syscall handling code and
implement ptrace_lwpinfo pl_flags PL_FLAG_SCE, PL_FLAG_SCX and
PL_FLAG_EXEC. The SCE and SCX flags notify the debugger that the
thread is stopped at syscall entry or return point respectively. The
EXEC flag augments SCX and notifies debugger that the process address
space was changed by one of exec(2)-family syscalls.
The i386, amd64, sparc64, sun4v, powerpc and ia64 syscall()s are
changed to use syscallenter()/syscallret(). MIPS and arm are not
converted and use the mostly unchanged syscall() implementation.
Reviewed by: jhb, marcel, marius, nwhitehorn, stas
Tested by: marcel (ia64), marius (sparc64), nwhitehorn (powerpc),
stas (mips)
MFC after: 1 month
2010-05-23 18:32:02 +00:00
|
|
|
.sv_flags = SV_ABI_UNDEF | SV_IA32 | SV_ILP32,
|
|
|
|
.sv_set_syscall_retval = cpu_set_syscall_retval,
|
|
|
|
.sv_fetch_syscall_args = cpu_fetch_syscall_args,
|
|
|
|
.sv_syscallnames = NULL,
|
2011-03-08 19:01:45 +00:00
|
|
|
.sv_schedtail = NULL,
|
1999-01-30 06:29:48 +00:00
|
|
|
};
|
|
|
|
|
- Implement svr4_emul_find() using kern_alternate_path(). This changes
the semantics in that the returned filename to use is now a kernel
pointer rather than a user space pointer. This required changing the
arguments to the CHECKALT*() macros some and changing the various system
calls that used pathnames to use the kern_foo() functions that can accept
kernel space filename pointers instead of calling the system call
directly.
- Use kern_open(), kern_access(), kern_msgctl(), kern_execve(),
kern_mkfifo(), kern_mknod(), kern_statfs(), kern_fstatfs(),
kern_setitimer(), kern_stat(), kern_lstat(), kern_fstat(), kern_utimes(),
kern_pathconf(), and kern_unlink().
2005-02-07 21:53:42 +00:00
|
|
|
const char svr4_emul_path[] = "/compat/svr4";
|
|
|
|
|
1999-01-30 06:29:48 +00:00
|
|
|
Elf32_Brandinfo svr4_brand = {
|
2008-09-24 10:14:37 +00:00
|
|
|
.brand = ELFOSABI_SYSV,
|
|
|
|
.machine = EM_386, /* XXX only implemented for x86 so far. */
|
|
|
|
.compat_3_brand = "SVR4",
|
|
|
|
.emul_path = svr4_emul_path,
|
|
|
|
.interp_path = "/lib/libc.so.1",
|
|
|
|
.sysvec = &svr4_sysvec,
|
|
|
|
.interp_newpath = NULL,
|
2009-03-13 16:40:51 +00:00
|
|
|
.brand_note = NULL,
|
2008-09-24 10:14:37 +00:00
|
|
|
.flags = 0
|
1999-01-30 06:29:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
2000-04-18 02:39:26 +00:00
|
|
|
svr4_fixup(register_t **stack_base, struct image_params *imgp)
|
1999-01-30 06:29:48 +00:00
|
|
|
{
|
2003-03-21 19:49:34 +00:00
|
|
|
Elf32_Auxargs *args;
|
2000-04-18 02:39:26 +00:00
|
|
|
register_t *pos;
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2008-03-12 22:17:06 +00:00
|
|
|
KASSERT(curthread->td_proc == imgp->proc,
|
2003-03-21 19:49:34 +00:00
|
|
|
("unsafe svr4_fixup(), should be curproc"));
|
|
|
|
args = (Elf32_Auxargs *)imgp->auxargs;
|
2005-01-29 23:12:00 +00:00
|
|
|
pos = *stack_base + (imgp->args->argc + imgp->args->envc + 2);
|
1999-01-30 06:29:48 +00:00
|
|
|
|
2003-03-21 19:49:34 +00:00
|
|
|
if (args->execfd != -1)
|
1999-01-30 06:29:48 +00:00
|
|
|
AUXARGS_ENTRY(pos, AT_EXECFD, args->execfd);
|
|
|
|
AUXARGS_ENTRY(pos, AT_PHDR, args->phdr);
|
|
|
|
AUXARGS_ENTRY(pos, AT_PHENT, args->phent);
|
|
|
|
AUXARGS_ENTRY(pos, AT_PHNUM, args->phnum);
|
|
|
|
AUXARGS_ENTRY(pos, AT_PAGESZ, args->pagesz);
|
|
|
|
AUXARGS_ENTRY(pos, AT_FLAGS, args->flags);
|
|
|
|
AUXARGS_ENTRY(pos, AT_ENTRY, args->entry);
|
|
|
|
AUXARGS_ENTRY(pos, AT_BASE, args->base);
|
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
|
|
|
AUXARGS_ENTRY(pos, AT_UID, imgp->proc->p_ucred->cr_ruid);
|
|
|
|
AUXARGS_ENTRY(pos, AT_EUID, imgp->proc->p_ucred->cr_svuid);
|
|
|
|
AUXARGS_ENTRY(pos, AT_GID, imgp->proc->p_ucred->cr_rgid);
|
|
|
|
AUXARGS_ENTRY(pos, AT_EGID, imgp->proc->p_ucred->cr_svgid);
|
1999-01-30 06:29:48 +00:00
|
|
|
AUXARGS_ENTRY(pos, AT_NULL, 0);
|
|
|
|
|
|
|
|
free(imgp->auxargs, M_TEMP);
|
|
|
|
imgp->auxargs = NULL;
|
|
|
|
|
|
|
|
(*stack_base)--;
|
2005-01-29 23:12:00 +00:00
|
|
|
**stack_base = (register_t)imgp->args->argc;
|
1999-01-30 06:29:48 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Search an alternate path before passing pathname arguments on
|
2001-02-06 11:21:58 +00:00
|
|
|
* to system calls. Useful for keeping a separate 'emulation tree'.
|
1999-01-30 06:29:48 +00:00
|
|
|
*
|
|
|
|
* If cflag is set, we check if an attempt can be made to create
|
|
|
|
* the named file, i.e. we check if the directory it should
|
|
|
|
* be in exists.
|
|
|
|
*/
|
|
|
|
int
|
- Implement svr4_emul_find() using kern_alternate_path(). This changes
the semantics in that the returned filename to use is now a kernel
pointer rather than a user space pointer. This required changing the
arguments to the CHECKALT*() macros some and changing the various system
calls that used pathnames to use the kern_foo() functions that can accept
kernel space filename pointers instead of calling the system call
directly.
- Use kern_open(), kern_access(), kern_msgctl(), kern_execve(),
kern_mkfifo(), kern_mknod(), kern_statfs(), kern_fstatfs(),
kern_setitimer(), kern_stat(), kern_lstat(), kern_fstat(), kern_utimes(),
kern_pathconf(), and kern_unlink().
2005-02-07 21:53:42 +00:00
|
|
|
svr4_emul_find(struct thread *td, char *path, enum uio_seg pathseg,
|
|
|
|
char **pbuf, int create)
|
1999-01-30 06:29:48 +00:00
|
|
|
{
|
|
|
|
|
- Implement svr4_emul_find() using kern_alternate_path(). This changes
the semantics in that the returned filename to use is now a kernel
pointer rather than a user space pointer. This required changing the
arguments to the CHECKALT*() macros some and changing the various system
calls that used pathnames to use the kern_foo() functions that can accept
kernel space filename pointers instead of calling the system call
directly.
- Use kern_open(), kern_access(), kern_msgctl(), kern_execve(),
kern_mkfifo(), kern_mknod(), kern_statfs(), kern_fstatfs(),
kern_setitimer(), kern_stat(), kern_lstat(), kern_fstat(), kern_utimes(),
kern_pathconf(), and kern_unlink().
2005-02-07 21:53:42 +00:00
|
|
|
return (kern_alternate_path(td, svr4_emul_path, path, pathseg, pbuf,
|
Implement the linux syscalls
openat, mkdirat, mknodat, fchownat, futimesat, fstatat, unlinkat,
renameat, linkat, symlinkat, readlinkat, fchmodat, faccessat.
Submitted by: rdivacky
Sponsored by: Google Summer of Code 2007
Tested by: pho
2008-04-08 09:45:49 +00:00
|
|
|
create, AT_FDCWD));
|
1999-01-30 06:29:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
svr4_elf_modevent(module_t mod, int type, void *data)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
|
|
|
|
error = 0;
|
|
|
|
|
|
|
|
switch(type) {
|
|
|
|
case MOD_LOAD:
|
2006-07-21 20:40:13 +00:00
|
|
|
if (elf32_insert_brand_entry(&svr4_brand) < 0) {
|
1999-01-30 06:29:48 +00:00
|
|
|
printf("cannot insert svr4 elf brand handler\n");
|
2006-07-21 20:40:13 +00:00
|
|
|
error = EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (bootverbose)
|
1999-01-30 06:29:48 +00:00
|
|
|
printf("svr4 ELF exec handler installed\n");
|
2006-07-21 20:40:13 +00:00
|
|
|
svr4_sockcache_init();
|
1999-01-30 06:29:48 +00:00
|
|
|
break;
|
|
|
|
case MOD_UNLOAD:
|
1999-02-04 12:43:17 +00:00
|
|
|
/* Only allow the emulator to be removed if it isn't in use. */
|
2002-07-20 02:56:12 +00:00
|
|
|
if (elf32_brand_inuse(&svr4_brand) != 0) {
|
1999-02-04 12:43:17 +00:00
|
|
|
error = EBUSY;
|
2002-07-20 02:56:12 +00:00
|
|
|
} else if (elf32_remove_brand_entry(&svr4_brand) < 0) {
|
1999-01-30 06:29:48 +00:00
|
|
|
error = EINVAL;
|
1999-02-04 12:43:17 +00:00
|
|
|
}
|
|
|
|
|
2006-07-21 20:40:13 +00:00
|
|
|
if (error) {
|
1999-02-04 12:43:17 +00:00
|
|
|
printf("Could not deinstall ELF interpreter entry (error %d)\n",
|
|
|
|
error);
|
2006-07-21 20:40:13 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (bootverbose)
|
1999-01-30 06:29:48 +00:00
|
|
|
printf("svr4 ELF exec handler removed\n");
|
2006-07-21 20:40:13 +00:00
|
|
|
svr4_sockcache_destroy();
|
1999-01-30 06:29:48 +00:00
|
|
|
break;
|
|
|
|
default:
|
2004-07-15 08:26:07 +00:00
|
|
|
return (EOPNOTSUPP);
|
1999-01-30 06:29:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
static moduledata_t svr4_elf_mod = {
|
|
|
|
"svr4elf",
|
|
|
|
svr4_elf_modevent,
|
|
|
|
0
|
|
|
|
};
|
2010-10-12 09:18:17 +00:00
|
|
|
DECLARE_MODULE_TIED(svr4elf, svr4_elf_mod, SI_SUB_EXEC, SI_ORDER_ANY);
|
2000-05-06 01:39:45 +00:00
|
|
|
MODULE_DEPEND(svr4elf, streams, 1, 1, 1);
|