os/ossrv/ofdbus/dbus/bus/services.c
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
/* -*- mode: C; c-file-style: "gnu" -*- */
sl@0
     2
/* services.c  Service management
sl@0
     3
 *
sl@0
     4
 * Copyright (C) 2003  Red Hat, Inc.
sl@0
     5
 * Copyright (C) 2003  CodeFactory AB
sl@0
     6
 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
sl@0
     7
 * Licensed under the Academic Free License version 2.1
sl@0
     8
 * 
sl@0
     9
 * This program is free software; you can redistribute it and/or modify
sl@0
    10
 * it under the terms of the GNU General Public License as published by
sl@0
    11
 * the Free Software Foundation; either version 2 of the License, or
sl@0
    12
 * (at your option) any later version.
sl@0
    13
 *
sl@0
    14
 * This program is distributed in the hope that it will be useful,
sl@0
    15
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    16
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
sl@0
    17
 * GNU General Public License for more details.
sl@0
    18
 * 
sl@0
    19
 * You should have received a copy of the GNU General Public License
sl@0
    20
 * along with this program; if not, write to the Free Software
sl@0
    21
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
sl@0
    22
 *
sl@0
    23
 */
sl@0
    24
 #ifndef __SYMBIAN32__
sl@0
    25
#include <dbus/dbus-hash.h>
sl@0
    26
#include <dbus/dbus-list.h>
sl@0
    27
#include <dbus/dbus-mempool.h>
sl@0
    28
#include <dbus/dbus-marshal-validate.h>
sl@0
    29
#else
sl@0
    30
#include "dbus-hash.h"
sl@0
    31
#include "dbus-list.h"
sl@0
    32
#include "dbus-mempool.h"
sl@0
    33
#include "dbus-marshal-validate.h"
sl@0
    34
#endif //__SYMBIAN32__
sl@0
    35
sl@0
    36
#include "driver.h"
sl@0
    37
#include "services.h"
sl@0
    38
#include "connection.h"
sl@0
    39
#include "utils.h"
sl@0
    40
#include "activation.h"
sl@0
    41
#include "policy.h"
sl@0
    42
#include "bus.h"
sl@0
    43
#include "selinux.h"
sl@0
    44
sl@0
    45
struct BusService
sl@0
    46
{
sl@0
    47
  int refcount;
sl@0
    48
sl@0
    49
  BusRegistry *registry;
sl@0
    50
  char *name;
sl@0
    51
  DBusList *owners;
sl@0
    52
};
sl@0
    53
sl@0
    54
struct BusOwner
sl@0
    55
{
sl@0
    56
  int refcount;
sl@0
    57
sl@0
    58
  BusService *service;
sl@0
    59
  DBusConnection *conn;
sl@0
    60
sl@0
    61
  unsigned int allow_replacement : 1;
sl@0
    62
  unsigned int do_not_queue : 1;
sl@0
    63
};
sl@0
    64
sl@0
    65
struct BusRegistry
sl@0
    66
{
sl@0
    67
  int refcount;
sl@0
    68
sl@0
    69
  BusContext *context;
sl@0
    70
  
sl@0
    71
  DBusHashTable *service_hash;
sl@0
    72
  DBusMemPool   *service_pool;
sl@0
    73
  DBusMemPool   *owner_pool;
sl@0
    74
sl@0
    75
  DBusHashTable *service_sid_table;
sl@0
    76
};
sl@0
    77
sl@0
    78
BusRegistry*
sl@0
    79
bus_registry_new (BusContext *context)
sl@0
    80
{
sl@0
    81
  BusRegistry *registry;
sl@0
    82
sl@0
    83
  registry = dbus_new0 (BusRegistry, 1);
sl@0
    84
  if (registry == NULL)
sl@0
    85
    return NULL;
sl@0
    86
sl@0
    87
  registry->refcount = 1;
sl@0
    88
  registry->context = context;
sl@0
    89
  
sl@0
    90
  registry->service_hash = _dbus_hash_table_new (DBUS_HASH_STRING,
sl@0
    91
                                                 NULL, NULL);
sl@0
    92
  if (registry->service_hash == NULL)
sl@0
    93
    goto failed;
sl@0
    94
  
sl@0
    95
  registry->service_pool = _dbus_mem_pool_new (sizeof (BusService),
sl@0
    96
                                               TRUE);
sl@0
    97
sl@0
    98
  if (registry->service_pool == NULL)
sl@0
    99
    goto failed;
sl@0
   100
sl@0
   101
  registry->owner_pool = _dbus_mem_pool_new (sizeof (BusOwner),
sl@0
   102
                                             TRUE);
sl@0
   103
sl@0
   104
  if (registry->owner_pool == NULL)
sl@0
   105
    goto failed;
sl@0
   106
sl@0
   107
  registry->service_sid_table = NULL;
sl@0
   108
  
sl@0
   109
  return registry;
sl@0
   110
sl@0
   111
 failed:
sl@0
   112
  bus_registry_unref (registry);
sl@0
   113
  return NULL;
sl@0
   114
}
sl@0
   115
sl@0
   116
BusRegistry *
sl@0
   117
bus_registry_ref (BusRegistry *registry)
sl@0
   118
{
sl@0
   119
  _dbus_assert (registry->refcount > 0);
sl@0
   120
  registry->refcount += 1;
sl@0
   121
sl@0
   122
  return registry;
sl@0
   123
}
sl@0
   124
sl@0
   125
void
sl@0
   126
bus_registry_unref  (BusRegistry *registry)
sl@0
   127
{
sl@0
   128
  _dbus_assert (registry->refcount > 0);
sl@0
   129
  registry->refcount -= 1;
sl@0
   130
sl@0
   131
  if (registry->refcount == 0)
sl@0
   132
    {
sl@0
   133
      if (registry->service_hash)
sl@0
   134
        _dbus_hash_table_unref (registry->service_hash);
sl@0
   135
      if (registry->service_pool)
sl@0
   136
        _dbus_mem_pool_free (registry->service_pool);
sl@0
   137
      if (registry->owner_pool)
sl@0
   138
        _dbus_mem_pool_free (registry->owner_pool);
sl@0
   139
      if (registry->service_sid_table)
sl@0
   140
        _dbus_hash_table_unref (registry->service_sid_table);
sl@0
   141
      
sl@0
   142
      dbus_free (registry);
sl@0
   143
    }
sl@0
   144
}
sl@0
   145
sl@0
   146
BusService*
sl@0
   147
bus_registry_lookup (BusRegistry      *registry,
sl@0
   148
                     const DBusString *service_name)
sl@0
   149
{
sl@0
   150
  BusService *service;
sl@0
   151
sl@0
   152
  service = _dbus_hash_table_lookup_string (registry->service_hash,
sl@0
   153
                                            _dbus_string_get_const_data (service_name));
sl@0
   154
sl@0
   155
  return service;
sl@0
   156
}
sl@0
   157
sl@0
   158
static DBusList *
sl@0
   159
_bus_service_find_owner_link (BusService *service,
sl@0
   160
                              DBusConnection *connection)
sl@0
   161
{
sl@0
   162
  DBusList *link;
sl@0
   163
  
sl@0
   164
  link = _dbus_list_get_first_link (&service->owners);
sl@0
   165
sl@0
   166
  while (link != NULL)
sl@0
   167
    {
sl@0
   168
      BusOwner *bus_owner;
sl@0
   169
sl@0
   170
      bus_owner = (BusOwner *) link->data;
sl@0
   171
      if (bus_owner->conn == connection) 
sl@0
   172
        break;
sl@0
   173
sl@0
   174
      link = _dbus_list_get_next_link (&service->owners, link);
sl@0
   175
    }
sl@0
   176
sl@0
   177
  return link;
sl@0
   178
}
sl@0
   179
