summaryrefslogtreecommitdiff
path: root/src/lib/process
diff options
context:
space:
mode:
authorAlexander Færøy <ahf@torproject.org>2018-11-23 03:44:59 +0100
committerNick Mathewson <nickm@torproject.org>2018-12-17 16:39:28 -0500
commit4f611a1df70d2c5e4cb6261f75c1b82c9ed04598 (patch)
tree8f6c89be62eb526fa3459782c2f4a1e5b5ec42a2 /src/lib/process
parent338137221c8bd89f6d611c0cd3bf7b8a85d02517 (diff)
downloadtor-4f611a1df70d2c5e4cb6261f75c1b82c9ed04598.tar.gz
tor-4f611a1df70d2c5e4cb6261f75c1b82c9ed04598.zip
Add process_terminate().
This patch adds support for process termination to the Process subsystem. See: https://bugs.torproject.org/28179
Diffstat (limited to 'src/lib/process')
-rw-r--r--src/lib/process/process.c20
-rw-r--r--src/lib/process/process.h1
-rw-r--r--src/lib/process/process_unix.c26
-rw-r--r--src/lib/process/process_unix.h1
-rw-r--r--src/lib/process/process_win32.c22
-rw-r--r--src/lib/process/process_win32.h1
6 files changed, 71 insertions, 0 deletions
diff --git a/src/lib/process/process.c b/src/lib/process/process.c
index ab19378a93..657b319f54 100644
--- a/src/lib/process/process.c
+++ b/src/lib/process/process.c
@@ -255,6 +255,26 @@ process_exec(process_t *process)
return status;
}
+/** Terminate the given process. Returns true on success,
+ * otherwise false. */
+bool
+process_terminate(process_t *process)
+{
+ tor_assert(process);
+
+ /* Terminating a non-running process isn't going to work. */
+ if (process_get_status(process) != PROCESS_STATUS_RUNNING)
+ return false;
+
+ log_debug(LD_PROCESS, "Terminating process");
+
+#ifndef _WIN32
+ return process_unix_terminate(process);
+#else
+ return process_win32_terminate(process);
+#endif
+}
+
/** Returns the unique process identifier for the given <b>process</b>. */
process_pid_t
process_get_pid(process_t *process)
diff --git a/src/lib/process/process.h b/src/lib/process/process.h
index 7fd6cf53d0..c6b733a065 100644
--- a/src/lib/process/process.h
+++ b/src/lib/process/process.h
@@ -66,6 +66,7 @@ void process_free_(process_t *process);
#define process_free(s) FREE_AND_NULL(process_t, process_free_, (s))
process_status_t process_exec(process_t *process);
+bool process_terminate(process_t *process);
process_pid_t process_get_pid(process_t *process);
diff --git a/src/lib/process/process_unix.c b/src/lib/process/process_unix.c
index 4f46bbd888..4a9aaa2edd 100644
--- a/src/lib/process/process_unix.c
+++ b/src/lib/process/process_unix.c
@@ -356,6 +356,32 @@ process_unix_exec(process_t *process)
return PROCESS_STATUS_RUNNING;
}
+/** Terminate the given process. Returns true on success, otherwise false. */
+bool
+process_unix_terminate(process_t *process)
+{
+ tor_assert(process);
+
+ process_unix_t *unix_process = process_get_unix_process(process);
+
+ /* All running processes should have a waitpid. */
+ if (BUG(unix_process->waitpid == NULL))
+ return false;
+
+ /* Send a SIGTERM to our child process. */
+ int ret;
+
+ ret = kill(unix_process->pid, SIGTERM);
+
+ if (ret == -1) {
+ log_warn(LD_PROCESS, "Unable to terminate process: %s",
+ strerror(errno));
+ return false;
+ }
+
+ return ret == 0;
+}
+
/** Returns the unique process identifier for the given <b>process</b>. */
process_pid_t
process_unix_get_pid(process_t *process)
diff --git a/src/lib/process/process_unix.h b/src/lib/process/process_unix.h
index 0474746b26..e17c59ea81 100644
--- a/src/lib/process/process_unix.h
+++ b/src/lib/process/process_unix.h
@@ -29,6 +29,7 @@ void process_unix_free_(process_unix_t *unix_process);
FREE_AND_NULL(process_unix_t, process_unix_free_, (s))
process_status_t process_unix_exec(struct process_t *process);
+bool process_unix_terminate(struct process_t *process);
process_pid_t process_unix_get_pid(struct process_t *process);
diff --git a/src/lib/process/process_win32.c b/src/lib/process/process_win32.c
index 7422493deb..7b18903c70 100644
--- a/src/lib/process/process_win32.c
+++ b/src/lib/process/process_win32.c
@@ -271,6 +271,28 @@ process_win32_exec(process_t *process)
return PROCESS_STATUS_RUNNING;
}
+/** Terminate the given process. Returns true on success, otherwise false. */
+bool
+process_win32_terminate(process_t *process)
+{
+ tor_assert(process);
+
+ process_win32_t *win32_process = process_get_win32_process(process);
+
+ /* Terminate our process. */
+ BOOL ret;
+
+ ret = TerminateProcess(win32_process->process_information.hProcess, 0);
+
+ if (! ret) {
+ log_warn(LD_PROCESS, "TerminateProcess() failed: %s",
+ format_win32_error(GetLastError()));
+ return false;
+ }
+
+ return true;
+}
+
/** Returns the unique process identifier for the given <b>process</b>. */
process_pid_t
process_win32_get_pid(process_t *process)
diff --git a/src/lib/process/process_win32.h b/src/lib/process/process_win32.h
index dbd264104c..9a42e6c713 100644
--- a/src/lib/process/process_win32.h
+++ b/src/lib/process/process_win32.h
@@ -33,6 +33,7 @@ void process_win32_init(void);
void process_win32_deinit(void);
process_status_t process_win32_exec(struct process_t *process);
+bool process_win32_terminate(struct process_t *process);
process_pid_t process_win32_get_pid(struct process_t *process);