Skip to content

Commit

Permalink
curl: Make socket callback during cleanup into no-op
Browse files Browse the repository at this point in the history
Because curl_multi_cleanup may invoke callbacks, we effectively have
some circular references going on here. See discussion in

curl/curl#14860

Basically what we do is the socket callback libcurl may invoke into a no-op when
we detect we're finalizing. The data structures are owned by this object and
not by the callbacks, and will be destroyed below. Note that
e.g. g_hash_table_unref() may itself invoke callbacks, which is where
some data is cleaned up.

Signed-off-by: Colin Walters <[email protected]>
Origin: upstream, 2024.8, commit:4d755a85225ea0a02d4580d088bb8a97138cb040
Bug: ostreedev/ostree#3299
Bug-Debian: https://bugs.debian.org/1082121
[smcv: Backport to 2022.7 by using gboolean instead of stdbool.h]
Signed-off-by: Simon McVittie <[email protected]>

Gbp-Pq: Name curl-Make-socket-callback-during-cleanup-into-no-op.patch
  • Loading branch information
cgwalters authored and smcv committed Oct 1, 2024
1 parent a743c71 commit 099095b
Showing 1 changed file with 14 additions and 0 deletions.
14 changes: 14 additions & 0 deletions src/libostree/ostree-fetcher-curl.c
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@ struct OstreeFetcher
char *proxy;
struct curl_slist *extra_headers;
int tmpdir_dfd;
gboolean finalizing; // Set if we're in the process of teardown
char *custom_user_agent;

GMainContext *mainctx;
Expand Down Expand Up @@ -174,6 +175,15 @@ _ostree_fetcher_finalize (GObject *object)
{
OstreeFetcher *self = OSTREE_FETCHER (object);

// Because curl_multi_cleanup may invoke callbacks, we effectively have
// some circular references going on here. See discussion in
// https://github.com/curl/curl/issues/14860
// Basically what we do is make most callbacks libcurl may invoke into no-ops when
// we detect we're finalizing. The data structures are owned by this object and
// not by the callbacks, and will be destroyed below. Note that
// e.g. g_hash_table_unref() may itself invoke callbacks, which is where
// some data is cleaned up.
self->finalizing = TRUE;
curl_multi_cleanup (self->multi);
g_free (self->remote_name);
g_free (self->tls_ca_db_path);
Expand Down Expand Up @@ -521,6 +531,10 @@ sock_cb (CURL *easy, curl_socket_t s, int what, void *cbp, void *sockp)
OstreeFetcher *fetcher = cbp;
SockInfo *fdp = (SockInfo*) sockp;

// We do nothing if we're in the process of teardown; see below.
if (fetcher->finalizing)
return 0;

if (what == CURL_POLL_REMOVE)
{
if (!g_hash_table_remove (fetcher->sockets, fdp))
Expand Down

0 comments on commit 099095b

Please sign in to comment.