os/ossrv/ofdbus/dbus/bus/connection.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
/* connection.c  Client connections
sl@0
     3
 *
sl@0
     4
 * Copyright (C) 2003  Red Hat, Inc.
sl@0
     5
 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
sl@0
     6
 * Licensed under the Academic Free License version 2.1
sl@0
     7
 * 
sl@0
     8
 * This program is free software; you can redistribute it and/or modify
sl@0
     9
 * it under the terms of the GNU General Public License as published by
sl@0
    10
 * the Free Software Foundation; either version 2 of the License, or
sl@0
    11
 * (at your option) any later version.
sl@0
    12
 *
sl@0
    13
 * This program is distributed in the hope that it will be useful,
sl@0
    14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
sl@0
    16
 * GNU General Public License for more details.
sl@0
    17
 * 
sl@0
    18
 * You should have received a copy of the GNU General Public License
sl@0
    19
 * along with this program; if not, write to the Free Software
sl@0
    20
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
sl@0
    21
 *
sl@0
    22
 */
sl@0
    23
#include "connection.h"
sl@0
    24
#include "dispatch.h"
sl@0
    25
#include "policy.h"
sl@0
    26
#include "services.h"
sl@0
    27
#include "utils.h"
sl@0
    28
#include "signals.h"
sl@0
    29
#include "expirelist.h"
sl@0
    30
#include "selinux.h"
sl@0
    31
#ifndef __SYMBIAN32__
sl@0
    32
#include <dbus/dbus-list.h>
sl@0
    33
#include <dbus/dbus-hash.h>
sl@0
    34
#include <dbus/dbus-timeout.h>
sl@0
    35
#else
sl@0
    36
#include "dbus-list.h"
sl@0
    37
#include "dbus-hash.h"
sl@0
    38
#include "dbus-timeout.h"
sl@0
    39
#endif //__SYMBIAN32__
sl@0
    40
sl@0
    41
#ifdef __SYMBIAN32__
sl@0
    42
#include "config.h"
sl@0
    43
#endif //__SYMBIAN32__
sl@0
    44
sl@0
    45
static void bus_connection_remove_transactions (DBusConnection *connection);
sl@0
    46
sl@0
    47
typedef struct
sl@0
    48
{
sl@0
    49
  BusExpireItem expire_item;
sl@0
    50
sl@0
    51
  DBusConnection *will_get_reply;
sl@0
    52
  DBusConnection *will_send_reply;
sl@0
    53
sl@0
    54
  dbus_uint32_t reply_serial;
sl@0
    55
  
sl@0
    56
} BusPendingReply;
sl@0
    57
sl@0
    58
struct BusConnections
sl@0
    59
{
sl@0
    60
  int refcount;
sl@0
    61
  DBusList *completed;  /**< List of all completed connections */
sl@0
    62
  int n_completed;      /**< Length of completed list */
sl@0
    63
  DBusList *incomplete; /**< List of all not-yet-active connections */
sl@0
    64
  int n_incomplete;     /**< Length of incomplete list */
sl@0
    65
  BusContext *context;
sl@0
    66
  DBusHashTable *completed_by_user; /**< Number of completed connections for each UID */
sl@0
    67
  DBusTimeout *expire_timeout; /**< Timeout for expiring incomplete connections. */
sl@0
    68
  int stamp;                   /**< Incrementing number */
sl@0
    69
  BusExpireList *pending_replies; /**< List of pending replies */
sl@0
    70
};
sl@0
    71
sl@0
    72
static dbus_int32_t connection_data_slot = -1;
sl@0
    73
sl@0
    74
typedef struct
sl@0
    75
{
sl@0
    76
  BusConnections *connections;
sl@0
    77
  DBusList *link_in_connection_list;
sl@0
    78
  DBusConnection *connection;
sl@0
    79
  DBusList *services_owned;
sl@0
    80
  int n_services_owned;
sl@0
    81
  DBusList *match_rules;
sl@0
    82
  int n_match_rules;
sl@0
    83
  char *name;
sl@0
    84
  DBusList *transaction_messages; /**< Stuff we need to send as part of a transaction */
sl@0
    85
  DBusMessage *oom_message;
sl@0
    86
  DBusPreallocatedSend *oom_preallocated;
sl@0
    87
  BusClientPolicy *policy;
sl@0
    88
sl@0
    89
  BusSELinuxID *selinux_id;
sl@0
    90
sl@0
    91
  long connection_tv_sec;  /**< Time when we connected (seconds component) */
sl@0
    92
  long connection_tv_usec; /**< Time when we connected (microsec component) */
sl@0
    93
  int stamp;               /**< connections->stamp last time we were traversed */
sl@0
    94
} BusConnectionData;
sl@0
    95
sl@0
    96
static dbus_bool_t bus_pending_reply_expired (BusExpireList *list,
sl@0
    97
                                              DBusList      *link,
sl@0
    98
                                              void          *data);
sl@0
    99
sl@0
   100
static void bus_connection_drop_pending_replies (BusConnections  *connections,
sl@0
   101
                                                 DBusConnection  *connection);
sl@0
   102
sl@0
   103
static dbus_bool_t expire_incomplete_timeout (void *data);
sl@0
   104
sl@0
   105
#define BUS_CONNECTION_DATA(connection) (dbus_connection_get_data ((connection), connection_data_slot))
sl@0
   106
sl@0
   107
static DBusLoop*
sl@0
   108
connection_get_loop (DBusConnection *connection)
sl@0
   109
{
sl@0
   110
  BusConnectionData *d;
sl@0
   111
sl@0
   112
  d = BUS_CONNECTION_DATA (connection);
sl@0
   113
sl@0
   114
  return bus_context_get_loop (d->connections->context);
sl@0
   115
}
sl@0
   116
sl@0
   117
sl@0
   118
static int
sl@0
   119
get_connections_for_uid (BusConnections *connections,
sl@0
   120
                         dbus_uid_t      uid)
sl@0
   121
{
sl@0
   122
  void *val;
sl@0
   123
  int current_count;
sl@0
   124
sl@0
   125
  /* val is NULL is 0 when it isn't in the hash yet */
sl@0
   126
  
sl@0
   127
  val = _dbus_hash_table_lookup_ulong (connections->completed_by_user,
sl@0
   128
                                       uid);
sl@0
   129
sl@0
   130
  current_count = _DBUS_POINTER_TO_INT (val);
sl@0
   131
sl@0
   132
  return current_count;
sl@0
   133
}
sl@0
   134
sl@0
   135
static dbus_bool_t
sl@0
   136
adjust_connections_for_uid (BusConnections *connections,
sl@0
   137
                            dbus_uid_t      uid,
sl@0
   138
                            int             adjustment)
sl@0
   139
{
sl@0
   140
  int current_count;
sl@0
   141
sl@0
   142
  current_count = get_connections_for_uid (connections, uid);
sl@0
   143
sl@0
   144
  _dbus_verbose ("Adjusting connection count for UID " DBUS_UID_FORMAT
sl@0
   145
                 ": was %d adjustment %d making %d\n",
sl@0
   146
                 uid, current_count, adjustment, current_count + adjustment);
sl@0
   147
  
sl@0
   148
  _dbus_assert (current_count >= 0);
sl@0
   149
  
sl@0
   150
  current_count += adjustment;
sl@0
   151
sl@0
   152
  _dbus_assert (current_count >= 0);
sl@0
   153
sl@0
   154
  if (current_count == 0)
sl@0
   155
    {
sl@0
   156
      _dbus_hash_table_remove_ulong (connections->completed_by_user, uid);
sl@0
   157
      return TRUE;
sl@0
   158
    }
sl@0
   159
  else
sl@0
   160
    {
sl@0
   161
      dbus_bool_t retval;
sl@0
   162
      
sl@0
   163
      retval = _dbus_hash_table_insert_ulong (connections->completed_by_user,
sl@0
   164
                                              uid, _DBUS_INT_TO_POINTER (current_count));
sl@0
   165
sl@0
   166
      /* only positive adjustment can fail as otherwise
sl@0
   167
       * a hash entry should already exist
sl@0
   168
       */
sl@0
   169
      _dbus_assert (adjustment > 0 ||
sl@0
   170
                    (adjustment <= 0 && retval));
sl@0
   171
sl@0
   172
      return retval;
sl@0
   173
    }
sl@0
   174
}
sl@0
   175
sl@0
   176
void
sl@0
   177
bus_connection_disconnected (DBusConnection *connection)
sl@0
   178
{
sl@0
   179
  BusConnectionData *d;
sl@0
   180
  BusService *service;
sl@0
   181
  BusMatchmaker *matchmaker;
sl@0
   182
  
sl@0
   183
  d = BUS_CONNECTION_DATA (connection);
sl@0
   184
  _dbus_assert (d != NULL);
sl@0
   185
sl@0
   186
  _dbus_verbose ("%s disconnected, dropping all service ownership and releasing\n",
sl@0
   187
                 d->name ? d->name : "(inactive)");
sl@0
   188
sl@0
   189
  /* Delete our match rules */
sl@0
   190
  if (d->n_match_rules > 0)
sl@0
   191
    {
sl@0
   192
      matchmaker = bus_context_get_matchmaker (d->connections->context);
sl@0
   193
      bus_matchmaker_disconnected (matchmaker, connection);
sl@0
   194
    }
sl@0
   195
  
sl@0
   196
  /* Drop any service ownership. Unfortunately, this requires
sl@0
   197
   * memory allocation and there doesn't seem to be a good way to
sl@0
   198
   * handle it other than sleeping; we can't "fail" the operation of
sl@0
   199
   * disconnecting a client, and preallocating a broadcast "service is
sl@0
   200
   * now gone" message for every client-service pair seems kind of
sl@0
   201
   * involved.
sl@0
   202
   */
sl@0
   203
  while ((service = _dbus_list_get_last (&d->services_owned)))
sl@0
   204
    {
sl@0
   205
      BusTransaction *transaction;
sl@0
   206
      DBusError error;
sl@0
   207
sl@0
   208
    retry:
sl@0
   209
      
sl@0
   210
      dbus_error_init (&error);
sl@0
   211
        
sl@0
   212
      while ((transaction = bus_transaction_new (d->connections->context)) == NULL)
sl@0
   213
        _dbus_wait_for_memory ();
sl@0
   214
        
sl@0
   215
      if (!bus_service_remove_owner (service, connection,
sl@0
   216
                                     transaction, &error))
sl@0
   217
        {
sl@0
   218
          _DBUS_ASSERT_ERROR_IS_SET (&error);
sl@0
   219
          
sl@0
   220
          if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
sl@0
   221
            {
sl@0
   222
              dbus_error_free (&error);
sl@0
   223
              bus_transaction_cancel_and_free (transaction);
sl@0
   224
              _dbus_wait_for_memory ();
sl@0
   225
              goto retry;
sl@0
   226
            }
sl@0
   227
          else
sl@0
   228
            {
sl@0
   229
              _dbus_verbose ("Failed to remove service owner: %s %s\n",
sl@0
   230
                             error.name, error.message);
sl@0
   231
              _dbus_assert_not_reached ("Removing service owner failed for non-memory-related reason");
sl@0
   232
            }
sl@0
   233
        }
sl@0
   234
        
sl@0
   235
      bus_transaction_execute_and_free (transaction);
sl@0
   236
    }
sl@0
   237
sl@0
   238
  bus_dispatch_remove_connection (connection);
sl@0
   239
  
sl@0
   240
  /* no more watching */
sl@0
   241
  if (!dbus_connection_set_watch_functions (connection,
sl@0
   242
                                            NULL, NULL, NULL,
sl@0
   243
                                            connection,
sl@0
   244
                                            NULL))
sl@0
   245
    _dbus_assert_not_reached ("setting watch functions to NULL failed");
sl@0
   246
sl@0
   247
  if (!dbus_connection_set_timeout_functions (connection,
sl@0
   248
                                              NULL, NULL, NULL,
sl@0
   249
                                              connection,
sl@0
   250
                                              NULL))
sl@0
   251
    _dbus_assert_not_reached ("setting timeout functions to NULL failed");
sl@0
   252
  
sl@0
   253
  dbus_connection_set_unix_user_function (connection,
sl@0
   254
                                          NULL, NULL, NULL);
sl@0
   255
sl@0
   256
  dbus_connection_set_dispatch_status_function (connection,
sl@0
   257
                                                NULL, NULL, NULL);
sl@0
   258
  
sl@0
   259
  bus_connection_remove_transactions (connection);
sl@0
   260
sl@0
   261
  if (d->link_in_connection_list != NULL)
sl@0
   262
    {
sl@0
   263
      if (d->name != NULL)
sl@0
   264
        {
sl@0
   265
          unsigned long uid;
sl@0
   266
          
sl@0
   267
          _dbus_list_remove_link (&d->connections->completed, d->link_in_connection_list);
sl@0
   268
          d->link_in_connection_list = NULL;
sl@0
   269
          d->connections->n_completed -= 1;
sl@0
   270
sl@0
   271
          if (dbus_connection_get_unix_user (connection, &uid))
sl@0
   272
            {
sl@0
   273
              if (!adjust_connections_for_uid (d->connections,
sl@0
   274
                                               uid, -1))
sl@0
   275
                _dbus_assert_not_reached ("adjusting downward should never fail");
sl@0
   276
            }
sl@0
   277
        }
sl@0
   278
      else
sl@0
   279
        {
sl@0
   280
          _dbus_list_remove_link (&d->connections->incomplete, d->link_in_connection_list);
sl@0
   281
          d->link_in_connection_list = NULL;
sl@0
   282
          d->connections->n_incomplete -= 1;
sl@0
   283
        }
sl@0
   284
      
sl@0
   285
      _dbus_assert (d->connections->n_incomplete >= 0);
sl@0
   286
      _dbus_assert (d->connections->n_completed >= 0);
sl@0
   287
    }
sl@0
   288
sl@0
   289
  bus_connection_drop_pending_replies (d->connections, connection);
sl@0
   290
  
sl@0
   291
  /* frees "d" as side effect */
sl@0
   292
  dbus_connection_set_data (connection,
sl@0
   293
                            connection_data_slot,
sl@0
   294
                            NULL, NULL);
sl@0
   295
  
sl@0
   296
  dbus_connection_unref (connection);
sl@0
   297
}
sl@0
   298
