blob: 33ac62378693e1636f2ed98ea319ae1292f203d1 [file] [log] [blame]
/*
*
* 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-client.h"
#include "connman-dbus.h"
#include "connman-dbus-glue.h"
#include "marshal.h"
#ifdef DEBUG
#define DBG(fmt, arg...) printf("%s:%s() " fmt "\n", __FILE__, __FUNCTION__ , ## arg)
#else
#define DBG(fmt...)
#endif
#ifndef DBUS_TYPE_G_OBJECT_PATH_ARRAY
#define DBUS_TYPE_G_OBJECT_PATH_ARRAY \
(dbus_g_type_get_collection("GPtrArray", DBUS_TYPE_G_OBJECT_PATH))
#endif
#ifndef DBUS_TYPE_G_DICTIONARY
#define DBUS_TYPE_G_DICTIONARY \
(dbus_g_type_get_map("GHashTable", G_TYPE_STRING, G_TYPE_VALUE))
#endif
static DBusGConnection *connection = NULL;
typedef gboolean (*IterSearchFunc) (GtkTreeStore *store,
GtkTreeIter *iter, gconstpointer user_data);
static gboolean iter_search(GtkTreeStore *store,
GtkTreeIter *iter, GtkTreeIter *parent,
IterSearchFunc func, gconstpointer user_data)
{
gboolean cont, found = FALSE;
if (parent == NULL)
cont = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store),
iter);
else
cont = gtk_tree_model_iter_children(GTK_TREE_MODEL(store),
iter, parent);
while (cont == TRUE) {
GtkTreeIter child;
found = func(store, iter, user_data);
if (found == TRUE)
break;
found = iter_search(store, &child, iter, func, user_data);
if (found == TRUE) {
*iter = child;
break;
}
cont = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), iter);
}
return found;
}
static gboolean compare_proxy(GtkTreeStore *store, GtkTreeIter *iter,
gconstpointer user_data)
{
const char *path = user_data;
DBusGProxy *proxy;
gboolean found = FALSE;
gtk_tree_model_get(GTK_TREE_MODEL(store), iter,
CONNMAN_COLUMN_PROXY, &proxy, -1);
if (proxy != NULL) {
found = g_str_equal(path, dbus_g_proxy_get_path(proxy));
g_object_unref(proxy);
}
return found;
}
static gboolean get_iter_from_proxy(GtkTreeStore *store,
GtkTreeIter *iter, DBusGProxy *proxy)
{
const char *path;
if (proxy == NULL)
return FALSE;
path = dbus_g_proxy_get_path(proxy);
if (path == NULL)
return FALSE;
return iter_search(store, iter, NULL, compare_proxy, path);
}
static gboolean get_iter_from_path(GtkTreeStore *store,
GtkTreeIter *iter, const char *path)
{
if (path == NULL)
return FALSE;
return iter_search(store, iter, NULL, compare_proxy, path);
}
DBusGProxy *connman_dbus_get_proxy(GtkTreeStore *store, const gchar *path)
{
DBusGProxy *proxy;
GtkTreeIter iter;
if (iter_search(store, &iter, NULL, compare_proxy, path) == FALSE)
return NULL;
gtk_tree_model_get(GTK_TREE_MODEL(store), &iter,
CONNMAN_COLUMN_PROXY, &proxy, -1);
return proxy;
}
static gboolean compare_type(GtkTreeStore *store, GtkTreeIter *iter,
gconstpointer user_data)
{
guint type_target = GPOINTER_TO_UINT(user_data);
guint type;
gboolean found = FALSE;
gtk_tree_model_get(GTK_TREE_MODEL(store), iter,
CONNMAN_COLUMN_TYPE, &type, -1);
if (type != CONNMAN_TYPE_UNKNOWN)
found = (type == type_target);
return found;
}
static gboolean get_iter_from_type(GtkTreeStore *store, GtkTreeIter *iter, guint type)
{
return iter_search(store, iter, NULL, compare_type, GUINT_TO_POINTER(type));
}
gboolean connman_dbus_get_iter(GtkTreeStore *store, const gchar *path,
GtkTreeIter *iter)
{
return get_iter_from_path(store, iter, path);
}
static guint get_type(const GValue *value)
{
const char *type = value ? g_value_get_string(value) : NULL;
if (type == NULL)
return CONNMAN_TYPE_UNKNOWN;
else if (g_str_equal(type, "ethernet") == TRUE)
return CONNMAN_TYPE_ETHERNET;
else if (g_str_equal(type, "wifi") == TRUE)
return CONNMAN_TYPE_WIFI;
else if (g_str_equal(type, "wimax") == TRUE)
return CONNMAN_TYPE_WIMAX;
else if (g_str_equal(type, "bluetooth") == TRUE)
return CONNMAN_TYPE_BLUETOOTH;
else if (g_str_equal(type, "cellular") == TRUE)
return CONNMAN_TYPE_CELLULAR;
return CONNMAN_TYPE_UNKNOWN;
}
static const gchar *type2icon(guint type)
{
switch (type) {
case CONNMAN_TYPE_ETHERNET:
return "network-wired";
case CONNMAN_TYPE_WIFI:
case CONNMAN_TYPE_WIMAX:
return "network-wireless";
case CONNMAN_TYPE_CELLULAR:
return "network-cellular";
case CONNMAN_TYPE_BLUETOOTH:
return "bluetooth";
}
return NULL;
}
static void tech_changed(DBusGProxy *proxy, const char *property,
GValue *value, gpointer user_data)
{
GtkTreeStore *store = user_data;
const char *path = dbus_g_proxy_get_path(proxy);
GtkTreeIter iter;
DBG("store %p proxy %p property %s", store, proxy, property);
if (property == NULL || value == NULL)
return;
if (get_iter_from_path(store, &iter, path) == FALSE)
return;
if (g_str_equal(property, "Powered") == TRUE) {
gboolean powered = g_value_get_boolean(value);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_POWERED, powered, -1);
}
}
static void tech_properties(DBusGProxy *proxy, GHashTable *hash,
GError *error, gpointer user_data)
{
GtkTreeStore *store = user_data;
GtkTreeIter iter;
gboolean powered = FALSE;
GValue *propval = 0;
const char *techtype = 0;
propval = g_hash_table_lookup(hash, "Type");
techtype = propval ? g_value_get_string(propval) : NULL;
propval = g_hash_table_lookup(hash, "Powered");
powered = propval ? g_value_get_boolean(propval) : FALSE;
if (g_str_equal("ethernet", techtype))
{
if (get_iter_from_type(store, &iter, CONNMAN_TYPE_LABEL_ETHERNET) == FALSE)
gtk_tree_store_append(store, &iter, NULL);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_PROXY, proxy,
CONNMAN_COLUMN_POWERED, powered,
CONNMAN_COLUMN_TYPE, CONNMAN_TYPE_LABEL_ETHERNET,
-1);
}
else if (g_str_equal ("wifi", techtype))
{
if (get_iter_from_type(store, &iter, CONNMAN_TYPE_LABEL_WIFI) == FALSE)
gtk_tree_store_append(store, &iter, NULL);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_PROXY, proxy,
CONNMAN_COLUMN_POWERED, powered,
CONNMAN_COLUMN_TYPE, CONNMAN_TYPE_LABEL_WIFI,
-1);
}
else if (g_str_equal ("3g", techtype))
{
if (get_iter_from_type(store, &iter, CONNMAN_TYPE_LABEL_CELLULAR) == FALSE)
gtk_tree_store_append(store, &iter, NULL);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_PROXY, proxy,
CONNMAN_COLUMN_POWERED, powered,
CONNMAN_COLUMN_TYPE, CONNMAN_TYPE_LABEL_CELLULAR,
-1);
}
}
static void offline_mode_changed(GtkTreeStore *store, GValue *value)
{
GtkTreeIter iter;
gboolean offline_mode = g_value_get_boolean(value);
get_iter_from_type(store, &iter, CONNMAN_TYPE_SYSCONFIG);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_OFFLINEMODE, offline_mode,
-1);
}
static void tech_added(DBusGProxy *proxy, DBusGObjectPath *path,
GHashTable *hash, gpointer user_data)
{
GtkTreeStore *store = user_data;
GtkTreeIter iter;
DBG("store %p proxy %p hash %p", store, proxy, hash);
if (!get_iter_from_path(store, &iter, path)) {
DBusGProxy *tech_proxy = dbus_g_proxy_new_for_name(connection,
CONNMAN_SERVICE, path,
CONNMAN_TECHNOLOGY_INTERFACE);
if (tech_proxy == NULL)
return;
tech_properties(tech_proxy, hash, NULL, user_data);
dbus_g_proxy_add_signal(tech_proxy, "PropertyChanged",
G_TYPE_STRING, G_TYPE_VALUE, G_TYPE_INVALID);
dbus_g_proxy_connect_signal(tech_proxy, "PropertyChanged",
G_CALLBACK(tech_changed), store, NULL);
}
}
static void tech_removed(DBusGProxy *proxy, DBusGObjectPath *path,
gpointer user_data)
{
GtkTreeStore *store = user_data;
GtkTreeIter iter;
if (get_iter_from_path(store, &iter, path))
gtk_tree_store_remove(store, &iter);
}
static void offline_mode_properties(GtkTreeStore *store, DBusGProxy *proxy, GValue *value)
{
GtkTreeIter iter;
gboolean offline_mode = g_value_get_boolean(value);
if (get_iter_from_type(store, &iter, CONNMAN_TYPE_SYSCONFIG) == FALSE)
gtk_tree_store_insert(store, &iter, NULL, 0);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_PROXY, proxy,
CONNMAN_COLUMN_TYPE, CONNMAN_TYPE_SYSCONFIG,
CONNMAN_COLUMN_OFFLINEMODE, offline_mode,
-1);
}
static void service_changed(DBusGProxy *proxy, const char *property,
GValue *value, gpointer user_data)
{
GtkTreeStore *store = user_data;
const char *path = dbus_g_proxy_get_path(proxy);
GtkTreeIter iter;
GHashTable *ipv4;
const char *method, *addr, *netmask, *gateway;
GValue *ipv4_method, *ipv4_address, *ipv4_netmask, *ipv4_gateway;
DBG("store %p proxy %p property %s", store, proxy, property);
if (property == NULL || value == NULL)
return;
if (get_iter_from_path(store, &iter, path) == FALSE)
return;
if (g_str_equal(property, "IPv4") == TRUE) {
ipv4 = g_value_get_boxed (value);
if (!ipv4)
return;
ipv4_method = g_hash_table_lookup (ipv4, "Method");
method = ipv4_method ? g_value_get_string(ipv4_method) : NULL;
ipv4_address = g_hash_table_lookup (ipv4, "Address");
addr = ipv4_address ? g_value_get_string(ipv4_address) : NULL;
ipv4_netmask = g_hash_table_lookup (ipv4, "Netmask");
netmask = ipv4_netmask ? g_value_get_string(ipv4_netmask) : NULL;
ipv4_gateway = g_hash_table_lookup (ipv4, "Gateway");
gateway = ipv4_gateway ? g_value_get_string(ipv4_gateway) : NULL;
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_METHOD, method,
CONNMAN_COLUMN_ADDRESS, addr,
CONNMAN_COLUMN_NETMASK, netmask,
CONNMAN_COLUMN_GATEWAY, gateway,
-1);
} else if (g_str_equal(property, "State") == TRUE) {
const char *state = value ? g_value_get_string(value) : NULL;
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_STATE, state, -1);
} else if (g_str_equal(property, "Favorite") == TRUE) {
gboolean favorite = g_value_get_boolean(value);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_FAVORITE, favorite, -1);
} else if (g_str_equal(property, "Security") == TRUE) {
char **array = value ? g_value_get_boxed(value) : NULL;
const char *security = g_strjoinv(" ", array);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_SECURITY, security,
-1);
} else if (g_str_equal(property, "Strength") == TRUE) {
guint strength = g_value_get_uchar(value);
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_STRENGTH, strength, -1);
}
}
static void service_properties(DBusGProxy *proxy, GHashTable *hash,
gpointer user_data)
{
GtkTreeStore *store = user_data;
GValue *value;
const gchar *name, *icon, *security, *state;
guint type, strength;
gboolean favorite;
GtkTreeIter iter;
GHashTable *ipv4;
GValue *ipv4_method, *ipv4_address, *ipv4_netmask, *ipv4_gateway;
const char *method, *addr, *netmask, *gateway;
DBG("store %p proxy %p hash %p", store, proxy, hash);
if (hash == NULL)
goto done;
value = g_hash_table_lookup(hash, "Name");
name = value ? g_value_get_string(value) : NULL;
value = g_hash_table_lookup(hash, "Type");
type = get_type(value);
icon = type2icon(type);
value = g_hash_table_lookup(hash, "State");
state = value ? g_value_get_string(value) : NULL;
value = g_hash_table_lookup(hash, "Favorite");
favorite = value ? g_value_get_boolean(value) : FALSE;
value = g_hash_table_lookup(hash, "Strength");
strength = value ? g_value_get_uchar(value) : 0;
value = g_hash_table_lookup(hash, "Security");
security = value ? g_strjoinv(" ", g_value_get_boxed(value)) : NULL;
DBG("name %s type %d icon %s", name, type, icon);
value = g_hash_table_lookup(hash, "IPv4.Configuration");
ipv4 = g_value_get_boxed (value);
if (!ipv4)
goto done;
ipv4_method = g_hash_table_lookup (ipv4, "Method");
method = ipv4_method ? g_value_get_string(ipv4_method) : NULL;
ipv4_address = g_hash_table_lookup (ipv4, "Address");
addr = ipv4_address ? g_value_get_string(ipv4_address) : NULL;
ipv4_netmask = g_hash_table_lookup (ipv4, "Netmask");
netmask = ipv4_netmask ? g_value_get_string(ipv4_netmask) : NULL;
ipv4_gateway = g_hash_table_lookup (ipv4, "Gateway");
gateway = ipv4_gateway ? g_value_get_string(ipv4_gateway) : NULL;
if (get_iter_from_proxy(store, &iter, proxy) == FALSE) {
GtkTreeIter label_iter;
guint label_type;
switch (type) {
case CONNMAN_TYPE_ETHERNET:
label_type = CONNMAN_TYPE_LABEL_ETHERNET;
break;
case CONNMAN_TYPE_WIFI:
label_type = CONNMAN_TYPE_LABEL_WIFI;
break;
case CONNMAN_TYPE_CELLULAR:
label_type = CONNMAN_TYPE_LABEL_CELLULAR;
break;
default:
label_type = CONNMAN_TYPE_UNKNOWN;
break;
}
get_iter_from_type(store, &label_iter, label_type);
gtk_tree_store_insert_after(store, &iter, NULL, &label_iter);
dbus_g_proxy_add_signal(proxy, "PropertyChanged",
G_TYPE_STRING, G_TYPE_VALUE, G_TYPE_INVALID);
dbus_g_proxy_connect_signal(proxy, "PropertyChanged",
G_CALLBACK(service_changed), store, NULL);
}
gtk_tree_store_set(store, &iter,
CONNMAN_COLUMN_PROXY, proxy,
CONNMAN_COLUMN_NAME, name,
CONNMAN_COLUMN_ICON, icon,
CONNMAN_COLUMN_TYPE, type,
CONNMAN_COLUMN_STATE, state,
CONNMAN_COLUMN_FAVORITE, favorite,
CONNMAN_COLUMN_SECURITY, security,
CONNMAN_COLUMN_STRENGTH, strength,
CONNMAN_COLUMN_METHOD, method,
CONNMAN_COLUMN_ADDRESS, addr,
CONNMAN_COLUMN_NETMASK, netmask,
CONNMAN_COLUMN_GATEWAY, gateway,
-1);
done:
g_object_unref(proxy);
}
static void manager_changed(DBusGProxy *proxy, const char *property,
GValue *value, gpointer user_data)
{
GtkTreeStore *store = user_data;
DBG("store %p proxy %p property %s", store, proxy, property);
if (property == NULL || value == NULL)
return;
if (g_str_equal(property, "OfflineMode") == TRUE)
offline_mode_changed(store, value);
}
static void manager_properties(DBusGProxy *proxy, GHashTable *hash,
GError *error, gpointer user_data)
{
GtkTreeStore *store = user_data;
GValue *value;
DBG("store %p proxy %p hash %p", store, proxy, hash);
if (error != NULL || hash == NULL)
return;
value = g_hash_table_lookup(hash, "OfflineMode");
if (value != NULL)
offline_mode_properties(store, proxy, value);
}
static void manager_services(DBusGProxy *proxy, GPtrArray *array,
GError *error, gpointer user_data)
{
unsigned int i;
DBG("proxy %p array %p", proxy, array);
if (error != NULL || array == NULL)
return;
for (i = 0; i < array->len; i++)
{
GHashTable *props;
GValueArray *item = g_ptr_array_index(array, i);
DBusGObjectPath *path = (DBusGObjectPath *)g_value_get_boxed(g_value_array_get_nth(item, 0));
DBusGProxy *service_proxy = dbus_g_proxy_new_for_name(connection,
CONNMAN_SERVICE, path,
CONNMAN_SERVICE_INTERFACE);
if (service_proxy == NULL)
continue;
props = (GHashTable *)g_value_get_boxed(g_value_array_get_nth(item, 1));
service_properties(service_proxy, props, user_data);
}
}
static void manager_technologies(DBusGProxy *proxy, GPtrArray *array,
GError *error, gpointer user_data)
{
unsigned int i;
DBG("proxy %p array %p", proxy, array);
if (error != NULL || array == NULL)
return;
for (i = 0; i < array->len; i++)
{
GValueArray *item = g_ptr_array_index(array, i);
DBusGObjectPath *path = (DBusGObjectPath *)g_value_get_boxed(g_value_array_get_nth(item, 0));
GHashTable *props = (GHashTable *)g_value_get_boxed(g_value_array_get_nth(item, 1));
tech_added(proxy, path, props, user_data);
}
}
static void services_added(DBusGProxy *proxy, GPtrArray *array,
gpointer user_data)
{
DBG("proxy %p array %p", proxy, array);
manager_services(proxy, array, NULL, user_data);
}
static void services_removed(DBusGProxy *proxy, GPtrArray *array,
gpointer user_data)
{
GtkTreeStore *store = user_data;
GtkTreeIter iter;
unsigned int i;
DBG("store %p proxy %p array %p", store, proxy, array);
for (i = 0; i < array->len; i++)
{
DBusGObjectPath *path = (DBusGObjectPath *)g_ptr_array_index(array, i);
if (get_iter_from_path(store, &iter, path))
gtk_tree_store_remove(store, &iter);
}
}
DBusGProxy *connman_dbus_create_manager(DBusGConnection *conn,
GtkTreeStore *store)
{
DBusGProxy *proxy;
GType otype;
connection = dbus_g_connection_ref(conn);
proxy = dbus_g_proxy_new_for_name(connection, CONNMAN_SERVICE,
CONNMAN_MANAGER_PATH, CONNMAN_MANAGER_INTERFACE);
DBG("store %p proxy %p", store, proxy);
dbus_g_proxy_add_signal(proxy, "PropertyChanged",
G_TYPE_STRING, G_TYPE_VALUE, G_TYPE_INVALID);
dbus_g_proxy_connect_signal(proxy, "PropertyChanged",
G_CALLBACK(manager_changed), store, NULL);
otype = dbus_g_type_get_struct("GValueArray", DBUS_TYPE_G_OBJECT_PATH, DBUS_TYPE_G_DICTIONARY, G_TYPE_INVALID);
otype = dbus_g_type_get_collection("GPtrArray", otype);
dbus_g_object_register_marshaller(marshal_VOID__BOXED, G_TYPE_NONE, otype, G_TYPE_INVALID);
dbus_g_proxy_add_signal(proxy, "ServicesAdded",
otype, G_TYPE_INVALID);
dbus_g_proxy_connect_signal(proxy, "ServicesAdded",
G_CALLBACK(services_added), store, NULL);
otype = DBUS_TYPE_G_OBJECT_PATH_ARRAY;
dbus_g_object_register_marshaller(marshal_VOID__BOXED, G_TYPE_NONE, otype, G_TYPE_INVALID);
dbus_g_proxy_add_signal(proxy, "ServicesRemoved",
otype, G_TYPE_INVALID);
dbus_g_proxy_connect_signal(proxy, "ServicesRemoved",
G_CALLBACK(services_removed), store, NULL);
dbus_g_object_register_marshaller(marshal_VOID__STRING_BOXED, G_TYPE_NONE, DBUS_TYPE_G_OBJECT_PATH, DBUS_TYPE_G_DICTIONARY, G_TYPE_INVALID);
dbus_g_proxy_add_signal(proxy, "TechnologyAdded",
DBUS_TYPE_G_OBJECT_PATH, DBUS_TYPE_G_DICTIONARY, G_TYPE_INVALID);
dbus_g_proxy_connect_signal(proxy, "TechnologyAdded",
G_CALLBACK(tech_added), store, NULL);
dbus_g_object_register_marshaller(marshal_VOID__STRING, G_TYPE_NONE, DBUS_TYPE_G_OBJECT_PATH, G_TYPE_INVALID);
dbus_g_proxy_add_signal(proxy, "TechnologyRemoved",
DBUS_TYPE_G_OBJECT_PATH, G_TYPE_INVALID);
dbus_g_proxy_connect_signal(proxy, "TechnologyRemoved",
G_CALLBACK(tech_removed), store, NULL);
DBG("getting manager properties");
connman_get_properties_async(proxy, manager_properties, store);
DBG("getting technologies");
connman_get_technologies_async(proxy, manager_technologies, store);
DBG("getting services");
connman_get_services_async(proxy, manager_services, store);
return proxy;
}
void connman_dbus_destroy_manager(DBusGProxy *proxy, GtkTreeStore *store)
{
DBG("store %p proxy %p", store, proxy);
g_signal_handlers_disconnect_by_func(proxy, manager_changed, store);
g_object_unref(proxy);
dbus_g_connection_unref(connection);
}