sl@0
   180
static void
sl@0
   181
bus_owner_set_flags (BusOwner *owner,
sl@0
   182
                     dbus_uint32_t flags)
sl@0
   183
{
sl@0
   184
   owner->allow_replacement = 
sl@0
   185
        (flags & DBUS_NAME_FLAG_ALLOW_REPLACEMENT) != FALSE;
sl@0
   186
sl@0
   187
   owner->do_not_queue =
sl@0
   188
        (flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) != FALSE;
sl@0
   189
}
sl@0
   190
sl@0
   191
static BusOwner *
sl@0
   192
bus_owner_new (BusService *service, 
sl@0
   193
               DBusConnection *conn, 
sl@0
   194
	       dbus_uint32_t flags)
sl@0
   195
{
sl@0
   196
  BusOwner *result;
sl@0
   197
sl@0
   198
  result = _dbus_mem_pool_alloc (service->registry->owner_pool);
sl@0
   199
  if (result != NULL)
sl@0
   200
    {
sl@0
   201
      result->refcount = 1;
sl@0
   202
      /* don't ref the connection because we don't want
sl@0
   203
         to block the connection from going away.
sl@0
   204
         transactions take care of reffing the connection
sl@0
   205
         but we need to use refcounting on the owner
sl@0
   206
         so that the owner does not get freed before
sl@0
   207
         we can deref the connection in the transaction
sl@0
   208
       */
sl@0
   209
      result->conn = conn;
sl@0
   210
      result->service = service;
sl@0
   211
sl@0
   212
      if (!bus_connection_add_owned_service (conn, service))
sl@0
   213
        {
sl@0
   214
          _dbus_mem_pool_dealloc (service->registry->owner_pool, result);
sl@0
   215
          return NULL;
sl@0
   216
        }
sl@0
   217
        
sl@0
   218
      bus_owner_set_flags (result, flags);
sl@0
   219
    }
sl@0
   220
  return result;
sl@0
   221
}
sl@0
   222
sl@0
   223
static BusOwner *
sl@0
   224
bus_owner_ref (BusOwner *owner)
sl@0
   225
{
sl@0
   226
  _dbus_assert (owner->refcount > 0);
sl@0
   227
  owner->refcount += 1;
sl@0
   228
sl@0
   229
  return owner;
sl@0
   230
}
sl@0
   231
sl@0
   232
static void
sl@0
   233
bus_owner_unref  (BusOwner *owner)
sl@0
   234
{
sl@0
   235
  _dbus_assert (owner->refcount > 0);
sl@0
   236
  owner->refcount -= 1;
sl@0
   237
sl@0
   238
  if (owner->refcount == 0)
sl@0
   239
    {
sl@0
   240
      bus_connection_remove_owned_service (owner->conn, owner->service);
sl@0
   241
      _dbus_mem_pool_dealloc (owner->service->registry->owner_pool, owner);
sl@0
   242
    }
sl@0
   243
}
sl@0
   244
sl@0
   245
BusService*
sl@0
   246
bus_registry_ensure (BusRegistry               *registry,
sl@0
   247
                     const DBusString          *service_name,
sl@0
   248
                     DBusConnection            *owner_connection_if_created,
sl@0
   249
                     dbus_uint32_t              flags,
sl@0
   250
                     BusTransaction            *transaction,
sl@0
   251
                     DBusError                 *error)
sl@0
   252
{
sl@0
   253
  BusService *service;
sl@0
   254
sl@0
   255
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   256
  
sl@0
   257
  _dbus_assert (owner_connection_if_created != NULL);
sl@0
   258
  _dbus_assert (transaction != NULL);
sl@0
   259
sl@0
   260
  service = _dbus_hash_table_lookup_string (registry->service_hash,
sl@0
   261
                                            _dbus_string_get_const_data (service_name));
sl@0
   262
  if (service != NULL)
sl@0
   263
    return service;
sl@0
   264
  
sl@0
   265
  service = _dbus_mem_pool_alloc (registry->service_pool);
sl@0
   266
  if (service == NULL)
sl@0
   267
    {
sl@0
   268
      BUS_SET_OOM (error);
sl@0
   269
      return NULL;
sl@0
   270
    }
sl@0
   271
sl@0
   272
  service->registry = registry;  
sl@0
   273
  service->refcount = 1;
sl@0
   274
sl@0
   275
  _dbus_verbose ("copying string %p '%s' to service->name\n",
sl@0
   276
                 service_name, _dbus_string_get_const_data (service_name));
sl@0
   277
  if (!_dbus_string_copy_data (service_name, &service->name))
sl@0
   278
    {
sl@0
   279
      _dbus_mem_pool_dealloc (registry->service_pool, service);
sl@0
   280
      BUS_SET_OOM (error);
sl@0
   281
      return NULL;
sl@0
   282
    }
sl@0
   283
  _dbus_verbose ("copied string %p '%s' to '%s'\n",
sl@0
   284
                 service_name, _dbus_string_get_const_data (service_name),
sl@0
   285
                 service->name);
sl@0
   286
sl@0
   287
  if (!bus_driver_send_service_owner_changed (service->name, 
sl@0
   288
					      NULL,
sl@0
   289
					      bus_connection_get_name (owner_connection_if_created),
sl@0
   290
					      transaction, error))
sl@0
   291
    {
sl@0
   292
      bus_service_unref (service);
sl@0
   293
      return NULL;
sl@0
   294
    }
sl@0
   295
sl@0
   296
  if (!bus_activation_service_created (bus_context_get_activation (registry->context),
sl@0
   297
				       service->name, transaction, error))
sl@0
   298
    {
sl@0
   299
      bus_service_unref (service);
sl@0
   300
      return NULL;
sl@0
   301
    }
sl@0
   302
  
sl@0
   303
  if (!bus_service_add_owner (service, owner_connection_if_created, flags,
sl@0
   304
                                              transaction, error))
sl@0
   305
    {
sl@0
   306
      bus_service_unref (service);
sl@0
   307
      return NULL;
sl@0
   308
    }
sl@0
   309
  
sl@0
   310
  if (!_dbus_hash_table_insert_string (registry->service_hash,
sl@0
   311
                                       service->name,
sl@0
   312
                                       service))
sl@0
   313
    {
sl@0
   314
      /* The add_owner gets reverted on transaction cancel */
sl@0
   315
      BUS_SET_OOM (error);
sl@0
   316
      return NULL;
sl@0
   317
    }
sl@0
   318
  
sl@0
   319
  return service;
sl@0
   320
}
sl@0
   321
sl@0
   322
void
sl@0
   323
bus_registry_foreach (BusRegistry               *registry,
sl@0
   324
                      BusServiceForeachFunction  function,
sl@0
   325
                      void                      *data)
sl@0
   326
{
sl@0
   327
  DBusHashIter iter;
sl@0
   328
  
sl@0
   329
  _dbus_hash_iter_init (registry->service_hash, &iter);
sl@0
   330
  while (_dbus_hash_iter_next (&iter))
sl@0
   331
    {
sl@0
   332
      BusService *service = _dbus_hash_iter_get_value (&iter);
sl@0
   333
sl@0
   334
      (* function) (service, data);
sl@0
   335
    }
sl@0
   336
}
sl@0
   337
sl@0
   338
dbus_bool_t
sl@0
   339
bus_registry_list_services (BusRegistry *registry,
sl@0
   340
                            char      ***listp,
sl@0
   341
                            int         *array_len)