sl@0
   299
static dbus_bool_t
sl@0
   300
connection_watch_callback (DBusWatch     *watch,
sl@0
   301
                           unsigned int   condition,
sl@0
   302
                           void          *data)
sl@0
   303
{
sl@0
   304
 /* FIXME this can be done in dbus-mainloop.c
sl@0
   305
  * if the code in activation.c for the babysitter
sl@0
   306
  * watch handler is fixed.
sl@0
   307
  */
sl@0
   308
  
sl@0
   309
#if 0
sl@0
   310
  _dbus_verbose ("Calling handle_watch\n");
sl@0
   311
#endif
sl@0
   312
  return dbus_watch_handle (watch, condition);
sl@0
   313
}
sl@0
   314
sl@0
   315
static dbus_bool_t
sl@0
   316
add_connection_watch (DBusWatch      *watch,
sl@0
   317
                      void           *data)
sl@0
   318
{
sl@0
   319
  DBusConnection *connection = data;
sl@0
   320
sl@0
   321
  return _dbus_loop_add_watch (connection_get_loop (connection),
sl@0
   322
                               watch, connection_watch_callback, connection,
sl@0
   323
                               NULL);
sl@0
   324
}
sl@0
   325
sl@0
   326
static void
sl@0
   327
remove_connection_watch (DBusWatch      *watch,
sl@0
   328
                         void           *data)
sl@0
   329
{
sl@0
   330
  DBusConnection *connection = data;
sl@0
   331
  
sl@0
   332
  _dbus_loop_remove_watch (connection_get_loop (connection),
sl@0
   333
                           watch, connection_watch_callback, connection);
sl@0
   334
}
sl@0
   335
sl@0
   336
static void
sl@0
   337
connection_timeout_callback (DBusTimeout   *timeout,
sl@0
   338
                             void          *data)
sl@0
   339
{
sl@0
   340
  /* DBusConnection *connection = data; */
sl@0
   341
sl@0
   342
  /* can return FALSE on OOM but we just let it fire again later */
sl@0
   343
  dbus_timeout_handle (timeout);
sl@0
   344
}
sl@0
   345
sl@0
   346
static dbus_bool_t
sl@0
   347
add_connection_timeout (DBusTimeout    *timeout,
sl@0
   348
                        void           *data)
sl@0
   349
{
sl@0
   350
  DBusConnection *connection = data;
sl@0
   351
  
sl@0
   352
  return _dbus_loop_add_timeout (connection_get_loop (connection),
sl@0
   353
                                 timeout, connection_timeout_callback, connection, NULL);
sl@0
   354
}
sl@0
   355
sl@0
   356
static void
sl@0
   357
remove_connection_timeout (DBusTimeout    *timeout,
sl@0
   358
                           void           *data)
sl@0
   359
{
sl@0
   360
  DBusConnection *connection = data;
sl@0
   361
  
sl@0
   362
  _dbus_loop_remove_timeout (connection_get_loop (connection),
sl@0
   363
                             timeout, connection_timeout_callback, connection);
sl@0
   364
}
sl@0
   365
sl@0
   366
static void
sl@0
   367
dispatch_status_function (DBusConnection    *connection,
sl@0
   368
                          DBusDispatchStatus new_status,
sl@0
   369
                          void              *data)
sl@0
   370
{
sl@0
   371
  DBusLoop *loop = data;
sl@0
   372
  
sl@0
   373
  if (new_status != DBUS_DISPATCH_COMPLETE)
sl@0
   374
    {
sl@0
   375
      while (!_dbus_loop_queue_dispatch (loop, connection))
sl@0
   376
        _dbus_wait_for_memory ();
sl@0
   377
    }
sl@0
   378
}
sl@0
   379
sl@0
   380
static dbus_bool_t
sl@0
   381
allow_user_function (DBusConnection *connection,
sl@0
   382
                     unsigned long   uid,
sl@0
   383
                     void           *data)
sl@0
   384
{
sl@0
   385
  BusConnectionData *d;
sl@0
   386
    
sl@0
   387
  d = BUS_CONNECTION_DATA (connection);
sl@0
   388
sl@0
   389
  _dbus_assert (d != NULL);
sl@0
   390
  
sl@0
   391
  return bus_context_allow_user (d->connections->context, uid);
sl@0
   392
}
sl@0
   393
sl@0
   394
static void
sl@0
   395
free_connection_data (void *data)
sl@0
   396
{
sl@0
   397
  BusConnectionData *d = data;
sl@0
   398
sl@0
   399
  /* services_owned should be NULL since we should be disconnected */
sl@0
   400
  _dbus_assert (d->services_owned == NULL);
sl@0
   401
  _dbus_assert (d->n_services_owned == 0);
sl@0
   402
  /* similarly */
sl@0
   403
  _dbus_assert (d->transaction_messages == NULL);
sl@0
   404
sl@0
   405
  if (d->oom_preallocated)
sl@0
   406
    dbus_connection_free_preallocated_send (d->connection, d->oom_preallocated);
sl@0
   407
sl@0
   408
  if (d->oom_message)
sl@0
   409
    dbus_message_unref (d->oom_message);
sl@0
   410
sl@0
   411
  if (d->policy)
sl@0
   412
    bus_client_policy_unref (d->policy);
sl@0
   413
sl@0
   414
  if (d->selinux_id)
sl@0
   415
    bus_selinux_id_unref (d->selinux_id);
sl@0
   416
  
sl@0
   417
  dbus_free (d->name);
sl@0
   418
  
sl@0
   419
  dbus_free (d);
sl@0
   420
}
sl@0
   421
sl@0
   422
static void
sl@0
   423
call_timeout_callback (DBusTimeout   *timeout,
sl@0
   424
                       void          *data)
sl@0
   425
{
sl@0
   426
  /* can return FALSE on OOM but we just let it fire again later */
sl@0
   427
  dbus_timeout_handle (timeout);
sl@0
   428
}
sl@0
   429
sl@0
   430
BusConnections*
sl@0
   431
bus_connections_new (BusContext *context)
sl@0
   432
{
sl@0
   433
  BusConnections *connections;
sl@0
   434
sl@0
   435
  if (!dbus_connection_allocate_data_slot (&connection_data_slot))
sl@0
   436
    goto failed_0;
sl@0
   437
sl@0
   438
  connections = dbus_new0 (BusConnections, 1);
sl@0
   439
  if (connections == NULL)
sl@0
   440
    goto failed_1;
sl@0
   441
sl@0
   442
  connections->completed_by_user = _dbus_hash_table_new (DBUS_HASH_ULONG,
sl@0
   443
                                                         NULL, NULL);
sl@0
   444
  if (connections->completed_by_user == NULL)
sl@0
   445
    goto failed_2;
sl@0
   446
sl@0
   447
  connections->expire_timeout = _dbus_timeout_new (100, /* irrelevant */
sl@0
   448
                                                   expire_incomplete_timeout,
sl@0
   449
                                                   connections, NULL);
sl@0
   450
  if (connections->expire_timeout == NULL)
sl@0
   451
    goto failed_3;
sl@0
   452
sl@0
   453
  _dbus_timeout_set_enabled (connections->expire_timeout, FALSE);
sl@0
   454
sl@0
   455
  connections->pending_replies = bus_expire_list_new (bus_context_get_loop (context),
sl@0
   456
                                                      bus_context_get_reply_timeout (context),
sl@0
   457
                                                      bus_pending_reply_expired,
sl@0
   458
                                                      connections);
sl@0
   459
  if (connections->pending_replies == NULL)
sl@0
   460
    goto failed_4;
sl@0
   461
  
sl@0
   462
  if (!_dbus_loop_add_timeout (bus_context_get_loop (context),
sl@0
   463
                               connections->expire_timeout,
sl@0
   464
                               call_timeout_callback, NULL, NULL))
sl@0
   465
    goto failed_5;
sl@0
   466
  
sl@0
   467
  connections->refcount = 1;
sl@0
   468
  connections->context = context;
sl@0
   469
  
sl@0
   470
  return connections;
sl@0
   471
sl@0
   472
 failed_5:
sl@0
   473
  bus_expire_list_free (connections->pending_replies);
sl@0
   474
 failed_4:
sl@0
   475
  _dbus_timeout_unref (connections->expire_timeout);
sl@0
   476
 failed_3:
sl@0
   477
  _dbus_hash_table_unref (connections->completed_by_user);
sl@0
   478
 failed_2:
sl@0
   479
  dbus_free (connections);
sl@0
   480
 failed_1:
sl@0
   481
  dbus_connection_free_data_slot (&connection_data_slot);
sl@0
   482
 failed_0:
sl@0
   483
  return NULL;
sl@0
   484
}
sl@0
   485
sl@0
   486
BusConnections *
sl@0
   487
bus_connections_ref (BusConnections *connections)
sl@0
   488
{
sl@0
   489
  _dbus_assert (connections->refcount > 0);
sl@0
   490
  connections->refcount += 1;
sl@0
   491
sl@0
   492
  return connections;
sl@0
   493
}
sl@0
   494
sl@0
   495
void
sl@0
   496
bus_connections_unref (BusConnections *connections)
sl@0
   497
{
sl@0
   498
  _dbus_assert (connections->refcount > 0);
sl@0
   499
  connections->refcount -= 1;
sl@0
   500
  if (connections->refcount == 0)
sl@0
   501
    {
sl@0
   502
      /* drop all incomplete */
sl@0
   503
      while (connections->incomplete != NULL)
sl@0
   504
        {
sl@0
   505
          DBusConnection *connection;
sl@0
   506
sl@0
   507
          connection = connections->incomplete->data;
sl@0
   508
sl@0
   509
          dbus_connection_ref (connection);
sl@0
   510
          dbus_connection_close (connection);
sl@0
   511
          bus_connection_disconnected (connection);
sl@0
   512
          dbus_connection_unref (connection);
sl@0
   513
        }
sl@0
   514
sl@0
   515
      _dbus_assert (connections->n_incomplete == 0);
sl@0
   516
      
sl@0
   517
      /* drop all real connections */
sl@0
   518
      while (connections->completed != NULL)
sl@0
   519
        {
sl@0
   520
          DBusConnection *connection;
sl@0
   521
sl@0
   522
          connection = connections->completed->data;
sl@0
   523
sl@0
   524
          dbus_connection_ref (connection);
sl@0
   525
          dbus_connection_close (connection);
sl@0
   526
          bus_connection_disconnected (connection);
sl@0
   527
          dbus_connection_unref (connection);
sl@0
   528
        }
sl@0
   529
sl@0
   530
      _dbus_assert (connections->n_completed == 0);
sl@0
   531
sl@0
   532
      bus_expire_list_free (connections->pending_replies);
sl@0
   533
      
sl@0
   534
      _dbus_loop_remove_timeout (bus_context_get_loop (connections->context),
sl@0
   535
                                 connections->expire_timeout,
sl@0
   536
                                 call_timeout_callback, NULL);
sl@0
   537
      
sl@0
   538
      _dbus_timeout_unref (connections->expire_timeout);
sl@0
   539
      
sl@0
   540
      _dbus_hash_table_unref (connections->completed_by_user);
sl@0
   541
      
sl@0
   542
      dbus_free (connections);
sl@0
   543
sl@0
   544
      dbus_connection_free_data_slot (&connection_data_slot);
sl@0
   545
    }
sl@0
   546
}
sl@0
   547
sl@0
   548
dbus_bool_t
sl@0
   549
bus_connections_setup_connection (BusConnections *connections,
sl@0
   550
                                  DBusConnection *connection)
