Coverage Report

Created: 2025-11-11 06:08

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/rauc/subprojects/glib-2.76.5/gio/gnetworkaddress.c
Line
Count
Source
1
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
2
3
/* GIO - GLib Input, Output and Streaming Library
4
 *
5
 * Copyright (C) 2008 Red Hat, Inc.
6
 * Copyright (C) 2018 Igalia S.L.
7
 *
8
 * SPDX-License-Identifier: LGPL-2.1-or-later
9
 *
10
 * This library is free software; you can redistribute it and/or
11
 * modify it under the terms of the GNU Lesser General Public
12
 * License as published by the Free Software Foundation; either
13
 * version 2.1 of the License, or (at your option) any later version.
14
 *
15
 * This library is distributed in the hope that it will be useful,
16
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18
 * Lesser General Public License for more details.
19
 *
20
 * You should have received a copy of the GNU Lesser General
21
 * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
22
 */
23
24
#include "config.h"
25
#include <glib.h>
26
#include "glibintl.h"
27
28
#include <stdlib.h>
29
#include "gnetworkaddress.h"
30
#include "gasyncresult.h"
31
#include "ginetaddress.h"
32
#include "ginetsocketaddress.h"
33
#include "gnetworkingprivate.h"
34
#include "gproxyaddressenumerator.h"
35
#include "gresolver.h"
36
#include "gtask.h"
37
#include "gsocketaddressenumerator.h"
38
#include "gioerror.h"
39
#include "gsocketconnectable.h"
40
41
#include <string.h>
42
43
/* As recommended by RFC 8305 this is the time it waits for a following
44
   DNS response to come in (ipv4 waiting on ipv6 generally)
45
 */
46
0
#define HAPPY_EYEBALLS_RESOLUTION_DELAY_MS 50
47
48
/**
49
 * SECTION:gnetworkaddress
50
 * @short_description: A GSocketConnectable for resolving hostnames
51
 * @include: gio/gio.h
52
 *
53
 * #GNetworkAddress provides an easy way to resolve a hostname and
54
 * then attempt to connect to that host, handling the possibility of
55
 * multiple IP addresses and multiple address families.
56
 *
57
 * The enumeration results of resolved addresses *may* be cached as long
58
 * as this object is kept alive which may have unexpected results if
59
 * alive for too long.
60
 *
61
 * See #GSocketConnectable for an example of using the connectable
62
 * interface.
63
 */
64
65
/**
66
 * GNetworkAddress:
67
 *
68
 * A #GSocketConnectable for resolving a hostname and connecting to
69
 * that host.
70
 */
71
72
struct _GNetworkAddressPrivate {
73
  gchar *hostname;
74
  guint16 port;
75
  GList *cached_sockaddrs;
76
  gchar *scheme;
77
78
  gint64 resolver_serial;
79
};
80
81
enum {
82
  PROP_0,
83
  PROP_HOSTNAME,
84
  PROP_PORT,
85
  PROP_SCHEME,
86
};
87
88
static void g_network_address_set_property (GObject      *object,
89
                                            guint         prop_id,
90
                                            const GValue *value,
91
                                            GParamSpec   *pspec);
92
static void g_network_address_get_property (GObject      *object,
93
                                            guint         prop_id,
94
                                            GValue       *value,
95
                                            GParamSpec   *pspec);
96
97
static void                      g_network_address_connectable_iface_init       (GSocketConnectableIface *iface);
98
static GSocketAddressEnumerator *g_network_address_connectable_enumerate        (GSocketConnectable      *connectable);
99
static GSocketAddressEnumerator *g_network_address_connectable_proxy_enumerate  (GSocketConnectable      *connectable);
100
static gchar                    *g_network_address_connectable_to_string        (GSocketConnectable      *connectable);
101
102
0
G_DEFINE_TYPE_WITH_CODE (GNetworkAddress, g_network_address, G_TYPE_OBJECT,
103
0
                         G_ADD_PRIVATE (GNetworkAddress)
104
0
                         G_IMPLEMENT_INTERFACE (G_TYPE_SOCKET_CONNECTABLE,
105
0
                                                g_network_address_connectable_iface_init))
106
0
107
0
static void
108
0
g_network_address_finalize (GObject *object)
109
0
{
110
0
  GNetworkAddress *addr = G_NETWORK_ADDRESS (object);
111
112
0
  g_free (addr->priv->hostname);
113
0
  g_free (addr->priv->scheme);
114
0
  g_list_free_full (addr->priv->cached_sockaddrs, g_object_unref);
115
116
0
  G_OBJECT_CLASS (g_network_address_parent_class)->finalize (object);
117
0
}
118
119
static void
120
g_network_address_class_init (GNetworkAddressClass *klass)
121
0
{
122
0
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
123
124
0
  gobject_class->set_property = g_network_address_set_property;
125
0
  gobject_class->get_property = g_network_address_get_property;
126
0
  gobject_class->finalize = g_network_address_finalize;
127
128
0
  g_object_class_install_property (gobject_class, PROP_HOSTNAME,
129
0
                                   g_param_spec_string ("hostname",
130
0
                                                        P_("Hostname"),
131
0
                                                        P_("Hostname to resolve"),
132
0
                                                        NULL,
133
0
                                                        G_PARAM_READWRITE |
134
0
                                                        G_PARAM_CONSTRUCT_ONLY |
135
0
                                                        G_PARAM_STATIC_STRINGS));
136
0
  g_object_class_install_property (gobject_class, PROP_PORT,
137
0
                                   g_param_spec_uint ("port",
138
0
                                                      P_("Port"),
139
0
                                                      P_("Network port"),
140
0
                                                      0, 65535, 0,
141
0
                                                      G_PARAM_READWRITE |
142
0
                                                      G_PARAM_CONSTRUCT_ONLY |
143
0
                                                      G_PARAM_STATIC_STRINGS));
144
145
0
  g_object_class_install_property (gobject_class, PROP_SCHEME,
146
0
                                   g_param_spec_string ("scheme",
147
0
                                                        P_("Scheme"),
148
0
                                                        P_("URI Scheme"),
149
0
                                                        NULL,
150
0
                                                        G_PARAM_READWRITE |
151
0
                                                        G_PARAM_CONSTRUCT_ONLY |
152
0
                                                        G_PARAM_STATIC_STRINGS));
153
0
}
154
155
static void
156
g_network_address_connectable_iface_init (GSocketConnectableIface *connectable_iface)
157
0
{
158
0
  connectable_iface->enumerate  = g_network_address_connectable_enumerate;
159
0
  connectable_iface->proxy_enumerate = g_network_address_connectable_proxy_enumerate;
160
0
  connectable_iface->to_string = g_network_address_connectable_to_string;
161
0
}
162
163
static void
164
g_network_address_init (GNetworkAddress *addr)
165
0
{
166
0
  addr->priv = g_network_address_get_instance_private (addr);
167
0
}
168
169
static void
170
g_network_address_set_property (GObject      *object,
171
                                guint         prop_id,
172
                                const GValue *value,
173
                                GParamSpec   *pspec)
