[GRASS-SVN] r40644 - grass/branches/releasebranch_6_4/lib/gis

svn_grass at osgeo.org svn_grass at osgeo.org
Mon Jan 25 04:19:29 EST 2010


Author: neteler
Date: 2010-01-25 04:19:28 -0500 (Mon, 25 Jan 2010)
New Revision: 40644

Modified:
   grass/branches/releasebranch_6_4/lib/gis/spawn.c
Log:
sync'ed to G6.5 (trac #714)

Modified: grass/branches/releasebranch_6_4/lib/gis/spawn.c
===================================================================
--- grass/branches/releasebranch_6_4/lib/gis/spawn.c	2010-01-25 09:17:20 UTC (rev 40643)
+++ grass/branches/releasebranch_6_4/lib/gis/spawn.c	2010-01-25 09:19:28 UTC (rev 40644)
@@ -26,6 +26,8 @@
 
 #ifndef __MINGW32__
 #include <sys/wait.h>
+#else
+#include <windows.h>
 #endif
 #include <grass/config.h>
 #include <grass/gis.h>
@@ -56,188 +58,430 @@
  * \return process status on success
  */
 
+struct redirect
+{
+    int dst_fd;
+    int src_fd;
+    const char *file;
+    int mode;
+};
+
+struct signal
+{
+    int which;
+    int action;
+    int signum;
+    int valid;
+#ifndef __MINGW32__
+    struct sigaction old_act;
+    sigset_t old_mask;
+#endif
+};
+
+struct binding
+{
+    const char *var;
+    const char *val;
+};
+
+struct spawn
+{
+    const char *args[MAX_ARGS];
+    int num_args;
+    struct redirect redirects[MAX_REDIRECTS];
+    int num_redirects;
+    struct signal signals[MAX_SIGNALS];
+    int num_signals;
+    struct binding bindings[MAX_BINDINGS];
+    int num_bindings;
+    int background;
+    const char *directory;
+};
+
+static void parse_arglist(struct spawn *sp, va_list va);
+static void parse_argvec(struct spawn *sp, const char **va);
+
 #ifdef __MINGW32__
 
-int G_spawn(const char *command, ...)
+struct buffer {
+    char *str;
+    size_t len;
+    size_t size;
+};
+
+static const int INCREMENT = 50;
+
+static void clear(struct buffer *b)
 {
-    va_list va;
-    char *args[MAX_ARGS];
-    int num_args = 0;
+    b->len = 0;
+    b->str[b->len] = '\0';
+}
 
-    va_start(va, command);
+static void init(struct buffer *b)
+{
+    b->str = G_malloc(1);
+    b->size = 1;
+    clear(b);
+}
 
-    for (num_args = 0; num_args < MAX_ARGS;) {
-	char *arg = va_arg(va, char *);
+static char *release(struct buffer *b)
+{
+    char *p = b->str;
 
-	args[num_args++] = arg;
-	if (!arg)
-	    break;
-    }
+    b->str = NULL;
+    b->size = 0;
+    b->len = 0;
 
-    va_end(va);
+    return p;
+}
 