sl@0
   551
{
sl@0
   552
  BusConnectionData *d;
sl@0
   553
  dbus_bool_t retval;
sl@0
   554
  DBusError error;
sl@0
   555
  
sl@0
   556
  d = dbus_new0 (BusConnectionData, 1);
sl@0
   557
  
sl@0
   558
  if (d == NULL)
sl@0
   559
    return FALSE;
sl@0
   560
sl@0
   561
  d->connections = connections;
sl@0
   562
  d->connection = connection;
sl@0
   563
  
sl@0
   564
  _dbus_get_current_time (&d->connection_tv_sec,
sl@0
   565
                          &d->connection_tv_usec);
sl@0
   566
  
sl@0
   567
  _dbus_assert (connection_data_slot >= 0);
sl@0
   568
  
sl@0
   569
  if (!dbus_connection_set_data (connection,
sl@0
   570
                                 connection_data_slot,
sl@0
   571
                                 d, free_connection_data))
sl@0
   572
    {
sl@0
   573
      dbus_free (d);
sl@0
   574
      return FALSE;
sl@0
   575
    }
sl@0
   576
sl@0
   577
  dbus_connection_set_route_peer_messages (connection, TRUE);
sl@0
   578
  
sl@0
   579
  retval = FALSE;
sl@0
   580
sl@0
   581
  dbus_error_init (&error);
sl@0
   582
  d->selinux_id = bus_selinux_init_connection_id (connection,
sl@0
   583
                                                  &error);
sl@0
   584
  if (dbus_error_is_set (&error))
sl@0
   585
    {
sl@0
   586
      /* This is a bit bogus because we pretend all errors
sl@0
   587
       * are OOM; this is done because we know that in bus.c
sl@0
   588
       * an OOM error disconnects the connection, which is
sl@0
   589
       * the same thing we want on any other error.
sl@0
   590
       */
sl@0
   591
      dbus_error_free (&error);
sl@0
   592
      goto out;
sl@0
   593
    }
sl@0
   594
  
sl@0
   595
  if (!dbus_connection_set_watch_functions (connection,
sl@0
   596
                                            add_connection_watch,
sl@0
   597
                                            remove_connection_watch,
sl@0
   598
                                            NULL,
sl@0
   599
                                            connection,
sl@0
   600
                                            NULL))
sl@0
   601
    goto out;
sl@0
   602
  
sl@0
   603
  if (!dbus_connection_set_timeout_functions (connection,
sl@0
   604
                                              add_connection_timeout,
sl@0
   605
                                              remove_connection_timeout,
sl@0
   606
                                              NULL,
sl@0
   607
                                              connection, NULL))
sl@0
   608
    goto out;
sl@0
   609
  
sl@0
   610
  dbus_connection_set_unix_user_function (connection,
sl@0
   611
                                          allow_user_function,
sl@0
   612
                                          NULL, NULL);
sl@0
   613
sl@0
   614
  dbus_connection_set_dispatch_status_function (connection,
sl@0
   615
                                                dispatch_status_function,
sl@0
   616
                                                bus_context_get_loop (connections->context),
sl@0
   617
                                                NULL);
sl@0
   618
sl@0
   619
  d->link_in_connection_list = _dbus_list_alloc_link (connection);
sl@0
   620
  if (d->link_in_connection_list == NULL)
sl@0
   621
    goto out;
sl@0
   622
  
sl@0
   623
  /* Setup the connection with the dispatcher */
sl@0
   624
  if (!bus_dispatch_add_connection (connection))
sl@0
   625
    goto out;
sl@0
   626
sl@0
   627
  if (dbus_connection_get_dispatch_status (connection) != DBUS_DISPATCH_COMPLETE)
sl@0
   628
    {
sl@0
   629
      if (!_dbus_loop_queue_dispatch (bus_context_get_loop (connections->context), connection))
sl@0
   630
        {
sl@0
   631
          bus_dispatch_remove_connection (connection);
sl@0
   632
          goto out;
sl@0
   633
        }
sl@0
   634
    }
sl@0
   635
sl@0
   636
  _dbus_list_append_link (&connections->incomplete, d->link_in_connection_list);
sl@0
   637
  connections->n_incomplete += 1;
sl@0
   638
  
sl@0
   639
  dbus_connection_ref (connection);
sl@0
   640
sl@0
   641
  /* Note that we might disconnect ourselves here, but it only takes
sl@0
   642
   * effect on return to the main loop. We call this to free up
sl@0
   643
   * expired connections if possible, and to queue the timeout for our
sl@0
   644
   * own expiration.
sl@0
   645
   */
sl@0
   646
  bus_connections_expire_incomplete (connections);
sl@0
   647
  
sl@0
   648
  /* And we might also disconnect ourselves here, but again it
sl@0
   649
   * only takes effect on return to main loop.
sl@0
   650
   */
sl@0
   651
  if (connections->n_incomplete >
sl@0
   652
      bus_context_get_max_incomplete_connections (connections->context))
sl@0
   653
    {
sl@0
   654
      _dbus_verbose ("Number of incomplete connections exceeds max, dropping oldest one\n");
sl@0
   655
      
sl@0
   656
      _dbus_assert (connections->incomplete != NULL);
sl@0
   657
      /* Disconnect the oldest unauthenticated connection.  FIXME
sl@0
   658
       * would it be more secure to drop a *random* connection?  This
sl@0
   659
       * algorithm seems to mean that if someone can create new
sl@0
   660
       * connections quickly enough, they can keep anyone else from
sl@0
   661
       * completing authentication. But random may or may not really
sl@0
   662
       * help with that, a more elaborate solution might be required.
sl@0
   663
       */
sl@0
   664
      dbus_connection_close (connections->incomplete->data);
sl@0
   665
    }
sl@0
   666
  
sl@0
   667
  retval = TRUE;
sl@0
   668
sl@0
   669
 out:
sl@0
   670
  if (!retval)
sl@0
   671
    {
sl@0
   672
      if (d->selinux_id)
sl@0
   673
        bus_selinux_id_unref (d->selinux_id);
sl@0
   674
      d->selinux_id = NULL;
sl@0
   675
      
sl@0
   676
      if (!dbus_connection_set_watch_functions (connection,
sl@0
   677
                                                NULL, NULL, NULL,
sl@0
   678
                                                connection,
sl@0
   679
                                                NULL))
sl@0
   680
        _dbus_assert_not_reached ("setting watch functions to NULL failed");
sl@0
   681
      
sl@0
   682
      if (!dbus_connection_set_timeout_functions (connection,
sl@0
   683
                                                  NULL, NULL, NULL,
sl@0
   684
                                                  connection,
sl@0
   685
                                                  NULL))
sl@0
   686
        _dbus_assert_not_reached ("setting timeout functions to NULL failed");
sl@0
   687
sl@0
   688
      dbus_connection_set_unix_user_function (connection,
sl@0
   689
                                              NULL, NULL, NULL);
sl@0
   690
sl@0
   691
      dbus_connection_set_dispatch_status_function (connection,
sl@0
   692
                                                    NULL, NULL, NULL);
sl@0
   693
sl@0
   694
      if (d->link_in_connection_list != NULL)
sl@0
   695
        {
sl@0
   696
          _dbus_assert (d->link_in_connection_list->next == NULL);
sl@0
   697
          _dbus_assert (d->link_in_connection_list->prev == NULL);
sl@0
   698
          _dbus_list_free_link (d->link_in_connection_list);
sl@0
   699
          d->link_in_connection_list = NULL;
sl@0
   700
        }
sl@0
   701
      
sl@0
   702
      if (!dbus_connection_set_data (connection,
sl@0
   703
                                     connection_data_slot,
sl@0
   704
                                     NULL, NULL))
sl@0
   705
        _dbus_assert_not_reached ("failed to set connection data to null");
sl@0
   706
sl@0
   707
      /* "d" has now been freed */
sl@0
   708
    }
sl@0
   709
  
sl@0
   710
  return retval;
sl@0
   711
}
sl@0
   712
sl@0
   713
void
sl@0
   714
bus_connections_expire_incomplete (BusConnections *connections)
sl@0
   715
{    
sl@0
   716
  int next_interval;
sl@0
   717
sl@0
   718
  next_interval = -1;
sl@0
   719
  
sl@0
   720
  if (connections->incomplete != NULL)
sl@0
   721
    {
sl@0
   722
      long tv_sec, tv_usec;
sl@0
   723
      DBusList *link;
sl@0
   724
      int auth_timeout;
sl@0
   725
      
sl@0
   726
      _dbus_get_current_time (&tv_sec, &tv_usec);
sl@0
   727
      auth_timeout = bus_context_get_auth_timeout (connections->context);
sl@0
   728
  
sl@0
   729
      link = _dbus_list_get_first_link (&connections->incomplete);
sl@0
   730
      while (link != NULL)
sl@0
   731
        {
sl@0
   732
          DBusList *next = _dbus_list_get_next_link (&connections->incomplete, link);
sl@0
   733
          DBusConnection *connection;
sl@0
   734
          BusConnectionData *d;
sl@0
   735
          double elapsed;
sl@0
   736
      
sl@0
   737
          connection = link->data;
sl@0
   738
      
sl@0
   739
          d = BUS_CONNECTION_DATA (connection);
sl@0
   740
      
sl@0
   741
          _dbus_assert (d != NULL);
sl@0
   742
      
sl@0
   743
          elapsed = ELAPSED_MILLISECONDS_SINCE (d->connection_tv_sec,
sl@0
   744
                                                d->connection_tv_usec,
sl@0
   745
                                                tv_sec, tv_usec);
sl@0
   746
sl@0
   747
          if (elapsed >= (double) auth_timeout)
sl@0
   748
            {
sl@0
   749
              _dbus_verbose ("Timing out authentication for connection %p\n", connection);
sl@0
   750
              dbus_connection_close (connection);
sl@0
   751
            }
sl@0
   752
          else
sl@0
   753
            {
sl@0
   754
              /* We can end the loop, since the connections are in oldest-first order */
sl@0
   755
              next_interval = ((double)auth_timeout) - elapsed;
sl@0
   756
              _dbus_verbose ("Connection %p authentication expires in %d milliseconds\n",
sl@0
   757
                             connection, next_interval);
sl@0
   758
          
sl@0
   759
              break;
sl@0
   760
            }
sl@0
   761
      
sl@0
   762
          link = next;
sl@0
   763
        }
sl@0
   764
    }
sl@0
   765
sl@0
   766
  bus_expire_timeout_set_interval (connections->expire_timeout,
sl@0
   767
                                   next_interval);
sl@0
   768
}
sl@0
   769
sl@0
   770
static dbus_bool_t
sl@0
   771
expire_incomplete_timeout (void *data)
sl@0
   772
{
sl@0
   773
  BusConnections *connections = data;
sl@0
   774
sl@0
   775
  _dbus_verbose ("Running %s\n", _DBUS_FUNCTION_NAME);
sl@0
   776
  
sl@0
   777
  /* note that this may remove the timeout */
sl@0
   778
  bus_connections_expire_incomplete (connections);
sl@0
   779
sl@0
   780
  return TRUE;
sl@0
   781
}
sl@0
   782
sl@0
   783
dbus_bool_t
sl@0
   784
bus_connection_get_groups  (DBusConnection   *connection,
sl@0
   785
                            unsigned long   **groups,
sl@0
   786
                            int              *n_groups,
sl@0
   787
                            DBusError        *error)
sl@0
   788
{
sl@0
   789
  BusConnectionData *d;
sl@0
   790
  unsigned long uid;
sl@0
   791
  DBusUserDatabase *user_database;
sl@0
   792
  
sl@0
   793
  d = BUS_CONNECTION_DATA (connection);
sl@0
   794
sl@0
   795
  _dbus_assert (d != NULL);
sl@0
   796
sl@0
   797
  user_database = bus_context_get_user_database (d->connections->context);
sl@0
   798
  
sl@0
   799
  *groups = NULL;
sl@0
   800
  *n_groups = 0;
sl@0
   801
sl@0
   802
  if (dbus_connection_get_unix_user (connection, &uid))
sl@0
   803
    {
sl@0
   804
      if (!_dbus_user_database_get_groups (user_database,
sl@0
   805
                                           uid, groups, n_groups,
sl@0
   806
                                           error))
sl@0
   807
        {
sl@0
   808
          _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
   809
          _dbus_verbose ("Did not get any groups for UID %lu\n",
sl@0
   810
                         uid);
sl@0
   811
          return FALSE;
sl@0
   812
        }
sl@0
   813
      else
sl@0
   814
        {
sl@0
   815
          _dbus_verbose ("Got %d groups for UID %lu\n",
sl@0
   816
                         *n_groups, uid);
sl@0
   817
          return TRUE;
sl@0
   818
        }
sl@0
   819
    }
sl@0
   820
  else
sl@0
   821
    return TRUE; /* successfully got 0 groups */
sl@0
   822
}
sl@0
   823
sl@0
   824
dbus_bool_t
sl@0
   825
bus_connection_is_in_group (DBusConnection *connection,
sl@0
   826
                            unsigned long   gid)