174
0
{
175
0
  GNetworkAddress *addr = G_NETWORK_ADDRESS (object);
176
177
0
  switch (prop_id)
178
0
    {
179
0
    case PROP_HOSTNAME:
180
0
      g_free (addr->priv->hostname);
181
0
      addr->priv->hostname = g_value_dup_string (value);
182
0
      break;
183
184
0
    case PROP_PORT:
185
0
      addr->priv->port = g_value_get_uint (value);
186
0
      break;
187
188
0
    case PROP_SCHEME:
189
0
      g_free (addr->priv->scheme);
190
0
      addr->priv->scheme = g_value_dup_string (value);
191
0
      break;
192
193
0
    default:
194
0
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
195
0
      break;
196
0
    }
197
198
0
}
199
200
static void
201
g_network_address_get_property (GObject    *object,
202
                                guint       prop_id,
203
                                GValue     *value,
204
                                GParamSpec *pspec)
205
0
{
206
0
  GNetworkAddress *addr = G_NETWORK_ADDRESS (object);
207
208
0
  switch (prop_id)
209
0
    {
210
0
    case PROP_HOSTNAME:
211
0
      g_value_set_string (value, addr->priv->hostname);
212
0
      break;
213
214
0
    case PROP_PORT:
215
0
      g_value_set_uint (value, addr->priv->port);
216
0
      break;
217
218
0
    case PROP_SCHEME:
219
0
      g_value_set_string (value, addr->priv->scheme);
220
0
      break;
221
222
0
    default:
223
0
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
224
0
      break;
225
0
    }
226
227
0
}
228
229
/*
230
 * inet_addresses_to_inet_socket_addresses:
231
 * @addresses: (transfer full): #GList of #GInetAddress
232
 *
233
 * Returns: (transfer full): #GList of #GInetSocketAddress
234
 */
235
static GList *
236
inet_addresses_to_inet_socket_addresses (GNetworkAddress *addr,
237
                                         GList           *addresses)
238
0
{
239
0
  GList *a, *socket_addresses = NULL;
240
241
0
  for (a = addresses; a; a = a->next)
242
0
    {
243
0
      GSocketAddress *sockaddr = g_inet_socket_address_new (a->data, addr->priv->port);
244
0
      socket_addresses = g_list_append (socket_addresses, g_steal_pointer (&sockaddr));
245
0
      g_object_unref (a->data);
246
0
    }
247
248
0
  g_list_free (addresses);
249
0
  return socket_addresses;
250
0
}
251
252
/*
253
 * g_network_address_set_cached_addresses:
254
 * @addr: A #GNetworkAddress
255
 * @addresses: (transfer full): List of #GInetAddress or #GInetSocketAddress
256
 * @resolver_serial: Serial of #GResolver used
257
 *
258
 * Consumes @addresses and uses them to replace the current internal list.
259
 */
260
static void
261
g_network_address_set_cached_addresses (GNetworkAddress *addr,
262
                                        GList           *addresses,
263
                                        guint64          resolver_serial)
264
0
{
265
0
  g_assert (addresses != NULL);
266
267
0
  if (addr->priv->cached_sockaddrs)
268
0
    g_list_free_full (addr->priv->cached_sockaddrs, g_object_unref);
269
270
0
  if (G_IS_INET_SOCKET_ADDRESS (addresses->data))
271
0
    addr->priv->cached_sockaddrs = g_steal_pointer (&addresses);
272
0
  else
273
0
    addr->priv->cached_sockaddrs = inet_addresses_to_inet_socket_addresses (addr, g_steal_pointer (&addresses));
274
0
  addr->priv->resolver_serial = resolver_serial;
275
0
}
276
277
static gboolean
278
g_network_address_parse_sockaddr (GNetworkAddress *addr)
279
0
{
280
0
  GSocketAddress *sockaddr;
281
282
0
  g_assert (addr->priv->cached_sockaddrs == NULL);
283
284
0
  sockaddr = g_inet_socket_address_new_from_string (addr->priv->hostname,
285
0
                                                    addr->priv->port);
286
0
  if (sockaddr)
287
0
    {
288
0
      addr->priv->cached_sockaddrs = g_list_append (addr->priv->cached_sockaddrs, sockaddr);
289
0
      return TRUE;
290
0
    }
291
0
  else
292
0
    return FALSE;
293
0
}
294
295
/**
296
 * g_network_address_new:
297
 * @hostname: the hostname
298
 * @port: the port
299
 *
300
 * Creates a new #GSocketConnectable for connecting to the given
301
 * @hostname and @port.
302
 *
303
 * Note that depending on the configuration of the machine, a
304
 * @hostname of `localhost` may refer to the IPv4 loopback address
305
 * only, or to both IPv4 and IPv6; use
306
 * g_network_address_new_loopback() to create a #GNetworkAddress that
307
 * is guaranteed to resolve to both addresses.
308
 *
309
 * Returns: (transfer full) (type GNetworkAddress): the new #GNetworkAddress
310
 *
311
 * Since: 2.22
312
 */
313
GSocketConnectable *
314
g_network_address_new (const gchar *hostname,
315
                       guint16      port)