sl@0
   342
{
sl@0
   343
  int i, j, len;
sl@0
   344
  char **retval;
sl@0
   345
  DBusHashIter iter;
sl@0
   346
   
sl@0
   347
  len = _dbus_hash_table_get_n_entries (registry->service_hash);
sl@0
   348
  retval = dbus_new (char *, len + 1);
sl@0
   349
sl@0
   350
  if (retval == NULL)
sl@0
   351
    return FALSE;
sl@0
   352
sl@0
   353
  _dbus_hash_iter_init (registry->service_hash, &iter);
sl@0
   354
  i = 0;
sl@0
   355
  while (_dbus_hash_iter_next (&iter))
sl@0
   356
    {
sl@0
   357
      BusService *service = _dbus_hash_iter_get_value (&iter);
sl@0
   358
sl@0
   359
      retval[i] = _dbus_strdup (service->name);
sl@0
   360
      if (retval[i] == NULL)
sl@0
   361
	goto error;
sl@0
   362
sl@0
   363
      i++;
sl@0
   364
    }
sl@0
   365
sl@0
   366
  retval[i] = NULL;
sl@0
   367
  
sl@0
   368
  if (array_len)
sl@0
   369
    *array_len = len;
sl@0
   370
  
sl@0
   371
  *listp = retval;
sl@0
   372
  return TRUE;
sl@0
   373
  
sl@0
   374
 error:
sl@0
   375
  for (j = 0; j < i; j++)
sl@0
   376
    dbus_free (retval[i]);
sl@0
   377
  dbus_free (retval);
sl@0
   378
sl@0
   379
  return FALSE;
sl@0
   380
}
sl@0
   381
sl@0
   382
dbus_bool_t
sl@0
   383
bus_registry_acquire_service (BusRegistry      *registry,
sl@0
   384
                              DBusConnection   *connection,
sl@0
   385
                              const DBusString *service_name,
sl@0
   386
                              dbus_uint32_t     flags,
sl@0
   387
                              dbus_uint32_t    *result,
sl@0
   388
                              BusTransaction   *transaction,
sl@0
   389
                              DBusError        *error)
sl@0
   390
{
sl@0
   391
  dbus_bool_t retval;
sl@0
   392
  DBusConnection *old_owner_conn;
sl@0
   393
  DBusConnection *current_owner_conn;
sl@0
   394
  BusClientPolicy *policy;
sl@0
   395
  BusService *service;
sl@0
   396
  BusActivation  *activation;
sl@0
   397
  BusSELinuxID *sid;
sl@0
   398
  BusOwner *primary_owner;
sl@0
   399
 
sl@0
   400
  retval = FALSE;
sl@0
   401
sl@0
   402
  if (!_dbus_validate_bus_name (service_name, 0,
sl@0
   403
                                _dbus_string_get_length (service_name)))
sl@0
   404
    {
sl@0
   405
      dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
sl@0
   406
                      "Requested bus name \"%s\" is not valid",
sl@0
   407
                      _dbus_string_get_const_data (service_name));
sl@0
   408
      
sl@0
   409
      _dbus_verbose ("Attempt to acquire invalid service name\n");
sl@0
   410
      
sl@0
   411
      goto out;
sl@0
   412
    }
sl@0
   413
  
sl@0
   414
  if (_dbus_string_get_byte (service_name, 0) == ':')
sl@0
   415
    {
sl@0
   416
      /* Not allowed; only base services can start with ':' */
sl@0
   417
      dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
sl@0
   418
                      "Cannot acquire a service starting with ':' such as \"%s\"",
sl@0
   419
                      _dbus_string_get_const_data (service_name));
sl@0
   420
      
sl@0
   421
      _dbus_verbose ("Attempt to acquire invalid base service name \"%s\"",
sl@0
   422
                     _dbus_string_get_const_data (service_name));
sl@0
   423
      
sl@0
   424
      goto out;
sl@0
   425
    }
sl@0
   426
sl@0
   427
  if (_dbus_string_equal_c_str (service_name, DBUS_SERVICE_DBUS))
sl@0
   428
    {
sl@0
   429
      dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
sl@0
   430
                      "Connection \"%s\" is not allowed to own the service \"%s\"because "
sl@0
   431
                      "it is reserved for D-Bus' use only",
sl@0
   432
                      bus_connection_is_active (connection) ?
sl@0
   433
                      bus_connection_get_name (connection) :
sl@0
   434
                      "(inactive)",
sl@0
   435
                      DBUS_SERVICE_DBUS);
sl@0
   436
      goto out;
sl@0
   437
    }
sl@0
   438
sl@0
   439
  policy = bus_connection_get_policy (connection);
sl@0
   440
  _dbus_assert (policy != NULL);
sl@0
   441
sl@0
   442
  /* Note that if sid is #NULL then the bus's own context gets used
sl@0
   443
   * in bus_connection_selinux_allows_acquire_service()
sl@0
   444
   */
sl@0
   445
  sid = bus_selinux_id_table_lookup (registry->service_sid_table,
sl@0
   446
                                     service_name);
sl@0
   447
sl@0
   448
  if (!bus_selinux_allows_acquire_service (connection, sid,
sl@0
   449
					   _dbus_string_get_const_data (service_name), error))
sl@0
   450
    {
sl@0
   451
sl@0
   452
      if (dbus_error_is_set (error) &&
sl@0
   453
	  dbus_error_has_name (error, DBUS_ERROR_NO_MEMORY))
sl@0
   454
	{
sl@0
   455
	  goto out;
sl@0
   456
	}
sl@0
   457
sl@0
   458
      dbus_set_error (error, DBUS_ERROR_ACCESS_DENIED,
sl@0
   459
                      "Connection \"%s\" is not allowed to own the service \"%s\" due "
sl@0
   460
                      "to SELinux policy",
sl@0
   461
                      bus_connection_is_active (connection) ?
sl@0
   462
                      bus_connection_get_name (connection) :
sl@0
   463
                      "(inactive)",
sl@0
   464
                      _dbus_string_get_const_data (service_name));
sl@0
   465
      goto out;
sl@0
   466
    }
sl@0
   467
  
sl@0
   468
  if (!bus_client_policy_check_can_own (policy, connection,
sl@0
   469
                                        service_name))
sl@0
   470
    {
sl@0
   471
      dbus_set_error (error, DBUS_ERROR_ACCESS_DENIED,
sl@0
   472
                      "Connection \"%s\" is not allowed to own the service \"%s\" due "
sl@0
   473
                      "to security policies in the configuration file",
sl@0
   474
                      bus_connection_is_active (connection) ?
sl@0
   475
                      bus_connection_get_name (connection) :
sl@0
   476
                      "(inactive)",
sl@0
   477
                      _dbus_string_get_const_data (service_name));
sl@0
   478
      goto out;
sl@0
   479
    }
sl@0
   480
sl@0
   481
  if (bus_connection_get_n_services_owned (connection) >=
sl@0
   482
      bus_context_get_max_services_per_connection (registry->context))
sl@0
   483
    {
sl@0
   484
      dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
sl@0
   485
                      "Connection \"%s\" is not allowed to own more services "
sl@0
   486
                      "(increase limits in configuration file if required)",
sl@0
   487
                      bus_connection_is_active (connection) ?
sl@0
   488
                      bus_connection_get_name (connection) :
sl@0
   489
                      "(inactive)");
sl@0
   490
      goto out;
sl@0
   491
    }
sl@0
   492
  
sl@0
   493
  service = bus_registry_lookup (registry, service_name);
sl@0
   494
sl@0
   495
  if (service != NULL)
