mirror of
https://github.com/janet-lang/janet
synced 2025-01-13 09:00:26 +00:00
Fix os/spawn piping on windows and free handles on errors.
This commit is contained in:
parent
4c9624db64
commit
32bf70571a
9
.gitattributes
vendored
9
.gitattributes
vendored
@ -1 +1,10 @@
|
|||||||
*.janet linguist-language=Clojure
|
*.janet linguist-language=Clojure
|
||||||
|
|
||||||
|
*.janet text eol=lf
|
||||||
|
*.c text eol=lf
|
||||||
|
*.h text eol=lf
|
||||||
|
*.md text eol=lf
|
||||||
|
*.yml text eol=lf
|
||||||
|
*.build text eol=lf
|
||||||
|
*.txt text eol=lf
|
||||||
|
*.sh text eol=lf
|
||||||
|
125
src/core/os.c
125
src/core/os.c
@ -221,7 +221,8 @@ static char **os_execute_env(int32_t argc, const Janet *argv) {
|
|||||||
return envp;
|
return envp;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free memory from os_execute */
|
/* Free memory from os_execute. Not actually needed, but doesn't pressure the GC
|
||||||
|
in the happy path. */
|
||||||
static void os_execute_cleanup(char **envp, const char **child_argv) {
|
static void os_execute_cleanup(char **envp, const char **child_argv) {
|
||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
(void) child_argv;
|
(void) child_argv;
|
||||||
@ -414,33 +415,66 @@ static Janet os_proc_kill(int32_t argc, Janet *argv) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Create piped file for os/execute and os/spawn. */
|
static void swap_handles(JanetHandle *handles) {
|
||||||
static JanetFile *make_pipes(JanetHandle *handle, int reverse) {
|
JanetHandle temp = handles[0];
|
||||||
|
handles[0] = handles[1];
|
||||||
|
handles[1] = temp;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void close_handle(JanetHandle handle) {
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
CloseHandle(handle);
|
||||||
|
#else
|
||||||
|
close(handle);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Create piped file for os/execute and os/spawn. Need to be careful that we mark
|
||||||
|
the error flag if we can't create pipe and don't leak handles. *handle will be cleaned
|
||||||
|
up by the calling function. If everything goes well, *handle is owned by the calling function,
|
||||||
|
(if it is set) and the returned JanetFile owns the other end of the pipe, which will be closed
|
||||||
|
on GC or fclose. */
|
||||||
|
static JanetFile *make_pipes(JanetHandle *handle, int reverse, int *errflag) {
|
||||||
JanetHandle handles[2];
|
JanetHandle handles[2];
|
||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
if (!CreatePipe(handles, handles + 1, NULL, 0)) janet_panic("failed to create pipe");
|
SECURITY_ATTRIBUTES saAttr;
|
||||||
if (reverse) {
|
memset(&saAttr, 0, sizeof(saAttr));
|
||||||
JanetHandle temp = handles[0];
|
saAttr.nLength = sizeof(saAttr);
|
||||||
handles[0] = handles[1];
|
saAttr.bInheritHandle = TRUE;
|
||||||
handles[1] = temp;
|
if (!CreatePipe(handles, handles + 1, &saAttr, 0)) goto error_pipe;
|
||||||
}
|
if (reverse) swap_handles(handles);
|
||||||
|
/* Don't inherit the side of the pipe owned by this process */
|
||||||
|
if (!SetHandleInformation(handles[0], HANDLE_FLAG_INHERIT, 0)) goto error_set_handle_info;
|
||||||
*handle = handles[1];
|
*handle = handles[1];
|
||||||
int fd = _open_osfhandle((intptr_t) handles[0], reverse ? _O_WRONLY : _O_RDONLY);
|
int fd = _open_osfhandle((intptr_t) handles[0], reverse ? _O_WRONLY : _O_RDONLY);
|
||||||
if (fd == -1) janet_panic("could not create file for piping");
|
if (fd == -1) goto error_open_osfhandle;
|
||||||
FILE *f = _fdopen(fd, reverse ? "w" : "r");
|
FILE *f = _fdopen(fd, reverse ? "w" : "r");
|
||||||
if (NULL == f) janet_panic(strerror(errno));
|
if (NULL == f) goto error_fdopen;
|
||||||
return janet_makejfile(f, reverse ? JANET_FILE_WRITE : JANET_FILE_READ);
|
return janet_makejfile(f, reverse ? JANET_FILE_WRITE : JANET_FILE_READ);
|
||||||
|
error_fdopen:
|
||||||
|
_close(fd); /* we need to close the fake file descriptor instead of the handle, as ownership has been transfered. */
|
||||||
|
*errflag = 1;
|
||||||
|
return NULL;
|
||||||
|
error_set_handle_info:
|
||||||
|
error_open_osfhandle:
|
||||||
|
close_handle(handles[0]);
|
||||||
|
/* fallthrough */
|
||||||
|
error_pipe:
|
||||||
|
*errflag = 1;
|
||||||
|
return NULL;
|
||||||
#else
|
#else
|
||||||
if (pipe(handles)) janet_panic(strerror(errno));
|
if (pipe(handles)) goto error_pipe;
|
||||||
if (reverse) {
|
if (reverse) swap_handles(handles);
|
||||||
JanetHandle temp = handles[0];
|
|
||||||
handles[0] = handles[1];
|
|
||||||
handles[1] = temp;
|
|
||||||
}
|
|
||||||
*handle = handles[1];
|
*handle = handles[1];
|
||||||
FILE *f = fdopen(handles[0], reverse ? "w" : "r");
|
FILE *f = fdopen(handles[0], reverse ? "w" : "r");
|
||||||
if (NULL == f) janet_panic(strerror(errno));
|
if (NULL == f) goto error_fdopen;
|
||||||
return janet_makejfile(f, reverse ? JANET_FILE_WRITE : JANET_FILE_READ);
|
return janet_makejfile(f, reverse ? JANET_FILE_WRITE : JANET_FILE_READ);
|
||||||
|
error_fdopen:
|
||||||
|
close_handle(handles[0]);
|
||||||
|
/* fallthrough */
|
||||||
|
error_pipe:
|
||||||
|
*errflag = 1;
|
||||||
|
return NULL;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -490,6 +524,7 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, int is_async) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Get environment */
|
/* Get environment */
|
||||||
|
int use_environ = !janet_flag_at(flags, 0);
|
||||||
char **envp = os_execute_env(argc, argv);
|
char **envp = os_execute_env(argc, argv);
|
||||||
|
|
||||||
/* Get arguments */
|
/* Get arguments */
|
||||||
@ -501,6 +536,7 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, int is_async) {
|
|||||||
/* Optional stdio redirections */
|
/* Optional stdio redirections */
|
||||||
JanetFile *new_in = NULL, *new_out = NULL, *new_err = NULL;
|
JanetFile *new_in = NULL, *new_out = NULL, *new_err = NULL;
|
||||||
JanetHandle pipe_in = JANET_HANDLE_NONE, pipe_out = JANET_HANDLE_NONE, pipe_err = JANET_HANDLE_NONE;
|
JanetHandle pipe_in = JANET_HANDLE_NONE, pipe_out = JANET_HANDLE_NONE, pipe_err = JANET_HANDLE_NONE;
|
||||||
|
int pipe_errflag = 0; /* Track errors setting up pipes */
|
||||||
|
|
||||||
/* Get optional redirections */
|
/* Get optional redirections */
|
||||||
if (argc > 2) {
|
if (argc > 2) {
|
||||||
@ -509,34 +545,46 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, int is_async) {
|
|||||||
Janet maybe_stdout = janet_dictionary_get(tab.kvs, tab.cap, janet_ckeywordv("out"));
|
Janet maybe_stdout = janet_dictionary_get(tab.kvs, tab.cap, janet_ckeywordv("out"));
|
||||||
Janet maybe_stderr = janet_dictionary_get(tab.kvs, tab.cap, janet_ckeywordv("err"));
|
Janet maybe_stderr = janet_dictionary_get(tab.kvs, tab.cap, janet_ckeywordv("err"));
|
||||||
if (janet_keyeq(maybe_stdin, "pipe")) {
|
if (janet_keyeq(maybe_stdin, "pipe")) {
|
||||||
new_in = make_pipes(&pipe_in, 1);
|
new_in = make_pipes(&pipe_in, 1, &pipe_errflag);
|
||||||
} else if (!janet_checktype(maybe_stdin, JANET_NIL)) {
|
} else if (!janet_checktype(maybe_stdin, JANET_NIL)) {
|
||||||
new_in = janet_getjfile(&maybe_stdin, 0);
|
new_in = janet_getjfile(&maybe_stdin, 0);
|
||||||
}
|
}
|
||||||
if (janet_keyeq(maybe_stdout, "pipe")) {
|
if (janet_keyeq(maybe_stdout, "pipe")) {
|
||||||
new_out = make_pipes(&pipe_out, 0);
|
new_out = make_pipes(&pipe_out, 0, &pipe_errflag);
|
||||||
} else if (!janet_checktype(maybe_stdout, JANET_NIL)) {
|
} else if (!janet_checktype(maybe_stdout, JANET_NIL)) {
|
||||||
new_out = janet_getjfile(&maybe_stdout, 0);
|
new_out = janet_getjfile(&maybe_stdout, 0);
|
||||||
}
|
}
|
||||||
if (janet_keyeq(maybe_stderr, "err")) {
|
if (janet_keyeq(maybe_stderr, "err")) {
|
||||||
new_err = make_pipes(&pipe_err, 0);
|
new_err = make_pipes(&pipe_err, 0, &pipe_errflag);
|
||||||
} else if (!janet_checktype(maybe_stderr, JANET_NIL)) {
|
} else if (!janet_checktype(maybe_stderr, JANET_NIL)) {
|
||||||
new_err = janet_getjfile(&maybe_stderr, 0);
|
new_err = janet_getjfile(&maybe_stderr, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Clean up if any of the pipes have any issues */
|
||||||
|
if (pipe_errflag) {
|
||||||
|
if (pipe_in != JANET_HANDLE_NONE) close_handle(pipe_in);
|
||||||
|
if (pipe_out != JANET_HANDLE_NONE) close_handle(pipe_out);
|
||||||
|
if (pipe_err != JANET_HANDLE_NONE) close_handle(pipe_err);
|
||||||
|
janet_panic("failed to create pipes");
|
||||||
|
}
|
||||||
|
|
||||||
/* Result */
|
/* Result */
|
||||||
int status = 0;
|
int status = 0;
|
||||||
|
|
||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
|
|
||||||
HANDLE pHandle, tHandle;
|
HANDLE pHandle, tHandle;
|
||||||
|
SECURITY_ATTRIBUTES saAttr;
|
||||||
PROCESS_INFORMATION processInfo;
|
PROCESS_INFORMATION processInfo;
|
||||||
STARTUPINFO startupInfo;
|
STARTUPINFO startupInfo;
|
||||||
|
memset(&saAttr, 0, sizeof(saAttr));
|
||||||
memset(&processInfo, 0, sizeof(processInfo));
|
memset(&processInfo, 0, sizeof(processInfo));
|
||||||
memset(&startupInfo, 0, sizeof(startupInfo));
|
memset(&startupInfo, 0, sizeof(startupInfo));
|
||||||
startupInfo.cb = sizeof(startupInfo);
|
startupInfo.cb = sizeof(startupInfo);
|
||||||
startupInfo.dwFlags |= STARTF_USESTDHANDLES;
|
startupInfo.dwFlags |= STARTF_USESTDHANDLES;
|
||||||
|
saAttr.nLength = sizeof(saAttr);
|
||||||
|
saAttr.bInheritHandle = TRUE;
|
||||||
|
|
||||||
JanetBuffer *buf = os_exec_escape(exargs);
|
JanetBuffer *buf = os_exec_escape(exargs);
|
||||||
if (buf->count > 8191) {
|
if (buf->count > 8191) {
|
||||||
@ -550,47 +598,58 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, int is_async) {
|
|||||||
startupInfo.hStdInput = pipe_in;
|
startupInfo.hStdInput = pipe_in;
|
||||||
} else if (new_in != NULL) {
|
} else if (new_in != NULL) {
|
||||||
startupInfo.hStdInput = (HANDLE) _get_osfhandle(_fileno(new_in->file));
|
startupInfo.hStdInput = (HANDLE) _get_osfhandle(_fileno(new_in->file));
|
||||||
|
} else {
|
||||||
|
startupInfo.hStdInput = (HANDLE) _get_osfhandle(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (pipe_out != JANET_HANDLE_NONE) {
|
if (pipe_out != JANET_HANDLE_NONE) {
|
||||||
startupInfo.hStdInput = pipe_out;
|
startupInfo.hStdOutput = pipe_out;
|
||||||
} else if (new_out != NULL) {
|
} else if (new_out != NULL) {
|
||||||
startupInfo.hStdOutput = (HANDLE) _get_osfhandle(_fileno(new_out->file));
|
startupInfo.hStdOutput = (HANDLE) _get_osfhandle(_fileno(new_out->file));
|
||||||
|
} else {
|
||||||
|
startupInfo.hStdOutput = (HANDLE) _get_osfhandle(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pipe_err != JANET_HANDLE_NONE) {
|
if (pipe_err != JANET_HANDLE_NONE) {
|
||||||
startupInfo.hStdInput = pipe_err;
|
startupInfo.hStdError = pipe_err;
|
||||||
} else if (new_err != NULL) {
|
} else if (new_err != NULL) {
|
||||||
startupInfo.hStdError = (HANDLE) _get_osfhandle(_fileno(new_err->file));
|
startupInfo.hStdError = (HANDLE) _get_osfhandle(_fileno(new_err->file));
|
||||||
|
} else {
|
||||||
|
startupInfo.hStdError = (HANDLE) _get_osfhandle(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Use _spawn family of functions. */
|
/* Use _spawn family of functions. */
|
||||||
/* Windows docs say do this before any spawns. */
|
/* Windows docs say do this before any spawns. */
|
||||||
_flushall();
|
_flushall();
|
||||||
|
|
||||||
/* TODO - redirection, :p flag */
|
int cp_failed = 0;
|
||||||
if (!CreateProcess(janet_flag_at(flags, 1) ? NULL : path, /* NULL? */
|
if (!CreateProcess(janet_flag_at(flags, 1) ? NULL : path,
|
||||||
(char *) buf->data, /* Single CLI argument */
|
(char *) buf->data, /* Single CLI argument */
|
||||||
NULL, /* no proc inheritance */
|
&saAttr, /* no proc inheritance */
|
||||||
NULL, /* no thread inheritance */
|
&saAttr, /* no thread inheritance */
|
||||||
TRUE, /* handle inheritance */
|
TRUE, /* handle inheritance */
|
||||||
0, /* flags */
|
0, /* flags */
|
||||||
envp, /* pass in environment */
|
use_environ ? NULL : envp, /* pass in environment */
|
||||||
NULL, /* use parents starting directory */
|
NULL, /* use parents starting directory */
|
||||||
&startupInfo,
|
&startupInfo,
|
||||||
&processInfo)) {
|
&processInfo)) {
|
||||||
janet_panic("failed to create process");
|
cp_failed = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pipe_in != JANET_HANDLE_NONE) CloseHandle(pipe_in);
|
if (pipe_in != JANET_HANDLE_NONE) CloseHandle(pipe_in);
|
||||||
if (pipe_out != JANET_HANDLE_NONE) CloseHandle(pipe_out);
|
if (pipe_out != JANET_HANDLE_NONE) CloseHandle(pipe_out);
|
||||||
if (pipe_err != JANET_HANDLE_NONE) CloseHandle(pipe_err);
|
if (pipe_err != JANET_HANDLE_NONE) CloseHandle(pipe_err);
|
||||||
|
|
||||||
|
os_execute_cleanup(envp, NULL);
|
||||||
|
|
||||||
|
if (cp_failed) {
|
||||||
|
janet_panic("failed to create process");
|
||||||
|
}
|
||||||
|
|
||||||
pHandle = processInfo.hProcess;
|
pHandle = processInfo.hProcess;
|
||||||
tHandle = processInfo.hThread;
|
tHandle = processInfo.hThread;
|
||||||
|
|
||||||
os_execute_cleanup(envp, NULL);
|
|
||||||
|
|
||||||
/* Wait and cleanup immedaitely */
|
/* Wait and cleanup immedaitely */
|
||||||
if (!is_async) {
|
if (!is_async) {
|
||||||
DWORD code;
|
DWORD code;
|
||||||
@ -612,8 +671,6 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, int is_async) {
|
|||||||
|
|
||||||
/* Use posix_spawn to spawn new process */
|
/* Use posix_spawn to spawn new process */
|
||||||
|
|
||||||
int use_environ = !janet_flag_at(flags, 0);
|
|
||||||
|
|
||||||
if (use_environ) {
|
if (use_environ) {
|
||||||
janet_lock_environ();
|
janet_lock_environ();
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user