316
0
{
317
0
  return g_object_new (G_TYPE_NETWORK_ADDRESS,
318
0
                       "hostname", hostname,
319
0
                       "port", port,
320
0
                       NULL);
321
0
}
322
323
/**
324
 * g_network_address_new_loopback:
325
 * @port: the port
326
 *
327
 * Creates a new #GSocketConnectable for connecting to the local host
328
 * over a loopback connection to the given @port. This is intended for
329
 * use in connecting to local services which may be running on IPv4 or
330
 * IPv6.
331
 *
332
 * The connectable will return IPv4 and IPv6 loopback addresses,
333
 * regardless of how the host resolves `localhost`. By contrast,
334
 * g_network_address_new() will often only return an IPv4 address when
335
 * resolving `localhost`, and an IPv6 address for `localhost6`.
336
 *
337
 * g_network_address_get_hostname() will always return `localhost` for
338
 * a #GNetworkAddress created with this constructor.
339
 *
340
 * Returns: (transfer full) (type GNetworkAddress): the new #GNetworkAddress
341
 *
342
 * Since: 2.44
343
 */
344
GSocketConnectable *
345
g_network_address_new_loopback (guint16 port)
346
0
{
347
0
  GNetworkAddress *addr;
348
0
  GList *addrs = NULL;
349
350
0
  addr = g_object_new (G_TYPE_NETWORK_ADDRESS,
351
0
                       "hostname", "localhost",
352
0
                       "port", port,
353
0
                       NULL);
354
355
0
  addrs = g_list_append (addrs, g_inet_address_new_loopback (AF_INET6));
356
0
  addrs = g_list_append (addrs, g_inet_address_new_loopback (AF_INET));
357
0
  g_network_address_set_cached_addresses (addr, g_steal_pointer (&addrs), 0);
358
359
0
  return G_SOCKET_CONNECTABLE (addr);
360
0
}
361
362
/**
363
 * g_network_address_parse:
364
 * @host_and_port: the hostname and optionally a port
365
 * @default_port: the default port if not in @host_and_port
366
 * @error: a pointer to a #GError, or %NULL
367
 *
368
 * Creates a new #GSocketConnectable for connecting to the given
369
 * @hostname and @port. May fail and return %NULL in case
370
 * parsing @host_and_port fails.
371
 *
372
 * @host_and_port may be in any of a number of recognised formats; an IPv6
373
 * address, an IPv4 address, or a domain name (in which case a DNS
374
 * lookup is performed). Quoting with [] is supported for all address
375
 * types. A port override may be specified in the usual way with a
376
 * colon.
377
 *
378
 * If no port is specified in @host_and_port then @default_port will be
379
 * used as the port number to connect to.
380
 *
381
 * In general, @host_and_port is expected to be provided by the user
382
 * (allowing them to give the hostname, and a port override if necessary)
383
 * and @default_port is expected to be provided by the application.
384
 *
385
 * (The port component of @host_and_port can also be specified as a
386
 * service name rather than as a numeric port, but this functionality
387
 * is deprecated, because it depends on the contents of /etc/services,
388
 * which is generally quite sparse on platforms other than Linux.)
389
 *
390
 * Returns: (transfer full) (type GNetworkAddress): the new
391
 *   #GNetworkAddress, or %NULL on error
392
 *
393
 * Since: 2.22
394
 */
395
GSocketConnectable *
396
g_network_address_parse (const gchar  *host_and_port,
397
                         guint16       default_port,
398
                         GError      **error)
399
0
{
400
0
  GSocketConnectable *connectable;
401
0
  const gchar *port;
402
0
  guint16 portnum;
403
0
  gchar *name;
404
405
0
  g_return_val_if_fail (host_and_port != NULL, NULL);
406
407
0
  port = NULL;
408
0
  if (host_and_port[0] == '[')
409
    /* escaped host part (to allow, eg. "[2001:db8::1]:888") */
410
0
    {
411
0
      const gchar *end;
412
413
0
      end = strchr (host_and_port, ']');
414
0
      if (end == NULL)
415
0
        {
416
0
          g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
417
0
                       _("Hostname “%s” contains “[” but not “]”"), host_and_port);
418
0
          return NULL;
419
0
        }
420
421
0
      if (end[1] == '\0')
422
0
        port = NULL;
423
0
      else if (end[1] == ':')
424
0
        port = &end[2];
425
0
      else
426
0
        {
427
0
          g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
428
0
                       "The ']' character (in hostname '%s') must come at the"
429
0
                       " end or be immediately followed by ':' and a port",
430
0
                       host_and_port);
431
0
          return NULL;
432
0
        }
433
434
0
      name = g_strndup (host_and_port + 1, end - host_and_port - 1);
435
0
    }
436
437
0
  else if ((port = strchr (host_and_port, ':')))
438
    /* string has a ':' in it */
439
0
    {
440
      /* skip ':' */
441
0
      port++;
442
443
0
      if (strchr (port, ':'))
444
        /* more than one ':' in string */
445
0
        {
446
          /* this is actually an unescaped IPv6 address */
447
0
          name = g_strdup (host_and_port);
448
0
          port = NULL;
449
0
        }
450
0
      else
451
0
        name = g_strndup (host_and_port, port - host_and_port - 1);
452
0
    }
453
454
0
  else
455
    /* plain hostname, no port */
456
0
    name = g_strdup (host_and_port);
457
458
0
  if (port != NULL)
459
0
    {
460
0
      if (port[0] == '\0')
461
0
        {
462
0
          g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
463
0
                       "If a ':' character is given, it must be followed by a "
464
0
                       "port (in hostname '%s').", host_and_port);
465
0
          g_free (name);
466
0
          return NULL;
467
0
        }
468
469
0
      else if ('0' <= port[0] && port[0] <= '9')
470
0
        {
471
0
          char *end;
472
0
          long value;
473
474
0
          value = strtol (port, &end, 10);
475
0
          if (*end != '\0' || value < 0 || value > G_MAXUINT16)
476
0
            {
477
0
              g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
478
0
                           "Invalid numeric port '%s' specified in hostname '%s'",
479
0
                           port, host_and_port);
480
0
              g_free (name);
481
0
              return NULL;
482
0
            }
483
484
0
          portnum = value;
485
0
        }
486
487
0
      else