sl@0
   827
{
sl@0
   828
  int i;
sl@0
   829
  unsigned long *group_ids;
sl@0
   830
  int n_group_ids;
sl@0
   831
sl@0
   832
  if (!bus_connection_get_groups (connection, &group_ids, &n_group_ids,
sl@0
   833
                                  NULL))
sl@0
   834
    return FALSE;
sl@0
   835
sl@0
   836
  i = 0;
sl@0
   837
  while (i < n_group_ids)
sl@0
   838
    {
sl@0
   839
      if (group_ids[i] == gid)
sl@0
   840
        {
sl@0
   841
          dbus_free (group_ids);
sl@0
   842
          return TRUE;
sl@0
   843
        }
sl@0
   844
      ++i;
sl@0
   845
    }
sl@0
   846
sl@0
   847
  dbus_free (group_ids);
sl@0
   848
  return FALSE;
sl@0
   849
}
sl@0
   850
sl@0
   851
BusClientPolicy*
sl@0
   852
bus_connection_get_policy (DBusConnection *connection)
sl@0
   853
{
sl@0
   854
  BusConnectionData *d;
sl@0
   855
    
sl@0
   856
  d = BUS_CONNECTION_DATA (connection);
sl@0
   857
sl@0
   858
  _dbus_assert (d != NULL);
sl@0
   859
  _dbus_assert (d->policy != NULL);
sl@0
   860
  
sl@0
   861
  return d->policy;
sl@0
   862
}
sl@0
   863
sl@0
   864
static dbus_bool_t
sl@0
   865
foreach_active (BusConnections               *connections,
sl@0
   866
                BusConnectionForeachFunction  function,
sl@0
   867
                void                         *data)
sl@0
   868
{
sl@0
   869
  DBusList *link;
sl@0
   870
  
sl@0
   871
  link = _dbus_list_get_first_link (&connections->completed);
sl@0
   872
  while (link != NULL)
sl@0
   873
    {
sl@0
   874
      DBusConnection *connection = link->data;
sl@0
   875
      DBusList *next = _dbus_list_get_next_link (&connections->completed, link);
sl@0
   876
sl@0
   877
      if (!(* function) (connection, data))
sl@0
   878
        return FALSE;
sl@0
   879
      
sl@0
   880
      link = next;
sl@0
   881
    }
sl@0
   882
sl@0
   883
  return TRUE;
sl@0
   884
}
sl@0
   885
sl@0
   886
static dbus_bool_t
sl@0
   887
foreach_inactive (BusConnections               *connections,
sl@0
   888
                  BusConnectionForeachFunction  function,
sl@0
   889
                  void                         *data)
sl@0
   890
{
sl@0
   891
  DBusList *link;
sl@0
   892
  
sl@0
   893
  link = _dbus_list_get_first_link (&connections->incomplete);
sl@0
   894
  while (link != NULL)
sl@0
   895
    {
sl@0
   896
      DBusConnection *connection = link->data;
sl@0
   897
      DBusList *next = _dbus_list_get_next_link (&connections->incomplete, link);
sl@0
   898
sl@0
   899
      if (!(* function) (connection, data))
sl@0
   900
        return FALSE;
sl@0
   901
      
sl@0
   902
      link = next;
sl@0
   903
    }
sl@0
   904
sl@0
   905
  return TRUE;
sl@0
   906
}
sl@0
   907
sl@0
   908
/**
sl@0
   909
 * Calls function on each active connection; if the function returns
sl@0
   910
 * #FALSE, stops iterating. Active connections are authenticated
sl@0
   911
 * and have sent a Hello message.
sl@0
   912
 *
sl@0
   913
 * @param connections the connections object
sl@0
   914
 * @param function the function
sl@0
   915
 * @param data data to pass to it as a second arg
sl@0
   916
 */
sl@0
   917
void
sl@0
   918
bus_connections_foreach_active (BusConnections               *connections,
sl@0
   919
                                BusConnectionForeachFunction  function,
sl@0
   920
                                void                         *data)
sl@0
   921
{
sl@0
   922
  foreach_active (connections, function, data);
sl@0
   923
}
sl@0
   924
sl@0
   925
/**
sl@0
   926
 * Calls function on each connection; if the function returns
sl@0
   927
 * #FALSE, stops iterating.
sl@0
   928
 *
sl@0
   929
 * @param connections the connections object
sl@0
   930
 * @param function the function
sl@0
   931
 * @param data data to pass to it as a second arg
sl@0
   932
 */
sl@0
   933
void
sl@0
   934
bus_connections_foreach (BusConnections               *connections,
sl@0
   935
                         BusConnectionForeachFunction  function,
sl@0
   936
                         void                         *data)
sl@0
   937
{
sl@0
   938
  if (!foreach_active (connections, function, data))
sl@0
   939
    return;
sl@0
   940
sl@0
   941
  foreach_inactive (connections, function, data);
sl@0
   942
}
sl@0
   943
sl@0
   944
BusContext*
sl@0
   945
bus_connections_get_context (BusConnections *connections)
sl@0
   946
{
sl@0
   947
  return connections->context;
sl@0
   948
}
sl@0
   949
sl@0
   950
/*
sl@0
   951
 * This is used to avoid covering the same connection twice when
sl@0
   952
 * traversing connections. Note that it assumes we will
sl@0
   953
 * bus_connection_mark_stamp() each connection at least once per
sl@0
   954
 * INT_MAX increments of the global stamp, or wraparound would break
sl@0
   955
 * things.
sl@0
   956
 */
sl@0
   957
void
sl@0
   958
bus_connections_increment_stamp (BusConnections *connections)
sl@0
   959
{
sl@0
   960
  connections->stamp += 1;
sl@0
   961
}
sl@0
   962
sl@0
   963
/* Mark connection with current stamp, return TRUE if it
sl@0
   964
 * didn't already have that stamp
sl@0
   965
 */
sl@0
   966
dbus_bool_t
sl@0
   967
bus_connection_mark_stamp (DBusConnection *connection)
sl@0
   968
{
sl@0
   969
  BusConnectionData *d;
sl@0
   970
  
sl@0
   971
  d = BUS_CONNECTION_DATA (connection);
sl@0
   972
  
sl@0
   973
  _dbus_assert (d != NULL);
sl@0
   974
sl@0
   975
  if (d->stamp == d->connections->stamp)
sl@0
   976
    return FALSE;
sl@0
   977
  else
sl@0
   978
    {
sl@0
   979
      d->stamp = d->connections->stamp;
sl@0
   980
      return TRUE;
sl@0
   981
    }
sl@0
   982
}
sl@0
   983
sl@0
   984
BusContext*
sl@0
   985
bus_connection_get_context (DBusConnection *connection)
sl@0
   986
{
sl@0
   987
  BusConnectionData *d;
sl@0
   988
sl@0
   989
  d = BUS_CONNECTION_DATA (connection);
sl@0
   990
sl@0
   991
  _dbus_assert (d != NULL);
sl@0
   992
sl@0
   993
  return d->connections->context;
sl@0
   994
}
sl@0
   995
sl@0
   996
BusConnections*
sl@0
   997
bus_connection_get_connections (DBusConnection *connection)
sl@0
   998
{
sl@0
   999
  BusConnectionData *d;
sl@0
  1000
    
sl@0
  1001
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1002
sl@0
  1003
  _dbus_assert (d != NULL);
sl@0
  1004
sl@0
  1005
  return d->connections;
sl@0
  1006
}
sl@0
  1007
sl@0
  1008
BusRegistry*
sl@0
  1009
bus_connection_get_registry (DBusConnection *connection)
sl@0
  1010
{
sl@0
  1011
  BusConnectionData *d;
sl@0
  1012
sl@0
  1013
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1014
sl@0
  1015
  _dbus_assert (d != NULL);
sl@0
  1016
sl@0
  1017
  return bus_context_get_registry (d->connections->context);
sl@0
  1018
}
sl@0
  1019
sl@0
  1020
BusActivation*
sl@0
  1021
bus_connection_get_activation (DBusConnection *connection)
sl@0
  1022
{
sl@0
  1023
  BusConnectionData *d;
sl@0
  1024
sl@0
  1025
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1026
sl@0
  1027
  _dbus_assert (d != NULL);
sl@0
  1028
sl@0
  1029
  return bus_context_get_activation (d->connections->context);
sl@0
  1030
}
sl@0
  1031
sl@0
  1032
BusMatchmaker*
sl@0
  1033
bus_connection_get_matchmaker (DBusConnection *connection)
sl@0
  1034
{
sl@0
  1035
  BusConnectionData *d;
sl@0
  1036
sl@0
  1037
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1038
sl@0
  1039
  _dbus_assert (d != NULL);
sl@0
  1040
sl@0
  1041
  return bus_context_get_matchmaker (d->connections->context);
sl@0
  1042
}
sl@0
  1043
sl@0
  1044
BusSELinuxID*
sl@0
  1045
bus_connection_get_selinux_id (DBusConnection *connection)
sl@0
  1046
{
sl@0
  1047
  BusConnectionData *d;
sl@0
  1048
sl@0
  1049
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1050
sl@0
  1051
  _dbus_assert (d != NULL);
sl@0
  1052
sl@0
  1053
  return d->selinux_id;
sl@0
  1054
}
sl@0
  1055
sl@0
  1056
/**
sl@0
  1057
 * Checks whether the connection is registered with the message bus.
sl@0
  1058
 *
sl@0
  1059
 * @param connection the connection
sl@0
  1060
 * @returns #TRUE if we're an active message bus participant
sl@0
  1061
 */
sl@0
  1062
dbus_bool_t
sl@0
  1063
bus_connection_is_active (DBusConnection *connection)
sl@0
  1064
{
sl@0
  1065
  BusConnectionData *d;
sl@0
  1066
sl@0
  1067
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1068
  
sl@0
  1069
  return d != NULL && d->name != NULL;
sl@0
  1070
}
sl@0
  1071
sl@0
  1072
dbus_bool_t
sl@0
  1073
bus_connection_preallocate_oom_error (DBusConnection *connection)
sl@0
  1074
{
sl@0
  1075
  DBusMessage *message;
sl@0
  1076
  DBusPreallocatedSend *preallocated;
sl@0
  1077
  BusConnectionData *d;
sl@0
  1078
sl@0
  1079
  d = BUS_CONNECTION_DATA (connection);  
sl@0
  1080
sl@0
  1081
  _dbus_assert (d != NULL);
sl@0
  1082
sl@0
  1083
  if (d->oom_preallocated != NULL)
sl@0
  1084
    return TRUE;
sl@0
  1085
  
sl@0
  1086
  preallocated = dbus_connection_preallocate_send (connection);
sl@0
  1087
  if (preallocated == NULL)
sl@0
  1088
    return FALSE;
sl@0
  1089
sl@0
  1090
  message = dbus_message_new (DBUS_MESSAGE_TYPE_ERROR);
sl@0
  1091
sl@0
  1092
  if (message == NULL)
sl@0
  1093
    {
sl@0
  1094
      dbus_connection_free_preallocated_send (connection, preallocated);
sl@0
  1095
      return FALSE;
sl@0
  1096
    }
sl@0
  1097
sl@0
  1098
  /* d->name may be NULL, but that is OK */
sl@0
  1099
  if (!dbus_message_set_error_name (message, DBUS_ERROR_NO_MEMORY) ||
sl@0
  1100
      !dbus_message_set_destination (message, d->name) ||
sl@0
  1101
      !dbus_message_set_sender (message,
sl@0
  1102
                                DBUS_SERVICE_DBUS))
sl@0
  1103
    {
sl@0
  1104
      dbus_connection_free_preallocated_send (connection, preallocated);
sl@0
  1105
      dbus_message_unref (message);
sl@0
  1106
      return FALSE;
sl@0
  1107
    }
sl@0
  1108
  
sl@0
  1109
  /* set reply serial to placeholder value just so space is already allocated
sl@0
  1110
   * for it.
sl@0
  1111
   */
sl@0
  1112
  if (!dbus_message_set_reply_serial (message, 14))
sl@0
  1113
    {
sl@0
  1114
      dbus_connection_free_preallocated_send (connection, preallocated);
sl@0
  1115
      dbus_message_unref (message);
sl@0
  1116
      return FALSE;
sl@0
  1117
    }
sl@0
  1118
sl@0
  1119
  d->oom_message = message;
sl@0
  1120
  d->oom_preallocated = preallocated;
sl@0
  1121
  
sl@0
  1122
  return TRUE;
sl@0
  1123
}
sl@0
  1124
sl@0
  1125
void
sl@0
  1126
bus_connection_send_oom_error (DBusConnection *connection,
sl@0
  1127
                               DBusMessage    *in_reply_to)