-    if (num_args >= MAX_ARGS) {
-	G_warning(_("Too many arguments"));
-	return -1;
+static void finish(struct buffer *b)
+{
+    if (b->str)
+	G_free(b->str);
+    release(b);
+}
+
+static void ensure(struct buffer *b, size_t n)
+{
+    if (b->size <= b->len + n + 1) {
+	b->size = b->len + n + INCREMENT;
+	b->str = G_realloc(b->str, b->size);
     }
+}
 
-    G_debug(3, "spawning '%s' ...", command);
+static void append(struct buffer *b, const char *str)
+{
+    size_t n = strlen(str);
 
-    return _spawnvp(_P_WAIT, (char *)command, args);
+    ensure(b, n);
+    memcpy(&b->str[b->len], str, n);
+    b->len += n;
+    b->str[b->len] = '\0';
 }
 
-#else
+static void append_char(struct buffer *b, char c)
+{
+    ensure(b, 1);
+    b->str[b->len] = c;
+    b->len++;
+    b->str[b->len] = '\0';
+}
 
-int G_spawn(const char *command, ...)
+static void escape_arg(struct buffer *result, const char *arg)
 {
-    va_list va;
-    char *args[MAX_ARGS];
-    int num_args = 0;
-    struct sigaction act, intr, quit;
-    sigset_t block, oldmask;
-    int status = -1;
-    pid_t pid;
+    struct buffer buf;
+    int quote, j;
 
-    va_start(va, command);
+    init(&buf);
 
-    for (num_args = 0; num_args < MAX_ARGS;) {
-	char *arg = va_arg(va, char *);
+    quote = arg[0] == '\0' || strchr(arg, ' ') || strchr(arg, '\t');
 
-	args[num_args++] = arg;
-	if (!arg)
+    if (quote)
+	append_char(result, '\"');
+
+    for (j = 0; arg[j]; j++) {
+	int c = arg[j];
+	int k;
+
+	switch (c) {
+	case '\\':
+	    append_char(&buf, '\\');
 	    break;
+	case '\"':
+	    for (k = 0; k < buf.len; k++)
+		append(result, "\\\\");
+	    clear(&buf);
+	    append(result, "\\\"");
+	    break;
+	default:
+	    if (buf.len > 0) {
+		append(result, buf.str);
+		clear(&buf);
+	    }
+	    append_char(result, c);
+	}
     }
 
-    va_end(va);
+    if (buf.len > 0)
+	append(result, buf.str);
 
-    if (num_args >= MAX_ARGS) {
-	G_warning(_("Too many arguments"));
-	return -1;
+    if (quote) {
+	append(result, buf.str);
+	append_char(result, '\"');
     }
 
-    sigemptyset(&act.sa_mask);
-    act.sa_flags = SA_RESTART;
+    finish(&buf);
+}
 
-    act.sa_handler = SIG_IGN;
-    if (sigaction(SIGINT, &act, &intr) < 0)
-	goto error_1;
-    if (sigaction(SIGQUIT, &act, &quit) < 0)
-	goto error_2;
+static char *check_program(const char *pgm, const char *dir, const char *ext)
+{
+    char pathname[GPATH_MAX];
 
-    sigemptyset(&block);
-    sigaddset(&block, SIGCHLD);
-    if (sigprocmask(SIG_BLOCK, &block, &oldmask) < 0)
-	goto error_3;
+    sprintf(pathname, "%s%s%s%s", dir, *dir ? "\\" : "", pgm, ext);
+    return access(pathname, 0) == 0
+	? G_store(pathname)
+	: NULL;
+}
 
-    G_debug(3, "forking '%s' ...", command);
+static char *find_program_ext(const char *pgm, const char *dir, char **pathext)
+{
+    char *result;
+    int i;
 
-    pid = fork();
+    if (result = check_program(pgm, dir, ""), result)
+	return result;
 
-    if (pid < 0) {
-	G_warning(_("Unable to create a new process"));
-	goto error_4;
+    for (i = 0; pathext[i]; i++) {
+	const char *ext = pathext[i];
+	if (result = check_program(pgm, dir, ext), result)
+	    return result;
     }
 
-    if (pid == 0) {
-	sigaction(SIGINT, &intr, NULL);
-	sigaction(SIGQUIT, &quit, NULL);
+    return NULL;
+}
 
-	execvp(command, args);
-	G_warning(_("Unable to execute command"));
-	_exit(127);
+static char *find_program_dir_ext(const char *pgm, char **path, char **pathext)
+{
+    char *result = NULL;
+    int i;
+
+    if (strchr(pgm, '\\') || strchr(pgm, '/')) {
+	if (result = find_program_ext(pgm, "", pathext), result)
+	    return result;
     }
     else {
-	pid_t n;
+	if (result = find_program_ext(pgm, ".", pathext), result)
+	    return result;
 
-	do
-	    n = waitpid(pid, &status, 0);
-	while (n == (pid_t) - 1 && errno == EINTR);
+	for (i = 0; path[i]; i++) {
+	    const char *dir = path[i];
+	    if (result = find_program_ext(pgm, dir, pathext), result)
+		return result;
+	}
+    }
 
-	if (n != pid)
-	    status = -1;
+    return NULL;
+}
+
+static char *find_program(const char *pgm)
+{
+    char **path = G_tokenize(getenv("PATH"), ";");
+    char **pathext = G_tokenize(getenv("PATHEXT"), ";");
+    char *result = find_program_dir_ext(pgm, path, pathext);
+    G_free_tokens(path);
+    G_free_tokens(pathext);
+    return result;
+}
+
+static char *make_command_line(int shell, const char *cmd, const char **argv)
+{
+    struct buffer result;
+    int i;
+
+    init(&result);
+
+    if (shell) {
+	const char *comspec = getenv("COMSPEC");
+	append(&result, comspec ? comspec : "cmd.exe");
+	append(&result, " /c \"");
+	escape_arg(&result, cmd);
     }
 
-  error_4:
-    sigprocmask(SIG_SETMASK, &oldmask, NULL);
-  error_3:
-    sigaction(SIGQUIT, &quit, NULL);
-  error_2:
-    sigaction(SIGINT, &intr, NULL);
-  error_1:
-    return status;
+    for (i = shell ? 1 : 0; argv[i]; i++) {
+	if (result.len > 0)
+	    append_char(&result, ' ');
+	escape_arg(&result, argv[i]);
+    }
+
+    append(&result, "\"");
+
+    return release(&result);
 }
 
-#endif /*__MINGW32__*/
+static char *make_environment(const char **envp)
+{
+    struct buffer result;
+    int i;
 
-struct redirect
+    init(&result);
+
+    for (i = 0; envp[i]; i++) {
+	const char *env = envp[i];
+
+	append(&result, env);
+	append_char(&result, '\0');
+    }
+
+    return release(&result);
+}
+
+static HANDLE get_handle(int fd)
 {
-    int dst_fd;
-    int src_fd;
-    const char *file;
-    int mode;
-};
+    HANDLE h1, h2;
 
-struct signal
+    if (fd < 0)
+	return INVALID_HANDLE_VALUE;
+
+    h1 = (HANDLE) _get_osfhandle(fd);
+    if (!DuplicateHandle(GetCurrentProcess(), h1,
+			 GetCurrentProcess(), &h2,
+			 0, TRUE, DUPLICATE_SAME_ACCESS))
+	return INVALID_HANDLE_VALUE;
+
+    return h2;
+}
+
+static int win_spawn(const char *cmd, const char **argv, const char **envp,
+		     const char *cwd, HANDLE handles[3], int background,
+		     int shell)
 {
-    int which;
-    int action;
-    int signum;
-    int valid;
-#ifndef __MINGW32__
-    struct sigaction old_act;
-    sigset_t old_mask;
-#endif
-};
+    char *args = make_command_line(shell, cmd, argv);
+    char *env = make_environment(envp);
+    char *program = shell ? NULL : find_program(cmd);
+    STARTUPINFO si;
+    PROCESS_INFORMATION pi;
+    BOOL result;
+    DWORD exitcode;
 
-struct binding
+    if (!shell) {
+	G_debug(3, "win_spawn: program = %s", program);
+
+	if (!program) {
+	    G_free(args);
+	    G_free(env);
+	    return -1;
+	}
+    }
+
+    G_debug(3, "win_spawn: args = %s", args);
+
+    memset(&si, 0, sizeof(si));
+    si.cb = sizeof(si);
+
+    si.dwFlags |= STARTF_USESTDHANDLES;
+    si.hStdInput  = handles[0];
+    si.hStdOutput = handles[1];
+    si.hStdError  = handles[2];
+
+    result = CreateProcess(
+	program,	/* lpApplicationName */
+	args,		/* lpCommandLine */
+	NULL,		/* lpProcessAttributes */
+	NULL,		/* lpThreadAttributes */
+	1,		/* bInheritHandles */
+	0,		/* dwCreationFlags */
+	env,		/* lpEnvironment */
+	cwd,		/* lpCurrentDirectory */
+	&si,		/* lpStartupInfo */
+	&pi		/* lpProcessInformation */
+	);
+
+    G_free(args);
+    G_free(env);
+    G_free(program);
+
+    if (!result) {
+	G_warning(_("CreateProcess() failed: error = %d"), GetLastError());
+	return -1;
+    }
+
+    if (!background) {
+	WaitForSingleObject(pi.hProcess, INFINITE);
+	if (!GetExitCodeProcess(pi.hProcess, &exitcode))
+	    return -1;
+	return (int) exitcode;
+    }
+
+    return pi.dwProcessId;
+}
+
+static void do_redirects(struct redirect *redirects, int num_redirects, HANDLE handles[3])
 {
-    const char *var;
-    const char *val;
-};
+    int i;
 
-static const char *args[MAX_ARGS];
-static int num_args;
-static struct redirect redirects[MAX_REDIRECTS];
-static int num_redirects;
-static struct signal signals[MAX_SIGNALS];
-static int num_signals;
-static struct binding bindings[MAX_BINDINGS];
-static int num_bindings;
-static int background;
-static const char *directory;
+    for (i = 0; i < 3; i++)
+	handles[i] = get_handle(i);
 
-#ifdef __MINGW32__
+    for (i = 0; i < num_redirects; i++) {
+	struct redirect *r = &redirects[i];
 
-static void do_redirects(struct redirect *redirects, int num_redirects)
+	if (r->dst_fd < 0 || r->dst_fd > 2) {
+	    G_warning(_("G_spawn: unable to redirect descriptor %d"), r->dst_fd);
+	    continue;
+	}
+
+	if (r->file) {
+	    r->src_fd = open(r->file, r->mode, 0666);
+
+	    if (r->src_fd < 0) {
+		G_warning(_("G_spawn: unable to open file %s"), r->file);
+		_exit(127);
+	    }
+
+	    handles[r->dst_fd] = get_handle(r->src_fd);
+
+	    close(r->src_fd);
+
+	}
+	else if (r->src_fd >= 0) {
+	    handles[r->dst_fd] = get_handle(r->src_fd);
+	}
+	else
+	    handles[r->dst_fd] = INVALID_HANDLE_VALUE;
+    }
+}
+
+static void add_binding(const char **env, int *pnum, const struct binding *b)
 {
-    if (num_redirects > 0)
-	G_fatal_error
-	    ("G_spawn_ex: redirection not (yet) supported on Windows");
+    char *str = G_malloc(strlen(b->var) + strlen(b->val) + 2);
+    int n = *pnum;
+    int i;
+
+    sprintf(str, "%s=%s", b->var, b->val);
+
+    for (i = 0; i < n; i++)
+	if (G_strcasecmp(env[i], b->var) == 0) {
+	    env[i] = str;
+	    return;
+	}
+
+    env[n++] = str;
+    *pnum = n;
 }
 
-static char **do_bindings(char **env, struct binding *bindings,
-			  int num_bindings)
+static const char **do_bindings(const struct binding *bindings, int num_bindings)
 {
-    if (num_bindings > 0)
-	G_fatal_error
-	    ("G_spawn_ex: redirection not (yet) supported on Windows");
+    const char **newenv;
+    int i, n;
 
-    return env;
+    for (i = 0; _environ[i]; i++)
+	;
+    n = i;
+
+    newenv = G_malloc((num_bindings + n + 1) * sizeof(char *));
+
+    for (i = 0; i < n; i++)
+	newenv[i] = _environ[i];
+
+    for (i = 0; i < num_bindings; i++)
+	add_binding(newenv, &n, &bindings[i]);
+
+    return newenv;
 }
 
-static int do_spawn(const char *command)
+static int do_spawn(struct spawn *sp, const char *command)
 {
-    char **env;
+    HANDLE handles[3];
+    const char **env;
     int status;
 
-    do_redirects(redirects, num_redirects);
-    env = do_bindings(_environ, bindings, num_bindings);
+    do_redirects(sp->redirects, sp->num_redirects, handles);
+    env = do_bindings(sp->bindings, sp->num_bindings);
 
-    status =
-	spawnvpe(background ? _P_NOWAIT : _P_WAIT, command, (char **)args,
-		 env);
+    status = win_spawn(command, sp->args, env, sp->directory, handles, sp->background, 1);
 
-    if (!background && status < 0)
+    if (!sp->background && status < 0)
 	G_warning(_("Unable to execute command"));
 
     return status;
@@ -245,13 +489,13 @@
 
 #else /* __MINGW32__ */
 
-static int undo_signals(struct signal *signals, int num_signals, int which)
+static int undo_signals(const struct signal *signals, int num_signals, int which)
 {
     int error = 0;
     int i;
 
     for (i = num_signals - 1; i >= 0; i--) {
-	struct signal *s = &signals[i];
+	const struct signal *s = &signals[i];
 
 	if (s->which != which)
 	    continue;
@@ -378,60 +622,59 @@
     }
 }
 
-static void do_bindings(struct binding *bindings, int num_bindings)
+static void do_bindings(const struct binding *bindings, int num_bindings)
 {
     int i;
 
     for (i = 0; i < num_bindings; i++) {
-	struct binding *b = &bindings[i];
-	static char *str = NULL;
+	const struct binding *b = &bindings[i];
+	char *str = G_malloc(strlen(b->var) + strlen(b->val) + 2);
 
-	str = G_realloc(str, strlen(b->var) + strlen(b->val) + 2);
 	sprintf(str, "%s=%s", b->var, b->val);
 	putenv(str);
     }
 }
 
-static int do_spawn(const char *command)
+static int do_spawn(struct spawn *sp, const char *command)
 {
     int status = -1;
     pid_t pid;
 
-    if (!do_signals(signals, num_signals, SST_PRE))
+    if (!do_signals(sp->signals, sp->num_signals, SST_PRE))
 	return status;
 
     pid = fork();
     if (pid < 0) {
 	G_warning(_("Unable to create a new process"));
-	undo_signals(signals, num_signals, SST_PRE);
+	undo_signals(sp->signals, sp->num_signals, SST_PRE);
 
 	return status;
     }
 
     if (pid == 0) {
-	if (!undo_signals(signals, num_signals, SST_PRE))
+	if (!undo_signals(sp->signals, sp->num_signals, SST_PRE))
 	    _exit(127);
 
-	if (!do_signals(signals, num_signals, SST_CHILD))
+	if (!do_signals(sp->signals, sp->num_signals, SST_CHILD))
 	    _exit(127);
 
-	if (directory)
-	    if (chdir(directory) < 0) {
-		G_warning(_("Unable to change directory to %s"), directory);
+	if (sp->directory)
+	    if (chdir(sp->directory) < 0) {
+		G_warning(_("Unable to change directory to %s"), sp->directory);
 		_exit(127);
 	    }
 
-	do_redirects(redirects, num_redirects);
-	do_bindings(bindings, num_bindings);
+	do_redirects(sp->redirects, sp->num_redirects);
+	do_bindings(sp->bindings, sp->num_bindings);
 
-	execvp(command, (char **)args);
+	execvp(command, (char **)sp->args);
 	G_warning(_("Unable to execute command"));
 	_exit(127);
     }
 
-    do_signals(signals, num_signals, SST_POST);
+    do_signals(sp->signals, sp->num_signals, SST_POST);
 
-    if (background)
+    if (sp->background)
 	status = (int)pid;
     else {
 	pid_t n;
@@ -444,171 +687,158 @@
 	    status = -1;
     }
 
-    undo_signals(signals, num_signals, SST_POST);
-    undo_signals(signals, num_signals, SST_PRE);
+    undo_signals(sp->signals, sp->num_signals, SST_POST);
+    undo_signals(sp->signals, sp->num_signals, SST_PRE);
 
     return status;
 }
 
 #endif /* __MINGW32__ */
 
-static void begin_spawn(void)
+static void begin_spawn(struct spawn *sp)
 {
-    num_args = 0;
-    num_redirects = 0;
-    num_signals = 0;
-    num_bindings = 0;
-    background = 0;
-    directory = NULL;
+    sp->num_args = 0;
+    sp->num_redirects = 0;
+    sp->num_signals = 0;
+    sp->num_bindings = 0;
+    sp->background = 0;
+    sp->directory = NULL;
 }
 
-#define NEXT_ARG(var, type) ((type) *(var)++)
+#define NEXT_ARG(var, type) ((type) (ssize_t) *(var)++)
 
-static void parse_argvec(const char **va)
+static void parse_argvec(struct spawn *sp, const char **va)
 {
     for (;;) {
 	const char *arg = NEXT_ARG(va, const char *);
 	const char *var, *val;
 
-	switch ((int)arg) {
-	case 0:
-	    args[num_args++] = NULL;
+	if (!arg) {
+	    sp->args[sp->num_args++] = NULL;
 	    break;
-	case ((int)SF_REDIRECT_FILE):
-	    redirects[num_redirects].dst_fd = NEXT_ARG(va, int);
+	}
+	else if (arg == SF_REDIRECT_FILE) {
+	    sp->redirects[sp->num_redirects].dst_fd = NEXT_ARG(va, int);
 
-	    redirects[num_redirects].src_fd = -1;
-	    redirects[num_redirects].mode = NEXT_ARG(va, int);
-	    redirects[num_redirects].file = NEXT_ARG(va, const char *);
+	    sp->redirects[sp->num_redirects].src_fd = -1;
+	    sp->redirects[sp->num_redirects].mode = NEXT_ARG(va, int);
+	    sp->redirects[sp->num_redirects].file = NEXT_ARG(va, const char *);
 
-	    num_redirects++;
-	    break;
-	case ((int)SF_REDIRECT_DESCRIPTOR):
-	    redirects[num_redirects].dst_fd = NEXT_ARG(va, int);
-	    redirects[num_redirects].src_fd = NEXT_ARG(va, int);
+	    sp->num_redirects++;
+	}
+	else if (arg == SF_REDIRECT_DESCRIPTOR) {
+	    sp->redirects[sp->num_redirects].dst_fd = NEXT_ARG(va, int);
+	    sp->redirects[sp->num_redirects].src_fd = NEXT_ARG(va, int);
 
-	    redirects[num_redirects].file = NULL;
-	    num_redirects++;
-	    break;
-	case ((int)SF_CLOSE_DESCRIPTOR):
-	    redirects[num_redirects].dst_fd = NEXT_ARG(va, int);
+	    sp->redirects[sp->num_redirects].file = NULL;
+	    sp->num_redirects++;
+	}
+	else if (arg == SF_CLOSE_DESCRIPTOR) {
+	    sp->redirects[sp->num_redirects].dst_fd = NEXT_ARG(va, int);
 
-	    redirects[num_redirects].src_fd = -1;
-	    redirects[num_redirects].file = NULL;
-	    num_redirects++;
-	    break;
-	case ((int)SF_SIGNAL):
-	    signals[num_signals].which = NEXT_ARG(va, int);
-	    signals[num_signals].action = NEXT_ARG(va, int);
-	    signals[num_signals].signum = NEXT_ARG(va, int);
+	    sp->redirects[sp->num_redirects].src_fd = -1;
+	    sp->redirects[sp->num_redirects].file = NULL;
+	    sp->num_redirects++;
+	}
+	else if (arg == SF_SIGNAL) {
+	    sp->signals[sp->num_signals].which = NEXT_ARG(va, int);
+	    sp->signals[sp->num_signals].action = NEXT_ARG(va, int);
+	    sp->signals[sp->num_signals].signum = NEXT_ARG(va, int);
 
-	    signals[num_signals].valid = 0;
-	    num_signals++;
-	    break;
-	case ((int)SF_VARIABLE):
+	    sp->signals[sp->num_signals].valid = 0;
+	    sp->num_signals++;
+	}
+	else if (arg == SF_VARIABLE) {
 	    var = NEXT_ARG(va, const char *);
 
 	    val = getenv(var);
-	    args[num_args++] = val ? val : "";
-	    break;
-	case ((int)SF_BINDING):
-	    bindings[num_bindings].var = NEXT_ARG(va, const char *);
-	    bindings[num_bindings].val = NEXT_ARG(va, const char *);
+	    sp->args[sp->num_args++] = val ? val : "";
+	}
+	else if (arg == SF_BINDING) {
+	    sp->bindings[sp->num_bindings].var = NEXT_ARG(va, const char *);
+	    sp->bindings[sp->num_bindings].val = NEXT_ARG(va, const char *);
 
-	    num_bindings++;
-	    break;
-	case ((int)SF_BACKGROUND):
-	    background = 1;
-	    break;
-	case ((int)SF_DIRECTORY):
-	    directory = NEXT_ARG(va, const char *);
+	    sp->num_bindings++;
+	}
+	else if (arg == SF_BACKGROUND) {
+	    sp->background = 1;
+	}
+	else if (arg == SF_DIRECTORY) {
+	    sp->directory = NEXT_ARG(va, const char *);
 
-	    break;
-	case ((int)SF_ARGVEC):
-	    parse_argvec(NEXT_ARG(va, const char **));
-
-	    break;
-	default:
-	    args[num_args++] = arg;
-	    break;
 	}
-
-	if (!arg)
-	    break;
+	else if (arg == SF_ARGVEC) {
+	    parse_argvec(sp, NEXT_ARG(va, const char **));
+	}
+	else
+	    sp->args[sp->num_args++] = arg;
     }
 }
 
-static void parse_arglist(va_list va)
+static void parse_arglist(struct spawn *sp, va_list va)
 {
     for (;;) {
 	const char *arg = va_arg(va, const char *);
 	const char *var, *val;
 
-	switch ((int)arg) {
-	case 0:
-	    args[num_args++] = NULL;
+	if (!arg) {
+	    sp->args[sp->num_args++] = NULL;
 	    break;
-	case ((int)SF_REDIRECT_FILE):
-	    redirects[num_redirects].dst_fd = va_arg(va, int);
+	}
+	else if (arg == SF_REDIRECT_FILE) {
+	    sp->redirects[sp->num_redirects].dst_fd = va_arg(va, int);
 
-	    redirects[num_redirects].src_fd = -1;
-	    redirects[num_redirects].mode = va_arg(va, int);
-	    redirects[num_redirects].file = va_arg(va, const char *);
+	    sp->redirects[sp->num_redirects].src_fd = -1;
+	    sp->redirects[sp->num_redirects].mode = va_arg(va, int);
+	    sp->redirects[sp->num_redirects].file = va_arg(va, const char *);
 
-	    num_redirects++;
-	    break;
-	case ((int)SF_REDIRECT_DESCRIPTOR):
-	    redirects[num_redirects].dst_fd = va_arg(va, int);
-	    redirects[num_redirects].src_fd = va_arg(va, int);
+	    sp->num_redirects++;
+	}
+	else if (arg == SF_REDIRECT_DESCRIPTOR) {
+	    sp->redirects[sp->num_redirects].dst_fd = va_arg(va, int);
+	    sp->redirects[sp->num_redirects].src_fd = va_arg(va, int);
 
-	    redirects[num_redirects].file = NULL;
-	    num_redirects++;
-	    break;
-	case ((int)SF_CLOSE_DESCRIPTOR):
-	    redirects[num_redirects].dst_fd = va_arg(va, int);
+	    sp->redirects[sp->num_redirects].file = NULL;
+	    sp->num_redirects++;
+	}
+	else if (arg == SF_CLOSE_DESCRIPTOR) {
+	    sp->redirects[sp->num_redirects].dst_fd = va_arg(va, int);
 
-	    redirects[num_redirects].src_fd = -1;
-	    redirects[num_redirects].file = NULL;
-	    num_redirects++;
-	    break;
-	case ((int)SF_SIGNAL):
-	    signals[num_signals].which = va_arg(va, int);
-	    signals[num_signals].action = va_arg(va, int);
-	    signals[num_signals].signum = va_arg(va, int);
+	    sp->redirects[sp->num_redirects].src_fd = -1;
+	    sp->redirects[sp->num_redirects].file = NULL;
+	    sp->num_redirects++;
+	}
+	else if (arg == SF_SIGNAL) {
+	    sp->signals[sp->num_signals].which = va_arg(va, int);
+	    sp->signals[sp->num_signals].action = va_arg(va, int);
+	    sp->signals[sp->num_signals].signum = va_arg(va, int);
 
-	    signals[num_signals].valid = 0;
-	    num_signals++;
-	    break;
-	case ((int)SF_VARIABLE):
+	    sp->signals[sp->num_signals].valid = 0;
+	    sp->num_signals++;
+	}
+	else if (arg == SF_VARIABLE) {
 	    var = va_arg(va, char *);
 
 	    val = getenv(var);
-	    args[num_args++] = val ? val : "";
-	    break;
-	case ((int)SF_BINDING):
-	    bindings[num_bindings].var = va_arg(va, const char *);
-	    bindings[num_bindings].val = va_arg(va, const char *);
+	    sp->args[sp->num_args++] = val ? val : "";
+	}
+	else if (arg == SF_BINDING) {
+	    sp->bindings[sp->num_bindings].var = va_arg(va, const char *);
+	    sp->bindings[sp->num_bindings].val = va_arg(va, const char *);
 
-	    num_bindings++;
-	    break;
-	case ((int)SF_BACKGROUND):
-	    background = 1;
-	    break;
-	case ((int)SF_DIRECTORY):
-	    directory = va_arg(va, const char *);
-
-	    break;
-	case ((int)SF_ARGVEC):
-	    parse_argvec(va_arg(va, const char **));
-
-	    break;
-	default:
-	    args[num_args++] = arg;
-	    break;
+	    sp->num_bindings++;
 	}
-
-	if (!arg)
-	    break;
+	else if (arg == SF_BACKGROUND) {
+	    sp->background = 1;
+	}
+	else if (arg == SF_DIRECTORY) {
+	    sp->directory = va_arg(va, const char *);
+	}
+	else if (arg == SF_ARGVEC) {
+	    parse_argvec(sp, va_arg(va, const char **));
+	}
+	else
+	    sp->args[sp->num_args++] = arg;
     }
 }
 
@@ -624,11 +854,13 @@
  */
 int G_vspawn_ex(const char *command, const char **args)
 {
-    begin_spawn();
+    struct spawn sp;
 
-    parse_argvec(args);
+    begin_spawn(&sp);
 
-    return do_spawn(command);
+    parse_argvec(&sp, args);
+
+    return do_spawn(&sp, command);
 }
 
 /**
@@ -643,13 +875,54 @@
 
 int G_spawn_ex(const char *command, ...)
 {
+    struct spawn sp;
     va_list va;
 
-    begin_spawn();
+    begin_spawn(&sp);
 
     va_start(va, command);
-    parse_arglist(va);
+    parse_arglist(&sp, va);
     va_end(va);
 
-    return do_spawn(command);
+    return do_spawn(&sp, command);
 }
+
+/**
+ * \brief Spawn new process based on <b>command</b>.
+ *
+ * \param[in] command
+ * \return -1 on error
+ * \return process status on success
+ */
+
+int G_spawn(const char *command, ...)
+{
+    const char *args[MAX_ARGS];
+    int num_args = 0, i;
+    va_list va;
+    int status = -1;
+
+    va_start(va, command);
+
+    for (i = 0; ; i++) {
+	const char *arg = va_arg(va, const char *);
+	args[num_args++] = arg;
+	if (!arg)
+	    break;
+    }
+
+    va_end(va);
+
+    status = G_spawn_ex(
+	command,
+#ifndef __MINGW32__
+	SF_SIGNAL, SST_PRE, SSA_IGNORE, SIGINT,
+	SF_SIGNAL, SST_PRE, SSA_IGNORE, SIGQUIT,
+	SF_SIGNAL, SST_PRE, SSA_BLOCK, SIGCHLD,
+#endif
+	SF_ARGVEC, args,
+	NULL);
+
+    return status;
+}
+



More information about the grass-commit mailing list