488
0
        {
489
0
          struct servent *entry;
490
491
0
          entry = getservbyname (port, "tcp");
492
0
          if (entry == NULL)
493
0
            {
494
0
              g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
495
0
                           "Unknown service '%s' specified in hostname '%s'",
496
0
                           port, host_and_port);
497
0
#ifdef HAVE_ENDSERVENT
498
0
              endservent ();
499
0
#endif
500
0
              g_free (name);
501
0
              return NULL;
502
0
            }
503
504
0
          portnum = g_ntohs (entry->s_port);
505
506
0
#ifdef HAVE_ENDSERVENT
507
0
          endservent ();
508
0
#endif
509
0
        }
510
0
    }
511
0
  else
512
0
    {
513
      /* No port in host_and_port */
514
0
      portnum = default_port;
515
0
    }
516
517
0
  connectable = g_network_address_new (name, portnum);
518
0
  g_free (name);
519
520
0
  return connectable;
521
0
}
522
523
/**
524
 * g_network_address_parse_uri:
525
 * @uri: the hostname and optionally a port
526
 * @default_port: The default port if none is found in the URI
527
 * @error: a pointer to a #GError, or %NULL
528
 *
529
 * Creates a new #GSocketConnectable for connecting to the given
530
 * @uri. May fail and return %NULL in case parsing @uri fails.
531
 *
532
 * Using this rather than g_network_address_new() or
533
 * g_network_address_parse() allows #GSocketClient to determine
534
 * when to use application-specific proxy protocols.
535
 *
536
 * Returns: (transfer full) (type GNetworkAddress): the new
537
 *   #GNetworkAddress, or %NULL on error
538
 *
539
 * Since: 2.26
540
 */
541
GSocketConnectable *
542
g_network_address_parse_uri (const gchar  *uri,
543
               guint16       default_port,
544
           GError      **error)
545
0
{
546
0
  GSocketConnectable *conn = NULL;
547
0
  gchar *scheme = NULL;
548
0
  gchar *hostname = NULL;
549
0
  gint port;
550
551
0
  if (!g_uri_split_network (uri, G_URI_FLAGS_NONE,
552
0
                            &scheme, &hostname, &port, NULL))
553
0
    {
554
0
      g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
555
0
                   "Invalid URI ‘%s’", uri);
556
0
      return NULL;
557
0
    }
558
559
0
  if (port <= 0)
560
0
    port = default_port;
561
562
0
  conn = g_object_new (G_TYPE_NETWORK_ADDRESS,
563
0
                       "hostname", hostname,
564
0
                       "port", (guint) port,
565
0
                       "scheme", scheme,
566
0
                       NULL);
567
0
  g_free (scheme);
568
0
  g_free (hostname);
569
570
0
  return conn;
571
0
}
572
573
/**
574
 * g_network_address_get_hostname:
575
 * @addr: a #GNetworkAddress
576
 *
577
 * Gets @addr's hostname. This might be either UTF-8 or ASCII-encoded,
578
 * depending on what @addr was created with.
579
 *
580
 * Returns: @addr's hostname
581
 *
582
 * Since: 2.22
583
 */
584
const gchar *
585
g_network_address_get_hostname (GNetworkAddress *addr)
586
0
{
587
0
  g_return_val_if_fail (G_IS_NETWORK_ADDRESS (addr), NULL);
588
589
0
  return addr->priv->hostname;
590
0
}
591
592
/**
593
 * g_network_address_get_port:
594
 * @addr: a #GNetworkAddress
595
 *
596
 * Gets @addr's port number
597
 *
598
 * Returns: @addr's port (which may be 0)
599
 *
600
 * Since: 2.22
601
 */
602
guint16
603
g_network_address_get_port (GNetworkAddress *addr)
604
0
{
605
0
  g_return_val_if_fail (G_IS_NETWORK_ADDRESS (addr), 0);
606
607
0
  return addr->priv->port;
608
0
}
609
610
/**
611
 * g_network_address_get_scheme:
612
 * @addr: a #GNetworkAddress
613
 *
614
 * Gets @addr's scheme
615
 *
616
 * Returns: (nullable): @addr's scheme (%NULL if not built from URI)
617
 *
618
 * Since: 2.26
619
 */
620
const gchar *
621
g_network_address_get_scheme (GNetworkAddress *addr)
622
0
{
623
0
  g_return_val_if_fail (G_IS_NETWORK_ADDRESS (addr), NULL);
624
625
0
  return addr->priv->scheme;
626
0
}
627
628
0
#define G_TYPE_NETWORK_ADDRESS_ADDRESS_ENUMERATOR (_g_network_address_address_enumerator_get_type ())
629
0
#define G_NETWORK_ADDRESS_ADDRESS_ENUMERATOR(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), G_TYPE_NETWORK_ADDRESS_ADDRESS_ENUMERATOR, GNetworkAddressAddressEnumerator))
630
631
typedef enum {
632
  RESOLVE_STATE_NONE = 0,
633
  RESOLVE_STATE_WAITING_ON_IPV4 = 1 << 0,
634
  RESOLVE_STATE_WAITING_ON_IPV6 = 1 << 1,
635
} ResolveState;
636
637
typedef struct {
638
  GSocketAddressEnumerator parent_instance;
639
640
  GNetworkAddress *addr; /* (owned) */
641
  GList *addresses; /* (owned) (nullable) */
642
  GList *current_item; /* (unowned) (nullable) */
643
  GTask *queued_task; /* (owned) (nullable) */
644
  GTask *waiting_task; /* (owned) (nullable) */
645
  GError *last_error; /* (owned) (nullable) */
646
  GSource *wait_source; /* (owned) (nullable) */
647
  GMainContext *context; /* (owned) (nullable) */
648
  ResolveState state;
649
} GNetworkAddressAddressEnumerator;
650
651
typedef struct {
652
  GSocketAddressEnumeratorClass parent_class;
653
654
} GNetworkAddressAddressEnumeratorClass;
655
656
static GType _g_network_address_address_enumerator_get_type (void);
657
0
G_DEFINE_TYPE (GNetworkAddressAddressEnumerator, _g_network_address_address_enumerator, G_TYPE_SOCKET_ADDRESS_ENUMERATOR)
658
0
659
0
static void
660
0
g_network_address_address_enumerator_finalize (GObject *object)
661
0
{
662
0
  GNetworkAddressAddressEnumerator *addr_enum =
663
0
    G_NETWORK_ADDRESS_ADDRESS_ENUMERATOR (object);
664
665
0
  if (addr_enum->wait_source)
666
0
    {
667
0
      g_source_destroy (addr_enum->wait_source);
668
0
      g_clear_pointer (&addr_enum->wait_source, g_source_unref);
669
0
    }
670
0
  g_clear_object (&addr_enum->queued_task);
671
0
  g_clear_object (&addr_enum->waiting_task);
672
0
  g_clear_error (&addr_enum->last_error);
673
0
  g_object_unref (addr_enum->addr);
674
0
  g_clear_pointer (&addr_enum->context, g_main_context_unref);
675
0
  g_list_free_full (addr_enum->addresses, g_object_unref);
676
677
0
  G_OBJECT_CLASS (_g_network_address_address_enumerator_parent_class)->finalize (object);
678
0
}
679
680
static inline GSocketFamily
681
get_address_family (GInetSocketAddress *address)
682
0
{
683
0
  return g_inet_address_get_family (g_inet_socket_address_get_address (address));
684
0
}
685
686
static void
687
list_split_families (GList  *list,
688
                     GList **out_ipv4,
689
                     GList **out_ipv6)
