os/ossrv/ofdbus/dbus/bus/services.c
changeset 0 bde4ae8d615e
     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 +}