sl@0
   496
    {
sl@0
   497
      primary_owner = bus_service_get_primary_owner (service);
sl@0
   498
      if (primary_owner != NULL)
sl@0
   499
        old_owner_conn = primary_owner->conn;
sl@0
   500
      else
sl@0
   501
        old_owner_conn = NULL;
sl@0
   502
    }
sl@0
   503
  else
sl@0
   504
    old_owner_conn = NULL;
sl@0
   505
      
sl@0
   506
  if (service == NULL)
sl@0
   507
    {
sl@0
   508
      service = bus_registry_ensure (registry,
sl@0
   509
                                     service_name, connection, flags,
sl@0
   510
                                     transaction, error);
sl@0
   511
      if (service == NULL)
sl@0
   512
        goto out;
sl@0
   513
    }
sl@0
   514
sl@0
   515
  primary_owner = bus_service_get_primary_owner (service);
sl@0
   516
  if (primary_owner == NULL)
sl@0
   517
    goto out;
sl@0
   518
    
sl@0
   519
  current_owner_conn = primary_owner->conn;
sl@0
   520
     
sl@0
   521
  if (old_owner_conn == NULL)
sl@0
   522
    {
sl@0
   523
      _dbus_assert (current_owner_conn == connection);
sl@0
   524
sl@0
   525
      *result = DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER;      
sl@0
   526
    }
sl@0
   527
  else if (old_owner_conn == connection)
sl@0
   528
    {
sl@0
   529
      bus_owner_set_flags (primary_owner, flags);
sl@0
   530
      *result = DBUS_REQUEST_NAME_REPLY_ALREADY_OWNER;
sl@0
   531
    }
sl@0
   532
  else if (((flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
sl@0
   533
           !(bus_service_get_allow_replacement (service))) ||
sl@0
   534
	   ((flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
sl@0
   535
           !(flags & DBUS_NAME_FLAG_REPLACE_EXISTING))) 
sl@0
   536
    {
sl@0
   537
      DBusList *link;
sl@0
   538
      BusOwner *temp_owner;
sl@0
   539
    /* Since we can't be queued if we are already in the queue
sl@0
   540
       remove us */
sl@0
   541
sl@0
   542
      link = _bus_service_find_owner_link (service, connection);
sl@0
   543
      if (link != NULL)
sl@0
   544
        {
sl@0
   545
          _dbus_list_unlink (&service->owners, link);
sl@0
   546
          temp_owner = (BusOwner *)link->data;
sl@0
   547
          bus_owner_unref (temp_owner); 
sl@0
   548
          _dbus_list_free_link (link);
sl@0
   549
        }
sl@0
   550
      
sl@0
   551
      *result = DBUS_REQUEST_NAME_REPLY_EXISTS;
sl@0
   552
    }
sl@0
   553
  else if (!(flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
sl@0
   554
           (!(flags & DBUS_NAME_FLAG_REPLACE_EXISTING) ||
sl@0
   555
	    !(bus_service_get_allow_replacement (service))))
sl@0
   556
    {
sl@0
   557
      /* Queue the connection */
sl@0
   558
      if (!bus_service_add_owner (service, connection, 
sl@0
   559
                                  flags,
sl@0
   560
                                  transaction, error))
sl@0
   561
        goto out;
sl@0
   562
      
sl@0
   563
      *result = DBUS_REQUEST_NAME_REPLY_IN_QUEUE;
sl@0
   564
    }
sl@0
   565
  else
sl@0
   566
    {
sl@0
   567
      /* Replace the current owner */
sl@0
   568
sl@0
   569
      /* We enqueue the new owner and remove the first one because
sl@0
   570
       * that will cause NameAcquired and NameLost messages to
sl@0
   571
       * be sent.
sl@0
   572
       */
sl@0
   573
      
sl@0
   574
      if (!bus_service_add_owner (service, connection,
sl@0
   575
                                  flags,
sl@0
   576
                                  transaction, error))
sl@0
   577
        goto out;
sl@0
   578
sl@0
   579
      if (primary_owner->do_not_queue)
sl@0
   580
        {
sl@0
   581
          if (!bus_service_remove_owner (service, old_owner_conn,
sl@0
   582
                                         transaction, error))
sl@0
   583
            goto out;
sl@0
   584
        }
sl@0
   585
      else
sl@0
   586
        {
sl@0
   587
          if (!bus_service_swap_owner (service, old_owner_conn,
sl@0
   588
                                       transaction, error))
sl@0
   589
            goto out;
sl@0
   590
        }
sl@0
   591
        
sl@0
   592
    
sl@0
   593
      _dbus_assert (connection == bus_service_get_primary_owner (service)->conn);
sl@0
   594
      *result = DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER;
sl@0
   595
    }
sl@0
   596
sl@0
   597
  activation = bus_context_get_activation (registry->context);
sl@0
   598
  retval = bus_activation_send_pending_auto_activation_messages (activation,
sl@0
   599
								 service,
sl@0
   600
								 transaction,
sl@0
   601
								 error);
sl@0
   602
  
sl@0
   603
 out:
sl@0
   604
  return retval;
sl@0
   605
}
sl@0
   606
sl@0
   607
dbus_bool_t
sl@0
   608
bus_registry_release_service (BusRegistry      *registry,
sl@0
   609
                              DBusConnection   *connection,
sl@0
   610
                              const DBusString *service_name,
sl@0
   611
                              dbus_uint32_t    *result,
sl@0
   612
                              BusTransaction   *transaction,
sl@0
   613
                              DBusError        *error)
sl@0
   614
{
sl@0
   615
  dbus_bool_t retval;
sl@0
   616
  BusService *service;
sl@0
   617
sl@0
   618
  retval = FALSE;
sl@0
   619
sl@0
   620
  if (!_dbus_validate_bus_name (service_name, 0,
sl@0
   621
                                _dbus_string_get_length (service_name)))
sl@0
   622
    {
sl@0
   623
      dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
sl@0
   624
                      "Given bus name \"%s\" is not valid",
sl@0
   625
                      _dbus_string_get_const_data (service_name));
sl@0
   626
sl@0
   627
      _dbus_verbose ("Attempt to release invalid service name\n");
sl@0
   628
sl@0
   629
      goto out;
sl@0
   630
    }
sl@0
   631
sl@0
   632
  if (_dbus_string_get_byte (service_name, 0) == ':')
sl@0
   633
    {
sl@0
   634
      /* Not allowed; the base service name cannot be created or released */
sl@0
   635
      dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
sl@0
   636
                      "Cannot release a service starting with ':' such as \"%s\"",
sl@0
   637
                      _dbus_string_get_const_data (service_name));
sl@0
   638
sl@0
   639
      _dbus_verbose ("Attempt to release invalid base service name \"%s\"",
sl@0
   640
                     _dbus_string_get_const_data (service_name));
sl@0
   641
sl@0
   642
      goto out;
sl@0
   643
    }
sl@0
   644
sl@0
   645
   if (_dbus_string_equal_c_str (service_name, DBUS_SERVICE_DBUS))
sl@0
   646
    {
sl@0
   647
      /* Not allowed; the base service name cannot be created or released */
sl@0
   648
      dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
sl@0
   649
                      "Cannot release the %s service because it is owned by the bus",
sl@0
   650
                     DBUS_SERVICE_DBUS);
sl@0
   651
sl@0
   652
      _dbus_verbose ("Attempt to release service name \"%s\"",
sl@0
   653
                     DBUS_SERVICE_DBUS);
sl@0
   654
sl@0
   655
      goto out;
sl@0
   656
    }
sl@0
   657
sl@0
   658
  service = bus_registry_lookup (registry, service_name);
sl@0
   659
sl@0
   660
  if (service == NULL)