690
0
{
691
0
  g_assert (out_ipv4);
692
0
  g_assert (out_ipv6);
693
694
0
  while (list)
695
0
    {
696
0
      GSocketFamily family = get_address_family (list->data);
697
0
      switch (family)
698
0
        {
699
0
          case G_SOCKET_FAMILY_IPV4:
700
0
            *out_ipv4 = g_list_prepend (*out_ipv4, list->data);
701
0
            break;
702
0
          case G_SOCKET_FAMILY_IPV6:
703
0
            *out_ipv6 = g_list_prepend (*out_ipv6, list->data);
704
0
            break;
705
0
          case G_SOCKET_FAMILY_INVALID:
706
0
          case G_SOCKET_FAMILY_UNIX:
707
0
            g_assert_not_reached ();
708
0
        }
709
710
0
      list = g_list_next (list);
711
0
    }
712
713
0
  *out_ipv4 = g_list_reverse (*out_ipv4);
714
0
  *out_ipv6 = g_list_reverse (*out_ipv6);
715
0
}
716
717
static GList *
718
list_interleave_families (GList *list1,
719
                          GList *list2)
720
0
{
721
0
  GList *interleaved = NULL;
722
723
0
  while (list1 || list2)
724
0
    {
725
0
      if (list1)
726
0
        {
727
0
          interleaved = g_list_append (interleaved, list1->data);
728
0
          list1 = g_list_delete_link (list1, list1);
729
0
        }
730
0
      if (list2)
731
0
        {
732
0
          interleaved = g_list_append (interleaved, list2->data);
733
0
          list2 = g_list_delete_link (list2, list2);
734
0
        }
735
0
    }
736
737
0
  return interleaved;
738
0
}
739
740
/* list_copy_interleaved:
741
 * @list: (transfer container): List to copy
742
 *
743
 * Does a shallow copy of a list with address families interleaved.
744
 *
745
 * For example:
746
 *   Input: [ipv6, ipv6, ipv4, ipv4]
747
 *   Output: [ipv6, ipv4, ipv6, ipv4]
748
 *
749
 * Returns: (transfer container): A new list
750
 */
751
static GList *
752
list_copy_interleaved (GList *list)
753
0
{
754
0
  GList *ipv4 = NULL, *ipv6 = NULL;
755
756
0
  list_split_families (list, &ipv4, &ipv6);
757
0
  return list_interleave_families (ipv6, ipv4);
758
0
}
759
760
/* list_concat_interleaved:
761
 * @parent_list: (transfer container): Already existing list
762
 * @current_item: (transfer container): Item after which to resort
763
 * @new_list: (transfer container): New list to be interleaved and concatenated
764
 *
765
 * This differs from g_list_concat() + list_copy_interleaved() in that it sorts
766
 * items in the previous list starting from @current_item and concats the results
767
 * to @parent_list.
768
 *
769
 * Returns: (transfer container): New start of list
770
 */
771
static GList *
772
list_concat_interleaved (GList *parent_list,
773
                         GList *current_item,
774
                         GList *new_list)
775
0
{
776
0
  GList *ipv4 = NULL, *ipv6 = NULL, *interleaved, *trailing = NULL;
777
0
  GSocketFamily last_family = G_SOCKET_FAMILY_IPV4; /* Default to starting with ipv6 */
778
779
0
  if (current_item)
780
0
    {
781
0
      last_family = get_address_family (current_item->data);
782
783
      /* Unused addresses will get removed, resorted, then readded */
784
0
      trailing = g_list_next (current_item);
785
0
      current_item->next = NULL;
786
0
    }
787
788
0
  list_split_families (trailing, &ipv4, &ipv6);
789
0
  list_split_families (new_list, &ipv4, &ipv6);
790
0
  g_list_free (new_list);
791
792
0
  if (trailing)
793
0
    g_list_free (trailing);
794
795
0
  if (last_family == G_SOCKET_FAMILY_IPV4)
796
0
    interleaved = list_interleave_families (ipv6, ipv4);
797
0
  else
798
0
    interleaved = list_interleave_families (ipv4, ipv6);
799
800
0
  return g_list_concat (parent_list, interleaved);
801
0
}
802
803
static void
804
maybe_update_address_cache (GNetworkAddressAddressEnumerator *addr_enum,
805
                            GResolver                        *resolver)
806
0
{
807
0
  GList *addresses, *p;
808
809
  /* Only cache complete results */
810
0
  if (addr_enum->state & RESOLVE_STATE_WAITING_ON_IPV4 || addr_enum->state & RESOLVE_STATE_WAITING_ON_IPV6)
811
0
    return;
812
813
  /* The enumerators list will not necessarily be fully sorted */
814
0
  addresses = list_copy_interleaved (addr_enum->addresses);
815
0
  for (p = addresses; p; p = p->next)
816
0
    g_object_ref (p->data);
817
818
0
  g_network_address_set_cached_addresses (addr_enum->addr, g_steal_pointer (&addresses), g_resolver_get_serial (resolver));
819
0
}
820
821
static void
822
g_network_address_address_enumerator_add_addresses (GNetworkAddressAddressEnumerator *addr_enum,
823
                                                    GList                            *addresses,
824
                                                    GResolver                        *resolver)
