*
* Connection Manager
*
- * Copyright (C) 2007-2012 Intel Corporation. All rights reserved.
+ * Copyright (C) 2013 Intel Corporation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
#include <config.h>
#endif
-#include <stdio.h>
#include <errno.h>
-#include <stdlib.h>
#include <string.h>
-#include <netinet/ether.h>
-
-#include <gdbus.h>
#define CONNMAN_API_SUBJECT_TO_CHANGE
#include <connman/plugin.h>
+#include <connman/dbus.h>
#include <connman/technology.h>
#include <connman/device.h>
#include <connman/inet.h>
-#include <connman/dbus.h>
-#include <connman/log.h>
-
-#define BLUEZ_SERVICE "org.bluez"
-#define BLUEZ_MANAGER_INTERFACE BLUEZ_SERVICE ".Manager"
-#define BLUEZ_ADAPTER_INTERFACE BLUEZ_SERVICE ".Adapter"
-#define BLUEZ_DEVICE_INTERFACE BLUEZ_SERVICE ".Device"
-#define BLUEZ_NETWORK_INTERFACE BLUEZ_SERVICE ".Network"
-#define BLUEZ_NETWORK_SERVER BLUEZ_SERVICE ".NetworkServer"
-
-#define LIST_ADAPTERS "ListAdapters"
-#define ADAPTER_ADDED "AdapterAdded"
-#define ADAPTER_REMOVED "AdapterRemoved"
-#define DEVICE_REMOVED "DeviceRemoved"
-
-#define PROPERTY_CHANGED "PropertyChanged"
-#define GET_PROPERTIES "GetProperties"
-#define SET_PROPERTY "SetProperty"
+#include <gdbus.h>
-#define CONNECT "Connect"
-#define DISCONNECT "Disconnect"
+#define BLUEZ_SERVICE "org.bluez"
+#define BLUEZ_PATH "/org/bluez"
+#define BLUETOOTH_PAN_PANU "00001115-0000-1000-8000-00805f9b34fb"
+#define BLUETOOTH_PAN_NAP "00001116-0000-1000-8000-00805f9b34fb"
+#define BLUETOOTH_PAN_GN "00001117-0000-1000-8000-00805f9b34fb"
-#define REGISTER "Register"
-#define UNREGISTER "Unregister"
+#define BLUETOOTH_ADDR_LEN 6
-#define UUID_NAP "00001116-0000-1000-8000-00805f9b34fb"
+static DBusConnection *connection;
+static GDBusClient *client;
+static GHashTable *devices;
+static GHashTable *networks;
+static bool bluetooth_tethering;
-#define TIMEOUT 5000
+struct bluetooth_pan {
+ struct connman_network *network;
+ GDBusProxy *btdevice_proxy;
+ GDBusProxy *btnetwork_proxy;
+ const char *pan_role;
+};
-static DBusConnection *connection;
+static void address2ident(const char *address, char *ident)
+{
+ int i;
-static GHashTable *bluetooth_devices = NULL;
-static GHashTable *bluetooth_networks = NULL;
+ for (i = 0; i < BLUETOOTH_ADDR_LEN; i++) {
+ ident[i * 2] = address[i * 3];
+ ident[i * 2 + 1] = address[i * 3 + 1];
+ }
+ ident[BLUETOOTH_ADDR_LEN * 2] = '\0';
+}
-static int pan_probe(struct connman_network *network)
+static const char *proxy_get_string(GDBusProxy *proxy, const char *property)
{
- DBG("network %p", network);
+ DBusMessageIter iter;
+ const char *str;
- return 0;
+ if (!g_dbus_proxy_get_property(proxy, property, &iter))
+ return NULL;
+ dbus_message_iter_get_basic(&iter, &str);
+ return str;
}
-static void pan_remove(struct connman_network *network)
+static bool proxy_get_bool(GDBusProxy *proxy, const char *property)
{
- DBG("network %p", network);
+ DBusMessageIter iter;
+ dbus_bool_t value;
+
+ if (!g_dbus_proxy_get_property(proxy, property, &iter))
+ return false;
+ dbus_message_iter_get_basic(&iter, &value);
+ return value;
}
-static void connect_reply(DBusPendingCall *call, void *user_data)
+static const char *proxy_get_role(GDBusProxy *proxy)
{
- struct connman_network *network = user_data;
- DBusMessage *reply;
- DBusError error;
- const char *interface = NULL;
- int index;
+ DBusMessageIter iter, value;
+ const char *pref = NULL;
- DBG("network %p", network);
+ if (!proxy)
+ return NULL;
- reply = dbus_pending_call_steal_reply(call);
+ if (!g_dbus_proxy_get_property(proxy, "UUIDs", &iter))
+ return NULL;
- dbus_error_init(&error);
+ dbus_message_iter_recurse(&iter, &value);
+ while (dbus_message_iter_get_arg_type(&value) == DBUS_TYPE_STRING) {
+ const char *uuid;
- if (dbus_set_error_from_message(&error, reply) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
+ dbus_message_iter_get_basic(&value, &uuid);
+ /*
+ * If a device offers more than one role, we prefer NAP,
+ * then GN, then PANU.
+ */
+ if (!strcmp(uuid, BLUETOOTH_PAN_NAP))
+ return "nap";
+ if (!strcmp(uuid, BLUETOOTH_PAN_GN))
+ pref = "gn";
+ if (!strcmp(uuid, BLUETOOTH_PAN_PANU) && !pref)
+ pref = "panu";
- goto err;
- }
+ dbus_message_iter_next(&value);
+ }
- if (dbus_message_get_args(reply, &error,
- DBUS_TYPE_STRING, &interface,
- DBUS_TYPE_INVALID) == FALSE) {
- if (dbus_error_is_set(&error) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- } else
- connman_error("Wrong arguments for connect");
- goto err;
- }
+ return pref;
+}
- if (interface == NULL)
- goto err;
+static int bluetooth_pan_probe(struct connman_network *network)
+{
+ GHashTableIter iter;
+ gpointer key, value;
- DBG("interface %s", interface);
+ DBG("network %p", network);
- index = connman_inet_ifindex(interface);
+ g_hash_table_iter_init(&iter, networks);
- connman_network_set_index(network, index);
+ while (g_hash_table_iter_next(&iter, &key, &value)) {
+ struct bluetooth_pan *pan = value;
- connman_network_set_connected(network, TRUE);
+ if (network == pan->network)
+ return 0;
+ }
+
+ return -EOPNOTSUPP;
+}
- dbus_message_unref(reply);
+static void pan_remove_nap(struct bluetooth_pan *pan)
+{
+ struct connman_device *device;
+ struct connman_network *network = pan->network;
- dbus_pending_call_unref(call);
+ DBG("network %p pan %p", pan->network, pan);
- return;
-err:
+ if (!network)
+ return;
- connman_network_set_connected(network, FALSE);
+ pan->network = NULL;
+ connman_network_set_data(network, NULL);
- dbus_message_unref(reply);
+ device = connman_network_get_device(network);
+ if (device)
+ connman_device_remove_network(device, network);
- dbus_pending_call_unref(call);
+ connman_network_unref(network);
}
-static int pan_connect(struct connman_network *network)
+static void bluetooth_pan_remove(struct connman_network *network)
{
- const char *path = connman_network_get_string(network, "Path");
- const char *uuid = "nap";
- DBusMessage *message;
- DBusPendingCall *call;
+ struct bluetooth_pan *pan = connman_network_get_data(network);
- DBG("network %p", network);
+ DBG("network %p pan %p", network, pan);
- if (path == NULL)
- return -EINVAL;
-
- message = dbus_message_new_method_call(BLUEZ_SERVICE, path,
- BLUEZ_NETWORK_INTERFACE, CONNECT);
- if (message == NULL)
- return -ENOMEM;
+ connman_network_set_data(network, NULL);
- dbus_message_set_auto_start(message, FALSE);
+ if (pan)
+ pan_remove_nap(pan);
+}
- dbus_message_append_args(message, DBUS_TYPE_STRING, &uuid,
- DBUS_TYPE_INVALID);
+static bool pan_connect(struct bluetooth_pan *pan,
+ const char *iface)
+{
+ int index;
- if (dbus_connection_send_with_reply(connection, message,
- &call, TIMEOUT * 10) == FALSE) {
- connman_error("Failed to connect service");
- dbus_message_unref(message);
- return -EINVAL;
+ if (!iface) {
+ if (!proxy_get_bool(pan->btnetwork_proxy, "Connected"))
+ return false;
+ iface = proxy_get_string(pan->btnetwork_proxy, "Interface");
}
- if (call == NULL) {
- connman_error("D-Bus connection not available");
- dbus_message_unref(message);
- return -EINVAL;
- }
+ if (!iface)
+ return false;
- dbus_pending_call_set_notify(call, connect_reply, network, NULL);
+ index = connman_inet_ifindex(iface);
+ if (index < 0) {
+ DBG("network %p invalid index %d", pan->network, index);
+ return false;
+ }
- dbus_message_unref(message);
+#if defined TIZEN_EXT
+ if (pan->network) {
+#endif
+ connman_network_set_index(pan->network, index);
+ connman_network_set_connected(pan->network, true);
+#if defined TIZEN_EXT
+ }
+#endif
- return -EINPROGRESS;
+ return true;
}
-static void disconnect_reply(DBusPendingCall *call, void *user_data)
+static void pan_connect_cb(DBusMessage *message, void *user_data)
{
- struct connman_network *network = user_data;
- DBusMessage *reply;
- DBusError error;
-
- DBG("network %p", network);
+ const char *path = user_data;
+ const char *iface = NULL;
+ struct bluetooth_pan *pan;
+ DBusMessageIter iter;
- reply = dbus_pending_call_steal_reply(call);
+ pan = g_hash_table_lookup(networks, path);
+ if (!pan || !pan->network) {
+ DBG("network already removed");
+ return;
+ }
- dbus_error_init(&error);
+ if (dbus_message_get_type(message) == DBUS_MESSAGE_TYPE_ERROR) {
+ const char *dbus_error = dbus_message_get_error_name(message);
- if (dbus_set_error_from_message(&error, reply) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- goto done;
- }
+ DBG("network %p %s", pan->network, dbus_error);
- if (dbus_message_get_args(reply, &error, DBUS_TYPE_INVALID) == FALSE) {
- if (dbus_error_is_set(&error) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- } else
- connman_error("Wrong arguments for disconnect");
- goto done;
+ if (strcmp(dbus_error,
+ "org.bluez.Error.AlreadyConnected") != 0) {
+ connman_network_set_associating(pan->network, false);
+ connman_network_set_error(pan->network,
+ CONNMAN_NETWORK_ERROR_ASSOCIATE_FAIL);
+ return;
+ }
+ } else {
+ if (dbus_message_iter_init(message, &iter) &&
+ dbus_message_iter_get_arg_type(&iter) ==
+ DBUS_TYPE_STRING)
+ dbus_message_iter_get_basic(&iter, &iface);
}
- connman_network_set_connected(network, FALSE);
+ DBG("network %p interface %s", pan->network, iface);
-done:
- dbus_message_unref(reply);
+ pan_connect(pan, iface);
+}
- dbus_pending_call_unref(call);
+static void pan_connect_append(DBusMessageIter *iter,
+ void *user_data)
+{
+ const char *path = user_data;
+ struct bluetooth_pan *pan;
- connman_network_unref(network);
+ pan = g_hash_table_lookup(networks, path);
+ dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &pan->pan_role);
}
-static int pan_disconnect(struct connman_network *network)
+static int bluetooth_pan_connect(struct connman_network *network)
{
- const char *path = connman_network_get_string(network, "Path");
- DBusMessage *message;
- DBusPendingCall *call;
+ struct bluetooth_pan *pan = connman_network_get_data(network);
+ const char *path;
DBG("network %p", network);
- if (path == NULL)
+ if (!pan)
return -EINVAL;
- message = dbus_message_new_method_call(BLUEZ_SERVICE, path,
- BLUEZ_NETWORK_INTERFACE, DISCONNECT);
- if (message == NULL)
- return -ENOMEM;
+ path = g_dbus_proxy_get_path(pan->btnetwork_proxy);
- dbus_message_set_auto_start(message, FALSE);
+ if (!g_dbus_proxy_method_call(pan->btnetwork_proxy, "Connect",
+ pan_connect_append, pan_connect_cb,
+ g_strdup(path), g_free))
+ return -EIO;
- dbus_message_append_args(message, DBUS_TYPE_INVALID);
+#if defined TIZEN_EXT
+ if (pan->network)
+#endif
+ connman_network_set_associating(pan->network, true);
- if (dbus_connection_send_with_reply(connection, message,
- &call, TIMEOUT) == FALSE) {
- connman_error("Failed to disconnect service");
- dbus_message_unref(message);
- return -EINVAL;
- }
+ return -EINPROGRESS;
+}
- if (call == NULL) {
- connman_error("D-Bus connection not available");
- dbus_message_unref(message);
- return -EINVAL;
- }
+static void pan_disconnect_cb(DBusMessage *message, void *user_data)
+{
+ const char *path = user_data;
+ struct bluetooth_pan *pan;
- connman_network_ref(network);
+ pan = g_hash_table_lookup(networks, path);
+ if (!pan || !pan->network) {
+ DBG("network already removed");
+ return;
+ }
- connman_network_set_associating(network, FALSE);
+ if (dbus_message_get_type(message) == DBUS_MESSAGE_TYPE_ERROR) {
+ const char *dbus_error = dbus_message_get_error_name(message);
- dbus_pending_call_set_notify(call, disconnect_reply, network, NULL);
+ DBG("network %p %s", pan->network, dbus_error);
+ }
- dbus_message_unref(message);
+ DBG("network %p", pan->network);
- return 0;
+#if defined TIZEN_EXT
+ if (pan->network)
+#endif
+ connman_network_set_connected(pan->network, false);
}
-static struct connman_network_driver pan_driver = {
- .name = "bluetooth-pan",
- .type = CONNMAN_NETWORK_TYPE_BLUETOOTH_PAN,
- .probe = pan_probe,
- .remove = pan_remove,
- .connect = pan_connect,
- .disconnect = pan_disconnect,
-};
-
-static gboolean network_changed(DBusConnection *conn,
- DBusMessage *message, void *user_data)
+static int bluetooth_pan_disconnect(struct connman_network *network)
{
- const char *path = dbus_message_get_path(message);
- struct connman_network *network;
- DBusMessageIter iter, value;
- const char *key;
-
- DBG("path %s", path);
-
- network = g_hash_table_lookup(bluetooth_networks, path);
- if (network == NULL)
- return TRUE;
-
- if (dbus_message_iter_init(message, &iter) == FALSE)
- return TRUE;
-
- dbus_message_iter_get_basic(&iter, &key);
+ struct bluetooth_pan *pan = connman_network_get_data(network);
+ const char *path;
- dbus_message_iter_next(&iter);
- dbus_message_iter_recurse(&iter, &value);
+ DBG("network %p", network);
- if (g_str_equal(key, "Connected") == TRUE) {
- dbus_bool_t connected;
+ if (!pan)
+ return -EINVAL;
- dbus_message_iter_get_basic(&value, &connected);
+#if defined TIZEN_EXT
+ if (connman_network_get_associating(network) == TRUE)
+ connman_network_clear_associating(network);
+#endif
- if (connected == TRUE)
- return TRUE;
+ path = g_dbus_proxy_get_path(pan->btnetwork_proxy);
- connman_network_set_associating(network, FALSE);
- connman_network_set_connected(network, FALSE);
- }
+ if (!g_dbus_proxy_method_call(pan->btnetwork_proxy, "Disconnect",
+ NULL, pan_disconnect_cb, g_strdup(path), g_free))
+ return -EIO;
- return TRUE;
+ return -EINPROGRESS;
}
-static void extract_properties(DBusMessage *reply, const char **parent,
- const char **address,
- const char **name,
- const char **alias,
- dbus_bool_t *powered,
- dbus_bool_t *scanning,
- DBusMessageIter *uuids,
- DBusMessageIter *networks)
+static void btnetwork_property_change(GDBusProxy *proxy, const char *name,
+ DBusMessageIter *iter, void *user_data)
{
- DBusMessageIter array, dict;
+ struct bluetooth_pan *pan;
+ dbus_bool_t connected;
+ bool proxy_connected, network_connected;
- if (dbus_message_iter_init(reply, &array) == FALSE)
+ if (strcmp(name, "Connected") != 0)
return;
- if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
+ pan = g_hash_table_lookup(networks, g_dbus_proxy_get_path(proxy));
+ if (!pan || !pan->network)
return;
- dbus_message_iter_recurse(&array, &dict);
-
- while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
- DBusMessageIter entry, value;
- const char *key;
-
- dbus_message_iter_recurse(&dict, &entry);
- dbus_message_iter_get_basic(&entry, &key);
-
- dbus_message_iter_next(&entry);
- dbus_message_iter_recurse(&entry, &value);
-
- if (g_str_equal(key, "Adapter") == TRUE) {
- if (parent != NULL)
- dbus_message_iter_get_basic(&value, parent);
- } else if (g_str_equal(key, "Address") == TRUE) {
- if (address != NULL)
- dbus_message_iter_get_basic(&value, address);
- } else if (g_str_equal(key, "Name") == TRUE) {
- if (name != NULL)
- dbus_message_iter_get_basic(&value, name);
- } else if (g_str_equal(key, "Alias") == TRUE) {
- if (alias != NULL)
- dbus_message_iter_get_basic(&value, alias);
- } else if (g_str_equal(key, "Powered") == TRUE) {
- if (powered != NULL)
- dbus_message_iter_get_basic(&value, powered);
- } else if (g_str_equal(key, "Discovering") == TRUE) {
- if (scanning != NULL)
- dbus_message_iter_get_basic(&value, scanning);
- } else if (g_str_equal(key, "Devices") == TRUE) {
- if (networks != NULL)
- memcpy(networks, &value, sizeof(value));
- } else if (g_str_equal(key, "UUIDs") == TRUE) {
- if (uuids != NULL)
- memcpy(uuids, &value, sizeof(value));
- }
-
- dbus_message_iter_next(&dict);
- }
-}
-
-static dbus_bool_t has_pan(DBusMessageIter *array)
-{
- DBusMessageIter value;
-
- if (dbus_message_iter_get_arg_type(array) != DBUS_TYPE_ARRAY)
- return FALSE;
-
- dbus_message_iter_recurse(array, &value);
-
- while (dbus_message_iter_get_arg_type(&value) == DBUS_TYPE_STRING) {
- const char *uuid;
-
- dbus_message_iter_get_basic(&value, &uuid);
+ dbus_message_iter_get_basic(iter, &connected);
+ proxy_connected = connected;
- if (g_strcmp0(uuid, UUID_NAP) == 0)
- return TRUE;
+ network_connected = connman_network_get_connected(pan->network);
- dbus_message_iter_next(&value);
- }
+ DBG("network %p network connected %d proxy connected %d",
+ pan->network, network_connected, proxy_connected);
- return FALSE;
+ if (network_connected != proxy_connected)
+ connman_network_set_connected(pan->network, proxy_connected);
}
-static void network_properties_reply(DBusPendingCall *call, void *user_data)
+static void pan_create_nap(struct bluetooth_pan *pan)
{
- char *path = user_data;
struct connman_device *device;
- struct connman_network *network;
- DBusMessage *reply;
- DBusMessageIter uuids;
- const char *parent = NULL, *address = NULL, *name = NULL;
- struct ether_addr addr;
- char ident[13];
+ const char* role;
+ const char *adapter;
- reply = dbus_pending_call_steal_reply(call);
-
- extract_properties(reply, &parent, &address, NULL, &name,
- NULL, NULL, &uuids, NULL);
-
- if (parent == NULL)
- goto done;
+ role = proxy_get_role(pan->btdevice_proxy);
+ if (!role) {
+ pan_remove_nap(pan);
+ return;
+ }
- device = g_hash_table_lookup(bluetooth_devices, parent);
- if (device == NULL)
- goto done;
+ adapter = proxy_get_string(pan->btdevice_proxy, "Adapter");
- if (address == NULL)
- goto done;
+ if (!adapter)
+ return;
- ether_aton_r(address, &addr);
+ device = g_hash_table_lookup(devices, adapter);
- snprintf(ident, 13, "%02x%02x%02x%02x%02x%02x",
- addr.ether_addr_octet[0],
- addr.ether_addr_octet[1],
- addr.ether_addr_octet[2],
- addr.ether_addr_octet[3],
- addr.ether_addr_octet[4],
- addr.ether_addr_octet[5]);
+ if (!device || !connman_device_get_powered(device))
+ return;
- if (has_pan(&uuids) == FALSE)
- goto done;
+ if (!pan->network) {
+ const char *address;
+ char ident[BLUETOOTH_ADDR_LEN * 2 + 1];
+ const char *name, *path;
- network = connman_device_get_network(device, ident);
- if (network != NULL)
- goto done;
+ address = proxy_get_string(pan->btdevice_proxy, "Address");
+ if (!address) {
+ connman_warn("Bluetooth device address missing");
+ return;
+ }
- network = connman_network_create(ident,
- CONNMAN_NETWORK_TYPE_BLUETOOTH_PAN);
- if (network == NULL)
- goto done;
+ address2ident(address, ident);
- connman_network_set_string(network, "Path", path);
+ pan->network = connman_network_create(ident,
+ CONNMAN_NETWORK_TYPE_BLUETOOTH_PAN);
- connman_network_set_name(network, name);
+ name = proxy_get_string(pan->btdevice_proxy, "Alias");
+ path = g_dbus_proxy_get_path(pan->btnetwork_proxy);
- connman_device_add_network(device, network);
+ DBG("network %p %s %s", pan->network, path, name);
- connman_network_set_group(network, ident);
+ if (!pan->network) {
+ connman_warn("Bluetooth network %s creation failed",
+ path);
+ return;
+ }
- g_hash_table_replace(bluetooth_networks, g_strdup(path), network);
+ connman_network_set_data(pan->network, pan);
+ connman_network_set_name(pan->network, name);
+ connman_network_set_group(pan->network, ident);
+ }
-done:
- dbus_message_unref(reply);
+ pan->pan_role = role;
+ connman_device_add_network(device, pan->network);
- dbus_pending_call_unref(call);
+ if (pan_connect(pan, NULL))
+ DBG("network %p already connected", pan->network);
}
-static void add_network(const char *path)
+static void btdevice_property_change(GDBusProxy *proxy, const char *name,
+ DBusMessageIter *iter, void *user_data)
{
- DBusMessage *message;
- DBusPendingCall *call;
+ struct bluetooth_pan *pan;
+ const char *old_role = NULL;
+ const char *new_role;
- DBG("path %s", path);
+ if (strcmp(name, "UUIDs"))
+ return;
- message = dbus_message_new_method_call(BLUEZ_SERVICE, path,
- BLUEZ_DEVICE_INTERFACE, GET_PROPERTIES);
- if (message == NULL)
+ pan = g_hash_table_lookup(networks, g_dbus_proxy_get_path(proxy));
+ if (!pan)
return;
- dbus_message_set_auto_start(message, FALSE);
+ if (pan->network &&
+ connman_network_get_device(pan->network))
+ old_role = pan->pan_role;
+ new_role = proxy_get_role(pan->btdevice_proxy);
- if (dbus_connection_send_with_reply(connection, message,
- &call, TIMEOUT) == FALSE) {
- connman_error("Failed to get network properties for %s", path);
- goto done;
- }
-
- if (call == NULL) {
- connman_error("D-Bus connection not available");
- goto done;
- }
+ DBG("network %p network role %s proxy role %s", pan->network, old_role,
+ new_role);
- dbus_pending_call_set_notify(call, network_properties_reply,
- g_strdup(path), g_free);
+ if (old_role && new_role && !strcmp(old_role, new_role))
+ return;
-done:
- dbus_message_unref(message);
+ pan_create_nap(pan);
}
-static void check_networks(DBusMessageIter *array)
+static void pan_free(gpointer data)
{
- DBusMessageIter value;
-
- if (dbus_message_iter_get_arg_type(array) != DBUS_TYPE_ARRAY)
- return;
+ struct bluetooth_pan *pan = data;
- dbus_message_iter_recurse(array, &value);
-
- while (dbus_message_iter_get_arg_type(&value) == DBUS_TYPE_OBJECT_PATH) {
- const char *path;
+ if (pan->btnetwork_proxy) {
+ g_dbus_proxy_unref(pan->btnetwork_proxy);
+ pan->btnetwork_proxy = NULL;
+ }
- dbus_message_iter_get_basic(&value, &path);
+ if (pan->btdevice_proxy) {
+ g_dbus_proxy_unref(pan->btdevice_proxy);
+ pan->btdevice_proxy = NULL;
+ }
- add_network(path);
+ pan_remove_nap(pan);
- dbus_message_iter_next(&value);
- }
+ g_free(pan);
}
-static gboolean adapter_changed(DBusConnection *conn,
- DBusMessage *message, void *user_data)
+static void pan_create(GDBusProxy *network_proxy)
{
- const char *path = dbus_message_get_path(message);
- struct connman_device *device;
- DBusMessageIter iter, value;
- const char *key;
-
- DBG("path %s", path);
-
- device = g_hash_table_lookup(bluetooth_devices, path);
- if (device == NULL)
- return TRUE;
+ const char *path = g_dbus_proxy_get_path(network_proxy);
+ struct bluetooth_pan *pan;
- if (dbus_message_iter_init(message, &iter) == FALSE)
- return TRUE;
+ pan = g_try_new0(struct bluetooth_pan, 1);
- dbus_message_iter_get_basic(&iter, &key);
-
- dbus_message_iter_next(&iter);
- dbus_message_iter_recurse(&iter, &value);
-
- if (g_str_equal(key, "Powered") == TRUE) {
- dbus_bool_t val;
-
- dbus_message_iter_get_basic(&value, &val);
- connman_device_set_powered(device, val);
- } else if (g_str_equal(key, "Discovering") == TRUE) {
- dbus_bool_t val;
-
- dbus_message_iter_get_basic(&value, &val);
- connman_device_set_scanning(device, val);
- } else if (g_str_equal(key, "Devices") == TRUE) {
- check_networks(&value);
+ if (!pan) {
+ connman_error("Out of memory creating PAN NAP");
+ return;
}
- return TRUE;
-}
-
-static gboolean device_removed(DBusConnection *conn,
- DBusMessage *message, void *user_data)
-{
- const char *network_path;
- struct connman_network *network;
- struct connman_device *device;
- DBusMessageIter iter;
-
- DBG("");
+ g_hash_table_replace(networks, g_strdup(path), pan);
- if (dbus_message_iter_init(message, &iter) == FALSE)
- return TRUE;
+ pan->btnetwork_proxy = g_dbus_proxy_ref(network_proxy);
+ pan->btdevice_proxy = g_dbus_proxy_new(client, path,
+ "org.bluez.Device1");
- dbus_message_iter_get_basic(&iter, &network_path);
+ if (!pan->btdevice_proxy) {
+ connman_error("Cannot create BT PAN watcher %s", path);
+ g_hash_table_remove(networks, path);
+ return;
+ }
- network = g_hash_table_lookup(bluetooth_networks, network_path);
- if (network == NULL)
- return TRUE;
+ g_dbus_proxy_set_property_watch(pan->btnetwork_proxy,
+ btnetwork_property_change, NULL);
- device = connman_network_get_device(network);
- if (device == NULL)
- return TRUE;
+ g_dbus_proxy_set_property_watch(pan->btdevice_proxy,
+ btdevice_property_change, NULL);
- g_hash_table_remove(bluetooth_networks, network_path);
+ DBG("pan %p %s role %s", pan, path, proxy_get_role(pan->btdevice_proxy));
- return TRUE;
+ pan_create_nap(pan);
}
-static gboolean device_changed(DBusConnection *conn,
- DBusMessage *message, void *user_data)
-{
- const char *path = dbus_message_get_path(message);
- DBusMessageIter iter, value;
- const char *key;
-
- DBG("path %s", path);
-
- if (dbus_message_iter_init(message, &iter) == FALSE)
- return TRUE;
-
- dbus_message_iter_get_basic(&iter, &key);
-
- dbus_message_iter_next(&iter);
- dbus_message_iter_recurse(&iter, &value);
-
- DBG("key %s", key);
-
- if (g_str_equal(key, "UUIDs") == TRUE)
- add_network(path);
-
- return TRUE;
-}
+static struct connman_network_driver network_driver = {
+ .name = "bluetooth",
+ .type = CONNMAN_NETWORK_TYPE_BLUETOOTH_PAN,
+ .probe = bluetooth_pan_probe,
+ .remove = bluetooth_pan_remove,
+ .connect = bluetooth_pan_connect,
+ .disconnect = bluetooth_pan_disconnect,
+};
-static void remove_device_networks(struct connman_device *device)
+static void enable_device(struct connman_device *device, const char *path)
{
GHashTableIter iter;
gpointer key, value;
- GSList *key_list = NULL;
- GSList *list;
-
- if (bluetooth_networks == NULL)
- return;
-
- g_hash_table_iter_init(&iter, bluetooth_networks);
-
- while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
- struct connman_network *network = value;
- if (connman_network_get_device(network) != device)
- continue;
+ DBG("device %p %s", device, path);
+ connman_device_set_powered(device, true);
- key_list = g_slist_append(key_list, key);
- }
+ g_hash_table_iter_init(&iter, networks);
+ while (g_hash_table_iter_next(&iter, &key, &value)) {
+ struct bluetooth_pan *pan = value;
- for (list = key_list; list != NULL; list = list->next) {
- const char *network_path = list->data;
+ if (g_strcmp0(proxy_get_string(pan->btdevice_proxy, "Adapter"),
+ path) == 0) {
- g_hash_table_remove(bluetooth_networks, network_path);
+ DBG("enable network %p", pan->network);
+ pan_create_nap(pan);
+ }
}
-
- g_slist_free(key_list);
}
-static void adapter_properties_reply(DBusPendingCall *call, void *user_data)
+static void device_enable_cb(const DBusError *error, void *user_data)
{
char *path = user_data;
struct connman_device *device;
- DBusMessage *reply;
- DBusMessageIter networks;
- const char *address = NULL, *name = NULL;
- dbus_bool_t powered = FALSE, scanning = FALSE;
- struct ether_addr addr;
- char ident[13];
- DBG("path %s", path);
+ device = g_hash_table_lookup(devices, path);
+ if (!device) {
+ DBG("device already removed");
+ goto out;
+ }
- reply = dbus_pending_call_steal_reply(call);
+ if (dbus_error_is_set(error)) {
+ connman_warn("Bluetooth device %s not enabled %s",
+ path, error->message);
+ goto out;
+ }
- if (path == NULL)
- goto done;
+#if !defined TIZEN_EXT
+ enable_device(device, path);
+#endif
+out:
+ g_free(path);
+}
- extract_properties(reply, NULL, &address, &name, NULL,
- &powered, &scanning, NULL, &networks);
+static int bluetooth_device_enable(struct connman_device *device)
+{
+ GDBusProxy *proxy = connman_device_get_data(device);
+ dbus_bool_t device_powered = TRUE;
+ const char *path;
- if (address == NULL)
- goto done;
+ if (!proxy)
+ return 0;
- if (g_strcmp0(address, "00:00:00:00:00:00") == 0)
- goto done;
+ path = g_dbus_proxy_get_path(proxy);
- device = g_hash_table_lookup(bluetooth_devices, path);
- if (device != NULL)
- goto update;
+ if (proxy_get_bool(proxy, "Powered")) {
+ DBG("already enabled %p %s", device, path);
+ return -EALREADY;
+ }
- g_hash_table_insert(bluetooth_devices, g_strdup(path), device);
+ DBG("device %p %s", device, path);
- ether_aton_r(address, &addr);
+ g_dbus_proxy_set_property_basic(proxy, "Powered",
+ DBUS_TYPE_BOOLEAN, &device_powered,
+ device_enable_cb, g_strdup(path), NULL);
- snprintf(ident, 13, "%02x%02x%02x%02x%02x%02x",
- addr.ether_addr_octet[0],
- addr.ether_addr_octet[1],
- addr.ether_addr_octet[2],
- addr.ether_addr_octet[3],
- addr.ether_addr_octet[4],
- addr.ether_addr_octet[5]);
+ return -EINPROGRESS;
+}
- device = connman_device_create(ident, CONNMAN_DEVICE_TYPE_BLUETOOTH);
- if (device == NULL)
- goto done;
+static void disable_device(struct connman_device *device, const char *path)
+{
+ GHashTableIter iter;
+ gpointer key, value;
- connman_device_set_ident(device, ident);
+ DBG("device %p %s", device, path);
+ connman_device_set_powered(device, false);
- connman_device_set_string(device, "Path", path);
+ g_hash_table_iter_init(&iter, networks);
+ while (g_hash_table_iter_next(&iter, &key, &value)) {
+ struct bluetooth_pan *pan = value;
- if (connman_device_register(device) < 0) {
- connman_device_unref(device);
- g_hash_table_remove(bluetooth_devices, path);
- goto done;
+ if (pan->network && connman_network_get_device(pan->network)
+ == device) {
+ DBG("disable network %p", pan->network);
+ connman_device_remove_network(device, pan->network);
+ }
}
-
-update:
- connman_device_set_string(device, "Address", address);
- connman_device_set_string(device, "Name", name);
- connman_device_set_string(device, "Path", path);
-
- connman_device_set_powered(device, powered);
- connman_device_set_scanning(device, scanning);
-
- if (powered == TRUE)
- check_networks(&networks);
- else
- remove_device_networks(device);
-
-done:
- dbus_message_unref(reply);
-
- dbus_pending_call_unref(call);
}
-static void add_adapter(DBusConnection *conn, const char *path)
+static void device_disable_cb(const DBusError *error, void *user_data)
{
- DBusMessage *message;
- DBusPendingCall *call;
-
- DBG("path %s", path);
-
- message = dbus_message_new_method_call(BLUEZ_SERVICE, path,
- BLUEZ_ADAPTER_INTERFACE, GET_PROPERTIES);
- if (message == NULL)
- return;
-
- dbus_message_set_auto_start(message, FALSE);
+ char *path = user_data;
+ struct connman_device *device;
- if (dbus_connection_send_with_reply(conn, message,
- &call, TIMEOUT) == FALSE) {
- connman_error("Failed to get adapter properties for %s", path);
- goto done;
+ device = g_hash_table_lookup(devices, path);
+ if (!device) {
+ DBG("device already removed");
+ goto out;
}
- if (call == NULL) {
- connman_error("D-Bus connection not available");
- goto done;
+ if (dbus_error_is_set(error)) {
+ connman_warn("Bluetooth device %s not disabled: %s",
+ path, error->message);
+ goto out;
}
- dbus_pending_call_set_notify(call, adapter_properties_reply,
- g_strdup(path), g_free);
+#if !defined TIZEN_EXT
+ disable_device(device, path);
+#endif
-done:
- dbus_message_unref(message);
+out:
+ g_free(path);
}
-static gboolean adapter_added(DBusConnection *conn, DBusMessage *message,
- void *user_data)
+static int bluetooth_device_disable(struct connman_device *device)
{
+ GDBusProxy *proxy = connman_device_get_data(device);
+ dbus_bool_t device_powered = FALSE;
const char *path;
- dbus_message_get_args(message, NULL, DBUS_TYPE_OBJECT_PATH, &path,
- DBUS_TYPE_INVALID);
- add_adapter(conn, path);
- return TRUE;
-}
+ if (!proxy)
+ return 0;
-static void remove_adapter(DBusConnection *conn, const char *path)
-{
- DBG("path %s", path);
+ path = g_dbus_proxy_get_path(proxy);
- g_hash_table_remove(bluetooth_devices, path);
-}
+ if (!proxy_get_bool(proxy, "Powered")) {
+ DBG("already disabled %p %s", device, path);
+ return -EALREADY;
+ }
-static gboolean adapter_removed(DBusConnection *conn, DBusMessage *message,
- void *user_data)
-{
- const char *path;
+ DBG("device %p %s", device, path);
+
+ g_dbus_proxy_set_property_basic(proxy, "Powered",
+ DBUS_TYPE_BOOLEAN, &device_powered,
+ device_disable_cb, g_strdup(path), NULL);
- dbus_message_get_args(message, NULL, DBUS_TYPE_OBJECT_PATH, &path,
- DBUS_TYPE_INVALID);
- remove_adapter(conn, path);
- return TRUE;
+ return -EINPROGRESS;
}
-static void list_adapters_reply(DBusPendingCall *call, void *user_data)
+static void adapter_property_change(GDBusProxy *proxy, const char *name,
+ DBusMessageIter *iter, void *user_data)
{
- DBusMessage *reply;
- DBusError error;
- char **adapters;
- int i, num_adapters;
+ struct connman_device *device;
+ const char *path;
+ bool adapter_powered, device_powered;
- DBG("");
+ if (strcmp(name, "Powered") != 0)
+ return;
- reply = dbus_pending_call_steal_reply(call);
+ path = g_dbus_proxy_get_path(proxy);
+ device = g_hash_table_lookup(devices, path);
- dbus_error_init(&error);
+ adapter_powered = proxy_get_bool(proxy, "Powered");
+ device_powered = connman_device_get_powered(device);
- if (dbus_set_error_from_message(&error, reply) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- goto done;
- }
+ DBG("device %p %s device powered %d adapter powered %d", device, path,
+ device_powered, adapter_powered);
- if (dbus_message_get_args(reply, &error,
- DBUS_TYPE_ARRAY, DBUS_TYPE_OBJECT_PATH,
- &adapters, &num_adapters,
- DBUS_TYPE_INVALID) == FALSE) {
- if (dbus_error_is_set(&error) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- } else
- connman_error("Wrong arguments for adapter list");
- goto done;
+ if (device_powered != adapter_powered) {
+ if (adapter_powered)
+ enable_device(device, path);
+ else
+ disable_device(device, path);
}
-
- for (i = 0; i < num_adapters; i++)
- add_adapter(connection, adapters[i]);
-
- g_strfreev(adapters);
-
-done:
- dbus_message_unref(reply);
-
- dbus_pending_call_unref(call);
}
-static void unregister_device(gpointer data)
+static void device_free(gpointer data)
{
struct connman_device *device = data;
+ GDBusProxy *proxy = connman_device_get_data(device);
- DBG("");
-
- remove_device_networks(device);
+ connman_device_set_data(device, NULL);
+ if (proxy)
+ g_dbus_proxy_unref(proxy);
connman_device_unregister(device);
connman_device_unref(device);
}
-static void remove_network(gpointer data)
-{
- struct connman_network *network = data;
- struct connman_device *device;
-
- DBG("network %p", network);
+struct tethering_info {
+ struct connman_technology *technology;
+ char *bridge;
+ bool enable;
+};
- device = connman_network_get_device(network);
- if (device != NULL)
- connman_device_remove_network(device, network);
+static void tethering_free(void *user_data)
+{
+ struct tethering_info *tethering = user_data;
- connman_network_unref(network);
+ g_free(tethering->bridge);
+ g_free(tethering);
}
-static void bluetooth_connect(DBusConnection *conn, void *user_data)
+static void tethering_create_cb(DBusMessage *message, void *user_data)
{
- DBusMessage *message;
- DBusPendingCall *call;
-
- DBG("connection %p", conn);
+ struct tethering_info *tethering = user_data;
- bluetooth_devices = g_hash_table_new_full(g_str_hash, g_str_equal,
- g_free, unregister_device);
+ if (dbus_message_get_type(message) == DBUS_MESSAGE_TYPE_ERROR) {
+ const char *dbus_error = dbus_message_get_error_name(message);
- bluetooth_networks = g_hash_table_new_full(g_str_hash, g_str_equal,
- g_free, remove_network);
-
- message = dbus_message_new_method_call(BLUEZ_SERVICE, "/",
- BLUEZ_MANAGER_INTERFACE, LIST_ADAPTERS);
- if (message == NULL)
+ DBG("%s tethering failed: %s",
+ tethering->enable ? "enable" : "disable",
+ dbus_error);
return;
-
- dbus_message_set_auto_start(message, FALSE);
-
- if (dbus_connection_send_with_reply(conn, message,
- &call, TIMEOUT) == FALSE) {
- connman_error("Failed to get Bluetooth adapters");
- goto done;
- }
-
- if (call == NULL) {
- connman_error("D-Bus connection not available");
- goto done;
}
- dbus_pending_call_set_notify(call, list_adapters_reply, NULL, NULL);
+ DBG("bridge %s %s", tethering->bridge, tethering->enable ?
+ "enabled": "disabled");
-done:
- dbus_message_unref(message);
+ if (tethering->technology)
+ connman_technology_tethering_notify(tethering->technology,
+ tethering->enable);
}
-static void bluetooth_disconnect(DBusConnection *conn, void *user_data)
+static void tethering_append(DBusMessageIter *iter, void *user_data)
{
- DBG("connection %p", conn);
-
- if (bluetooth_devices == NULL)
- return;
+ struct tethering_info *tethering = user_data;
+ const char *nap = "nap";
- g_hash_table_destroy(bluetooth_networks);
- bluetooth_networks = NULL;
- g_hash_table_destroy(bluetooth_devices);
- bluetooth_devices = NULL;
+ dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &nap);
+ if (tethering->enable)
+ dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING,
+ &tethering->bridge);
}
-static int bluetooth_probe(struct connman_device *device)
+static bool tethering_create(const char *path,
+ struct connman_technology *technology, const char *bridge,
+ bool enabled)
{
- GHashTableIter iter;
- gpointer key, value;
-
- DBG("device %p", device);
+ struct tethering_info *tethering = g_new0(struct tethering_info, 1);
+ GDBusProxy *proxy;
+ const char *method;
+ bool result;
- if (bluetooth_devices == NULL)
- return -ENOTSUP;
+ DBG("path %s bridge %s", path, bridge);
- g_hash_table_iter_init(&iter, bluetooth_devices);
-
- while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
- struct connman_device *device_pan = value;
-
- if (device == device_pan)
- return 0;
+ if (!bridge) {
+ g_free(tethering);
+ return false;
}
- return -ENOTSUP;
-}
-
-static void bluetooth_remove(struct connman_device *device)
-{
- DBG("device %p", device);
-}
-
-static void powered_reply(DBusPendingCall *call, void *user_data)
-{
- DBusError error;
- DBusMessage *reply;
-
- DBG("");
+ proxy = g_dbus_proxy_new(client, path, "org.bluez.NetworkServer1");
+ if (!proxy) {
+ g_free(tethering);
+ return false;
+ }
- reply = dbus_pending_call_steal_reply(call);
+ tethering->technology = technology;
+ tethering->bridge = g_strdup(bridge);
+ tethering->enable = enabled;
- dbus_error_init(&error);
+ if (tethering->enable)
+ method = "Register";
+ else
+ method = "Unregister";
- if (dbus_set_error_from_message(&error, reply) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- dbus_message_unref(reply);
- dbus_pending_call_unref(call);
- return;
- }
+ result = g_dbus_proxy_method_call(proxy, method, tethering_append,
+ tethering_create_cb, tethering, tethering_free);
- dbus_message_unref(reply);
- dbus_pending_call_unref(call);
+ g_dbus_proxy_unref(proxy);
- add_adapter(connection, user_data);
+ return result;
}
-static int change_powered(DBusConnection *conn, const char *path,
- dbus_bool_t powered)
+static void device_create(GDBusProxy *proxy)
{
- DBusMessage *message;
- DBusMessageIter iter;
- DBusPendingCall *call;
+ struct connman_device *device = NULL;
+ const char *path = g_dbus_proxy_get_path(proxy);
+ const char *address;
+ char ident[BLUETOOTH_ADDR_LEN * 2 + 1];
+ bool powered;
- DBG("");
+ address = proxy_get_string(proxy, "Address");
+ if (!address)
+ return;
- if (path == NULL)
- return -EINVAL;
+ address2ident(address, ident);
- message = dbus_message_new_method_call(BLUEZ_SERVICE, path,
- BLUEZ_ADAPTER_INTERFACE, SET_PROPERTY);
- if (message == NULL)
- return -ENOMEM;
+ device = connman_device_create("bluetooth",
+ CONNMAN_DEVICE_TYPE_BLUETOOTH);
+ if (!device)
+ return;
- dbus_message_set_auto_start(message, FALSE);
+ connman_device_set_data(device, g_dbus_proxy_ref(proxy));
+ connman_device_set_ident(device, ident);
- dbus_message_iter_init_append(message, &iter);
- connman_dbus_property_append_basic(&iter, "Powered",
- DBUS_TYPE_BOOLEAN, &powered);
+ g_hash_table_replace(devices, g_strdup(path), device);
- if (dbus_connection_send_with_reply(conn, message,
- &call, TIMEOUT) == FALSE) {
- connman_error("Failed to change Powered property");
- dbus_message_unref(message);
- return -EINVAL;
- }
+ DBG("device %p %s device powered %d adapter powered %d", device,
+ path, connman_device_get_powered(device),
+ proxy_get_bool(proxy, "Powered"));
- if (call == NULL) {
- connman_error("D-Bus connection not available");
- dbus_message_unref(message);
- return -EINVAL;
+ if (connman_device_register(device) < 0) {
+ g_hash_table_remove(devices, device);
+ return;
}
- dbus_pending_call_set_notify(call, powered_reply,
- g_strdup(path), g_free);
+ g_dbus_proxy_set_property_watch(proxy, adapter_property_change, NULL);
- dbus_message_unref(message);
-
- return -EINPROGRESS;
-}
-
-static int bluetooth_enable(struct connman_device *device)
-{
- const char *path = connman_device_get_string(device, "Path");
-
- DBG("device %p", device);
+ powered = proxy_get_bool(proxy, "Powered");
+ connman_device_set_powered(device, powered);
- return change_powered(connection, path, TRUE);
+ if (proxy_get_role(proxy) && !bluetooth_tethering)
+ tethering_create(path, NULL, NULL, false);
}
-static int bluetooth_disable(struct connman_device *device)
+static void object_added(GDBusProxy *proxy, void *user_data)
{
- const char *path = connman_device_get_string(device, "Path");
-
- DBG("device %p", device);
-
- return change_powered(connection, path, FALSE);
-}
+ const char *interface;
-static struct connman_device_driver bluetooth_driver = {
- .name = "bluetooth",
- .type = CONNMAN_DEVICE_TYPE_BLUETOOTH,
- .probe = bluetooth_probe,
- .remove = bluetooth_remove,
- .enable = bluetooth_enable,
- .disable = bluetooth_disable,
-};
+ interface = g_dbus_proxy_get_interface(proxy);
+ if (!interface) {
+ connman_warn("Interface or proxy missing when adding "
+ "bluetooth object");
+ return;
+ }
-static int tech_probe(struct connman_technology *technology)
-{
- return 0;
-}
+ if (strcmp(interface, "org.bluez.Adapter1") == 0) {
+ DBG("%s %s", interface, g_dbus_proxy_get_path(proxy));
+ device_create(proxy);
+ return;
+ }
-static void tech_remove(struct connman_technology *technology)
-{
+ if (strcmp(interface, "org.bluez.Network1") == 0) {
+ DBG("%s %s", interface, g_dbus_proxy_get_path(proxy));
+ pan_create(proxy);
+ return;
+ }
}
-static void server_register_reply(DBusPendingCall *call, void *user_data)
+static void object_removed(GDBusProxy *proxy, void *user_data)
{
- struct connman_technology *technology = user_data;
- DBusError error;
- DBusMessage *reply;
-
- DBG("");
-
- reply = dbus_pending_call_steal_reply(call);
-
- dbus_error_init(&error);
+ const char *interface, *path;
- if (dbus_set_error_from_message(&error, reply) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- dbus_message_unref(reply);
- dbus_pending_call_unref(call);
+ interface = g_dbus_proxy_get_interface(proxy);
+ if (!interface) {
+ connman_warn("Interface or proxy missing when removing "
+ "bluetooth object");
return;
}
- dbus_message_unref(reply);
- dbus_pending_call_unref(call);
+ if (strcmp(interface, "org.bluez.Adapter1") == 0) {
+ path = g_dbus_proxy_get_path(proxy);
+ DBG("%s %s", interface, path);
- connman_technology_tethering_notify(technology, TRUE);
-}
-
-static void server_unregister_reply(DBusPendingCall *call, void *user_data)
-{
- struct connman_technology *technology = user_data;
- DBusError error;
- DBusMessage *reply;
-
- DBG("");
-
- reply = dbus_pending_call_steal_reply(call);
+ g_hash_table_remove(devices, path);
+ }
- dbus_error_init(&error);
+ if (strcmp(interface, "org.bluez.Network1") == 0) {
+ path = g_dbus_proxy_get_path(proxy);
+ DBG("%s %s", interface, path);
- if (dbus_set_error_from_message(&error, reply) == TRUE) {
- connman_error("%s", error.message);
- dbus_error_free(&error);
- dbus_message_unref(reply);
- dbus_pending_call_unref(call);
- return;
+ g_hash_table_remove(networks, path);
}
- dbus_message_unref(reply);
- dbus_pending_call_unref(call);
-
- connman_technology_tethering_notify(technology, FALSE);
}
-
-static void server_register(const char *path, const char *uuid,
- struct connman_technology *technology,
- const char *bridge, connman_bool_t enabled)
+static int bluetooth_device_probe(struct connman_device *device)
{
- DBusMessage *message;
- DBusPendingCall *call;
- char *command;
-
- DBG("path %s enabled %d", path, enabled);
-
- command = enabled ? REGISTER : UNREGISTER;
-
- message = dbus_message_new_method_call(BLUEZ_SERVICE, path,
- BLUEZ_NETWORK_SERVER, command);
- if (message == NULL)
- return;
-
- dbus_message_set_auto_start(message, FALSE);
-
- dbus_message_append_args(message, DBUS_TYPE_STRING, &uuid,
- DBUS_TYPE_INVALID);
+ GHashTableIter iter;
+ gpointer key, value;
- if (enabled == TRUE)
- dbus_message_append_args(message, DBUS_TYPE_STRING, &bridge,
- DBUS_TYPE_INVALID);
+ g_hash_table_iter_init(&iter, devices);
- if (dbus_connection_send_with_reply(connection, message,
- &call, TIMEOUT) == FALSE) {
- connman_error("Failed to enable PAN server");
- dbus_message_unref(message);
- return;
- }
+ while (g_hash_table_iter_next(&iter, &key, &value)) {
+ struct connman_device *known = value;
- if (call == NULL) {
- connman_error("D-Bus connection not available");
- dbus_message_unref(message);
- return;
+ if (device == known)
+ return 0;
}
- if (enabled == TRUE)
- dbus_pending_call_set_notify(call, server_register_reply,
- technology, NULL);
- else
- dbus_pending_call_set_notify(call, server_unregister_reply,
- technology, NULL);
+ return -EOPNOTSUPP;
+}
- dbus_message_unref(message);
+static void bluetooth_device_remove(struct connman_device *device)
+{
+ DBG("%p", device);
}
-struct tethering_info {
- struct connman_technology *technology;
- const char *bridge;
+static struct connman_device_driver device_driver = {
+ .name = "bluetooth",
+ .type = CONNMAN_DEVICE_TYPE_BLUETOOTH,
+ .probe = bluetooth_device_probe,
+ .remove = bluetooth_device_remove,
+ .enable = bluetooth_device_enable,
+ .disable = bluetooth_device_disable,
};
-static void enable_nap(gpointer key, gpointer value, gpointer user_data)
+static int bluetooth_tech_probe(struct connman_technology *technology)
{
- struct tethering_info *info = user_data;
- struct connman_device *device = value;
- const char *path;
-
- DBG("");
+ return 0;
+}
- path = connman_device_get_string(device, "Path");
+static void bluetooth_tech_remove(struct connman_technology *technology)
+{
- server_register(path, "nap", info->technology, info->bridge, TRUE);
}
-static void disable_nap(gpointer key, gpointer value, gpointer user_data)
+static int bluetooth_tech_set_tethering(struct connman_technology *technology,
+ const char *identifier, const char *passphrase,
+ const char *bridge, bool enabled)
{
- struct tethering_info *info = user_data;
- struct connman_device *device = value;
- const char *path;
+ GHashTableIter hash_iter;
+ gpointer key, value;
+ int i = 0;
- DBG("");
+ bluetooth_tethering = enabled;
- path = connman_device_get_string(device, "Path");
+ g_hash_table_iter_init(&hash_iter, devices);
- server_register(path, "nap", info->technology, info->bridge, FALSE);
-}
+ while (g_hash_table_iter_next(&hash_iter, &key, &value)) {
+ const char *path = key;
+ struct connman_device *device = value;
-static int tech_set_tethering(struct connman_technology *technology,
- const char *identifier, const char *passphrase,
- const char *bridge, connman_bool_t enabled)
-{
- struct tethering_info info = {
- .technology = technology,
- .bridge = bridge,
- };
+ DBG("device %p", device);
- DBG("bridge %s", bridge);
+ if (tethering_create(path, technology, bridge, enabled)
+ )
+ i++;
+ }
- if (enabled)
- g_hash_table_foreach(bluetooth_devices, enable_nap, &info);
- else
- g_hash_table_foreach(bluetooth_devices, disable_nap, &info);
+ DBG("%s %d device(s)", enabled ? "enabled" : "disabled", i);
+
+ if (i == 0)
+ return -ENODEV;
return 0;
}
static struct connman_technology_driver tech_driver = {
.name = "bluetooth",
.type = CONNMAN_SERVICE_TYPE_BLUETOOTH,
- .probe = tech_probe,
- .remove = tech_remove,
- .set_tethering = tech_set_tethering,
+ .probe = bluetooth_tech_probe,
+ .remove = bluetooth_tech_remove,
+ .set_tethering = bluetooth_tech_set_tethering,
};
-static guint watch;
-static guint added_watch;
-static guint removed_watch;
-static guint adapter_watch;
-static guint device_watch;
-static guint device_removed_watch;
-static guint network_watch;
-
static int bluetooth_init(void)
{
- int err;
-
connection = connman_dbus_get_connection();
- if (connection == NULL)
- return -EIO;
+ if (!connection)
+ goto out;
- watch = g_dbus_add_service_watch(connection, BLUEZ_SERVICE,
- bluetooth_connect, bluetooth_disconnect, NULL, NULL);
-
- added_watch = g_dbus_add_signal_watch(connection, BLUEZ_SERVICE, NULL,
- BLUEZ_MANAGER_INTERFACE,
- ADAPTER_ADDED, adapter_added,
- NULL, NULL);
-
- removed_watch = g_dbus_add_signal_watch(connection, BLUEZ_SERVICE, NULL,
- BLUEZ_MANAGER_INTERFACE,
- ADAPTER_REMOVED, adapter_removed,
- NULL, NULL);
-
- adapter_watch = g_dbus_add_signal_watch(connection, BLUEZ_SERVICE,
- NULL, BLUEZ_ADAPTER_INTERFACE,
- PROPERTY_CHANGED, adapter_changed,
- NULL, NULL);
-
- device_removed_watch = g_dbus_add_signal_watch(connection,
- BLUEZ_SERVICE, NULL,
- BLUEZ_ADAPTER_INTERFACE,
- DEVICE_REMOVED, device_removed,
- NULL, NULL);
-
- device_watch = g_dbus_add_signal_watch(connection, BLUEZ_SERVICE, NULL,
- BLUEZ_DEVICE_INTERFACE,
- PROPERTY_CHANGED, device_changed,
- NULL, NULL);
-
- network_watch = g_dbus_add_signal_watch(connection, BLUEZ_SERVICE,
- NULL, BLUEZ_NETWORK_INTERFACE,
- PROPERTY_CHANGED, network_changed,
- NULL, NULL);
-
- if (watch == 0 || added_watch == 0 || removed_watch == 0
- || adapter_watch == 0 || network_watch == 0
- || device_watch == 0
- || device_removed_watch == 0) {
- err = -EIO;
- goto remove;
+ if (connman_technology_driver_register(&tech_driver) < 0) {
+ connman_warn("Failed to initialize technology for Bluez 5");
+ goto out;
}
- err = connman_network_driver_register(&pan_driver);
- if (err < 0)
- goto remove;
+ devices = g_hash_table_new_full(g_str_hash, g_str_equal, g_free,
+ device_free);
- err = connman_device_driver_register(&bluetooth_driver);
- if (err < 0) {
- connman_network_driver_unregister(&pan_driver);
- goto remove;
+ if (connman_device_driver_register(&device_driver) < 0) {
+ connman_warn("Failed to initialize device driver for "
+ BLUEZ_SERVICE);
+ connman_technology_driver_unregister(&tech_driver);
+ goto out;
}
- err = connman_technology_driver_register(&tech_driver);
- if (err < 0) {
- connman_device_driver_unregister(&bluetooth_driver);
- connman_network_driver_unregister(&pan_driver);
- goto remove;
+ if (connman_network_driver_register(&network_driver) < 0) {
+ connman_technology_driver_unregister(&tech_driver);
+ connman_device_driver_unregister(&device_driver);
+ goto out;
}
+ networks = g_hash_table_new_full(g_str_hash, g_str_equal, g_free,
+ pan_free);
+
+ client = g_dbus_client_new(connection, BLUEZ_SERVICE, BLUEZ_PATH);
+ if (!client) {
+ connman_warn("Failed to initialize D-Bus client for "
+ BLUEZ_SERVICE);
+ goto out;
+ }
+
+ g_dbus_client_set_proxy_handlers(client, object_added, object_removed,
+ NULL, NULL);
+
return 0;
-remove:
- g_dbus_remove_watch(connection, watch);
- g_dbus_remove_watch(connection, added_watch);
- g_dbus_remove_watch(connection, removed_watch);
- g_dbus_remove_watch(connection, adapter_watch);
- g_dbus_remove_watch(connection, device_removed_watch);
- g_dbus_remove_watch(connection, device_watch);
- g_dbus_remove_watch(connection, network_watch);
+out:
+ if (networks)
+ g_hash_table_destroy(networks);
- dbus_connection_unref(connection);
+ if (devices)
+ g_hash_table_destroy(devices);
+
+ if (client)
+ g_dbus_client_unref(client);
+
+ if (connection)
+ dbus_connection_unref(connection);
- return err;
+ return -EIO;
}
static void bluetooth_exit(void)
{
- g_dbus_remove_watch(connection, watch);
- g_dbus_remove_watch(connection, added_watch);
- g_dbus_remove_watch(connection, removed_watch);
- g_dbus_remove_watch(connection, adapter_watch);
- g_dbus_remove_watch(connection, device_removed_watch);
- g_dbus_remove_watch(connection, device_watch);
- g_dbus_remove_watch(connection, network_watch);
+ /*
+ * We unset the disabling of the Bluetooth device when shutting down
+ * so that non-PAN BT connections are not affected.
+ */
+ device_driver.disable = NULL;
- bluetooth_disconnect(connection, NULL);
+ g_dbus_client_unref(client);
- connman_technology_driver_unregister(&tech_driver);
+ connman_network_driver_unregister(&network_driver);
+ g_hash_table_destroy(networks);
- connman_device_driver_unregister(&bluetooth_driver);
- connman_network_driver_unregister(&pan_driver);
+ connman_device_driver_unregister(&device_driver);
+ g_hash_table_destroy(devices);
+ connman_technology_driver_unregister(&tech_driver);
dbus_connection_unref(connection);
}
CONNMAN_PLUGIN_DEFINE(bluetooth, "Bluetooth technology plugin", VERSION,
- CONNMAN_PLUGIN_PRIORITY_DEFAULT, bluetooth_init, bluetooth_exit)
+ CONNMAN_PLUGIN_PRIORITY_DEFAULT, bluetooth_init, bluetooth_exit)