sl@0
   661
    {
sl@0
   662
      *result = DBUS_RELEASE_NAME_REPLY_NON_EXISTENT;
sl@0
   663
    }
sl@0
   664
  else if (!bus_service_has_owner (service, connection))
sl@0
   665
    {
sl@0
   666
      *result = DBUS_RELEASE_NAME_REPLY_NOT_OWNER;
sl@0
   667
    }
sl@0
   668
  else
sl@0
   669
    {
sl@0
   670
      if (!bus_service_remove_owner (service, connection,
sl@0
   671
                                     transaction, error))
sl@0
   672
        goto out;
sl@0
   673
sl@0
   674
      _dbus_assert (!bus_service_has_owner (service, connection));
sl@0
   675
      *result = DBUS_RELEASE_NAME_REPLY_RELEASED;
sl@0
   676
    }
sl@0
   677
sl@0
   678
  retval = TRUE;
sl@0
   679
sl@0
   680
 out:
sl@0
   681
  return retval;
sl@0
   682
}
sl@0
   683
sl@0
   684
dbus_bool_t
sl@0
   685
bus_registry_set_service_context_table (BusRegistry   *registry,
sl@0
   686
					DBusHashTable *table)
sl@0
   687
{
sl@0
   688
  DBusHashTable *new_table;
sl@0
   689
  DBusHashIter iter;
sl@0
   690
  
sl@0
   691
  new_table = bus_selinux_id_table_new ();
sl@0
   692
  if (!new_table)
sl@0
   693
    return FALSE;
sl@0
   694
sl@0
   695
  _dbus_hash_iter_init (table, &iter);
sl@0
   696
  while (_dbus_hash_iter_next (&iter))
sl@0
   697
    {
sl@0
   698
      const char *service = _dbus_hash_iter_get_string_key (&iter);
sl@0
   699
      const char *context = _dbus_hash_iter_get_value (&iter);
sl@0
   700
sl@0
   701
      if (!bus_selinux_id_table_insert (new_table,
sl@0
   702
					service,
sl@0
   703
					context))
sl@0
   704
	return FALSE;
sl@0
   705
    }
sl@0
   706
  
sl@0
   707
  if (registry->service_sid_table)
sl@0
   708
    _dbus_hash_table_unref (registry->service_sid_table);
sl@0
   709
  registry->service_sid_table = new_table;
sl@0
   710
  return TRUE;
sl@0
   711
}
sl@0
   712
sl@0
   713
static void
sl@0
   714
bus_service_unlink_owner (BusService      *service,
sl@0
   715
                          BusOwner        *owner)
sl@0
   716
{
sl@0
   717
  _dbus_list_remove_last (&service->owners, owner);
sl@0
   718
  bus_owner_unref (owner);
sl@0
   719
}
sl@0
   720
sl@0
   721
static void
sl@0
   722
bus_service_unlink (BusService *service)
sl@0
   723
{
sl@0
   724
  _dbus_assert (service->owners == NULL);
sl@0
   725
sl@0
   726
  /* the service may not be in the hash, if
sl@0
   727
   * the failure causing transaction cancel
sl@0
   728
   * was in the right place, but that's OK
sl@0
   729
   */
sl@0
   730
  _dbus_hash_table_remove_string (service->registry->service_hash,
sl@0
   731
                                  service->name);
sl@0
   732
  
sl@0
   733
  bus_service_unref (service);
sl@0
   734
}
sl@0
   735
sl@0
   736
static void
sl@0
   737
bus_service_relink (BusService           *service,
sl@0
   738
                    DBusPreallocatedHash *preallocated)
sl@0
   739
{
sl@0
   740
  _dbus_assert (service->owners == NULL);
sl@0
   741
  _dbus_assert (preallocated != NULL);
sl@0
   742
sl@0
   743
  _dbus_hash_table_insert_string_preallocated (service->registry->service_hash,
sl@0
   744
                                               preallocated,
sl@0
   745
                                               service->name,
sl@0
   746
                                               service);
sl@0
   747
  
sl@0
   748
  bus_service_ref (service);
sl@0
   749
}
sl@0
   750
sl@0
   751
/**
sl@0
   752
 * Data used to represent an ownership cancellation in
sl@0
   753
 * a bus transaction.
sl@0
   754
 */
sl@0
   755
typedef struct
sl@0
   756
{
sl@0
   757
  BusOwner *owner;            /**< the owner */
sl@0
   758
  BusService *service;        /**< service to cancel ownership of */
sl@0
   759
} OwnershipCancelData;
sl@0
   760
sl@0
   761
static void
sl@0
   762
cancel_ownership (void *data)
sl@0
   763
{
sl@0
   764
  OwnershipCancelData *d = data;
sl@0
   765
sl@0
   766
  /* We don't need to send messages notifying of these
sl@0
   767
   * changes, since we're reverting something that was
sl@0
   768
   * cancelled (effectively never really happened)
sl@0
   769
   */
sl@0
   770
  bus_service_unlink_owner (d->service, d->owner);
sl@0
   771
  
sl@0
   772
  if (d->service->owners == NULL)
sl@0
   773
    bus_service_unlink (d->service);
sl@0
   774
}
sl@0
   775
sl@0
   776
static void
sl@0
   777
free_ownership_cancel_data (void *data)
sl@0
   778
{
sl@0
   779
  OwnershipCancelData *d = data;
sl@0
   780
sl@0
   781
  dbus_connection_unref (d->owner->conn);
sl@0
   782
  bus_owner_unref (d->owner);
sl@0
   783
  bus_service_unref (d->service);
sl@0
   784
  
sl@0
   785
  dbus_free (d);
sl@0
   786
}
sl@0
   787
sl@0
   788
static dbus_bool_t
sl@0
   789
add_cancel_ownership_to_transaction (BusTransaction *transaction,
sl@0
   790
                                     BusService     *service,
sl@0
   791
                                     BusOwner       *owner)
sl@0
   792
{
sl@0
   793
  OwnershipCancelData *d;
sl@0
   794
sl@0
   795
  d = dbus_new (OwnershipCancelData, 1);
sl@0
   796
  if (d == NULL)
sl@0
   797
    return FALSE;
sl@0
   798
  
sl@0
   799
  d->service = service;
sl@0
   800
  d->owner = owner;
sl@0
   801
sl@0
   802
  if (!bus_transaction_add_cancel_hook (transaction, cancel_ownership, d,
sl@0
   803
                                        free_ownership_cancel_data))
sl@0
   804
    {
sl@0
   805
      dbus_free (d);
sl@0
   806
      return FALSE;
sl@0
   807
    }
sl@0
   808
sl@0
   809
  bus_service_ref (d->service);
sl@0
   810
  bus_owner_ref (owner);
sl@0
   811
  dbus_connection_ref (d->owner->conn);
sl@0
   812
 
sl@0
   813
  return TRUE;
sl@0
   814
}
sl@0
   815
sl@0
   816
/* this function is self-cancelling if you cancel the transaction */
sl@0
   817
dbus_bool_t
sl@0
   818
bus_service_add_owner (BusService     *service,
sl@0
   819
                       DBusConnection *connection,
sl@0
   820
                       dbus_uint32_t  flags,
sl@0
   821
                       BusTransaction *transaction,
sl@0
   822
                       DBusError      *error)
