/* -*- mode: C; c-file-style: "gnu" -*- */ /* services.c Service management * * Copyright (C) 2003 Red Hat, Inc. * Copyright (C) 2003 CodeFactory AB * * Licensed under the Academic Free License version 1.2 * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ #include #include #include #include "driver.h" #include "services.h" #include "connection.h" #include "utils.h" #include "activation.h" struct BusService { BusRegistry *registry; char *name; DBusList *owners; unsigned int prohibit_replacement : 1; }; struct BusRegistry { int refcount; BusContext *context; DBusHashTable *service_hash; DBusMemPool *service_pool; }; BusRegistry* bus_registry_new (BusContext *context) { BusRegistry *registry; registry = dbus_new0 (BusRegistry, 1); if (registry == NULL) return NULL; registry->refcount = 1; registry->context = context; registry->service_hash = _dbus_hash_table_new (DBUS_HASH_STRING, NULL, NULL); if (registry->service_hash == NULL) goto failed; registry->service_pool = _dbus_mem_pool_new (sizeof (BusService), TRUE); if (registry->service_pool == NULL) goto failed; return registry; failed: bus_registry_unref (registry); return NULL; } void bus_registry_ref (BusRegistry *registry) { _dbus_assert (registry->refcount > 0); registry->refcount += 1; } void bus_registry_unref (BusRegistry *registry) { _dbus_assert (registry->refcount > 0); registry->refcount -= 1; if (registry->refcount == 0) { if (registry->service_hash) _dbus_hash_table_unref (registry->service_hash); if (registry->service_pool) _dbus_mem_pool_free (registry->service_pool); dbus_free (registry); } } BusService* bus_registry_lookup (BusRegistry *registry, const DBusString *service_name) { BusService *service; service = _dbus_hash_table_lookup_string (registry->service_hash, _dbus_string_get_const_data (service_name)); return service; } BusService* bus_registry_ensure (BusRegistry *registry, const DBusString *service_name, DBusConnection *owner_if_created, BusTransaction *transaction, DBusError *error) { BusService *service; _DBUS_ASSERT_ERROR_IS_CLEAR (error); _dbus_assert (owner_if_created != NULL); _dbus_assert (transaction != NULL); service = _dbus_hash_table_lookup_string (registry->service_hash, _dbus_string_get_const_data (service_name)); if (service != NULL) return service; service = _dbus_mem_pool_alloc (registry->service_pool); if (service == NULL) { BUS_SET_OOM (error); return NULL; } service->registry = registry; if (!_dbus_string_copy_data (service_name, &service->name)) { _dbus_mem_pool_dealloc (registry->service_pool, service); BUS_SET_OOM (error); return NULL; } if (!bus_driver_send_service_created (service->name, transaction, error)) { dbus_free (service->name); _dbus_mem_pool_dealloc (registry->service_pool, service); return NULL; } if (!bus_activation_service_created (bus_context_get_activation (registry->context), service->name, transaction, error)) { dbus_free (service->name); _dbus_mem_pool_dealloc (registry->service_pool, service); return NULL; } if (!bus_service_add_owner (service, owner_if_created, transaction, error)) { dbus_free (service->name); _dbus_mem_pool_dealloc (registry->service_pool, service); return NULL; } if (!_dbus_hash_table_insert_string (registry->service_hash, service->name, service)) { bus_connection_remove_owned_service (owner_if_created, service); _dbus_list_clear (&service->owners); dbus_free (service->name); _dbus_mem_pool_dealloc (registry->service_pool, service); BUS_SET_OOM (error); return NULL; } return service; } void bus_registry_foreach (BusRegistry *registry, BusServiceForeachFunction function, void *data) { DBusHashIter iter; _dbus_hash_iter_init (registry->service_hash, &iter); while (_dbus_hash_iter_next (&iter)) { BusService *service = _dbus_hash_iter_get_value (&iter); (* function) (service, data); } } dbus_bool_t bus_registry_list_services (BusRegistry *registry, char ***listp, int *array_len) { int i, j, len; char **retval; DBusHashIter iter; len = _dbus_hash_table_get_n_entries (registry->service_hash); retval = dbus_new (char *, len + 1); if (retval == NULL) return FALSE; _dbus_hash_iter_init (registry->service_hash, &iter); i = 0; while (_dbus_hash_iter_next (&iter)) { BusService *service = _dbus_hash_iter_get_value (&iter); retval[i] = _dbus_strdup (service->name); if (retval[i] == NULL) goto error; i++; } retval[i] = NULL; if (array_len) *array_len = len; *listp = retval; return TRUE; error: for (j = 0; j < i; j++) dbus_free (retval[i]); dbus_free (retval); return FALSE; } dbus_bool_t bus_service_add_owner (BusService *service, DBusConnection *owner, BusTransaction *transaction, DBusError *error) { _DBUS_ASSERT_ERROR_IS_CLEAR (error); /* Send service acquired message first, OOM will result * in cancelling the transaction */ if (service->owners == NULL) { if (!bus_driver_send_service_acquired (owner, service->name, transaction, error)) return FALSE; } if (!_dbus_list_append (&service->owners, owner)) { BUS_SET_OOM (error); return FALSE; } if (!bus_connection_add_owned_service (owner, service)) { _dbus_list_remove_last (&service->owners, owner); BUS_SET_OOM (error); return FALSE; } return TRUE; } dbus_bool_t bus_service_remove_owner (BusService *service, DBusConnection *owner, BusTransaction *transaction, DBusError *error) { _DBUS_ASSERT_ERROR_IS_CLEAR (error); /* We send out notifications before we do any work we * might have to undo if the notification-sending failed */ /* Send service lost message */ if (bus_service_get_primary_owner (service) == owner) { if (!bus_driver_send_service_lost (owner, service->name, transaction, error)) return FALSE; } if (service->owners == NULL) { _dbus_assert_not_reached ("Tried to remove owner of a service that has no owners"); } else if (_dbus_list_length_is_one (&service->owners)) { /* We are the only owner - send service deleted */ if (!bus_driver_send_service_deleted (service->name, transaction, error)) return FALSE; } else { DBusList *link; link = _dbus_list_get_first (&service->owners); _dbus_assert (link != NULL); link = _dbus_list_get_next_link (&service->owners, link); if (link != NULL) { /* This will be our new owner */ if (!bus_driver_send_service_acquired (link->data, service->name, transaction, error)) return FALSE; } } _dbus_list_remove_last (&service->owners, owner); bus_connection_remove_owned_service (owner, service); if (service->owners == NULL) { /* Delete service (already sent message that it was deleted above) */ _dbus_hash_table_remove_string (service->registry->service_hash, service->name); dbus_free (service->name); _dbus_mem_pool_dealloc (service->registry->service_pool, service); } return TRUE; } DBusConnection* bus_service_get_primary_owner (BusService *service) { return _dbus_list_get_first (&service->owners); } const char* bus_service_get_name (BusService *service) { return service->name; } void bus_service_set_prohibit_replacement (BusService *service, dbus_bool_t prohibit_replacement) { service->prohibit_replacement = prohibit_replacement != FALSE; } dbus_bool_t bus_service_get_prohibit_replacement (BusService *service) { return service->prohibit_replacement; } dbus_bool_t bus_service_has_owner (BusService *service, DBusConnection *owner) { DBusList *link; link = _dbus_list_get_first_link (&service->owners); while (link != NULL) { if (link->data == owner) return TRUE; link = _dbus_list_get_next_link (&service->owners, link); } return FALSE; }