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, ...
This commit is contained in:
Waldemar Zimpel 2024-09-26 03:37:19 +02:00 committed by David Goulet
parent 93df26b11a
commit 6feaea8fa4
7 changed files with 102 additions and 29 deletions

3
changes/thread-memleak Normal file
View File

@ -0,0 +1,3 @@
o Minor bugfixes (memory):
- Fix memory leaks of the CPU worker code during shutdown. Fixes bug 833;
bugfix on 0.3.5.1-alpha.

View File

@ -1,7 +1,7 @@
/* Copyright (c) 2001 Matej Pfajfar.
* Copyright (c) 2001-2004, Roger Dingledine.
* Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
* Copyright (c) 2007-2021, The Tor Project, Inc. */
* Copyright (c) 2007-2024, The Tor Project, Inc. */
/* See LICENSE for licensing information */
/**
@ -19,6 +19,7 @@
#include "app/main/subsysmgr.h"
#include "core/mainloop/connection.h"
#include "core/mainloop/mainloop_pubsub.h"
#include "core/mainloop/cpuworker.h"
#include "core/or/channeltls.h"
#include "core/or/circuitlist.h"
#include "core/or/circuitmux_ewma.h"
@ -112,6 +113,7 @@ tor_free_all(int postfork)
if (!postfork) {
evdns_shutdown(1);
}
cpuworker_free_all();
geoip_free_all();
geoip_stats_free_all();
routerlist_free_all();

View File

@ -1,6 +1,6 @@
/* Copyright (c) 2003-2004, Roger Dingledine.
* Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
* Copyright (c) 2007-2021, The Tor Project, Inc. */
* Copyright (c) 2007-2024, The Tor Project, Inc. */
/* See LICENSE for licensing information */
/**
@ -74,7 +74,6 @@ worker_state_free_void(void *arg)
worker_state_free_(arg);
}
static replyqueue_t *replyqueue = NULL;
static threadpool_t *threadpool = NULL;
static uint32_t total_pending_tasks = 0;
@ -120,31 +119,33 @@ cpuworker_consensus_has_changed(const networkstatus_t *ns)
void
cpuworker_init(void)
{
if (!replyqueue) {
replyqueue = replyqueue_new(0);
}
if (!threadpool) {
/*
In our threadpool implementation, half the threads are permissive and
half are strict (when it comes to running lower-priority tasks). So we
always make sure we have at least two threads, so that there will be at
least one thread of each kind.
*/
const int n_threads = MAX(get_num_cpus(get_options()), 2);
threadpool = threadpool_new(n_threads,
replyqueue,
worker_state_new,
worker_state_free_void,
NULL);
/*
In our threadpool implementation, half the threads are permissive and
half are strict (when it comes to running lower-priority tasks). So we
always make sure we have at least two threads, so that there will be at
least one thread of each kind.
*/
const int n_threads = MAX(get_num_cpus(get_options()), 2);
threadpool = threadpool_new(n_threads,
replyqueue_new(0),
worker_state_new,
worker_state_free_void,
NULL);
int r = threadpool_register_reply_event(threadpool, NULL);
int r = threadpool_register_reply_event(threadpool, NULL);
tor_assert(r == 0);
}
tor_assert(r == 0);
set_max_pending_tasks(NULL);
}
/** Free all resources allocated by cpuworker. */
void
cpuworker_free_all(void)
{
threadpool_free(threadpool);
}
/** Return the number of threads configured for our CPU worker. */
unsigned int
cpuworker_get_n_threads(void)

View File

@ -1,7 +1,7 @@
/* Copyright (c) 2001 Matej Pfajfar.
* Copyright (c) 2001-2004, Roger Dingledine.
* Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
* Copyright (c) 2007-2021, The Tor Project, Inc. */
* Copyright (c) 2007-2024, The Tor Project, Inc. */
/* See LICENSE for licensing information */
/**
@ -13,6 +13,7 @@
#define TOR_CPUWORKER_H
void cpuworker_init(void);
void cpuworker_free_all(void);
void cpuworkers_rotate_keyinfo(void);
void cpuworker_consensus_has_changed(const networkstatus_t *ns);

View File

@ -1,4 +1,4 @@
/* Copyright (c) 2009-2021, The Tor Project, Inc. */
/* Copyright (c) 2009-2024, The Tor Project, Inc. */
/* See LICENSE for licensing information */
/**
@ -19,6 +19,7 @@ void configure_libevent_logging(void);
void suppress_libevent_log_msg(const char *msg);
#define tor_event_new event_new
#define tor_event_del event_del
#define tor_evtimer_new evtimer_new
#define tor_evsignal_new evsignal_new
#define tor_evdns_add_server_port(sock, tcp, cb, data) \

View File

@ -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
}
@ -363,6 +365,15 @@ workerthread_new(int32_t lower_priority_chance,
return thr;
}
/**
* 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
@ -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

View File

@ -1,4 +1,4 @@
/* Copyright (c) 2013-2021, The Tor Project, Inc. */
/* Copyright (c) 2013-2024, The Tor Project, Inc. */
/* See LICENSE for licensing information */
/**
@ -58,6 +58,7 @@ threadpool_t *threadpool_new(int n_threads,
void *(*new_thread_state_fn)(void*),
void (*free_thread_state_fn)(void*),
void *arg);
void threadpool_free(threadpool_t *tp);
replyqueue_t *threadpool_get_replyqueue(threadpool_t *tp);
replyqueue_t *replyqueue_new(uint32_t alertsocks_flags);