X-Git-Url: http://review.tizen.org/git/?a=blobdiff_plain;f=gdbus%2Fwatch.c;h=fba58c32ba8748985b9456e83a7ac18381b1158b;hb=0baf5237502d01fcc977ef973aafd9800065548d;hp=df3944ca4d2167a314bb76dc51c3acd46e38f9c7;hpb=4b84588d1efa628e4cec71749f108ee3a9b3677f;p=framework%2Fconnectivity%2Fconnman.git diff --git a/gdbus/watch.c b/gdbus/watch.c index df3944c..fba58c3 100644 --- a/gdbus/watch.c +++ b/gdbus/watch.c @@ -2,7 +2,7 @@ * * D-Bus helper library * - * Copyright (C) 2004-2009 Marcel Holtmann + * Copyright (C) 2004-2011 Marcel Holtmann * * * This program is free software; you can redistribute it and/or modify @@ -43,11 +43,21 @@ static DBusHandlerResult message_filter(DBusConnection *connection, static guint listener_id = 0; static GSList *listeners = NULL; +struct service_data { + DBusConnection *conn; + DBusPendingCall *call; + char *name; + const char *owner; + guint id; + struct filter_callback *callback; +}; + struct filter_callback { GDBusWatchFunction conn_func; GDBusWatchFunction disc_func; GDBusSignalFunction signal_func; GDBusDestroyFunction destroy_func; + struct service_data *data; void *user_data; guint id; }; @@ -55,19 +65,22 @@ struct filter_callback { struct filter_data { DBusConnection *connection; DBusHandleMessageFunction handle_func; - char *sender; + char *name; + char *owner; char *path; char *interface; char *member; char *argument; GSList *callbacks; GSList *processed; + guint name_watch; gboolean lock; gboolean registered; }; static struct filter_data *filter_data_find(DBusConnection *connection, - const char *sender, + const char *name, + const char *owner, const char *path, const char *interface, const char *member, @@ -82,8 +95,12 @@ static struct filter_data *filter_data_find(DBusConnection *connection, if (connection != data->connection) continue; - if (sender && data->sender && - g_str_equal(sender, data->sender) == FALSE) + if (name && data->name && + g_str_equal(name, data->name) == FALSE) + continue; + + if (owner && data->owner && + g_str_equal(owner, data->owner) == FALSE) continue; if (path && data->path && @@ -110,13 +127,15 @@ static struct filter_data *filter_data_find(DBusConnection *connection, static void format_rule(struct filter_data *data, char *rule, size_t size) { + const char *sender; int offset; offset = snprintf(rule, size, "type='signal'"); + sender = data->name ? : data->owner; - if (data->sender) + if (sender) offset += snprintf(rule + offset, size - offset, - ",sender='%s'", data->sender); + ",sender='%s'", sender); if (data->path) offset += snprintf(rule + offset, size - offset, ",path='%s'", data->path); @@ -183,8 +202,9 @@ static struct filter_data *filter_data_get(DBusConnection *connection, const char *argument) { struct filter_data *data; + const char *name = NULL, *owner = NULL; - if (!filter_data_find(connection, NULL, NULL, NULL, NULL, NULL)) { + if (filter_data_find(connection, NULL, NULL, NULL, NULL, NULL, NULL) == NULL) { if (!dbus_connection_add_filter(connection, message_filter, NULL, NULL)) { error("dbus_connection_add_filter() failed"); @@ -192,15 +212,25 @@ static struct filter_data *filter_data_get(DBusConnection *connection, } } - data = filter_data_find(connection, sender, path, interface, member, - argument); + if (sender == NULL) + goto proceed; + + if (sender[0] == ':') + owner = sender; + else + name = sender; + +proceed: + data = filter_data_find(connection, name, owner, path, interface, + member, argument); if (data) return data; data = g_new0(struct filter_data, 1); data->connection = dbus_connection_ref(connection); - data->sender = g_strdup(sender); + data->name = name ? g_strdup(name) : NULL; + data->owner = owner ? g_strdup(owner) : NULL; data->path = g_strdup(path); data->interface = g_strdup(interface); data->member = g_strdup(member); @@ -244,7 +274,9 @@ static void filter_data_free(struct filter_data *data) g_free(l->data); g_slist_free(data->callbacks); - g_free(data->sender); + g_dbus_remove_watch(data->connection, data->name_watch); + g_free(data->name); + g_free(data->owner); g_free(data->path); g_free(data->interface); g_free(data->member); @@ -279,7 +311,7 @@ static struct filter_callback *filter_data_add_callback( { struct filter_callback *cb = NULL; - cb = g_new(struct filter_callback, 1); + cb = g_new0(struct filter_callback, 1); cb->conn_func = connect; cb->disc_func = disconnect; @@ -296,12 +328,39 @@ static struct filter_callback *filter_data_add_callback( return cb; } +static void service_data_free(struct service_data *data) +{ + struct filter_callback *callback = data->callback; + + dbus_connection_unref(data->conn); + + if (data->call) + dbus_pending_call_unref(data->call); + + if (data->id) + g_source_remove(data->id); + + g_free(data->name); + g_free(data); + + callback->data = NULL; +} + static gboolean filter_data_remove_callback(struct filter_data *data, struct filter_callback *cb) { + DBusConnection *connection; + data->callbacks = g_slist_remove(data->callbacks, cb); data->processed = g_slist_remove(data->processed, cb); + /* Cancel pending operations */ + if (cb->data) { + if (cb->data->call) + dbus_pending_call_cancel(cb->data->call); + service_data_free(cb->data); + } + if (cb->destroy_func) cb->destroy_func(cb->user_data); @@ -315,15 +374,18 @@ static gboolean filter_data_remove_callback(struct filter_data *data, if (data->registered && !remove_match(data)) return FALSE; + connection = dbus_connection_ref(data->connection); + listeners = g_slist_remove(listeners, data); + filter_data_free(data); + /* Remove filter if there are no listeners left for the connection */ - data = filter_data_find(data->connection, NULL, NULL, NULL, NULL, + data = filter_data_find(connection, NULL, NULL, NULL, NULL, NULL, NULL); - if (!data) - dbus_connection_remove_filter(data->connection, message_filter, + if (data == NULL) + dbus_connection_remove_filter(connection, message_filter, NULL); - listeners = g_slist_remove(listeners, data); - filter_data_free(data); + dbus_connection_unref(connection); return TRUE; } @@ -355,6 +417,37 @@ static DBusHandlerResult signal_filter(DBusConnection *connection, return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; } +static void update_name_cache(const char *name, const char *owner) +{ + GSList *l; + + for (l = listeners; l != NULL; l = l->next) { + struct filter_data *data = l->data; + + if (g_strcmp0(data->name, name) != 0) + continue; + + g_free(data->owner); + data->owner = g_strdup(owner); + } +} + +static const char *check_name_cache(const char *name) +{ + GSList *l; + + for (l = listeners; l != NULL; l = l->next) { + struct filter_data *data = l->data; + + if (g_strcmp0(data->name, name) != 0) + continue; + + return data->owner; + } + + return NULL; +} + static DBusHandlerResult service_filter(DBusConnection *connection, DBusMessage *message, void *user_data) { @@ -371,6 +464,8 @@ static DBusHandlerResult service_filter(DBusConnection *connection, return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; } + update_name_cache(name, new); + while (data->callbacks) { cb = data->callbacks->data; @@ -387,13 +482,14 @@ static DBusHandlerResult service_filter(DBusConnection *connection, if (!g_slist_find(data->callbacks, cb)) continue; - data->callbacks = g_slist_remove(data->callbacks, cb); - - if (!cb->conn_func || !cb->disc_func) { - g_free(cb); + /* Only auto remove if it is a bus name watch */ + if (data->argument[0] == ':' && + (cb->conn_func == NULL || cb->disc_func == NULL)) { + filter_data_remove_callback(data, cb); continue; } + data->callbacks = g_slist_remove(data->callbacks, cb); data->processed = g_slist_append(data->processed, cb); } @@ -417,8 +513,10 @@ static DBusHandlerResult message_filter(DBusConnection *connection, member = dbus_message_get_member(message); dbus_message_get_args(message, NULL, DBUS_TYPE_STRING, &arg, DBUS_TYPE_INVALID); - data = filter_data_find(connection, sender, path, iface, member, arg); - if (!data) { + /* Sender is always bus name */ + data = filter_data_find(connection, NULL, sender, path, iface, member, + arg); + if (data == NULL) { error("Got %s.%s signal which has no listeners", iface, member); return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; } @@ -442,57 +540,66 @@ static DBusHandlerResult message_filter(DBusConnection *connection, filter_data_free(data); /* Remove filter if there no listener left for the connection */ - data = filter_data_find(connection, NULL, NULL, NULL, NULL, NULL); - if (!data) + data = filter_data_find(connection, NULL, NULL, NULL, NULL, NULL, + NULL); + if (data == NULL) dbus_connection_remove_filter(connection, message_filter, NULL); return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; } -struct service_data { - DBusConnection *conn; - GDBusWatchFunction conn_func; - void *user_data; -}; +static gboolean update_service(void *user_data) +{ + struct service_data *data = user_data; + struct filter_callback *cb = data->callback; + + update_name_cache(data->name, data->owner); + if (cb->conn_func) + cb->conn_func(data->conn, cb->user_data); + + service_data_free(data); + + return FALSE; +} static void service_reply(DBusPendingCall *call, void *user_data) { struct service_data *data = user_data; DBusMessage *reply; - DBusError error; - dbus_bool_t has_owner; + DBusError err; reply = dbus_pending_call_steal_reply(call); if (reply == NULL) return; - dbus_error_init(&error); + dbus_error_init(&err); - if (dbus_message_get_args(reply, &error, - DBUS_TYPE_BOOLEAN, &has_owner, - DBUS_TYPE_INVALID) == FALSE) { - if (dbus_error_is_set(&error) == TRUE) { - error("%s", error.message); - dbus_error_free(&error); - } else { - error("Wrong arguments for NameHasOwner reply"); - } - goto done; - } + if (dbus_set_error_from_message(&err, reply)) + goto fail; + + if (dbus_message_get_args(reply, &err, + DBUS_TYPE_STRING, &data->owner, + DBUS_TYPE_INVALID) == FALSE) + goto fail; + + update_service(data); - if (has_owner && data->conn_func) - data->conn_func(data->conn, data->user_data); + goto done; +fail: + error("%s", err.message); + dbus_error_free(&err); + service_data_free(data); done: dbus_message_unref(reply); } -static void check_service(DBusConnection *connection, const char *name, - GDBusWatchFunction connect, void *user_data) +static void check_service(DBusConnection *connection, + const char *name, + struct filter_callback *callback) { DBusMessage *message; - DBusPendingCall *call; struct service_data *data; data = g_try_malloc0(sizeof(*data)); @@ -501,12 +608,19 @@ static void check_service(DBusConnection *connection, const char *name, return; } - data->conn = connection; - data->conn_func = connect; - data->user_data = user_data; + data->conn = dbus_connection_ref(connection); + data->name = g_strdup(name); + data->callback = callback; + callback->data = data; + + data->owner = check_name_cache(name); + if (data->owner != NULL) { + data->id = g_idle_add(update_service, data); + return; + } message = dbus_message_new_method_call(DBUS_SERVICE_DBUS, - DBUS_PATH_DBUS, DBUS_INTERFACE_DBUS, "NameHasOwner"); + DBUS_PATH_DBUS, DBUS_INTERFACE_DBUS, "GetNameOwner"); if (message == NULL) { error("Can't allocate new message"); g_free(data); @@ -517,19 +631,19 @@ static void check_service(DBusConnection *connection, const char *name, DBUS_TYPE_INVALID); if (dbus_connection_send_with_reply(connection, message, - &call, -1) == FALSE) { + &data->call, -1) == FALSE) { error("Failed to execute method call"); g_free(data); goto done; } - if (call == NULL) { + if (data->call == NULL) { error("D-Bus connection not available"); g_free(data); goto done; } - dbus_pending_call_set_notify(call, service_reply, data, NULL); + dbus_pending_call_set_notify(data->call, service_reply, data, NULL); done: dbus_message_unref(message); @@ -543,22 +657,22 @@ guint g_dbus_add_service_watch(DBusConnection *connection, const char *name, struct filter_data *data; struct filter_callback *cb; - if (!name) + if (name == NULL) return 0; data = filter_data_get(connection, service_filter, NULL, NULL, DBUS_INTERFACE_DBUS, "NameOwnerChanged", name); - if (!data) + if (data == NULL) return 0; cb = filter_data_add_callback(data, connect, disconnect, NULL, NULL, user_data); - if (!cb) + if (cb == NULL) return 0; if (connect) - check_service(connection, name, connect, user_data); + check_service(connection, name, cb); return cb->id; } @@ -582,14 +696,19 @@ guint g_dbus_add_signal_watch(DBusConnection *connection, data = filter_data_get(connection, signal_filter, sender, path, interface, member, NULL); - if (!data) + if (data == NULL) return 0; cb = filter_data_add_callback(data, NULL, NULL, function, destroy, user_data); - if (!cb) + if (cb == NULL) return 0; + if (data->name != NULL && data->name_watch == 0) + data->name_watch = g_dbus_add_service_watch(connection, + data->name, NULL, + NULL, NULL, NULL); + return cb->id; } @@ -619,7 +738,8 @@ void g_dbus_remove_all_watches(DBusConnection *connection) { struct filter_data *data; - while ((data = filter_data_find(connection, NULL, NULL, NULL, NULL, NULL))) { + while ((data = filter_data_find(connection, NULL, NULL, NULL, NULL, + NULL, NULL))) { listeners = g_slist_remove(listeners, data); filter_data_call_and_free(data); }