mirror of
https://gitlab.freedesktop.org/NetworkManager/NetworkManager.git
synced 2026-05-22 14:48:11 +02:00
- use _NM_GET_PRIVATE() and _NM_GET_PRIVATE_PTR() everywhere. - reorder statements, to have GObject related functions (init, dispose, constructed) at the bottom of each file and in a consistent order w.r.t. each other. - unify whitespaces in signal and properties declarations. - use NM_GOBJECT_PROPERTIES_DEFINE() and _notify() - drop unused signal slots in class structures - drop unused header files for device factories
387 lines
12 KiB
C
387 lines
12 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
|
|
/* NetworkManager -- Network link manager
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Copyright (C) 2007 - 2008 Novell, Inc.
|
|
* Copyright (C) 2007 - 2013 Red Hat, Inc.
|
|
* Copyright (C) 2013 Intel Corporation.
|
|
*/
|
|
|
|
#include "nm-default.h"
|
|
|
|
#include "nm-bluez5-manager.h"
|
|
|
|
#include <signal.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "nm-core-internal.h"
|
|
|
|
#include "nm-bluez-device.h"
|
|
#include "nm-bluez-common.h"
|
|
#include "nm-settings.h"
|
|
|
|
/*****************************************************************************/
|
|
|
|
enum {
|
|
BDADDR_ADDED,
|
|
LAST_SIGNAL,
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
typedef struct {
|
|
NMSettings *settings;
|
|
|
|
GDBusProxy *proxy;
|
|
|
|
GHashTable *devices;
|
|
} NMBluez5ManagerPrivate;
|
|
|
|
struct _NMBluez5Manager {
|
|
GObject parent;
|
|
NMBluez5ManagerPrivate _priv;
|
|
};
|
|
|
|
struct _NMBluez5ManagerClass {
|
|
GObjectClass parent;
|
|
};
|
|
|
|
G_DEFINE_TYPE (NMBluez5Manager, nm_bluez5_manager, G_TYPE_OBJECT)
|
|
|
|
#define NM_BLUEZ5_MANAGER_GET_PRIVATE(self) _NM_GET_PRIVATE (self, NMBluez5Manager, NM_IS_BLUEZ5_MANAGER)
|
|
|
|
/*****************************************************************************/
|
|
|
|
static void device_initialized (NMBluezDevice *device, gboolean success, NMBluez5Manager *self);
|
|
static void device_usable (NMBluezDevice *device, GParamSpec *pspec, NMBluez5Manager *self);
|
|
|
|
/*****************************************************************************/
|
|
|
|
static void
|
|
emit_bdaddr_added (NMBluez5Manager *self, NMBluezDevice *device)
|
|
{
|
|
g_signal_emit (self, signals[BDADDR_ADDED], 0,
|
|
device,
|
|
nm_bluez_device_get_address (device),
|
|
nm_bluez_device_get_name (device),
|
|
nm_bluez_device_get_path (device),
|
|
nm_bluez_device_get_capabilities (device));
|
|
}
|
|
|
|
void
|
|
nm_bluez5_manager_query_devices (NMBluez5Manager *self)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
NMBluezDevice *device;
|
|
GHashTableIter iter;
|
|
|
|
g_hash_table_iter_init (&iter, priv->devices);
|
|
while (g_hash_table_iter_next (&iter, NULL, (gpointer) &device)) {
|
|
if (nm_bluez_device_get_usable (device))
|
|
emit_bdaddr_added (self, device);
|
|
}
|
|
}
|
|
|
|
static void
|
|
remove_device (NMBluez5Manager *self, NMBluezDevice *device)
|
|
{
|
|
g_signal_handlers_disconnect_by_func (device, G_CALLBACK (device_initialized), self);
|
|
g_signal_handlers_disconnect_by_func (device, G_CALLBACK (device_usable), self);
|
|
if (nm_bluez_device_get_usable (device))
|
|
g_signal_emit_by_name (device, NM_BLUEZ_DEVICE_REMOVED);
|
|
}
|
|
|
|
static void
|
|
remove_all_devices (NMBluez5Manager *self)
|
|
{
|
|
GHashTableIter iter;
|
|
NMBluezDevice *device;
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
|
|
g_hash_table_iter_init (&iter, priv->devices);
|
|
while (g_hash_table_iter_next (&iter, NULL, (gpointer) &device)) {
|
|
g_hash_table_iter_steal (&iter);
|
|
remove_device (self, device);
|
|
g_object_unref (device);
|
|
}
|
|
}
|
|
|
|
static void
|
|
device_usable (NMBluezDevice *device, GParamSpec *pspec, NMBluez5Manager *self)
|
|
{
|
|
gboolean usable = nm_bluez_device_get_usable (device);
|
|
|
|
nm_log_dbg (LOGD_BT, "(%s): bluez device now %s",
|
|
nm_bluez_device_get_path (device),
|
|
usable ? "usable" : "unusable");
|
|
|
|
if (usable) {
|
|
nm_log_dbg (LOGD_BT, "(%s): bluez device address %s",
|
|
nm_bluez_device_get_path (device),
|
|
nm_bluez_device_get_address (device));
|
|
emit_bdaddr_added (self, device);
|
|
} else
|
|
g_signal_emit_by_name (device, NM_BLUEZ_DEVICE_REMOVED);
|
|
}
|
|
|
|
static void
|
|
device_initialized (NMBluezDevice *device, gboolean success, NMBluez5Manager *self)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
|
|
nm_log_dbg (LOGD_BT, "(%s): bluez device %s",
|
|
nm_bluez_device_get_path (device),
|
|
success ? "initialized" : "failed to initialize");
|
|
if (!success)
|
|
g_hash_table_remove (priv->devices, nm_bluez_device_get_path (device));
|
|
}
|
|
|
|
static void
|
|
device_added (GDBusProxy *proxy, const gchar *path, NMBluez5Manager *self)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
NMBluezDevice *device;
|
|
|
|
device = nm_bluez_device_new (path, NULL, priv->settings, 5);
|
|
g_signal_connect (device, "initialized", G_CALLBACK (device_initialized), self);
|
|
g_signal_connect (device, "notify::usable", G_CALLBACK (device_usable), self);
|
|
g_hash_table_insert (priv->devices, (gpointer) nm_bluez_device_get_path (device), device);
|
|
|
|
nm_log_dbg (LOGD_BT, "(%s): new bluez device found", path);
|
|
}
|
|
|
|
static void
|
|
device_removed (GDBusProxy *proxy, const gchar *path, NMBluez5Manager *self)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
NMBluezDevice *device;
|
|
|
|
nm_log_dbg (LOGD_BT, "(%s): bluez device removed", path);
|
|
|
|
device = g_hash_table_lookup (priv->devices, path);
|
|
if (device) {
|
|
g_hash_table_steal (priv->devices, nm_bluez_device_get_path (device));
|
|
remove_device (NM_BLUEZ5_MANAGER (self), device);
|
|
g_object_unref (device);
|
|
}
|
|
}
|
|
|
|
static void
|
|
object_manager_interfaces_added (GDBusProxy *proxy,
|
|
const char *path,
|
|
GVariant *dict,
|
|
NMBluez5Manager *self)
|
|
{
|
|
if (g_variant_lookup (dict, BLUEZ5_DEVICE_INTERFACE, "a{sv}", NULL))
|
|
device_added (proxy, path, self);
|
|
}
|
|
|
|
static void
|
|
object_manager_interfaces_removed (GDBusProxy *proxy,
|
|
const char *path,
|
|
const char **ifaces,
|
|
NMBluez5Manager *self)
|
|
{
|
|
if (ifaces && g_strv_contains (ifaces, BLUEZ5_DEVICE_INTERFACE))
|
|
device_removed (proxy, path, self);
|
|
}
|
|
|
|
static void
|
|
get_managed_objects_cb (GDBusProxy *proxy,
|
|
GAsyncResult *res,
|
|
NMBluez5Manager *self)
|
|
{
|
|
GVariant *variant, *ifaces;
|
|
GVariantIter i;
|
|
GError *error = NULL;
|
|
const char *path;
|
|
|
|
variant = _nm_dbus_proxy_call_finish (proxy, res,
|
|
G_VARIANT_TYPE ("(a{oa{sa{sv}}})"),
|
|
&error);
|
|
if (!variant) {
|
|
if (g_error_matches (error, G_DBUS_ERROR, G_DBUS_ERROR_UNKNOWN_METHOD))
|
|
nm_log_warn (LOGD_BT, "Couldn't get managed objects: not running Bluez5?");
|
|
else {
|
|
g_dbus_error_strip_remote_error (error);
|
|
nm_log_warn (LOGD_BT, "Couldn't get managed objects: %s", error->message);
|
|
}
|
|
g_clear_error (&error);
|
|
return;
|
|
}
|
|
g_variant_iter_init (&i, g_variant_get_child_value (variant, 0));
|
|
while ((g_variant_iter_next (&i, "{&o*}", &path, &ifaces))) {
|
|
if (g_variant_lookup_value (ifaces, BLUEZ5_DEVICE_INTERFACE,
|
|
G_VARIANT_TYPE_DICTIONARY)) {
|
|
device_added (proxy, path, self);
|
|
}
|
|
g_variant_unref (ifaces);
|
|
}
|
|
|
|
g_variant_unref (variant);
|
|
}
|
|
|
|
static void name_owner_changed_cb (GObject *object, GParamSpec *pspec, gpointer user_data);
|
|
|
|
static void
|
|
on_proxy_acquired (GObject *object,
|
|
GAsyncResult *res,
|
|
NMBluez5Manager *self)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
GError *error = NULL;
|
|
|
|
priv->proxy = g_dbus_proxy_new_for_bus_finish (res, &error);
|
|
|
|
if (!priv->proxy) {
|
|
nm_log_warn (LOGD_BT, "Couldn't acquire object manager proxy: %s", error->message);
|
|
g_clear_error (&error);
|
|
return;
|
|
}
|
|
|
|
g_signal_connect (priv->proxy, "notify::g-name-owner",
|
|
G_CALLBACK (name_owner_changed_cb), self);
|
|
|
|
/* Get already managed devices. */
|
|
g_dbus_proxy_call (priv->proxy, "GetManagedObjects",
|
|
NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
(GAsyncReadyCallback) get_managed_objects_cb,
|
|
self);
|
|
|
|
_nm_dbus_signal_connect (priv->proxy, "InterfacesAdded", G_VARIANT_TYPE ("(oa{sa{sv}})"),
|
|
G_CALLBACK (object_manager_interfaces_added), self);
|
|
_nm_dbus_signal_connect (priv->proxy, "InterfacesRemoved", G_VARIANT_TYPE ("(oas)"),
|
|
G_CALLBACK (object_manager_interfaces_removed), self);
|
|
}
|
|
|
|
static void
|
|
bluez_connect (NMBluez5Manager *self)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
|
|
g_return_if_fail (priv->proxy == NULL);
|
|
|
|
g_dbus_proxy_new_for_bus (G_BUS_TYPE_SYSTEM,
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
NULL,
|
|
BLUEZ_SERVICE,
|
|
BLUEZ_MANAGER_PATH,
|
|
OBJECT_MANAGER_INTERFACE,
|
|
NULL,
|
|
(GAsyncReadyCallback) on_proxy_acquired,
|
|
self);
|
|
}
|
|
|
|
static void
|
|
name_owner_changed_cb (GObject *object,
|
|
GParamSpec *pspec,
|
|
gpointer user_data)
|
|
{
|
|
NMBluez5Manager *self = NM_BLUEZ5_MANAGER (user_data);
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
char *owner;
|
|
|
|
if (priv->devices) {
|
|
owner = g_dbus_proxy_get_name_owner (priv->proxy);
|
|
if (!owner)
|
|
remove_all_devices (self);
|
|
g_free (owner);
|
|
}
|
|
}
|
|
|
|
static void
|
|
bluez_cleanup (NMBluez5Manager *self, gboolean do_signal)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
|
|
if (priv->proxy) {
|
|
g_signal_handlers_disconnect_by_func (priv->proxy, G_CALLBACK (name_owner_changed_cb), self);
|
|
g_clear_object (&priv->proxy);
|
|
}
|
|
|
|
if (do_signal)
|
|
remove_all_devices (self);
|
|
else
|
|
g_hash_table_remove_all (priv->devices);
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
static void
|
|
nm_bluez5_manager_init (NMBluez5Manager *self)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE (self);
|
|
|
|
bluez_connect (self);
|
|
|
|
priv->devices = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
NULL, g_object_unref);
|
|
}
|
|
|
|
NMBluez5Manager *
|
|
nm_bluez5_manager_new (NMSettings *settings)
|
|
{
|
|
NMBluez5Manager *instance = NULL;
|
|
|
|
g_return_val_if_fail (NM_IS_SETTINGS (settings), NULL);
|
|
|
|
instance = g_object_new (NM_TYPE_BLUEZ5_MANAGER, NULL);
|
|
NM_BLUEZ5_MANAGER_GET_PRIVATE (instance)->settings = g_object_ref (settings);
|
|
return instance;
|
|
}
|
|
|
|
static void
|
|
dispose (GObject *object)
|
|
{
|
|
NMBluez5Manager *self = NM_BLUEZ5_MANAGER (object);
|
|
|
|
bluez_cleanup (self, FALSE);
|
|
|
|
G_OBJECT_CLASS (nm_bluez5_manager_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
finalize (GObject *object)
|
|
{
|
|
NMBluez5ManagerPrivate *priv = NM_BLUEZ5_MANAGER_GET_PRIVATE ((NMBluez5Manager *) object);
|
|
|
|
g_hash_table_destroy (priv->devices);
|
|
|
|
G_OBJECT_CLASS (nm_bluez5_manager_parent_class)->finalize (object);
|
|
|
|
g_object_unref (priv->settings);
|
|
}
|
|
|
|
static void
|
|
nm_bluez5_manager_class_init (NMBluez5ManagerClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->dispose = dispose;
|
|
object_class->finalize = finalize;
|
|
|
|
signals[BDADDR_ADDED] =
|
|
g_signal_new (NM_BLUEZ_MANAGER_BDADDR_ADDED,
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
0, NULL, NULL, NULL,
|
|
G_TYPE_NONE, 5, G_TYPE_OBJECT, G_TYPE_STRING,
|
|
G_TYPE_STRING, G_TYPE_STRING, G_TYPE_UINT);
|
|
}
|