aboutsummaryrefslogtreecommitdiff
path: root/src/lib/evloop/workqueue.c
diff options
context:
space:
mode:
authorWaldemar Zimpel <w.zimpel@dev.utilizer.de>2024-09-26 03:37:19 +0200
committerDavid Goulet <dgoulet@torproject.org>2024-10-10 09:55:46 -0400
commit6feaea8fa44d4594388232f4e104a7c656013e19 (patch)
tree4ded9a86ed8eb13c32d5feff7f7a64bbf0ab48c3 /src/lib/evloop/workqueue.c
parent93df26b11a67423e05c200cd93cba6398b9f2a49 (diff)
downloadtor-6feaea8fa44d4594388232f4e104a7c656013e19.tar.gz
tor-6feaea8fa44d4594388232f4e104a7c656013e19.zip
Fix: Memory leaks in cpuworker on shutdown
Resources allocated by cpuworker weren't being freed on clean shutdown. This applies for worker threads, worker thread pool, reply queue, reply event, ...
Diffstat (limited to 'src/lib/evloop/workqueue.c')
-rw-r--r--src/lib/evloop/workqueue.c72
1 files changed, 68 insertions, 4 deletions
diff --git a/src/lib/evloop/workqueue.c b/src/lib/evloop/workqueue.c
index 9a0c02fbd0..20b611f7cb 100644
--- a/src/lib/evloop/workqueue.c
+++ b/src/lib/evloop/workqueue.c
@@ -1,5 +1,5 @@
-/* copyright (c) 2013-2015, The Tor Project, Inc. */
+/* copyright (c) 2013-2024, The Tor Project, Inc. */
/* See LICENSE for licensing information */
/**
@@ -143,6 +143,8 @@ typedef struct workerthread_t {
} workerthread_t;
static void queue_reply(replyqueue_t *queue, workqueue_entry_t *work);
+static void workerthread_free(workerthread_t *thread);
+static void replyqueue_free(replyqueue_t *queue);
/** Allocate and return a new workqueue_entry_t, set up to run the function
* <b>fn</b> in the worker thread, and <b>reply_fn</b> in the main
@@ -355,7 +357,7 @@ workerthread_new(int32_t lower_priority_chance,
//LCOV_EXCL_START
tor_assert_nonfatal_unreached();
log_err(LD_GENERAL, "Can't launch worker thread.");
- tor_free(thr);
+ workerthread_free(thr);
return NULL;
//LCOV_EXCL_STOP
}
@@ -364,6 +366,15 @@ workerthread_new(int32_t lower_priority_chance,
}
/**
+ * Free up the resources allocated by a worker thread.
+ */
+static void
+workerthread_free(workerthread_t *thread)
+{
+ tor_free(thread);
+}
+
+/**
* Queue an item of work for a thread in a thread pool. The function
* <b>fn</b> will be run in a worker thread, and will receive as arguments the
* thread's state object, and the provided object <b>arg</b>. It must return
@@ -566,7 +577,7 @@ threadpool_new(int n_threads,
tor_assert_nonfatal_unreached();
tor_cond_uninit(&pool->condition);
tor_mutex_uninit(&pool->lock);
- tor_free(pool);
+ threadpool_free(pool);
return NULL;
//LCOV_EXCL_STOP
}
@@ -574,6 +585,39 @@ threadpool_new(int n_threads,
return pool;
}
+/**
+ * Free up the resources allocated by worker threads, worker thread pool, ...
+ */
+void
+threadpool_free(threadpool_t *pool)
+{
+ if (!pool)
+ return;
+
+ if (pool->threads) {
+ for (int i = 0; i != pool->n_threads; ++i)
+ workerthread_free(pool->threads[i]);
+
+ tor_free(pool->threads);
+ }
+
+ if (pool->update_args)
+ pool->free_update_arg_fn(pool->update_args);
+
+ if (pool->reply_event) {
+ tor_event_del(pool->reply_event);
+ tor_event_free(pool->reply_event);
+ }
+
+ if (pool->reply_queue)
+ replyqueue_free(pool->reply_queue);
+
+ if (pool->new_thread_state_arg)
+ pool->free_thread_state_fn(pool->new_thread_state_arg);
+
+ tor_free(pool);
+}
+
/** Return the reply queue associated with a given thread pool. */
replyqueue_t *
threadpool_get_replyqueue(threadpool_t *tp)
@@ -593,7 +637,7 @@ replyqueue_new(uint32_t alertsocks_flags)
rq = tor_malloc_zero(sizeof(replyqueue_t));
if (alert_sockets_create(&rq->alert, alertsocks_flags) < 0) {
//LCOV_EXCL_START
- tor_free(rq);
+ replyqueue_free(rq);
return NULL;
//LCOV_EXCL_STOP
}
@@ -604,6 +648,26 @@ replyqueue_new(uint32_t alertsocks_flags)
return rq;
}
+/**
+ * Free up the resources allocated by a reply queue.
+ */
+static void
+replyqueue_free(replyqueue_t *queue)
+{
+ if (!queue)
+ return;
+
+ workqueue_entry_t *work;
+
+ while (!TOR_TAILQ_EMPTY(&queue->answers)) {
+ work = TOR_TAILQ_FIRST(&queue->answers);
+ TOR_TAILQ_REMOVE(&queue->answers, work, next_work);
+ workqueue_entry_free(work);
+ }
+
+ tor_free(queue);
+}
+
/** Internal: Run from the libevent mainloop when there is work to handle in
* the reply queue handler. */
static void