sl@0
   823
{
sl@0
   824
  BusOwner *bus_owner;
sl@0
   825
  DBusList *bus_owner_link;
sl@0
   826
  
sl@0
   827
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   828
  
sl@0
   829
 /* Send service acquired message first, OOM will result
sl@0
   830
  * in cancelling the transaction
sl@0
   831
  */
sl@0
   832
  if (service->owners == NULL)
sl@0
   833
    {
sl@0
   834
      if (!bus_driver_send_service_acquired (connection, service->name, transaction, error))
sl@0
   835
        return FALSE;
sl@0
   836
    }
sl@0
   837
  
sl@0
   838
  bus_owner_link = _bus_service_find_owner_link (service, connection);
sl@0
   839
  
sl@0
   840
  if (bus_owner_link == NULL)
sl@0
   841
    {
sl@0
   842
      bus_owner = bus_owner_new (service, connection, flags);
sl@0
   843
      if (bus_owner == NULL)
sl@0
   844
        {
sl@0
   845
          BUS_SET_OOM (error);
sl@0
   846
          return FALSE;
sl@0
   847
        }
sl@0
   848
sl@0
   849
      bus_owner_set_flags (bus_owner, flags);
sl@0
   850
      if (!(flags & DBUS_NAME_FLAG_REPLACE_EXISTING) || service->owners == NULL)
sl@0
   851
        {
sl@0
   852
          if (!_dbus_list_append (&service->owners,
sl@0
   853
                                  bus_owner))
sl@0
   854
            {
sl@0
   855
              bus_owner_unref (bus_owner);
sl@0
   856
              BUS_SET_OOM (error);
sl@0
   857
              return FALSE;
sl@0
   858
            }
sl@0
   859
        }
sl@0
   860
      else
sl@0
   861
        {
sl@0
   862
          if (!_dbus_list_insert_after (&service->owners,
sl@0
   863
                                         _dbus_list_get_first_link (&service->owners),
sl@0
   864
                                         bus_owner))
sl@0
   865
            {
sl@0
   866
              bus_owner_unref (bus_owner);
sl@0
   867
              BUS_SET_OOM (error);
sl@0
   868
              return FALSE;
sl@0
   869
            }
sl@0
   870
        }      
sl@0
   871
    } 
sl@0
   872
  else 
sl@0
   873
    {
sl@0
   874
      /* Update the link since we are already in the queue
sl@0
   875
       * No need for operations that can produce OOM
sl@0
   876
       */
sl@0
   877
sl@0
   878
      bus_owner = (BusOwner *) bus_owner_link->data;
sl@0
   879
      if (flags & DBUS_NAME_FLAG_REPLACE_EXISTING)
sl@0
   880
        {
sl@0
   881
	  DBusList *link;
sl@0
   882
          _dbus_list_unlink (&service->owners, bus_owner_link);
sl@0
   883
	  link = _dbus_list_get_first_link (&service->owners);
sl@0
   884
	  _dbus_assert (link != NULL);
sl@0
   885
	  
sl@0
   886
          _dbus_list_insert_after_link (&service->owners, link, bus_owner_link);
sl@0
   887
        }
sl@0
   888
      
sl@0
   889
      bus_owner_set_flags (bus_owner, flags);
sl@0
   890
      return TRUE;
sl@0
   891
    }
sl@0
   892
sl@0
   893
  if (!add_cancel_ownership_to_transaction (transaction,
sl@0
   894
                                            service,
sl@0
   895
                                            bus_owner))
sl@0
   896
    {
sl@0
   897
      bus_service_unlink_owner (service, bus_owner);
sl@0
   898
      BUS_SET_OOM (error);
sl@0
   899
      return FALSE;
sl@0
   900
    }
sl@0
   901
sl@0
   902
  return TRUE;
sl@0
   903
}
sl@0
   904
sl@0
   905
typedef struct
sl@0
   906
{
sl@0
   907
  BusOwner       *owner;
sl@0
   908
  BusService     *service;
sl@0
   909
  BusOwner       *before_owner; /* restore to position before this connection in owners list */
sl@0
   910
  DBusList       *owner_link;
sl@0
   911
  DBusList       *service_link;
sl@0
   912
  DBusPreallocatedHash *hash_entry;
sl@0
   913
} OwnershipRestoreData;
sl@0
   914
sl@0
   915
static void
sl@0
   916
restore_ownership (void *data)
sl@0
   917
{
sl@0
   918
  OwnershipRestoreData *d = data;
sl@0
   919
  DBusList *link;
sl@0
   920
sl@0
   921
  _dbus_assert (d->service_link != NULL);
sl@0
   922
  _dbus_assert (d->owner_link != NULL);
sl@0
   923
  
sl@0
   924
  if (d->service->owners == NULL)
sl@0
   925
    {
sl@0
   926
      _dbus_assert (d->hash_entry != NULL);
sl@0
   927
      bus_service_relink (d->service, d->hash_entry);
sl@0
   928
    }
sl@0
   929
  else
sl@0
   930
    {
sl@0
   931
      _dbus_assert (d->hash_entry == NULL);
sl@0
   932
    }
sl@0
   933
  
sl@0
   934
  /* We don't need to send messages notifying of these
sl@0
   935
   * changes, since we're reverting something that was
sl@0
   936
   * cancelled (effectively never really happened)
sl@0
   937
   */
sl@0
   938
  link = _dbus_list_get_first_link (&d->service->owners);
sl@0
   939
  while (link != NULL)
sl@0
   940
    {
sl@0
   941
      if (link->data == d->before_owner)
sl@0
   942
        break;
sl@0
   943
sl@0
   944
      link = _dbus_list_get_next_link (&d->service->owners, link);
sl@0
   945
    }
sl@0
   946
  
sl@0
   947
  _dbus_list_insert_before_link (&d->service->owners, link, d->owner_link);
sl@0
   948
sl@0
   949
  /* Note that removing then restoring this changes the order in which
sl@0
   950
   * ServiceDeleted messages are sent on destruction of the
sl@0
   951
   * connection.  This should be OK as the only guarantee there is
sl@0
   952
   * that the base service is destroyed last, and we never even
sl@0
   953
   * tentatively remove the base service.
sl@0
   954
   */
sl@0
   955
  bus_connection_add_owned_service_link (d->owner->conn, d->service_link);
sl@0
   956
  
sl@0
   957
  d->hash_entry = NULL;
sl@0
   958
  d->service_link = NULL;
sl@0
   959
  d->owner_link = NULL;
sl@0
   960
}
sl@0
   961
sl@0
   962
static void
sl@0
   963
free_ownership_restore_data (void *data)
sl@0
   964
{
sl@0
   965
  OwnershipRestoreData *d = data;
sl@0
   966
sl@0
   967
  if (d->service_link)
sl@0
   968
    _dbus_list_free_link (d->service_link);
sl@0
   969
  if (d->owner_link)
sl@0
   970
    _dbus_list_free_link (d->owner_link);
sl@0
   971
  if (d->hash_entry)
sl@0
   972
    _dbus_hash_table_free_preallocated_entry (d->service->registry->service_hash,
sl@0
   973
                                              d->hash_entry);
sl@0
   974
sl@0
   975
  dbus_connection_unref (d->owner->conn);
sl@0
   976
  bus_owner_unref (d->owner);
sl@0
   977
  bus_service_unref (d->service);
sl@0
   978
  
sl@0
   979
  dbus_free (d);
sl@0
   980
}
sl@0
   981
sl@0
   982
static dbus_bool_t
sl@0
   983
add_restore_ownership_to_transaction (BusTransaction *transaction,
sl@0
   984
                                      BusService     *service,
sl@0
   985
                                      BusOwner       *owner)