sl@0
  1128
{
sl@0
  1129
  BusConnectionData *d;
sl@0
  1130
sl@0
  1131
  d = BUS_CONNECTION_DATA (connection);  
sl@0
  1132
sl@0
  1133
  _dbus_assert (d != NULL);  
sl@0
  1134
  _dbus_assert (d->oom_message != NULL);
sl@0
  1135
sl@0
  1136
  /* should always succeed since we set it to a placeholder earlier */
sl@0
  1137
  if (!dbus_message_set_reply_serial (d->oom_message,
sl@0
  1138
                                      dbus_message_get_serial (in_reply_to)))
sl@0
  1139
    _dbus_assert_not_reached ("Failed to set reply serial for preallocated oom message");
sl@0
  1140
sl@0
  1141
  _dbus_assert (dbus_message_get_sender (d->oom_message) != NULL);
sl@0
  1142
  
sl@0
  1143
  dbus_connection_send_preallocated (connection, d->oom_preallocated,
sl@0
  1144
                                     d->oom_message, NULL);
sl@0
  1145
sl@0
  1146
  dbus_message_unref (d->oom_message);
sl@0
  1147
  d->oom_message = NULL;
sl@0
  1148
  d->oom_preallocated = NULL;
sl@0
  1149
}
sl@0
  1150
sl@0
  1151
void
sl@0
  1152
bus_connection_add_match_rule_link (DBusConnection *connection,
sl@0
  1153
                                    DBusList       *link)
sl@0
  1154
{
sl@0
  1155
  BusConnectionData *d;
sl@0
  1156
sl@0
  1157
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1158
  _dbus_assert (d != NULL);
sl@0
  1159
sl@0
  1160
  _dbus_list_append_link (&d->match_rules, link);
sl@0
  1161
sl@0
  1162
  d->n_match_rules += 1;
sl@0
  1163
}
sl@0
  1164
sl@0
  1165
dbus_bool_t
sl@0
  1166
bus_connection_add_match_rule (DBusConnection *connection,
sl@0
  1167
                               BusMatchRule   *rule)
sl@0
  1168
{
sl@0
  1169
    DBusList *link;
sl@0
  1170
sl@0
  1171
  link = _dbus_list_alloc_link (rule);
sl@0
  1172
sl@0
  1173
  if (link == NULL)
sl@0
  1174
    return FALSE;
sl@0
  1175
sl@0
  1176
  bus_connection_add_match_rule_link (connection, link);
sl@0
  1177
sl@0
  1178
  return TRUE;
sl@0
  1179
}
sl@0
  1180
sl@0
  1181
void
sl@0
  1182
bus_connection_remove_match_rule (DBusConnection *connection,
sl@0
  1183
                                  BusMatchRule   *rule)
sl@0
  1184
{
sl@0
  1185
  BusConnectionData *d;
sl@0
  1186
sl@0
  1187
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1188
  _dbus_assert (d != NULL);
sl@0
  1189
sl@0
  1190
  _dbus_list_remove_last (&d->match_rules, rule);
sl@0
  1191
sl@0
  1192
  d->n_match_rules -= 1;
sl@0
  1193
  _dbus_assert (d->n_match_rules >= 0);
sl@0
  1194
}
sl@0
  1195
sl@0
  1196
int
sl@0
  1197
bus_connection_get_n_match_rules (DBusConnection *connection)
sl@0
  1198
{
sl@0
  1199
  BusConnectionData *d;
sl@0
  1200
sl@0
  1201
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1202
  _dbus_assert (d != NULL);
sl@0
  1203
  
sl@0
  1204
  return d->n_match_rules;
sl@0
  1205
}
sl@0
  1206
sl@0
  1207
void
sl@0
  1208
bus_connection_add_owned_service_link (DBusConnection *connection,
sl@0
  1209
                                       DBusList       *link)
sl@0
  1210
{
sl@0
  1211
  BusConnectionData *d;
sl@0
  1212
sl@0
  1213
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1214
  _dbus_assert (d != NULL);
sl@0
  1215
sl@0
  1216
  _dbus_list_append_link (&d->services_owned, link);
sl@0
  1217
sl@0
  1218
  d->n_services_owned += 1;
sl@0
  1219
}
sl@0
  1220
sl@0
  1221
dbus_bool_t
sl@0
  1222
bus_connection_add_owned_service (DBusConnection *connection,
sl@0
  1223
                                  BusService     *service)
sl@0
  1224
{
sl@0
  1225
  DBusList *link;
sl@0
  1226
sl@0
  1227
  link = _dbus_list_alloc_link (service);
sl@0
  1228
sl@0
  1229
  if (link == NULL)
sl@0
  1230
    return FALSE;
sl@0
  1231
sl@0
  1232
  bus_connection_add_owned_service_link (connection, link);
sl@0
  1233
sl@0
  1234
  return TRUE;
sl@0
  1235
}
sl@0
  1236
sl@0
  1237
void
sl@0
  1238
bus_connection_remove_owned_service (DBusConnection *connection,
sl@0
  1239
                                     BusService     *service)
sl@0
  1240
{
sl@0
  1241
  BusConnectionData *d;
sl@0
  1242
sl@0
  1243
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1244
  _dbus_assert (d != NULL);
sl@0
  1245
sl@0
  1246
  _dbus_list_remove_last (&d->services_owned, service);
sl@0
  1247
sl@0
  1248
  d->n_services_owned -= 1;
sl@0
  1249
  _dbus_assert (d->n_services_owned >= 0);
sl@0
  1250
}
sl@0
  1251
sl@0
  1252
int
sl@0
  1253
bus_connection_get_n_services_owned (DBusConnection *connection)
sl@0
  1254
{
sl@0
  1255
  BusConnectionData *d;
sl@0
  1256
sl@0
  1257
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1258
  _dbus_assert (d != NULL);
sl@0
  1259
  
sl@0
  1260
  return d->n_services_owned;
sl@0
  1261
}
sl@0
  1262
sl@0
  1263
dbus_bool_t
sl@0
  1264
bus_connection_complete (DBusConnection   *connection,
sl@0
  1265
			 const DBusString *name,
sl@0
  1266
                         DBusError        *error)
sl@0
  1267
{
sl@0
  1268
  BusConnectionData *d;
sl@0
  1269
  unsigned long uid;
sl@0
  1270
  
sl@0
  1271
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1272
  _dbus_assert (d != NULL);
sl@0
  1273
  _dbus_assert (d->name == NULL);
sl@0
  1274
  _dbus_assert (d->policy == NULL);
sl@0
  1275
sl@0
  1276
  _dbus_assert (!bus_connection_is_active (connection));
sl@0
  1277
  
sl@0
  1278
  if (!_dbus_string_copy_data (name, &d->name))
sl@0
  1279
    {
sl@0
  1280
      BUS_SET_OOM (error);
sl@0
  1281
      return FALSE;
sl@0
  1282
    }
sl@0
  1283
sl@0
  1284
  _dbus_assert (d->name != NULL);
sl@0
  1285
  
sl@0
  1286
  _dbus_verbose ("Name %s assigned to %p\n", d->name, connection);
sl@0
  1287
sl@0
  1288
  d->policy = bus_context_create_client_policy (d->connections->context,
sl@0
  1289
                                                connection,
sl@0
  1290
                                                error);
sl@0
  1291
sl@0
  1292
  /* we may have a NULL policy on OOM or error getting list of
sl@0
  1293
   * groups for a user. In the latter case we don't handle it so
sl@0
  1294
   * well currently, as it will just keep failing over and over.
sl@0
  1295
   */
sl@0
  1296
sl@0
  1297
  if (d->policy == NULL)
sl@0
  1298
    {
sl@0
  1299
      _dbus_verbose ("Failed to create security policy for connection %p\n",
sl@0
  1300
                     connection);
sl@0
  1301
      _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1302
      dbus_free (d->name);
sl@0
  1303
      d->name = NULL;
sl@0
  1304
      return FALSE;
sl@0
  1305
    }
sl@0
  1306
  
sl@0
  1307
  if (dbus_connection_get_unix_user (connection, &uid))
sl@0
  1308
    {
sl@0
  1309
      if (!adjust_connections_for_uid (d->connections,
sl@0
  1310
                                       uid, 1))
sl@0
  1311
        {
sl@0
  1312
          BUS_SET_OOM (error);
sl@0
  1313
          dbus_free (d->name);
sl@0
  1314
          d->name = NULL;
sl@0
  1315
          return FALSE;
sl@0
  1316
        }
sl@0
  1317
    }
sl@0
  1318
  
sl@0
  1319
  /* Now the connection is active, move it between lists */
sl@0
  1320
  _dbus_list_unlink (&d->connections->incomplete,
sl@0
  1321
                     d->link_in_connection_list);
sl@0
  1322
  d->connections->n_incomplete -= 1;
sl@0
  1323
  _dbus_list_append_link (&d->connections->completed,
sl@0
  1324
                          d->link_in_connection_list);
sl@0
  1325
  d->connections->n_completed += 1;
sl@0
  1326
sl@0
  1327
  _dbus_assert (d->connections->n_incomplete >= 0);
sl@0
  1328
  _dbus_assert (d->connections->n_completed > 0);
sl@0
  1329
sl@0
  1330
  /* See if we can remove the timeout */
sl@0
  1331
  bus_connections_expire_incomplete (d->connections);
sl@0
  1332
sl@0
  1333
  _dbus_assert (bus_connection_is_active (connection));
sl@0
  1334
  
sl@0
  1335
  return TRUE;
sl@0
  1336
}
sl@0
  1337
sl@0
  1338
const char *
sl@0
  1339
bus_connection_get_name (DBusConnection *connection)
sl@0
  1340
{
sl@0
  1341
  BusConnectionData *d;
sl@0
  1342
  
sl@0
  1343
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1344
  _dbus_assert (d != NULL);
sl@0
  1345
  
sl@0
  1346
  return d->name;
sl@0
  1347
}
sl@0
  1348
sl@0
  1349
/**
sl@0
  1350
 * Check whether completing the passed-in connection would
sl@0
  1351
 * exceed limits, and if so set error and return #FALSE
sl@0
  1352
 */
sl@0
  1353
dbus_bool_t
sl@0
  1354
bus_connections_check_limits (BusConnections  *connections,
sl@0
  1355
                              DBusConnection  *requesting_completion,
sl@0
  1356
                              DBusError       *error)
sl@0
  1357
{
sl@0
  1358
  BusConnectionData *d;
sl@0
  1359
  unsigned long uid;
sl@0
  1360
  
sl@0
  1361
  d = BUS_CONNECTION_DATA (requesting_completion);
sl@0
  1362
  _dbus_assert (d != NULL);
sl@0
  1363
sl@0
  1364
  _dbus_assert (d->name == NULL);
sl@0
  1365
sl@0
  1366
  if (connections->n_completed >=
sl@0
  1367
      bus_context_get_max_completed_connections (connections->context))
sl@0
  1368
    {
sl@0
  1369
      dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
sl@0
  1370
                      "The maximum number of active connections has been reached");
sl@0
  1371
      return FALSE;
sl@0
  1372
    }
sl@0
  1373
  
sl@0
  1374
  if (dbus_connection_get_unix_user (requesting_completion, &uid))
sl@0
  1375
    {
sl@0
  1376
      if (get_connections_for_uid (connections, uid) >=
sl@0
  1377
          bus_context_get_max_connections_per_user (connections->context))
sl@0
  1378
        {
sl@0
  1379
          dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
sl@0
  1380
                          "The maximum number of active connections for UID %lu has been reached",
sl@0
  1381
                          uid);
sl@0
  1382
          return FALSE;
sl@0
  1383
        }
sl@0
  1384
    }
sl@0
  1385
  
sl@0
  1386
  return TRUE;
sl@0
  1387
}
sl@0
  1388
sl@0
  1389
static void
sl@0
  1390
bus_pending_reply_free (BusPendingReply *pending)
sl@0
  1391
{
sl@0
  1392
  _dbus_verbose ("Freeing pending reply %p, replier %p receiver %p serial %u\n",
sl@0
  1393
                 pending,
sl@0
  1394
                 pending->will_send_reply,
sl@0
  1395
                 pending->will_get_reply,
sl@0
  1396
                 pending->reply_serial);
sl@0
  1397
sl@0
  1398
  dbus_free (pending);
sl@0
  1399
}
sl@0
  1400
sl@0
  1401
static dbus_bool_t
sl@0
  1402
bus_pending_reply_send_no_reply (BusConnections  *connections,
sl@0
  1403
                                 BusTransaction  *transaction,
sl@0
  1404
                                 BusPendingReply *pending)
