Blame sysdeps/posix/system.c

Packit Service 82fcde
/* Copyright (C) 1991-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <signal.h>
Packit Service 82fcde
#include <stddef.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <unistd.h>
Packit Service 82fcde
#include <sys/types.h>
Packit Service 82fcde
#include <sys/wait.h>
Packit Service 82fcde
#include <libc-lock.h>
Packit Service 82fcde
#include <sysdep-cancel.h>
Packit Service 82fcde
#include <sigsetops.h>
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
#define	SHELL_PATH	"/bin/sh"	/* Path of the shell.  */
Packit Service 82fcde
#define	SHELL_NAME	"sh"		/* Name to give it.  */
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
#ifdef _LIBC_REENTRANT
Packit Service 82fcde
static struct sigaction intr, quit;
Packit Service 82fcde
static int sa_refcntr;
Packit Service 82fcde
__libc_lock_define_initialized (static, lock);
Packit Service 82fcde
Packit Service 82fcde
# define DO_LOCK() __libc_lock_lock (lock)
Packit Service 82fcde
# define DO_UNLOCK() __libc_lock_unlock (lock)
Packit Service 82fcde
# define INIT_LOCK() ({ __libc_lock_init (lock); sa_refcntr = 0; })
Packit Service 82fcde
# define ADD_REF() sa_refcntr++
Packit Service 82fcde
# define SUB_REF() --sa_refcntr
Packit Service 82fcde
#else
Packit Service 82fcde
# define DO_LOCK()
Packit Service 82fcde
# define DO_UNLOCK()
Packit Service 82fcde
# define INIT_LOCK()
Packit Service 82fcde
# define ADD_REF() 0
Packit Service 82fcde
# define SUB_REF() 0
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
/* Execute LINE as a shell command, returning its status.  */
Packit Service 82fcde
static int
Packit Service 82fcde
do_system (const char *line)
Packit Service 82fcde
{
Packit Service 82fcde
  int status, save;
Packit Service 82fcde
  pid_t pid;
Packit Service 82fcde
  struct sigaction sa;
Packit Service 82fcde
#ifndef _LIBC_REENTRANT
Packit Service 82fcde
  struct sigaction intr, quit;
Packit Service 82fcde
#endif
Packit Service 82fcde
  sigset_t omask;
Packit Service 82fcde
Packit Service 82fcde
  sa.sa_handler = SIG_IGN;
Packit Service 82fcde
  sa.sa_flags = 0;
Packit Service 82fcde
  __sigemptyset (&sa.sa_mask);
Packit Service 82fcde
Packit Service 82fcde
  DO_LOCK ();
Packit Service 82fcde
  if (ADD_REF () == 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (__sigaction (SIGINT, &sa, &intr) < 0)
Packit Service 82fcde
	{
Packit Service 82fcde
	  (void) SUB_REF ();
Packit Service 82fcde
	  goto out;
Packit Service 82fcde
	}
Packit Service 82fcde
      if (__sigaction (SIGQUIT, &sa, &quit) < 0)
Packit Service 82fcde
	{
Packit Service 82fcde
	  save = errno;
Packit Service 82fcde
	  (void) SUB_REF ();
Packit Service 82fcde
	  goto out_restore_sigint;
Packit Service 82fcde
	}
Packit Service 82fcde
    }
Packit Service 82fcde
  DO_UNLOCK ();
Packit Service 82fcde
Packit Service 82fcde
  /* We reuse the bitmap in the 'sa' structure.  */
Packit Service 82fcde
  __sigaddset (&sa.sa_mask, SIGCHLD);
Packit Service 82fcde
  save = errno;
Packit Service 82fcde
  if (__sigprocmask (SIG_BLOCK, &sa.sa_mask, &omask) < 0)
Packit Service 82fcde
    {
Packit Service 82fcde
#ifndef _LIBC
Packit Service 82fcde
      if (errno == ENOSYS)
Packit Service 82fcde
	__set_errno (save);
Packit Service 82fcde
      else
Packit Service 82fcde
#endif
Packit Service 82fcde
	{
Packit Service 82fcde
	  DO_LOCK ();
Packit Service 82fcde
	  if (SUB_REF () == 0)
Packit Service 82fcde
	    {
Packit Service 82fcde
	      save = errno;
Packit Service 82fcde
	      (void) __sigaction (SIGQUIT, &quit, (struct sigaction *) NULL);
Packit Service 82fcde
	    out_restore_sigint:
Packit Service 82fcde
	      (void) __sigaction (SIGINT, &intr, (struct sigaction *) NULL);
Packit Service 82fcde
	      __set_errno (save);
Packit Service 82fcde
	    }
Packit Service 82fcde
	out:
Packit Service 82fcde
	  DO_UNLOCK ();
Packit Service 82fcde
	  return -1;
Packit Service 82fcde
	}
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
#ifdef CLEANUP_HANDLER
Packit Service 82fcde
  CLEANUP_HANDLER;
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
#ifdef FORK
Packit Service 82fcde
  pid = FORK ();
Packit Service 82fcde
#else
Packit Service 82fcde
  pid = __fork ();
Packit Service 82fcde
#endif
Packit Service 82fcde
  if (pid == (pid_t) 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* Child side.  */
Packit Service 82fcde
      const char *new_argv[4];
Packit Service 82fcde
      new_argv[0] = SHELL_NAME;
Packit Service 82fcde
      new_argv[1] = "-c";
Packit Service 82fcde
      new_argv[2] = line;
Packit Service 82fcde
      new_argv[3] = NULL;
Packit Service 82fcde
Packit Service 82fcde
      /* Restore the signals.  */
Packit Service 82fcde
      (void) __sigaction (SIGINT, &intr, (struct sigaction *) NULL);
Packit Service 82fcde
      (void) __sigaction (SIGQUIT, &quit, (struct sigaction *) NULL);
Packit Service 82fcde
      (void) __sigprocmask (SIG_SETMASK, &omask, (sigset_t *) NULL);
Packit Service 82fcde
      INIT_LOCK ();
Packit Service 82fcde
Packit Service 82fcde
      /* Exec the shell.  */
Packit Service 82fcde
      (void) __execve (SHELL_PATH, (char *const *) new_argv, __environ);
Packit Service 82fcde
      _exit (127);
Packit Service 82fcde
    }
Packit Service 82fcde
  else if (pid < (pid_t) 0)
Packit Service 82fcde
    /* The fork failed.  */
Packit Service 82fcde
    status = -1;
Packit Service 82fcde
  else
Packit Service 82fcde
    /* Parent side.  */
Packit Service 82fcde
    {
Packit Service 82fcde
      /* Note the system() is a cancellation point.  But since we call
Packit Service 82fcde
	 waitpid() which itself is a cancellation point we do not
Packit Service 82fcde
	 have to do anything here.  */
Packit Service 82fcde
      if (TEMP_FAILURE_RETRY (__waitpid (pid, &status, 0)) != pid)
Packit Service 82fcde
	status = -1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
#ifdef CLEANUP_HANDLER
Packit Service 82fcde
  CLEANUP_RESET;
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
  save = errno;
Packit Service 82fcde
  DO_LOCK ();
Packit Service 82fcde
  if ((SUB_REF () == 0
Packit Service 82fcde
       && (__sigaction (SIGINT, &intr, (struct sigaction *) NULL)
Packit Service 82fcde
	   | __sigaction (SIGQUIT, &quit, (struct sigaction *) NULL)) != 0)
Packit Service 82fcde
      || __sigprocmask (SIG_SETMASK, &omask, (sigset_t *) NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
#ifndef _LIBC
Packit Service 82fcde
      /* glibc cannot be used on systems without waitpid.  */
Packit Service 82fcde
      if (errno == ENOSYS)
Packit Service 82fcde
	__set_errno (save);
Packit Service 82fcde
      else
Packit Service 82fcde
#endif
Packit Service 82fcde
	status = -1;
Packit Service 82fcde
    }
Packit Service 82fcde
  DO_UNLOCK ();
Packit Service 82fcde
Packit Service 82fcde
  return status;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
__libc_system (const char *line)
Packit Service 82fcde
{
Packit Service 82fcde
  if (line == NULL)
Packit Service 82fcde
    /* Check that we have a command processor available.  It might
Packit Service 82fcde
       not be available after a chroot(), for example.  */
Packit Service 82fcde
    return do_system ("exit 0") == 0;
Packit Service 82fcde
Packit Service 82fcde
  return do_system (line);
Packit Service 82fcde
}
Packit Service 82fcde
weak_alias (__libc_system, system)