sl@0
   986
{
sl@0
   987
  OwnershipRestoreData *d;
sl@0
   988
  DBusList *link;
sl@0
   989
sl@0
   990
  d = dbus_new (OwnershipRestoreData, 1);
sl@0
   991
  if (d == NULL)
sl@0
   992
    return FALSE;
sl@0
   993
  
sl@0
   994
  d->service = service;
sl@0
   995
  d->owner = owner;
sl@0
   996
  d->service_link = _dbus_list_alloc_link (service);
sl@0
   997
  d->owner_link = _dbus_list_alloc_link (owner);
sl@0
   998
  d->hash_entry = _dbus_hash_table_preallocate_entry (service->registry->service_hash);
sl@0
   999
  
sl@0
  1000
  bus_service_ref (d->service);
sl@0
  1001
  bus_owner_ref (d->owner);
sl@0
  1002
  dbus_connection_ref (d->owner->conn);
sl@0
  1003
sl@0
  1004
  d->before_owner = NULL;
sl@0
  1005
  link = _dbus_list_get_first_link (&service->owners);
sl@0
  1006
  while (link != NULL)
sl@0
  1007
    {
sl@0
  1008
      if (link->data == owner)
sl@0
  1009
        {
sl@0
  1010
          link = _dbus_list_get_next_link (&service->owners, link);
sl@0
  1011
sl@0
  1012
          if (link)
sl@0
  1013
            d->before_owner = link->data;
sl@0
  1014
sl@0
  1015
          break;
sl@0
  1016
        }
sl@0
  1017
      
sl@0
  1018
      link = _dbus_list_get_next_link (&service->owners, link);
sl@0
  1019
    }
sl@0
  1020
  
sl@0
  1021
  if (d->service_link == NULL ||
sl@0
  1022
      d->owner_link == NULL ||
sl@0
  1023
      d->hash_entry == NULL ||
sl@0
  1024
      !bus_transaction_add_cancel_hook (transaction, restore_ownership, d,
sl@0
  1025
                                        free_ownership_restore_data))
sl@0
  1026
    {
sl@0
  1027
      free_ownership_restore_data (d);
sl@0
  1028
      return FALSE;
sl@0
  1029
    }
sl@0
  1030
  
sl@0
  1031
  return TRUE;
sl@0
  1032
}
sl@0
  1033
sl@0
  1034
dbus_bool_t
sl@0
  1035
bus_service_swap_owner (BusService     *service,
sl@0
  1036
                        DBusConnection *connection,
sl@0
  1037
                        BusTransaction *transaction,
sl@0
  1038
                        DBusError      *error)
sl@0
  1039
{
sl@0
  1040
  DBusList *swap_link;
sl@0
  1041
  BusOwner *primary_owner;
sl@0
  1042
sl@0
  1043
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1044
sl@0
  1045
  /* We send out notifications before we do any work we
sl@0
  1046
   * might have to undo if the notification-sending failed
sl@0
  1047
   */
sl@0
  1048
  
sl@0
  1049
  /* Send service lost message */
sl@0
  1050
  primary_owner = bus_service_get_primary_owner (service);
sl@0
  1051
  if (primary_owner == NULL || primary_owner->conn != connection)
sl@0
  1052
    _dbus_assert_not_reached ("Tried to swap a non primary owner");
sl@0
  1053
sl@0
  1054
    
sl@0
  1055
  if (!bus_driver_send_service_lost (connection, service->name,
sl@0
  1056
                                     transaction, error))
sl@0
  1057
    return FALSE;
sl@0
  1058
sl@0
  1059
  if (service->owners == NULL)
sl@0
  1060
    {
sl@0
  1061
      _dbus_assert_not_reached ("Tried to swap owner of a service that has no owners");
sl@0
  1062
    }
sl@0
  1063
  else if (_dbus_list_length_is_one (&service->owners))
sl@0
  1064
    {
sl@0
  1065
      _dbus_assert_not_reached ("Tried to swap owner of a service that has no other owners in the queue");
sl@0
  1066
    }
sl@0
  1067
  else
sl@0
  1068
    {
sl@0
  1069
      DBusList *link;
sl@0
  1070
      BusOwner *new_owner;
sl@0
  1071
      DBusConnection *new_owner_conn;
sl@0
  1072
      link = _dbus_list_get_first_link (&service->owners);
sl@0
  1073
      _dbus_assert (link != NULL);
sl@0
  1074
      link = _dbus_list_get_next_link (&service->owners, link);
sl@0
  1075
      _dbus_assert (link != NULL);
sl@0
  1076
sl@0
  1077
      new_owner = (BusOwner *)link->data;
sl@0
  1078
      new_owner_conn = new_owner->conn;
sl@0
  1079
sl@0
  1080
      if (!bus_driver_send_service_owner_changed (service->name,
sl@0
  1081
 						  bus_connection_get_name (connection),
sl@0
  1082
 						  bus_connection_get_name (new_owner_conn),
sl@0
  1083
 						  transaction, error))
sl@0
  1084
        return FALSE;
sl@0
  1085
sl@0
  1086
      /* This will be our new owner */
sl@0
  1087
      if (!bus_driver_send_service_acquired (new_owner_conn,
sl@0
  1088
                                             service->name,
sl@0
  1089
                                             transaction,
sl@0
  1090
                                             error))
sl@0
  1091
        return FALSE;
sl@0
  1092
    }
sl@0
  1093
sl@0
  1094
  if (!add_restore_ownership_to_transaction (transaction, service, primary_owner))
sl@0
  1095
    {
sl@0
  1096
      BUS_SET_OOM (error);
sl@0
  1097
      return FALSE;
sl@0
  1098
    }
sl@0
  1099
sl@0
  1100
  /* unlink the primary and make it the second link */
sl@0
  1101
  swap_link = _dbus_list_get_first_link (&service->owners);
sl@0
  1102
  _dbus_list_unlink (&service->owners, swap_link);
sl@0
  1103
sl@0
  1104
  _dbus_list_insert_after_link (&service->owners,
sl@0
  1105
                                _dbus_list_get_first_link (&service->owners),
sl@0
  1106
				swap_link);
sl@0
  1107
sl@0
  1108
  return TRUE;
sl@0
  1109
}
sl@0
  1110
sl@0
  1111
/* this function is self-cancelling if you cancel the transaction */
sl@0
  1112
dbus_bool_t
sl@0
  1113
bus_service_remove_owner (BusService     *service,
sl@0
  1114
                          DBusConnection *connection,
sl@0
  1115
                          BusTransaction *transaction,
sl@0
  1116
                          DBusError      *error)