825
0
{
826
0
  GList *new_addresses = inet_addresses_to_inet_socket_addresses (addr_enum->addr, addresses);
827
828
0
  if (addr_enum->addresses == NULL)
829
0
    addr_enum->addresses = g_steal_pointer (&new_addresses);
830
0
  else
831
0
    addr_enum->addresses = list_concat_interleaved (addr_enum->addresses, addr_enum->current_item, g_steal_pointer (&new_addresses));
832
833
0
  maybe_update_address_cache (addr_enum, resolver);
834
0
}
835
836
static gpointer
837
copy_object (gconstpointer src,
838
             gpointer      user_data)
839
0
{
840
0
  return g_object_ref (G_OBJECT (src));
841
0
}
842
843
static GSocketAddress *
844
init_and_query_next_address (GNetworkAddressAddressEnumerator *addr_enum)
845
0
{
846
0
  GList *next_item;
847
848
0
  if (addr_enum->addresses == NULL)
849
0
    addr_enum->addresses = g_list_copy_deep (addr_enum->addr->priv->cached_sockaddrs,
850
0
                                             copy_object, NULL);
851
852
  /* We always want to look at the next item at call time to get the latest results.
853
     That means that sometimes ->next is NULL this call but is valid next call.
854
   */
855
0
  if (addr_enum->current_item == NULL)
856
0
    next_item = addr_enum->current_item = addr_enum->addresses;
857
0
  else
858
0
    next_item = g_list_next (addr_enum->current_item);
859
860
0
  if (next_item)
861
0
    {
862
0
      addr_enum->current_item = next_item;
863
0
      return g_object_ref (addr_enum->current_item->data);
864
0
    }
865
0
  else
866
0
    return NULL;
867
0
}
868
869
static GSocketAddress *
870
g_network_address_address_enumerator_next (GSocketAddressEnumerator  *enumerator,
871
                                           GCancellable              *cancellable,
872
                                           GError                   **error)
873
0
{
874
0
  GNetworkAddressAddressEnumerator *addr_enum =
875
0
    G_NETWORK_ADDRESS_ADDRESS_ENUMERATOR (enumerator);
876
877
0
  if (addr_enum->addresses == NULL)
878
0
    {
879
0
      GNetworkAddress *addr = addr_enum->addr;
880
0
      GResolver *resolver = g_resolver_get_default ();
881
0
      gint64 serial = g_resolver_get_serial (resolver);
882
883
0
      if (addr->priv->resolver_serial != 0 &&
884
0
          addr->priv->resolver_serial != serial)
885
0
        {
886
          /* Resolver has reloaded, discard cached addresses */
887
0
          g_list_free_full (addr->priv->cached_sockaddrs, g_object_unref);
888
0
          addr->priv->cached_sockaddrs = NULL;
889
0
        }
890
891
0
      if (!addr->priv->cached_sockaddrs)
892
0
        g_network_address_parse_sockaddr (addr);
893
0
      if (!addr->priv->cached_sockaddrs)
894
0
        {
895
0
          GList *addresses;
896
897
0
          addresses = g_resolver_lookup_by_name (resolver,
898
0
                                                 addr->priv->hostname,
899
0
                                                 cancellable, error);
900
0
          if (!addresses)
901
0
            {
902
0
              g_object_unref (resolver);
903
0
              return NULL;
904
0
            }
905
906
0
          g_network_address_set_cached_addresses (addr, g_steal_pointer (&addresses), serial);
907
0
        }
908
909
0
      g_object_unref (resolver);
910
0
    }
911
912
0
  return init_and_query_next_address (addr_enum);
913
0
}
914
915
static void
916
complete_queued_task (GNetworkAddressAddressEnumerator *addr_enum,
917
                      GTask                            *task,
918
                      GError                           *error)
919
0
{
920
0
  if (error)
921
0
    g_task_return_error (task, error);
922
0
  else
923
0
    {
924
0
      GSocketAddress *sockaddr = init_and_query_next_address (addr_enum);
925
0
      g_task_return_pointer (task, g_steal_pointer (&sockaddr), g_object_unref);
926
0
    }
927
0
  g_object_unref (task);
928
0
}
929
930
static int
931
on_address_timeout (gpointer user_data)
932
0
{
933
0
  GNetworkAddressAddressEnumerator *addr_enum = user_data;
934
935
  /* Upon completion it may get unref'd by the owner */
936
0
  g_object_ref (addr_enum);
937
938
0
  if (addr_enum->queued_task != NULL)
939
0
      complete_queued_task (addr_enum, g_steal_pointer (&addr_enum->queued_task),
940
0
                            g_steal_pointer (&addr_enum->last_error));
941
0
  else if (addr_enum->waiting_task != NULL)
942
0
      complete_queued_task (addr_enum, g_steal_pointer (&addr_enum->waiting_task),
943
0
                            NULL);
944
945
0
  g_clear_pointer (&addr_enum->wait_source, g_source_unref);
946
0
  g_object_unref (addr_enum);
947
948
0
  return G_SOURCE_REMOVE;
949
0
}
950
951
static void
952
got_ipv6_addresses (GObject      *source_object,
953
                    GAsyncResult *result,
954
                    gpointer      user_data)