sl@0
  1405
{
sl@0
  1406
  DBusMessage *message;
sl@0
  1407
  DBusMessageIter iter;
sl@0
  1408
  dbus_bool_t retval;
sl@0
  1409
  const char *errmsg;
sl@0
  1410
sl@0
  1411
  retval = FALSE;
sl@0
  1412
  
sl@0
  1413
  message = dbus_message_new (DBUS_MESSAGE_TYPE_ERROR);
sl@0
  1414
  if (message == NULL)
sl@0
  1415
    return FALSE;
sl@0
  1416
  
sl@0
  1417
  dbus_message_set_no_reply (message, TRUE);
sl@0
  1418
  
sl@0
  1419
  if (!dbus_message_set_reply_serial (message,
sl@0
  1420
                                      pending->reply_serial))
sl@0
  1421
    goto out;
sl@0
  1422
sl@0
  1423
  if (!dbus_message_set_error_name (message,
sl@0
  1424
                                    DBUS_ERROR_NO_REPLY))
sl@0
  1425
    goto out;
sl@0
  1426
sl@0
  1427
  errmsg = "Message did not receive a reply (timeout by message bus)";
sl@0
  1428
  dbus_message_iter_init_append (message, &iter);
sl@0
  1429
  if (!dbus_message_iter_append_basic (&iter, DBUS_TYPE_STRING, &errmsg))
sl@0
  1430
    goto out;
sl@0
  1431
    
sl@0
  1432
  if (!bus_transaction_send_from_driver (transaction, pending->will_get_reply,
sl@0
  1433
                                         message))
sl@0
  1434
    goto out;
sl@0
  1435
sl@0
  1436
  retval = TRUE;
sl@0
  1437
sl@0
  1438
 out:
sl@0
  1439
  dbus_message_unref (message);
sl@0
  1440
  return retval;
sl@0
  1441
}
sl@0
  1442
sl@0
  1443
static dbus_bool_t
sl@0
  1444
bus_pending_reply_expired (BusExpireList *list,
sl@0
  1445
                           DBusList      *link,
sl@0
  1446
                           void          *data)
sl@0
  1447
{
sl@0
  1448
  BusPendingReply *pending = link->data;
sl@0
  1449
  BusConnections *connections = data;
sl@0
  1450
  BusTransaction *transaction;
sl@0
  1451
  
sl@0
  1452
  /* No reply is forthcoming. So nuke it if we can. If not,
sl@0
  1453
   * leave it in the list to try expiring again later when we
sl@0
  1454
   * get more memory.
sl@0
  1455
   */
sl@0
  1456
sl@0
  1457
  _dbus_verbose ("Expiring pending reply %p, replier %p receiver %p serial %u\n",
sl@0
  1458
                 pending,
sl@0
  1459
                 pending->will_send_reply,
sl@0
  1460
                 pending->will_get_reply,
sl@0
  1461
                 pending->reply_serial);
sl@0
  1462
  
sl@0
  1463
  transaction = bus_transaction_new (connections->context);
sl@0
  1464
  if (transaction == NULL)
sl@0
  1465
    return FALSE;
sl@0
  1466
  
sl@0
  1467
  if (!bus_pending_reply_send_no_reply (connections,
sl@0
  1468
                                        transaction,
sl@0
  1469
                                        pending))
sl@0
  1470
    {
sl@0
  1471
      bus_transaction_cancel_and_free (transaction);
sl@0
  1472
      return FALSE;
sl@0
  1473
    }
sl@0
  1474
  
sl@0
  1475
  _dbus_list_remove_link (&connections->pending_replies->items,
sl@0
  1476
                          link);
sl@0
  1477
  bus_pending_reply_free (pending);
sl@0
  1478
  bus_transaction_execute_and_free (transaction);
sl@0
  1479
sl@0
  1480
  return TRUE;
sl@0
  1481
}
sl@0
  1482
sl@0
  1483
static void
sl@0
  1484
bus_connection_drop_pending_replies (BusConnections  *connections,
sl@0
  1485
                                     DBusConnection  *connection)
sl@0
  1486
{
sl@0
  1487
  /* The DBusConnection is almost 100% finalized here, so you can't
sl@0
  1488
   * do anything with it except check for pointer equality
sl@0
  1489
   */
sl@0
  1490
  DBusList *link;
sl@0
  1491
sl@0
  1492
  _dbus_verbose ("Dropping pending replies that involve connection %p\n",
sl@0
  1493
                 connection);
sl@0
  1494
  
sl@0
  1495
  link = _dbus_list_get_first_link (&connections->pending_replies->items);
sl@0
  1496
  while (link != NULL)
sl@0
  1497
    {
sl@0
  1498
      DBusList *next;
sl@0
  1499
      BusPendingReply *pending;
sl@0
  1500
sl@0
  1501
      next = _dbus_list_get_next_link (&connections->pending_replies->items,
sl@0
  1502
                                       link);
sl@0
  1503
      pending = link->data;
sl@0
  1504
sl@0
  1505
      if (pending->will_get_reply == connection)
sl@0
  1506
        {
sl@0
  1507
          /* We don't need to track this pending reply anymore */
sl@0
  1508
sl@0
  1509
          _dbus_verbose ("Dropping pending reply %p, replier %p receiver %p serial %u\n",
sl@0
  1510
                         pending,
sl@0
  1511
                         pending->will_send_reply,
sl@0
  1512
                         pending->will_get_reply,
sl@0
  1513
                         pending->reply_serial);
sl@0
  1514
          
sl@0
  1515
          _dbus_list_remove_link (&connections->pending_replies->items,
sl@0
  1516
                                  link);
sl@0
  1517
          bus_pending_reply_free (pending);
sl@0
  1518
        }
sl@0
  1519
      else if (pending->will_send_reply == connection)
sl@0
  1520
        {
sl@0
  1521
          /* The reply isn't going to be sent, so set things
sl@0
  1522
           * up so it will be expired right away
sl@0
  1523
           */
sl@0
  1524
          _dbus_verbose ("Will expire pending reply %p, replier %p receiver %p serial %u\n",
sl@0
  1525
                         pending,
sl@0
  1526
                         pending->will_send_reply,
sl@0
  1527
                         pending->will_get_reply,
sl@0
  1528
                         pending->reply_serial);
sl@0
  1529
          
sl@0
  1530
          pending->will_send_reply = NULL;
sl@0
  1531
          pending->expire_item.added_tv_sec = 0;
sl@0
  1532
          pending->expire_item.added_tv_usec = 0;
sl@0
  1533
sl@0
  1534
          bus_expire_timeout_set_interval (connections->pending_replies->timeout,
sl@0
  1535
                                           0);
sl@0
  1536
        }
sl@0
  1537
      
sl@0
  1538
      link = next;
sl@0
  1539
    }
sl@0
  1540
}
sl@0
  1541
sl@0
  1542
sl@0
  1543
typedef struct
sl@0
  1544
{
sl@0
  1545
  BusPendingReply *pending;
sl@0
  1546
  BusConnections  *connections;
sl@0
  1547
} CancelPendingReplyData;
sl@0
  1548
sl@0
  1549
static void
sl@0
  1550
cancel_pending_reply (void *data)
sl@0
  1551
{
sl@0
  1552
  CancelPendingReplyData *d = data;
sl@0
  1553
sl@0
  1554
  _dbus_verbose ("%s: d = %p\n", _DBUS_FUNCTION_NAME, d);
sl@0
  1555
  
sl@0
  1556
  if (!_dbus_list_remove (&d->connections->pending_replies->items,
sl@0
  1557
                          d->pending))
sl@0
  1558
    _dbus_assert_not_reached ("pending reply did not exist to be cancelled");
sl@0
  1559
sl@0
  1560
  bus_pending_reply_free (d->pending); /* since it's been cancelled */
sl@0
  1561
}
sl@0
  1562
sl@0
  1563
static void
sl@0
  1564
cancel_pending_reply_data_free (void *data)
sl@0
  1565
{
sl@0
  1566
  CancelPendingReplyData *d = data;
sl@0
  1567
sl@0
  1568
  _dbus_verbose ("%s: d = %p\n", _DBUS_FUNCTION_NAME, d);
sl@0
  1569
  
sl@0
  1570
  /* d->pending should be either freed or still
sl@0
  1571
   * in the list of pending replies (owned by someone
sl@0
  1572
   * else)
sl@0
  1573
   */
sl@0
  1574
  
sl@0
  1575
  dbus_free (d);
sl@0
  1576
}
sl@0
  1577
sl@0
  1578
/*
sl@0
  1579
 * Record that a reply is allowed; return TRUE on success.
sl@0
  1580
 */
sl@0
  1581
dbus_bool_t
sl@0
  1582
bus_connections_expect_reply (BusConnections  *connections,
sl@0
  1583
                              BusTransaction  *transaction,
sl@0
  1584
                              DBusConnection  *will_get_reply,
sl@0
  1585
                              DBusConnection  *will_send_reply,
sl@0
  1586
                              DBusMessage     *reply_to_this,
sl@0
  1587
                              DBusError       *error)
sl@0
  1588
{
sl@0
  1589
  BusPendingReply *pending;
sl@0
  1590
  dbus_uint32_t reply_serial;
sl@0
  1591
  DBusList *link;
sl@0
  1592
  CancelPendingReplyData *cprd;
sl@0
  1593
  int count;
sl@0
  1594
sl@0
  1595
  _dbus_assert (will_get_reply != NULL);
sl@0
  1596
  _dbus_assert (will_send_reply != NULL);
sl@0
  1597
  _dbus_assert (reply_to_this != NULL);
sl@0
  1598
  
sl@0
  1599
  if (dbus_message_get_no_reply (reply_to_this))
sl@0
  1600
    return TRUE; /* we won't allow a reply, since client doesn't care for one. */
sl@0
  1601
  
sl@0
  1602
  reply_serial = dbus_message_get_serial (reply_to_this);
sl@0
  1603
sl@0
  1604
  link = _dbus_list_get_first_link (&connections->pending_replies->items);
sl@0
  1605
  count = 0;
sl@0
  1606
  while (link != NULL)
sl@0
  1607
    {
sl@0
  1608
      pending = link->data;
sl@0
  1609
sl@0
  1610
      if (pending->reply_serial == reply_serial &&
sl@0
  1611
          pending->will_get_reply == will_get_reply &&
sl@0
  1612
          pending->will_send_reply == will_send_reply)
sl@0
  1613
        {
sl@0
  1614
          dbus_set_error (error, DBUS_ERROR_ACCESS_DENIED,
sl@0
  1615
                          "Message has the same reply serial as a currently-outstanding existing method call");
sl@0
  1616
          return FALSE;
sl@0
  1617
        }
sl@0
  1618
      
sl@0
  1619
      link = _dbus_list_get_next_link (&connections->pending_replies->items,
sl@0
  1620
                                       link);
sl@0
  1621
      if (pending->will_get_reply == will_get_reply)
sl@0
  1622
        ++count;
sl@0
  1623
    }
sl@0
  1624
  
sl@0
  1625
  if (count >=
sl@0
  1626
      bus_context_get_max_replies_per_connection (connections->context))
sl@0
  1627
    {
sl@0
  1628
      dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
sl@0
  1629
		      "The maximum number of pending replies per connection has been reached");
sl@0
  1630
      return FALSE;
sl@0
  1631
    }
sl@0
  1632
sl@0
  1633
  pending = dbus_new0 (BusPendingReply, 1);
sl@0
  1634
  if (pending == NULL)
sl@0
  1635
    {
sl@0
  1636
      BUS_SET_OOM (error);
sl@0
  1637
      return FALSE;
sl@0
  1638
    }
sl@0
  1639
sl@0
  1640
#ifdef DBUS_ENABLE_VERBOSE_MODE
sl@0
  1641
  /* so we can see a not-yet-added pending reply */
sl@0
  1642
  pending->expire_item.added_tv_sec = 1;
sl@0
  1643
  pending->expire_item.added_tv_usec = 1;
sl@0
  1644
#endif
sl@0
  1645
sl@0
  1646
  pending->will_get_reply = will_get_reply;
sl@0
  1647
  pending->will_send_reply = will_send_reply;
sl@0
  1648
  pending->reply_serial = reply_serial;
sl@0
  1649
  
sl@0
  1650
  cprd = dbus_new0 (CancelPendingReplyData, 1);
sl@0
  1651
  if (cprd == NULL)
sl@0
  1652
    {
sl@0
  1653
      BUS_SET_OOM (error);
sl@0
  1654
      bus_pending_reply_free (pending);
sl@0
  1655
      return FALSE;
sl@0
  1656
    }
sl@0
  1657
  
sl@0
  1658
  if (!_dbus_list_prepend (&connections->pending_replies->items,
sl@0
  1659
                           pending))
sl@0
  1660
    {
sl@0
  1661
      BUS_SET_OOM (error);
sl@0
  1662
      dbus_free (cprd);
sl@0
  1663
      bus_pending_reply_free (pending);
sl@0
  1664
      return FALSE;
sl@0
  1665
    }
sl@0
  1666
sl@0
  1667
  if (!bus_transaction_add_cancel_hook (transaction,
sl@0
  1668
                                        cancel_pending_reply,
sl@0
  1669
                                        cprd,
sl@0
  1670
                                        cancel_pending_reply_data_free))
sl@0
  1671
    {
sl@0
  1672
      BUS_SET_OOM (error);
sl@0
  1673
      _dbus_list_remove (&connections->pending_replies->items, pending);
sl@0
  1674
      dbus_free (cprd);
sl@0
  1675
      bus_pending_reply_free (pending);
sl@0
  1676
      return FALSE;
sl@0
  1677
    }
sl@0
  1678
                                        
sl@0
  1679
  cprd->pending = pending;
sl@0
  1680
  cprd->connections = connections;
sl@0
  1681
  
