ack/lang/cem/libcc.ansi/stdlib/system.c
2018-06-21 22:33:47 +02:00

68 lines
1.4 KiB
C

/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*/
/* $Id$ */
#if defined(_POSIX_SOURCE)
#include <sys/types.h>
#endif
#include <stdlib.h>
#include <signal.h>
extern char** environ;
extern int _fork(void);
extern int _wait(int*);
extern void _exit(int);
extern void _execve(const char* path, const char** argv, const char** envp);
extern void _close(int);
#define FAIL 127
static const char* exec_tab[] = {
"sh", /* argv[0] */
"-c", /* argument to the shell */
NULL, /* to be filled with user command */
NULL /* terminating NULL */
};
int system(const char* str)
{
int pid, exitstatus, waitval;
int i;
if ((pid = _fork()) < 0)
return str ? -1 : 0;
if (pid == 0)
{
for (i = 3; i <= 20; i++)
_close(i);
if (!str)
str = "cd ."; /* just testing for a shell */
exec_tab[2] = str; /* fill in command */
_execve("/bin/sh", exec_tab, (char const**)environ);
/* get here if execve fails ... */
_exit(FAIL); /* see manual page */
}
while ((waitval = _wait(&exitstatus)) != pid)
{
if (waitval == -1)
break;
}
if (waitval == -1)
{
/* no child ??? or maybe interrupted ??? */
exitstatus = -1;
}
if (!str)
{
if (exitstatus == FAIL << 8) /* execve() failed */
exitstatus = 0;
else
exitstatus = 1; /* /bin/sh exists */
}
return exitstatus;
}