1998-03-05 07:53:47 +03:00
|
|
|
/* 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
|
2000-08-23 02:50:00 +04:00
|
|
|
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
1998-03-05 07:53:47 +03:00
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <signal.h> /* my_system */
|
|
|
|
#include <limits.h> /* INT_MAX */
|
|
|
|
#include <sys/time.h> /* select: timeout */
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#ifdef HAVE_SYS_WAIT_H
|
|
|
|
# include <sys/wait.h> /* my_system */
|
|
|
|
#endif
|
|
|
|
#include <errno.h> /* my_system */
|
|
|
|
#include <pwd.h>
|
|
|
|
#include <grp.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#ifdef HAVE_SYS_SELECT_H
|
|
|
|
# include <sys/select.h>
|
|
|
|
#endif
|
1998-04-07 23:08:31 +04:00
|
|
|
#include <gnome.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <gdk/gdkprivate.h>
|
1999-01-27 04:14:57 +03:00
|
|
|
#include "global.h"
|
1998-03-05 07:53:47 +03:00
|
|
|
|
2001-07-25 03:07:26 +04:00
|
|
|
int my_system (int flags, const char *shell, const char *command)
|
1998-03-05 07:53:47 +03:00
|
|
|
{
|
2001-07-25 03:07:26 +04:00
|
|
|
pid_t pid;
|
1998-03-18 09:24:20 +03:00
|
|
|
struct sigaction ignore, save_intr, save_quit, save_stop;
|
|
|
|
int status = 0, i;
|
1999-02-12 00:16:36 +03:00
|
|
|
|
1998-03-18 09:24:20 +03:00
|
|
|
ignore.sa_handler = SIG_IGN;
|
|
|
|
sigemptyset (&ignore.sa_mask);
|
|
|
|
ignore.sa_flags = 0;
|
1998-03-05 07:53:47 +03:00
|
|
|
|
1998-03-18 09:24:20 +03:00
|
|
|
sigaction (SIGINT, &ignore, &save_intr);
|
|
|
|
sigaction (SIGQUIT, &ignore, &save_quit);
|
|
|
|
|
2001-07-25 03:07:26 +04:00
|
|
|
if ((pid = fork ()) < 0){
|
1998-03-18 09:24:20 +03:00
|
|
|
return -1;
|
|
|
|
}
|
2001-07-25 03:07:26 +04:00
|
|
|
if (pid == 0){
|
1998-04-07 23:08:31 +04:00
|
|
|
const int top = max_open_files ();
|
1999-03-09 06:56:54 +03:00
|
|
|
struct sigaction default_pipe;
|
|
|
|
|
1998-03-18 09:24:20 +03:00
|
|
|
sigaction (SIGINT, &save_intr, NULL);
|
|
|
|
sigaction (SIGQUIT, &save_quit, NULL);
|
|
|
|
|
1999-03-09 06:56:54 +03:00
|
|
|
/*
|
|
|
|
* 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++)
|
1998-03-18 09:24:20 +03:00
|
|
|
close (i);
|
|
|
|
|
1999-03-09 06:56:54 +03:00
|
|
|
/* Setup the file descriptor for the child */
|
|
|
|
|
|
|
|
/* stdin */
|
2001-07-25 03:07:26 +04:00
|
|
|
open ("/dev/null", O_RDONLY);
|
1999-03-09 06:56:54 +03:00
|
|
|
|
|
|
|
/* stdout */
|
2001-07-25 03:07:26 +04:00
|
|
|
open ("/dev/null", O_WRONLY);
|
1999-03-09 06:56:54 +03:00
|
|
|
|
|
|
|
/* stderr */
|
2001-07-25 03:07:26 +04:00
|
|
|
open ("/dev/null", O_WRONLY);
|
1999-03-09 06:56:54 +03:00
|
|
|
|
1999-01-11 03:48:23 +03:00
|
|
|
if (!(flags & EXECUTE_WAIT))
|
2001-07-25 03:07:26 +04:00
|
|
|
pid = fork ();
|
1999-01-11 03:48:23 +03:00
|
|
|
|
2001-07-25 03:07:26 +04:00
|
|
|
if (pid == 0){
|
1998-04-23 23:31:32 +04:00
|
|
|
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;
|
1999-02-08 20:39:31 +03:00
|
|
|
|
1999-02-12 00:16:36 +03:00
|
|
|
if (flags & EXECUTE_WAIT)
|
2001-07-25 03:07:26 +04:00
|
|
|
waitpid (pid, &status, 0);
|
1998-04-23 23:31:32 +04:00
|
|
|
}
|
|
|
|
/* We need to use _exit instead of exit to avoid
|
|
|
|
* calling the atexit handlers (specifically the gdk atexit
|
|
|
|
* handler
|
|
|
|
*/
|
|
|
|
_exit (0);
|
1998-03-18 09:24:20 +03:00
|
|
|
}
|
2001-07-25 03:07:26 +04:00
|
|
|
waitpid (pid, &status, 0);
|
1998-03-05 07:53:47 +03:00
|
|
|
sigaction (SIGINT, &save_intr, NULL);
|
|
|
|
sigaction (SIGQUIT, &save_quit, NULL);
|
1998-03-18 09:24:20 +03:00
|
|
|
sigaction (SIGTSTP, &save_stop, NULL);
|
|
|
|
|
|
|
|
return WEXITSTATUS(status);
|
|
|
|
}
|