sl@0
  1117
{
sl@0
  1118
  BusOwner *primary_owner;
sl@0
  1119
  
sl@0
  1120
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1121
  
sl@0
  1122
  /* We send out notifications before we do any work we
sl@0
  1123
   * might have to undo if the notification-sending failed
sl@0
  1124
   */
sl@0
  1125
  
sl@0
  1126
  /* Send service lost message */
sl@0
  1127
  primary_owner = bus_service_get_primary_owner (service);
sl@0
  1128
  if (primary_owner != NULL && primary_owner->conn == connection)
sl@0
  1129
    {
sl@0
  1130
      if (!bus_driver_send_service_lost (connection, service->name,
sl@0
  1131
                                         transaction, error))
sl@0
  1132
        return FALSE;
sl@0
  1133
    }
sl@0
  1134
  else
sl@0
  1135
    {
sl@0
  1136
      /* if we are not the primary owner then just remove us from the queue */
sl@0
  1137
      DBusList *link;
sl@0
  1138
      BusOwner *temp_owner;
sl@0
  1139
sl@0
  1140
      link = _bus_service_find_owner_link (service, connection);
sl@0
  1141
      _dbus_list_unlink (&service->owners, link);
sl@0
  1142
      temp_owner = (BusOwner *)link->data;
sl@0
  1143
      bus_owner_unref (temp_owner); 
sl@0
  1144
      _dbus_list_free_link (link);
sl@0
  1145
sl@0
  1146
      return TRUE; 
sl@0
  1147
    }
sl@0
  1148
sl@0
  1149
  if (service->owners == NULL)
sl@0
  1150
    {
sl@0
  1151
      _dbus_assert_not_reached ("Tried to remove owner of a service that has no owners");
sl@0
  1152
    }
sl@0
  1153
  else if (_dbus_list_length_is_one (&service->owners))
sl@0
  1154
    {
sl@0
  1155
      if (!bus_driver_send_service_owner_changed (service->name,
sl@0
  1156
 						  bus_connection_get_name (connection),
sl@0
  1157
 						  NULL,
sl@0
  1158
 						  transaction, error))
sl@0
  1159
        return FALSE;
sl@0
  1160
    }
sl@0
  1161
  else
sl@0
  1162
    {
sl@0
  1163
      DBusList *link;
sl@0
  1164
      BusOwner *new_owner;
sl@0
  1165
      DBusConnection *new_owner_conn;
sl@0
  1166
      link = _dbus_list_get_first_link (&service->owners);
sl@0
  1167
      _dbus_assert (link != NULL);
sl@0
  1168
      link = _dbus_list_get_next_link (&service->owners, link);
sl@0
  1169
      _dbus_assert (link != NULL);
sl@0
  1170
sl@0
  1171
      new_owner = (BusOwner *)link->data;
sl@0
  1172
      new_owner_conn = new_owner->conn;
sl@0
  1173
sl@0
  1174
      if (!bus_driver_send_service_owner_changed (service->name,
sl@0
  1175
 						  bus_connection_get_name (connection),
sl@0
  1176
 						  bus_connection_get_name (new_owner_conn),
sl@0
  1177
 						  transaction, error))
sl@0
  1178
        return FALSE;
sl@0
  1179
sl@0
  1180
      /* This will be our new owner */
sl@0
  1181
      if (!bus_driver_send_service_acquired (new_owner_conn,
sl@0
  1182
                                             service->name,
sl@0
  1183
                                             transaction,
sl@0
  1184
                                             error))
sl@0
  1185
        return FALSE;
sl@0
  1186
    }
sl@0
  1187
sl@0
  1188
  if (!add_restore_ownership_to_transaction (transaction, service, primary_owner))
sl@0
  1189
    {
sl@0
  1190
      BUS_SET_OOM (error);
sl@0
  1191
      return FALSE;
sl@0
  1192
    }
sl@0
  1193
 
sl@0
  1194
  bus_service_unlink_owner (service, primary_owner);
sl@0
  1195
sl@0
  1196
  if (service->owners == NULL)
sl@0
  1197
    bus_service_unlink (service);
sl@0
  1198
sl@0
  1199
  return TRUE;
sl@0
  1200
}
sl@0
  1201
sl@0
  1202
BusService *
sl@0
  1203
bus_service_ref (BusService *service)
sl@0
  1204
{
sl@0
  1205
  _dbus_assert (service->refcount > 0);
sl@0
  1206
  
sl@0
  1207
  service->refcount += 1;
sl@0
  1208
sl@0
  1209
  return service;
sl@0
  1210
}
sl@0
  1211
sl@0
  1212
void
sl@0
  1213
bus_service_unref (BusService *service)
sl@0
  1214
{
sl@0
  1215
  _dbus_assert (service->refcount > 0);
sl@0
  1216
  
sl@0
  1217
  service->refcount -= 1;
sl@0
  1218
sl@0
  1219
  if (service->refcount == 0)
sl@0
  1220
    {
sl@0
  1221
      _dbus_assert (service->owners == NULL);
sl@0
  1222
      
sl@0
  1223
      dbus_free (service->name);
sl@0
  1224
      _dbus_mem_pool_dealloc (service->registry->service_pool, service);
sl@0
  1225
    }
sl@0
  1226
}
sl@0
  1227
sl@0
  1228
DBusConnection *
sl@0
  1229
bus_service_get_primary_owners_connection (BusService *service)
sl@0
  1230
{
sl@0
  1231
  BusOwner *owner;
sl@0
  1232
sl@0
  1233
  owner = bus_service_get_primary_owner (service);
sl@0
  1234
sl@0
  1235
  if (owner != NULL)
sl@0
  1236
    return owner->conn;
sl@0
  1237
  else
sl@0
  1238
    return NULL;
sl@0
  1239
}
sl@0
  1240
sl@0
  1241
BusOwner*
sl@0
  1242
bus_service_get_primary_owner (BusService *service)
sl@0
  1243
{
sl@0
  1244
  return _dbus_list_get_first (&service->owners);
sl@0
  1245
}
sl@0
  1246
sl@0
  1247
const char*
sl@0
  1248
bus_service_get_name (BusService *service)
sl@0
  1249
{
sl@0
  1250
  return service->name;
sl@0
  1251
}
sl@0
  1252
sl@0
  1253
dbus_bool_t
sl@0
  1254
bus_service_get_allow_replacement (BusService *service)
sl@0
  1255
{
sl@0
  1256
  BusOwner *owner;
sl@0
  1257
  DBusList *link;
sl@0
  1258
 
sl@0
  1259
  _dbus_assert (service->owners != NULL);
sl@0
  1260
sl@0
  1261
  link = _dbus_list_get_first_link (&service->owners);
sl@0
  1262
  owner = (BusOwner *) link->data;
sl@0
  1263
sl@0
  1264
  return owner->allow_replacement;
sl@0
  1265
}
sl@0
  1266
sl@0
  1267
dbus_bool_t
sl@0
  1268
bus_service_has_owner (BusService     *service,
sl@0
  1269
		       DBusConnection *connection)
sl@0
  1270
{
sl@0
  1271
  DBusList *link;
sl@0
  1272
sl@0
  1273
  link = _bus_service_find_owner_link (service, connection);
sl@0
  1274
 
sl@0
  1275
  if (link == NULL)
sl@0
  1276
    return FALSE;
sl@0
  1277
  else
sl@0
  1278
    return TRUE;
sl@0
  1279
}
sl@0
  1280
sl@0
  1281
dbus_bool_t 
sl@0
  1282
bus_service_list_queued_owners (BusService *service,
sl@0
  1283
                                DBusList  **return_list,
sl@0
  1284
                                DBusError  *error)
sl@0
  1285
{
sl@0
  1286
  DBusList *link;
sl@0
  1287
sl@0
  1288
  _dbus_assert (*return_list == NULL);
sl@0
  1289
sl@0
  1290
  link = _dbus_list_get_first_link (&service->owners);
sl@0
  1291
  _dbus_assert (link != NULL);
sl@0
  1292
  
sl@0
  1293
  while (link != NULL)
sl@0
  1294
    {
sl@0
  1295
      BusOwner *owner;
sl@0
  1296
      const char *uname;
sl@0
  1297
sl@0
  1298
      owner = (BusOwner *) link->data;
sl@0
  1299
      uname = bus_connection_get_name (owner->conn);
sl@0
  1300
sl@0
  1301
      if (!_dbus_list_append (return_list, (char *)uname))
sl@0
  1302
        goto oom;
sl@0
  1303
sl@0
  1304
      link = _dbus_list_get_next_link (&service->owners, link);
sl@0
  1305
    }
sl@0
  1306
  
sl@0
  1307
  return TRUE;
sl@0
  1308
  
sl@0
  1309
 oom:
sl@0
  1310
  _dbus_list_clear (return_list);
sl@0
  1311
  BUS_SET_OOM (error);
sl@0
  1312
  return FALSE;
sl@0
  1313
}