mirror of
https://github.com/janet-lang/janet
synced 2025-01-26 15:16:51 +00:00
Make os/execute not leak memory on panics.
Since many calls can panic, it's best to only use scratch memory for temporary values.
This commit is contained in:
parent
c150f2f2c1
commit
e697cc3811
@ -129,10 +129,7 @@ static char **os_execute_env(int32_t argc, const Janet *argv) {
|
|||||||
char **envp = NULL;
|
char **envp = NULL;
|
||||||
if (argc > 2) {
|
if (argc > 2) {
|
||||||
JanetDictView dict = janet_getdictionary(argv, 2);
|
JanetDictView dict = janet_getdictionary(argv, 2);
|
||||||
envp = malloc(sizeof(char *) * (dict.len + 1));
|
envp = janet_smalloc(sizeof(char *) * (dict.len + 1));
|
||||||
if (NULL == envp) {
|
|
||||||
JANET_OUT_OF_MEMORY;
|
|
||||||
}
|
|
||||||
int32_t j = 0;
|
int32_t j = 0;
|
||||||
for (int32_t i = 0; i < dict.cap; i++) {
|
for (int32_t i = 0; i < dict.cap; i++) {
|
||||||
const JanetKV *kv = dict.kvs + i;
|
const JanetKV *kv = dict.kvs + i;
|
||||||
@ -151,10 +148,7 @@ static char **os_execute_env(int32_t argc, const Janet *argv) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (skip) continue;
|
if (skip) continue;
|
||||||
char *envitem = malloc(klen + vlen + 2);
|
char *envitem = janet_smalloc(klen + vlen + 2);
|
||||||
if (NULL == envitem) {
|
|
||||||
JANET_OUT_OF_MEMORY;
|
|
||||||
}
|
|
||||||
memcpy(envitem, keys, klen);
|
memcpy(envitem, keys, klen);
|
||||||
envitem[klen] = '=';
|
envitem[klen] = '=';
|
||||||
memcpy(envitem + klen + 1, vals, vlen);
|
memcpy(envitem + klen + 1, vals, vlen);
|
||||||
@ -171,16 +165,16 @@ static void os_execute_cleanup(char **envp, const char **child_argv) {
|
|||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
(void) child_argv;
|
(void) child_argv;
|
||||||
#else
|
#else
|
||||||
free((void *)child_argv);
|
janet_sfree((void *)child_argv);
|
||||||
#endif
|
#endif
|
||||||
if (NULL != envp) {
|
if (NULL != envp) {
|
||||||
char **envitem = envp;
|
char **envitem = envp;
|
||||||
while (*envitem != NULL) {
|
while (*envitem != NULL) {
|
||||||
free(*envitem);
|
janet_sfree(*envitem);
|
||||||
envitem++;
|
envitem++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
free(envp);
|
janet_sfree(envp);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
@ -313,13 +307,9 @@ static Janet os_execute(int32_t argc, Janet *argv) {
|
|||||||
return janet_wrap_integer(status);
|
return janet_wrap_integer(status);
|
||||||
#else
|
#else
|
||||||
|
|
||||||
const char **child_argv = malloc(sizeof(char *) * (exargs.len + 1));
|
const char **child_argv = janet_smalloc(sizeof(char *) * (exargs.len + 1));
|
||||||
if (NULL == child_argv) {
|
for (int32_t i = 0; i < exargs.len; i++)
|
||||||
JANET_OUT_OF_MEMORY;
|
|
||||||
}
|
|
||||||
for (int32_t i = 0; i < exargs.len; i++) {
|
|
||||||
child_argv[i] = janet_getcstring(exargs.items, i);
|
child_argv[i] = janet_getcstring(exargs.items, i);
|
||||||
}
|
|
||||||
child_argv[exargs.len] = NULL;
|
child_argv[exargs.len] = NULL;
|
||||||
/* Coerce to form that works for spawn. I'm fairly confident no implementation
|
/* Coerce to form that works for spawn. I'm fairly confident no implementation
|
||||||
* of posix_spawn would modify the argv array passed in. */
|
* of posix_spawn would modify the argv array passed in. */
|
||||||
|
Loading…
Reference in New Issue
Block a user