Blame libio/oldiopopen.c

Packit Service 82fcde
/* Copyright (C) 1998-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Written by Per Bothner <bothner@cygnus.com>.
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
   As a special exception, if you link the code in this file with
Packit Service 82fcde
   files compiled with a GNU compiler to produce an executable,
Packit Service 82fcde
   that does not cause the resulting executable to be covered by
Packit Service 82fcde
   the GNU Lesser General Public License.  This exception does not
Packit Service 82fcde
   however invalidate any other reasons why the executable file
Packit Service 82fcde
   might be covered by the GNU Lesser General Public License.
Packit Service 82fcde
   This exception applies to code released by its copyright holders
Packit Service 82fcde
   in files containing the exception.  */
Packit Service 82fcde
Packit Service 82fcde
#define _IO_USE_OLD_IO_FILE
Packit Service 82fcde
#include "libioP.h"
Packit Service 82fcde
#include <signal.h>
Packit Service 82fcde
#include <unistd.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
Packit Service 82fcde
#include <shlib-compat.h>
Packit Service 82fcde
#if SHLIB_COMPAT (libc, GLIBC_2_0, GLIBC_2_1)
Packit Service 82fcde
Packit Service 82fcde
struct _IO_proc_file
Packit Service 82fcde
{
Packit Service 82fcde
  struct _IO_FILE_complete_plus file;
Packit Service 82fcde
  /* Following fields must match those in class procbuf (procbuf.h) */
Packit Service 82fcde
  pid_t pid;
Packit Service 82fcde
  struct _IO_proc_file *next;
Packit Service 82fcde
};
Packit Service 82fcde
typedef struct _IO_proc_file _IO_proc_file;
Packit Service 82fcde
Packit Service 82fcde
static struct _IO_proc_file *old_proc_file_chain;
Packit Service 82fcde
Packit Service 82fcde
#ifdef _IO_MTSAFE_IO
Packit Service 82fcde
static _IO_lock_t proc_file_chain_lock = _IO_lock_initializer;
Packit Service 82fcde
Packit Service 82fcde
static void
Packit Service 82fcde
unlock (void *not_used)
Packit Service 82fcde
{
Packit Service 82fcde
  _IO_lock_unlock (proc_file_chain_lock);
Packit Service 82fcde
}
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
FILE *
Packit Service 82fcde
attribute_compat_text_section
Packit Service 82fcde
_IO_old_proc_open (FILE *fp, const char *command, const char *mode)
Packit Service 82fcde
{
Packit Service 82fcde
  volatile int read_or_write;
Packit Service 82fcde
  volatile int parent_end, child_end;
Packit Service 82fcde
  int pipe_fds[2];
Packit Service 82fcde
  pid_t child_pid;
Packit Service 82fcde
  if (_IO_file_is_open (fp))
Packit Service 82fcde
    return NULL;
Packit Service 82fcde
  if (__pipe (pipe_fds) < 0)
Packit Service 82fcde
    return NULL;
Packit Service 82fcde
  if (mode[0] == 'r' && mode[1] == '\0')
Packit Service 82fcde
    {
Packit Service 82fcde
      parent_end = pipe_fds[0];
Packit Service 82fcde
      child_end = pipe_fds[1];
Packit Service 82fcde
      read_or_write = _IO_NO_WRITES;
Packit Service 82fcde
    }
Packit Service 82fcde
  else if (mode[0] == 'w' && mode[1] == '\0')
Packit Service 82fcde
    {
Packit Service 82fcde
      parent_end = pipe_fds[1];
Packit Service 82fcde
      child_end = pipe_fds[0];
Packit Service 82fcde
      read_or_write = _IO_NO_READS;
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    {
Packit Service 82fcde
      __close (pipe_fds[0]);
Packit Service 82fcde
      __close (pipe_fds[1]);
Packit Service 82fcde
      __set_errno (EINVAL);
Packit Service 82fcde
      return NULL;
Packit Service 82fcde
    }
Packit Service 82fcde
  ((_IO_proc_file *) fp)->pid = child_pid = __fork ();
Packit Service 82fcde
  if (child_pid == 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      int child_std_end = mode[0] == 'r' ? 1 : 0;
Packit Service 82fcde
      struct _IO_proc_file *p;
Packit Service 82fcde
Packit Service 82fcde
      __close (parent_end);
Packit Service 82fcde
      if (child_end != child_std_end)
Packit Service 82fcde
	{
Packit Service 82fcde
	  __dup2 (child_end, child_std_end);
Packit Service 82fcde
	  __close (child_end);
Packit Service 82fcde
	}
Packit Service 82fcde
      /* POSIX.2:  "popen() shall ensure that any streams from previous
Packit Service 82fcde
         popen() calls that remain open in the parent process are closed
Packit Service 82fcde
	 in the new child process." */
Packit Service 82fcde
      for (p = old_proc_file_chain; p; p = p->next)
Packit Service 82fcde
	__close (_IO_fileno ((FILE *) p));
Packit Service 82fcde
Packit Service 82fcde
      execl ("/bin/sh", "sh", "-c", command, (char *) 0);
Packit Service 82fcde
      _exit (127);
Packit Service 82fcde
    }
Packit Service 82fcde
  __close (child_end);
Packit Service 82fcde
  if (child_pid < 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      __close (parent_end);
Packit Service 82fcde
      return NULL;
Packit Service 82fcde
    }
Packit Service 82fcde
  _IO_fileno (fp) = parent_end;
Packit Service 82fcde
Packit Service 82fcde
  /* Link into old_proc_file_chain. */
Packit Service 82fcde
#ifdef _IO_MTSAFE_IO
Packit Service 82fcde
  _IO_cleanup_region_start_noarg (unlock);
Packit Service 82fcde
  _IO_lock_lock (proc_file_chain_lock);
Packit Service 82fcde
#endif
Packit Service 82fcde
  ((_IO_proc_file *) fp)->next = old_proc_file_chain;
Packit Service 82fcde
  old_proc_file_chain = (_IO_proc_file *) fp;
Packit Service 82fcde
#ifdef _IO_MTSAFE_IO
Packit Service 82fcde
  _IO_lock_unlock (proc_file_chain_lock);
Packit Service 82fcde
  _IO_cleanup_region_end (0);
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
  _IO_mask_flags (fp, read_or_write, _IO_NO_READS|_IO_NO_WRITES);
Packit Service 82fcde
  return fp;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
FILE *
Packit Service 82fcde
attribute_compat_text_section
Packit Service 82fcde
_IO_old_popen (const char *command, const char *mode)
Packit Service 82fcde
{
Packit Service 82fcde
  struct locked_FILE
Packit Service 82fcde
  {
Packit Service 82fcde
    struct _IO_proc_file fpx;
Packit Service 82fcde
#ifdef _IO_MTSAFE_IO
Packit Service 82fcde
    _IO_lock_t lock;
Packit Service 82fcde
#endif
Packit Service 82fcde
  } *new_f;
Packit Service 82fcde
  FILE *fp;
Packit Service 82fcde
Packit Service 82fcde
  new_f = (struct locked_FILE *) malloc (sizeof (struct locked_FILE));
Packit Service 82fcde
  if (new_f == NULL)
Packit Service 82fcde
    return NULL;
Packit Service 82fcde
#ifdef _IO_MTSAFE_IO
Packit Service 82fcde
  new_f->fpx.file.file._file._lock = &new_f->lock;
Packit Service 82fcde
#endif
Packit Service 82fcde
  fp = &new_f->fpx.file.file._file;
Packit Service 82fcde
  _IO_old_init (fp, 0);
Packit Service 82fcde
  _IO_JUMPS_FILE_plus (&new_f->fpx.file) = &_IO_old_proc_jumps;
Packit Service 82fcde
  _IO_old_file_init_internal ((struct _IO_FILE_plus *) &new_f->fpx.file);
Packit Service 82fcde
  if (_IO_old_proc_open (fp, command, mode) != NULL)
Packit Service 82fcde
    return fp;
Packit Service 82fcde
  _IO_un_link ((struct _IO_FILE_plus *) &new_f->fpx.file);
Packit Service 82fcde
  free (new_f);
Packit Service 82fcde
  return NULL;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
attribute_compat_text_section
Packit Service 82fcde
_IO_old_proc_close (FILE *fp)
Packit Service 82fcde
{
Packit Service 82fcde
  /* This is not name-space clean. FIXME! */
Packit Service 82fcde
  int wstatus;
Packit Service 82fcde
  _IO_proc_file **ptr = &old_proc_file_chain;
Packit Service 82fcde
  pid_t wait_pid;
Packit Service 82fcde
  int status = -1;
Packit Service 82fcde
Packit Service 82fcde
  /* Unlink from old_proc_file_chain. */
Packit Service 82fcde
#ifdef _IO_MTSAFE_IO
Packit Service 82fcde
  _IO_cleanup_region_start_noarg (unlock);
Packit Service 82fcde
  _IO_lock_lock (proc_file_chain_lock);
Packit Service 82fcde
#endif
Packit Service 82fcde
  for ( ; *ptr != NULL; ptr = &(*ptr)->next)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (*ptr == (_IO_proc_file *) fp)
Packit Service 82fcde
	{
Packit Service 82fcde
	  *ptr = (*ptr)->next;
Packit Service 82fcde
	  status = 0;
Packit Service 82fcde
	  break;
Packit Service 82fcde
	}
Packit Service 82fcde
    }
Packit Service 82fcde
#ifdef _IO_MTSAFE_IO
Packit Service 82fcde
  _IO_lock_unlock (proc_file_chain_lock);
Packit Service 82fcde
  _IO_cleanup_region_end (0);
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
  if (status < 0 || __close (_IO_fileno(fp)) < 0)
Packit Service 82fcde
    return -1;
Packit Service 82fcde
  /* POSIX.2 Rationale:  "Some historical implementations either block
Packit Service 82fcde
     or ignore the signals SIGINT, SIGQUIT, and SIGHUP while waiting
Packit Service 82fcde
     for the child process to terminate.  Since this behavior is not
Packit Service 82fcde
     described in POSIX.2, such implementations are not conforming." */
Packit Service 82fcde
  do
Packit Service 82fcde
    {
Packit Service 82fcde
      wait_pid = __waitpid (((_IO_proc_file *) fp)->pid, &wstatus, 0);
Packit Service 82fcde
    }
Packit Service 82fcde
  while (wait_pid == -1 && errno == EINTR);
Packit Service 82fcde
  if (wait_pid == -1)
Packit Service 82fcde
    return -1;
Packit Service 82fcde
  return wstatus;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
const struct _IO_jump_t _IO_old_proc_jumps libio_vtable = {
Packit Service 82fcde
  JUMP_INIT_DUMMY,
Packit Service 82fcde
  JUMP_INIT(finish, _IO_old_file_finish),
Packit Service 82fcde
  JUMP_INIT(overflow, _IO_old_file_overflow),
Packit Service 82fcde
  JUMP_INIT(underflow, _IO_old_file_underflow),
Packit Service 82fcde
  JUMP_INIT(uflow, _IO_default_uflow),
Packit Service 82fcde
  JUMP_INIT(pbackfail, _IO_default_pbackfail),
Packit Service 82fcde
  JUMP_INIT(xsputn, _IO_old_file_xsputn),
Packit Service 82fcde
  JUMP_INIT(xsgetn, _IO_default_xsgetn),
Packit Service 82fcde
  JUMP_INIT(seekoff, _IO_old_file_seekoff),
Packit Service 82fcde
  JUMP_INIT(seekpos, _IO_default_seekpos),
Packit Service 82fcde
  JUMP_INIT(setbuf, _IO_old_file_setbuf),
Packit Service 82fcde
  JUMP_INIT(sync, _IO_old_file_sync),
Packit Service 82fcde
  JUMP_INIT(doallocate, _IO_file_doallocate),
Packit Service 82fcde
  JUMP_INIT(read, _IO_file_read),
Packit Service 82fcde
  JUMP_INIT(write, _IO_old_file_write),
Packit Service 82fcde
  JUMP_INIT(seek, _IO_file_seek),
Packit Service 82fcde
  JUMP_INIT(close, _IO_old_proc_close),
Packit Service 82fcde
  JUMP_INIT(stat, _IO_file_stat),
Packit Service 82fcde
  JUMP_INIT(showmanyc, _IO_default_showmanyc),
Packit Service 82fcde
  JUMP_INIT(imbue, _IO_default_imbue)
Packit Service 82fcde
};
Packit Service 82fcde
Packit Service 82fcde
strong_alias (_IO_old_popen, __old_popen)
Packit Service 82fcde
compat_symbol (libc, _IO_old_popen, _IO_popen, GLIBC_2_0);
Packit Service 82fcde
compat_symbol (libc, __old_popen, popen, GLIBC_2_0);
Packit Service 82fcde
compat_symbol (libc, _IO_old_proc_open, _IO_proc_open, GLIBC_2_0);
Packit Service 82fcde
compat_symbol (libc, _IO_old_proc_close, _IO_proc_close, GLIBC_2_0);
Packit Service 82fcde
Packit Service 82fcde
#endif