sl@0
  1682
  _dbus_get_current_time (&pending->expire_item.added_tv_sec,
sl@0
  1683
                          &pending->expire_item.added_tv_usec);
sl@0
  1684
sl@0
  1685
  _dbus_verbose ("Added pending reply %p, replier %p receiver %p serial %u\n",
sl@0
  1686
                 pending,
sl@0
  1687
                 pending->will_send_reply,
sl@0
  1688
                 pending->will_get_reply,
sl@0
  1689
                 pending->reply_serial);
sl@0
  1690
  
sl@0
  1691
  return TRUE;
sl@0
  1692
}
sl@0
  1693
sl@0
  1694
typedef struct
sl@0
  1695
{
sl@0
  1696
  DBusList        *link;
sl@0
  1697
  BusConnections  *connections;
sl@0
  1698
} CheckPendingReplyData;
sl@0
  1699
sl@0
  1700
static void
sl@0
  1701
cancel_check_pending_reply (void *data)
sl@0
  1702
{
sl@0
  1703
  CheckPendingReplyData *d = data;
sl@0
  1704
sl@0
  1705
  _dbus_verbose ("%s: d = %p\n", _DBUS_FUNCTION_NAME, d);
sl@0
  1706
  
sl@0
  1707
  _dbus_list_prepend_link (&d->connections->pending_replies->items,
sl@0
  1708
                           d->link);
sl@0
  1709
  d->link = NULL;
sl@0
  1710
}
sl@0
  1711
sl@0
  1712
static void
sl@0
  1713
check_pending_reply_data_free (void *data)
sl@0
  1714
{
sl@0
  1715
  CheckPendingReplyData *d = data;
sl@0
  1716
sl@0
  1717
  _dbus_verbose ("%s: d = %p\n", _DBUS_FUNCTION_NAME, d);
sl@0
  1718
  
sl@0
  1719
  if (d->link != NULL)
sl@0
  1720
    {
sl@0
  1721
      BusPendingReply *pending = d->link->data;
sl@0
  1722
      
sl@0
  1723
      _dbus_assert (_dbus_list_find_last (&d->connections->pending_replies->items,
sl@0
  1724
                                          pending) == NULL);
sl@0
  1725
      
sl@0
  1726
      bus_pending_reply_free (pending);
sl@0
  1727
      _dbus_list_free_link (d->link);
sl@0
  1728
    }
sl@0
  1729
  
sl@0
  1730
  dbus_free (d);
sl@0
  1731
}
sl@0
  1732
sl@0
  1733
/*
sl@0
  1734
 * Check whether a reply is allowed, remove BusPendingReply
sl@0
  1735
 * if so, return TRUE if so.
sl@0
  1736
 */
sl@0
  1737
dbus_bool_t
sl@0
  1738
bus_connections_check_reply (BusConnections *connections,
sl@0
  1739
                             BusTransaction *transaction,
sl@0
  1740
                             DBusConnection *sending_reply,
sl@0
  1741
                             DBusConnection *receiving_reply,
sl@0
  1742
                             DBusMessage    *reply,
sl@0
  1743
                             DBusError      *error)
sl@0
  1744
{
sl@0
  1745
  CheckPendingReplyData *cprd;
sl@0
  1746
  DBusList *link;
sl@0
  1747
  dbus_uint32_t reply_serial;
sl@0
  1748
  
sl@0
  1749
  _dbus_assert (sending_reply != NULL);
sl@0
  1750
  _dbus_assert (receiving_reply != NULL);
sl@0
  1751
sl@0
  1752
  reply_serial = dbus_message_get_reply_serial (reply);
sl@0
  1753
sl@0
  1754
  link = _dbus_list_get_first_link (&connections->pending_replies->items);
sl@0
  1755
  while (link != NULL)
sl@0
  1756
    {
sl@0
  1757
      BusPendingReply *pending = link->data;
sl@0
  1758
sl@0
  1759
      if (pending->reply_serial == reply_serial &&
sl@0
  1760
          pending->will_get_reply == receiving_reply &&
sl@0
  1761
          pending->will_send_reply == sending_reply)
sl@0
  1762
        {
sl@0
  1763
          _dbus_verbose ("Found pending reply with serial %u\n", reply_serial);
sl@0
  1764
          break;
sl@0
  1765
        }
sl@0
  1766
      
sl@0
  1767
      link = _dbus_list_get_next_link (&connections->pending_replies->items,
sl@0
  1768
                                       link);
sl@0
  1769
    }
sl@0
  1770
sl@0
  1771
  if (link == NULL)
sl@0
  1772
    {
sl@0
  1773
      _dbus_verbose ("No pending reply expected\n");
sl@0
  1774
sl@0
  1775
      return FALSE;
sl@0
  1776
    }
sl@0
  1777
sl@0
  1778
  cprd = dbus_new0 (CheckPendingReplyData, 1);
sl@0
  1779
  if (cprd == NULL)
sl@0
  1780
    {
sl@0
  1781
      BUS_SET_OOM (error);
sl@0
  1782
      return FALSE;
sl@0
  1783
    }
sl@0
  1784
  
sl@0
  1785
  if (!bus_transaction_add_cancel_hook (transaction,
sl@0
  1786
                                        cancel_check_pending_reply,
sl@0
  1787
                                        cprd,
sl@0
  1788
                                        check_pending_reply_data_free))
sl@0
  1789
    {
sl@0
  1790
      BUS_SET_OOM (error);
sl@0
  1791
      dbus_free (cprd);
sl@0
  1792
      return FALSE;
sl@0
  1793
    }
sl@0
  1794
sl@0
  1795
  cprd->link = link;
sl@0
  1796
  cprd->connections = connections;
sl@0
  1797
  
sl@0
  1798
  _dbus_list_unlink (&connections->pending_replies->items,
sl@0
  1799
                     link);
sl@0
  1800
  
sl@0
  1801
  _dbus_assert (_dbus_list_find_last (&connections->pending_replies->items,
sl@0
  1802
                                      link->data) == NULL);
sl@0
  1803
sl@0
  1804
  return TRUE;
sl@0
  1805
}
sl@0
  1806
sl@0
  1807
/*
sl@0
  1808
 * Transactions
sl@0
  1809
 *
sl@0
  1810
 * Note that this is fairly fragile; in particular, don't try to use
sl@0
  1811
 * one transaction across any main loop iterations.
sl@0
  1812
 */
sl@0
  1813
sl@0
  1814
typedef struct
sl@0
  1815
{
sl@0
  1816
  BusTransaction *transaction;
sl@0
  1817
  DBusMessage    *message;
sl@0
  1818
  DBusPreallocatedSend *preallocated;
sl@0
  1819
} MessageToSend;
sl@0
  1820
sl@0
  1821
typedef struct
sl@0
  1822
{
sl@0
  1823
  BusTransactionCancelFunction cancel_function;
sl@0
  1824
  DBusFreeFunction free_data_function;
sl@0
  1825
  void *data;
sl@0
  1826
} CancelHook;
sl@0
  1827
sl@0
  1828
struct BusTransaction
sl@0
  1829
{
sl@0
  1830
  DBusList *connections;
sl@0
  1831
  BusContext *context;
sl@0
  1832
  DBusList *cancel_hooks;
sl@0
  1833
};
sl@0
  1834
sl@0
  1835
static void
sl@0
  1836
message_to_send_free (DBusConnection *connection,
sl@0
  1837
                      MessageToSend  *to_send)
sl@0
  1838
{
sl@0
  1839
  if (to_send->message)
sl@0
  1840
    dbus_message_unref (to_send->message);
sl@0
  1841
sl@0
  1842
  if (to_send->preallocated)
sl@0
  1843
    dbus_connection_free_preallocated_send (connection, to_send->preallocated);
sl@0
  1844
sl@0
  1845
  dbus_free (to_send);
sl@0
  1846
}
sl@0
  1847
sl@0
  1848
static void
sl@0
  1849
cancel_hook_cancel (void *element,
sl@0
  1850
                    void *data)
sl@0
  1851
{
sl@0
  1852
  CancelHook *ch = element;
sl@0
  1853
sl@0
  1854
  _dbus_verbose ("Running transaction cancel hook\n");
sl@0
  1855
  
sl@0
  1856
  if (ch->cancel_function)
sl@0
  1857
    (* ch->cancel_function) (ch->data);  
sl@0
  1858
}
sl@0
  1859
sl@0
  1860
static void
sl@0
  1861
cancel_hook_free (void *element,
sl@0
  1862
                  void *data)
sl@0
  1863
{
sl@0
  1864
  CancelHook *ch = element;
sl@0
  1865
sl@0
  1866
  if (ch->free_data_function)
sl@0
  1867
    (* ch->free_data_function) (ch->data);
sl@0
  1868
sl@0
  1869
  dbus_free (ch);
sl@0
  1870
}
sl@0
  1871
sl@0
  1872
static void
sl@0
  1873
free_cancel_hooks (BusTransaction *transaction)
sl@0
  1874
{
sl@0
  1875
  _dbus_list_foreach (&transaction->cancel_hooks,
sl@0
  1876
                      cancel_hook_free, NULL);
sl@0
  1877
  
sl@0
  1878
  _dbus_list_clear (&transaction->cancel_hooks);
sl@0
  1879
}
sl@0
  1880
sl@0
  1881
BusTransaction*
sl@0
  1882
bus_transaction_new (BusContext *context)
sl@0
  1883
{
sl@0
  1884
  BusTransaction *transaction;
sl@0
  1885
sl@0
  1886
  transaction = dbus_new0 (BusTransaction, 1);
sl@0
  1887
  if (transaction == NULL)
sl@0
  1888
    return NULL;
sl@0
  1889
sl@0
  1890
  transaction->context = context;
sl@0
  1891
  
sl@0
  1892
  return transaction;
sl@0
  1893
}
sl@0
  1894
sl@0
  1895
BusContext*
sl@0
  1896
bus_transaction_get_context (BusTransaction  *transaction)
sl@0
  1897
{
sl@0
  1898
  return transaction->context;
sl@0
  1899
}
sl@0
  1900
sl@0
  1901
BusConnections*
sl@0
  1902
bus_transaction_get_connections (BusTransaction  *transaction)
sl@0
  1903
{
sl@0
  1904
  return bus_context_get_connections (transaction->context);
sl@0
  1905
}
sl@0
  1906
sl@0
  1907
dbus_bool_t
sl@0
  1908
bus_transaction_send_from_driver (BusTransaction *transaction,
sl@0
  1909
                                  DBusConnection *connection,
sl@0
  1910
                                  DBusMessage    *message)
sl@0
  1911
{
sl@0
  1912
  /* We have to set the sender to the driver, and have
sl@0
  1913
   * to check security policy since it was not done in
sl@0
  1914
   * dispatch.c
sl@0
  1915
   */
sl@0
  1916
  _dbus_verbose ("Sending %s %s %s from driver\n",
sl@0
  1917
                 dbus_message_get_interface (message) ?
sl@0
  1918
                 dbus_message_get_interface (message) : "(no interface)",
sl@0
  1919
                 dbus_message_get_member (message) ?
sl@0
  1920
                 dbus_message_get_member (message) : "(no member)",
sl@0
  1921
                 dbus_message_get_error_name (message) ?
sl@0
  1922
                 dbus_message_get_error_name (message) : "(no error name)");
sl@0
  1923
                 
sl@0
  1924
  if (!dbus_message_set_sender (message, DBUS_SERVICE_DBUS))
sl@0
  1925
    return FALSE;
sl@0
  1926
sl@0
  1927
  if (bus_connection_is_active (connection))
sl@0
  1928
    {
sl@0
  1929
      if (!dbus_message_set_destination (message,
sl@0
  1930
                                         bus_connection_get_name (connection)))
sl@0
  1931
        return FALSE;
sl@0
  1932
    }
sl@0
  1933
  
sl@0
  1934
  /* bus driver never wants a reply */
sl@0
  1935
  dbus_message_set_no_reply (message, TRUE);
sl@0
  1936
  
sl@0
  1937
  /* If security policy doesn't allow the message, we silently
sl@0
  1938
   * eat it; the driver doesn't care about getting a reply.
sl@0
  1939
   */
sl@0
  1940
  if (!bus_context_check_security_policy (bus_transaction_get_context (transaction),
sl@0
  1941
                                          transaction,
sl@0
  1942
                                          NULL, connection, connection, message, NULL))
sl@0
  1943
    return TRUE;
sl@0
  1944
sl@0
  1945
  return bus_transaction_send (transaction, connection, message);
sl@0
  1946
}
sl@0
  1947
sl@0
  1948
dbus_bool_t
sl@0
  1949
bus_transaction_send (BusTransaction *transaction,
sl@0
  1950
                      DBusConnection *connection,
sl@0
  1951
                      DBusMessage    *message)
