1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/ofdbus/dbus/bus/services.c Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1313 @@
1.4 +/* -*- mode: C; c-file-style: "gnu" -*- */
1.5 +/* services.c Service management
1.6 + *
1.7 + * Copyright (C) 2003 Red Hat, Inc.
1.8 + * Copyright (C) 2003 CodeFactory AB
1.9 + * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
1.10 + * Licensed under the Academic Free License version 2.1
1.11 + *
1.12 + * This program is free software; you can redistribute it and/or modify
1.13 + * it under the terms of the GNU General Public License as published by
1.14 + * the Free Software Foundation; either version 2 of the License, or
1.15 + * (at your option) any later version.
1.16 + *
1.17 + * This program is distributed in the hope that it will be useful,
1.18 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
1.19 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1.20 + * GNU General Public License for more details.
1.21 + *
1.22 + * You should have received a copy of the GNU General Public License
1.23 + * along with this program; if not, write to the Free Software
1.24 + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
1.25 + *
1.26 + */
1.27 + #ifndef __SYMBIAN32__
1.28 +#include <dbus/dbus-hash.h>
1.29 +#include <dbus/dbus-list.h>
1.30 +#include <dbus/dbus-mempool.h>
1.31 +#include <dbus/dbus-marshal-validate.h>
1.32 +#else
1.33 +#include "dbus-hash.h"
1.34 +#include "dbus-list.h"
1.35 +#include "dbus-mempool.h"
1.36 +#include "dbus-marshal-validate.h"
1.37 +#endif //__SYMBIAN32__
1.38 +
1.39 +#include "driver.h"
1.40 +#include "services.h"
1.41 +#include "connection.h"
1.42 +#include "utils.h"
1.43 +#include "activation.h"
1.44 +#include "policy.h"
1.45 +#include "bus.h"
1.46 +#include "selinux.h"
1.47 +
1.48 +struct BusService
1.49 +{
1.50 + int refcount;
1.51 +
1.52 + BusRegistry *registry;
1.53 + char *name;
1.54 + DBusList *owners;
1.55 +};
1.56 +
1.57 +struct BusOwner
1.58 +{
1.59 + int refcount;
1.60 +
1.61 + BusService *service;
1.62 + DBusConnection *conn;
1.63 +
1.64 + unsigned int allow_replacement : 1;
1.65 + unsigned int do_not_queue : 1;
1.66 +};
1.67 +
1.68 +struct BusRegistry
1.69 +{
1.70 + int refcount;
1.71 +
1.72 + BusContext *context;
1.73 +
1.74 + DBusHashTable *service_hash;
1.75 + DBusMemPool *service_pool;
1.76 + DBusMemPool *owner_pool;
1.77 +
1.78 + DBusHashTable *service_sid_table;
1.79 +};
1.80 +
1.81 +BusRegistry*
1.82 +bus_registry_new (BusContext *context)
1.83 +{
1.84 + BusRegistry *registry;
1.85 +
1.86 + registry = dbus_new0 (BusRegistry, 1);
1.87 + if (registry == NULL)
1.88 + return NULL;
1.89 +
1.90 + registry->refcount = 1;
1.91 + registry->context = context;
1.92 +
1.93 + registry->service_hash = _dbus_hash_table_new (DBUS_HASH_STRING,
1.94 + NULL, NULL);
1.95 + if (registry->service_hash == NULL)
1.96 + goto failed;
1.97 +
1.98 + registry->service_pool = _dbus_mem_pool_new (sizeof (BusService),
1.99 + TRUE);
1.100 +
1.101 + if (registry->service_pool == NULL)
1.102 + goto failed;
1.103 +
1.104 + registry->owner_pool = _dbus_mem_pool_new (sizeof (BusOwner),
1.105 + TRUE);
1.106 +
1.107 + if (registry->owner_pool == NULL)
1.108 + goto failed;
1.109 +
1.110 + registry->service_sid_table = NULL;
1.111 +
1.112 + return registry;
1.113 +
1.114 + failed:
1.115 + bus_registry_unref (registry);
1.116 + return NULL;
1.117 +}
1.118 +
1.119 +BusRegistry *
1.120 +bus_registry_ref (BusRegistry *registry)
1.121 +{
1.122 + _dbus_assert (registry->refcount > 0);
1.123 + registry->refcount += 1;
1.124 +
1.125 + return registry;
1.126 +}
1.127 +
1.128 +void
1.129 +bus_registry_unref (BusRegistry *registry)
1.130 +{
1.131 + _dbus_assert (registry->refcount > 0);
1.132 + registry->refcount -= 1;
1.133 +
1.134 + if (registry->refcount == 0)
1.135 + {
1.136 + if (registry->service_hash)
1.137 + _dbus_hash_table_unref (registry->service_hash);
1.138 + if (registry->service_pool)
1.139 + _dbus_mem_pool_free (registry->service_pool);
1.140 + if (registry->owner_pool)
1.141 + _dbus_mem_pool_free (registry->owner_pool);
1.142 + if (registry->service_sid_table)
1.143 + _dbus_hash_table_unref (registry->service_sid_table);
1.144 +
1.145 + dbus_free (registry);
1.146 + }
1.147 +}
1.148 +
1.149 +BusService*
1.150 +bus_registry_lookup (BusRegistry *registry,
1.151 + const DBusString *service_name)
1.152 +{
1.153 + BusService *service;
1.154 +
1.155 + service = _dbus_hash_table_lookup_string (registry->service_hash,
1.156 + _dbus_string_get_const_data (service_name));
1.157 +
1.158 + return service;
1.159 +}
1.160 +
1.161 +static DBusList *
1.162 +_bus_service_find_owner_link (BusService *service,
1.163 + DBusConnection *connection)
1.164 +{
1.165 + DBusList *link;
1.166 +
1.167 + link = _dbus_list_get_first_link (&service->owners);
1.168 +
1.169 + while (link != NULL)
1.170 + {
1.171 + BusOwner *bus_owner;
1.172 +
1.173 + bus_owner = (BusOwner *) link->data;
1.174 + if (bus_owner->conn == connection)
1.175 + break;
1.176 +
1.177 + link = _dbus_list_get_next_link (&service->owners, link);
1.178 + }
1.179 +
1.180 + return link;
1.181 +}
1.182 +
1.183 +static void
1.184 +bus_owner_set_flags (BusOwner *owner,
1.185 + dbus_uint32_t flags)
1.186 +{
1.187 + owner->allow_replacement =
1.188 + (flags & DBUS_NAME_FLAG_ALLOW_REPLACEMENT) != FALSE;
1.189 +
1.190 + owner->do_not_queue =
1.191 + (flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) != FALSE;
1.192 +}
1.193 +
1.194 +static BusOwner *
1.195 +bus_owner_new (BusService *service,
1.196 + DBusConnection *conn,
1.197 + dbus_uint32_t flags)
1.198 +{
1.199 + BusOwner *result;
1.200 +
1.201 + result = _dbus_mem_pool_alloc (service->registry->owner_pool);
1.202 + if (result != NULL)
1.203 + {
1.204 + result->refcount = 1;
1.205 + /* don't ref the connection because we don't want
1.206 + to block the connection from going away.
1.207 + transactions take care of reffing the connection
1.208 + but we need to use refcounting on the owner
1.209 + so that the owner does not get freed before
1.210 + we can deref the connection in the transaction
1.211 + */
1.212 + result->conn = conn;
1.213 + result->service = service;
1.214 +
1.215 + if (!bus_connection_add_owned_service (conn, service))
1.216 + {
1.217 + _dbus_mem_pool_dealloc (service->registry->owner_pool, result);
1.218 + return NULL;
1.219 + }
1.220 +
1.221 + bus_owner_set_flags (result, flags);
1.222 + }
1.223 + return result;
1.224 +}
1.225 +
1.226 +static BusOwner *
1.227 +bus_owner_ref (BusOwner *owner)
1.228 +{
1.229 + _dbus_assert (owner->refcount > 0);
1.230 + owner->refcount += 1;
1.231 +
1.232 + return owner;
1.233 +}
1.234 +
1.235 +static void
1.236 +bus_owner_unref (BusOwner *owner)
1.237 +{
1.238 + _dbus_assert (owner->refcount > 0);
1.239 + owner->refcount -= 1;
1.240 +
1.241 + if (owner->refcount == 0)
1.242 + {
1.243 + bus_connection_remove_owned_service (owner->conn, owner->service);
1.244 + _dbus_mem_pool_dealloc (owner->service->registry->owner_pool, owner);
1.245 + }
1.246 +}
1.247 +
1.248 +BusService*
1.249 +bus_registry_ensure (BusRegistry *registry,
1.250 + const DBusString *service_name,
1.251 + DBusConnection *owner_connection_if_created,
1.252 + dbus_uint32_t flags,
1.253 + BusTransaction *transaction,
1.254 + DBusError *error)
1.255 +{
1.256 + BusService *service;
1.257 +
1.258 + _DBUS_ASSERT_ERROR_IS_CLEAR (error);
1.259 +
1.260 + _dbus_assert (owner_connection_if_created != NULL);
1.261 + _dbus_assert (transaction != NULL);
1.262 +
1.263 + service = _dbus_hash_table_lookup_string (registry->service_hash,
1.264 + _dbus_string_get_const_data (service_name));
1.265 + if (service != NULL)
1.266 + return service;
1.267 +
1.268 + service = _dbus_mem_pool_alloc (registry->service_pool);
1.269 + if (service == NULL)
1.270 + {
1.271 + BUS_SET_OOM (error);
1.272 + return NULL;
1.273 + }
1.274 +
1.275 + service->registry = registry;
1.276 + service->refcount = 1;
1.277 +
1.278 + _dbus_verbose ("copying string %p '%s' to service->name\n",
1.279 + service_name, _dbus_string_get_const_data (service_name));
1.280 + if (!_dbus_string_copy_data (service_name, &service->name))
1.281 + {
1.282 + _dbus_mem_pool_dealloc (registry->service_pool, service);
1.283 + BUS_SET_OOM (error);
1.284 + return NULL;
1.285 + }
1.286 + _dbus_verbose ("copied string %p '%s' to '%s'\n",
1.287 + service_name, _dbus_string_get_const_data (service_name),
1.288 + service->name);
1.289 +
1.290 + if (!bus_driver_send_service_owner_changed (service->name,
1.291 + NULL,
1.292 + bus_connection_get_name (owner_connection_if_created),
1.293 + transaction, error))
1.294 + {
1.295 + bus_service_unref (service);
1.296 + return NULL;
1.297 + }
1.298 +
1.299 + if (!bus_activation_service_created (bus_context_get_activation (registry->context),
1.300 + service->name, transaction, error))
1.301 + {
1.302 + bus_service_unref (service);
1.303 + return NULL;
1.304 + }
1.305 +
1.306 + if (!bus_service_add_owner (service, owner_connection_if_created, flags,
1.307 + transaction, error))
1.308 + {
1.309 + bus_service_unref (service);
1.310 + return NULL;
1.311 + }
1.312 +
1.313 + if (!_dbus_hash_table_insert_string (registry->service_hash,
1.314 + service->name,
1.315 + service))
1.316 + {
1.317 + /* The add_owner gets reverted on transaction cancel */
1.318 + BUS_SET_OOM (error);
1.319 + return NULL;
1.320 + }
1.321 +
1.322 + return service;
1.323 +}
1.324 +
1.325 +void
1.326 +bus_registry_foreach (BusRegistry *registry,
1.327 + BusServiceForeachFunction function,
1.328 + void *data)
1.329 +{
1.330 + DBusHashIter iter;
1.331 +
1.332 + _dbus_hash_iter_init (registry->service_hash, &iter);
1.333 + while (_dbus_hash_iter_next (&iter))
1.334 + {
1.335 + BusService *service = _dbus_hash_iter_get_value (&iter);
1.336 +
1.337 + (* function) (service, data);
1.338 + }
1.339 +}
1.340 +
1.341 +dbus_bool_t
1.342 +bus_registry_list_services (BusRegistry *registry,
1.343 + char ***listp,
1.344 + int *array_len)
1.345 +{
1.346 + int i, j, len;
1.347 + char **retval;
1.348 + DBusHashIter iter;
1.349 +
1.350 + len = _dbus_hash_table_get_n_entries (registry->service_hash);
1.351 + retval = dbus_new (char *, len + 1);
1.352 +
1.353 + if (retval == NULL)
1.354 + return FALSE;
1.355 +
1.356 + _dbus_hash_iter_init (registry->service_hash, &iter);
1.357 + i = 0;
1.358 + while (_dbus_hash_iter_next (&iter))
1.359 + {
1.360 + BusService *service = _dbus_hash_iter_get_value (&iter);
1.361 +
1.362 + retval[i] = _dbus_strdup (service->name);
1.363 + if (retval[i] == NULL)
1.364 + goto error;
1.365 +
1.366 + i++;
1.367 + }
1.368 +
1.369 + retval[i] = NULL;
1.370 +
1.371 + if (array_len)
1.372 + *array_len = len;
1.373 +
1.374 + *listp = retval;
1.375 + return TRUE;
1.376 +
1.377 + error:
1.378 + for (j = 0; j < i; j++)
1.379 + dbus_free (retval[i]);
1.380 + dbus_free (retval);
1.381 +
1.382 + return FALSE;
1.383 +}
1.384 +
1.385 +dbus_bool_t
1.386 +bus_registry_acquire_service (BusRegistry *registry,
1.387 + DBusConnection *connection,
1.388 + const DBusString *service_name,
1.389 + dbus_uint32_t flags,
1.390 + dbus_uint32_t *result,
1.391 + BusTransaction *transaction,
1.392 + DBusError *error)
1.393 +{
1.394 + dbus_bool_t retval;
1.395 + DBusConnection *old_owner_conn;
1.396 + DBusConnection *current_owner_conn;
1.397 + BusClientPolicy *policy;
1.398 + BusService *service;
1.399 + BusActivation *activation;
1.400 + BusSELinuxID *sid;
1.401 + BusOwner *primary_owner;
1.402 +
1.403 + retval = FALSE;
1.404 +
1.405 + if (!_dbus_validate_bus_name (service_name, 0,
1.406 + _dbus_string_get_length (service_name)))
1.407 + {
1.408 + dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
1.409 + "Requested bus name \"%s\" is not valid",
1.410 + _dbus_string_get_const_data (service_name));
1.411 +
1.412 + _dbus_verbose ("Attempt to acquire invalid service name\n");
1.413 +
1.414 + goto out;
1.415 + }
1.416 +
1.417 + if (_dbus_string_get_byte (service_name, 0) == ':')
1.418 + {
1.419 + /* Not allowed; only base services can start with ':' */
1.420 + dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
1.421 + "Cannot acquire a service starting with ':' such as \"%s\"",
1.422 + _dbus_string_get_const_data (service_name));
1.423 +
1.424 + _dbus_verbose ("Attempt to acquire invalid base service name \"%s\"",
1.425 + _dbus_string_get_const_data (service_name));
1.426 +
1.427 + goto out;
1.428 + }
1.429 +
1.430 + if (_dbus_string_equal_c_str (service_name, DBUS_SERVICE_DBUS))
1.431 + {
1.432 + dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
1.433 + "Connection \"%s\" is not allowed to own the service \"%s\"because "
1.434 + "it is reserved for D-Bus' use only",
1.435 + bus_connection_is_active (connection) ?
1.436 + bus_connection_get_name (connection) :
1.437 + "(inactive)",
1.438 + DBUS_SERVICE_DBUS);
1.439 + goto out;
1.440 + }
1.441 +
1.442 + policy = bus_connection_get_policy (connection);
1.443 + _dbus_assert (policy != NULL);
1.444 +
1.445 + /* Note that if sid is #NULL then the bus's own context gets used
1.446 + * in bus_connection_selinux_allows_acquire_service()
1.447 + */
1.448 + sid = bus_selinux_id_table_lookup (registry->service_sid_table,
1.449 + service_name);
1.450 +
1.451 + if (!bus_selinux_allows_acquire_service (connection, sid,
1.452 + _dbus_string_get_const_data (service_name), error))
1.453 + {
1.454 +
1.455 + if (dbus_error_is_set (error) &&
1.456 + dbus_error_has_name (error, DBUS_ERROR_NO_MEMORY))
1.457 + {
1.458 + goto out;
1.459 + }
1.460 +
1.461 + dbus_set_error (error, DBUS_ERROR_ACCESS_DENIED,
1.462 + "Connection \"%s\" is not allowed to own the service \"%s\" due "
1.463 + "to SELinux policy",
1.464 + bus_connection_is_active (connection) ?
1.465 + bus_connection_get_name (connection) :
1.466 + "(inactive)",
1.467 + _dbus_string_get_const_data (service_name));
1.468 + goto out;
1.469 + }
1.470 +
1.471 + if (!bus_client_policy_check_can_own (policy, connection,
1.472 + service_name))
1.473 + {
1.474 + dbus_set_error (error, DBUS_ERROR_ACCESS_DENIED,
1.475 + "Connection \"%s\" is not allowed to own the service \"%s\" due "
1.476 + "to security policies in the configuration file",
1.477 + bus_connection_is_active (connection) ?
1.478 + bus_connection_get_name (connection) :
1.479 + "(inactive)",
1.480 + _dbus_string_get_const_data (service_name));
1.481 + goto out;
1.482 + }
1.483 +
1.484 + if (bus_connection_get_n_services_owned (connection) >=
1.485 + bus_context_get_max_services_per_connection (registry->context))
1.486 + {
1.487 + dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
1.488 + "Connection \"%s\" is not allowed to own more services "
1.489 + "(increase limits in configuration file if required)",
1.490 + bus_connection_is_active (connection) ?
1.491 + bus_connection_get_name (connection) :
1.492 + "(inactive)");
1.493 + goto out;
1.494 + }
1.495 +
1.496 + service = bus_registry_lookup (registry, service_name);
1.497 +
1.498 + if (service != NULL)
1.499 + {
1.500 + primary_owner = bus_service_get_primary_owner (service);
1.501 + if (primary_owner != NULL)
1.502 + old_owner_conn = primary_owner->conn;
1.503 + else
1.504 + old_owner_conn = NULL;
1.505 + }
1.506 + else
1.507 + old_owner_conn = NULL;
1.508 +
1.509 + if (service == NULL)
1.510 + {
1.511 + service = bus_registry_ensure (registry,
1.512 + service_name, connection, flags,
1.513 + transaction, error);
1.514 + if (service == NULL)
1.515 + goto out;
1.516 + }
1.517 +
1.518 + primary_owner = bus_service_get_primary_owner (service);
1.519 + if (primary_owner == NULL)
1.520 + goto out;
1.521 +
1.522 + current_owner_conn = primary_owner->conn;
1.523 +
1.524 + if (old_owner_conn == NULL)
1.525 + {
1.526 + _dbus_assert (current_owner_conn == connection);
1.527 +
1.528 + *result = DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER;
1.529 + }
1.530 + else if (old_owner_conn == connection)
1.531 + {
1.532 + bus_owner_set_flags (primary_owner, flags);
1.533 + *result = DBUS_REQUEST_NAME_REPLY_ALREADY_OWNER;
1.534 + }
1.535 + else if (((flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
1.536 + !(bus_service_get_allow_replacement (service))) ||
1.537 + ((flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
1.538 + !(flags & DBUS_NAME_FLAG_REPLACE_EXISTING)))
1.539 + {
1.540 + DBusList *link;
1.541 + BusOwner *temp_owner;
1.542 + /* Since we can't be queued if we are already in the queue
1.543 + remove us */
1.544 +
1.545 + link = _bus_service_find_owner_link (service, connection);
1.546 + if (link != NULL)
1.547 + {
1.548 + _dbus_list_unlink (&service->owners, link);
1.549 + temp_owner = (BusOwner *)link->data;
1.550 + bus_owner_unref (temp_owner);
1.551 + _dbus_list_free_link (link);
1.552 + }
1.553 +
1.554 + *result = DBUS_REQUEST_NAME_REPLY_EXISTS;
1.555 + }
1.556 + else if (!(flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
1.557 + (!(flags & DBUS_NAME_FLAG_REPLACE_EXISTING) ||
1.558 + !(bus_service_get_allow_replacement (service))))
1.559 + {
1.560 + /* Queue the connection */
1.561 + if (!bus_service_add_owner (service, connection,
1.562 + flags,
1.563 + transaction, error))
1.564 + goto out;
1.565 +
1.566 + *result = DBUS_REQUEST_NAME_REPLY_IN_QUEUE;
1.567 + }
1.568 + else
1.569 + {
1.570 + /* Replace the current owner */
1.571 +
1.572 + /* We enqueue the new owner and remove the first one because
1.573 + * that will cause NameAcquired and NameLost messages to
1.574 + * be sent.
1.575 + */
1.576 +
1.577 + if (!bus_service_add_owner (service, connection,
1.578 + flags,
1.579 + transaction, error))
1.580 + goto out;
1.581 +
1.582 + if (primary_owner->do_not_queue)
1.583 + {
1.584 + if (!bus_service_remove_owner (service, old_owner_conn,
1.585 + transaction, error))
1.586 + goto out;
1.587 + }
1.588 + else
1.589 + {
1.590 + if (!bus_service_swap_owner (service, old_owner_conn,
1.591 + transaction, error))
1.592 + goto out;
1.593 + }
1.594 +
1.595 +
1.596 + _dbus_assert (connection == bus_service_get_primary_owner (service)->conn);
1.597 + *result = DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER;
1.598 + }
1.599 +
1.600 + activation = bus_context_get_activation (registry->context);
1.601 + retval = bus_activation_send_pending_auto_activation_messages (activation,
1.602 + service,
1.603 + transaction,
1.604 + error);
1.605 +
1.606 + out:
1.607 + return retval;
1.608 +}
1.609 +
1.610 +dbus_bool_t
1.611 +bus_registry_release_service (BusRegistry *registry,
1.612 + DBusConnection *connection,
1.613 + const DBusString *service_name,
1.614 + dbus_uint32_t *result,
1.615 + BusTransaction *transaction,
1.616 + DBusError *error)
1.617 +{
1.618 + dbus_bool_t retval;
1.619 + BusService *service;
1.620 +
1.621 + retval = FALSE;
1.622 +
1.623 + if (!_dbus_validate_bus_name (service_name, 0,
1.624 + _dbus_string_get_length (service_name)))
1.625 + {
1.626 + dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
1.627 + "Given bus name \"%s\" is not valid",
1.628 + _dbus_string_get_const_data (service_name));
1.629 +
1.630 + _dbus_verbose ("Attempt to release invalid service name\n");
1.631 +
1.632 + goto out;
1.633 + }
1.634 +
1.635 + if (_dbus_string_get_byte (service_name, 0) == ':')
1.636 + {
1.637 + /* Not allowed; the base service name cannot be created or released */
1.638 + dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
1.639 + "Cannot release a service starting with ':' such as \"%s\"",
1.640 + _dbus_string_get_const_data (service_name));
1.641 +
1.642 + _dbus_verbose ("Attempt to release invalid base service name \"%s\"",
1.643 + _dbus_string_get_const_data (service_name));
1.644 +
1.645 + goto out;
1.646 + }
1.647 +
1.648 + if (_dbus_string_equal_c_str (service_name, DBUS_SERVICE_DBUS))
1.649 + {
1.650 + /* Not allowed; the base service name cannot be created or released */
1.651 + dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
1.652 + "Cannot release the %s service because it is owned by the bus",
1.653 + DBUS_SERVICE_DBUS);
1.654 +
1.655 + _dbus_verbose ("Attempt to release service name \"%s\"",
1.656 + DBUS_SERVICE_DBUS);
1.657 +
1.658 + goto out;
1.659 + }
1.660 +
1.661 + service = bus_registry_lookup (registry, service_name);
1.662 +
1.663 + if (service == NULL)
1.664 + {
1.665 + *result = DBUS_RELEASE_NAME_REPLY_NON_EXISTENT;
1.666 + }
1.667 + else if (!bus_service_has_owner (service, connection))
1.668 + {
1.669 + *result = DBUS_RELEASE_NAME_REPLY_NOT_OWNER;
1.670 + }
1.671 + else
1.672 + {
1.673 + if (!bus_service_remove_owner (service, connection,
1.674 + transaction, error))
1.675 + goto out;
1.676 +
1.677 + _dbus_assert (!bus_service_has_owner (service, connection));
1.678 + *result = DBUS_RELEASE_NAME_REPLY_RELEASED;
1.679 + }
1.680 +
1.681 + retval = TRUE;
1.682 +
1.683 + out:
1.684 + return retval;
1.685 +}
1.686 +
1.687 +dbus_bool_t
1.688 +bus_registry_set_service_context_table (BusRegistry *registry,
1.689 + DBusHashTable *table)
1.690 +{
1.691 + DBusHashTable *new_table;
1.692 + DBusHashIter iter;
1.693 +
1.694 + new_table = bus_selinux_id_table_new ();
1.695 + if (!new_table)
1.696 + return FALSE;
1.697 +
1.698 + _dbus_hash_iter_init (table, &iter);
1.699 + while (_dbus_hash_iter_next (&iter))
1.700 + {
1.701 + const char *service = _dbus_hash_iter_get_string_key (&iter);
1.702 + const char *context = _dbus_hash_iter_get_value (&iter);
1.703 +
1.704 + if (!bus_selinux_id_table_insert (new_table,
1.705 + service,
1.706 + context))
1.707 + return FALSE;
1.708 + }
1.709 +
1.710 + if (registry->service_sid_table)
1.711 + _dbus_hash_table_unref (registry->service_sid_table);
1.712 + registry->service_sid_table = new_table;
1.713 + return TRUE;
1.714 +}
1.715 +
1.716 +static void
1.717 +bus_service_unlink_owner (BusService *service,
1.718 + BusOwner *owner)
1.719 +{
1.720 + _dbus_list_remove_last (&service->owners, owner);
1.721 + bus_owner_unref (owner);
1.722 +}
1.723 +
1.724 +static void
1.725 +bus_service_unlink (BusService *service)
1.726 +{
1.727 + _dbus_assert (service->owners == NULL);
1.728 +
1.729 + /* the service may not be in the hash, if
1.730 + * the failure causing transaction cancel
1.731 + * was in the right place, but that's OK
1.732 + */
1.733 + _dbus_hash_table_remove_string (service->registry->service_hash,
1.734 + service->name);
1.735 +
1.736 + bus_service_unref (service);
1.737 +}
1.738 +
1.739 +static void
1.740 +bus_service_relink (BusService *service,
1.741 + DBusPreallocatedHash *preallocated)
1.742 +{
1.743 + _dbus_assert (service->owners == NULL);
1.744 + _dbus_assert (preallocated != NULL);
1.745 +
1.746 + _dbus_hash_table_insert_string_preallocated (service->registry->service_hash,
1.747 + preallocated,
1.748 + service->name,
1.749 + service);
1.750 +
1.751 + bus_service_ref (service);
1.752 +}
1.753 +
1.754 +/**
1.755 + * Data used to represent an ownership cancellation in
1.756 + * a bus transaction.
1.757 + */
1.758 +typedef struct
1.759 +{
1.760 + BusOwner *owner; /**< the owner */
1.761 + BusService *service; /**< service to cancel ownership of */
1.762 +} OwnershipCancelData;
1.763 +
1.764 +static void
1.765 +cancel_ownership (void *data)
1.766 +{
1.767 + OwnershipCancelData *d = data;
1.768 +
1.769 + /* We don't need to send messages notifying of these
1.770 + * changes, since we're reverting something that was
1.771 + * cancelled (effectively never really happened)
1.772 + */
1.773 + bus_service_unlink_owner (d->service, d->owner);
1.774 +
1.775 + if (d->service->owners == NULL)
1.776 + bus_service_unlink (d->service);
1.777 +}
1.778 +
1.779 +static void
1.780 +free_ownership_cancel_data (void *data)
1.781 +{
1.782 + OwnershipCancelData *d = data;
1.783 +
1.784 + dbus_connection_unref (d->owner->conn);
1.785 + bus_owner_unref (d->owner);
1.786 + bus_service_unref (d->service);
1.787 +
1.788 + dbus_free (d);
1.789 +}
1.790 +
1.791 +static dbus_bool_t
1.792 +add_cancel_ownership_to_transaction (BusTransaction *transaction,
1.793 + BusService *service,
1.794 + BusOwner *owner)
1.795 +{
1.796 + OwnershipCancelData *d;
1.797 +
1.798 + d = dbus_new (OwnershipCancelData, 1);
1.799 + if (d == NULL)
1.800 + return FALSE;
1.801 +
1.802 + d->service = service;
1.803 + d->owner = owner;
1.804 +
1.805 + if (!bus_transaction_add_cancel_hook (transaction, cancel_ownership, d,
1.806 + free_ownership_cancel_data))
1.807 + {
1.808 + dbus_free (d);
1.809 + return FALSE;
1.810 + }
1.811 +
1.812 + bus_service_ref (d->service);
1.813 + bus_owner_ref (owner);
1.814 + dbus_connection_ref (d->owner->conn);
1.815 +
1.816 + return TRUE;
1.817 +}
1.818 +
1.819 +/* this function is self-cancelling if you cancel the transaction */
1.820 +dbus_bool_t
1.821 +bus_service_add_owner (BusService *service,
1.822 + DBusConnection *connection,
1.823 + dbus_uint32_t flags,
1.824 + BusTransaction *transaction,
1.825 + DBusError *error)
1.826 +{
1.827 + BusOwner *bus_owner;
1.828 + DBusList *bus_owner_link;
1.829 +
1.830 + _DBUS_ASSERT_ERROR_IS_CLEAR (error);
1.831 +
1.832 + /* Send service acquired message first, OOM will result
1.833 + * in cancelling the transaction
1.834 + */
1.835 + if (service->owners == NULL)
1.836 + {
1.837 + if (!bus_driver_send_service_acquired (connection, service->name, transaction, error))
1.838 + return FALSE;
1.839 + }
1.840 +
1.841 + bus_owner_link = _bus_service_find_owner_link (service, connection);
1.842 +
1.843 + if (bus_owner_link == NULL)
1.844 + {
1.845 + bus_owner = bus_owner_new (service, connection, flags);
1.846 + if (bus_owner == NULL)
1.847 + {
1.848 + BUS_SET_OOM (error);
1.849 + return FALSE;
1.850 + }
1.851 +
1.852 + bus_owner_set_flags (bus_owner, flags);
1.853 + if (!(flags & DBUS_NAME_FLAG_REPLACE_EXISTING) || service->owners == NULL)
1.854 + {
1.855 + if (!_dbus_list_append (&service->owners,
1.856 + bus_owner))
1.857 + {
1.858 + bus_owner_unref (bus_owner);
1.859 + BUS_SET_OOM (error);
1.860 + return FALSE;
1.861 + }
1.862 + }
1.863 + else
1.864 + {
1.865 + if (!_dbus_list_insert_after (&service->owners,
1.866 + _dbus_list_get_first_link (&service->owners),
1.867 + bus_owner))
1.868 + {
1.869 + bus_owner_unref (bus_owner);
1.870 + BUS_SET_OOM (error);
1.871 + return FALSE;
1.872 + }
1.873 + }
1.874 + }
1.875 + else
1.876 + {
1.877 + /* Update the link since we are already in the queue
1.878 + * No need for operations that can produce OOM
1.879 + */
1.880 +
1.881 + bus_owner = (BusOwner *) bus_owner_link->data;
1.882 + if (flags & DBUS_NAME_FLAG_REPLACE_EXISTING)
1.883 + {
1.884 + DBusList *link;
1.885 + _dbus_list_unlink (&service->owners, bus_owner_link);
1.886 + link = _dbus_list_get_first_link (&service->owners);
1.887 + _dbus_assert (link != NULL);
1.888 +
1.889 + _dbus_list_insert_after_link (&service->owners, link, bus_owner_link);
1.890 + }
1.891 +
1.892 + bus_owner_set_flags (bus_owner, flags);
1.893 + return TRUE;
1.894 + }
1.895 +
1.896 + if (!add_cancel_ownership_to_transaction (transaction,
1.897 + service,
1.898 + bus_owner))
1.899 + {
1.900 + bus_service_unlink_owner (service, bus_owner);
1.901 + BUS_SET_OOM (error);
1.902 + return FALSE;
1.903 + }
1.904 +
1.905 + return TRUE;
1.906 +}
1.907 +
1.908 +typedef struct
1.909 +{
1.910 + BusOwner *owner;
1.911 + BusService *service;
1.912 + BusOwner *before_owner; /* restore to position before this connection in owners list */
1.913 + DBusList *owner_link;
1.914 + DBusList *service_link;
1.915 + DBusPreallocatedHash *hash_entry;
1.916 +} OwnershipRestoreData;
1.917 +
1.918 +static void
1.919 +restore_ownership (void *data)
1.920 +{
1.921 + OwnershipRestoreData *d = data;
1.922 + DBusList *link;
1.923 +
1.924 + _dbus_assert (d->service_link != NULL);
1.925 + _dbus_assert (d->owner_link != NULL);
1.926 +
1.927 + if (d->service->owners == NULL)
1.928 + {
1.929 + _dbus_assert (d->hash_entry != NULL);
1.930 + bus_service_relink (d->service, d->hash_entry);
1.931 + }
1.932 + else
1.933 + {
1.934 + _dbus_assert (d->hash_entry == NULL);
1.935 + }
1.936 +
1.937 + /* We don't need to send messages notifying of these
1.938 + * changes, since we're reverting something that was
1.939 + * cancelled (effectively never really happened)
1.940 + */
1.941 + link = _dbus_list_get_first_link (&d->service->owners);
1.942 + while (link != NULL)
1.943 + {
1.944 + if (link->data == d->before_owner)
1.945 + break;
1.946 +
1.947 + link = _dbus_list_get_next_link (&d->service->owners, link);
1.948 + }
1.949 +
1.950 + _dbus_list_insert_before_link (&d->service->owners, link, d->owner_link);
1.951 +
1.952 + /* Note that removing then restoring this changes the order in which
1.953 + * ServiceDeleted messages are sent on destruction of the
1.954 + * connection. This should be OK as the only guarantee there is
1.955 + * that the base service is destroyed last, and we never even
1.956 + * tentatively remove the base service.
1.957 + */
1.958 + bus_connection_add_owned_service_link (d->owner->conn, d->service_link);
1.959 +
1.960 + d->hash_entry = NULL;
1.961 + d->service_link = NULL;
1.962 + d->owner_link = NULL;
1.963 +}
1.964 +
1.965 +static void
1.966 +free_ownership_restore_data (void *data)
1.967 +{
1.968 + OwnershipRestoreData *d = data;
1.969 +
1.970 + if (d->service_link)
1.971 + _dbus_list_free_link (d->service_link);
1.972 + if (d->owner_link)
1.973 + _dbus_list_free_link (d->owner_link);
1.974 + if (d->hash_entry)
1.975 + _dbus_hash_table_free_preallocated_entry (d->service->registry->service_hash,
1.976 + d->hash_entry);
1.977 +
1.978 + dbus_connection_unref (d->owner->conn);
1.979 + bus_owner_unref (d->owner);
1.980 + bus_service_unref (d->service);
1.981 +
1.982 + dbus_free (d);
1.983 +}
1.984 +
1.985 +static dbus_bool_t
1.986 +add_restore_ownership_to_transaction (BusTransaction *transaction,
1.987 + BusService *service,
1.988 + BusOwner *owner)
1.989 +{
1.990 + OwnershipRestoreData *d;
1.991 + DBusList *link;
1.992 +
1.993 + d = dbus_new (OwnershipRestoreData, 1);
1.994 + if (d == NULL)
1.995 + return FALSE;
1.996 +
1.997 + d->service = service;
1.998 + d->owner = owner;
1.999 + d->service_link = _dbus_list_alloc_link (service);
1.1000 + d->owner_link = _dbus_list_alloc_link (owner);
1.1001 + d->hash_entry = _dbus_hash_table_preallocate_entry (service->registry->service_hash);
1.1002 +
1.1003 + bus_service_ref (d->service);
1.1004 + bus_owner_ref (d->owner);
1.1005 + dbus_connection_ref (d->owner->conn);
1.1006 +
1.1007 + d->before_owner = NULL;
1.1008 + link = _dbus_list_get_first_link (&service->owners);
1.1009 + while (link != NULL)
1.1010 + {
1.1011 + if (link->data == owner)
1.1012 + {
1.1013 + link = _dbus_list_get_next_link (&service->owners, link);
1.1014 +
1.1015 + if (link)
1.1016 + d->before_owner = link->data;
1.1017 +
1.1018 + break;
1.1019 + }
1.1020 +
1.1021 + link = _dbus_list_get_next_link (&service->owners, link);
1.1022 + }
1.1023 +
1.1024 + if (d->service_link == NULL ||
1.1025 + d->owner_link == NULL ||
1.1026 + d->hash_entry == NULL ||
1.1027 + !bus_transaction_add_cancel_hook (transaction, restore_ownership, d,
1.1028 + free_ownership_restore_data))
1.1029 + {
1.1030 + free_ownership_restore_data (d);
1.1031 + return FALSE;
1.1032 + }
1.1033 +
1.1034 + return TRUE;
1.1035 +}
1.1036 +
1.1037 +dbus_bool_t
1.1038 +bus_service_swap_owner (BusService *service,
1.1039 + DBusConnection *connection,
1.1040 + BusTransaction *transaction,
1.1041 + DBusError *error)
1.1042 +{
1.1043 + DBusList *swap_link;
1.1044 + BusOwner *primary_owner;
1.1045 +
1.1046 + _DBUS_ASSERT_ERROR_IS_CLEAR (error);
1.1047 +
1.1048 + /* We send out notifications before we do any work we
1.1049 + * might have to undo if the notification-sending failed
1.1050 + */
1.1051 +
1.1052 + /* Send service lost message */
1.1053 + primary_owner = bus_service_get_primary_owner (service);
1.1054 + if (primary_owner == NULL || primary_owner->conn != connection)
1.1055 + _dbus_assert_not_reached ("Tried to swap a non primary owner");
1.1056 +
1.1057 +
1.1058 + if (!bus_driver_send_service_lost (connection, service->name,
1.1059 + transaction, error))
1.1060 + return FALSE;
1.1061 +
1.1062 + if (service->owners == NULL)
1.1063 + {
1.1064 + _dbus_assert_not_reached ("Tried to swap owner of a service that has no owners");
1.1065 + }
1.1066 + else if (_dbus_list_length_is_one (&service->owners))
1.1067 + {
1.1068 + _dbus_assert_not_reached ("Tried to swap owner of a service that has no other owners in the queue");
1.1069 + }
1.1070 + else
1.1071 + {
1.1072 + DBusList *link;
1.1073 + BusOwner *new_owner;
1.1074 + DBusConnection *new_owner_conn;
1.1075 + link = _dbus_list_get_first_link (&service->owners);
1.1076 + _dbus_assert (link != NULL);
1.1077 + link = _dbus_list_get_next_link (&service->owners, link);
1.1078 + _dbus_assert (link != NULL);
1.1079 +
1.1080 + new_owner = (BusOwner *)link->data;
1.1081 + new_owner_conn = new_owner->conn;
1.1082 +
1.1083 + if (!bus_driver_send_service_owner_changed (service->name,
1.1084 + bus_connection_get_name (connection),
1.1085 + bus_connection_get_name (new_owner_conn),
1.1086 + transaction, error))
1.1087 + return FALSE;
1.1088 +
1.1089 + /* This will be our new owner */
1.1090 + if (!bus_driver_send_service_acquired (new_owner_conn,
1.1091 + service->name,
1.1092 + transaction,
1.1093 + error))
1.1094 + return FALSE;
1.1095 + }
1.1096 +
1.1097 + if (!add_restore_ownership_to_transaction (transaction, service, primary_owner))
1.1098 + {
1.1099 + BUS_SET_OOM (error);
1.1100 + return FALSE;
1.1101 + }
1.1102 +
1.1103 + /* unlink the primary and make it the second link */
1.1104 + swap_link = _dbus_list_get_first_link (&service->owners);
1.1105 + _dbus_list_unlink (&service->owners, swap_link);
1.1106 +
1.1107 + _dbus_list_insert_after_link (&service->owners,
1.1108 + _dbus_list_get_first_link (&service->owners),
1.1109 + swap_link);
1.1110 +
1.1111 + return TRUE;
1.1112 +}
1.1113 +
1.1114 +/* this function is self-cancelling if you cancel the transaction */
1.1115 +dbus_bool_t
1.1116 +bus_service_remove_owner (BusService *service,
1.1117 + DBusConnection *connection,
1.1118 + BusTransaction *transaction,
1.1119 + DBusError *error)
1.1120 +{
1.1121 + BusOwner *primary_owner;
1.1122 +
1.1123 + _DBUS_ASSERT_ERROR_IS_CLEAR (error);
1.1124 +
1.1125 + /* We send out notifications before we do any work we
1.1126 + * might have to undo if the notification-sending failed
1.1127 + */
1.1128 +
1.1129 + /* Send service lost message */
1.1130 + primary_owner = bus_service_get_primary_owner (service);
1.1131 + if (primary_owner != NULL && primary_owner->conn == connection)
1.1132 + {
1.1133 + if (!bus_driver_send_service_lost (connection, service->name,
1.1134 + transaction, error))
1.1135 + return FALSE;
1.1136 + }
1.1137 + else
1.1138 + {
1.1139 + /* if we are not the primary owner then just remove us from the queue */
1.1140 + DBusList *link;
1.1141 + BusOwner *temp_owner;
1.1142 +
1.1143 + link = _bus_service_find_owner_link (service, connection);
1.1144 + _dbus_list_unlink (&service->owners, link);
1.1145 + temp_owner = (BusOwner *)link->data;
1.1146 + bus_owner_unref (temp_owner);
1.1147 + _dbus_list_free_link (link);
1.1148 +
1.1149 + return TRUE;
1.1150 + }
1.1151 +
1.1152 + if (service->owners == NULL)
1.1153 + {
1.1154 + _dbus_assert_not_reached ("Tried to remove owner of a service that has no owners");
1.1155 + }
1.1156 + else if (_dbus_list_length_is_one (&service->owners))
1.1157 + {
1.1158 + if (!bus_driver_send_service_owner_changed (service->name,
1.1159 + bus_connection_get_name (connection),
1.1160 + NULL,
1.1161 + transaction, error))
1.1162 + return FALSE;
1.1163 + }
1.1164 + else
1.1165 + {
1.1166 + DBusList *link;
1.1167 + BusOwner *new_owner;
1.1168 + DBusConnection *new_owner_conn;
1.1169 + link = _dbus_list_get_first_link (&service->owners);
1.1170 + _dbus_assert (link != NULL);
1.1171 + link = _dbus_list_get_next_link (&service->owners, link);
1.1172 + _dbus_assert (link != NULL);
1.1173 +
1.1174 + new_owner = (BusOwner *)link->data;
1.1175 + new_owner_conn = new_owner->conn;
1.1176 +
1.1177 + if (!bus_driver_send_service_owner_changed (service->name,
1.1178 + bus_connection_get_name (connection),
1.1179 + bus_connection_get_name (new_owner_conn),
1.1180 + transaction, error))
1.1181 + return FALSE;
1.1182 +
1.1183 + /* This will be our new owner */
1.1184 + if (!bus_driver_send_service_acquired (new_owner_conn,
1.1185 + service->name,
1.1186 + transaction,
1.1187 + error))
1.1188 + return FALSE;
1.1189 + }
1.1190 +
1.1191 + if (!add_restore_ownership_to_transaction (transaction, service, primary_owner))
1.1192 + {
1.1193 + BUS_SET_OOM (error);
1.1194 + return FALSE;
1.1195 + }
1.1196 +
1.1197 + bus_service_unlink_owner (service, primary_owner);
1.1198 +
1.1199 + if (service->owners == NULL)
1.1200 + bus_service_unlink (service);
1.1201 +
1.1202 + return TRUE;
1.1203 +}
1.1204 +
1.1205 +BusService *
1.1206 +bus_service_ref (BusService *service)
1.1207 +{
1.1208 + _dbus_assert (service->refcount > 0);
1.1209 +
1.1210 + service->refcount += 1;
1.1211 +
1.1212 + return service;
1.1213 +}
1.1214 +
1.1215 +void
1.1216 +bus_service_unref (BusService *service)
1.1217 +{
1.1218 + _dbus_assert (service->refcount > 0);
1.1219 +
1.1220 + service->refcount -= 1;
1.1221 +
1.1222 + if (service->refcount == 0)
1.1223 + {
1.1224 + _dbus_assert (service->owners == NULL);
1.1225 +
1.1226 + dbus_free (service->name);
1.1227 + _dbus_mem_pool_dealloc (service->registry->service_pool, service);
1.1228 + }
1.1229 +}
1.1230 +
1.1231 +DBusConnection *
1.1232 +bus_service_get_primary_owners_connection (BusService *service)
1.1233 +{
1.1234 + BusOwner *owner;
1.1235 +
1.1236 + owner = bus_service_get_primary_owner (service);
1.1237 +
1.1238 + if (owner != NULL)
1.1239 + return owner->conn;
1.1240 + else
1.1241 + return NULL;
1.1242 +}
1.1243 +
1.1244 +BusOwner*
1.1245 +bus_service_get_primary_owner (BusService *service)
1.1246 +{
1.1247 + return _dbus_list_get_first (&service->owners);
1.1248 +}
1.1249 +
1.1250 +const char*
1.1251 +bus_service_get_name (BusService *service)
1.1252 +{
1.1253 + return service->name;
1.1254 +}
1.1255 +
1.1256 +dbus_bool_t
1.1257 +bus_service_get_allow_replacement (BusService *service)
1.1258 +{
1.1259 + BusOwner *owner;
1.1260 + DBusList *link;
1.1261 +
1.1262 + _dbus_assert (service->owners != NULL);
1.1263 +
1.1264 + link = _dbus_list_get_first_link (&service->owners);
1.1265 + owner = (BusOwner *) link->data;
1.1266 +
1.1267 + return owner->allow_replacement;
1.1268 +}
1.1269 +
1.1270 +dbus_bool_t
1.1271 +bus_service_has_owner (BusService *service,
1.1272 + DBusConnection *connection)
1.1273 +{
1.1274 + DBusList *link;
1.1275 +
1.1276 + link = _bus_service_find_owner_link (service, connection);
1.1277 +
1.1278 + if (link == NULL)
1.1279 + return FALSE;
1.1280 + else
1.1281 + return TRUE;
1.1282 +}
1.1283 +
1.1284 +dbus_bool_t
1.1285 +bus_service_list_queued_owners (BusService *service,
1.1286 + DBusList **return_list,
1.1287 + DBusError *error)
1.1288 +{
1.1289 + DBusList *link;
1.1290 +
1.1291 + _dbus_assert (*return_list == NULL);
1.1292 +
1.1293 + link = _dbus_list_get_first_link (&service->owners);
1.1294 + _dbus_assert (link != NULL);
1.1295 +
1.1296 + while (link != NULL)
1.1297 + {
1.1298 + BusOwner *owner;
1.1299 + const char *uname;
1.1300 +
1.1301 + owner = (BusOwner *) link->data;
1.1302 + uname = bus_connection_get_name (owner->conn);
1.1303 +
1.1304 + if (!_dbus_list_append (return_list, (char *)uname))
1.1305 + goto oom;
1.1306 +
1.1307 + link = _dbus_list_get_next_link (&service->owners, link);
1.1308 + }
1.1309 +
1.1310 + return TRUE;
1.1311 +
1.1312 + oom:
1.1313 + _dbus_list_clear (return_list);
1.1314 + BUS_SET_OOM (error);
1.1315 + return FALSE;
1.1316 +}