2015-04-11 15:54:41 +03:00
|
|
|
/* $NetBSD: hijack.c,v 1.117 2015/04/11 12:54:41 riastradh Exp $ */
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
|
|
|
/*-
|
|
|
|
* Copyright (c) 2011 Antti Kantee. 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.
|
|
|
|
*
|
|
|
|
* 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 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.
|
|
|
|
*/
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
/*
|
|
|
|
* XXX: rumphijack sort of works on glibc Linux. But it's not
|
|
|
|
* the same quality working as on NetBSD.
|
|
|
|
* autoconf HAVE_FOO vs. __NetBSD__ / __linux__ could be further
|
|
|
|
* improved.
|
|
|
|
*/
|
2013-09-10 20:53:06 +04:00
|
|
|
#include <rump/rumpuser_port.h>
|
2012-08-25 22:00:06 +04:00
|
|
|
|
2013-07-19 02:58:35 +04:00
|
|
|
#if !defined(lint)
|
2015-04-11 15:54:41 +03:00
|
|
|
__RCSID("$NetBSD: hijack.c,v 1.117 2015/04/11 12:54:41 riastradh Exp $");
|
2013-07-19 02:58:35 +04:00
|
|
|
#endif
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/ioctl.h>
|
2011-02-21 16:19:35 +03:00
|
|
|
#include <sys/mman.h>
|
2011-02-17 20:18:08 +03:00
|
|
|
#include <sys/mount.h>
|
|
|
|
#include <sys/socket.h>
|
2012-08-25 22:00:06 +04:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/time.h>
|
2013-07-20 22:46:15 +04:00
|
|
|
#include <sys/uio.h>
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2014-04-02 21:09:23 +04:00
|
|
|
#include <sys/statvfs.h>
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_KQUEUE
|
2012-08-25 22:00:06 +04:00
|
|
|
#include <sys/event.h>
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
#include <sys/quotactl.h>
|
|
|
|
#endif
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <err.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <poll.h>
|
|
|
|
#include <pthread.h>
|
2011-01-08 21:11:46 +03:00
|
|
|
#include <signal.h>
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
#include <stdarg.h>
|
2011-01-17 19:27:54 +03:00
|
|
|
#include <stdbool.h>
|
2012-08-25 22:00:06 +04:00
|
|
|
#include <stdint.h>
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2011-02-06 18:48:20 +03:00
|
|
|
#include <string.h>
|
2011-01-08 21:11:46 +03:00
|
|
|
#include <time.h>
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
#include <unistd.h>
|
|
|
|
|
2012-08-25 22:00:06 +04:00
|
|
|
#include <rump/rumpclient.h>
|
|
|
|
#include <rump/rump_syscalls.h>
|
|
|
|
|
2011-02-23 18:23:15 +03:00
|
|
|
#include "hijack.h"
|
|
|
|
|
2012-08-25 22:00:06 +04:00
|
|
|
/*
|
|
|
|
* XXX: Consider autogenerating this, syscnames[] and syscalls[] with
|
|
|
|
* a DSL where the tool also checks the symbols exported by this library
|
|
|
|
* to make sure all relevant calls are accounted for.
|
|
|
|
*/
|
2011-01-25 15:18:33 +03:00
|
|
|
enum dualcall {
|
2011-02-21 15:51:06 +03:00
|
|
|
DUALCALL_WRITE, DUALCALL_WRITEV, DUALCALL_PWRITE, DUALCALL_PWRITEV,
|
2011-01-25 15:18:33 +03:00
|
|
|
DUALCALL_IOCTL, DUALCALL_FCNTL,
|
|
|
|
DUALCALL_SOCKET, DUALCALL_ACCEPT, DUALCALL_BIND, DUALCALL_CONNECT,
|
|
|
|
DUALCALL_GETPEERNAME, DUALCALL_GETSOCKNAME, DUALCALL_LISTEN,
|
|
|
|
DUALCALL_RECVFROM, DUALCALL_RECVMSG,
|
|
|
|
DUALCALL_SENDTO, DUALCALL_SENDMSG,
|
|
|
|
DUALCALL_GETSOCKOPT, DUALCALL_SETSOCKOPT,
|
|
|
|
DUALCALL_SHUTDOWN,
|
2011-02-21 15:51:06 +03:00
|
|
|
DUALCALL_READ, DUALCALL_READV, DUALCALL_PREAD, DUALCALL_PREADV,
|
2011-02-15 17:01:52 +03:00
|
|
|
DUALCALL_DUP2,
|
2011-02-08 17:45:35 +03:00
|
|
|
DUALCALL_CLOSE,
|
2011-01-25 15:18:33 +03:00
|
|
|
DUALCALL_POLLTS,
|
2012-09-03 15:33:35 +04:00
|
|
|
|
|
|
|
#ifndef __linux__
|
2011-02-17 15:23:58 +03:00
|
|
|
DUALCALL_STAT, DUALCALL_LSTAT, DUALCALL_FSTAT,
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
DUALCALL_CHMOD, DUALCALL_LCHMOD, DUALCALL_FCHMOD,
|
|
|
|
DUALCALL_CHOWN, DUALCALL_LCHOWN, DUALCALL_FCHOWN,
|
|
|
|
DUALCALL_OPEN,
|
|
|
|
DUALCALL_CHDIR, DUALCALL_FCHDIR,
|
|
|
|
DUALCALL_LSEEK,
|
|
|
|
DUALCALL_UNLINK, DUALCALL_SYMLINK, DUALCALL_READLINK,
|
2012-08-04 07:56:47 +04:00
|
|
|
DUALCALL_LINK, DUALCALL_RENAME,
|
2011-02-17 15:23:58 +03:00
|
|
|
DUALCALL_MKDIR, DUALCALL_RMDIR,
|
|
|
|
DUALCALL_UTIMES, DUALCALL_LUTIMES, DUALCALL_FUTIMES,
|
2015-03-03 04:24:39 +03:00
|
|
|
DUALCALL_UTIMENSAT, DUALCALL_FUTIMENS,
|
2011-02-17 15:23:58 +03:00
|
|
|
DUALCALL_TRUNCATE, DUALCALL_FTRUNCATE,
|
2012-08-25 22:00:06 +04:00
|
|
|
DUALCALL_FSYNC,
|
2011-02-23 18:29:21 +03:00
|
|
|
DUALCALL_ACCESS,
|
2012-09-03 15:33:35 +04:00
|
|
|
|
|
|
|
#ifndef __linux__
|
|
|
|
DUALCALL___GETCWD,
|
2012-09-03 16:07:42 +04:00
|
|
|
DUALCALL_GETDENTS,
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __linux__
|
2011-02-25 21:29:00 +03:00
|
|
|
DUALCALL_MKNOD,
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-03-09 18:03:18 +03:00
|
|
|
DUALCALL_GETFH, DUALCALL_FHOPEN, DUALCALL_FHSTAT, DUALCALL_FHSTATVFS1,
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_KQUEUE
|
2012-09-03 15:33:35 +04:00
|
|
|
DUALCALL_KEVENT,
|
|
|
|
#endif
|
2012-08-25 22:00:06 +04:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
DUALCALL___SYSCTL,
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
DUALCALL_NFSSVC,
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
DUALCALL_STATVFS1, DUALCALL_FSTATVFS1, DUALCALL_GETVFSSTAT,
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
DUALCALL_MOUNT, DUALCALL_UNMOUNT,
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_FSYNC_RANGE
|
2012-08-25 22:00:06 +04:00
|
|
|
DUALCALL_FSYNC_RANGE,
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_CHFLAGS
|
2012-08-25 22:00:06 +04:00
|
|
|
DUALCALL_CHFLAGS, DUALCALL_LCHFLAGS, DUALCALL_FCHFLAGS,
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE___QUOTACTL
|
2011-03-09 21:45:30 +03:00
|
|
|
DUALCALL_QUOTACTL,
|
2011-03-10 12:47:32 +03:00
|
|
|
#endif
|
2011-01-25 15:18:33 +03:00
|
|
|
DUALCALL__NUM
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
};
|
|
|
|
|
2011-01-17 19:27:54 +03:00
|
|
|
#define RSYS_STRING(a) __STRING(a)
|
|
|
|
#define RSYS_NAME(a) RSYS_STRING(__CONCAT(RUMP_SYS_RENAME_,a))
|
|
|
|
|
2011-01-18 22:41:02 +03:00
|
|
|
/*
|
|
|
|
* Would be nice to get this automatically in sync with libc.
|
2012-09-03 15:33:35 +04:00
|
|
|
* Also, this does not work for compat-using binaries (we should
|
|
|
|
* provide all previous interfaces, not just the current ones)
|
2011-01-18 22:41:02 +03:00
|
|
|
*/
|
2012-09-03 15:33:35 +04:00
|
|
|
#if defined(__NetBSD__)
|
|
|
|
|
2011-01-18 22:41:02 +03:00
|
|
|
#if !__NetBSD_Prereq__(5,99,7)
|
2011-02-07 13:28:18 +03:00
|
|
|
#define REALSELECT select
|
|
|
|
#define REALPOLLTS pollts
|
2011-02-08 17:45:35 +03:00
|
|
|
#define REALKEVENT kevent
|
2011-02-17 15:23:58 +03:00
|
|
|
#define REALSTAT __stat30
|
|
|
|
#define REALLSTAT __lstat30
|
|
|
|
#define REALFSTAT __fstat30
|
|
|
|
#define REALUTIMES utimes
|
|
|
|
#define REALLUTIMES lutimes
|
|
|
|
#define REALFUTIMES futimes
|
2011-02-25 21:29:00 +03:00
|
|
|
#define REALMKNOD mknod
|
2011-03-09 18:03:18 +03:00
|
|
|
#define REALFHSTAT __fhstat40
|
2012-08-25 22:00:06 +04:00
|
|
|
#else /* >= 5.99.7 */
|
2011-02-07 13:28:18 +03:00
|
|
|
#define REALSELECT _sys___select50
|
|
|
|
#define REALPOLLTS _sys___pollts50
|
2011-02-08 17:45:35 +03:00
|
|
|
#define REALKEVENT _sys___kevent50
|
2011-02-17 15:23:58 +03:00
|
|
|
#define REALSTAT __stat50
|
|
|
|
#define REALLSTAT __lstat50
|
|
|
|
#define REALFSTAT __fstat50
|
|
|
|
#define REALUTIMES __utimes50
|
|
|
|
#define REALLUTIMES __lutimes50
|
2011-02-25 21:36:36 +03:00
|
|
|
#define REALFUTIMES __futimes50
|
2011-02-25 21:29:00 +03:00
|
|
|
#define REALMKNOD __mknod50
|
2011-03-09 18:03:18 +03:00
|
|
|
#define REALFHSTAT __fhstat50
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif /* < 5.99.7 */
|
2012-09-03 15:33:35 +04:00
|
|
|
|
2011-02-07 15:23:05 +03:00
|
|
|
#define REALREAD _sys_read
|
2011-02-21 15:51:06 +03:00
|
|
|
#define REALPREAD _sys_pread
|
|
|
|
#define REALPWRITE _sys_pwrite
|
2011-02-17 15:23:58 +03:00
|
|
|
#define REALGETDENTS __getdents30
|
2011-02-17 20:18:08 +03:00
|
|
|
#define REALMOUNT __mount50
|
2011-03-09 18:03:18 +03:00
|
|
|
#define REALGETFH __getfh30
|
|
|
|
#define REALFHOPEN __fhopen40
|
|
|
|
#define REALFHSTATVFS1 __fhstatvfs140
|
2012-08-25 22:00:06 +04:00
|
|
|
#define REALSOCKET __socket30
|
|
|
|
|
|
|
|
#define LSEEK_ALIAS _lseek
|
|
|
|
#define VFORK __vfork14
|
|
|
|
|
2012-09-03 15:33:35 +04:00
|
|
|
int REALSTAT(const char *, struct stat *);
|
|
|
|
int REALLSTAT(const char *, struct stat *);
|
|
|
|
int REALFSTAT(int, struct stat *);
|
|
|
|
int REALMKNOD(const char *, mode_t, dev_t);
|
2012-09-03 16:07:42 +04:00
|
|
|
int REALGETDENTS(int, char *, size_t);
|
2012-09-03 15:33:35 +04:00
|
|
|
|
|
|
|
int __getcwd(char *, size_t);
|
|
|
|
|
|
|
|
#elif defined(__linux__) /* glibc, really */
|
2012-08-25 22:00:06 +04:00
|
|
|
|
|
|
|
#define REALREAD read
|
|
|
|
#define REALPREAD pread
|
|
|
|
#define REALPWRITE pwrite
|
|
|
|
#define REALSELECT select
|
|
|
|
#define REALPOLLTS ppoll
|
|
|
|
#define REALUTIMES utimes
|
|
|
|
#define REALLUTIMES lutimes
|
|
|
|
#define REALFUTIMES futimes
|
|
|
|
#define REALFHSTAT fhstat
|
|
|
|
#define REALSOCKET socket
|
|
|
|
|
2012-09-03 15:33:35 +04:00
|
|
|
#else /* !NetBSD && !linux */
|
|
|
|
|
|
|
|
#error platform not supported
|
|
|
|
|
|
|
|
#endif /* platform */
|
2011-01-18 22:41:02 +03:00
|
|
|
|
2011-02-07 13:28:18 +03:00
|
|
|
int REALSELECT(int, fd_set *, fd_set *, fd_set *, struct timeval *);
|
|
|
|
int REALPOLLTS(struct pollfd *, nfds_t,
|
2011-01-25 20:37:00 +03:00
|
|
|
const struct timespec *, const sigset_t *);
|
2011-02-08 17:45:35 +03:00
|
|
|
int REALKEVENT(int, const struct kevent *, size_t, struct kevent *, size_t,
|
|
|
|
const struct timespec *);
|
2011-02-07 15:23:05 +03:00
|
|
|
ssize_t REALREAD(int, void *, size_t);
|
2011-02-21 15:51:06 +03:00
|
|
|
ssize_t REALPREAD(int, void *, size_t, off_t);
|
|
|
|
ssize_t REALPWRITE(int, const void *, size_t, off_t);
|
2011-02-17 15:23:58 +03:00
|
|
|
int REALUTIMES(const char *, const struct timeval [2]);
|
|
|
|
int REALLUTIMES(const char *, const struct timeval [2]);
|
|
|
|
int REALFUTIMES(int, const struct timeval [2]);
|
2011-02-17 20:18:08 +03:00
|
|
|
int REALMOUNT(const char *, const char *, int, void *, size_t);
|
2011-03-09 18:03:18 +03:00
|
|
|
int REALGETFH(const char *, void *, size_t *);
|
|
|
|
int REALFHOPEN(const void *, size_t, int);
|
|
|
|
int REALFHSTAT(const void *, size_t, struct stat *);
|
|
|
|
int REALFHSTATVFS1(const void *, size_t, struct statvfs *, int);
|
2012-08-25 22:00:06 +04:00
|
|
|
int REALSOCKET(int, int, int);
|
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
#define S(a) __STRING(a)
|
|
|
|
struct sysnames {
|
|
|
|
enum dualcall scm_callnum;
|
|
|
|
const char *scm_hostname;
|
|
|
|
const char *scm_rumpname;
|
2011-03-10 12:47:32 +03:00
|
|
|
} syscnames[] = {
|
2012-08-25 22:00:06 +04:00
|
|
|
{ DUALCALL_SOCKET, S(REALSOCKET), RSYS_NAME(SOCKET) },
|
2011-01-25 15:18:33 +03:00
|
|
|
{ DUALCALL_ACCEPT, "accept", RSYS_NAME(ACCEPT) },
|
|
|
|
{ DUALCALL_BIND, "bind", RSYS_NAME(BIND) },
|
|
|
|
{ DUALCALL_CONNECT, "connect", RSYS_NAME(CONNECT) },
|
|
|
|
{ DUALCALL_GETPEERNAME, "getpeername", RSYS_NAME(GETPEERNAME) },
|
|
|
|
{ DUALCALL_GETSOCKNAME, "getsockname", RSYS_NAME(GETSOCKNAME) },
|
|
|
|
{ DUALCALL_LISTEN, "listen", RSYS_NAME(LISTEN) },
|
|
|
|
{ DUALCALL_RECVFROM, "recvfrom", RSYS_NAME(RECVFROM) },
|
|
|
|
{ DUALCALL_RECVMSG, "recvmsg", RSYS_NAME(RECVMSG) },
|
|
|
|
{ DUALCALL_SENDTO, "sendto", RSYS_NAME(SENDTO) },
|
|
|
|
{ DUALCALL_SENDMSG, "sendmsg", RSYS_NAME(SENDMSG) },
|
|
|
|
{ DUALCALL_GETSOCKOPT, "getsockopt", RSYS_NAME(GETSOCKOPT) },
|
|
|
|
{ DUALCALL_SETSOCKOPT, "setsockopt", RSYS_NAME(SETSOCKOPT) },
|
|
|
|
{ DUALCALL_SHUTDOWN, "shutdown", RSYS_NAME(SHUTDOWN) },
|
2011-02-07 15:23:05 +03:00
|
|
|
{ DUALCALL_READ, S(REALREAD), RSYS_NAME(READ) },
|
2011-01-25 15:18:33 +03:00
|
|
|
{ DUALCALL_READV, "readv", RSYS_NAME(READV) },
|
2011-02-21 15:51:06 +03:00
|
|
|
{ DUALCALL_PREAD, S(REALPREAD), RSYS_NAME(PREAD) },
|
|
|
|
{ DUALCALL_PREADV, "preadv", RSYS_NAME(PREADV) },
|
2011-01-25 15:18:33 +03:00
|
|
|
{ DUALCALL_WRITE, "write", RSYS_NAME(WRITE) },
|
|
|
|
{ DUALCALL_WRITEV, "writev", RSYS_NAME(WRITEV) },
|
2011-02-21 15:51:06 +03:00
|
|
|
{ DUALCALL_PWRITE, S(REALPWRITE), RSYS_NAME(PWRITE) },
|
|
|
|
{ DUALCALL_PWRITEV, "pwritev", RSYS_NAME(PWRITEV) },
|
2011-01-25 15:18:33 +03:00
|
|
|
{ DUALCALL_IOCTL, "ioctl", RSYS_NAME(IOCTL) },
|
|
|
|
{ DUALCALL_FCNTL, "fcntl", RSYS_NAME(FCNTL) },
|
|
|
|
{ DUALCALL_DUP2, "dup2", RSYS_NAME(DUP2) },
|
|
|
|
{ DUALCALL_CLOSE, "close", RSYS_NAME(CLOSE) },
|
2011-02-07 13:28:18 +03:00
|
|
|
{ DUALCALL_POLLTS, S(REALPOLLTS), RSYS_NAME(POLLTS) },
|
2012-09-03 15:33:35 +04:00
|
|
|
#ifndef __linux__
|
2011-02-17 15:23:58 +03:00
|
|
|
{ DUALCALL_STAT, S(REALSTAT), RSYS_NAME(STAT) },
|
|
|
|
{ DUALCALL_LSTAT, S(REALLSTAT), RSYS_NAME(LSTAT) },
|
|
|
|
{ DUALCALL_FSTAT, S(REALFSTAT), RSYS_NAME(FSTAT) },
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
{ DUALCALL_CHOWN, "chown", RSYS_NAME(CHOWN) },
|
|
|
|
{ DUALCALL_LCHOWN, "lchown", RSYS_NAME(LCHOWN) },
|
|
|
|
{ DUALCALL_FCHOWN, "fchown", RSYS_NAME(FCHOWN) },
|
|
|
|
{ DUALCALL_CHMOD, "chmod", RSYS_NAME(CHMOD) },
|
|
|
|
{ DUALCALL_LCHMOD, "lchmod", RSYS_NAME(LCHMOD) },
|
|
|
|
{ DUALCALL_FCHMOD, "fchmod", RSYS_NAME(FCHMOD) },
|
|
|
|
{ DUALCALL_UTIMES, S(REALUTIMES), RSYS_NAME(UTIMES) },
|
|
|
|
{ DUALCALL_LUTIMES, S(REALLUTIMES), RSYS_NAME(LUTIMES) },
|
|
|
|
{ DUALCALL_FUTIMES, S(REALFUTIMES), RSYS_NAME(FUTIMES) },
|
2015-03-03 03:19:07 +03:00
|
|
|
{ DUALCALL_UTIMENSAT, "utimensat", RSYS_NAME(UTIMENSAT) },
|
2015-03-03 04:24:39 +03:00
|
|
|
{ DUALCALL_FUTIMENS, "futimens", RSYS_NAME(FUTIMENS) },
|
2011-02-17 15:23:58 +03:00
|
|
|
{ DUALCALL_OPEN, "open", RSYS_NAME(OPEN) },
|
|
|
|
{ DUALCALL_CHDIR, "chdir", RSYS_NAME(CHDIR) },
|
|
|
|
{ DUALCALL_FCHDIR, "fchdir", RSYS_NAME(FCHDIR) },
|
2011-02-21 15:55:21 +03:00
|
|
|
{ DUALCALL_LSEEK, "lseek", RSYS_NAME(LSEEK) },
|
2011-02-17 15:23:58 +03:00
|
|
|
{ DUALCALL_UNLINK, "unlink", RSYS_NAME(UNLINK) },
|
|
|
|
{ DUALCALL_SYMLINK, "symlink", RSYS_NAME(SYMLINK) },
|
|
|
|
{ DUALCALL_READLINK, "readlink", RSYS_NAME(READLINK) },
|
2012-08-04 07:56:47 +04:00
|
|
|
{ DUALCALL_LINK, "link", RSYS_NAME(LINK) },
|
2011-02-17 15:23:58 +03:00
|
|
|
{ DUALCALL_RENAME, "rename", RSYS_NAME(RENAME) },
|
|
|
|
{ DUALCALL_MKDIR, "mkdir", RSYS_NAME(MKDIR) },
|
|
|
|
{ DUALCALL_RMDIR, "rmdir", RSYS_NAME(RMDIR) },
|
|
|
|
{ DUALCALL_TRUNCATE, "truncate", RSYS_NAME(TRUNCATE) },
|
|
|
|
{ DUALCALL_FTRUNCATE, "ftruncate", RSYS_NAME(FTRUNCATE) },
|
|
|
|
{ DUALCALL_FSYNC, "fsync", RSYS_NAME(FSYNC) },
|
2011-02-23 18:29:21 +03:00
|
|
|
{ DUALCALL_ACCESS, "access", RSYS_NAME(ACCESS) },
|
2012-09-03 15:33:35 +04:00
|
|
|
|
|
|
|
#ifndef __linux__
|
|
|
|
{ DUALCALL___GETCWD, "__getcwd", RSYS_NAME(__GETCWD) },
|
2012-09-03 16:07:42 +04:00
|
|
|
{ DUALCALL_GETDENTS, S(REALGETDENTS),RSYS_NAME(GETDENTS) },
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __linux__
|
2011-02-25 21:29:00 +03:00
|
|
|
{ DUALCALL_MKNOD, S(REALMKNOD), RSYS_NAME(MKNOD) },
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-03-09 21:06:22 +03:00
|
|
|
{ DUALCALL_GETFH, S(REALGETFH), RSYS_NAME(GETFH) },
|
2012-09-03 15:33:35 +04:00
|
|
|
{ DUALCALL_FHOPEN, S(REALFHOPEN), RSYS_NAME(FHOPEN) },
|
|
|
|
{ DUALCALL_FHSTAT, S(REALFHSTAT), RSYS_NAME(FHSTAT) },
|
2011-03-09 21:06:22 +03:00
|
|
|
{ DUALCALL_FHSTATVFS1, S(REALFHSTATVFS1),RSYS_NAME(FHSTATVFS1) },
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_KQUEUE
|
2012-09-03 15:33:35 +04:00
|
|
|
{ DUALCALL_KEVENT, S(REALKEVENT), RSYS_NAME(KEVENT) },
|
|
|
|
#endif
|
2012-08-25 22:00:06 +04:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
{ DUALCALL___SYSCTL, "__sysctl", RSYS_NAME(__SYSCTL) },
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
{ DUALCALL_NFSSVC, "nfssvc", RSYS_NAME(NFSSVC) },
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
{ DUALCALL_STATVFS1, "statvfs1", RSYS_NAME(STATVFS1) },
|
|
|
|
{ DUALCALL_FSTATVFS1, "fstatvfs1", RSYS_NAME(FSTATVFS1) },
|
|
|
|
{ DUALCALL_GETVFSSTAT, "getvfsstat", RSYS_NAME(GETVFSSTAT) },
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
{ DUALCALL_MOUNT, S(REALMOUNT), RSYS_NAME(MOUNT) },
|
|
|
|
{ DUALCALL_UNMOUNT, "unmount", RSYS_NAME(UNMOUNT) },
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_FSYNC_RANGE
|
2012-08-25 22:00:06 +04:00
|
|
|
{ DUALCALL_FSYNC_RANGE, "fsync_range", RSYS_NAME(FSYNC_RANGE) },
|
|
|
|
#endif
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_CHFLAGS
|
2012-08-25 22:00:06 +04:00
|
|
|
{ DUALCALL_CHFLAGS, "chflags", RSYS_NAME(CHFLAGS) },
|
|
|
|
{ DUALCALL_LCHFLAGS, "lchflags", RSYS_NAME(LCHFLAGS) },
|
|
|
|
{ DUALCALL_FCHFLAGS, "fchflags", RSYS_NAME(FCHFLAGS) },
|
2014-11-04 22:05:17 +03:00
|
|
|
#endif /* HAVE_CHFLAGS */
|
2012-08-25 22:00:06 +04:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE___QUOTACTL
|
2012-02-01 09:34:38 +04:00
|
|
|
{ DUALCALL_QUOTACTL, "__quotactl", RSYS_NAME(__QUOTACTL) },
|
2014-11-04 22:05:17 +03:00
|
|
|
#endif /* HAVE___QUOTACTL */
|
2012-08-25 22:00:06 +04:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
};
|
|
|
|
#undef S
|
|
|
|
|
|
|
|
struct bothsys {
|
|
|
|
void *bs_host;
|
|
|
|
void *bs_rump;
|
|
|
|
} syscalls[DUALCALL__NUM];
|
|
|
|
#define GETSYSCALL(which, name) syscalls[DUALCALL_##name].bs_##which
|
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
static pid_t (*host_fork)(void);
|
|
|
|
static int (*host_daemon)(int, int);
|
|
|
|
static void * (*host_mmap)(void *, size_t, int, int, int, off_t);
|
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
/*
|
|
|
|
* This tracks if our process is in a subdirectory of /rump.
|
|
|
|
* It's preserved over exec.
|
|
|
|
*/
|
|
|
|
static bool pwdinrump;
|
|
|
|
|
|
|
|
enum pathtype { PATH_HOST, PATH_RUMP, PATH_RUMPBLANKET };
|
|
|
|
|
|
|
|
static bool fd_isrump(int);
|
|
|
|
static enum pathtype path_isrump(const char *);
|
2011-02-28 22:57:36 +03:00
|
|
|
|
2011-03-14 18:13:26 +03:00
|
|
|
/* default FD_SETSIZE is 256 ==> default fdoff is 128 */
|
|
|
|
static int hijack_fdoff = FD_SETSIZE/2;
|
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
/*
|
|
|
|
* Maintain a mapping table for the usual dup2 suspects.
|
2011-03-01 13:54:06 +03:00
|
|
|
* Could use atomic ops to operate on dup2vec, but an application
|
|
|
|
* racing there is not well-defined, so don't bother.
|
2011-02-28 22:57:36 +03:00
|
|
|
*/
|
|
|
|
/* note: you cannot change this without editing the env-passing code */
|
|
|
|
#define DUP2HIGH 2
|
|
|
|
static uint32_t dup2vec[DUP2HIGH+1];
|
|
|
|
#define DUP2BIT (1<<31)
|
|
|
|
#define DUP2ALIAS (1<<30)
|
|
|
|
#define DUP2FDMASK ((1<<30)-1)
|
|
|
|
|
|
|
|
static bool
|
|
|
|
isdup2d(int fd)
|
|
|
|
{
|
|
|
|
|
|
|
|
return fd <= DUP2HIGH && fd >= 0 && dup2vec[fd] & DUP2BIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
mapdup2(int hostfd)
|
|
|
|
{
|
|
|
|
|
|
|
|
_DIAGASSERT(isdup2d(hostfd));
|
|
|
|
return dup2vec[hostfd] & DUP2FDMASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
unmapdup2(int rumpfd)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i <= DUP2HIGH; i++) {
|
2011-02-28 23:39:07 +03:00
|
|
|
if (dup2vec[i] & DUP2BIT &&
|
|
|
|
(dup2vec[i] & DUP2FDMASK) == (unsigned)rumpfd)
|
2011-02-28 22:57:36 +03:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setdup2(int hostfd, int rumpfd)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (hostfd > DUP2HIGH) {
|
|
|
|
_DIAGASSERT(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dup2vec[hostfd] = DUP2BIT | DUP2ALIAS | rumpfd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
clrdup2(int hostfd)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (hostfd > DUP2HIGH) {
|
|
|
|
_DIAGASSERT(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dup2vec[hostfd] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
killdup2alias(int rumpfd)
|
|
|
|
{
|
|
|
|
int hostfd;
|
|
|
|
|
|
|
|
if ((hostfd = unmapdup2(rumpfd)) == -1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (dup2vec[hostfd] & DUP2ALIAS) {
|
|
|
|
dup2vec[hostfd] &= ~DUP2ALIAS;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2011-01-25 15:18:33 +03:00
|
|
|
|
|
|
|
//#define DEBUGJACK
|
|
|
|
#ifdef DEBUGJACK
|
|
|
|
#define DPRINTF(x) mydprintf x
|
|
|
|
static void
|
|
|
|
mydprintf(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
if (isdup2d(STDERR_FILENO))
|
2011-01-25 15:18:33 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vfprintf(stderr, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
}
|
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
static const char *
|
|
|
|
whichfd(int fd)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (fd == -1)
|
|
|
|
return "-1";
|
|
|
|
else if (fd_isrump(fd))
|
|
|
|
return "rump";
|
|
|
|
else
|
|
|
|
return "host";
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
whichpath(const char *path)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (path_isrump(path))
|
|
|
|
return "rump";
|
|
|
|
else
|
|
|
|
return "host";
|
|
|
|
}
|
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
#else
|
|
|
|
#define DPRINTF(x)
|
2011-01-18 22:41:02 +03:00
|
|
|
#endif
|
|
|
|
|
2015-03-05 02:31:49 +03:00
|
|
|
#define ATCALL(type, name, rcname, args, proto, vars) \
|
|
|
|
type name args \
|
|
|
|
{ \
|
|
|
|
type (*fun) proto; \
|
|
|
|
int isrump = -1; \
|
|
|
|
\
|
2015-03-05 02:42:26 +03:00
|
|
|
if (fd == AT_FDCWD || *path == '/') { \
|
2015-03-05 02:31:49 +03:00
|
|
|
isrump = path_isrump(path); \
|
|
|
|
} else { \
|
|
|
|
isrump = fd_isrump(fd); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
DPRINTF(("%s -> %d:%s (%s)\n", __STRING(name), \
|
|
|
|
fd, path, isrump ? "rump" : "host")); \
|
|
|
|
\
|
|
|
|
assert(isrump != -1); \
|
|
|
|
if (isrump) { \
|
|
|
|
fun = syscalls[rcname].bs_rump; \
|
|
|
|
if (fd != AT_FDCWD) \
|
|
|
|
fd = fd_host2rump(fd); \
|
|
|
|
path = path_host2rump(path); \
|
|
|
|
} else { \
|
|
|
|
fun = syscalls[rcname].bs_host; \
|
|
|
|
} \
|
|
|
|
return fun vars; \
|
|
|
|
}
|
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
#define FDCALL(type, name, rcname, args, proto, vars) \
|
|
|
|
type name args \
|
|
|
|
{ \
|
|
|
|
type (*fun) proto; \
|
|
|
|
\
|
2011-02-28 22:57:36 +03:00
|
|
|
DPRINTF(("%s -> %d (%s)\n", __STRING(name), fd, whichfd(fd))); \
|
2011-01-25 15:18:33 +03:00
|
|
|
if (fd_isrump(fd)) { \
|
|
|
|
fun = syscalls[rcname].bs_rump; \
|
|
|
|
fd = fd_host2rump(fd); \
|
|
|
|
} else { \
|
|
|
|
fun = syscalls[rcname].bs_host; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
return fun vars; \
|
|
|
|
}
|
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
#define PATHCALL(type, name, rcname, args, proto, vars) \
|
|
|
|
type name args \
|
|
|
|
{ \
|
|
|
|
type (*fun) proto; \
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
enum pathtype pt; \
|
2011-02-17 15:23:58 +03:00
|
|
|
\
|
2011-02-28 22:57:36 +03:00
|
|
|
DPRINTF(("%s -> %s (%s)\n", __STRING(name), path, \
|
|
|
|
whichpath(path))); \
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if ((pt = path_isrump(path)) != PATH_HOST) { \
|
2011-02-17 15:23:58 +03:00
|
|
|
fun = syscalls[rcname].bs_rump; \
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if (pt == PATH_RUMP) \
|
|
|
|
path = path_host2rump(path); \
|
2011-02-17 15:23:58 +03:00
|
|
|
} else { \
|
|
|
|
fun = syscalls[rcname].bs_host; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
return fun vars; \
|
|
|
|
}
|
|
|
|
|
2011-03-09 18:03:18 +03:00
|
|
|
#define VFSCALL(bit, type, name, rcname, args, proto, vars) \
|
|
|
|
type name args \
|
|
|
|
{ \
|
|
|
|
type (*fun) proto; \
|
|
|
|
\
|
|
|
|
DPRINTF(("%s (0x%x, 0x%x)\n", __STRING(name), bit, vfsbits)); \
|
|
|
|
if (vfsbits & bit) { \
|
|
|
|
fun = syscalls[rcname].bs_rump; \
|
|
|
|
} else { \
|
|
|
|
fun = syscalls[rcname].bs_host; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
return fun vars; \
|
|
|
|
}
|
|
|
|
|
2011-02-18 14:41:32 +03:00
|
|
|
/*
|
|
|
|
* These variables are set from the RUMPHIJACK string and control
|
|
|
|
* which operations can product rump kernel file descriptors.
|
|
|
|
* This should be easily extendable for future needs.
|
|
|
|
*/
|
|
|
|
#define RUMPHIJACK_DEFAULT "path=/rump,socket=all:nolocal"
|
|
|
|
static bool rumpsockets[PF_MAX];
|
|
|
|
static const char *rumpprefix;
|
|
|
|
static size_t rumpprefixlen;
|
|
|
|
|
|
|
|
static struct {
|
|
|
|
int pf;
|
|
|
|
const char *name;
|
|
|
|
} socketmap[] = {
|
2011-02-18 17:25:04 +03:00
|
|
|
{ PF_LOCAL, "local" },
|
2011-02-18 14:41:32 +03:00
|
|
|
{ PF_INET, "inet" },
|
2012-08-25 22:00:06 +04:00
|
|
|
#ifdef PF_LINK
|
2011-02-18 14:41:32 +03:00
|
|
|
{ PF_LINK, "link" },
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
2011-02-19 16:07:53 +03:00
|
|
|
#ifdef PF_OROUTE
|
2011-02-19 16:09:40 +03:00
|
|
|
{ PF_OROUTE, "oroute" },
|
|
|
|
#endif
|
2011-02-18 14:41:32 +03:00
|
|
|
{ PF_ROUTE, "route" },
|
|
|
|
{ PF_INET6, "inet6" },
|
2011-02-19 16:07:53 +03:00
|
|
|
#ifdef PF_MPLS
|
|
|
|
{ PF_MPLS, "mpls" },
|
|
|
|
#endif
|
2011-02-18 14:41:32 +03:00
|
|
|
{ -1, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
sockparser(char *buf)
|
|
|
|
{
|
2012-08-25 22:00:06 +04:00
|
|
|
char *p, *l = NULL;
|
2011-02-18 14:41:32 +03:00
|
|
|
bool value;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* if "all" is present, it must be specified first */
|
|
|
|
if (strncmp(buf, "all", strlen("all")) == 0) {
|
2011-02-18 16:04:52 +03:00
|
|
|
for (i = 0; i < (int)__arraycount(rumpsockets); i++) {
|
2011-02-18 14:41:32 +03:00
|
|
|
rumpsockets[i] = true;
|
|
|
|
}
|
|
|
|
buf += strlen("all");
|
|
|
|
if (*buf == ':')
|
|
|
|
buf++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (p = strtok_r(buf, ":", &l); p; p = strtok_r(NULL, ":", &l)) {
|
|
|
|
value = true;
|
|
|
|
if (strncmp(p, "no", strlen("no")) == 0) {
|
|
|
|
value = false;
|
|
|
|
p += strlen("no");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; socketmap[i].name; i++) {
|
|
|
|
if (strcmp(p, socketmap[i].name) == 0) {
|
|
|
|
rumpsockets[socketmap[i].pf] = value;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (socketmap[i].name == NULL) {
|
2011-03-09 18:03:18 +03:00
|
|
|
errx(1, "invalid socket specifier %s", p);
|
2011-02-18 14:41:32 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
pathparser(char *buf)
|
|
|
|
{
|
|
|
|
|
2011-02-19 16:10:35 +03:00
|
|
|
/* sanity-check */
|
2011-02-18 14:41:32 +03:00
|
|
|
if (*buf != '/')
|
|
|
|
errx(1, "hijack path specifier must begin with ``/''");
|
2011-02-19 16:10:35 +03:00
|
|
|
rumpprefixlen = strlen(buf);
|
|
|
|
if (rumpprefixlen < 2)
|
|
|
|
errx(1, "invalid hijack prefix: %s", buf);
|
|
|
|
if (buf[rumpprefixlen-1] == '/' && strspn(buf, "/") != rumpprefixlen)
|
|
|
|
errx(1, "hijack prefix may end in slash only if pure "
|
|
|
|
"slash, gave %s", buf);
|
2011-02-18 14:41:32 +03:00
|
|
|
|
|
|
|
if ((rumpprefix = strdup(buf)) == NULL)
|
|
|
|
err(1, "strdup");
|
|
|
|
rumpprefixlen = strlen(rumpprefix);
|
|
|
|
}
|
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
static struct blanket {
|
|
|
|
const char *pfx;
|
|
|
|
size_t len;
|
|
|
|
} *blanket;
|
|
|
|
static int nblanket;
|
|
|
|
|
|
|
|
static void
|
|
|
|
blanketparser(char *buf)
|
|
|
|
{
|
2012-08-25 22:00:06 +04:00
|
|
|
char *p, *l = NULL;
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (nblanket = 0, p = buf; p; p = strchr(p+1, ':'), nblanket++)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
blanket = malloc(nblanket * sizeof(*blanket));
|
|
|
|
if (blanket == NULL)
|
|
|
|
err(1, "alloc blanket %d", nblanket);
|
|
|
|
|
|
|
|
for (p = strtok_r(buf, ":", &l), i = 0; p;
|
|
|
|
p = strtok_r(NULL, ":", &l), i++) {
|
|
|
|
blanket[i].pfx = strdup(p);
|
|
|
|
if (blanket[i].pfx == NULL)
|
|
|
|
err(1, "strdup blanket");
|
|
|
|
blanket[i].len = strlen(p);
|
|
|
|
|
|
|
|
if (blanket[i].len == 0 || *blanket[i].pfx != '/')
|
|
|
|
errx(1, "invalid blanket specifier %s", p);
|
|
|
|
if (*(blanket[i].pfx + blanket[i].len-1) == '/')
|
|
|
|
errx(1, "invalid blanket specifier %s", p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-09 18:03:18 +03:00
|
|
|
#define VFSBIT_NFSSVC 0x01
|
|
|
|
#define VFSBIT_GETVFSSTAT 0x02
|
|
|
|
#define VFSBIT_FHCALLS 0x04
|
|
|
|
static unsigned vfsbits;
|
|
|
|
|
|
|
|
static struct {
|
|
|
|
int bit;
|
|
|
|
const char *name;
|
|
|
|
} vfscalls[] = {
|
|
|
|
{ VFSBIT_NFSSVC, "nfssvc" },
|
|
|
|
{ VFSBIT_GETVFSSTAT, "getvfsstat" },
|
|
|
|
{ VFSBIT_FHCALLS, "fhcalls" },
|
|
|
|
{ -1, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
vfsparser(char *buf)
|
|
|
|
{
|
2012-08-25 22:00:06 +04:00
|
|
|
char *p, *l = NULL;
|
2011-03-09 18:03:18 +03:00
|
|
|
bool turnon;
|
|
|
|
unsigned int fullmask;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* build the full mask and sanity-check while we're at it */
|
|
|
|
fullmask = 0;
|
|
|
|
for (i = 0; vfscalls[i].name != NULL; i++) {
|
|
|
|
if (fullmask & vfscalls[i].bit)
|
|
|
|
errx(1, "problem exists between vi and chair");
|
|
|
|
fullmask |= vfscalls[i].bit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* if "all" is present, it must be specified first */
|
|
|
|
if (strncmp(buf, "all", strlen("all")) == 0) {
|
|
|
|
vfsbits = fullmask;
|
|
|
|
buf += strlen("all");
|
|
|
|
if (*buf == ':')
|
|
|
|
buf++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (p = strtok_r(buf, ":", &l); p; p = strtok_r(NULL, ":", &l)) {
|
|
|
|
turnon = true;
|
|
|
|
if (strncmp(p, "no", strlen("no")) == 0) {
|
|
|
|
turnon = false;
|
|
|
|
p += strlen("no");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; vfscalls[i].name; i++) {
|
|
|
|
if (strcmp(p, vfscalls[i].name) == 0) {
|
|
|
|
if (turnon)
|
|
|
|
vfsbits |= vfscalls[i].bit;
|
|
|
|
else
|
|
|
|
vfsbits &= ~vfscalls[i].bit;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (vfscalls[i].name == NULL) {
|
|
|
|
errx(1, "invalid vfscall specifier %s", p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool rumpsysctl = false;
|
|
|
|
|
|
|
|
static void
|
|
|
|
sysctlparser(char *buf)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (buf == NULL) {
|
|
|
|
rumpsysctl = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcasecmp(buf, "y") == 0 || strcasecmp(buf, "yes") == 0 ||
|
|
|
|
strcasecmp(buf, "yep") == 0 || strcasecmp(buf, "tottakai") == 0) {
|
|
|
|
rumpsysctl = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (strcasecmp(buf, "n") == 0 || strcasecmp(buf, "no") == 0) {
|
|
|
|
rumpsysctl = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
errx(1, "sysctl value should be y(es)/n(o), gave: %s", buf);
|
|
|
|
}
|
|
|
|
|
2011-03-14 18:13:26 +03:00
|
|
|
static void
|
|
|
|
fdoffparser(char *buf)
|
|
|
|
{
|
|
|
|
unsigned long fdoff;
|
|
|
|
char *ep;
|
|
|
|
|
|
|
|
if (*buf == '-') {
|
|
|
|
errx(1, "fdoff must not be negative");
|
|
|
|
}
|
|
|
|
fdoff = strtoul(buf, &ep, 10);
|
|
|
|
if (*ep != '\0')
|
|
|
|
errx(1, "invalid fdoff specifier \"%s\"", buf);
|
|
|
|
if (fdoff >= INT_MAX/2 || fdoff < 3)
|
|
|
|
errx(1, "fdoff out of range");
|
|
|
|
hijack_fdoff = fdoff;
|
|
|
|
}
|
|
|
|
|
2011-02-18 14:41:32 +03:00
|
|
|
static struct {
|
|
|
|
void (*parsefn)(char *);
|
|
|
|
const char *name;
|
2011-03-09 18:03:18 +03:00
|
|
|
bool needvalues;
|
2011-02-18 14:41:32 +03:00
|
|
|
} hijackparse[] = {
|
2011-03-09 18:03:18 +03:00
|
|
|
{ sockparser, "socket", true },
|
|
|
|
{ pathparser, "path", true },
|
|
|
|
{ blanketparser, "blanket", true },
|
|
|
|
{ vfsparser, "vfs", true },
|
|
|
|
{ sysctlparser, "sysctl", false },
|
2011-03-14 18:15:47 +03:00
|
|
|
{ fdoffparser, "fdoff", true },
|
2011-03-09 18:03:18 +03:00
|
|
|
{ NULL, NULL, false },
|
2011-02-18 14:41:32 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
parsehijack(char *hijack)
|
|
|
|
{
|
|
|
|
char *p, *p2, *l;
|
|
|
|
const char *hijackcopy;
|
2011-03-09 18:03:18 +03:00
|
|
|
bool nop2;
|
2011-02-18 14:41:32 +03:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if ((hijackcopy = strdup(hijack)) == NULL)
|
|
|
|
err(1, "strdup");
|
|
|
|
|
|
|
|
/* disable everything explicitly */
|
|
|
|
for (i = 0; i < PF_MAX; i++)
|
|
|
|
rumpsockets[i] = false;
|
|
|
|
|
|
|
|
for (p = strtok_r(hijack, ",", &l); p; p = strtok_r(NULL, ",", &l)) {
|
2011-03-09 18:03:18 +03:00
|
|
|
nop2 = false;
|
2011-02-18 14:41:32 +03:00
|
|
|
p2 = strchr(p, '=');
|
2011-03-09 18:03:18 +03:00
|
|
|
if (!p2) {
|
|
|
|
nop2 = true;
|
|
|
|
p2 = p + strlen(p);
|
|
|
|
}
|
2011-02-18 14:41:32 +03:00
|
|
|
|
|
|
|
for (i = 0; hijackparse[i].parsefn; i++) {
|
|
|
|
if (strncmp(hijackparse[i].name, p,
|
|
|
|
(size_t)(p2-p)) == 0) {
|
2011-03-09 18:03:18 +03:00
|
|
|
if (nop2 && hijackparse[i].needvalues)
|
|
|
|
errx(1, "invalid hijack specifier: %s",
|
|
|
|
hijackcopy);
|
|
|
|
hijackparse[i].parsefn(nop2 ? NULL : p2+1);
|
2011-02-18 14:41:32 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-03-09 23:48:57 +03:00
|
|
|
|
|
|
|
if (hijackparse[i].parsefn == NULL)
|
|
|
|
errx(1, "invalid hijack specifier name in %s", p);
|
2011-02-18 14:41:32 +03:00
|
|
|
}
|
2011-02-17 15:23:58 +03:00
|
|
|
|
2011-02-18 14:41:32 +03:00
|
|
|
}
|
2011-01-09 22:56:33 +03:00
|
|
|
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
static void __attribute__((constructor))
|
|
|
|
rcinit(void)
|
|
|
|
{
|
2011-02-18 14:41:32 +03:00
|
|
|
char buf[1024];
|
2011-01-25 15:53:45 +03:00
|
|
|
unsigned i, j;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-08 17:19:27 +03:00
|
|
|
host_fork = dlsym(RTLD_NEXT, "fork");
|
2011-02-05 19:57:39 +03:00
|
|
|
host_daemon = dlsym(RTLD_NEXT, "daemon");
|
2011-02-21 16:19:35 +03:00
|
|
|
host_mmap = dlsym(RTLD_NEXT, "mmap");
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
/*
|
|
|
|
* In theory cannot print anything during lookups because
|
|
|
|
* we might not have the call vector set up. so, the errx()
|
|
|
|
* is a bit of a strech, but it might work.
|
|
|
|
*/
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
for (i = 0; i < DUALCALL__NUM; i++) {
|
|
|
|
/* build runtime O(1) access */
|
2011-03-10 12:47:32 +03:00
|
|
|
for (j = 0; j < __arraycount(syscnames); j++) {
|
|
|
|
if (syscnames[j].scm_callnum == i)
|
|
|
|
break;
|
2011-03-09 23:48:57 +03:00
|
|
|
}
|
2011-01-09 13:28:46 +03:00
|
|
|
|
2011-03-10 12:47:32 +03:00
|
|
|
if (j == __arraycount(syscnames))
|
|
|
|
errx(1, "rumphijack error: syscall pos %d missing", i);
|
|
|
|
|
2011-01-27 21:12:19 +03:00
|
|
|
syscalls[i].bs_host = dlsym(RTLD_NEXT,
|
2011-03-10 12:47:32 +03:00
|
|
|
syscnames[j].scm_hostname);
|
|
|
|
if (syscalls[i].bs_host == NULL)
|
|
|
|
errx(1, "hostcall %s not found!",
|
|
|
|
syscnames[j].scm_hostname);
|
2011-01-09 13:28:46 +03:00
|
|
|
|
2011-01-27 21:12:19 +03:00
|
|
|
syscalls[i].bs_rump = dlsym(RTLD_NEXT,
|
2011-03-10 12:47:32 +03:00
|
|
|
syscnames[j].scm_rumpname);
|
2011-01-25 15:18:33 +03:00
|
|
|
if (syscalls[i].bs_rump == NULL)
|
2011-02-27 14:32:12 +03:00
|
|
|
errx(1, "rumpcall %s not found!",
|
2011-03-10 12:47:32 +03:00
|
|
|
syscnames[j].scm_rumpname);
|
2011-01-25 15:18:33 +03:00
|
|
|
}
|
2011-01-09 13:28:46 +03:00
|
|
|
|
2011-01-27 21:05:16 +03:00
|
|
|
if (rumpclient_init() == -1)
|
2011-01-25 15:18:33 +03:00
|
|
|
err(1, "rumpclient init");
|
2011-02-06 18:48:20 +03:00
|
|
|
|
2011-02-18 14:41:32 +03:00
|
|
|
/* check which syscalls we're supposed to hijack */
|
|
|
|
if (getenv_r("RUMPHIJACK", buf, sizeof(buf)) == -1) {
|
|
|
|
strcpy(buf, RUMPHIJACK_DEFAULT);
|
|
|
|
}
|
|
|
|
parsehijack(buf);
|
|
|
|
|
2011-02-06 18:48:20 +03:00
|
|
|
/* set client persistence level */
|
2011-02-16 22:26:58 +03:00
|
|
|
if (getenv_r("RUMPHIJACK_RETRYCONNECT", buf, sizeof(buf)) != -1) {
|
2011-02-06 18:48:20 +03:00
|
|
|
if (strcmp(buf, "die") == 0)
|
|
|
|
rumpclient_setconnretry(RUMPCLIENT_RETRYCONN_DIE);
|
|
|
|
else if (strcmp(buf, "inftime") == 0)
|
|
|
|
rumpclient_setconnretry(RUMPCLIENT_RETRYCONN_INFTIME);
|
|
|
|
else if (strcmp(buf, "once") == 0)
|
|
|
|
rumpclient_setconnretry(RUMPCLIENT_RETRYCONN_ONCE);
|
|
|
|
else {
|
|
|
|
time_t timeout;
|
2011-02-16 22:26:58 +03:00
|
|
|
char *ep;
|
2011-02-06 18:48:20 +03:00
|
|
|
|
2011-02-16 22:26:58 +03:00
|
|
|
timeout = (time_t)strtoll(buf, &ep, 10);
|
|
|
|
if (timeout <= 0 || ep != buf + strlen(buf))
|
|
|
|
errx(1, "RUMPHIJACK_RETRYCONNECT must be "
|
|
|
|
"keyword or integer, got: %s", buf);
|
2011-02-06 18:48:20 +03:00
|
|
|
|
|
|
|
rumpclient_setconnretry(timeout);
|
|
|
|
}
|
|
|
|
}
|
2011-02-14 17:56:23 +03:00
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
if (getenv_r("RUMPHIJACK__DUP2INFO", buf, sizeof(buf)) == 0) {
|
|
|
|
if (sscanf(buf, "%u,%u,%u",
|
|
|
|
&dup2vec[0], &dup2vec[1], &dup2vec[2]) != 3) {
|
|
|
|
warnx("invalid dup2mask: %s", buf);
|
|
|
|
memset(dup2vec, 0, sizeof(dup2vec));
|
|
|
|
}
|
|
|
|
unsetenv("RUMPHIJACK__DUP2INFO");
|
2011-02-17 15:23:58 +03:00
|
|
|
}
|
|
|
|
if (getenv_r("RUMPHIJACK__PWDINRUMP", buf, sizeof(buf)) == 0) {
|
2011-02-18 14:41:32 +03:00
|
|
|
pwdinrump = true;
|
2011-02-17 15:23:58 +03:00
|
|
|
unsetenv("RUMPHIJACK__PWDINRUMP");
|
2011-02-14 17:56:23 +03:00
|
|
|
}
|
2011-01-25 15:18:33 +03:00
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-08 17:19:27 +03:00
|
|
|
static int
|
|
|
|
fd_rump2host(int fd)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (fd == -1)
|
|
|
|
return fd;
|
2011-03-14 18:13:26 +03:00
|
|
|
return fd + hijack_fdoff;
|
2011-02-28 22:57:36 +03:00
|
|
|
}
|
2011-01-08 17:19:27 +03:00
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
static int
|
|
|
|
fd_rump2host_withdup(int fd)
|
|
|
|
{
|
|
|
|
int hfd;
|
2011-01-08 17:19:27 +03:00
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
_DIAGASSERT(fd != -1);
|
|
|
|
hfd = unmapdup2(fd);
|
|
|
|
if (hfd != -1) {
|
|
|
|
_DIAGASSERT(hfd <= DUP2HIGH);
|
|
|
|
return hfd;
|
|
|
|
}
|
|
|
|
return fd_rump2host(fd);
|
2011-01-08 17:19:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
fd_host2rump(int fd)
|
|
|
|
{
|
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
if (!isdup2d(fd))
|
2011-03-14 18:13:26 +03:00
|
|
|
return fd - hijack_fdoff;
|
2011-02-28 22:57:36 +03:00
|
|
|
else
|
|
|
|
return mapdup2(fd);
|
2011-01-08 17:19:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
fd_isrump(int fd)
|
|
|
|
{
|
|
|
|
|
2011-03-14 18:13:26 +03:00
|
|
|
return isdup2d(fd) || fd >= hijack_fdoff;
|
2011-01-08 17:19:27 +03:00
|
|
|
}
|
|
|
|
|
2011-03-14 18:13:26 +03:00
|
|
|
#define assertfd(_fd_) assert(ISDUP2D(_fd_) || (_fd_) >= hijack_fdoff)
|
2011-02-15 16:59:28 +03:00
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
static enum pathtype
|
2011-02-17 15:23:58 +03:00
|
|
|
path_isrump(const char *path)
|
|
|
|
{
|
2011-03-09 00:36:01 +03:00
|
|
|
size_t plen;
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
int i;
|
2011-02-17 15:23:58 +03:00
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if (rumpprefix == NULL && nblanket == 0)
|
|
|
|
return PATH_HOST;
|
2011-02-18 14:41:32 +03:00
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
if (*path == '/') {
|
2011-03-09 00:36:01 +03:00
|
|
|
plen = strlen(path);
|
|
|
|
if (rumpprefix && plen >= rumpprefixlen) {
|
|
|
|
if (strncmp(path, rumpprefix, rumpprefixlen) == 0
|
|
|
|
&& (plen == rumpprefixlen
|
|
|
|
|| *(path + rumpprefixlen) == '/')) {
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
return PATH_RUMP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = 0; i < nblanket; i++) {
|
|
|
|
if (strncmp(path, blanket[i].pfx, blanket[i].len) == 0)
|
|
|
|
return PATH_RUMPBLANKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PATH_HOST;
|
2011-02-17 15:23:58 +03:00
|
|
|
} else {
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
return pwdinrump ? PATH_RUMP : PATH_HOST;
|
2011-02-17 15:23:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *rootpath = "/";
|
|
|
|
static const char *
|
|
|
|
path_host2rump(const char *path)
|
|
|
|
{
|
|
|
|
const char *rv;
|
|
|
|
|
|
|
|
if (*path == '/') {
|
2011-02-18 14:41:32 +03:00
|
|
|
rv = path + rumpprefixlen;
|
2011-02-17 15:23:58 +03:00
|
|
|
if (*rv == '\0')
|
|
|
|
rv = rootpath;
|
|
|
|
} else {
|
|
|
|
rv = path;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2011-02-15 16:59:28 +03:00
|
|
|
static int
|
|
|
|
dodup(int oldd, int minfd)
|
|
|
|
{
|
|
|
|
int (*op_fcntl)(int, int, ...);
|
|
|
|
int newd;
|
|
|
|
int isrump;
|
|
|
|
|
|
|
|
DPRINTF(("dup -> %d (minfd %d)\n", oldd, minfd));
|
|
|
|
if (fd_isrump(oldd)) {
|
|
|
|
op_fcntl = GETSYSCALL(rump, FCNTL);
|
|
|
|
oldd = fd_host2rump(oldd);
|
2011-03-14 18:13:26 +03:00
|
|
|
if (minfd >= hijack_fdoff)
|
|
|
|
minfd -= hijack_fdoff;
|
2011-02-15 16:59:28 +03:00
|
|
|
isrump = 1;
|
|
|
|
} else {
|
|
|
|
op_fcntl = GETSYSCALL(host, FCNTL);
|
|
|
|
isrump = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
newd = op_fcntl(oldd, F_DUPFD, minfd);
|
|
|
|
|
|
|
|
if (isrump)
|
|
|
|
newd = fd_rump2host(newd);
|
|
|
|
DPRINTF(("dup <- %d\n", newd));
|
|
|
|
|
|
|
|
return newd;
|
|
|
|
}
|
2011-01-08 17:19:27 +03:00
|
|
|
|
2011-02-17 18:20:10 +03:00
|
|
|
/*
|
2011-03-14 18:13:26 +03:00
|
|
|
* Check that host fd value does not exceed fdoffset and if necessary
|
|
|
|
* dup the file descriptor so that it doesn't collide with the dup2mask.
|
2011-02-17 18:20:10 +03:00
|
|
|
*/
|
|
|
|
static int
|
2011-03-14 18:13:26 +03:00
|
|
|
fd_host2host(int fd)
|
2011-02-17 18:20:10 +03:00
|
|
|
{
|
|
|
|
int (*op_fcntl)(int, int, ...) = GETSYSCALL(host, FCNTL);
|
|
|
|
int (*op_close)(int) = GETSYSCALL(host, CLOSE);
|
|
|
|
int ofd, i;
|
|
|
|
|
2011-03-14 18:13:26 +03:00
|
|
|
if (fd >= hijack_fdoff) {
|
|
|
|
op_close(fd);
|
|
|
|
errno = ENFILE;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
for (i = 1; isdup2d(fd); i++) {
|
2011-02-17 18:20:10 +03:00
|
|
|
ofd = fd;
|
|
|
|
fd = op_fcntl(ofd, F_DUPFD, i);
|
|
|
|
op_close(ofd);
|
|
|
|
}
|
|
|
|
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
int
|
|
|
|
open(const char *path, int flags, ...)
|
|
|
|
{
|
|
|
|
int (*op_open)(const char *, int, ...);
|
|
|
|
bool isrump;
|
|
|
|
va_list ap;
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
enum pathtype pt;
|
2011-02-17 15:23:58 +03:00
|
|
|
int fd;
|
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
DPRINTF(("open -> %s (%s)\n", path, whichpath(path)));
|
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if ((pt = path_isrump(path)) != PATH_HOST) {
|
|
|
|
if (pt == PATH_RUMP)
|
|
|
|
path = path_host2rump(path);
|
2011-02-17 15:23:58 +03:00
|
|
|
op_open = GETSYSCALL(rump, OPEN);
|
|
|
|
isrump = true;
|
|
|
|
} else {
|
|
|
|
op_open = GETSYSCALL(host, OPEN);
|
|
|
|
isrump = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
va_start(ap, flags);
|
|
|
|
fd = op_open(path, flags, va_arg(ap, mode_t));
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
if (isrump)
|
|
|
|
fd = fd_rump2host(fd);
|
2011-02-17 18:20:10 +03:00
|
|
|
else
|
2011-03-14 18:13:26 +03:00
|
|
|
fd = fd_host2host(fd);
|
2011-02-28 22:57:36 +03:00
|
|
|
|
|
|
|
DPRINTF(("open <- %d (%s)\n", fd, whichfd(fd)));
|
2011-02-17 15:23:58 +03:00
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
chdir(const char *path)
|
|
|
|
{
|
|
|
|
int (*op_chdir)(const char *);
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
enum pathtype pt;
|
2011-02-17 15:23:58 +03:00
|
|
|
int rv;
|
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if ((pt = path_isrump(path)) != PATH_HOST) {
|
2011-02-17 15:23:58 +03:00
|
|
|
op_chdir = GETSYSCALL(rump, CHDIR);
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if (pt == PATH_RUMP)
|
|
|
|
path = path_host2rump(path);
|
2011-02-17 15:23:58 +03:00
|
|
|
} else {
|
|
|
|
op_chdir = GETSYSCALL(host, CHDIR);
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = op_chdir(path);
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if (rv == 0)
|
|
|
|
pwdinrump = pt != PATH_HOST;
|
2011-02-17 15:23:58 +03:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
fchdir(int fd)
|
|
|
|
{
|
|
|
|
int (*op_fchdir)(int);
|
|
|
|
bool isrump;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
if (fd_isrump(fd)) {
|
|
|
|
op_fchdir = GETSYSCALL(rump, FCHDIR);
|
|
|
|
isrump = true;
|
|
|
|
fd = fd_host2rump(fd);
|
|
|
|
} else {
|
|
|
|
op_fchdir = GETSYSCALL(host, FCHDIR);
|
|
|
|
isrump = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = op_fchdir(fd);
|
|
|
|
if (rv == 0) {
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
pwdinrump = isrump;
|
2011-02-17 15:23:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2012-09-03 15:33:35 +04:00
|
|
|
#ifndef __linux__
|
2011-02-19 16:10:35 +03:00
|
|
|
int
|
|
|
|
__getcwd(char *bufp, size_t len)
|
|
|
|
{
|
|
|
|
int (*op___getcwd)(char *, size_t);
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
size_t prefixgap;
|
|
|
|
bool iamslash;
|
2011-02-19 16:10:35 +03:00
|
|
|
int rv;
|
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if (pwdinrump && rumpprefix) {
|
2011-02-19 16:10:35 +03:00
|
|
|
if (rumpprefix[rumpprefixlen-1] == '/')
|
|
|
|
iamslash = true;
|
|
|
|
else
|
|
|
|
iamslash = false;
|
|
|
|
|
|
|
|
if (iamslash)
|
|
|
|
prefixgap = rumpprefixlen - 1; /* ``//+path'' */
|
|
|
|
else
|
|
|
|
prefixgap = rumpprefixlen; /* ``/pfx+/path'' */
|
|
|
|
if (len <= prefixgap) {
|
2011-02-23 18:44:38 +03:00
|
|
|
errno = ERANGE;
|
|
|
|
return -1;
|
2011-02-19 16:10:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
op___getcwd = GETSYSCALL(rump, __GETCWD);
|
|
|
|
rv = op___getcwd(bufp + prefixgap, len - prefixgap);
|
|
|
|
if (rv == -1)
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
/* augment the "/" part only for a non-root path */
|
|
|
|
memcpy(bufp, rumpprefix, rumpprefixlen);
|
|
|
|
|
|
|
|
/* append / only to non-root cwd */
|
|
|
|
if (rv != 2)
|
|
|
|
bufp[prefixgap] = '/';
|
|
|
|
|
|
|
|
/* don't append extra slash in the purely-slash case */
|
|
|
|
if (rv == 2 && !iamslash)
|
|
|
|
bufp[rumpprefixlen] = '\0';
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
} else if (pwdinrump) {
|
|
|
|
/* assume blanket. we can't provide a prefix here */
|
|
|
|
op___getcwd = GETSYSCALL(rump, __GETCWD);
|
|
|
|
rv = op___getcwd(bufp, len);
|
2011-02-19 16:10:35 +03:00
|
|
|
} else {
|
|
|
|
op___getcwd = GETSYSCALL(host, __GETCWD);
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
rv = op___getcwd(bufp, len);
|
2011-02-19 16:10:35 +03:00
|
|
|
}
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
|
|
|
|
return rv;
|
2011-02-19 16:10:35 +03:00
|
|
|
}
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
2011-02-19 16:10:35 +03:00
|
|
|
|
2012-08-04 07:56:47 +04:00
|
|
|
static int
|
|
|
|
moveish(const char *from, const char *to,
|
|
|
|
int (*rump_op)(const char *, const char *),
|
|
|
|
int (*host_op)(const char *, const char *))
|
2011-02-18 17:33:11 +03:00
|
|
|
{
|
2012-08-04 07:56:47 +04:00
|
|
|
int (*op)(const char *, const char *);
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
enum pathtype ptf, ptt;
|
2011-02-18 17:33:11 +03:00
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if ((ptf = path_isrump(from)) != PATH_HOST) {
|
|
|
|
if ((ptt = path_isrump(to)) == PATH_HOST) {
|
2011-02-23 18:44:38 +03:00
|
|
|
errno = EXDEV;
|
|
|
|
return -1;
|
|
|
|
}
|
2011-02-18 17:33:11 +03:00
|
|
|
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if (ptf == PATH_RUMP)
|
|
|
|
from = path_host2rump(from);
|
|
|
|
if (ptt == PATH_RUMP)
|
|
|
|
to = path_host2rump(to);
|
2012-08-04 07:56:47 +04:00
|
|
|
op = rump_op;
|
2011-02-18 17:33:11 +03:00
|
|
|
} else {
|
Add ``blanket''. It acts like path, except that the prefix does
_not_ get removed if the call goes to the rump namespace.
So, now it's possible to use e.g. tcpdump (and most other utilities
which hardcore a /dev pathname) on a rump kernel:
golem> setenv RUMPHIJACK blanket=/dev/bpf
golem> tcpdump -n -i virt0
tcpdump: WARNING: SIOCGIFADDR: virt0: Device not configured
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on virt0, link-type EN10MB (Ethernet), capture size 96 bytes
21:55:38.925596 IP 192.168.2.101 > 204.152.190.12: ICMP echo request, id 47811, seq 0, length 64
21:55:39.095596 IP 204.152.190.12 > 192.168.2.101: ICMP echo reply, id 47811, seq 0, length 64
(if you additionally set socket=all in RUMPHIJACK, tcpdump doesn't
whine about the "not configured" interface)
2011-03-08 23:59:01 +03:00
|
|
|
if (path_isrump(to) != PATH_HOST) {
|
2011-02-23 18:44:38 +03:00
|
|
|
errno = EXDEV;
|
|
|
|
return -1;
|
|
|
|
}
|
2011-02-18 17:44:46 +03:00
|
|
|
|
2012-08-04 07:56:47 +04:00
|
|
|
op = host_op;
|
2011-02-18 17:33:11 +03:00
|
|
|
}
|
|
|
|
|
2012-08-04 07:56:47 +04:00
|
|
|
return op(from, to);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
link(const char *from, const char *to)
|
|
|
|
{
|
|
|
|
return moveish(from, to,
|
|
|
|
GETSYSCALL(rump, LINK), GETSYSCALL(host, LINK));
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
rename(const char *from, const char *to)
|
|
|
|
{
|
|
|
|
return moveish(from, to,
|
|
|
|
GETSYSCALL(rump, RENAME), GETSYSCALL(host, RENAME));
|
2011-02-18 17:33:11 +03:00
|
|
|
}
|
|
|
|
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
int
|
2012-08-25 22:00:06 +04:00
|
|
|
REALSOCKET(int domain, int type, int protocol)
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
2011-01-25 15:18:33 +03:00
|
|
|
int (*op_socket)(int, int, int);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
int fd;
|
2011-02-18 14:41:32 +03:00
|
|
|
bool isrump;
|
2011-01-09 22:56:33 +03:00
|
|
|
|
2011-02-18 14:41:32 +03:00
|
|
|
isrump = domain < PF_MAX && rumpsockets[domain];
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-02-18 14:41:32 +03:00
|
|
|
if (isrump)
|
2011-01-25 15:18:33 +03:00
|
|
|
op_socket = GETSYSCALL(rump, SOCKET);
|
2011-02-18 14:41:32 +03:00
|
|
|
else
|
|
|
|
op_socket = GETSYSCALL(host, SOCKET);
|
2011-01-25 15:18:33 +03:00
|
|
|
fd = op_socket(domain, type, protocol);
|
2011-01-08 17:19:27 +03:00
|
|
|
|
2011-02-18 14:41:32 +03:00
|
|
|
if (isrump)
|
2011-01-09 22:56:33 +03:00
|
|
|
fd = fd_rump2host(fd);
|
2011-02-17 18:20:10 +03:00
|
|
|
else
|
2011-03-14 18:13:26 +03:00
|
|
|
fd = fd_host2host(fd);
|
2011-01-09 22:56:33 +03:00
|
|
|
DPRINTF(("socket <- %d\n", fd));
|
2011-01-08 17:19:27 +03:00
|
|
|
|
2011-01-09 22:56:33 +03:00
|
|
|
return fd;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
accept(int s, struct sockaddr *addr, socklen_t *addrlen)
|
|
|
|
{
|
2011-01-25 15:18:33 +03:00
|
|
|
int (*op_accept)(int, struct sockaddr *, socklen_t *);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
int fd;
|
2011-01-09 22:56:33 +03:00
|
|
|
bool isrump;
|
|
|
|
|
|
|
|
isrump = fd_isrump(s);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-08 17:19:27 +03:00
|
|
|
DPRINTF(("accept -> %d", s));
|
2011-01-09 22:56:33 +03:00
|
|
|
if (isrump) {
|
2011-01-25 15:18:33 +03:00
|
|
|
op_accept = GETSYSCALL(rump, ACCEPT);
|
2011-01-09 22:56:33 +03:00
|
|
|
s = fd_host2rump(s);
|
|
|
|
} else {
|
2011-01-25 15:18:33 +03:00
|
|
|
op_accept = GETSYSCALL(host, ACCEPT);
|
2011-01-09 22:56:33 +03:00
|
|
|
}
|
2011-01-25 15:18:33 +03:00
|
|
|
fd = op_accept(s, addr, addrlen);
|
2011-01-09 22:56:33 +03:00
|
|
|
if (fd != -1 && isrump)
|
|
|
|
fd = fd_rump2host(fd);
|
2011-02-17 18:20:10 +03:00
|
|
|
else
|
2011-03-14 18:13:26 +03:00
|
|
|
fd = fd_host2host(fd);
|
2011-01-09 22:56:33 +03:00
|
|
|
|
|
|
|
DPRINTF((" <- %d\n", fd));
|
2011-01-08 17:19:27 +03:00
|
|
|
|
2011-01-09 22:56:33 +03:00
|
|
|
return fd;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
/*
|
2013-07-22 16:11:03 +04:00
|
|
|
* ioctl() and fcntl() are varargs calls and need special treatment.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Various [Linux] libc's have various signatures for ioctl so we
|
|
|
|
* need to handle the discrepancies. On NetBSD, we use the
|
|
|
|
* one with unsigned long cmd.
|
2011-01-25 15:18:33 +03:00
|
|
|
*/
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
int
|
2013-07-22 16:11:03 +04:00
|
|
|
#ifdef HAVE_IOCTL_CMD_INT
|
|
|
|
ioctl(int fd, int cmd, ...)
|
|
|
|
{
|
|
|
|
int (*op_ioctl)(int, int cmd, ...);
|
|
|
|
#else
|
2011-01-25 15:18:33 +03:00
|
|
|
ioctl(int fd, unsigned long cmd, ...)
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
2011-01-25 15:18:33 +03:00
|
|
|
int (*op_ioctl)(int, unsigned long cmd, ...);
|
2013-07-22 16:11:03 +04:00
|
|
|
#endif
|
2011-01-25 15:18:33 +03:00
|
|
|
va_list ap;
|
|
|
|
int rv;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
DPRINTF(("ioctl -> %d\n", fd));
|
|
|
|
if (fd_isrump(fd)) {
|
|
|
|
fd = fd_host2rump(fd);
|
|
|
|
op_ioctl = GETSYSCALL(rump, IOCTL);
|
2011-01-09 22:56:33 +03:00
|
|
|
} else {
|
2011-01-25 15:18:33 +03:00
|
|
|
op_ioctl = GETSYSCALL(host, IOCTL);
|
2011-01-09 22:56:33 +03:00
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
va_start(ap, cmd);
|
|
|
|
rv = op_ioctl(fd, cmd, va_arg(ap, void *));
|
|
|
|
va_end(ap);
|
|
|
|
return rv;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2011-01-25 15:18:33 +03:00
|
|
|
fcntl(int fd, int cmd, ...)
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
2011-01-25 15:18:33 +03:00
|
|
|
int (*op_fcntl)(int, int, ...);
|
|
|
|
va_list ap;
|
2012-08-25 22:00:06 +04:00
|
|
|
int rv, minfd;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-02-15 16:59:28 +03:00
|
|
|
DPRINTF(("fcntl -> %d (cmd %d)\n", fd, cmd));
|
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
case F_DUPFD:
|
|
|
|
va_start(ap, cmd);
|
|
|
|
minfd = va_arg(ap, int);
|
|
|
|
va_end(ap);
|
|
|
|
return dodup(fd, minfd);
|
|
|
|
|
2012-08-25 22:00:06 +04:00
|
|
|
#ifdef F_CLOSEM
|
|
|
|
case F_CLOSEM: {
|
|
|
|
int maxdup2, i;
|
|
|
|
|
2011-02-15 16:59:28 +03:00
|
|
|
/*
|
|
|
|
* So, if fd < HIJACKOFF, we want to do a host closem.
|
|
|
|
*/
|
|
|
|
|
2011-03-14 18:13:26 +03:00
|
|
|
if (fd < hijack_fdoff) {
|
2011-02-15 16:59:28 +03:00
|
|
|
int closemfd = fd;
|
|
|
|
|
|
|
|
if (rumpclient__closenotify(&closemfd,
|
2011-02-14 17:56:23 +03:00
|
|
|
RUMPCLIENT_CLOSE_FCLOSEM) == -1)
|
|
|
|
return -1;
|
2011-02-15 16:59:28 +03:00
|
|
|
op_fcntl = GETSYSCALL(host, FCNTL);
|
|
|
|
rv = op_fcntl(closemfd, cmd);
|
|
|
|
if (rv)
|
|
|
|
return rv;
|
|
|
|
}
|
2011-01-19 14:27:01 +03:00
|
|
|
|
2011-02-15 16:59:28 +03:00
|
|
|
/*
|
|
|
|
* Additionally, we want to do a rump closem, but only
|
2011-02-28 22:57:36 +03:00
|
|
|
* for the file descriptors not dup2'd.
|
2011-02-15 16:59:28 +03:00
|
|
|
*/
|
|
|
|
|
2014-08-13 03:47:09 +04:00
|
|
|
for (i = 0, maxdup2 = -1; i <= DUP2HIGH; i++) {
|
2011-02-28 23:39:07 +03:00
|
|
|
if (dup2vec[i] & DUP2BIT) {
|
|
|
|
int val;
|
|
|
|
|
|
|
|
val = dup2vec[i] & DUP2FDMASK;
|
|
|
|
maxdup2 = MAX(val, maxdup2);
|
|
|
|
}
|
2011-02-15 16:59:28 +03:00
|
|
|
}
|
|
|
|
|
2011-03-14 18:13:26 +03:00
|
|
|
if (fd >= hijack_fdoff)
|
|
|
|
fd -= hijack_fdoff;
|
2011-02-15 16:59:28 +03:00
|
|
|
else
|
|
|
|
fd = 0;
|
2011-02-28 22:57:36 +03:00
|
|
|
fd = MAX(maxdup2+1, fd);
|
2011-02-15 16:59:28 +03:00
|
|
|
|
|
|
|
/* hmm, maybe we should close rump fd's not within dup2mask? */
|
|
|
|
return rump_sys_fcntl(fd, F_CLOSEM);
|
2012-08-25 22:00:06 +04:00
|
|
|
}
|
|
|
|
#endif /* F_CLOSEM */
|
2011-02-15 16:59:28 +03:00
|
|
|
|
2012-08-25 22:00:06 +04:00
|
|
|
#ifdef F_MAXFD
|
2011-02-15 16:59:28 +03:00
|
|
|
case F_MAXFD:
|
|
|
|
/*
|
|
|
|
* For maxfd, if there's a rump kernel fd, return
|
|
|
|
* it hostified. Otherwise, return host's MAXFD
|
|
|
|
* return value.
|
|
|
|
*/
|
|
|
|
if ((rv = rump_sys_fcntl(fd, F_MAXFD)) != -1) {
|
|
|
|
/*
|
|
|
|
* This might go a little wrong in case
|
|
|
|
* of dup2 to [012], but I'm not sure if
|
|
|
|
* there's a justification for tracking
|
|
|
|
* that info. Consider e.g.
|
|
|
|
* dup2(rumpfd, 2) followed by rump_sys_open()
|
|
|
|
* returning 1. We should return 1+HIJACKOFF,
|
|
|
|
* not 2+HIJACKOFF. However, if [01] is not
|
|
|
|
* open, the correct return value is 2.
|
|
|
|
*/
|
|
|
|
return fd_rump2host(fd);
|
|
|
|
} else {
|
|
|
|
op_fcntl = GETSYSCALL(host, FCNTL);
|
|
|
|
return op_fcntl(fd, F_MAXFD);
|
|
|
|
}
|
|
|
|
/*NOTREACHED*/
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif /* F_MAXFD */
|
2011-02-15 16:59:28 +03:00
|
|
|
|
|
|
|
default:
|
|
|
|
if (fd_isrump(fd)) {
|
|
|
|
fd = fd_host2rump(fd);
|
|
|
|
op_fcntl = GETSYSCALL(rump, FCNTL);
|
|
|
|
} else {
|
|
|
|
op_fcntl = GETSYSCALL(host, FCNTL);
|
|
|
|
}
|
|
|
|
|
|
|
|
va_start(ap, cmd);
|
|
|
|
rv = op_fcntl(fd, cmd, va_arg(ap, void *));
|
|
|
|
va_end(ap);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
/*NOTREACHED*/
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
2011-02-14 17:56:23 +03:00
|
|
|
int
|
|
|
|
close(int fd)
|
|
|
|
{
|
|
|
|
int (*op_close)(int);
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
DPRINTF(("close -> %d\n", fd));
|
|
|
|
if (fd_isrump(fd)) {
|
2011-02-28 22:57:36 +03:00
|
|
|
bool undup2 = false;
|
|
|
|
int ofd;
|
|
|
|
|
|
|
|
if (isdup2d(ofd = fd)) {
|
|
|
|
undup2 = true;
|
|
|
|
}
|
2011-02-14 17:56:23 +03:00
|
|
|
|
2011-02-17 18:20:10 +03:00
|
|
|
fd = fd_host2rump(fd);
|
2011-02-28 22:57:36 +03:00
|
|
|
if (!undup2 && killdup2alias(fd)) {
|
2011-02-17 18:20:10 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:56:23 +03:00
|
|
|
op_close = GETSYSCALL(rump, CLOSE);
|
|
|
|
rv = op_close(fd);
|
2011-02-28 22:57:36 +03:00
|
|
|
if (rv == 0 && undup2) {
|
|
|
|
clrdup2(ofd);
|
|
|
|
}
|
2011-02-14 17:56:23 +03:00
|
|
|
} else {
|
|
|
|
if (rumpclient__closenotify(&fd, RUMPCLIENT_CLOSE_CLOSE) == -1)
|
|
|
|
return -1;
|
|
|
|
op_close = GETSYSCALL(host, CLOSE);
|
|
|
|
rv = op_close(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
/*
|
|
|
|
* write cannot issue a standard debug printf due to recursion
|
|
|
|
*/
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
ssize_t
|
2011-01-25 15:18:33 +03:00
|
|
|
write(int fd, const void *buf, size_t blen)
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
2011-01-25 15:18:33 +03:00
|
|
|
ssize_t (*op_write)(int, const void *, size_t);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
if (fd_isrump(fd)) {
|
|
|
|
fd = fd_host2rump(fd);
|
|
|
|
op_write = GETSYSCALL(rump, WRITE);
|
2011-01-09 22:56:33 +03:00
|
|
|
} else {
|
2011-01-25 15:18:33 +03:00
|
|
|
op_write = GETSYSCALL(host, WRITE);
|
2011-01-09 22:56:33 +03:00
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
return op_write(fd, buf, blen);
|
2011-01-08 17:19:27 +03:00
|
|
|
}
|
|
|
|
|
2012-06-29 17:20:25 +04:00
|
|
|
/*
|
|
|
|
* file descriptor passing
|
|
|
|
*
|
|
|
|
* we intercept sendmsg and recvmsg to convert file descriptors in
|
|
|
|
* control messages. an attempt to send a descriptor from a different kernel
|
|
|
|
* is rejected. (ENOTSUP)
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int
|
|
|
|
msg_convert(struct msghdr *msg, int (*func)(int))
|
|
|
|
{
|
|
|
|
struct cmsghdr *cmsg;
|
|
|
|
|
|
|
|
for (cmsg = CMSG_FIRSTHDR(msg); cmsg != NULL;
|
|
|
|
cmsg = CMSG_NXTHDR(msg, cmsg)) {
|
|
|
|
if (cmsg->cmsg_level == SOL_SOCKET &&
|
|
|
|
cmsg->cmsg_type == SCM_RIGHTS) {
|
|
|
|
int *fdp = (void *)CMSG_DATA(cmsg);
|
|
|
|
const size_t size =
|
|
|
|
cmsg->cmsg_len - __CMSG_ALIGN(sizeof(*cmsg));
|
2012-09-12 14:35:10 +04:00
|
|
|
const int nfds = (int)(size / sizeof(int));
|
2012-06-29 17:20:25 +04:00
|
|
|
const int * const efdp = fdp + nfds;
|
|
|
|
|
|
|
|
while (fdp < efdp) {
|
|
|
|
const int newval = func(*fdp);
|
|
|
|
|
|
|
|
if (newval < 0) {
|
|
|
|
return ENOTSUP;
|
|
|
|
}
|
|
|
|
*fdp = newval;
|
|
|
|
fdp++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t
|
|
|
|
recvmsg(int fd, struct msghdr *msg, int flags)
|
|
|
|
{
|
|
|
|
ssize_t (*op_recvmsg)(int, struct msghdr *, int);
|
|
|
|
ssize_t ret;
|
|
|
|
const bool isrump = fd_isrump(fd);
|
|
|
|
|
|
|
|
if (isrump) {
|
|
|
|
fd = fd_host2rump(fd);
|
|
|
|
op_recvmsg = GETSYSCALL(rump, RECVMSG);
|
|
|
|
} else {
|
|
|
|
op_recvmsg = GETSYSCALL(host, RECVMSG);
|
|
|
|
}
|
|
|
|
ret = op_recvmsg(fd, msg, flags);
|
|
|
|
if (ret == -1) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* convert descriptors in the message.
|
|
|
|
*/
|
|
|
|
if (isrump) {
|
|
|
|
msg_convert(msg, fd_rump2host);
|
|
|
|
} else {
|
|
|
|
msg_convert(msg, fd_host2host);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-09-03 15:33:35 +04:00
|
|
|
ssize_t
|
|
|
|
recv(int fd, void *buf, size_t len, int flags)
|
|
|
|
{
|
|
|
|
|
|
|
|
return recvfrom(fd, buf, len, flags, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t
|
|
|
|
send(int fd, const void *buf, size_t len, int flags)
|
|
|
|
{
|
|
|
|
|
|
|
|
return sendto(fd, buf, len, flags, NULL, 0);
|
|
|
|
}
|
|
|
|
|
2012-06-29 17:20:25 +04:00
|
|
|
static int
|
|
|
|
fd_check_rump(int fd)
|
|
|
|
{
|
|
|
|
|
|
|
|
return fd_isrump(fd) ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
fd_check_host(int fd)
|
|
|
|
{
|
|
|
|
|
|
|
|
return !fd_isrump(fd) ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t
|
|
|
|
sendmsg(int fd, const struct msghdr *msg, int flags)
|
|
|
|
{
|
|
|
|
ssize_t (*op_sendmsg)(int, const struct msghdr *, int);
|
|
|
|
const bool isrump = fd_isrump(fd);
|
|
|
|
int error;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* reject descriptors from a different kernel.
|
|
|
|
*/
|
|
|
|
error = msg_convert(__UNCONST(msg),
|
|
|
|
isrump ? fd_check_rump: fd_check_host);
|
|
|
|
if (error != 0) {
|
|
|
|
errno = error;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* convert descriptors in the message to raw values.
|
|
|
|
*/
|
|
|
|
if (isrump) {
|
|
|
|
fd = fd_host2rump(fd);
|
|
|
|
/*
|
|
|
|
* XXX we directly modify the given message assuming:
|
|
|
|
* - cmsg is writable (typically on caller's stack)
|
|
|
|
* - caller don't care cmsg's contents after calling sendmsg.
|
|
|
|
* (thus no need to restore values)
|
|
|
|
*
|
|
|
|
* it's safer to copy and modify instead.
|
|
|
|
*/
|
|
|
|
msg_convert(__UNCONST(msg), fd_host2rump);
|
|
|
|
op_sendmsg = GETSYSCALL(rump, SENDMSG);
|
|
|
|
} else {
|
|
|
|
op_sendmsg = GETSYSCALL(host, SENDMSG);
|
|
|
|
}
|
|
|
|
return op_sendmsg(fd, msg, flags);
|
|
|
|
}
|
|
|
|
|
2011-01-08 17:19:27 +03:00
|
|
|
/*
|
|
|
|
* dup2 is special. we allow dup2 of a rump kernel fd to 0-2 since
|
|
|
|
* many programs do that. dup2 of a rump kernel fd to another value
|
|
|
|
* not >= fdoff is an error.
|
|
|
|
*
|
|
|
|
* Note: cannot rump2host newd, because it is often hardcoded.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
dup2(int oldd, int newd)
|
|
|
|
{
|
2011-01-25 15:18:33 +03:00
|
|
|
int (*host_dup2)(int, int);
|
2011-01-08 17:19:27 +03:00
|
|
|
int rv;
|
|
|
|
|
|
|
|
DPRINTF(("dup2 -> %d (o) -> %d (n)\n", oldd, newd));
|
|
|
|
|
|
|
|
if (fd_isrump(oldd)) {
|
2011-02-28 22:57:36 +03:00
|
|
|
int (*op_close)(int) = GETSYSCALL(host, CLOSE);
|
|
|
|
|
|
|
|
/* only allow fd 0-2 for cross-kernel dup */
|
|
|
|
if (!(newd >= 0 && newd <= 2 && !fd_isrump(newd))) {
|
2011-02-23 18:44:38 +03:00
|
|
|
errno = EBADF;
|
|
|
|
return -1;
|
|
|
|
}
|
2011-02-28 22:57:36 +03:00
|
|
|
|
|
|
|
/* regular dup2? */
|
|
|
|
if (fd_isrump(newd)) {
|
|
|
|
newd = fd_host2rump(newd);
|
|
|
|
rv = rump_sys_dup2(oldd, newd);
|
|
|
|
return fd_rump2host(rv);
|
2011-02-17 18:20:10 +03:00
|
|
|
}
|
2011-02-28 22:57:36 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* dup2 rump => host? just establish an
|
|
|
|
* entry in the mapping table.
|
|
|
|
*/
|
|
|
|
op_close(newd);
|
|
|
|
setdup2(newd, fd_host2rump(oldd));
|
|
|
|
rv = 0;
|
2011-01-08 17:19:27 +03:00
|
|
|
} else {
|
2011-01-25 15:18:33 +03:00
|
|
|
host_dup2 = syscalls[DUALCALL_DUP2].bs_host;
|
2011-02-14 17:56:23 +03:00
|
|
|
if (rumpclient__closenotify(&newd, RUMPCLIENT_CLOSE_DUP2) == -1)
|
|
|
|
return -1;
|
2011-01-18 14:04:10 +03:00
|
|
|
rv = host_dup2(oldd, newd);
|
2011-01-08 17:19:27 +03:00
|
|
|
}
|
2011-01-18 14:04:10 +03:00
|
|
|
|
|
|
|
return rv;
|
2011-01-08 17:19:27 +03:00
|
|
|
}
|
|
|
|
|
2011-02-08 17:45:35 +03:00
|
|
|
int
|
|
|
|
dup(int oldd)
|
|
|
|
{
|
|
|
|
|
2011-02-15 16:59:28 +03:00
|
|
|
return dodup(oldd, 0);
|
2011-02-08 17:45:35 +03:00
|
|
|
}
|
|
|
|
|
2011-01-08 17:19:27 +03:00
|
|
|
pid_t
|
2012-06-26 02:32:43 +04:00
|
|
|
fork(void)
|
2011-01-08 17:19:27 +03:00
|
|
|
{
|
|
|
|
pid_t rv;
|
|
|
|
|
|
|
|
DPRINTF(("fork\n"));
|
|
|
|
|
2011-02-16 20:56:46 +03:00
|
|
|
rv = rumpclient__dofork(host_fork);
|
2011-01-08 17:19:27 +03:00
|
|
|
|
|
|
|
DPRINTF(("fork returns %d\n", rv));
|
|
|
|
return rv;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2012-08-25 22:00:06 +04:00
|
|
|
#ifdef VFORK
|
2011-02-16 20:56:46 +03:00
|
|
|
/* we do not have the luxury of not requiring a stackframe */
|
2015-04-11 15:54:41 +03:00
|
|
|
#define __strong_alias_macro(m, f) __strong_alias(m, f)
|
|
|
|
__strong_alias_macro(VFORK,fork);
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-02-05 19:57:39 +03:00
|
|
|
int
|
|
|
|
daemon(int nochdir, int noclose)
|
|
|
|
{
|
|
|
|
struct rumpclient_fork *rf;
|
|
|
|
|
|
|
|
if ((rf = rumpclient_prefork()) == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (host_daemon(nochdir, noclose) == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (rumpclient_fork_init(rf) == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:56:23 +03:00
|
|
|
int
|
2011-02-16 18:33:46 +03:00
|
|
|
execve(const char *path, char *const argv[], char *const envp[])
|
2011-02-14 17:56:23 +03:00
|
|
|
{
|
|
|
|
char buf[128];
|
|
|
|
char *dup2str;
|
2011-02-18 14:41:32 +03:00
|
|
|
const char *pwdinrumpstr;
|
2011-02-16 18:33:46 +03:00
|
|
|
char **newenv;
|
|
|
|
size_t nelem;
|
|
|
|
int rv, sverrno;
|
2011-02-28 22:57:36 +03:00
|
|
|
int bonus = 2, i = 0;
|
2011-02-17 15:23:58 +03:00
|
|
|
|
2011-02-28 22:57:36 +03:00
|
|
|
snprintf(buf, sizeof(buf), "RUMPHIJACK__DUP2INFO=%u,%u,%u",
|
|
|
|
dup2vec[0], dup2vec[1], dup2vec[2]);
|
|
|
|
dup2str = strdup(buf);
|
|
|
|
if (dup2str == NULL) {
|
|
|
|
errno = ENOMEM;
|
|
|
|
return -1;
|
2011-02-17 15:23:58 +03:00
|
|
|
}
|
2011-02-14 17:56:23 +03:00
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
if (pwdinrump) {
|
2011-02-18 14:41:32 +03:00
|
|
|
pwdinrumpstr = "RUMPHIJACK__PWDINRUMP=true";
|
2011-02-17 15:23:58 +03:00
|
|
|
bonus++;
|
|
|
|
} else {
|
|
|
|
pwdinrumpstr = NULL;
|
|
|
|
}
|
2011-02-14 17:56:23 +03:00
|
|
|
|
2011-02-16 18:33:46 +03:00
|
|
|
for (nelem = 0; envp && envp[nelem]; nelem++)
|
|
|
|
continue;
|
2011-02-28 22:57:36 +03:00
|
|
|
newenv = malloc(sizeof(*newenv) * (nelem+bonus));
|
2011-02-16 18:33:46 +03:00
|
|
|
if (newenv == NULL) {
|
2011-02-14 17:56:23 +03:00
|
|
|
free(dup2str);
|
2011-02-23 18:44:38 +03:00
|
|
|
errno = ENOMEM;
|
|
|
|
return -1;
|
2011-02-14 17:56:23 +03:00
|
|
|
}
|
2011-02-16 18:33:46 +03:00
|
|
|
memcpy(newenv, envp, nelem*sizeof(*newenv));
|
2011-02-28 22:57:36 +03:00
|
|
|
newenv[nelem+i] = dup2str;
|
|
|
|
i++;
|
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
if (pwdinrumpstr) {
|
2011-02-18 14:41:32 +03:00
|
|
|
newenv[nelem+i] = __UNCONST(pwdinrumpstr);
|
2011-02-17 15:23:58 +03:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
newenv[nelem+i] = NULL;
|
|
|
|
_DIAGASSERT(i < bonus);
|
2011-02-16 18:33:46 +03:00
|
|
|
|
|
|
|
rv = rumpclient_exec(path, argv, newenv);
|
|
|
|
|
|
|
|
_DIAGASSERT(rv != 0);
|
|
|
|
sverrno = errno;
|
|
|
|
free(newenv);
|
|
|
|
free(dup2str);
|
|
|
|
errno = sverrno;
|
2011-02-14 17:56:23 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
/*
|
2011-01-25 15:18:33 +03:00
|
|
|
* select is done by calling poll.
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
*/
|
|
|
|
int
|
2011-02-07 13:28:18 +03:00
|
|
|
REALSELECT(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds,
|
2011-01-09 00:30:24 +03:00
|
|
|
struct timeval *timeout)
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
2011-01-09 00:30:24 +03:00
|
|
|
struct pollfd *pfds;
|
|
|
|
struct timespec ts, *tsp = NULL;
|
2011-01-25 15:53:45 +03:00
|
|
|
nfds_t realnfds;
|
|
|
|
int i, j;
|
2011-01-09 00:30:24 +03:00
|
|
|
int rv, incr;
|
|
|
|
|
2012-10-16 16:56:10 +04:00
|
|
|
DPRINTF(("select %d %p %p %p %p\n", nfds,
|
|
|
|
readfds, writefds, exceptfds, timeout));
|
2011-01-09 22:56:33 +03:00
|
|
|
|
2011-01-09 00:30:24 +03:00
|
|
|
/*
|
|
|
|
* Well, first we must scan the fds to figure out how many
|
|
|
|
* fds there really are. This is because up to and including
|
2011-01-25 15:18:33 +03:00
|
|
|
* nb5 poll() silently refuses nfds > process_maxopen_fds.
|
2011-01-09 00:30:24 +03:00
|
|
|
* Seems to be fixed in current, thank the maker.
|
|
|
|
* god damn cluster...bomb.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for (i = 0, realnfds = 0; i < nfds; i++) {
|
|
|
|
if (readfds && FD_ISSET(i, readfds)) {
|
|
|
|
realnfds++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (writefds && FD_ISSET(i, writefds)) {
|
|
|
|
realnfds++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (exceptfds && FD_ISSET(i, exceptfds)) {
|
|
|
|
realnfds++;
|
|
|
|
continue;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-09 17:15:06 +03:00
|
|
|
if (realnfds) {
|
2011-02-12 13:25:46 +03:00
|
|
|
pfds = calloc(realnfds, sizeof(*pfds));
|
2011-01-09 17:15:06 +03:00
|
|
|
if (!pfds)
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
pfds = NULL;
|
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-09 00:30:24 +03:00
|
|
|
for (i = 0, j = 0; i < nfds; i++) {
|
|
|
|
incr = 0;
|
|
|
|
if (readfds && FD_ISSET(i, readfds)) {
|
|
|
|
pfds[j].fd = i;
|
|
|
|
pfds[j].events |= POLLIN;
|
|
|
|
incr=1;
|
|
|
|
}
|
|
|
|
if (writefds && FD_ISSET(i, writefds)) {
|
|
|
|
pfds[j].fd = i;
|
|
|
|
pfds[j].events |= POLLOUT;
|
|
|
|
incr=1;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-01-09 00:30:24 +03:00
|
|
|
if (exceptfds && FD_ISSET(i, exceptfds)) {
|
|
|
|
pfds[j].fd = i;
|
|
|
|
pfds[j].events |= POLLHUP|POLLERR;
|
|
|
|
incr=1;
|
|
|
|
}
|
|
|
|
if (incr)
|
|
|
|
j++;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-02-11 17:02:12 +03:00
|
|
|
assert(j == (int)realnfds);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-01-09 00:30:24 +03:00
|
|
|
if (timeout) {
|
|
|
|
TIMEVAL_TO_TIMESPEC(timeout, &ts);
|
|
|
|
tsp = &ts;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-02-07 13:28:18 +03:00
|
|
|
rv = REALPOLLTS(pfds, realnfds, tsp, NULL);
|
2011-02-11 15:46:41 +03:00
|
|
|
/*
|
|
|
|
* "If select() returns with an error the descriptor sets
|
|
|
|
* will be unmodified"
|
|
|
|
*/
|
|
|
|
if (rv < 0)
|
2011-01-09 00:30:24 +03:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
/*
|
2011-02-11 15:46:41 +03:00
|
|
|
* zero out results (can't use FD_ZERO for the
|
|
|
|
* obvious select-me-not reason). whee.
|
|
|
|
*
|
|
|
|
* We do this here since some software ignores the return
|
|
|
|
* value of select, and hence if the timeout expires, it may
|
|
|
|
* assume all input descriptors have activity.
|
2011-01-09 00:30:24 +03:00
|
|
|
*/
|
|
|
|
for (i = 0; i < nfds; i++) {
|
|
|
|
if (readfds)
|
|
|
|
FD_CLR(i, readfds);
|
|
|
|
if (writefds)
|
|
|
|
FD_CLR(i, writefds);
|
|
|
|
if (exceptfds)
|
|
|
|
FD_CLR(i, exceptfds);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-02-11 15:46:41 +03:00
|
|
|
if (rv == 0)
|
|
|
|
goto out;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-02-11 15:46:41 +03:00
|
|
|
/*
|
|
|
|
* We have >0 fds with activity. Harvest the results.
|
|
|
|
*/
|
2011-01-25 15:53:45 +03:00
|
|
|
for (i = 0; i < (int)realnfds; i++) {
|
2011-01-09 00:30:24 +03:00
|
|
|
if (readfds) {
|
|
|
|
if (pfds[i].revents & POLLIN) {
|
|
|
|
FD_SET(pfds[i].fd, readfds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (writefds) {
|
|
|
|
if (pfds[i].revents & POLLOUT) {
|
|
|
|
FD_SET(pfds[i].fd, writefds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (exceptfds) {
|
|
|
|
if (pfds[i].revents & (POLLHUP|POLLERR)) {
|
|
|
|
FD_SET(pfds[i].fd, exceptfds);
|
|
|
|
}
|
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-01-09 00:30:24 +03:00
|
|
|
|
|
|
|
out:
|
|
|
|
free(pfds);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
checkpoll(struct pollfd *fds, nfds_t nfds, int *hostcall, int *rumpcall)
|
|
|
|
{
|
|
|
|
nfds_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < nfds; i++) {
|
2011-01-18 17:51:14 +03:00
|
|
|
if (fds[i].fd == -1)
|
|
|
|
continue;
|
|
|
|
|
2011-01-08 17:19:27 +03:00
|
|
|
if (fd_isrump(fds[i].fd))
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
(*rumpcall)++;
|
2011-01-08 17:19:27 +03:00
|
|
|
else
|
|
|
|
(*hostcall)++;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-01-08 17:19:27 +03:00
|
|
|
adjustpoll(struct pollfd *fds, nfds_t nfds, int (*fdadj)(int))
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
|
|
|
nfds_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < nfds; i++) {
|
2011-01-08 17:19:27 +03:00
|
|
|
fds[i].fd = fdadj(fds[i].fd);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* poll is easy as long as the call comes in the fds only in one
|
|
|
|
* kernel. otherwise its quite tricky...
|
|
|
|
*/
|
|
|
|
struct pollarg {
|
|
|
|
struct pollfd *pfds;
|
|
|
|
nfds_t nfds;
|
2011-01-08 21:11:46 +03:00
|
|
|
const struct timespec *ts;
|
|
|
|
const sigset_t *sigmask;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
int pipefd;
|
|
|
|
int errnum;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void *
|
|
|
|
hostpoll(void *arg)
|
|
|
|
{
|
2011-01-25 15:18:33 +03:00
|
|
|
int (*op_pollts)(struct pollfd *, nfds_t, const struct timespec *,
|
|
|
|
const sigset_t *);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
struct pollarg *parg = arg;
|
|
|
|
intptr_t rv;
|
|
|
|
|
2011-02-08 22:12:54 +03:00
|
|
|
op_pollts = GETSYSCALL(host, POLLTS);
|
2011-01-25 15:18:33 +03:00
|
|
|
rv = op_pollts(parg->pfds, parg->nfds, parg->ts, parg->sigmask);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
if (rv == -1)
|
|
|
|
parg->errnum = errno;
|
|
|
|
rump_sys_write(parg->pipefd, &rv, sizeof(rv));
|
|
|
|
|
2012-04-18 14:37:37 +04:00
|
|
|
return (void *)rv;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2011-02-07 13:28:18 +03:00
|
|
|
REALPOLLTS(struct pollfd *fds, nfds_t nfds, const struct timespec *ts,
|
2011-01-08 21:11:46 +03:00
|
|
|
const sigset_t *sigmask)
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
2011-01-08 21:11:46 +03:00
|
|
|
int (*op_pollts)(struct pollfd *, nfds_t, const struct timespec *,
|
|
|
|
const sigset_t *);
|
2011-01-25 15:18:33 +03:00
|
|
|
int (*host_close)(int);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
int hostcall = 0, rumpcall = 0;
|
|
|
|
pthread_t pt;
|
|
|
|
nfds_t i;
|
|
|
|
int rv;
|
|
|
|
|
2012-10-16 16:56:10 +04:00
|
|
|
DPRINTF(("poll %p %d %p %p\n", fds, (int)nfds, ts, sigmask));
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
checkpoll(fds, nfds, &hostcall, &rumpcall);
|
|
|
|
|
|
|
|
if (hostcall && rumpcall) {
|
|
|
|
struct pollfd *pfd_host = NULL, *pfd_rump = NULL;
|
|
|
|
int rpipe[2] = {-1,-1}, hpipe[2] = {-1,-1};
|
|
|
|
struct pollarg parg;
|
2012-04-18 14:37:37 +04:00
|
|
|
void *trv_val;
|
2012-09-12 14:35:10 +04:00
|
|
|
int sverrno = 0, rv_rump, rv_host, errno_rump, errno_host;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* ok, this is where it gets tricky. We must support
|
|
|
|
* this since it's a very common operation in certain
|
|
|
|
* types of software (telnet, netcat, etc). We allocate
|
|
|
|
* two vectors and run two poll commands in separate
|
|
|
|
* threads. Whichever returns first "wins" and the
|
|
|
|
* other kernel's fds won't show activity.
|
|
|
|
*/
|
|
|
|
rv = -1;
|
|
|
|
|
|
|
|
/* allocate full vector for O(n) joining after call */
|
|
|
|
pfd_host = malloc(sizeof(*pfd_host)*(nfds+1));
|
|
|
|
if (!pfd_host)
|
|
|
|
goto out;
|
|
|
|
pfd_rump = malloc(sizeof(*pfd_rump)*(nfds+1));
|
|
|
|
if (!pfd_rump) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2011-02-21 02:47:04 +03:00
|
|
|
/*
|
|
|
|
* then, open two pipes, one for notifications
|
|
|
|
* to each kernel.
|
2011-03-01 13:47:29 +03:00
|
|
|
*
|
|
|
|
* At least the rump pipe should probably be
|
|
|
|
* cached, along with the helper threads. This
|
|
|
|
* should give a microbenchmark improvement (haven't
|
|
|
|
* experienced a macro-level problem yet, though).
|
2011-02-21 02:47:04 +03:00
|
|
|
*/
|
|
|
|
if ((rv = rump_sys_pipe(rpipe)) == -1) {
|
|
|
|
sverrno = errno;
|
|
|
|
}
|
|
|
|
if (rv == 0 && (rv = pipe(hpipe)) == -1) {
|
|
|
|
sverrno = errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* split vectors (or signal errors) */
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
for (i = 0; i < nfds; i++) {
|
2011-02-21 02:47:04 +03:00
|
|
|
int fd;
|
|
|
|
|
|
|
|
fds[i].revents = 0;
|
2011-01-08 21:11:46 +03:00
|
|
|
if (fds[i].fd == -1) {
|
|
|
|
pfd_host[i].fd = -1;
|
|
|
|
pfd_rump[i].fd = -1;
|
|
|
|
} else if (fd_isrump(fds[i].fd)) {
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
pfd_host[i].fd = -1;
|
2011-02-21 02:47:04 +03:00
|
|
|
fd = fd_host2rump(fds[i].fd);
|
|
|
|
if (fd == rpipe[0] || fd == rpipe[1]) {
|
|
|
|
fds[i].revents = POLLNVAL;
|
|
|
|
if (rv != -1)
|
|
|
|
rv++;
|
|
|
|
}
|
|
|
|
pfd_rump[i].fd = fd;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
pfd_rump[i].events = fds[i].events;
|
2011-01-08 17:19:27 +03:00
|
|
|
} else {
|
|
|
|
pfd_rump[i].fd = -1;
|
2011-02-21 02:47:04 +03:00
|
|
|
fd = fds[i].fd;
|
|
|
|
if (fd == hpipe[0] || fd == hpipe[1]) {
|
|
|
|
fds[i].revents = POLLNVAL;
|
|
|
|
if (rv != -1)
|
|
|
|
rv++;
|
|
|
|
}
|
|
|
|
pfd_host[i].fd = fd;
|
2011-01-08 17:19:27 +03:00
|
|
|
pfd_host[i].events = fds[i].events;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-02-14 17:56:23 +03:00
|
|
|
pfd_rump[i].revents = pfd_host[i].revents = 0;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-02-21 02:47:04 +03:00
|
|
|
if (rv) {
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
goto out;
|
2011-02-21 02:47:04 +03:00
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
|
|
|
pfd_host[nfds].fd = hpipe[0];
|
|
|
|
pfd_host[nfds].events = POLLIN;
|
|
|
|
pfd_rump[nfds].fd = rpipe[0];
|
|
|
|
pfd_rump[nfds].events = POLLIN;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* then, create a thread to do host part and meanwhile
|
|
|
|
* do rump kernel part right here
|
|
|
|
*/
|
|
|
|
|
|
|
|
parg.pfds = pfd_host;
|
|
|
|
parg.nfds = nfds+1;
|
2011-01-08 21:11:46 +03:00
|
|
|
parg.ts = ts;
|
|
|
|
parg.sigmask = sigmask;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
parg.pipefd = rpipe[1];
|
|
|
|
pthread_create(&pt, NULL, hostpoll, &parg);
|
|
|
|
|
2011-02-08 22:12:54 +03:00
|
|
|
op_pollts = GETSYSCALL(rump, POLLTS);
|
2012-09-12 14:35:10 +04:00
|
|
|
rv_rump = op_pollts(pfd_rump, nfds+1, ts, NULL);
|
|
|
|
errno_rump = errno;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
write(hpipe[1], &rv, sizeof(rv));
|
2012-04-18 14:37:37 +04:00
|
|
|
pthread_join(pt, &trv_val);
|
2012-09-12 14:35:10 +04:00
|
|
|
rv_host = (int)(intptr_t)trv_val;
|
|
|
|
errno_host = parg.errnum;
|
|
|
|
|
|
|
|
/* strip cross-thread notification from real results */
|
2014-07-21 18:23:43 +04:00
|
|
|
if (rv_host > 0 && pfd_host[nfds].revents & POLLIN) {
|
2012-09-12 14:35:10 +04:00
|
|
|
rv_host--;
|
|
|
|
}
|
2014-07-21 18:23:43 +04:00
|
|
|
if (rv_rump > 0 && pfd_rump[nfds].revents & POLLIN) {
|
2012-09-12 14:35:10 +04:00
|
|
|
rv_rump--;
|
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2012-09-12 14:35:10 +04:00
|
|
|
/* then merge the results into what's reported to the caller */
|
|
|
|
if (rv_rump > 0 || rv_host > 0) {
|
|
|
|
/* SUCCESS */
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2012-09-12 14:35:10 +04:00
|
|
|
rv = 0;
|
|
|
|
if (rv_rump > 0) {
|
|
|
|
for (i = 0; i < nfds; i++) {
|
|
|
|
if (pfd_rump[i].fd != -1)
|
|
|
|
fds[i].revents
|
|
|
|
= pfd_rump[i].revents;
|
|
|
|
}
|
|
|
|
rv += rv_rump;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2012-09-12 14:35:10 +04:00
|
|
|
if (rv_host > 0) {
|
|
|
|
for (i = 0; i < nfds; i++) {
|
|
|
|
if (pfd_host[i].fd != -1)
|
|
|
|
fds[i].revents
|
|
|
|
= pfd_host[i].revents;
|
|
|
|
}
|
|
|
|
rv += rv_host;
|
|
|
|
}
|
|
|
|
assert(rv > 0);
|
|
|
|
sverrno = 0;
|
|
|
|
} else if (rv_rump == -1 || rv_host == -1) {
|
|
|
|
/* ERROR */
|
|
|
|
|
|
|
|
/* just pick one kernel at "random" */
|
|
|
|
rv = -1;
|
|
|
|
if (rv_host == -1) {
|
|
|
|
sverrno = errno_host;
|
|
|
|
} else if (rv_rump == -1) {
|
|
|
|
sverrno = errno_rump;
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
} else {
|
2012-09-12 14:35:10 +04:00
|
|
|
/* TIMEOUT */
|
|
|
|
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
rv = 0;
|
2012-09-12 14:35:10 +04:00
|
|
|
assert(rv_rump == 0 && rv_host == 0);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
2011-02-08 22:12:54 +03:00
|
|
|
host_close = GETSYSCALL(host, CLOSE);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
if (rpipe[0] != -1)
|
|
|
|
rump_sys_close(rpipe[0]);
|
|
|
|
if (rpipe[1] != -1)
|
|
|
|
rump_sys_close(rpipe[1]);
|
|
|
|
if (hpipe[0] != -1)
|
2011-01-17 19:30:09 +03:00
|
|
|
host_close(hpipe[0]);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
if (hpipe[1] != -1)
|
2011-01-17 19:30:09 +03:00
|
|
|
host_close(hpipe[1]);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
free(pfd_host);
|
|
|
|
free(pfd_rump);
|
|
|
|
errno = sverrno;
|
|
|
|
} else {
|
|
|
|
if (hostcall) {
|
2011-02-08 22:12:54 +03:00
|
|
|
op_pollts = GETSYSCALL(host, POLLTS);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
} else {
|
2011-02-08 22:12:54 +03:00
|
|
|
op_pollts = GETSYSCALL(rump, POLLTS);
|
2011-01-08 17:19:27 +03:00
|
|
|
adjustpoll(fds, nfds, fd_host2rump);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
2011-01-08 21:11:46 +03:00
|
|
|
rv = op_pollts(fds, nfds, ts, sigmask);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
if (rumpcall)
|
2011-02-28 22:57:36 +03:00
|
|
|
adjustpoll(fds, nfds, fd_rump2host_withdup);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2011-02-05 15:38:19 +03:00
|
|
|
poll(struct pollfd *fds, nfds_t nfds, int timeout)
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
{
|
2011-01-08 21:11:46 +03:00
|
|
|
struct timespec ts;
|
|
|
|
struct timespec *tsp = NULL;
|
|
|
|
|
|
|
|
if (timeout != INFTIM) {
|
|
|
|
ts.tv_sec = timeout / 1000;
|
2011-01-18 17:45:30 +03:00
|
|
|
ts.tv_nsec = (timeout % 1000) * 1000*1000;
|
2011-01-08 21:11:46 +03:00
|
|
|
|
|
|
|
tsp = &ts;
|
|
|
|
}
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
|
2011-02-07 13:28:18 +03:00
|
|
|
return REALPOLLTS(fds, nfds, tsp, NULL);
|
Begin work on a syscall hijacking library which can be LD_PRELOADed
to convince non-rumped applications to communicate with a rump
kernel instead of the host kernel. The precision of what goes
where is not exactly surgical, but for example when wanting to
debug a web server's TCP/IP stack interaction, it might be enough.
When all you have is a hand grenade, all problems look like a ....
hmm?
There's still plenty to figure out. For example, I'm not sure what
the user interface will be like. Now it just attempts to hijack
network communication. It also needs to sync with symbol renaming
in libc, and maybe autogenerate the non-schizophrenic wrappers
where the communication is heading to exactly one destination, lest
I'll be a mummmy by the time I finish writing them all. As a fun
example of a non-non-schizophrenic one, consider poll().
Work in progress, but I managed to get two non-rumped netcats
talking to each other or fetching the index from a non-rumped
thttpd. telnet works in one direction (i can read the data from
netcat, but anything i send back is not printed). bozohttpd uses
dup2() which i haven't bothered to address yet, etcetc.
(not hooking this up the build for now)
2011-01-07 22:52:43 +03:00
|
|
|
}
|
2011-01-18 14:04:10 +03:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_KQUEUE
|
2011-01-18 14:04:10 +03:00
|
|
|
int
|
2011-02-08 17:45:35 +03:00
|
|
|
REALKEVENT(int kq, const struct kevent *changelist, size_t nchanges,
|
2011-01-18 14:04:10 +03:00
|
|
|
struct kevent *eventlist, size_t nevents,
|
|
|
|
const struct timespec *timeout)
|
|
|
|
{
|
2011-02-08 17:45:35 +03:00
|
|
|
int (*op_kevent)(int, const struct kevent *, size_t,
|
|
|
|
struct kevent *, size_t, const struct timespec *);
|
|
|
|
const struct kevent *ev;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that we don't attempt to kevent rump kernel fd's.
|
|
|
|
* That needs similar treatment to select/poll, but is slightly
|
|
|
|
* trickier since we need to manage to different kq descriptors.
|
|
|
|
* (TODO, in case you're wondering).
|
|
|
|
*/
|
|
|
|
for (i = 0; i < nchanges; i++) {
|
|
|
|
ev = &changelist[i];
|
|
|
|
if (ev->filter == EVFILT_READ || ev->filter == EVFILT_WRITE ||
|
|
|
|
ev->filter == EVFILT_VNODE) {
|
2011-02-23 18:44:38 +03:00
|
|
|
if (fd_isrump((int)ev->ident)) {
|
|
|
|
errno = ENOTSUP;
|
|
|
|
return -1;
|
|
|
|
}
|
2011-02-08 17:45:35 +03:00
|
|
|
}
|
|
|
|
}
|
2011-01-18 14:04:10 +03:00
|
|
|
|
2011-02-08 22:12:54 +03:00
|
|
|
op_kevent = GETSYSCALL(host, KEVENT);
|
2011-02-08 17:45:35 +03:00
|
|
|
return op_kevent(kq, changelist, nchanges, eventlist, nevents, timeout);
|
2011-01-18 14:04:10 +03:00
|
|
|
}
|
2014-11-04 22:05:17 +03:00
|
|
|
#endif /* HAVE_KQUEUE */
|
2011-01-25 15:18:33 +03:00
|
|
|
|
2011-02-21 16:19:35 +03:00
|
|
|
/*
|
|
|
|
* mmapping from a rump kernel is not supported, so disallow it.
|
|
|
|
*/
|
|
|
|
void *
|
|
|
|
mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (flags & MAP_FILE && fd_isrump(fd)) {
|
|
|
|
errno = ENOSYS;
|
|
|
|
return MAP_FAILED;
|
|
|
|
}
|
|
|
|
return host_mmap(addr, len, prot, flags, fd, offset);
|
|
|
|
}
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-03-09 18:03:18 +03:00
|
|
|
/*
|
|
|
|
* these go to one or the other on a per-process configuration
|
|
|
|
*/
|
|
|
|
int __sysctl(const int *, unsigned int, void *, size_t *, const void *, size_t);
|
|
|
|
int
|
|
|
|
__sysctl(const int *name, unsigned int namelen, void *old, size_t *oldlenp,
|
|
|
|
const void *new, size_t newlen)
|
|
|
|
{
|
|
|
|
int (*op___sysctl)(const int *, unsigned int, void *, size_t *,
|
|
|
|
const void *, size_t);
|
|
|
|
|
|
|
|
if (rumpsysctl) {
|
|
|
|
op___sysctl = GETSYSCALL(rump, __SYSCTL);
|
|
|
|
} else {
|
|
|
|
op___sysctl = GETSYSCALL(host, __SYSCTL);
|
|
|
|
/* we haven't inited yet */
|
|
|
|
if (__predict_false(op___sysctl == NULL)) {
|
2011-03-11 02:02:56 +03:00
|
|
|
op___sysctl = rumphijack_dlsym(RTLD_NEXT, "__sysctl");
|
2011-03-09 18:03:18 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return op___sysctl(name, namelen, old, oldlenp, new, newlen);
|
|
|
|
}
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
2011-03-09 18:03:18 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
/*
|
|
|
|
* Rest are std type calls.
|
|
|
|
*/
|
|
|
|
|
2015-03-05 03:26:17 +03:00
|
|
|
#ifdef HAVE_UTIMENSAT
|
2015-03-05 02:31:49 +03:00
|
|
|
ATCALL(int, utimensat, DUALCALL_UTIMENSAT, \
|
|
|
|
(int fd, const char *path, const struct timespec t[2], int f), \
|
|
|
|
(int, const char *, const struct timespec [2], int),
|
|
|
|
(fd, path, t, f))
|
2015-03-05 03:26:17 +03:00
|
|
|
#endif
|
2015-03-05 02:31:49 +03:00
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
FDCALL(int, bind, DUALCALL_BIND, \
|
|
|
|
(int fd, const struct sockaddr *name, socklen_t namelen), \
|
|
|
|
(int, const struct sockaddr *, socklen_t), \
|
|
|
|
(fd, name, namelen))
|
|
|
|
|
|
|
|
FDCALL(int, connect, DUALCALL_CONNECT, \
|
|
|
|
(int fd, const struct sockaddr *name, socklen_t namelen), \
|
|
|
|
(int, const struct sockaddr *, socklen_t), \
|
|
|
|
(fd, name, namelen))
|
|
|
|
|
|
|
|
FDCALL(int, getpeername, DUALCALL_GETPEERNAME, \
|
|
|
|
(int fd, struct sockaddr *name, socklen_t *namelen), \
|
|
|
|
(int, struct sockaddr *, socklen_t *), \
|
|
|
|
(fd, name, namelen))
|
|
|
|
|
|
|
|
FDCALL(int, getsockname, DUALCALL_GETSOCKNAME, \
|
|
|
|
(int fd, struct sockaddr *name, socklen_t *namelen), \
|
|
|
|
(int, struct sockaddr *, socklen_t *), \
|
|
|
|
(fd, name, namelen))
|
|
|
|
|
|
|
|
FDCALL(int, listen, DUALCALL_LISTEN, \
|
|
|
|
(int fd, int backlog), \
|
|
|
|
(int, int), \
|
|
|
|
(fd, backlog))
|
|
|
|
|
|
|
|
FDCALL(ssize_t, recvfrom, DUALCALL_RECVFROM, \
|
|
|
|
(int fd, void *buf, size_t len, int flags, \
|
|
|
|
struct sockaddr *from, socklen_t *fromlen), \
|
|
|
|
(int, void *, size_t, int, struct sockaddr *, socklen_t *), \
|
|
|
|
(fd, buf, len, flags, from, fromlen))
|
|
|
|
|
|
|
|
FDCALL(ssize_t, sendto, DUALCALL_SENDTO, \
|
|
|
|
(int fd, const void *buf, size_t len, int flags, \
|
|
|
|
const struct sockaddr *to, socklen_t tolen), \
|
|
|
|
(int, const void *, size_t, int, \
|
|
|
|
const struct sockaddr *, socklen_t), \
|
|
|
|
(fd, buf, len, flags, to, tolen))
|
|
|
|
|
|
|
|
FDCALL(int, getsockopt, DUALCALL_GETSOCKOPT, \
|
|
|
|
(int fd, int level, int optn, void *optval, socklen_t *optlen), \
|
|
|
|
(int, int, int, void *, socklen_t *), \
|
|
|
|
(fd, level, optn, optval, optlen))
|
|
|
|
|
|
|
|
FDCALL(int, setsockopt, DUALCALL_SETSOCKOPT, \
|
|
|
|
(int fd, int level, int optn, \
|
|
|
|
const void *optval, socklen_t optlen), \
|
|
|
|
(int, int, int, const void *, socklen_t), \
|
|
|
|
(fd, level, optn, optval, optlen))
|
|
|
|
|
|
|
|
FDCALL(int, shutdown, DUALCALL_SHUTDOWN, \
|
|
|
|
(int fd, int how), \
|
|
|
|
(int, int), \
|
|
|
|
(fd, how))
|
|
|
|
|
2011-02-07 15:23:05 +03:00
|
|
|
FDCALL(ssize_t, REALREAD, DUALCALL_READ, \
|
2011-01-25 15:18:33 +03:00
|
|
|
(int fd, void *buf, size_t buflen), \
|
|
|
|
(int, void *, size_t), \
|
|
|
|
(fd, buf, buflen))
|
|
|
|
|
2012-10-16 16:56:10 +04:00
|
|
|
#ifdef __linux__
|
|
|
|
ssize_t __read_chk(int, void *, size_t)
|
|
|
|
__attribute__((alias("read")));
|
|
|
|
#endif
|
|
|
|
|
2011-01-25 15:21:36 +03:00
|
|
|
FDCALL(ssize_t, readv, DUALCALL_READV, \
|
2011-01-25 15:18:33 +03:00
|
|
|
(int fd, const struct iovec *iov, int iovcnt), \
|
|
|
|
(int, const struct iovec *, int), \
|
|
|
|
(fd, iov, iovcnt))
|
|
|
|
|
2011-02-21 15:51:06 +03:00
|
|
|
FDCALL(ssize_t, REALPREAD, DUALCALL_PREAD, \
|
|
|
|
(int fd, void *buf, size_t nbytes, off_t offset), \
|
|
|
|
(int, void *, size_t, off_t), \
|
|
|
|
(fd, buf, nbytes, offset))
|
|
|
|
|
|
|
|
FDCALL(ssize_t, preadv, DUALCALL_PREADV, \
|
|
|
|
(int fd, const struct iovec *iov, int iovcnt, off_t offset), \
|
|
|
|
(int, const struct iovec *, int, off_t), \
|
|
|
|
(fd, iov, iovcnt, offset))
|
|
|
|
|
2011-01-25 15:18:33 +03:00
|
|
|
FDCALL(ssize_t, writev, DUALCALL_WRITEV, \
|
|
|
|
(int fd, const struct iovec *iov, int iovcnt), \
|
|
|
|
(int, const struct iovec *, int), \
|
|
|
|
(fd, iov, iovcnt))
|
2011-02-17 15:23:58 +03:00
|
|
|
|
2011-02-21 15:51:06 +03:00
|
|
|
FDCALL(ssize_t, REALPWRITE, DUALCALL_PWRITE, \
|
|
|
|
(int fd, const void *buf, size_t nbytes, off_t offset), \
|
|
|
|
(int, const void *, size_t, off_t), \
|
|
|
|
(fd, buf, nbytes, offset))
|
|
|
|
|
|
|
|
FDCALL(ssize_t, pwritev, DUALCALL_PWRITEV, \
|
|
|
|
(int fd, const struct iovec *iov, int iovcnt, off_t offset), \
|
|
|
|
(int, const struct iovec *, int, off_t), \
|
|
|
|
(fd, iov, iovcnt, offset))
|
|
|
|
|
2012-09-03 15:33:35 +04:00
|
|
|
#ifndef __linux__
|
2011-02-17 15:23:58 +03:00
|
|
|
FDCALL(int, REALFSTAT, DUALCALL_FSTAT, \
|
|
|
|
(int fd, struct stat *sb), \
|
|
|
|
(int, struct stat *), \
|
|
|
|
(fd, sb))
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-02-17 15:23:58 +03:00
|
|
|
FDCALL(int, fstatvfs1, DUALCALL_FSTATVFS1, \
|
|
|
|
(int fd, struct statvfs *buf, int flags), \
|
|
|
|
(int, struct statvfs *, int), \
|
|
|
|
(fd, buf, flags))
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
|
2011-02-21 15:55:21 +03:00
|
|
|
FDCALL(off_t, lseek, DUALCALL_LSEEK, \
|
2011-02-17 15:23:58 +03:00
|
|
|
(int fd, off_t offset, int whence), \
|
|
|
|
(int, off_t, int), \
|
|
|
|
(fd, offset, whence))
|
2012-08-25 22:00:06 +04:00
|
|
|
#ifdef LSEEK_ALIAS
|
|
|
|
__strong_alias(LSEEK_ALIAS,lseek);
|
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
|
2012-09-03 16:07:42 +04:00
|
|
|
#ifndef __linux__
|
2011-02-17 15:23:58 +03:00
|
|
|
FDCALL(int, REALGETDENTS, DUALCALL_GETDENTS, \
|
|
|
|
(int fd, char *buf, size_t nbytes), \
|
|
|
|
(int, char *, size_t), \
|
|
|
|
(fd, buf, nbytes))
|
2012-09-03 16:07:42 +04:00
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
|
|
|
|
FDCALL(int, fchown, DUALCALL_FCHOWN, \
|
|
|
|
(int fd, uid_t owner, gid_t group), \
|
|
|
|
(int, uid_t, gid_t), \
|
|
|
|
(fd, owner, group))
|
|
|
|
|
|
|
|
FDCALL(int, fchmod, DUALCALL_FCHMOD, \
|
|
|
|
(int fd, mode_t mode), \
|
|
|
|
(int, mode_t), \
|
|
|
|
(fd, mode))
|
|
|
|
|
|
|
|
FDCALL(int, ftruncate, DUALCALL_FTRUNCATE, \
|
|
|
|
(int fd, off_t length), \
|
|
|
|
(int, off_t), \
|
|
|
|
(fd, length))
|
|
|
|
|
|
|
|
FDCALL(int, fsync, DUALCALL_FSYNC, \
|
|
|
|
(int fd), \
|
|
|
|
(int), \
|
|
|
|
(fd))
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_FSYNC_RANGE
|
2011-02-17 15:23:58 +03:00
|
|
|
FDCALL(int, fsync_range, DUALCALL_FSYNC_RANGE, \
|
|
|
|
(int fd, int how, off_t start, off_t length), \
|
|
|
|
(int, int, off_t, off_t), \
|
|
|
|
(fd, how, start, length))
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
|
|
|
|
FDCALL(int, futimes, DUALCALL_FUTIMES, \
|
|
|
|
(int fd, const struct timeval *tv), \
|
|
|
|
(int, const struct timeval *), \
|
|
|
|
(fd, tv))
|
|
|
|
|
2015-03-03 04:24:39 +03:00
|
|
|
FDCALL(int, futimens, DUALCALL_FUTIMENS, \
|
|
|
|
(int fd, const struct timespec *ts), \
|
|
|
|
(int, const struct timespec *), \
|
|
|
|
(fd, ts))
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_CHFLAGS
|
2011-02-21 15:51:06 +03:00
|
|
|
FDCALL(int, fchflags, DUALCALL_FCHFLAGS, \
|
|
|
|
(int fd, u_long flags), \
|
|
|
|
(int, u_long), \
|
|
|
|
(fd, flags))
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
2011-02-21 15:51:06 +03:00
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
/*
|
|
|
|
* path-based selectors
|
|
|
|
*/
|
|
|
|
|
2012-09-03 15:33:35 +04:00
|
|
|
#ifndef __linux__
|
2011-02-17 15:23:58 +03:00
|
|
|
PATHCALL(int, REALSTAT, DUALCALL_STAT, \
|
|
|
|
(const char *path, struct stat *sb), \
|
|
|
|
(const char *, struct stat *), \
|
|
|
|
(path, sb))
|
|
|
|
|
|
|
|
PATHCALL(int, REALLSTAT, DUALCALL_LSTAT, \
|
|
|
|
(const char *path, struct stat *sb), \
|
|
|
|
(const char *, struct stat *), \
|
|
|
|
(path, sb))
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
|
|
|
|
PATHCALL(int, chown, DUALCALL_CHOWN, \
|
|
|
|
(const char *path, uid_t owner, gid_t group), \
|
|
|
|
(const char *, uid_t, gid_t), \
|
|
|
|
(path, owner, group))
|
|
|
|
|
|
|
|
PATHCALL(int, lchown, DUALCALL_LCHOWN, \
|
|
|
|
(const char *path, uid_t owner, gid_t group), \
|
|
|
|
(const char *, uid_t, gid_t), \
|
|
|
|
(path, owner, group))
|
|
|
|
|
|
|
|
PATHCALL(int, chmod, DUALCALL_CHMOD, \
|
|
|
|
(const char *path, mode_t mode), \
|
|
|
|
(const char *, mode_t), \
|
|
|
|
(path, mode))
|
|
|
|
|
|
|
|
PATHCALL(int, lchmod, DUALCALL_LCHMOD, \
|
|
|
|
(const char *path, mode_t mode), \
|
|
|
|
(const char *, mode_t), \
|
|
|
|
(path, mode))
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-02-17 15:23:58 +03:00
|
|
|
PATHCALL(int, statvfs1, DUALCALL_STATVFS1, \
|
|
|
|
(const char *path, struct statvfs *buf, int flags), \
|
|
|
|
(const char *, struct statvfs *, int), \
|
|
|
|
(path, buf, flags))
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
2011-02-17 15:23:58 +03:00
|
|
|
|
|
|
|
PATHCALL(int, unlink, DUALCALL_UNLINK, \
|
|
|
|
(const char *path), \
|
|
|
|
(const char *), \
|
|
|
|
(path))
|
|
|
|
|
|
|
|
PATHCALL(int, symlink, DUALCALL_SYMLINK, \
|
2011-02-19 22:17:33 +03:00
|
|
|
(const char *target, const char *path), \
|
2011-02-17 15:23:58 +03:00
|
|
|
(const char *, const char *), \
|
2011-02-19 22:17:33 +03:00
|
|
|
(target, path))
|
2011-02-17 15:23:58 +03:00
|
|
|
|
2012-10-16 16:56:10 +04:00
|
|
|
/*
|
|
|
|
* readlink() can be called from malloc which can be called
|
|
|
|
* from dlsym() during init
|
|
|
|
*/
|
|
|
|
ssize_t
|
|
|
|
readlink(const char *path, char *buf, size_t bufsiz)
|
|
|
|
{
|
|
|
|
int (*op_readlink)(const char *, char *, size_t);
|
|
|
|
enum pathtype pt;
|
|
|
|
|
|
|
|
if ((pt = path_isrump(path)) != PATH_HOST) {
|
|
|
|
op_readlink = GETSYSCALL(rump, READLINK);
|
|
|
|
if (pt == PATH_RUMP)
|
|
|
|
path = path_host2rump(path);
|
|
|
|
} else {
|
|
|
|
op_readlink = GETSYSCALL(host, READLINK);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (__predict_false(op_readlink == NULL)) {
|
|
|
|
errno = ENOENT;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return op_readlink(path, buf, bufsiz);
|
|
|
|
}
|
2011-02-17 15:23:58 +03:00
|
|
|
|
|
|
|
PATHCALL(int, mkdir, DUALCALL_MKDIR, \
|
|
|
|
(const char *path, mode_t mode), \
|
|
|
|
(const char *, mode_t), \
|
|
|
|
(path, mode))
|
|
|
|
|
|
|
|
PATHCALL(int, rmdir, DUALCALL_RMDIR, \
|
|
|
|
(const char *path), \
|
|
|
|
(const char *), \
|
|
|
|
(path))
|
|
|
|
|
|
|
|
PATHCALL(int, utimes, DUALCALL_UTIMES, \
|
|
|
|
(const char *path, const struct timeval *tv), \
|
|
|
|
(const char *, const struct timeval *), \
|
|
|
|
(path, tv))
|
|
|
|
|
|
|
|
PATHCALL(int, lutimes, DUALCALL_LUTIMES, \
|
|
|
|
(const char *path, const struct timeval *tv), \
|
|
|
|
(const char *, const struct timeval *), \
|
|
|
|
(path, tv))
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE_CHFLAGS
|
2011-02-21 15:51:06 +03:00
|
|
|
PATHCALL(int, chflags, DUALCALL_CHFLAGS, \
|
|
|
|
(const char *path, u_long flags), \
|
|
|
|
(const char *, u_long), \
|
|
|
|
(path, flags))
|
|
|
|
|
|
|
|
PATHCALL(int, lchflags, DUALCALL_LCHFLAGS, \
|
|
|
|
(const char *path, u_long flags), \
|
|
|
|
(const char *, u_long), \
|
|
|
|
(path, flags))
|
2014-11-04 22:05:17 +03:00
|
|
|
#endif /* HAVE_CHFLAGS */
|
2011-02-21 15:51:06 +03:00
|
|
|
|
2011-02-17 15:23:58 +03:00
|
|
|
PATHCALL(int, truncate, DUALCALL_TRUNCATE, \
|
|
|
|
(const char *path, off_t length), \
|
|
|
|
(const char *, off_t), \
|
|
|
|
(path, length))
|
2011-02-17 20:18:08 +03:00
|
|
|
|
2011-02-23 18:29:21 +03:00
|
|
|
PATHCALL(int, access, DUALCALL_ACCESS, \
|
|
|
|
(const char *path, int mode), \
|
|
|
|
(const char *, int), \
|
|
|
|
(path, mode))
|
|
|
|
|
2012-09-03 15:33:35 +04:00
|
|
|
#ifndef __linux__
|
2011-02-25 21:29:00 +03:00
|
|
|
PATHCALL(int, REALMKNOD, DUALCALL_MKNOD, \
|
|
|
|
(const char *path, mode_t mode, dev_t dev), \
|
|
|
|
(const char *, mode_t, dev_t), \
|
|
|
|
(path, mode, dev))
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
2011-02-25 21:29:00 +03:00
|
|
|
|
2011-02-17 20:18:08 +03:00
|
|
|
/*
|
|
|
|
* Note: with mount the decisive parameter is the mount
|
|
|
|
* destination directory. This is because we don't really know
|
|
|
|
* about the "source" directory in a generic call (and besides,
|
|
|
|
* it might not even exist, cf. nfs).
|
|
|
|
*/
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-02-17 20:18:08 +03:00
|
|
|
PATHCALL(int, REALMOUNT, DUALCALL_MOUNT, \
|
|
|
|
(const char *type, const char *path, int flags, \
|
|
|
|
void *data, size_t dlen), \
|
|
|
|
(const char *, const char *, int, void *, size_t), \
|
|
|
|
(type, path, flags, data, dlen))
|
|
|
|
|
|
|
|
PATHCALL(int, unmount, DUALCALL_UNMOUNT, \
|
|
|
|
(const char *path, int flags), \
|
|
|
|
(const char *, int), \
|
|
|
|
(path, flags))
|
2014-11-04 22:05:17 +03:00
|
|
|
#endif /* __NetBSD__ */
|
2011-03-09 18:03:18 +03:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef HAVE___QUOTACTL
|
2012-02-01 09:34:38 +04:00
|
|
|
PATHCALL(int, __quotactl, DUALCALL_QUOTACTL, \
|
|
|
|
(const char *path, struct quotactl_args *args), \
|
|
|
|
(const char *, struct quotactl_args *), \
|
|
|
|
(path, args))
|
2014-11-04 22:05:17 +03:00
|
|
|
#endif /* HAVE___QUOTACTL */
|
2011-03-09 21:45:30 +03:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-03-10 02:26:19 +03:00
|
|
|
PATHCALL(int, REALGETFH, DUALCALL_GETFH, \
|
|
|
|
(const char *path, void *fhp, size_t *fh_size), \
|
|
|
|
(const char *, void *, size_t *), \
|
|
|
|
(path, fhp, fh_size))
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
2011-03-10 02:26:19 +03:00
|
|
|
|
2011-03-09 18:03:18 +03:00
|
|
|
/*
|
|
|
|
* These act different on a per-process vfs configuration
|
|
|
|
*/
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-03-09 18:03:18 +03:00
|
|
|
VFSCALL(VFSBIT_GETVFSSTAT, int, getvfsstat, DUALCALL_GETVFSSTAT, \
|
|
|
|
(struct statvfs *buf, size_t buflen, int flags), \
|
|
|
|
(struct statvfs *, size_t, int), \
|
|
|
|
(buf, buflen, flags))
|
2012-08-25 22:00:06 +04:00
|
|
|
#endif
|
2011-03-09 18:03:18 +03:00
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2011-03-09 18:03:18 +03:00
|
|
|
VFSCALL(VFSBIT_FHCALLS, int, REALFHOPEN, DUALCALL_FHOPEN, \
|
|
|
|
(const void *fhp, size_t fh_size, int flags), \
|
|
|
|
(const char *, size_t, int), \
|
|
|
|
(fhp, fh_size, flags))
|
|
|
|
|
|
|
|
VFSCALL(VFSBIT_FHCALLS, int, REALFHSTAT, DUALCALL_FHSTAT, \
|
|
|
|
(const void *fhp, size_t fh_size, struct stat *sb), \
|
|
|
|
(const char *, size_t, struct stat *), \
|
|
|
|
(fhp, fh_size, sb))
|
|
|
|
|
|
|
|
VFSCALL(VFSBIT_FHCALLS, int, REALFHSTATVFS1, DUALCALL_FHSTATVFS1, \
|
|
|
|
(const void *fhp, size_t fh_size, struct statvfs *sb, int flgs),\
|
|
|
|
(const char *, size_t, struct statvfs *, int), \
|
|
|
|
(fhp, fh_size, sb, flgs))
|
2012-09-03 15:33:35 +04:00
|
|
|
#endif
|
2011-03-09 18:03:18 +03:00
|
|
|
|
|
|
|
|
2014-11-04 22:05:17 +03:00
|
|
|
#ifdef __NetBSD__
|
2012-08-25 22:00:06 +04:00
|
|
|
|
|
|
|
/* finally, put nfssvc here. "keep the namespace clean" */
|
2011-03-09 18:03:18 +03:00
|
|
|
#include <nfs/rpcv2.h>
|
|
|
|
#include <nfs/nfs.h>
|
|
|
|
|
|
|
|
int
|
|
|
|
nfssvc(int flags, void *argstructp)
|
|
|
|
{
|
|
|
|
int (*op_nfssvc)(int, void *);
|
|
|
|
|
|
|
|
if (vfsbits & VFSBIT_NFSSVC){
|
|
|
|
struct nfsd_args *nfsdargs;
|
|
|
|
|
|
|
|
/* massage the socket descriptor if necessary */
|
|
|
|
if (flags == NFSSVC_ADDSOCK) {
|
|
|
|
nfsdargs = argstructp;
|
|
|
|
nfsdargs->sock = fd_host2rump(nfsdargs->sock);
|
|
|
|
}
|
|
|
|
op_nfssvc = GETSYSCALL(rump, NFSSVC);
|
|
|
|
} else
|
|
|
|
op_nfssvc = GETSYSCALL(host, NFSSVC);
|
|
|
|
|
|
|
|
return op_nfssvc(flags, argstructp);
|
|
|
|
}
|
2014-11-04 22:05:17 +03:00
|
|
|
#endif /* __NetBSD__ */
|