sl@0
  1952
{
sl@0
  1953
  MessageToSend *to_send;
sl@0
  1954
  BusConnectionData *d;
sl@0
  1955
  DBusList *link;
sl@0
  1956
sl@0
  1957
  _dbus_verbose ("  trying to add %s interface=%s member=%s error=%s to transaction%s\n",
sl@0
  1958
                 dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR ? "error" :
sl@0
  1959
                 dbus_message_get_reply_serial (message) != 0 ? "reply" :
sl@0
  1960
                 "message",
sl@0
  1961
                 dbus_message_get_interface (message) ?
sl@0
  1962
                 dbus_message_get_interface (message) : "(unset)",
sl@0
  1963
                 dbus_message_get_member (message) ?
sl@0
  1964
                 dbus_message_get_member (message) : "(unset)",
sl@0
  1965
                 dbus_message_get_error_name (message) ?
sl@0
  1966
                 dbus_message_get_error_name (message) : "(unset)",
sl@0
  1967
                 dbus_connection_get_is_connected (connection) ?
sl@0
  1968
                 "" : " (disconnected)");
sl@0
  1969
sl@0
  1970
  _dbus_assert (dbus_message_get_sender (message) != NULL);
sl@0
  1971
  
sl@0
  1972
  if (!dbus_connection_get_is_connected (connection))
sl@0
  1973
    return TRUE; /* silently ignore disconnected connections */
sl@0
  1974
  
sl@0
  1975
  d = BUS_CONNECTION_DATA (connection);
sl@0
  1976
  _dbus_assert (d != NULL);
sl@0
  1977
  
sl@0
  1978
  to_send = dbus_new (MessageToSend, 1);
sl@0
  1979
  if (to_send == NULL)
sl@0
  1980
    {
sl@0
  1981
      return FALSE;
sl@0
  1982
    }
sl@0
  1983
sl@0
  1984
  to_send->preallocated = dbus_connection_preallocate_send (connection);
sl@0
  1985
  if (to_send->preallocated == NULL)
sl@0
  1986
    {
sl@0
  1987
      dbus_free (to_send);
sl@0
  1988
      return FALSE;
sl@0
  1989
    }  
sl@0
  1990
  
sl@0
  1991
  dbus_message_ref (message);
sl@0
  1992
  to_send->message = message;
sl@0
  1993
  to_send->transaction = transaction;
sl@0
  1994
sl@0
  1995
  _dbus_verbose ("about to prepend message\n");
sl@0
  1996
  
sl@0
  1997
  if (!_dbus_list_prepend (&d->transaction_messages, to_send))
sl@0
  1998
    {
sl@0
  1999
      message_to_send_free (connection, to_send);
sl@0
  2000
      return FALSE;
sl@0
  2001
    }
sl@0
  2002
sl@0
  2003
  _dbus_verbose ("prepended message\n");
sl@0
  2004
  
sl@0
  2005
  /* See if we already had this connection in the list
sl@0
  2006
   * for this transaction. If we have a pending message,
sl@0
  2007
   * then we should already be in transaction->connections
sl@0
  2008
   */
sl@0
  2009
  link = _dbus_list_get_first_link (&d->transaction_messages);
sl@0
  2010
  _dbus_assert (link->data == to_send);
sl@0
  2011
  link = _dbus_list_get_next_link (&d->transaction_messages, link);
sl@0
  2012
  while (link != NULL)
sl@0
  2013
    {
sl@0
  2014
      MessageToSend *m = link->data;
sl@0
  2015
      DBusList *next = _dbus_list_get_next_link (&d->transaction_messages, link);
sl@0
  2016
      
sl@0
  2017
      if (m->transaction == transaction)
sl@0
  2018
        break;
sl@0
  2019
        
sl@0
  2020
      link = next;
sl@0
  2021
    }
sl@0
  2022
sl@0
  2023
  if (link == NULL)
sl@0
  2024
    {
sl@0
  2025
      if (!_dbus_list_prepend (&transaction->connections, connection))
sl@0
  2026
        {
sl@0
  2027
          _dbus_list_remove (&d->transaction_messages, to_send);
sl@0
  2028
          message_to_send_free (connection, to_send);
sl@0
  2029
          return FALSE;
sl@0
  2030
        }
sl@0
  2031
    }
sl@0
  2032
sl@0
  2033
  return TRUE;
sl@0
  2034
}
sl@0
  2035
sl@0
  2036
static void
sl@0
  2037
connection_cancel_transaction (DBusConnection *connection,
sl@0
  2038
                               BusTransaction *transaction)
sl@0
  2039
{
sl@0
  2040
  DBusList *link;
sl@0
  2041
  BusConnectionData *d;
sl@0
  2042
  
sl@0
  2043
  d = BUS_CONNECTION_DATA (connection);
sl@0
  2044
  _dbus_assert (d != NULL);
sl@0
  2045
  
sl@0
  2046
  link = _dbus_list_get_first_link (&d->transaction_messages);
sl@0
  2047
  while (link != NULL)
sl@0
  2048
    {
sl@0
  2049
      MessageToSend *m = link->data;
sl@0
  2050
      DBusList *next = _dbus_list_get_next_link (&d->transaction_messages, link);
sl@0
  2051
      
sl@0
  2052
      if (m->transaction == transaction)
sl@0
  2053
        {
sl@0
  2054
          _dbus_list_remove_link (&d->transaction_messages,
sl@0
  2055
                                  link);
sl@0
  2056
          
sl@0
  2057
          message_to_send_free (connection, m);
sl@0
  2058
        }
sl@0
  2059
        
sl@0
  2060
      link = next;
sl@0
  2061
    }
sl@0
  2062
}
sl@0
  2063
sl@0
  2064
void
sl@0
  2065
bus_transaction_cancel_and_free (BusTransaction *transaction)
sl@0
  2066
{
sl@0
  2067
  DBusConnection *connection;
sl@0
  2068
sl@0
  2069
  _dbus_verbose ("TRANSACTION: cancelled\n");
sl@0
  2070
  
sl@0
  2071
  while ((connection = _dbus_list_pop_first (&transaction->connections)))
sl@0
  2072
    connection_cancel_transaction (connection, transaction);
sl@0
  2073
sl@0
  2074
  _dbus_assert (transaction->connections == NULL);
sl@0
  2075
sl@0
  2076
  _dbus_list_foreach (&transaction->cancel_hooks,
sl@0
  2077
                      cancel_hook_cancel, NULL);
sl@0
  2078
sl@0
  2079
  free_cancel_hooks (transaction);
sl@0
  2080
  
sl@0
  2081
  dbus_free (transaction);
sl@0
  2082
}
sl@0
  2083
sl@0
  2084
static void
sl@0
  2085
connection_execute_transaction (DBusConnection *connection,
sl@0
  2086
                                BusTransaction *transaction)
sl@0
  2087
{
sl@0
  2088
  DBusList *link;
sl@0
  2089
  BusConnectionData *d;
sl@0
  2090
  
sl@0
  2091
  d = BUS_CONNECTION_DATA (connection);
sl@0
  2092
  _dbus_assert (d != NULL);
sl@0
  2093
sl@0
  2094
  /* Send the queue in order (FIFO) */
sl@0
  2095
  link = _dbus_list_get_last_link (&d->transaction_messages);
sl@0
  2096
  while (link != NULL)
sl@0
  2097
    {
sl@0
  2098
      MessageToSend *m = link->data;
sl@0
  2099
      DBusList *prev = _dbus_list_get_prev_link (&d->transaction_messages, link);
sl@0
  2100
      
sl@0
  2101
      if (m->transaction == transaction)
sl@0
  2102
        {
sl@0
  2103
          _dbus_list_remove_link (&d->transaction_messages,
sl@0
  2104
                                  link);
sl@0
  2105
sl@0
  2106
          _dbus_assert (dbus_message_get_sender (m->message) != NULL);
sl@0
  2107
          
sl@0
  2108
          dbus_connection_send_preallocated (connection,
sl@0
  2109
                                             m->preallocated,
sl@0
  2110
                                             m->message,
sl@0
  2111
                                             NULL);
sl@0
  2112
sl@0
  2113
          m->preallocated = NULL; /* so we don't double-free it */
sl@0
  2114
          
sl@0
  2115
          message_to_send_free (connection, m);
sl@0
  2116
        }
sl@0
  2117
        
sl@0
  2118
      link = prev;
sl@0
  2119
    }
sl@0
  2120
}
sl@0
  2121
sl@0
  2122
void
sl@0
  2123
bus_transaction_execute_and_free (BusTransaction *transaction)
sl@0
  2124
{
sl@0
  2125
  /* For each connection in transaction->connections
sl@0
  2126
   * send the messages
sl@0
  2127
   */
sl@0
  2128
  DBusConnection *connection;
sl@0
  2129
sl@0
  2130
  _dbus_verbose ("TRANSACTION: executing\n");
sl@0
  2131
  
sl@0
  2132
  while ((connection = _dbus_list_pop_first (&transaction->connections)))
sl@0
  2133
    connection_execute_transaction (connection, transaction);
sl@0
  2134
sl@0
  2135
  _dbus_assert (transaction->connections == NULL);
sl@0
  2136
sl@0
  2137
  free_cancel_hooks (transaction);
sl@0
  2138
  
sl@0
  2139
  dbus_free (transaction);
sl@0
  2140
}
sl@0
  2141
sl@0
  2142
static void
sl@0
  2143
bus_connection_remove_transactions (DBusConnection *connection)
sl@0
  2144
{
sl@0
  2145
  MessageToSend *to_send;
sl@0
  2146
  BusConnectionData *d;
sl@0
  2147
  
sl@0
  2148
  d = BUS_CONNECTION_DATA (connection);
sl@0
  2149
  _dbus_assert (d != NULL);
sl@0
  2150
  
sl@0
  2151
  while ((to_send = _dbus_list_get_first (&d->transaction_messages)))
sl@0
  2152
    {
sl@0
  2153
      /* only has an effect for the first MessageToSend listing this transaction */
sl@0
  2154
      _dbus_list_remove (&to_send->transaction->connections,
sl@0
  2155
                         connection);
sl@0
  2156
sl@0
  2157
      _dbus_list_remove (&d->transaction_messages, to_send);
sl@0
  2158
      message_to_send_free (connection, to_send);
sl@0
  2159
    }
sl@0
  2160
}
sl@0
  2161
sl@0
  2162
/**
sl@0
  2163
 * Converts the DBusError to a message reply
sl@0
  2164
 */
sl@0
  2165
dbus_bool_t
sl@0
  2166
bus_transaction_send_error_reply (BusTransaction  *transaction,
sl@0
  2167
                                  DBusConnection  *connection,
sl@0
  2168
                                  const DBusError *error,
sl@0
  2169
                                  DBusMessage     *in_reply_to)
sl@0
  2170
{
sl@0
  2171
  DBusMessage *reply;
sl@0
  2172
  
sl@0
  2173
  _dbus_assert (error != NULL);
sl@0
  2174
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  2175
  
sl@0
  2176
  _dbus_verbose ("Sending error reply %s \"%s\"\n",
sl@0
  2177
                 error->name, error->message);
sl@0
  2178
sl@0
  2179
  reply = dbus_message_new_error (in_reply_to,
sl@0
  2180
                                  error->name,
sl@0
  2181
                                  error->message);
sl@0
  2182
  if (reply == NULL)
sl@0
  2183
    return FALSE;
sl@0
  2184
sl@0
  2185
  if (!bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  2186
    {
sl@0
  2187
      dbus_message_unref (reply);
sl@0
  2188
      return FALSE;
sl@0
  2189
    }
sl@0
  2190
sl@0
  2191
  dbus_message_unref (reply);
sl@0
  2192
  
sl@0
  2193
  return TRUE;
sl@0
  2194
}
sl@0
  2195
sl@0
  2196
dbus_bool_t
sl@0
  2197
bus_transaction_add_cancel_hook (BusTransaction               *transaction,
sl@0
  2198
                                 BusTransactionCancelFunction  cancel_function,
sl@0
  2199
                                 void                         *data,
sl@0
  2200
                                 DBusFreeFunction              free_data_function)
sl@0
  2201
{
sl@0
  2202
  CancelHook *ch;
sl@0
  2203
sl@0
  2204
  ch = dbus_new (CancelHook, 1);
sl@0
  2205
  if (ch == NULL)
sl@0
  2206
    return FALSE;
sl@0
  2207
sl@0
  2208
  _dbus_verbose ("     adding cancel hook function = %p data = %p\n",
sl@0
  2209
                 cancel_function, data);
sl@0
  2210
  
sl@0
  2211
  ch->cancel_function = cancel_function;
sl@0
  2212
  ch->data = data;
sl@0
  2213
  ch->free_data_function = free_data_function;
sl@0
  2214
sl@0
  2215
  /* It's important that the hooks get run in reverse order that they
sl@0
  2216
   * were added
sl@0
  2217
   */
sl@0
  2218
  if (!_dbus_list_prepend (&transaction->cancel_hooks, ch))
sl@0
  2219
    {
sl@0
  2220
      dbus_free (ch);
sl@0
  2221
      return FALSE;
sl@0
  2222
    }
sl@0
  2223
sl@0
  2224
  return TRUE;
sl@0
  2225
}