start to track Daniel's version
[util-vserver.git] / src / vcontext.c
diff --git a/src/vcontext.c b/src/vcontext.c
deleted file mode 100644 (file)
index 3461482..0000000
+++ /dev/null
@@ -1,454 +0,0 @@
-// $Id: vcontext.c 2578 2007-08-08 20:05:26Z dhozac $    --*- c -*--
-
-// Copyright (C) 2004-2006 Enrico Scholz <enrico.scholz@informatik.tu-chemnitz.de>
-//  
-// This program is free software; you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation; version 2 of the License.
-//  
-// This program is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-// GNU General Public License for more details.
-//  
-// You should have received a copy of the GNU General Public License
-// along with this program; if not, write to the Free Software
-// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
-
-
-#ifdef HAVE_CONFIG_H
-#  include <config.h>
-#endif
-
-#include "util.h"
-#include "lib/internal.h"
-#include "lib_internal/jail.h"
-#include "lib_internal/sys_personality.h"
-
-#include <vserver.h>
-#include <getopt.h>
-#include <fcntl.h>
-#include <errno.h>
-#include <sys/socket.h>
-#include <sys/un.h>
-#include <assert.h>
-#include <signal.h>
-#include <sys/types.h>
-#include <pwd.h>
-#include <grp.h>
-
-#include <linux/personality.h>
-
-#define ENSC_WRAPPERS_PREFIX   "vcontext: "
-#define ENSC_WRAPPERS_UNISTD   1
-#define ENSC_WRAPPERS_VSERVER  1
-#define ENSC_WRAPPERS_FCNTL    1
-#define ENSC_WRAPPERS_SOCKET   1
-#define ENSC_WRAPPERS_IOSOCK   1
-#include <wrappers.h>
-
-#define CMD_HELP               0x1000
-#define CMD_VERSION            0x1001
-#define CMD_XID                        0x4000
-#define CMD_CREATE             0x4001
-#define CMD_MIGRATE            0x4003
-#define CMD_INITPID            0x4002
-#define CMD_DISCONNECT         0x4004
-#define CMD_UID                        0x4005
-#define CMD_CHROOT             0x4006
-#define CMD_SILENT             0x4007
-#define CMD_SYNCSOCK           0x4008
-#define CMD_SYNCMSG            0x4009
-#define CMD_MIGRATESELF                0x400a
-#define CMD_ENDSETUP           0x400b
-#define CMD_SILENTEXIST                0x400c
-#define CMD_NAMESPACE          0x400d
-#define CMD_PERSTYPE           0x400e
-#define CMD_PERSFLAG           0x400f
-#define CMD_VLOGIN             0x4010
-
-
-struct option const
-CMDLINE_OPTIONS[] = {
-  { "help",       no_argument,       0, CMD_HELP },
-  { "version",    no_argument,       0, CMD_VERSION },
-  { "ctx",        required_argument, 0, CMD_XID },
-  { "xid",        required_argument, 0, CMD_XID },
-  { "create",     no_argument,       0, CMD_CREATE },
-  { "migrate",    no_argument,       0, CMD_MIGRATE },
-  { "migrate-self", no_argument,               0, CMD_MIGRATESELF },
-  { "initpid",      no_argument,               0, CMD_INITPID },
-  { "endsetup",     no_argument,        0, CMD_ENDSETUP },
-  { "disconnect",   no_argument,       0, CMD_DISCONNECT },
-  { "silent",       no_argument,               0, CMD_SILENT },
-  { "silentexist",  no_argument,               0, CMD_SILENTEXIST },
-  { "uid",          required_argument,  0, CMD_UID },
-  { "chroot",       no_argument,               0, CMD_CHROOT },
-  { "namespace",    no_argument,               0, CMD_NAMESPACE },
-  { "syncsock",     required_argument,         0, CMD_SYNCSOCK },
-  { "syncmsg",      required_argument,         0, CMD_SYNCMSG },
-  { "personality-type",  required_argument, 0, CMD_PERSTYPE },
-  { "personality-flags", required_argument, 0, CMD_PERSFLAG },
-  { "vlogin",       no_argument,        0, CMD_VLOGIN },
-#if 1  
-  { "fakeinit",     no_argument,               0, CMD_INITPID },       // compatibility
-#endif  
-  { 0,0,0,0 },
-};
-
-struct Arguments {
-    bool               do_create;
-    bool               do_migrate;
-    bool               do_migrateself;
-    bool               do_disconnect;
-    bool               do_endsetup;
-    bool               is_initpid;
-    bool               is_silentexist;
-    bool               set_namespace;
-    bool               do_vlogin;
-    uint_least32_t     personality_flags;
-    uint_least32_t     personality_type;
-    int                        verbosity;
-    bool               do_chroot;
-    char const *       uid;
-    xid_t              xid;
-    char const *       sync_sock;
-    char const *       sync_msg;
-};
-
-int            wrapper_exit_code = 255;
-
-void do_vlogin(int argc, char *argv[], int ind);
-
-static void
-showHelp(int fd, char const *cmd, int res)
-{
-  WRITE_MSG(fd, "Usage:\n    ");
-  WRITE_STR(fd, cmd);
-  WRITE_MSG(fd,
-           " --create [--xid <xid>] <opts>* [--] <program> <args>*\n    ");
-  WRITE_STR(fd, cmd);
-  WRITE_MSG(fd,
-           " [(--migrate --xid <xid>)|--migrate-self]  <opts>* [--] <program> <args>*\n"
-           "\n"
-           "<opts> can be:\n"
-           "    --chroot        ...  chroot into current directory\n"
-           "    --namespace     ...  execute namespace management operations\n"
-           "    --uid <uid>     ...  change uid\n"
-           "    --initpid       ...  set current process as general process reaper\n"
-           "                         for ctx (possible for --migrate only)\n"
-           "    --endsetup      ...  clear the setup flag; usefully for migrate only\n"
-           "    --disconnect    ...  start program in background\n"
-           "    --personality-type <type>\n"
-           "                    ...  execute <program> in the given execution domain\n"
-           "    --personality-flags <flags>+\n"
-           "                    ...  set special flags for the given execution domain\n"
-           "    --silent        ...  be silent\n"
-           "    --silentexist   ...  be silent when context exists already; usefully\n"
-           "                         for '--create' only\n"
-           "    --syncsock <file-name>\n"
-           "                    ...  before executing the program, send a message\n"
-           "                         to the socket and wait until it closes.\n"
-           "                         <file-name> must be a SOCK_STREAM unix socket\n"
-           "    --syncmsg <message>\n"
-           "                    ...  use <message> as synchronization message; by\n"
-           "                         default, 'ok' will be used\n"
-           "    --vlogin        ...  enable terminal proxy\n"
-           "\n"
-           "'vcontext --create' exits with code 254 iff the context exists already.\n"
-           "\n"
-           "Please report bugs to " PACKAGE_BUGREPORT "\n");
-
-  exit(res);
-}
-
-static void
-showVersion()
-{
-  WRITE_MSG(1,
-           "vcontext " VERSION " -- manages the creation of security contexts\n"
-           "This program is part of " PACKAGE_STRING "\n\n"
-           "Copyright (C) 2004-2006 Enrico Scholz\n"
-           VERSION_COPYRIGHT_DISCLAIMER);
-  exit(0);
-}
-
-#include "context-sync.hc"
-
-static inline ALWAYSINLINE void
-tellContext(xid_t ctx, bool do_it)
-{
-  char         buf[sizeof(xid_t)*3+2];
-  size_t       l;
-
-  if (!do_it) return;
-
-  l = utilvserver_fmt_long(buf,ctx);
-
-  WRITE_MSG(1, "New security context is ");
-  Vwrite   (1, buf, l);
-  WRITE_MSG(1, "\n");
-}
-
-static int
-connectExternalSync(char const *filename)
-{
-  int                  fd;
-  struct sockaddr_un   addr;
-  
-  if (filename==0) return -1;
-
-  ENSC_INIT_UNIX_SOCK(addr, filename);
-
-  fd = Esocket(PF_UNIX, SOCK_STREAM, 0);
-  Econnect(fd, &addr, sizeof(addr));
-
-  return fd;
-}
-
-static void
-setFlags(struct Arguments const *args, xid_t xid)
-{
-  struct vc_ctx_flags  flags = { 0,0 };
-
-  if (args->is_initpid)
-    flags.mask |=  VC_VXF_STATE_INIT;
-
-  if (args->do_endsetup)
-    flags.mask |=  VC_VXF_STATE_SETUP;
-
-  if (flags.mask!=0) {
-    DPRINTF("set_flags: mask=%08llx, flag=%08llx\n", flags.mask, flags.flagword);
-    Evc_set_cflags(xid, &flags);
-  }
-}
-
-static void
-doExternalSync(int fd, char const *msg)
-{
-  char         c;
-  
-  if (fd==-1) return;
-
-  if (msg) EsendAll(fd, msg, strlen(msg));
-  Eshutdown(fd, SHUT_WR);
-
-  if (TEMP_FAILURE_RETRY(recv(fd, &c, 1, MSG_NOSIGNAL))!=0) {
-    WRITE_MSG(2, ENSC_WRAPPERS_PREFIX "unexpected external synchronization event\n");
-    exit(wrapper_exit_code);
-  }
-
-  Eclose(fd);
-}
-
-static inline ALWAYSINLINE int
-doit(struct Arguments const *args, int argc, char *argv[])
-{
-  int                  p[2][2];
-  pid_t                        pid = initSync(p, args->do_disconnect);
-  
-  if (pid==0) {
-    xid_t                      xid;
-    int                                ext_sync_fd = connectExternalSync(args->sync_sock);
-
-    doSyncStage0(p, args->do_disconnect);  
-    
-    if (args->do_create) {
-      xid = vc_ctx_create(args->xid, NULL);
-      if (xid==VC_NOCTX) {
-       switch (errno) {
-         case EEXIST   :
-           if (!args->is_silentexist)
-             perror(ENSC_WRAPPERS_PREFIX "vc_ctx_create()");
-           return 254;
-         default       :
-           perror(ENSC_WRAPPERS_PREFIX "vc_ctx_create()");
-           return wrapper_exit_code;
-       }
-      }
-      tellContext(xid, args->verbosity>=1);
-    }
-    else
-      xid = args->xid;
-
-    if (args->do_chroot) {
-      Echroot(".");
-      if (args->set_namespace) {
-       if (args->do_migrateself)  Evc_set_namespace(xid, 0);
-       else if (args->do_migrate) Evc_enter_namespace(xid, 0);
-      }
-    }
-
-    setFlags(args, xid);
-
-    if (args->do_migrate && !args->do_migrateself)
-      Evc_ctx_migrate(xid, 0);
-
-    if (args->uid != NULL) {
-      uid_t uid = 0;
-      unsigned long tmp;
-
-      if (!isNumberUnsigned(args->uid, &tmp, false)) {
-#ifdef __dietlibc__
-       struct passwd *pw;
-       pw = getpwnam(args->uid);
-       if (pw == NULL) {
-         WRITE_MSG(2, ENSC_WRAPPERS_PREFIX "Username '");
-         WRITE_STR(2, args->uid);
-         WRITE_MSG(2, "' does not exist\n");
-         return wrapper_exit_code;
-       }
-       uid = pw->pw_uid;
-       Einitgroups(args->uid, pw->pw_gid);
-       Esetgid(pw->pw_gid);
-#else
-       WRITE_MSG(2, ENSC_WRAPPERS_PREFIX "Uid '");
-       WRITE_STR(2, args->uid);
-       WRITE_MSG(2, "' is not a number\n");
-       return wrapper_exit_code;
-#endif
-      }
-      else
-       uid = (uid_t) tmp;
-
-      Esetuid((uid_t) uid);
-      if (getuid()!=uid) {
-       WRITE_MSG(2, ENSC_WRAPPERS_PREFIX "Something went wrong while changing the UID\n");
-       exit(wrapper_exit_code);
-      }
-    }
-
-    if (args->personality_type!=VC_BAD_PERSONALITY &&
-       sys_personality(args->personality_type | args->personality_flags)==-1) {
-      perror(ENSC_WRAPPERS_PREFIX "personality()");
-      exit(wrapper_exit_code);
-    }
-
-    doExternalSync(ext_sync_fd, args->sync_msg);
-    doSyncStage1(p, args->do_disconnect);
-    DPRINTF("doit: pid=%u, ppid=%u\n", getpid(), getppid());
-    if (!args->do_vlogin)
-      execvp (argv[optind],argv+optind);
-    else
-      do_vlogin(argc, argv, optind);
-    doSyncStage2(p, args->do_disconnect);
-
-    PERROR_Q(ENSC_WRAPPERS_PREFIX "execvp", argv[optind]);
-    exit(wrapper_exit_code);
-  }
-
-  assert(args->do_disconnect);
-    
-  waitOnSync(pid, p, args->xid!=VC_DYNAMIC_XID && args->do_migrate);
-  return EXIT_SUCCESS;
-}
-
-static uint_least32_t
-parsePersonalityType(char const *str)
-{
-  uint_least32_t       res = vc_str2personalitytype(str, 0);
-  if (res==VC_BAD_PERSONALITY) {
-    WRITE_MSG(2, ENSC_WRAPPERS_PREFIX "bad personality type\n");
-    exit(wrapper_exit_code);
-  }
-
-  return res;
-}
-
-static uint_least32_t
-parsePersonalityFlags(char const *str)
-{
-  struct vc_err_listparser     err;
-  uint_least32_t               res;
-
-  if (vc_list2personalityflag(str, 0, &res, &err)==-1) {
-    WRITE_MSG(2, ENSC_WRAPPERS_PREFIX "bad personality flag '");
-    Vwrite(2, err.ptr, err.len);
-    WRITE_MSG(2, "'\n");
-    exit(wrapper_exit_code);
-  }
-
-  return res;
-}
-
-int main (int argc, char *argv[])
-{
-  struct Arguments             args = {
-    .do_create         = false,
-    .do_migrate        = false,
-    .do_migrateself    = false,
-    .do_disconnect     = false,
-    .do_endsetup       = false,
-    .do_vlogin         = false,
-    .is_initpid        = false,
-    .is_silentexist    = false,
-    .set_namespace     = false,
-    .verbosity         = 1,
-    .uid               = NULL,
-    .xid               = VC_DYNAMIC_XID,
-    .personality_type  = VC_BAD_PERSONALITY,
-    .personality_flags = 0,
-    .sync_msg          = "ok",
-  };
-  
-  while (1) {
-    int                c = getopt_long(argc, argv, "+", CMDLINE_OPTIONS, 0);
-    if (c==-1) break;
-    
-    switch (c) {
-      case CMD_HELP            :  showHelp(1, argv[0], 0);
-      case CMD_VERSION         :  showVersion();
-      case CMD_CREATE          :  args.do_create      = true;   break;
-      case CMD_MIGRATE         :  args.do_migrate     = true;   break;
-      case CMD_DISCONNECT      :  args.do_disconnect  = true;   break;
-      case CMD_ENDSETUP                :  args.do_endsetup    = true;   break;
-      case CMD_VLOGIN          :  args.do_vlogin      = true;   break;
-      case CMD_INITPID         :  args.is_initpid     = true;   break;
-      case CMD_CHROOT          :  args.do_chroot      = true;   break;
-      case CMD_NAMESPACE       :  args.set_namespace  = true;   break;
-      case CMD_SILENTEXIST     :  args.is_silentexist = true;   break;
-      case CMD_SYNCSOCK                :  args.sync_sock      = optarg; break;
-      case CMD_SYNCMSG         :  args.sync_msg       = optarg; break;
-      case CMD_UID             :  args.uid            = optarg; break;
-      case CMD_XID             :  args.xid = Evc_xidopt2xid(optarg,true); break;
-      case CMD_SILENT          :  --args.verbosity; break;
-      case CMD_PERSTYPE                :
-       args.personality_type   = parsePersonalityType(optarg);
-       break;
-      case CMD_PERSFLAG                :
-       args.personality_flags |= parsePersonalityFlags(optarg);
-       break;
-      case CMD_MIGRATESELF     :
-       args.do_migrate     = true;
-       args.do_migrateself = true;
-       break;
-
-      default          :
-       WRITE_MSG(2, "Try '");
-       WRITE_STR(2, argv[0]);
-       WRITE_MSG(2, " --help' for more information.\n");
-       return wrapper_exit_code;
-       break;
-    }
-  }
-
-  signal(SIGCHLD, SIG_DFL);
-  
-  if (args.do_migrateself)
-    args.xid = Evc_get_task_xid(0);
-  
-  if (!args.do_create && !args.do_migrate)
-    WRITE_MSG(2, "Neither '--create' nor '--migrate' specified; try '--help' for more information\n");
-  else if (args.do_create  &&  args.do_migrate)
-    WRITE_MSG(2, "Can not specify '--create' and '--migrate' at the same time; try '--help' for more information\n");
-  else if (!args.do_migrate && args.is_initpid)
-    WRITE_MSG(2, "'--initpid' is possible in combination with '--migrate' only\n");
-  else if (!args.do_create && args.xid==VC_DYNAMIC_XID)
-    WRITE_MSG(2, ENSC_WRAPPERS_PREFIX "Can not migrate to an unknown context\n");
-  else if (optind>=argc)
-    WRITE_MSG(2, "No command given; use '--help' for more information.\n");
-  else
-    return doit(&args, argc, argv);
-
-  return wrapper_exit_code;
-}