955
0
{
956
0
  GNetworkAddressAddressEnumerator *addr_enum = user_data;
957
0
  GResolver *resolver = G_RESOLVER (source_object);
958
0
  GList *addresses;
959
0
  GError *error = NULL;
960
961
0
  addr_enum->state ^= RESOLVE_STATE_WAITING_ON_IPV6;
962
963
0
  addresses = g_resolver_lookup_by_name_with_flags_finish (resolver, result, &error);
964
0
  if (!error)
965
0
    g_network_address_address_enumerator_add_addresses (addr_enum, g_steal_pointer (&addresses), resolver);
966
0
  else
967
0
    g_debug ("IPv6 DNS error: %s", error->message);
968
969
  /* If ipv4 was first and waiting on us it can stop waiting */
970
0
  if (addr_enum->wait_source)
971
0
    {
972
0
      g_source_destroy (addr_enum->wait_source);
973
0
      g_clear_pointer (&addr_enum->wait_source, g_source_unref);
974
0
    }
975
976
  /* If we got an error before ipv4 then let its response handle it.
977
   * If we get ipv6 response first or error second then
978
   * immediately complete the task.
979
   */
980
0
  if (error != NULL && !addr_enum->last_error && (addr_enum->state & RESOLVE_STATE_WAITING_ON_IPV4))
981
0
    {
982
      /* ipv6 lookup failed, but ipv4 is still outstanding.  wait. */
983
0
      addr_enum->last_error = g_steal_pointer (&error);
984
0
    }
985
0
  else if (addr_enum->waiting_task != NULL)
986
0
    {
987
0
      complete_queued_task (addr_enum, g_steal_pointer (&addr_enum->waiting_task), NULL);
988
0
    }
989
0
  else if (addr_enum->queued_task != NULL)
990
0
    {
991
0
      GError *task_error = NULL;
992
993
      /* If both errored just use the ipv6 one,
994
         but if ipv6 errored and ipv4 didn't we don't error */
995
0
      if (error != NULL && addr_enum->last_error)
996
0
          task_error = g_steal_pointer (&error);
997
998
0
      g_clear_error (&addr_enum->last_error);
999
0
      complete_queued_task (addr_enum, g_steal_pointer (&addr_enum->queued_task),
1000
0
                            g_steal_pointer (&task_error));
1001
0
    }
1002
1003
0
  g_clear_error (&error);
1004
0
  g_object_unref (addr_enum);
1005
0
}
1006
1007
static void
1008
got_ipv4_addresses (GObject      *source_object,
1009
                    GAsyncResult *result,
1010
                    gpointer      user_data)
1011
0
{
1012
0
  GNetworkAddressAddressEnumerator *addr_enum = user_data;
1013
0
  GResolver *resolver = G_RESOLVER (source_object);
1014
0
  GList *addresses;
1015
0
  GError *error = NULL;
1016
1017
0
  addr_enum->state ^= RESOLVE_STATE_WAITING_ON_IPV4;
1018
1019
0
  addresses = g_resolver_lookup_by_name_with_flags_finish (resolver, result, &error);
1020
0
  if (!error)
1021
0
    g_network_address_address_enumerator_add_addresses (addr_enum, g_steal_pointer (&addresses), resolver);
1022
0
  else
1023
0
    g_debug ("IPv4 DNS error: %s", error->message);
1024
1025
0
  if (addr_enum->wait_source)
1026
0
    {
1027
0
      g_source_destroy (addr_enum->wait_source);
1028
0
      g_clear_pointer (&addr_enum->wait_source, g_source_unref);
1029
0
    }
1030
1031
  /* If ipv6 already came in and errored then we return.
1032
   * If ipv6 returned successfully then we don't need to do anything unless
1033
   * another enumeration was waiting on us.
1034
   * If ipv6 hasn't come we should wait a short while for it as RFC 8305 suggests.
1035
   */
1036
0
  if (addr_enum->last_error)
1037
0
    {
1038
0
      g_assert (addr_enum->queued_task);
1039
0
      g_clear_error (&addr_enum->last_error);
1040
0
      complete_queued_task (addr_enum, g_steal_pointer (&addr_enum->queued_task),
1041
0
                            g_steal_pointer (&error));
1042
0
    }
1043
0
  else if (addr_enum->waiting_task != NULL)
1044
0
    {
1045
0
      complete_queued_task (addr_enum, g_steal_pointer (&addr_enum->waiting_task), NULL);
1046
0
    }
1047
0
  else if (addr_enum->queued_task != NULL)
1048
0
    {
1049
0
      addr_enum->last_error = g_steal_pointer (&error);
1050
0
      addr_enum->wait_source = g_timeout_source_new (HAPPY_EYEBALLS_RESOLUTION_DELAY_MS);
1051
0
      g_source_set_callback (addr_enum->wait_source,
1052
0
                             on_address_timeout,
1053
0
                             addr_enum, NULL);
1054
0
      g_source_attach (addr_enum->wait_source, addr_enum->context);
1055
0
    }
1056
1057
0
  g_clear_error (&error);
1058
0
  g_object_unref (addr_enum);
1059
0
}
1060
1061
static void
1062
g_network_address_address_enumerator_next_async (GSocketAddressEnumerator  *enumerator,
1063
                                                 GCancellable              *cancellable,
1064
                                                 GAsyncReadyCallback        callback,
1065
                                                 gpointer                   user_data)
