| /* |
| * |
| * Connection Manager |
| * |
| * Copyright (C) 2008 Intel Corporation. All rights reserved. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License version 2.1 as published by the Free Software Foundation. |
| * |
| * 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, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
| * |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include <config.h> |
| #endif |
| |
| #include "connman-dbus.h" |
| #include "connman-dbus-glue.h" |
| #include "connman-client.h" |
| |
| #include "marshal.h" |
| #include "marshal.c" |
| |
| #ifdef DEBUG |
| #define DBG(fmt, arg...) printf("%s:%s() " fmt "\n", __FILE__, __FUNCTION__ , ## arg) |
| #else |
| #define DBG(fmt...) |
| #endif |
| |
| #define CONNMAN_CLIENT_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE((obj), \ |
| CONNMAN_TYPE_CLIENT, ConnmanClientPrivate)) |
| |
| #ifndef DBUS_TYPE_G_DICTIONARY |
| #define DBUS_TYPE_G_DICTIONARY \ |
| (dbus_g_type_get_map("GHashTable", G_TYPE_STRING, G_TYPE_VALUE)) |
| #endif |
| |
| typedef struct _ConnmanClientPrivate ConnmanClientPrivate; |
| |
| struct _ConnmanClientPrivate { |
| GtkTreeStore *store; |
| DBusGProxy *dbus; |
| DBusGProxy *manager; |
| GHashTable *services; |
| ConnmanClientCallback callback; |
| gpointer userdata; |
| }; |
| |
| G_DEFINE_TYPE(ConnmanClient, connman_client, G_TYPE_OBJECT) |
| |
| static void name_owner_changed(DBusGProxy *dbus, const char *name, |
| const char *prev, const char *new, gpointer user_data) |
| { |
| ConnmanClient *client = user_data; |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| GtkTreeIter iter; |
| gboolean cont; |
| char *state, *oldstate; |
| |
| if (g_str_equal(name, CONNMAN_SERVICE) == FALSE) |
| return; |
| |
| if (*new != '\0') { |
| state = "offline"; |
| goto done; |
| } |
| |
| DBG("client %p name %s", client, name); |
| |
| cont = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(priv->store), |
| &iter); |
| |
| while (cont == TRUE) |
| cont = gtk_tree_store_remove(priv->store, &iter); |
| |
| state = "unavailable"; |
| |
| done: |
| oldstate = g_object_get_data(G_OBJECT(priv->store), "State"); |
| g_free(oldstate); |
| |
| g_object_set_data(G_OBJECT(priv->store), "State", g_strdup(state)); |
| |
| if (priv->callback != NULL) |
| priv->callback(state, priv->userdata); |
| } |
| |
| static DBusGConnection *connection = NULL; |
| |
| static void connman_client_init(ConnmanClient *client) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| |
| DBG("client %p", client); |
| |
| priv->store = gtk_tree_store_new(_CONNMAN_NUM_COLUMNS, |
| G_TYPE_OBJECT, /* proxy */ |
| G_TYPE_UINT, /* index */ |
| G_TYPE_STRING, /* name */ |
| G_TYPE_STRING, /* icon */ |
| G_TYPE_UINT, /* type */ |
| G_TYPE_STRING, /* state */ |
| G_TYPE_BOOLEAN, /* favorite */ |
| G_TYPE_UINT, /* strength */ |
| G_TYPE_STRING, /* security */ |
| G_TYPE_STRING, /* method */ |
| G_TYPE_STRING, /* address */ |
| G_TYPE_STRING, /* netmask */ |
| G_TYPE_STRING, /* gateway */ |
| G_TYPE_BOOLEAN, /* powered */ |
| G_TYPE_BOOLEAN);/* offline */ |
| |
| g_object_set_data(G_OBJECT(priv->store), |
| "State", g_strdup("unavailable")); |
| |
| priv->dbus = dbus_g_proxy_new_for_name(connection, DBUS_SERVICE_DBUS, |
| DBUS_PATH_DBUS, DBUS_INTERFACE_DBUS); |
| |
| dbus_g_proxy_add_signal(priv->dbus, "NameOwnerChanged", |
| G_TYPE_STRING, G_TYPE_STRING, |
| G_TYPE_STRING, G_TYPE_INVALID); |
| dbus_g_proxy_connect_signal(priv->dbus, "NameOwnerChanged", |
| G_CALLBACK(name_owner_changed), client, NULL); |
| |
| priv->manager = connman_dbus_create_manager(connection, priv->store); |
| } |
| |
| static void connman_client_finalize(GObject *client) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| |
| DBG("client %p", client); |
| |
| g_signal_handlers_disconnect_by_func(priv->dbus, |
| name_owner_changed, client); |
| g_object_unref(priv->dbus); |
| |
| connman_dbus_destroy_manager(priv->manager, priv->store); |
| |
| g_object_unref(priv->store); |
| |
| G_OBJECT_CLASS(connman_client_parent_class)->finalize(client); |
| } |
| |
| static void connman_client_class_init(ConnmanClientClass *klass) |
| { |
| GObjectClass *object_class = (GObjectClass *) klass; |
| GError *error = NULL; |
| |
| DBG("class %p", klass); |
| |
| g_type_class_add_private(klass, sizeof(ConnmanClientPrivate)); |
| |
| object_class->finalize = connman_client_finalize; |
| |
| dbus_g_object_register_marshaller(marshal_VOID__STRING_BOXED, |
| G_TYPE_NONE, G_TYPE_STRING, |
| G_TYPE_VALUE, G_TYPE_INVALID); |
| |
| connection = dbus_g_bus_get(DBUS_BUS_SYSTEM, &error); |
| |
| if (error != NULL) { |
| g_printerr("Connecting to system bus failed: %s\n", |
| error->message); |
| g_error_free(error); |
| } |
| } |
| |
| static ConnmanClient *connman_client = NULL; |
| |
| ConnmanClient *connman_client_new(void) |
| { |
| if (connman_client != NULL) |
| return g_object_ref(connman_client); |
| |
| connman_client = CONNMAN_CLIENT(g_object_new(CONNMAN_TYPE_CLIENT, |
| NULL)); |
| |
| DBG("client %p", connman_client); |
| |
| return connman_client; |
| } |
| |
| GtkTreeModel *connman_client_get_model(ConnmanClient *client) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| GtkTreeModel *model; |
| |
| DBG("client %p", client); |
| |
| model = g_object_ref(priv->store); |
| |
| return model; |
| } |
| |
| GtkTreeModel *connman_client_get_connection_model(ConnmanClient *client) |
| { |
| return connman_client_get_model(client); |
| } |
| |
| static gboolean device_filter(GtkTreeModel *model, |
| GtkTreeIter *iter, gpointer user_data) |
| { |
| DBusGProxy *proxy; |
| gboolean active; |
| guint type; |
| |
| gtk_tree_model_get(model, iter, CONNMAN_COLUMN_PROXY, &proxy, |
| CONNMAN_COLUMN_TYPE, &type, |
| -1); |
| |
| switch (type) { |
| case CONNMAN_TYPE_LABEL_ETHERNET: |
| case CONNMAN_TYPE_LABEL_WIFI: |
| case CONNMAN_TYPE_LABEL_CELLULAR: |
| case CONNMAN_TYPE_SYSCONFIG: |
| return TRUE; |
| } |
| |
| if (proxy == NULL) |
| return FALSE; |
| |
| active = g_str_equal(CONNMAN_SERVICE_INTERFACE, |
| dbus_g_proxy_get_interface(proxy)); |
| |
| g_object_unref(proxy); |
| |
| return active; |
| } |
| |
| GtkTreeModel *connman_client_get_device_model(ConnmanClient *client) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| GtkTreeModel *model; |
| |
| DBG("client %p", client); |
| |
| model = gtk_tree_model_filter_new(GTK_TREE_MODEL(priv->store), NULL); |
| |
| gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(model), |
| device_filter, NULL, NULL); |
| |
| return model; |
| } |
| |
| static void hash_table_value_string_insert( GHashTable *hash, gpointer key, const char *str ) |
| { |
| GValue *itemvalue; |
| |
| itemvalue = g_slice_new0(GValue); |
| g_value_init(itemvalue, G_TYPE_STRING); |
| g_value_set_string(itemvalue, str); |
| g_hash_table_insert(hash, key, itemvalue); |
| } |
| |
| gboolean connman_client_set_ipv4(ConnmanClient *client, const gchar *device, |
| struct ipv4_config *ipv4_config) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| DBusGProxy *proxy; |
| GValue value = { 0 }; |
| gboolean ret; |
| GHashTable *ipv4 = g_hash_table_new_full(g_str_hash, g_str_equal, NULL, g_free); |
| |
| DBG("client %p", client); |
| |
| if (device == NULL) |
| return FALSE; |
| |
| proxy = connman_dbus_get_proxy(priv->store, device); |
| if (proxy == NULL) |
| return FALSE; |
| |
| hash_table_value_string_insert(ipv4, "Method", ipv4_config->method); |
| if( g_strcmp0(ipv4_config->method, "dhcp" ) != 0 ) { |
| hash_table_value_string_insert(ipv4, "Address", ipv4_config->address); |
| hash_table_value_string_insert(ipv4, "Netmask", ipv4_config->netmask); |
| hash_table_value_string_insert(ipv4, "Gateway", ipv4_config->gateway); |
| } |
| |
| g_value_init(&value, DBUS_TYPE_G_DICTIONARY); |
| g_value_set_boxed(&value, ipv4); |
| ret = connman_set_property(proxy, "IPv4.Configuration", &value, NULL); |
| |
| g_object_unref(proxy); |
| |
| return ret; |
| } |
| |
| void connman_client_set_powered(ConnmanClient *client, const gchar *device, |
| gboolean powered) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| DBusGProxy *proxy; |
| GError *error; |
| GValue value = { 0 }; |
| |
| DBG("client %p device %s", client, device); |
| |
| if (device == NULL) |
| return; |
| |
| proxy = connman_dbus_get_proxy(priv->store, device); |
| if (proxy == NULL) |
| return; |
| |
| g_value_init(&value, G_TYPE_BOOLEAN); |
| g_value_set_boolean(&value, powered); |
| |
| error = NULL; |
| connman_set_property(proxy, "Powered", &value, &error); |
| if( error ) |
| fprintf (stderr, "error: %s\n", error->message); |
| |
| g_object_unref(proxy); |
| } |
| |
| void connman_client_scan(ConnmanClient *client, const gchar *device, |
| connman_scan_reply callback, gpointer user_data) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| DBusGProxy *proxy; |
| |
| DBG("client %p device %s", client, device); |
| |
| if (device == NULL) |
| return; |
| |
| proxy = connman_dbus_get_proxy(priv->store, device); |
| if (proxy == NULL) |
| return; |
| |
| connman_scan_async(proxy, callback, user_data); |
| |
| g_object_unref(proxy); |
| } |
| |
| gboolean connman_client_get_offline_status(ConnmanClient *client) |
| { |
| GHashTable *hash; |
| GValue *value; |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| gboolean ret; |
| |
| DBG("client %p", client); |
| |
| ret = connman_get_properties(priv->manager, &hash, NULL); |
| |
| if (ret == FALSE) |
| goto done; |
| |
| value = g_hash_table_lookup(hash, "OfflineMode"); |
| ret = value ? g_value_get_boolean(value) : FALSE; |
| |
| done: |
| return ret; |
| } |
| |
| void connman_client_set_offlinemode(ConnmanClient *client, gboolean status) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| GValue value = { 0 }; |
| |
| DBG("client %p", client); |
| |
| g_value_init(&value, G_TYPE_BOOLEAN); |
| g_value_set_boolean(&value, status); |
| |
| connman_set_property(priv->manager, "OfflineMode", &value, NULL); |
| } |
| |
| static gboolean network_disconnect(GtkTreeModel *model, GtkTreePath *path, |
| GtkTreeIter *iter, gpointer user_data) |
| { |
| DBusGProxy *proxy; |
| char *name; |
| guint type; |
| |
| gtk_tree_model_get(model, iter, CONNMAN_COLUMN_PROXY, &proxy, |
| CONNMAN_COLUMN_NAME, &name, |
| CONNMAN_COLUMN_TYPE, &type, |
| -1); |
| |
| if (proxy == NULL) |
| return TRUE; |
| |
| if (g_str_equal(dbus_g_proxy_get_interface(proxy), |
| CONNMAN_SERVICE_INTERFACE) == FALSE) |
| return TRUE; |
| |
| if (type == CONNMAN_TYPE_WIFI) |
| connman_disconnect(proxy, NULL); |
| |
| g_object_unref(proxy); |
| |
| return FALSE; |
| } |
| |
| void connman_client_connect(ConnmanClient *client, const gchar *network) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| DBusGProxy *proxy; |
| |
| DBG("client %p", client); |
| |
| if (network == NULL) |
| return; |
| |
| gtk_tree_model_foreach(GTK_TREE_MODEL(priv->store), |
| network_disconnect, NULL); |
| |
| proxy = connman_dbus_get_proxy(priv->store, network); |
| if (proxy == NULL) |
| return; |
| |
| connman_connect(proxy, NULL); |
| |
| g_object_unref(proxy); |
| } |
| |
| void connman_client_connect_async(ConnmanClient *client, const gchar *network, |
| connman_connect_reply callback, gpointer userdata) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| DBusGProxy *proxy; |
| |
| DBG("client %p", client); |
| DBG("network %s", network); |
| |
| if (network == NULL) |
| goto done; |
| |
| gtk_tree_model_foreach(GTK_TREE_MODEL(priv->store), |
| network_disconnect, NULL); |
| |
| proxy = connman_dbus_get_proxy(priv->store, network); |
| if (proxy == NULL) |
| goto done; |
| |
| connman_connect_async(proxy, callback, userdata); |
| |
| done: |
| return; |
| } |
| |
| static void connman_client_disconnect_all(ConnmanClient *client) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| |
| gtk_tree_model_foreach(GTK_TREE_MODEL(priv->store), |
| network_disconnect, NULL); |
| } |
| |
| void connman_client_disconnect(ConnmanClient *client, const gchar *network) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| DBusGProxy *proxy; |
| |
| DBG("client %p", client); |
| |
| if (network == NULL) { |
| connman_client_disconnect_all(client); |
| return; |
| } |
| |
| proxy = connman_dbus_get_proxy(priv->store, network); |
| if (proxy == NULL) |
| return; |
| |
| connman_disconnect(proxy, NULL); |
| |
| g_object_unref(proxy); |
| } |
| |
| gchar *connman_client_get_security(ConnmanClient *client, const gchar *network) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| GtkTreeIter iter; |
| gchar *security; |
| |
| DBG("client %p", client); |
| |
| if (network == NULL) |
| return CONNMAN_SECURITY_UNKNOWN; |
| |
| if (connman_dbus_get_iter(priv->store, network, &iter) == FALSE) |
| return CONNMAN_SECURITY_UNKNOWN; |
| |
| gtk_tree_model_get(GTK_TREE_MODEL(priv->store), &iter, |
| CONNMAN_COLUMN_SECURITY, &security, -1); |
| |
| return security; |
| } |
| |
| void connman_client_set_callback(ConnmanClient *client, |
| ConnmanClientCallback callback, gpointer user_data) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| gchar *state; |
| |
| DBG("client %p", client); |
| |
| priv->callback = callback; |
| priv->userdata = user_data; |
| |
| g_object_set_data(G_OBJECT(priv->store), "callback", callback); |
| g_object_set_data(G_OBJECT(priv->store), "userdata", user_data); |
| |
| state = g_object_get_data(G_OBJECT(priv->store), "State"); |
| if (state != NULL && priv->callback != NULL) |
| priv->callback(state, priv->userdata); |
| } |
| |
| void connman_client_remove(ConnmanClient *client, const gchar *network) |
| { |
| ConnmanClientPrivate *priv = CONNMAN_CLIENT_GET_PRIVATE(client); |
| DBusGProxy *proxy; |
| |
| if (network == NULL) |
| return; |
| |
| proxy = connman_dbus_get_proxy(priv->store, network); |
| if (proxy == NULL) |
| return; |
| |
| connman_remove(proxy, NULL); |
| |
| g_object_unref(proxy); |
| } |