/* GIO - GLib Input, Output and Streaming Library * * Copyright 2011 Red Hat, Inc. * * SPDX-License-Identifier: LGPL-2.1-or-later * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General * Public License along with this library; if not, see <http://www.gnu.org/licenses/>. */ #include "config.h" #include "gnetworkmonitorbase.h" #include "ginetaddress.h" #include "ginetaddressmask.h" #include "ginetsocketaddress.h" #include "ginitable.h" #include "gioerror.h" #include "giomodule-priv.h" #include "gnetworkmonitor.h" #include "gsocketaddressenumerator.h" #include "gsocketconnectable.h" #include "gtask.h" #include "glibintl.h" static void g_network_monitor_base_iface_init (GNetworkMonitorInterface *iface); static void g_network_monitor_base_initable_iface_init (GInitableIface *iface); enum { PROP_0, PROP_NETWORK_AVAILABLE, PROP_NETWORK_METERED, PROP_CONNECTIVITY }; struct _GNetworkMonitorBasePrivate { GHashTable *networks /* (element-type GInetAddressMask) (owned) */; gboolean have_ipv4_default_route; gboolean have_ipv6_default_route; gboolean is_available; GMainContext *context; GSource *network_changed_source; gboolean initializing; }; static guint network_changed_signal = 0; static void queue_network_changed (GNetworkMonitorBase *monitor); static guint inet_address_mask_hash (gconstpointer key); static gboolean inet_address_mask_equal (gconstpointer a, gconstpointer b); G_DEFINE_TYPE_WITH_CODE (GNetworkMonitorBase, g_network_monitor_base, G_TYPE_OBJECT, G_ADD_PRIVATE (GNetworkMonitorBase) G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE, g_network_monitor_base_initable_iface_init) G_IMPLEMENT_INTERFACE (G_TYPE_NETWORK_MONITOR, g_network_monitor_base_iface_init) _g_io_modules_ensure_extension_points_registered (); g_io_extension_point_implement (G_NETWORK_MONITOR_EXTENSION_POINT_NAME, g_define_type_id, "base", 0)) static void g_network_monitor_base_init (GNetworkMonitorBase *monitor) { monitor->priv = g_network_monitor_base_get_instance_private (monitor); monitor->priv->networks = g_hash_table_new_full (inet_address_mask_hash, inet_address_mask_equal, g_object_unref, NULL); monitor->priv->context = g_main_context_get_thread_default (); if (monitor->priv->context) g_main_context_ref (monitor->priv->context); monitor->priv->initializing = TRUE; } static void g_network_monitor_base_constructed (GObject *object) { GNetworkMonitorBase *monitor = G_NETWORK_MONITOR_BASE (object); if (G_OBJECT_TYPE (monitor) == G_TYPE_NETWORK_MONITOR_BASE) { GInetAddressMask *mask; /* We're the dumb base class, not a smarter subclass. So just * assume that the network is available. */ mask = g_inet_address_mask_new_from_string ("0.0.0.0/0", NULL); g_network_monitor_base_add_network (monitor, mask); g_object_unref (mask); mask = g_inet_address_mask_new_from_string ("::/0", NULL); if (mask) { /* On some environments (for example Windows without IPv6 support * enabled) the string "::/0" can't be processed and causes * g_inet_address_mask_new_from_string to return NULL */ g_network_monitor_base_add_network (monitor, mask); g_object_unref (mask); } } } static void g_network_monitor_base_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GNetworkMonitorBase *monitor = G_NETWORK_MONITOR_BASE (object); switch (prop_id) { case PROP_NETWORK_AVAILABLE: g_value_set_boolean (value, monitor->priv->is_available); break; case PROP_NETWORK_METERED: /* Default to FALSE in the unknown case. */ g_value_set_boolean (value, FALSE); break; case PROP_CONNECTIVITY: g_value_set_enum (value, monitor->priv->is_available ? G_NETWORK_CONNECTIVITY_FULL : G_NETWORK_CONNECTIVITY_LOCAL); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void g_network_monitor_base_finalize (GObject *object) { GNetworkMonitorBase *monitor = G_NETWORK_MONITOR_BASE (object); g_hash_table_unref (monitor->priv->networks); if (monitor->priv->network_changed_source) { g_source_destroy (monitor->priv->network_changed_source); g_source_unref (monitor->priv->network_changed_source); } if (monitor->priv->context) g_main_context_unref (monitor->priv->context); G_OBJECT_CLASS (g_network_monitor_base_parent_class)->finalize (object); } static void g_network_monitor_base_class_init (GNetworkMonitorBaseClass *monitor_class) { GObjectClass *gobject_class = G_OBJECT_CLASS (monitor_class); gobject_class->constructed = g_network_monitor_base_constructed; gobject_class->get_property = g_network_monitor_base_get_property; gobject_class->finalize = g_network_monitor_base_finalize; g_object_class_override_property (gobject_class, PROP_NETWORK_AVAILABLE, "network-available"); g_object_class_override_property (gobject_class, PROP_NETWORK_METERED, "network-metered"); g_object_class_override_property (gobject_class, PROP_CONNECTIVITY, "connectivity"); } static gboolean g_network_monitor_base_can_reach_sockaddr (GNetworkMonitorBase *base, GSocketAddress *sockaddr) { GInetAddress *iaddr; GHashTableIter iter; gpointer key; if (!G_IS_INET_SOCKET_ADDRESS (sockaddr)) return FALSE; iaddr = g_inet_socket_address_get_address (G_INET_SOCKET_ADDRESS (sockaddr)); g_hash_table_iter_init (&iter, base->priv->networks); while (g_hash_table_iter_next (&iter, &key, NULL)) { GInetAddressMask *mask = key; if (g_inet_address_mask_matches (mask, iaddr)) return TRUE; } return FALSE; } static gboolean g_network_monitor_base_can_reach (GNetworkMonitor *monitor, GSocketConnectable *connectable, GCancellable *cancellable, GError **error) { GNetworkMonitorBase *base = G_NETWORK_MONITOR_BASE (monitor); GSocketAddressEnumerator *enumerator; GSocketAddress *addr; if (g_hash_table_size (base->priv->networks) == 0) { g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NETWORK_UNREACHABLE, _("Network unreachable")); return FALSE; } enumerator = g_socket_connectable_proxy_enumerate (connectable); addr = g_socket_address_enumerator_next (enumerator, cancellable, error); if (!addr) { /* Either the user cancelled, or DNS resolution failed */ g_object_unref (enumerator); return FALSE; } if (base->priv->have_ipv4_default_route && base->priv->have_ipv6_default_route) { g_object_unref (enumerator); g_object_unref (addr); return TRUE; } while (addr) { if (g_network_monitor_base_can_reach_sockaddr (base, addr)) { g_object_unref (addr); g_object_unref (enumerator); return TRUE; } g_object_unref (addr); addr = g_socket_address_enumerator_next (enumerator, cancellable, error); } g_object_unref (enumerator); if (error && !*error) { g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_HOST_UNREACHABLE, _("Host unreachable")); } return FALSE; } static void can_reach_async_got_address (GObject *object, GAsyncResult *result, gpointer user_data) { GSocketAddressEnumerator *enumerator = G_SOCKET_ADDRESS_ENUMERATOR (object); GTask *task = user_data; GNetworkMonitorBase *base = g_task_get_source_object (task); GSocketAddress *addr; GError *error = NULL; addr = g_socket_address_enumerator_next_finish (enumerator, result, &error); if (!addr) { if (error) { /* Either the user cancelled, or DNS resolution failed */ g_task_return_error (task, error); g_object_unref (task); return; } else { /* Resolved all addresses, none matched */ g_task_return_new_error_literal (task, G_IO_ERROR, G_IO_ERROR_HOST_UNREACHABLE, _("Host unreachable")); g_object_unref (task); return; } } if (g_network_monitor_base_can_reach_sockaddr (base, addr)) { g_object_unref (addr); g_task_return_boolean (task, TRUE); g_object_unref (task); return; } g_object_unref (addr); g_socket_address_enumerator_next_async (enumerator, g_task_get_cancellable (task), can_reach_async_got_address, task); } static void g_network_monitor_base_can_reach_async (GNetworkMonitor *monitor, GSocketConnectable *connectable, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { GTask *task; GSocketAddressEnumerator *enumerator; task = g_task_new (monitor, cancellable, callback, user_data); g_task_set_source_tag (task, g_network_monitor_base_can_reach_async); if (g_hash_table_size (G_NETWORK_MONITOR_BASE (monitor)->priv->networks) == 0) { g_task_return_new_error_literal (task, G_IO_ERROR, G_IO_ERROR_NETWORK_UNREACHABLE, _("Network unreachable")); g_object_unref (task); return; } enumerator = g_socket_connectable_proxy_enumerate (connectable); g_socket_address_enumerator_next_async (enumerator, cancellable, can_reach_async_got_address, task); g_object_unref (enumerator); } static gboolean g_network_monitor_base_can_reach_finish (GNetworkMonitor *monitor, GAsyncResult *result, GError **error) { g_return_val_if_fail (g_task_is_valid (result, monitor), FALSE); return g_task_propagate_boolean (G_TASK (result), error); } static void g_network_monitor_base_iface_init (GNetworkMonitorInterface *monitor_iface) { monitor_iface->can_reach = g_network_monitor_base_can_reach; monitor_iface->can_reach_async = g_network_monitor_base_can_reach_async; monitor_iface->can_reach_finish = g_network_monitor_base_can_reach_finish; network_changed_signal = g_signal_lookup ("network-changed", G_TYPE_NETWORK_MONITOR); } static gboolean g_network_monitor_base_initable_init (GInitable *initable, GCancellable *cancellable, GError **error) { GNetworkMonitorBase *base = G_NETWORK_MONITOR_BASE (initable); base->priv->initializing = FALSE; return TRUE; } static void g_network_monitor_base_initable_iface_init (GInitableIface *iface) { iface->init = g_network_monitor_base_initable_init; } static guint inet_address_mask_hash (gconstpointer key) { GInetAddressMask *mask = G_INET_ADDRESS_MASK (key); guint addr_hash; guint mask_length = g_inet_address_mask_get_length (mask); GInetAddress *addr = g_inet_address_mask_get_address (mask); const guint8 *bytes = g_inet_address_to_bytes (addr); gsize bytes_length = g_inet_address_get_native_size (addr); union { const guint8 *bytes; guint32 *hash32; guint64 *hash64; } integerifier; /* If we can fit the entire address into the hash key, do it. Don’t worry * about endianness; the address should always be in network endianness. */ if (bytes_length == sizeof (guint32)) { integerifier.bytes = bytes; addr_hash = *integerifier.hash32; } else if (bytes_length == sizeof (guint64)) { integerifier.bytes = bytes; addr_hash = *integerifier.hash64; } else { gsize i; /* Otherwise, fall back to adding the bytes together. We do this, rather * than XORing them, as routes often have repeated tuples which would * cancel out under XOR. */ addr_hash = 0; for (i = 0; i < bytes_length; i++) addr_hash += bytes[i]; } return addr_hash + mask_length;; } static gboolean inet_address_mask_equal (gconstpointer a, gconstpointer b) { GInetAddressMask *mask_a = G_INET_ADDRESS_MASK (a); GInetAddressMask *mask_b = G_INET_ADDRESS_MASK (b); return g_inet_address_mask_equal (mask_a, mask_b); } static gboolean emit_network_changed (gpointer user_data) { GNetworkMonitorBase *monitor = user_data; gboolean is_available; if (g_source_is_destroyed (g_main_current_source ())) return FALSE; g_object_ref (monitor); is_available = (monitor->priv->have_ipv4_default_route || monitor->priv->have_ipv6_default_route); if (monitor->priv->is_available != is_available) { monitor->priv->is_available = is_available; g_object_notify (G_OBJECT (monitor), "network-available"); } g_signal_emit (monitor, network_changed_signal, 0, is_available); g_source_unref (monitor->priv->network_changed_source); monitor->priv->network_changed_source = NULL; g_object_unref (monitor); return FALSE; } static void queue_network_changed (GNetworkMonitorBase *monitor) { if (!monitor->priv->network_changed_source && !monitor->priv->initializing) { GSource *source; source = g_idle_source_new (); /* Use G_PRIORITY_HIGH_IDLE priority so that multiple * network-change-related notifications coming in at * G_PRIORITY_DEFAULT will get coalesced into one signal * emission. */ g_source_set_priority (source, G_PRIORITY_HIGH_IDLE); g_source_set_callback (source, emit_network_changed, monitor, NULL); g_source_set_static_name (source, "[gio] emit_network_changed"); g_source_attach (source, monitor->priv->context); monitor->priv->network_changed_source = source; } /* Normally we wait to update is_available until we emit the signal, * to keep things consistent. But when we're first creating the * object, we want it to be correct right away. */ if (monitor->priv->initializing) { monitor->priv->is_available = (monitor->priv->have_ipv4_default_route || monitor->priv->have_ipv6_default_route); } } /** * g_network_monitor_base_add_network: * @monitor: the #GNetworkMonitorBase * @network: (transfer none): a #GInetAddressMask * * Adds @network to @monitor's list of available networks. * * Since: 2.32 */ void g_network_monitor_base_add_network (GNetworkMonitorBase *monitor, GInetAddressMask *network) { if (!g_hash_table_add (monitor->priv->networks, g_object_ref (network))) return; if (g_inet_address_mask_get_length (network) == 0) { switch (g_inet_address_mask_get_family (network)) { case G_SOCKET_FAMILY_IPV4: monitor->priv->have_ipv4_default_route = TRUE; break; case G_SOCKET_FAMILY_IPV6: monitor->priv->have_ipv6_default_route = TRUE; break; default: break; } } /* Don't emit network-changed when multicast-link-local routing * changes. This rather arbitrary decision is mostly because it * seems to change quite often... */ if (g_inet_address_get_is_mc_link_local (g_inet_address_mask_get_address (network))) return; queue_network_changed (monitor); } /** * g_network_monitor_base_remove_network: * @monitor: the #GNetworkMonitorBase * @network: a #GInetAddressMask * * Removes @network from @monitor's list of available networks. * * Since: 2.32 */ void g_network_monitor_base_remove_network (GNetworkMonitorBase *monitor, GInetAddressMask *network) { if (!g_hash_table_remove (monitor->priv->networks, network)) return; if (g_inet_address_mask_get_length (network) == 0) { switch (g_inet_address_mask_get_family (network)) { case G_SOCKET_FAMILY_IPV4: monitor->priv->have_ipv4_default_route = FALSE; break; case G_SOCKET_FAMILY_IPV6: monitor->priv->have_ipv6_default_route = FALSE; break; default: break; } } queue_network_changed (monitor); } /** * g_network_monitor_base_set_networks: * @monitor: the #GNetworkMonitorBase * @networks: (array length=length): an array of #GInetAddressMask * @length: length of @networks * * Drops @monitor's current list of available networks and replaces * it with @networks. */ void g_network_monitor_base_set_networks (GNetworkMonitorBase *monitor, GInetAddressMask **networks, gint length) { int i; g_hash_table_remove_all (monitor->priv->networks); monitor->priv->have_ipv4_default_route = FALSE; monitor->priv->have_ipv6_default_route = FALSE; for (i = 0; i < length; i++) g_network_monitor_base_add_network (monitor, networks[i]); }