1066
0
{
1067
0
  GNetworkAddressAddressEnumerator *addr_enum =
1068
0
    G_NETWORK_ADDRESS_ADDRESS_ENUMERATOR (enumerator);
1069
0
  GSocketAddress *sockaddr;
1070
0
  GTask *task;
1071
1072
0
  task = g_task_new (addr_enum, cancellable, callback, user_data);
1073
0
  g_task_set_source_tag (task, g_network_address_address_enumerator_next_async);
1074
1075
0
  if (addr_enum->addresses == NULL && addr_enum->state == RESOLVE_STATE_NONE)
1076
0
    {
1077
0
      GNetworkAddress *addr = addr_enum->addr;
1078
0
      GResolver *resolver = g_resolver_get_default ();
1079
0
      gint64 serial = g_resolver_get_serial (resolver);
1080
1081
0
      if (addr->priv->resolver_serial != 0 &&
1082
0
          addr->priv->resolver_serial != serial)
1083
0
        {
1084
          /* Resolver has reloaded, discard cached addresses */
1085
0
          g_list_free_full (addr->priv->cached_sockaddrs, g_object_unref);
1086
0
          addr->priv->cached_sockaddrs = NULL;
1087
0
        }
1088
1089
0
      if (addr->priv->cached_sockaddrs == NULL)
1090
0
        {
1091
0
          if (g_network_address_parse_sockaddr (addr))
1092
0
            complete_queued_task (addr_enum, task, NULL);
1093
0
          else
1094
0
            {
1095
              /* It does not make sense for this to be called multiple
1096
               * times before the initial callback has been called */
1097
0
              g_assert (addr_enum->queued_task == NULL);
1098
1099
0
              addr_enum->state = RESOLVE_STATE_WAITING_ON_IPV4 | RESOLVE_STATE_WAITING_ON_IPV6;
1100
0
              addr_enum->queued_task = g_steal_pointer (&task);
1101
              /* Look up in parallel as per RFC 8305 */
1102
0
              g_resolver_lookup_by_name_with_flags_async (resolver,
1103
0
                                                          addr->priv->hostname,
1104
0
                                                          G_RESOLVER_NAME_LOOKUP_FLAGS_IPV6_ONLY,
1105
0
                                                          cancellable,
1106
0
                                                          got_ipv6_addresses, g_object_ref (addr_enum));
1107
0
              g_resolver_lookup_by_name_with_flags_async (resolver,
1108
0
                                                          addr->priv->hostname,
1109
0
                                                          G_RESOLVER_NAME_LOOKUP_FLAGS_IPV4_ONLY,
1110
0
                                                          cancellable,
1111
0
                                                          got_ipv4_addresses, g_object_ref (addr_enum));
1112
0
            }
1113
0
          g_object_unref (resolver);
1114
0
          return;
1115
0
        }
1116
1117
0
      g_object_unref (resolver);
1118
0
    }
1119
1120
0
  sockaddr = init_and_query_next_address (addr_enum);
1121
0
  if (sockaddr == NULL && (addr_enum->state & RESOLVE_STATE_WAITING_ON_IPV4 ||
1122
0
                           addr_enum->state & RESOLVE_STATE_WAITING_ON_IPV6))
1123
0
    {
1124
0
      addr_enum->waiting_task = task;
1125
0
    }
1126
0
  else
1127
0
    {
1128
0
      g_task_return_pointer (task, sockaddr, g_object_unref);
1129
0
      g_object_unref (task);
1130
0
    }
1131
0
}
1132
1133
static GSocketAddress *
1134
g_network_address_address_enumerator_next_finish (GSocketAddressEnumerator  *enumerator,
1135
                                                  GAsyncResult              *result,
1136
                                                  GError                   **error)
1137
0
{
1138
0
  g_return_val_if_fail (g_task_is_valid (result, enumerator), NULL);
1139
1140
0
  return g_task_propagate_pointer (G_TASK (result), error);
1141
0
}
1142
1143
static void
1144
_g_network_address_address_enumerator_init (GNetworkAddressAddressEnumerator *enumerator)
1145
0
{
1146
0
  enumerator->context = g_main_context_ref_thread_default ();
1147
0
}
1148
1149
static void
1150
_g_network_address_address_enumerator_class_init (GNetworkAddressAddressEnumeratorClass *addrenum_class)
1151
0
{
1152
0
  GObjectClass *object_class = G_OBJECT_CLASS (addrenum_class);
1153
0
  GSocketAddressEnumeratorClass *enumerator_class =
1154
0
    G_SOCKET_ADDRESS_ENUMERATOR_CLASS (addrenum_class);
1155
1156
0
  enumerator_class->next = g_network_address_address_enumerator_next;
1157
0
  enumerator_class->next_async = g_network_address_address_enumerator_next_async;
1158
0
  enumerator_class->next_finish = g_network_address_address_enumerator_next_finish;
1159
0
  object_class->finalize = g_network_address_address_enumerator_finalize;
1160
0
}
1161
1162
static GSocketAddressEnumerator *
1163
g_network_address_connectable_enumerate (GSocketConnectable *connectable)
1164
0
{
1165
0
  GNetworkAddressAddressEnumerator *addr_enum;
1166
1167
0
  addr_enum = g_object_new (G_TYPE_NETWORK_ADDRESS_ADDRESS_ENUMERATOR, NULL);
1168
0
  addr_enum->addr = g_object_ref (G_NETWORK_ADDRESS (connectable));
1169
1170
0
  return (GSocketAddressEnumerator *)addr_enum;
1171
0
}
1172
1173
static GSocketAddressEnumerator *
1174
g_network_address_connectable_proxy_enumerate (GSocketConnectable *connectable)
1175
0
{
1176
0
  GNetworkAddress *self = G_NETWORK_ADDRESS (connectable);
1177
0
  GSocketAddressEnumerator *proxy_enum;
1178
0
  gchar *uri;
1179
1180
0
  uri = g_uri_join (G_URI_FLAGS_NONE,
1181
0
                    self->priv->scheme ? self->priv->scheme : "none",
1182
0
                    NULL,
1183
0
                    self->priv->hostname,
1184
0
                    self->priv->port,
1185
0
                    "",
1186
0
                    NULL,
1187
0
                    NULL);
1188
1189
0
  proxy_enum = g_object_new (G_TYPE_PROXY_ADDRESS_ENUMERATOR,
1190
0
                             "connectable", connectable,
1191
0
                             "uri", uri,
1192
0
                             NULL);
1193
1194
0
  g_free (uri);
1195
1196
0
  return proxy_enum;
1197
0
}
1198
1199
static gchar *
1200
g_network_address_connectable_to_string (GSocketConnectable *connectable)
1201
0
{
1202
0
  GNetworkAddress *addr;
1203
0
  const gchar *scheme;
1204
0
  guint16 port;
1205
0
  GString *out;  /* owned */
1206
1207
0
  addr = G_NETWORK_ADDRESS (connectable);
1208
0
  out = g_string_new ("");
1209
1210
0
  scheme = g_network_address_get_scheme (addr);
1211
0
  if (scheme != NULL)
1212
0
    g_string_append_printf (out, "%s:", scheme);
1213
1214
0
  g_string_append (out, g_network_address_get_hostname (addr));
1215
1216
0
  port = g_network_address_get_port (addr);
1217
0
  if (port != 0)
1218
0
    g_string_append_printf (out, ":%u", port);
1219
1220
0
  return g_string_free (out, FALSE);
1221
0
}