gthreadedresolver: Combine closure structs for different lookup types

This introduces no functional changes, but will make a reorganisation of
the code simpler in the next commit.

Rather than dealing with three different closure types, this changes the
code to deal with one which is a tagged union of the three.

Signed-off-by: Philip Withnall <pwithnall@endlessos.org>
This commit is contained in:
Philip Withnall 2023-04-25 15:56:47 +01:00
parent 6e499764e4
commit 694394207c

View File

@ -71,24 +71,75 @@ g_resolver_error_from_addrinfo_error (gint err)
} }
typedef struct { typedef struct {
enum {
LOOKUP_BY_NAME,
LOOKUP_BY_ADDRESS,
LOOKUP_RECORDS,
} lookup_type;
union {
struct {
char *hostname; char *hostname;
int address_family; int address_family;
} lookup_by_name;
struct {
GInetAddress *address; /* (owned) */
} lookup_by_address;
struct {
char *rrname;
GResolverRecordType record_type;
} lookup_records;
};
} LookupData; } LookupData;
static LookupData * static LookupData *
lookup_data_new (const char *hostname, lookup_data_new_by_name (const char *hostname,
int address_family) int address_family)
{ {
LookupData *data = g_new (LookupData, 1); LookupData *data = g_new0 (LookupData, 1);
data->hostname = g_strdup (hostname); data->lookup_type = LOOKUP_BY_NAME;
data->address_family = address_family; data->lookup_by_name.hostname = g_strdup (hostname);
return data; data->lookup_by_name.address_family = address_family;
return g_steal_pointer (&data);
}
static LookupData *
lookup_data_new_by_address (GInetAddress *address)
{
LookupData *data = g_new0 (LookupData, 1);
data->lookup_type = LOOKUP_BY_ADDRESS;
data->lookup_by_address.address = g_object_ref (address);
return g_steal_pointer (&data);
}
static LookupData *
lookup_data_new_records (const gchar *rrname,
GResolverRecordType record_type)
{
LookupData *data = g_new0 (LookupData, 1);
data->lookup_type = LOOKUP_RECORDS;
data->lookup_records.rrname = g_strdup (rrname);
data->lookup_records.record_type = record_type;
return g_steal_pointer (&data);
} }
static void static void
lookup_data_free (LookupData *data) lookup_data_free (LookupData *data)
{ {
g_free (data->hostname); switch (data->lookup_type) {
case LOOKUP_BY_NAME:
g_free (data->lookup_by_name.hostname);
break;
case LOOKUP_BY_ADDRESS:
g_clear_object (&data->lookup_by_address.address);
break;
case LOOKUP_RECORDS:
g_free (data->lookup_records.rrname);
break;
default:
g_assert_not_reached ();
}
g_free (data); g_free (data);
} }
@ -99,7 +150,7 @@ do_lookup_by_name (GTask *task,
GCancellable *cancellable) GCancellable *cancellable)
{ {
LookupData *lookup_data = task_data; LookupData *lookup_data = task_data;
const char *hostname = lookup_data->hostname; const char *hostname = lookup_data->lookup_by_name.hostname;
struct addrinfo *res = NULL; struct addrinfo *res = NULL;
GList *addresses; GList *addresses;
gint retval; gint retval;
@ -115,7 +166,7 @@ do_lookup_by_name (GTask *task,
addrinfo_hints.ai_socktype = SOCK_STREAM; addrinfo_hints.ai_socktype = SOCK_STREAM;
addrinfo_hints.ai_protocol = IPPROTO_TCP; addrinfo_hints.ai_protocol = IPPROTO_TCP;
addrinfo_hints.ai_family = lookup_data->address_family; addrinfo_hints.ai_family = lookup_data->lookup_by_name.address_family;
retval = getaddrinfo (hostname, NULL, &addrinfo_hints, &res); retval = getaddrinfo (hostname, NULL, &addrinfo_hints, &res);
if (retval == 0) if (retval == 0)
@ -190,11 +241,11 @@ lookup_by_name (GResolver *resolver,
GList *addresses; GList *addresses;
LookupData *data; LookupData *data;
data = lookup_data_new (hostname, AF_UNSPEC); data = lookup_data_new_by_name (hostname, AF_UNSPEC);
task = g_task_new (resolver, cancellable, NULL, NULL); task = g_task_new (resolver, cancellable, NULL, NULL);
g_task_set_source_tag (task, lookup_by_name); g_task_set_source_tag (task, lookup_by_name);
g_task_set_name (task, "[gio] resolver lookup"); g_task_set_name (task, "[gio] resolver lookup");
g_task_set_task_data (task, data, (GDestroyNotify)lookup_data_free); g_task_set_task_data (task, g_steal_pointer (&data), (GDestroyNotify) lookup_data_free);
g_task_set_return_on_cancel (task, TRUE); g_task_set_return_on_cancel (task, TRUE);
g_task_run_in_thread_sync (task, do_lookup_by_name); g_task_run_in_thread_sync (task, do_lookup_by_name);
addresses = g_task_propagate_pointer (task, error); addresses = g_task_propagate_pointer (task, error);
@ -232,11 +283,11 @@ lookup_by_name_with_flags (GResolver *resolver,
GList *addresses; GList *addresses;
LookupData *data; LookupData *data;
data = lookup_data_new (hostname, flags_to_family (flags)); data = lookup_data_new_by_name (hostname, flags_to_family (flags));
task = g_task_new (resolver, cancellable, NULL, NULL); task = g_task_new (resolver, cancellable, NULL, NULL);
g_task_set_source_tag (task, lookup_by_name_with_flags); g_task_set_source_tag (task, lookup_by_name_with_flags);
g_task_set_name (task, "[gio] resolver lookup"); g_task_set_name (task, "[gio] resolver lookup");
g_task_set_task_data (task, data, (GDestroyNotify)lookup_data_free); g_task_set_task_data (task, g_steal_pointer (&data), (GDestroyNotify) lookup_data_free);
g_task_set_return_on_cancel (task, TRUE); g_task_set_return_on_cancel (task, TRUE);
g_task_run_in_thread_sync (task, do_lookup_by_name); g_task_run_in_thread_sync (task, do_lookup_by_name);
addresses = g_task_propagate_pointer (task, error); addresses = g_task_propagate_pointer (task, error);
@ -256,7 +307,7 @@ lookup_by_name_with_flags_async (GResolver *resolver,
GTask *task; GTask *task;
LookupData *data; LookupData *data;
data = lookup_data_new (hostname, flags_to_family (flags)); data = lookup_data_new_by_name (hostname, flags_to_family (flags));
task = g_task_new (resolver, cancellable, callback, user_data); task = g_task_new (resolver, cancellable, callback, user_data);
g_debug ("%s: starting new lookup for %s with GTask %p, LookupData %p", g_debug ("%s: starting new lookup for %s with GTask %p, LookupData %p",
@ -264,7 +315,7 @@ lookup_by_name_with_flags_async (GResolver *resolver,
g_task_set_source_tag (task, lookup_by_name_with_flags_async); g_task_set_source_tag (task, lookup_by_name_with_flags_async);
g_task_set_name (task, "[gio] resolver lookup"); g_task_set_name (task, "[gio] resolver lookup");
g_task_set_task_data (task, data, (GDestroyNotify)lookup_data_free); g_task_set_task_data (task, g_steal_pointer (&data), (GDestroyNotify) lookup_data_free);
g_task_set_return_on_cancel (task, TRUE); g_task_set_return_on_cancel (task, TRUE);
g_task_run_in_thread (task, do_lookup_by_name); g_task_run_in_thread (task, do_lookup_by_name);
g_object_unref (task); g_object_unref (task);
@ -311,7 +362,8 @@ do_lookup_by_address (GTask *task,
gpointer task_data, gpointer task_data,
GCancellable *cancellable) GCancellable *cancellable)
{ {
GInetAddress *address = task_data; LookupData *data = task_data;
GInetAddress *address = data->lookup_by_address.address;
struct sockaddr_storage sockaddr_address; struct sockaddr_storage sockaddr_address;
gsize sockaddr_address_size; gsize sockaddr_address_size;
GSocketAddress *gsockaddr; GSocketAddress *gsockaddr;
@ -358,13 +410,15 @@ lookup_by_address (GResolver *resolver,
GCancellable *cancellable, GCancellable *cancellable,
GError **error) GError **error)
{ {
LookupData *data = NULL;
GTask *task; GTask *task;
gchar *name; gchar *name;
data = lookup_data_new_by_address (address);
task = g_task_new (resolver, cancellable, NULL, NULL); task = g_task_new (resolver, cancellable, NULL, NULL);
g_task_set_source_tag (task, lookup_by_address); g_task_set_source_tag (task, lookup_by_address);
g_task_set_name (task, "[gio] resolver lookup"); g_task_set_name (task, "[gio] resolver lookup");
g_task_set_task_data (task, g_object_ref (address), g_object_unref); g_task_set_task_data (task, g_steal_pointer (&data), (GDestroyNotify) lookup_data_free);
g_task_set_return_on_cancel (task, TRUE); g_task_set_return_on_cancel (task, TRUE);
g_task_run_in_thread_sync (task, do_lookup_by_address); g_task_run_in_thread_sync (task, do_lookup_by_address);
name = g_task_propagate_pointer (task, error); name = g_task_propagate_pointer (task, error);
@ -380,12 +434,14 @@ lookup_by_address_async (GResolver *resolver,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data) gpointer user_data)
{ {
LookupData *data = NULL;
GTask *task; GTask *task;
data = lookup_data_new_by_address (address);
task = g_task_new (resolver, cancellable, callback, user_data); task = g_task_new (resolver, cancellable, callback, user_data);
g_task_set_source_tag (task, lookup_by_address_async); g_task_set_source_tag (task, lookup_by_address_async);
g_task_set_name (task, "[gio] resolver lookup"); g_task_set_name (task, "[gio] resolver lookup");
g_task_set_task_data (task, g_object_ref (address), g_object_unref); g_task_set_task_data (task, g_steal_pointer (&data), (GDestroyNotify) lookup_data_free);
g_task_set_return_on_cancel (task, TRUE); g_task_set_return_on_cancel (task, TRUE);
g_task_run_in_thread (task, do_lookup_by_address); g_task_run_in_thread (task, do_lookup_by_address);
g_object_unref (task); g_object_unref (task);
@ -1074,18 +1130,6 @@ g_resolver_records_from_DnsQuery (const gchar *rrname,
#endif #endif
typedef struct {
char *rrname;
GResolverRecordType record_type;
} LookupRecordsData;
static void
free_lookup_records_data (LookupRecordsData *lrd)
{
g_free (lrd->rrname);
g_slice_free (LookupRecordsData, lrd);
}
static void static void
free_records (GList *records) free_records (GList *records)
{ {
@ -1107,7 +1151,9 @@ do_lookup_records (GTask *task,
gpointer task_data, gpointer task_data,
GCancellable *cancellable) GCancellable *cancellable)
{ {
LookupRecordsData *lrd = task_data; LookupData *data = task_data;
const gchar *rrname = data->lookup_records.rrname;
GResolverRecordType record_type = data->lookup_records.record_type;
GList *records; GList *records;
GError *error = NULL; GError *error = NULL;
@ -1134,20 +1180,20 @@ do_lookup_records (GTask *task,
if (res_ninit (&res) != 0) if (res_ninit (&res) != 0)
{ {
g_task_return_new_error (task, G_RESOLVER_ERROR, G_RESOLVER_ERROR_INTERNAL, g_task_return_new_error (task, G_RESOLVER_ERROR, G_RESOLVER_ERROR_INTERNAL,
_("Error resolving “%s”"), lrd->rrname); _("Error resolving “%s”"), rrname);
return; return;
} }
#endif #endif
rrtype = g_resolver_record_type_to_rrtype (lrd->record_type); rrtype = g_resolver_record_type_to_rrtype (record_type);
answer = g_byte_array_new (); answer = g_byte_array_new ();
for (;;) for (;;)
{ {
g_byte_array_set_size (answer, len * 2); g_byte_array_set_size (answer, len * 2);
#if defined(HAVE_RES_NQUERY) #if defined(HAVE_RES_NQUERY)
len = res_nquery (&res, lrd->rrname, C_IN, rrtype, answer->data, answer->len); len = res_nquery (&res, rrname, C_IN, rrtype, answer->data, answer->len);
#else #else
len = res_query (lrd->rrname, C_IN, rrtype, answer->data, answer->len); len = res_query (rrname, C_IN, rrtype, answer->data, answer->len);
#endif #endif
/* If answer fit in the buffer then we're done */ /* If answer fit in the buffer then we're done */
@ -1161,7 +1207,7 @@ do_lookup_records (GTask *task,
} }
herr = h_errno; herr = h_errno;
records = g_resolver_records_from_res_query (lrd->rrname, rrtype, answer->data, len, herr, &error); records = g_resolver_records_from_res_query (rrname, rrtype, answer->data, len, herr, &error);
g_byte_array_free (answer, TRUE); g_byte_array_free (answer, TRUE);
#ifdef HAVE_RES_NQUERY #ifdef HAVE_RES_NQUERY
@ -1182,9 +1228,9 @@ do_lookup_records (GTask *task,
DNS_RECORD *results = NULL; DNS_RECORD *results = NULL;
WORD dnstype; WORD dnstype;
dnstype = g_resolver_record_type_to_dnstype (lrd->record_type); dnstype = g_resolver_record_type_to_dnstype (record_type);
status = DnsQuery_A (lrd->rrname, dnstype, DNS_QUERY_STANDARD, NULL, &results, NULL); status = DnsQuery_A (rrname, dnstype, DNS_QUERY_STANDARD, NULL, &results, NULL);
records = g_resolver_records_from_DnsQuery (lrd->rrname, dnstype, status, results, &error); records = g_resolver_records_from_DnsQuery (rrname, dnstype, status, results, &error);
if (results != NULL) if (results != NULL)
DnsRecordListFree (results, DnsFreeRecordList); DnsRecordListFree (results, DnsFreeRecordList);
@ -1205,16 +1251,14 @@ lookup_records (GResolver *resolver,
{ {
GTask *task; GTask *task;
GList *records; GList *records;
LookupRecordsData *lrd; LookupData *data = NULL;
task = g_task_new (resolver, cancellable, NULL, NULL); task = g_task_new (resolver, cancellable, NULL, NULL);
g_task_set_source_tag (task, lookup_records); g_task_set_source_tag (task, lookup_records);
g_task_set_name (task, "[gio] resolver lookup records"); g_task_set_name (task, "[gio] resolver lookup records");
lrd = g_slice_new (LookupRecordsData); data = lookup_data_new_records (rrname, record_type);
lrd->rrname = g_strdup (rrname); g_task_set_task_data (task, g_steal_pointer (&data), (GDestroyNotify) lookup_data_free);
lrd->record_type = record_type;
g_task_set_task_data (task, lrd, (GDestroyNotify) free_lookup_records_data);
g_task_set_return_on_cancel (task, TRUE); g_task_set_return_on_cancel (task, TRUE);
g_task_run_in_thread_sync (task, do_lookup_records); g_task_run_in_thread_sync (task, do_lookup_records);
@ -1233,16 +1277,14 @@ lookup_records_async (GResolver *resolver,
gpointer user_data) gpointer user_data)
{ {
GTask *task; GTask *task;
LookupRecordsData *lrd; LookupData *data = NULL;
task = g_task_new (resolver, cancellable, callback, user_data); task = g_task_new (resolver, cancellable, callback, user_data);
g_task_set_source_tag (task, lookup_records_async); g_task_set_source_tag (task, lookup_records_async);
g_task_set_name (task, "[gio] resolver lookup records"); g_task_set_name (task, "[gio] resolver lookup records");
lrd = g_slice_new (LookupRecordsData); data = lookup_data_new_records (rrname, record_type);
lrd->rrname = g_strdup (rrname); g_task_set_task_data (task, g_steal_pointer (&data), (GDestroyNotify) lookup_data_free);
lrd->record_type = record_type;
g_task_set_task_data (task, lrd, (GDestroyNotify) free_lookup_records_data);
g_task_set_return_on_cancel (task, TRUE); g_task_set_return_on_cancel (task, TRUE);
g_task_run_in_thread (task, do_lookup_records); g_task_run_in_thread (task, do_lookup_records);