/* Various utilities - Unix variants Copyright (C) 1998 the Free Software Foundation. Written 1998 by: Miguel de Icaza 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; either version 2 of the License, or (at your option) any later version. 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. */ #include #include #include #ifdef HAVE_UNISTD_H #include #endif #include #include /* my_system */ #include /* INT_MAX */ #include /* select: timeout */ #include #include #include #include #ifdef HAVE_SYS_WAIT_H # include /* my_system */ #endif #include /* my_system */ #include #include #include #include #ifdef HAVE_SYS_SELECT_H # include #endif #include #include #include #include "global.h" static struct sigaction previous_sigchld; static int child_died_notify_handler; /* * A list of childs that were executed with a temporary file * We remove the files when they die */ static GList *children; typedef struct { pid_t pid; char *temp_file; } Child; /* * Received when a child dies, notifies the high level routione * that new input is available */ static void gnome_sigchld_handler (int sig) { char c; if (previous_sigchld.sa_handler != SIG_IGN && previous_sigchld.sa_handler != SIG_DFL){ (*previous_sigchld.sa_handler)(sig); } write (child_died_notify_handler, &c, sizeof (c)); } /* * Invoked from the main loop when a child has died * deal with it */ static void gnome_child_died (gpointer data, gint source, GdkInputCondition condition) { GList *l; char c; read (source, &c, sizeof (c)); for (l = children; l; l = l->next){ int status; Child *child = l->data; if (child->pid == waitpid (child->pid, &status, WUNTRACED | WNOHANG)){ children = g_list_remove (children, child); unlink (child->temp_file); g_free (child->temp_file); g_free (child); } } } int my_system_get_child_pid (int flags, const char *shell, const char *command, pid_t *pid) { struct sigaction ignore, save_intr, save_quit, save_stop; int status = 0, i; static int gnome_sigchld_installed; ignore.sa_handler = SIG_IGN; sigemptyset (&ignore.sa_mask); ignore.sa_flags = 0; sigaction (SIGINT, &ignore, &save_intr); sigaction (SIGQUIT, &ignore, &save_quit); if (!gnome_sigchld_installed){ struct sigaction newsig; int monitors [2]; pipe (monitors); sigemptyset (&newsig.sa_mask); newsig.sa_flags = 0; newsig.sa_handler = gnome_sigchld_handler; sigaction (SIGCHLD, &newsig, &previous_sigchld); gnome_sigchld_installed = 1; gdk_input_add (monitors [0], GDK_INPUT_READ, gnome_child_died, NULL); } if ((*pid = fork ()) < 0){ return -1; } if (*pid == 0){ const int top = max_open_files (); struct sigaction default_pipe; sigaction (SIGINT, &save_intr, NULL); sigaction (SIGQUIT, &save_quit, NULL); /* * reset sigpipe */ default_pipe.sa_handler = SIG_DFL; sigemptyset (&default_pipe.sa_mask); default_pipe.sa_flags = 0; sigaction (SIGPIPE, &default_pipe, NULL); for (i = 0; i < top; i++) close (i); /* Setup the file descriptor for the child */ /* stdin */ open ("/dev/null", O_APPEND); /* stdout */ open ("/dev/null", O_RDONLY); /* stderr */ open ("/dev/null", O_RDONLY); if (!(flags & EXECUTE_WAIT)) *pid = fork (); if (*pid == 0){ if (flags & EXECUTE_AS_SHELL) execl (shell, shell, "-c", command, (char *) 0); else execlp (shell, shell, command, (char *) 0); /* See note below for why we use _exit () */ _exit (127); /* Exec error */ } else { int status; if (flags & EXECUTE_WAIT) waitpid (*pid, &status, 0); if (flags & EXECUTE_TEMPFILE){ Child *child; child = g_new (Child, 1); child->pid = *pid; child->temp_file = g_strdup (command); } } /* We need to use _exit instead of exit to avoid * calling the atexit handlers (specifically the gdk atexit * handler */ _exit (0); } waitpid (*pid, &status, 0); sigaction (SIGINT, &save_intr, NULL); sigaction (SIGQUIT, &save_quit, NULL); sigaction (SIGTSTP, &save_stop, NULL); return WEXITSTATUS(status); } int my_system (int flags, const char *shell, const char *command) { pid_t pid; return my_system_get_child_pid (flags, shell, command, &pid); }