diff --git a/src/core/os.c b/src/core/os.c index bf5673b7..4d3a3448 100644 --- a/src/core/os.c +++ b/src/core/os.c @@ -624,12 +624,99 @@ JANET_CORE_FN(os_proc_wait, #endif } +struct keyword_signal { + const char *keyword; + int signal; +}; + +#ifndef JANET_WINDOWS +static const struct keyword_signal signal_keywords[] = { +#ifdef SIGKILL + {"kill", SIGKILL}, +#endif + {"int", SIGINT}, + {"abrt", SIGABRT}, + {"fpe", SIGFPE}, + {"ill", SIGILL}, + {"segv", SIGSEGV}, +#ifdef SIGTERM + {"term", SIGTERM}, +#endif +#ifdef SIGARLM + {"alrm", SIGALRM}, +#endif +#ifdef SIGHUP + {"hup", SIGHUP}, +#endif +#ifdef SIGPIPE + {"pipe", SIGPIPE}, +#endif +#ifdef SIGQUIT + {"quit", SIGQUIT}, +#endif +#ifdef SIGUSR1 + {"usr1", SIGUSR1}, +#endif +#ifdef SIGUSR2 + {"usr2", SIGUSR2}, +#endif +#ifdef SIGCHLD + {"chld", SIGCHLD}, +#endif +#ifdef SIGCONT + {"cont", SIGCONT}, +#endif +#ifdef SIGSTOP + {"stop", SIGSTOP}, +#endif +#ifdef SIGTSTP + {"tstp", SIGTSTP}, +#endif +#ifdef SIGTTIN + {"ttin", SIGTTIN}, +#endif +#ifdef SIGTTOU + {"ttou", SIGTTOU}, +#endif +#ifdef SIGBUS + {"bus", SIGBUS}, +#endif +#ifdef SIGPOLL + {"poll", SIGPOLL}, +#endif +#ifdef SIGPROF + {"prof", SIGPROF}, +#endif +#ifdef SIGSYS + {"sys", SIGSYS}, +#endif +#ifdef SIGTRAP + {"trap", SIGTRAP}, +#endif +#ifdef SIGURG + {"urg", SIGURG}, +#endif +#ifdef SIGVTALRM + {"vtlarm", SIGVTALRM}, +#endif +#ifdef SIGXCPU + {"xcpu", SIGXCPU}, +#endif +#ifdef SIGXFSZ + {"xfsz", SIGXFSZ}, +#endif + {NULL, 0}, +}; +#endif + JANET_CORE_FN(os_proc_kill, - "(os/proc-kill proc &opt wait)", + "(os/proc-kill proc &opt wait signal)", "Kill a subprocess by sending SIGKILL to it on posix systems, or by closing the process " "handle on windows. If `wait` is truthy, will wait for the process to finish and " - "returns the exit code. Otherwise, returns `proc`.") { - janet_arity(argc, 1, 2); + "returns the exit code. Otherwise, returns `proc`. If signal is specified send it instead." + "Signal keywords are named after their C counterparts but in lowercase with the leading " + "`SIG` stripped. Signals are ignored on windows.") { + janet_arity(argc, 1, 3); JanetProc *proc = janet_getabstract(argv, 0, &ProcAT); if (proc->flags & JANET_PROC_WAITED) { janet_panicf("cannot kill process that has already finished"); @@ -643,7 +730,22 @@ JANET_CORE_FN(os_proc_kill, CloseHandle(proc->pHandle); CloseHandle(proc->tHandle); #else - int status = kill(proc->pid, SIGKILL); + int signal = -1; + if (argc == 3) { + JanetKeyword signal_kw = janet_getkeyword(argv, 2); + const struct keyword_signal *ptr = signal_keywords; + while (ptr->keyword) { + if (!janet_cstrcmp(signal_kw, ptr->keyword)) { + signal = ptr->signal; + break; + } + ptr++; + } + if (signal == -1) { + janet_panic("undefined signal"); + } + } + int status = kill(proc->pid, signal == -1 ? SIGKILL : signal); if (status) { janet_panic(strerror(errno)); } diff --git a/test/suite0009.janet b/test/suite0009.janet index 6b61797a..99f5232e 100644 --- a/test/suite0009.janet +++ b/test/suite0009.janet @@ -52,6 +52,11 @@ (def retval (os/proc-wait p)) (assert (not= retval 24) "Process was *not* terminated by parent")) +(let [p (os/spawn [janet "-e" `(do (ev/sleep 30) (os/exit 24)`] :p)] + (os/proc-kill p false :term) + (def retval (os/proc-wait p)) + (assert (not= retval 24) "Process was *not* terminated by parent")) + # Parallel subprocesses (defn calc-1