Blame support/support_can_chroot.c

Packit 6c4009
/* Return true if the process can perform a chroot operation.
Packit 6c4009
   Copyright (C) 2017-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <support/check.h>
Packit 6c4009
#include <support/namespace.h>
Packit 6c4009
#include <support/support.h>
Packit 6c4009
#include <support/xunistd.h>
Packit 6c4009
#include <sys/stat.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
callback (void *closure)
Packit 6c4009
{
Packit 6c4009
  int *result = closure;
Packit 6c4009
  struct stat64 before;
Packit 6c4009
  xstat ("/dev", &before);
Packit 6c4009
  if (chroot ("/dev") != 0)
Packit 6c4009
    {
Packit 6c4009
      *result = errno;
Packit 6c4009
      return;
Packit 6c4009
    }
Packit 6c4009
  struct stat64 after;
Packit 6c4009
  xstat ("/", &after);
Packit 6c4009
  TEST_VERIFY (before.st_dev == after.st_dev);
Packit 6c4009
  TEST_VERIFY (before.st_ino == after.st_ino);
Packit 6c4009
  *result = 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
bool
Packit 6c4009
support_can_chroot (void)
Packit 6c4009
{
Packit 6c4009
  int *result = support_shared_allocate (sizeof (*result));
Packit 6c4009
  *result = 0;
Packit 6c4009
  support_isolate_in_subprocess (callback, result);
Packit 6c4009
  bool ok = *result == 0;
Packit 6c4009
  if (!ok)
Packit 6c4009
    {
Packit 6c4009
      static bool already_warned;
Packit 6c4009
      if (!already_warned)
Packit 6c4009
        {
Packit 6c4009
          already_warned = true;
Packit 6c4009
          errno = *result;
Packit 6c4009
          printf ("warning: this process does not support chroot: %m\n");
Packit 6c4009
        }
Packit 6c4009
    }
Packit 6c4009
  support_shared_free (result);
Packit 6c4009
  return ok;
Packit 6c4009
}