os/ossrv/ofdbus/dbus/bus/driver.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
/* driver.c  Bus client (driver)
sl@0
     3
 *
sl@0
     4
 * Copyright (C) 2003 CodeFactory AB
sl@0
     5
 * Copyright (C) 2003, 2004, 2005 Red Hat, Inc.
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
sl@0
    25
#include "activation.h"
sl@0
    26
#include "connection.h"
sl@0
    27
#include "driver.h"
sl@0
    28
#include "dispatch.h"
sl@0
    29
#include "services.h"
sl@0
    30
#include "selinux.h"
sl@0
    31
#include "signals.h"
sl@0
    32
#include "utils.h"
sl@0
    33
#ifndef __SYMBIAN32__
sl@0
    34
#include <dbus/dbus-string.h>
sl@0
    35
#include <dbus/dbus-internals.h>
sl@0
    36
#include <dbus/dbus-marshal-recursive.h>
sl@0
    37
#else
sl@0
    38
#include "dbus-string.h"
sl@0
    39
#include "dbus-internals.h"
sl@0
    40
#include "dbus-marshal-recursive.h"
sl@0
    41
#endif //__SYMBIAN32__
sl@0
    42
#include <string.h>
sl@0
    43
sl@0
    44
static dbus_bool_t bus_driver_send_welcome_message (DBusConnection *connection,
sl@0
    45
                                                    DBusMessage    *hello_message,
sl@0
    46
                                                    BusTransaction *transaction,
sl@0
    47
                                                    DBusError      *error);
sl@0
    48
sl@0
    49
dbus_bool_t
sl@0
    50
bus_driver_send_service_owner_changed (const char     *service_name,
sl@0
    51
				       const char     *old_owner,
sl@0
    52
				       const char     *new_owner,
sl@0
    53
				       BusTransaction *transaction,
sl@0
    54
				       DBusError      *error)
sl@0
    55
{
sl@0
    56
  DBusMessage *message;
sl@0
    57
  dbus_bool_t retval;
sl@0
    58
  const char *null_service;
sl@0
    59
sl@0
    60
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
    61
sl@0
    62
  null_service = "";
sl@0
    63
  _dbus_verbose ("sending name owner changed: %s [%s -> %s]\n",
sl@0
    64
                 service_name, 
sl@0
    65
                 old_owner ? old_owner : null_service, 
sl@0
    66
                 new_owner ? new_owner : null_service);
sl@0
    67
sl@0
    68
  message = dbus_message_new_signal (DBUS_PATH_DBUS,
sl@0
    69
                                     DBUS_INTERFACE_DBUS,
sl@0
    70
                                     "NameOwnerChanged");
sl@0
    71
  
sl@0
    72
  if (message == NULL)
sl@0
    73
    {
sl@0
    74
      BUS_SET_OOM (error);
sl@0
    75
      return FALSE;
sl@0
    76
    }
sl@0
    77
  
sl@0
    78
  if (!dbus_message_set_sender (message, DBUS_SERVICE_DBUS))
sl@0
    79
    goto oom;
sl@0
    80
sl@0
    81
  if (!dbus_message_append_args (message,
sl@0
    82
                                 DBUS_TYPE_STRING, &service_name,
sl@0
    83
                                 DBUS_TYPE_STRING, old_owner ? &old_owner : &null_service,
sl@0
    84
                                 DBUS_TYPE_STRING, new_owner ? &new_owner : &null_service,
sl@0
    85
                                 DBUS_TYPE_INVALID))
sl@0
    86
    goto oom;
sl@0
    87
sl@0
    88
  _dbus_assert (dbus_message_has_signature (message, "sss"));
sl@0
    89
  
sl@0
    90
  retval = bus_dispatch_matches (transaction, NULL, NULL, message, error);
sl@0
    91
  dbus_message_unref (message);
sl@0
    92
sl@0
    93
  return retval;
sl@0
    94
sl@0
    95
 oom:
sl@0
    96
  dbus_message_unref (message);
sl@0
    97
  BUS_SET_OOM (error);
sl@0
    98
  return FALSE;
sl@0
    99
}
sl@0
   100
sl@0
   101
dbus_bool_t
sl@0
   102
bus_driver_send_service_lost (DBusConnection *connection,
sl@0
   103
			      const char     *service_name,
sl@0
   104
                              BusTransaction *transaction,
sl@0
   105
                              DBusError      *error)
sl@0
   106
{
sl@0
   107
  DBusMessage *message;
sl@0
   108
sl@0
   109
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   110
  
sl@0
   111
  message = dbus_message_new_signal (DBUS_PATH_DBUS,
sl@0
   112
                                     DBUS_INTERFACE_DBUS,
sl@0
   113
                                     "NameLost");
sl@0
   114
  
sl@0
   115
  if (message == NULL)
sl@0
   116
    {
sl@0
   117
      BUS_SET_OOM (error);
sl@0
   118
      return FALSE;
sl@0
   119
    }
sl@0
   120
  
sl@0
   121
  if (!dbus_message_set_destination (message, bus_connection_get_name (connection)) ||
sl@0
   122
      !dbus_message_append_args (message,
sl@0
   123
                                 DBUS_TYPE_STRING, &service_name,
sl@0
   124
                                 DBUS_TYPE_INVALID))
sl@0
   125
    {
sl@0
   126
      dbus_message_unref (message);
sl@0
   127
      BUS_SET_OOM (error);
sl@0
   128
      return FALSE;
sl@0
   129
    }
sl@0
   130
sl@0
   131
  if (!bus_transaction_send_from_driver (transaction, connection, message))
sl@0
   132
    {
sl@0
   133
      dbus_message_unref (message);
sl@0
   134
      BUS_SET_OOM (error);
sl@0
   135
      return FALSE;
sl@0
   136
    }
sl@0
   137
  else
sl@0
   138
    {
sl@0
   139
      dbus_message_unref (message);
sl@0
   140
      return TRUE;
sl@0
   141
    }
sl@0
   142
}
sl@0
   143
sl@0
   144
dbus_bool_t
sl@0
   145
bus_driver_send_service_acquired (DBusConnection *connection,
sl@0
   146
                                  const char     *service_name,
sl@0
   147
                                  BusTransaction *transaction,
sl@0
   148
                                  DBusError      *error)
sl@0
   149
{
sl@0
   150
  DBusMessage *message;
sl@0
   151
sl@0
   152
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   153
  
sl@0
   154
  message = dbus_message_new_signal (DBUS_PATH_DBUS,
sl@0
   155
                                     DBUS_INTERFACE_DBUS,
sl@0
   156
                                     "NameAcquired");
sl@0
   157
sl@0
   158
  if (message == NULL)
sl@0
   159
    {
sl@0
   160
      BUS_SET_OOM (error);
sl@0
   161
      return FALSE;
sl@0
   162
    }
sl@0
   163
  
sl@0
   164
  if (!dbus_message_set_destination (message, bus_connection_get_name (connection)) ||
sl@0
   165
      !dbus_message_append_args (message,
sl@0
   166
                                 DBUS_TYPE_STRING, &service_name,
sl@0
   167
                                 DBUS_TYPE_INVALID))
sl@0
   168
    {
sl@0
   169
      dbus_message_unref (message);
sl@0
   170
      BUS_SET_OOM (error);
sl@0
   171
      return FALSE;
sl@0
   172
    }
sl@0
   173
sl@0
   174
  if (!bus_transaction_send_from_driver (transaction, connection, message))
sl@0
   175
    {
sl@0
   176
      dbus_message_unref (message);
sl@0
   177
      BUS_SET_OOM (error);
sl@0
   178
      return FALSE;
sl@0
   179
    }
sl@0
   180
  else
sl@0
   181
    {
sl@0
   182
      dbus_message_unref (message);
sl@0
   183
      return TRUE;
sl@0
   184
    }
sl@0
   185
}
sl@0
   186
sl@0
   187
static dbus_bool_t
sl@0
   188
create_unique_client_name (BusRegistry *registry,
sl@0
   189
                           DBusString  *str)
sl@0
   190
{
sl@0
   191
  /* We never want to use the same unique client name twice, because
sl@0
   192
   * we want to guarantee that if you send a message to a given unique
sl@0
   193
   * name, you always get the same application. So we use two numbers
sl@0
   194
   * for INT_MAX * INT_MAX combinations, should be pretty safe against
sl@0
   195
   * wraparound.
sl@0
   196
   */
sl@0
   197
  /* FIXME these should be in BusRegistry rather than static vars */
sl@0
   198
  static int next_major_number = 0;
sl@0
   199
  static int next_minor_number = 0;
sl@0
   200
  int len;
sl@0
   201
  
sl@0
   202
  len = _dbus_string_get_length (str);
sl@0
   203
  
sl@0
   204
  while (TRUE)
sl@0
   205
    {
sl@0
   206
      /* start out with 1-0, go to 1-1, 1-2, 1-3,
sl@0
   207
       * up to 1-MAXINT, then 2-0, 2-1, etc.
sl@0
   208
       */
sl@0
   209
      if (next_minor_number <= 0)
sl@0
   210
        {
sl@0
   211
          next_major_number += 1;
sl@0
   212
          next_minor_number = 0;
sl@0
   213
          if (next_major_number <= 0)
sl@0
   214
            _dbus_assert_not_reached ("INT_MAX * INT_MAX clients were added");
sl@0
   215
        }
sl@0
   216
sl@0
   217
      _dbus_assert (next_major_number > 0);
sl@0
   218
      _dbus_assert (next_minor_number >= 0);
sl@0
   219
sl@0
   220
      /* appname:MAJOR-MINOR */
sl@0
   221
      
sl@0
   222
      if (!_dbus_string_append (str, ":"))
sl@0
   223
        return FALSE;
sl@0
   224
      
sl@0
   225
      if (!_dbus_string_append_int (str, next_major_number))
sl@0
   226
        return FALSE;
sl@0
   227
sl@0
   228
      if (!_dbus_string_append (str, "."))
sl@0
   229
        return FALSE;
sl@0
   230
      
sl@0
   231
      if (!_dbus_string_append_int (str, next_minor_number))
sl@0
   232
        return FALSE;
sl@0
   233
sl@0
   234
      next_minor_number += 1;
sl@0
   235
      
sl@0
   236
      /* Check if a client with the name exists */
sl@0
   237
      if (bus_registry_lookup (registry, str) == NULL)
sl@0
   238
	break;
sl@0
   239
sl@0
   240
      /* drop the number again, try the next one. */
sl@0
   241
      _dbus_string_set_length (str, len);
sl@0
   242
    }
sl@0
   243
sl@0
   244
  return TRUE;
sl@0
   245
}
sl@0
   246
sl@0
   247
static dbus_bool_t
sl@0
   248
bus_driver_handle_hello (DBusConnection *connection,
sl@0
   249
                         BusTransaction *transaction,
sl@0
   250
                         DBusMessage    *message,
sl@0
   251
                         DBusError      *error)
sl@0
   252
{
sl@0
   253
  DBusString unique_name;
sl@0
   254
  BusService *service;
sl@0
   255
  dbus_bool_t retval;
sl@0
   256
  BusRegistry *registry;
sl@0
   257
  BusConnections *connections;
sl@0
   258
sl@0
   259
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   260
sl@0
   261
  if (bus_connection_is_active (connection))
sl@0
   262
    {
sl@0
   263
      /* We already handled an Hello message for this connection. */
sl@0
   264
      dbus_set_error (error, DBUS_ERROR_FAILED,
sl@0
   265
                      "Already handled an Hello message");
sl@0
   266
      return FALSE;
sl@0
   267
    }
sl@0
   268
sl@0
   269
  /* Note that when these limits are exceeded we don't disconnect the
sl@0
   270
   * connection; we just sort of leave it hanging there until it times
sl@0
   271
   * out or disconnects itself or is dropped due to the max number of
sl@0
   272
   * incomplete connections. It's even OK if the connection wants to
sl@0
   273
   * retry the hello message, we support that.
sl@0
   274
   */
sl@0
   275
  connections = bus_connection_get_connections (connection);
sl@0
   276
  if (!bus_connections_check_limits (connections, connection,
sl@0
   277
                                     error))
sl@0
   278
    {
sl@0
   279
      _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
   280
      return FALSE;
sl@0
   281
    }
sl@0
   282
  
sl@0
   283
  if (!_dbus_string_init (&unique_name))
sl@0
   284
    {
sl@0
   285
      BUS_SET_OOM (error);
sl@0
   286
      return FALSE;
sl@0
   287
    }
sl@0
   288
sl@0
   289
  retval = FALSE;
sl@0
   290
sl@0
   291
  registry = bus_connection_get_registry (connection);
sl@0
   292
  
sl@0
   293
  if (!create_unique_client_name (registry, &unique_name))
sl@0
   294
    {
sl@0
   295
      BUS_SET_OOM (error);
sl@0
   296
      goto out_0;
sl@0
   297
    }
sl@0
   298
sl@0
   299
  if (!bus_connection_complete (connection, &unique_name, error))
sl@0
   300
    {
sl@0
   301
      _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
   302
      goto out_0;
sl@0
   303
    }
sl@0
   304
  
sl@0
   305
  if (!dbus_message_set_sender (message,
sl@0
   306
                                bus_connection_get_name (connection)))
sl@0
   307
    {
sl@0
   308
      BUS_SET_OOM (error);
sl@0
   309
      goto out_0;
sl@0
   310
    }
sl@0
   311
  
sl@0
   312
  if (!bus_driver_send_welcome_message (connection, message, transaction, error))
sl@0
   313
    goto out_0;
sl@0
   314
sl@0
   315
  /* Create the service */
sl@0
   316
  service = bus_registry_ensure (registry,
sl@0
   317
                                 &unique_name, connection, 0, transaction, error);
sl@0
   318
  if (service == NULL)
sl@0
   319
    goto out_0;
sl@0
   320
  
sl@0
   321
  _dbus_assert (bus_connection_is_active (connection));
sl@0
   322
  retval = TRUE;
sl@0
   323
  
sl@0
   324
 out_0:
sl@0
   325
  _dbus_string_free (&unique_name);
sl@0
   326
  return retval;
sl@0
   327
}
sl@0
   328
sl@0
   329
static dbus_bool_t
sl@0
   330
bus_driver_send_welcome_message (DBusConnection *connection,
sl@0
   331
                                 DBusMessage    *hello_message,
sl@0
   332
                                 BusTransaction *transaction,
sl@0
   333
                                 DBusError      *error)
sl@0
   334
{
sl@0
   335
  DBusMessage *welcome;
sl@0
   336
  const char *name;
sl@0
   337
sl@0
   338
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   339
  
sl@0
   340
  name = bus_connection_get_name (connection);
sl@0
   341
  _dbus_assert (name != NULL);
sl@0
   342
  
sl@0
   343
  welcome = dbus_message_new_method_return (hello_message);
sl@0
   344
  if (welcome == NULL)
sl@0
   345
    {
sl@0
   346
      BUS_SET_OOM (error);
sl@0
   347
      return FALSE;
sl@0
   348
    }
sl@0
   349
  
sl@0
   350
  if (!dbus_message_append_args (welcome,
sl@0
   351
                                 DBUS_TYPE_STRING, &name,
sl@0
   352
                                 DBUS_TYPE_INVALID))
sl@0
   353
    {
sl@0
   354
      dbus_message_unref (welcome);
sl@0
   355
      BUS_SET_OOM (error);
sl@0
   356
      return FALSE;
sl@0
   357
    }
sl@0
   358
sl@0
   359
  _dbus_assert (dbus_message_has_signature (welcome, DBUS_TYPE_STRING_AS_STRING));
sl@0
   360
  
sl@0
   361
  if (!bus_transaction_send_from_driver (transaction, connection, welcome))
sl@0
   362
    {
sl@0
   363
      dbus_message_unref (welcome);
sl@0
   364
      BUS_SET_OOM (error);
sl@0
   365
      return FALSE;
sl@0
   366
    }
sl@0
   367
  else
sl@0
   368
    {
sl@0
   369
      dbus_message_unref (welcome);
sl@0
   370
      return TRUE;
sl@0
   371
    }
sl@0
   372
}
sl@0
   373
sl@0
   374
static dbus_bool_t
sl@0
   375
bus_driver_handle_list_services (DBusConnection *connection,
sl@0
   376
                                 BusTransaction *transaction,
sl@0
   377
                                 DBusMessage    *message,
sl@0
   378
                                 DBusError      *error)
sl@0
   379
{
sl@0
   380
  DBusMessage *reply;
sl@0
   381
  int len;
sl@0
   382
  char **services;
sl@0
   383
  BusRegistry *registry;
sl@0
   384
  int i;
sl@0
   385
  DBusMessageIter iter;
sl@0
   386
  DBusMessageIter sub;
sl@0
   387
sl@0
   388
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   389
  
sl@0
   390
  registry = bus_connection_get_registry (connection);
sl@0
   391
  
sl@0
   392
  reply = dbus_message_new_method_return (message);
sl@0
   393
  if (reply == NULL)
sl@0
   394
    {
sl@0
   395
      BUS_SET_OOM (error);
sl@0
   396
      return FALSE;
sl@0
   397
    }
sl@0
   398
sl@0
   399
  if (!bus_registry_list_services (registry, &services, &len))
sl@0
   400
    {
sl@0
   401
      dbus_message_unref (reply);
sl@0
   402
      BUS_SET_OOM (error);
sl@0
   403
      return FALSE;
sl@0
   404
    }
sl@0
   405
sl@0
   406
  dbus_message_iter_init_append (reply, &iter);
sl@0
   407
  
sl@0
   408
  if (!dbus_message_iter_open_container (&iter, DBUS_TYPE_ARRAY,
sl@0
   409
                                         DBUS_TYPE_STRING_AS_STRING,
sl@0
   410
                                         &sub))
sl@0
   411
    {
sl@0
   412
      dbus_free_string_array (services);
sl@0
   413
      dbus_message_unref (reply);
sl@0
   414
      BUS_SET_OOM (error);
sl@0
   415
      return FALSE;
sl@0
   416
    }
sl@0
   417
sl@0
   418
  {
sl@0
   419
    /* Include the bus driver in the list */
sl@0
   420
    const char *v_STRING = DBUS_SERVICE_DBUS;
sl@0
   421
    if (!dbus_message_iter_append_basic (&sub, DBUS_TYPE_STRING,
sl@0
   422
                                         &v_STRING))
sl@0
   423
      {
sl@0
   424
        dbus_free_string_array (services);
sl@0
   425
        dbus_message_unref (reply);
sl@0
   426
        BUS_SET_OOM (error);
sl@0
   427
        return FALSE;
sl@0
   428
      }
sl@0
   429
  }
sl@0
   430
  
sl@0
   431
  i = 0;
sl@0
   432
  while (i < len)
sl@0
   433
    {
sl@0
   434
      if (!dbus_message_iter_append_basic (&sub, DBUS_TYPE_STRING,
sl@0
   435
                                           &services[i]))
sl@0
   436
        {
sl@0
   437
          dbus_free_string_array (services);
sl@0
   438
          dbus_message_unref (reply);
sl@0
   439
          BUS_SET_OOM (error);
sl@0
   440
          return FALSE;
sl@0
   441
        }
sl@0
   442
      ++i;
sl@0
   443
    }
sl@0
   444
sl@0
   445
  dbus_free_string_array (services);
sl@0
   446
sl@0
   447
  if (!dbus_message_iter_close_container (&iter, &sub))
sl@0
   448
    {
sl@0
   449
      dbus_message_unref (reply);
sl@0
   450
      BUS_SET_OOM (error);
sl@0
   451
      return FALSE;
sl@0
   452
    }
sl@0
   453
  
sl@0
   454
  if (!bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
   455
    {
sl@0
   456
      dbus_message_unref (reply);
sl@0
   457
      BUS_SET_OOM (error);
sl@0
   458
      return FALSE;
sl@0
   459
    }
sl@0
   460
  else
sl@0
   461
    {
sl@0
   462
      dbus_message_unref (reply);
sl@0
   463
      return TRUE;
sl@0
   464
    }
sl@0
   465
}
sl@0
   466
sl@0
   467
static dbus_bool_t
sl@0
   468
bus_driver_handle_list_activatable_services (DBusConnection *connection,
sl@0
   469
					     BusTransaction *transaction,
sl@0
   470
					     DBusMessage    *message,
sl@0
   471
					     DBusError      *error)
sl@0
   472
{
sl@0
   473
  DBusMessage *reply;
sl@0
   474
  int len;
sl@0
   475
  char **services;
sl@0
   476
  BusActivation *activation;
sl@0
   477
  int i;
sl@0
   478
  DBusMessageIter iter;
sl@0
   479
  DBusMessageIter sub;
sl@0
   480
sl@0
   481
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   482
sl@0
   483
  activation = bus_connection_get_activation (connection);
sl@0
   484
sl@0
   485
  reply = dbus_message_new_method_return (message);
sl@0
   486
  if (reply == NULL)
sl@0
   487
    {
sl@0
   488
      BUS_SET_OOM (error);
sl@0
   489
      return FALSE;
sl@0
   490
    }
sl@0
   491
sl@0
   492
  if (!bus_activation_list_services (activation, &services, &len))
sl@0
   493
    {
sl@0
   494
      dbus_message_unref (reply);
sl@0
   495
      BUS_SET_OOM (error);
sl@0
   496
      return FALSE;
sl@0
   497
    }
sl@0
   498
sl@0
   499
  dbus_message_iter_init_append (reply, &iter);
sl@0
   500
sl@0
   501
  if (!dbus_message_iter_open_container (&iter, DBUS_TYPE_ARRAY,
sl@0
   502
					 DBUS_TYPE_STRING_AS_STRING,
sl@0
   503
					 &sub))
sl@0
   504
    {
sl@0
   505
      dbus_free_string_array (services);
sl@0
   506
      dbus_message_unref (reply);
sl@0
   507
      BUS_SET_OOM (error);
sl@0
   508
      return FALSE;
sl@0
   509
    }
sl@0
   510
sl@0
   511
  {
sl@0
   512
    /* Include the bus driver in the list */
sl@0
   513
    const char *v_STRING = DBUS_SERVICE_DBUS;
sl@0
   514
    if (!dbus_message_iter_append_basic (&sub, DBUS_TYPE_STRING,
sl@0
   515
					 &v_STRING))
sl@0
   516
      {
sl@0
   517
	dbus_free_string_array (services);
sl@0
   518
	dbus_message_unref (reply);
sl@0
   519
	BUS_SET_OOM (error);
sl@0
   520
	return FALSE;
sl@0
   521
      }
sl@0
   522
  }
sl@0
   523
sl@0
   524
  i = 0;
sl@0
   525
  while (i < len)
sl@0
   526
    {
sl@0
   527
      if (!dbus_message_iter_append_basic (&sub, DBUS_TYPE_STRING,
sl@0
   528
					   &services[i]))
sl@0
   529
	{
sl@0
   530
	  dbus_free_string_array (services);
sl@0
   531
	  dbus_message_unref (reply);
sl@0
   532
	  BUS_SET_OOM (error);
sl@0
   533
	  return FALSE;
sl@0
   534
	}
sl@0
   535
      ++i;
sl@0
   536
    }
sl@0
   537
sl@0
   538
  dbus_free_string_array (services);
sl@0
   539
sl@0
   540
  if (!dbus_message_iter_close_container (&iter, &sub))
sl@0
   541
    {
sl@0
   542
      dbus_message_unref (reply);
sl@0
   543
      BUS_SET_OOM (error);
sl@0
   544
      return FALSE;
sl@0
   545
    }
sl@0
   546
sl@0
   547
  if (!bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
   548
    {
sl@0
   549
      dbus_message_unref (reply);
sl@0
   550
      BUS_SET_OOM (error);
sl@0
   551
      return FALSE;
sl@0
   552
    }
sl@0
   553
  else
sl@0
   554
    {
sl@0
   555
      dbus_message_unref (reply);
sl@0
   556
      return TRUE;
sl@0
   557
    }
sl@0
   558
}
sl@0
   559
sl@0
   560
static dbus_bool_t
sl@0
   561
bus_driver_handle_acquire_service (DBusConnection *connection,
sl@0
   562
                                   BusTransaction *transaction,
sl@0
   563
                                   DBusMessage    *message,
sl@0
   564
                                   DBusError      *error)
sl@0
   565
{
sl@0
   566
  DBusMessage *reply;
sl@0
   567
  DBusString service_name;
sl@0
   568
  const char *name;
sl@0
   569
  dbus_uint32_t service_reply;
sl@0
   570
  dbus_uint32_t flags;
sl@0
   571
  dbus_bool_t retval;
sl@0
   572
  BusRegistry *registry;
sl@0
   573
sl@0
   574
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   575
  
sl@0
   576
  registry = bus_connection_get_registry (connection);
sl@0
   577
  
sl@0
   578
  if (!dbus_message_get_args (message, error,
sl@0
   579
                              DBUS_TYPE_STRING, &name,
sl@0
   580
                              DBUS_TYPE_UINT32, &flags,
sl@0
   581
                              DBUS_TYPE_INVALID))
sl@0
   582
    return FALSE;
sl@0
   583
  
sl@0
   584
  _dbus_verbose ("Trying to own name %s with flags 0x%x\n", name, flags);
sl@0
   585
  
sl@0
   586
  retval = FALSE;
sl@0
   587
  reply = NULL;
sl@0
   588
sl@0
   589
  _dbus_string_init_const (&service_name, name);
sl@0
   590
sl@0
   591
  if (!bus_registry_acquire_service (registry, connection,
sl@0
   592
                                     &service_name, flags,
sl@0
   593
                                     &service_reply, transaction,
sl@0
   594
                                     error))
sl@0
   595
    goto out;
sl@0
   596
  
sl@0
   597
  reply = dbus_message_new_method_return (message);
sl@0
   598
  if (reply == NULL)
sl@0
   599
    {
sl@0
   600
      BUS_SET_OOM (error);
sl@0
   601
      goto out;
sl@0
   602
    }
sl@0
   603
sl@0
   604
  if (!dbus_message_append_args (reply, DBUS_TYPE_UINT32, &service_reply, DBUS_TYPE_INVALID))
sl@0
   605
    {
sl@0
   606
      BUS_SET_OOM (error);
sl@0
   607
      goto out;
sl@0
   608
    }
sl@0
   609
sl@0
   610
  if (!bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
   611
    {
sl@0
   612
      BUS_SET_OOM (error);
sl@0
   613
      goto out;
sl@0
   614
    }
sl@0
   615
sl@0
   616
  retval = TRUE;
sl@0
   617
  
sl@0
   618
 out:
sl@0
   619
  if (reply)
sl@0
   620
    dbus_message_unref (reply);
sl@0
   621
  return retval;
sl@0
   622
} 
sl@0
   623
sl@0
   624
static dbus_bool_t
sl@0
   625
bus_driver_handle_release_service (DBusConnection *connection,
sl@0
   626
                                   BusTransaction *transaction,
sl@0
   627
                                   DBusMessage    *message,
sl@0
   628
                                   DBusError      *error)
sl@0
   629
{
sl@0
   630
  DBusMessage *reply;
sl@0
   631
  DBusString service_name;
sl@0
   632
  const char *name;
sl@0
   633
  dbus_uint32_t service_reply;
sl@0
   634
  dbus_bool_t retval;
sl@0
   635
  BusRegistry *registry;
sl@0
   636
sl@0
   637
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   638
sl@0
   639
  registry = bus_connection_get_registry (connection);
sl@0
   640
sl@0
   641
  if (!dbus_message_get_args (message, error,
sl@0
   642
                              DBUS_TYPE_STRING, &name,
sl@0
   643
                              DBUS_TYPE_INVALID))
sl@0
   644
    return FALSE;
sl@0
   645
sl@0
   646
  _dbus_verbose ("Trying to release name %s\n", name);
sl@0
   647
sl@0
   648
  retval = FALSE;
sl@0
   649
  reply = NULL;
sl@0
   650
sl@0
   651
  _dbus_string_init_const (&service_name, name);
sl@0
   652
sl@0
   653
  if (!bus_registry_release_service (registry, connection,
sl@0
   654
                                     &service_name, &service_reply,
sl@0
   655
                                     transaction, error))
sl@0
   656
    goto out;
sl@0
   657
sl@0
   658
  reply = dbus_message_new_method_return (message);
sl@0
   659
  if (reply == NULL)
sl@0
   660
    {
sl@0
   661
      BUS_SET_OOM (error);
sl@0
   662
      goto out;
sl@0
   663
    }
sl@0
   664
sl@0
   665
  if (!dbus_message_append_args (reply, DBUS_TYPE_UINT32, &service_reply, DBUS_TYPE_INVALID))
sl@0
   666
    {
sl@0
   667
      BUS_SET_OOM (error);
sl@0
   668
      goto out;
sl@0
   669
    }
sl@0
   670
sl@0
   671
  if (!bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
   672
    {
sl@0
   673
      BUS_SET_OOM (error);
sl@0
   674
      goto out;
sl@0
   675
    }
sl@0
   676
sl@0
   677
  retval = TRUE;
sl@0
   678
sl@0
   679
 out:
sl@0
   680
  if (reply)
sl@0
   681
    dbus_message_unref (reply);
sl@0
   682
  return retval;
sl@0
   683
}
sl@0
   684
sl@0
   685
static dbus_bool_t
sl@0
   686
bus_driver_handle_service_exists (DBusConnection *connection,
sl@0
   687
                                  BusTransaction *transaction,
sl@0
   688
                                  DBusMessage    *message,
sl@0
   689
                                  DBusError      *error)
sl@0
   690
{
sl@0
   691
  DBusMessage *reply;
sl@0
   692
  DBusString service_name;
sl@0
   693
  BusService *service;
sl@0
   694
  dbus_bool_t service_exists;
sl@0
   695
  const char *name;
sl@0
   696
  dbus_bool_t retval;
sl@0
   697
  BusRegistry *registry;
sl@0
   698
sl@0
   699
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   700
  
sl@0
   701
  registry = bus_connection_get_registry (connection);
sl@0
   702
  
sl@0
   703
  if (!dbus_message_get_args (message, error,
sl@0
   704
                              DBUS_TYPE_STRING, &name,
sl@0
   705
                              DBUS_TYPE_INVALID))
sl@0
   706
    return FALSE;
sl@0
   707
sl@0
   708
  retval = FALSE;
sl@0
   709
sl@0
   710
  if (strcmp (name, DBUS_SERVICE_DBUS) == 0)
sl@0
   711
    {
sl@0
   712
      service_exists = TRUE;
sl@0
   713
    }
sl@0
   714
  else
sl@0
   715
    {
sl@0
   716
      _dbus_string_init_const (&service_name, name);
sl@0
   717
      service = bus_registry_lookup (registry, &service_name);
sl@0
   718
      service_exists = service != NULL;
sl@0
   719
    }
sl@0
   720
  
sl@0
   721
  reply = dbus_message_new_method_return (message);
sl@0
   722
  if (reply == NULL)
sl@0
   723
    {
sl@0
   724
      BUS_SET_OOM (error);
sl@0
   725
      goto out;
sl@0
   726
    }
sl@0
   727
sl@0
   728
  if (!dbus_message_append_args (reply,
sl@0
   729
                                 DBUS_TYPE_BOOLEAN, &service_exists,
sl@0
   730
                                 0))
sl@0
   731
    {
sl@0
   732
      BUS_SET_OOM (error);
sl@0
   733
      goto out;
sl@0
   734
    }
sl@0
   735
sl@0
   736
  if (!bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
   737
    {
sl@0
   738
      BUS_SET_OOM (error);
sl@0
   739
      goto out;
sl@0
   740
    }
sl@0
   741
sl@0
   742
  retval = TRUE;
sl@0
   743
  
sl@0
   744
 out:
sl@0
   745
  if (reply)
sl@0
   746
    dbus_message_unref (reply);
sl@0
   747
sl@0
   748
  return retval;
sl@0
   749
}
sl@0
   750
sl@0
   751
static dbus_bool_t
sl@0
   752
bus_driver_handle_activate_service (DBusConnection *connection,
sl@0
   753
                                    BusTransaction *transaction,
sl@0
   754
                                    DBusMessage    *message,
sl@0
   755
                                    DBusError      *error)
sl@0
   756
{
sl@0
   757
  dbus_uint32_t flags;
sl@0
   758
  const char *name;
sl@0
   759
  dbus_bool_t retval;
sl@0
   760
  BusActivation *activation;
sl@0
   761
sl@0
   762
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   763
  
sl@0
   764
  activation = bus_connection_get_activation (connection);
sl@0
   765
  
sl@0
   766
  if (!dbus_message_get_args (message, error,
sl@0
   767
                              DBUS_TYPE_STRING, &name,
sl@0
   768
                              DBUS_TYPE_UINT32, &flags,
sl@0
   769
                              DBUS_TYPE_INVALID))
sl@0
   770
    {
sl@0
   771
      _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
   772
      _dbus_verbose ("No memory to get arguments to StartServiceByName\n");
sl@0
   773
      return FALSE;
sl@0
   774
    }
sl@0
   775
sl@0
   776
  retval = FALSE;
sl@0
   777
sl@0
   778
  if (!bus_activation_activate_service (activation, connection, transaction, FALSE,
sl@0
   779
                                        message, name, error))
sl@0
   780
    {
sl@0
   781
      _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
   782
      _dbus_verbose ("bus_activation_activate_service() failed\n");
sl@0
   783
      goto out;
sl@0
   784
    }
sl@0
   785
sl@0
   786
  retval = TRUE;
sl@0
   787
  
sl@0
   788
 out:
sl@0
   789
  return retval;
sl@0
   790
}
sl@0
   791
sl@0
   792
static dbus_bool_t
sl@0
   793
send_ack_reply (DBusConnection *connection,
sl@0
   794
                BusTransaction *transaction,
sl@0
   795
                DBusMessage    *message,
sl@0
   796
                DBusError      *error)
sl@0
   797
{
sl@0
   798
  DBusMessage *reply;
sl@0
   799
sl@0
   800
  reply = dbus_message_new_method_return (message);
sl@0
   801
  if (reply == NULL)
sl@0
   802
    {
sl@0
   803
      BUS_SET_OOM (error);
sl@0
   804
      return FALSE;
sl@0
   805
    }
sl@0
   806
sl@0
   807
  if (!bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
   808
    {
sl@0
   809
      BUS_SET_OOM (error);
sl@0
   810
      dbus_message_unref (reply);
sl@0
   811
      return FALSE;
sl@0
   812
    }
sl@0
   813
sl@0
   814
  dbus_message_unref (reply);
sl@0
   815
  
sl@0
   816
  return TRUE;
sl@0
   817
}
sl@0
   818
sl@0
   819
static dbus_bool_t
sl@0
   820
bus_driver_handle_add_match (DBusConnection *connection,
sl@0
   821
                             BusTransaction *transaction,
sl@0
   822
                             DBusMessage    *message,
sl@0
   823
                             DBusError      *error)
sl@0
   824
{
sl@0
   825
  BusMatchRule *rule;
sl@0
   826
  const char *text;
sl@0
   827
  DBusString str;
sl@0
   828
  BusMatchmaker *matchmaker;
sl@0
   829
  
sl@0
   830
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   831
sl@0
   832
  text = NULL;
sl@0
   833
  rule = NULL;
sl@0
   834
sl@0
   835
  if (bus_connection_get_n_match_rules (connection) >=
sl@0
   836
      bus_context_get_max_match_rules_per_connection (bus_transaction_get_context (transaction)))
sl@0
   837
    {
sl@0
   838
      dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
sl@0
   839
                      "Connection \"%s\" is not allowed to add more match rules "
sl@0
   840
                      "(increase limits in configuration file if required)",
sl@0
   841
                      bus_connection_is_active (connection) ?
sl@0
   842
                      bus_connection_get_name (connection) :
sl@0
   843
                      "(inactive)");
sl@0
   844
      goto failed;
sl@0
   845
    }
sl@0
   846
  
sl@0
   847
  if (!dbus_message_get_args (message, error,
sl@0
   848
                              DBUS_TYPE_STRING, &text,
sl@0
   849
                              DBUS_TYPE_INVALID))
sl@0
   850
    {
sl@0
   851
      _dbus_verbose ("No memory to get arguments to AddMatch\n");
sl@0
   852
      goto failed;
sl@0
   853
    }
sl@0
   854
sl@0
   855
  _dbus_string_init_const (&str, text);
sl@0
   856
sl@0
   857
  rule = bus_match_rule_parse (connection, &str, error);
sl@0
   858
  if (rule == NULL)
sl@0
   859
    goto failed;
sl@0
   860
sl@0
   861
  matchmaker = bus_connection_get_matchmaker (connection);
sl@0
   862
sl@0
   863
  if (!bus_matchmaker_add_rule (matchmaker, rule))
sl@0
   864
    {
sl@0
   865
      BUS_SET_OOM (error);
sl@0
   866
      goto failed;
sl@0
   867
    }
sl@0
   868
sl@0
   869
  if (!send_ack_reply (connection, transaction,
sl@0
   870
                       message, error))
sl@0
   871
    {
sl@0
   872
      bus_matchmaker_remove_rule (matchmaker, rule);
sl@0
   873
      goto failed;
sl@0
   874
    }
sl@0
   875
  
sl@0
   876
  bus_match_rule_unref (rule);
sl@0
   877
  
sl@0
   878
  return TRUE;
sl@0
   879
sl@0
   880
 failed:
sl@0
   881
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
   882
  if (rule)
sl@0
   883
    bus_match_rule_unref (rule);
sl@0
   884
  return FALSE;
sl@0
   885
}
sl@0
   886
sl@0
   887
static dbus_bool_t
sl@0
   888
bus_driver_handle_remove_match (DBusConnection *connection,
sl@0
   889
                                BusTransaction *transaction,
sl@0
   890
                                DBusMessage    *message,
sl@0
   891
                                DBusError      *error)
sl@0
   892
{
sl@0
   893
  BusMatchRule *rule;
sl@0
   894
  const char *text;
sl@0
   895
  DBusString str;
sl@0
   896
  BusMatchmaker *matchmaker;
sl@0
   897
  
sl@0
   898
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   899
sl@0
   900
  text = NULL;
sl@0
   901
  rule = NULL;
sl@0
   902
  
sl@0
   903
  if (!dbus_message_get_args (message, error,
sl@0
   904
                              DBUS_TYPE_STRING, &text,
sl@0
   905
                              DBUS_TYPE_INVALID))
sl@0
   906
    {
sl@0
   907
      _dbus_verbose ("No memory to get arguments to RemoveMatch\n");
sl@0
   908
      goto failed;
sl@0
   909
    }
sl@0
   910
sl@0
   911
  _dbus_string_init_const (&str, text);
sl@0
   912
sl@0
   913
  rule = bus_match_rule_parse (connection, &str, error);
sl@0
   914
  if (rule == NULL)
sl@0
   915
    goto failed;
sl@0
   916
sl@0
   917
  /* Send the ack before we remove the rule, since the ack is undone
sl@0
   918
   * on transaction cancel, but rule removal isn't.
sl@0
   919
   */
sl@0
   920
  if (!send_ack_reply (connection, transaction,
sl@0
   921
                       message, error))
sl@0
   922
    goto failed;
sl@0
   923
  
sl@0
   924
  matchmaker = bus_connection_get_matchmaker (connection);
sl@0
   925
sl@0
   926
  if (!bus_matchmaker_remove_rule_by_value (matchmaker, rule, error))
sl@0
   927
    goto failed;
sl@0
   928
sl@0
   929
  bus_match_rule_unref (rule);
sl@0
   930
  
sl@0
   931
  return TRUE;
sl@0
   932
sl@0
   933
 failed:
sl@0
   934
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
   935
  if (rule)
sl@0
   936
    bus_match_rule_unref (rule);
sl@0
   937
  return FALSE;
sl@0
   938
}
sl@0
   939
sl@0
   940
static dbus_bool_t
sl@0
   941
bus_driver_handle_get_service_owner (DBusConnection *connection,
sl@0
   942
				     BusTransaction *transaction,
sl@0
   943
				     DBusMessage    *message,
sl@0
   944
				     DBusError      *error)
sl@0
   945
{
sl@0
   946
  const char *text;
sl@0
   947
  const char *base_name;
sl@0
   948
  DBusString str;
sl@0
   949
  BusRegistry *registry;
sl@0
   950
  BusService *service;
sl@0
   951
  DBusMessage *reply;
sl@0
   952
  
sl@0
   953
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
   954
sl@0
   955
  registry = bus_connection_get_registry (connection);
sl@0
   956
sl@0
   957
  text = NULL;
sl@0
   958
  reply = NULL;
sl@0
   959
sl@0
   960
  if (! dbus_message_get_args (message, error,
sl@0
   961
			       DBUS_TYPE_STRING, &text,
sl@0
   962
			       DBUS_TYPE_INVALID))
sl@0
   963
      goto failed;
sl@0
   964
sl@0
   965
  _dbus_string_init_const (&str, text);
sl@0
   966
  service = bus_registry_lookup (registry, &str);
sl@0
   967
  if (service == NULL &&
sl@0
   968
      _dbus_string_equal_c_str (&str, DBUS_SERVICE_DBUS))
sl@0
   969
    {
sl@0
   970
      /* ORG_FREEDESKTOP_DBUS owns itself */
sl@0
   971
      base_name = DBUS_SERVICE_DBUS;
sl@0
   972
    }
sl@0
   973
  else if (service == NULL)
sl@0
   974
    {
sl@0
   975
      dbus_set_error (error, 
sl@0
   976
                      DBUS_ERROR_NAME_HAS_NO_OWNER,
sl@0
   977
                      "Could not get owner of name '%s': no such name", text);
sl@0
   978
      goto failed;
sl@0
   979
    }
sl@0
   980
  else
sl@0
   981
    {
sl@0
   982
      base_name = bus_connection_get_name (bus_service_get_primary_owners_connection (service));
sl@0
   983
      if (base_name == NULL)
sl@0
   984
        {
sl@0
   985
          /* FIXME - how is this error possible? */
sl@0
   986
          dbus_set_error (error,
sl@0
   987
                          DBUS_ERROR_FAILED,
sl@0
   988
                          "Could not determine unique name for '%s'", text);
sl@0
   989
          goto failed;
sl@0
   990
        }
sl@0
   991
      _dbus_assert (*base_name == ':');      
sl@0
   992
    }
sl@0
   993
sl@0
   994
  _dbus_assert (base_name != NULL);
sl@0
   995
sl@0
   996
  reply = dbus_message_new_method_return (message);
sl@0
   997
  if (reply == NULL)
sl@0
   998
    goto oom;
sl@0
   999
sl@0
  1000
  if (! dbus_message_append_args (reply, 
sl@0
  1001
				  DBUS_TYPE_STRING, &base_name,
sl@0
  1002
				  DBUS_TYPE_INVALID))
sl@0
  1003
    goto oom;
sl@0
  1004
  
sl@0
  1005
  if (! bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  1006
    goto oom;
sl@0
  1007
sl@0
  1008
  dbus_message_unref (reply);
sl@0
  1009
sl@0
  1010
  return TRUE;
sl@0
  1011
sl@0
  1012
 oom:
sl@0
  1013
  BUS_SET_OOM (error);
sl@0
  1014
sl@0
  1015
 failed:
sl@0
  1016
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1017
  if (reply)
sl@0
  1018
    dbus_message_unref (reply);
sl@0
  1019
  return FALSE;
sl@0
  1020
}
sl@0
  1021
sl@0
  1022
static dbus_bool_t
sl@0
  1023
bus_driver_handle_list_queued_owners (DBusConnection *connection,
sl@0
  1024
				      BusTransaction *transaction,
sl@0
  1025
				      DBusMessage    *message,
sl@0
  1026
				      DBusError      *error)
sl@0
  1027
{
sl@0
  1028
  const char *text;
sl@0
  1029
  DBusList *base_names;
sl@0
  1030
  DBusList *link;
sl@0
  1031
  DBusString str;
sl@0
  1032
  BusRegistry *registry;
sl@0
  1033
  BusService *service;
sl@0
  1034
  DBusMessage *reply;
sl@0
  1035
  DBusMessageIter iter, array_iter;
sl@0
  1036
  char *dbus_service_name = DBUS_SERVICE_DBUS;
sl@0
  1037
  
sl@0
  1038
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1039
sl@0
  1040
  registry = bus_connection_get_registry (connection);
sl@0
  1041
sl@0
  1042
  base_names = NULL;
sl@0
  1043
  text = NULL;
sl@0
  1044
  reply = NULL;
sl@0
  1045
sl@0
  1046
  if (! dbus_message_get_args (message, error,
sl@0
  1047
			       DBUS_TYPE_STRING, &text,
sl@0
  1048
			       DBUS_TYPE_INVALID))
sl@0
  1049
      goto failed;
sl@0
  1050
sl@0
  1051
  _dbus_string_init_const (&str, text);
sl@0
  1052
  service = bus_registry_lookup (registry, &str);
sl@0
  1053
  if (service == NULL &&
sl@0
  1054
      _dbus_string_equal_c_str (&str, DBUS_SERVICE_DBUS))
sl@0
  1055
    {
sl@0
  1056
      /* ORG_FREEDESKTOP_DBUS owns itself */
sl@0
  1057
      if (! _dbus_list_append (&base_names, dbus_service_name))
sl@0
  1058
        goto oom;
sl@0
  1059
    }
sl@0
  1060
  else if (service == NULL)
sl@0
  1061
    {
sl@0
  1062
      dbus_set_error (error, 
sl@0
  1063
                      DBUS_ERROR_NAME_HAS_NO_OWNER,
sl@0
  1064
                      "Could not get owners of name '%s': no such name", text);
sl@0
  1065
      goto failed;
sl@0
  1066
    }
sl@0
  1067
  else
sl@0
  1068
    {
sl@0
  1069
      if (!bus_service_list_queued_owners (service, 
sl@0
  1070
                                           &base_names,
sl@0
  1071
                                           error))
sl@0
  1072
        goto failed;
sl@0
  1073
    }
sl@0
  1074
sl@0
  1075
  _dbus_assert (base_names != NULL);
sl@0
  1076
sl@0
  1077
  reply = dbus_message_new_method_return (message);
sl@0
  1078
  if (reply == NULL)
sl@0
  1079
    goto oom;
sl@0
  1080
sl@0
  1081
  dbus_message_iter_init_append (reply, &iter);
sl@0
  1082
  if (!dbus_message_iter_open_container (&iter,
sl@0
  1083
                                         DBUS_TYPE_ARRAY,
sl@0
  1084
                                         DBUS_TYPE_STRING_AS_STRING,
sl@0
  1085
                                         &array_iter))
sl@0
  1086
    goto oom;
sl@0
  1087
  
sl@0
  1088
  link = _dbus_list_get_first_link (&base_names);
sl@0
  1089
  while (link != NULL)
sl@0
  1090
    {
sl@0
  1091
      char *uname;
sl@0
  1092
sl@0
  1093
      _dbus_assert (link->data != NULL);
sl@0
  1094
      uname = (char *)link->data;
sl@0
  1095
    
sl@0
  1096
      if (!dbus_message_iter_append_basic (&array_iter, 
sl@0
  1097
                                           DBUS_TYPE_STRING,
sl@0
  1098
                                           &uname))
sl@0
  1099
        goto oom;
sl@0
  1100
sl@0
  1101
      link = _dbus_list_get_next_link (&base_names, link);
sl@0
  1102
    }
sl@0
  1103
sl@0
  1104
  if (! dbus_message_iter_close_container (&iter, &array_iter))
sl@0
  1105
    goto oom;
sl@0
  1106
                                    
sl@0
  1107
 
sl@0
  1108
  if (! bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  1109
    goto oom;
sl@0
  1110
sl@0
  1111
  dbus_message_unref (reply);
sl@0
  1112
sl@0
  1113
  return TRUE;
sl@0
  1114
sl@0
  1115
 oom:
sl@0
  1116
  BUS_SET_OOM (error);
sl@0
  1117
sl@0
  1118
 failed:
sl@0
  1119
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1120
  if (reply)
sl@0
  1121
    dbus_message_unref (reply);
sl@0
  1122
sl@0
  1123
  if (base_names)
sl@0
  1124
    _dbus_list_clear (&base_names);
sl@0
  1125
sl@0
  1126
  return FALSE;
sl@0
  1127
}
sl@0
  1128
sl@0
  1129
static dbus_bool_t
sl@0
  1130
bus_driver_handle_get_connection_unix_user (DBusConnection *connection,
sl@0
  1131
                                            BusTransaction *transaction,
sl@0
  1132
                                            DBusMessage    *message,
sl@0
  1133
                                            DBusError      *error)
sl@0
  1134
{
sl@0
  1135
  const char *service;
sl@0
  1136
  DBusString str;
sl@0
  1137
  BusRegistry *registry;
sl@0
  1138
  BusService *serv;
sl@0
  1139
  DBusConnection *conn;
sl@0
  1140
  DBusMessage *reply;
sl@0
  1141
  unsigned long uid;
sl@0
  1142
  dbus_uint32_t uid32;
sl@0
  1143
sl@0
  1144
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1145
sl@0
  1146
  registry = bus_connection_get_registry (connection);
sl@0
  1147
sl@0
  1148
  service = NULL;
sl@0
  1149
  reply = NULL;
sl@0
  1150
sl@0
  1151
  if (! dbus_message_get_args (message, error,
sl@0
  1152
			       DBUS_TYPE_STRING, &service,
sl@0
  1153
			       DBUS_TYPE_INVALID))
sl@0
  1154
      goto failed;
sl@0
  1155
sl@0
  1156
  _dbus_verbose ("asked for UID of connection %s\n", service);
sl@0
  1157
sl@0
  1158
  _dbus_string_init_const (&str, service);
sl@0
  1159
  serv = bus_registry_lookup (registry, &str);
sl@0
  1160
  if (serv == NULL)
sl@0
  1161
    {
sl@0
  1162
      dbus_set_error (error, 
sl@0
  1163
		      DBUS_ERROR_NAME_HAS_NO_OWNER,
sl@0
  1164
		      "Could not get UID of name '%s': no such name", service);
sl@0
  1165
      goto failed;
sl@0
  1166
    }
sl@0
  1167
sl@0
  1168
  conn = bus_service_get_primary_owners_connection (serv);
sl@0
  1169
sl@0
  1170
  reply = dbus_message_new_method_return (message);
sl@0
  1171
  if (reply == NULL)
sl@0
  1172
    goto oom;
sl@0
  1173
sl@0
  1174
  if (!dbus_connection_get_unix_user (conn, &uid))
sl@0
  1175
    {
sl@0
  1176
      dbus_set_error (error,
sl@0
  1177
                      DBUS_ERROR_FAILED,
sl@0
  1178
                      "Could not determine UID for '%s'", service);
sl@0
  1179
      goto failed;
sl@0
  1180
    }
sl@0
  1181
sl@0
  1182
  uid32 = uid;
sl@0
  1183
  if (! dbus_message_append_args (reply,
sl@0
  1184
                                  DBUS_TYPE_UINT32, &uid32,
sl@0
  1185
                                  DBUS_TYPE_INVALID))
sl@0
  1186
    goto oom;
sl@0
  1187
sl@0
  1188
  if (! bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  1189
    goto oom;
sl@0
  1190
sl@0
  1191
  dbus_message_unref (reply);
sl@0
  1192
sl@0
  1193
  return TRUE;
sl@0
  1194
sl@0
  1195
 oom:
sl@0
  1196
  BUS_SET_OOM (error);
sl@0
  1197
sl@0
  1198
 failed:
sl@0
  1199
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1200
  if (reply)
sl@0
  1201
    dbus_message_unref (reply);
sl@0
  1202
  return FALSE;
sl@0
  1203
}
sl@0
  1204
sl@0
  1205
static dbus_bool_t
sl@0
  1206
bus_driver_handle_get_connection_unix_process_id (DBusConnection *connection,
sl@0
  1207
						  BusTransaction *transaction,
sl@0
  1208
						  DBusMessage    *message,
sl@0
  1209
						  DBusError      *error)
sl@0
  1210
{
sl@0
  1211
  const char *service;
sl@0
  1212
  DBusString str;
sl@0
  1213
  BusRegistry *registry;
sl@0
  1214
  BusService *serv;
sl@0
  1215
  DBusConnection *conn;
sl@0
  1216
  DBusMessage *reply;
sl@0
  1217
  unsigned long pid;
sl@0
  1218
  dbus_uint32_t pid32;
sl@0
  1219
sl@0
  1220
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1221
sl@0
  1222
  registry = bus_connection_get_registry (connection);
sl@0
  1223
sl@0
  1224
  service = NULL;
sl@0
  1225
  reply = NULL;
sl@0
  1226
sl@0
  1227
  if (! dbus_message_get_args (message, error,
sl@0
  1228
			       DBUS_TYPE_STRING, &service,
sl@0
  1229
			       DBUS_TYPE_INVALID))
sl@0
  1230
      goto failed;
sl@0
  1231
sl@0
  1232
  _dbus_verbose ("asked for PID of connection %s\n", service);
sl@0
  1233
sl@0
  1234
  _dbus_string_init_const (&str, service);
sl@0
  1235
  serv = bus_registry_lookup (registry, &str);
sl@0
  1236
  if (serv == NULL)
sl@0
  1237
    {
sl@0
  1238
      dbus_set_error (error, 
sl@0
  1239
		      DBUS_ERROR_NAME_HAS_NO_OWNER,
sl@0
  1240
		      "Could not get PID of name '%s': no such name", service);
sl@0
  1241
      goto failed;
sl@0
  1242
    }
sl@0
  1243
sl@0
  1244
  conn = bus_service_get_primary_owners_connection (serv);
sl@0
  1245
sl@0
  1246
  reply = dbus_message_new_method_return (message);
sl@0
  1247
  if (reply == NULL)
sl@0
  1248
    goto oom;
sl@0
  1249
sl@0
  1250
  if (!dbus_connection_get_unix_process_id (conn, &pid))
sl@0
  1251
    {
sl@0
  1252
      dbus_set_error (error,
sl@0
  1253
                      DBUS_ERROR_UNIX_PROCESS_ID_UNKNOWN,
sl@0
  1254
                      "Could not determine PID for '%s'", service);
sl@0
  1255
      goto failed;
sl@0
  1256
    }
sl@0
  1257
sl@0
  1258
  pid32 = pid;
sl@0
  1259
  if (! dbus_message_append_args (reply,
sl@0
  1260
                                  DBUS_TYPE_UINT32, &pid32,
sl@0
  1261
                                  DBUS_TYPE_INVALID))
sl@0
  1262
    goto oom;
sl@0
  1263
sl@0
  1264
  if (! bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  1265
    goto oom;
sl@0
  1266
sl@0
  1267
  dbus_message_unref (reply);
sl@0
  1268
sl@0
  1269
  return TRUE;
sl@0
  1270
sl@0
  1271
 oom:
sl@0
  1272
  BUS_SET_OOM (error);
sl@0
  1273
sl@0
  1274
 failed:
sl@0
  1275
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1276
  if (reply)
sl@0
  1277
    dbus_message_unref (reply);
sl@0
  1278
  return FALSE;
sl@0
  1279
}
sl@0
  1280
sl@0
  1281
static dbus_bool_t
sl@0
  1282
bus_driver_handle_get_connection_selinux_security_context (DBusConnection *connection,
sl@0
  1283
							   BusTransaction *transaction,
sl@0
  1284
							   DBusMessage    *message,
sl@0
  1285
							   DBusError      *error)
sl@0
  1286
{
sl@0
  1287
  const char *service;
sl@0
  1288
  DBusString str;
sl@0
  1289
  BusRegistry *registry;
sl@0
  1290
  BusService *serv;
sl@0
  1291
  DBusConnection *conn;
sl@0
  1292
  DBusMessage *reply;
sl@0
  1293
  BusSELinuxID *context;
sl@0
  1294
sl@0
  1295
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1296
sl@0
  1297
  registry = bus_connection_get_registry (connection);
sl@0
  1298
sl@0
  1299
  service = NULL;
sl@0
  1300
  reply = NULL;
sl@0
  1301
sl@0
  1302
  if (! dbus_message_get_args (message, error,
sl@0
  1303
			       DBUS_TYPE_STRING, &service,
sl@0
  1304
			       DBUS_TYPE_INVALID))
sl@0
  1305
      goto failed;
sl@0
  1306
sl@0
  1307
  _dbus_verbose ("asked for security context of connection %s\n", service);
sl@0
  1308
sl@0
  1309
  _dbus_string_init_const (&str, service);
sl@0
  1310
  serv = bus_registry_lookup (registry, &str);
sl@0
  1311
  if (serv == NULL)
sl@0
  1312
    {
sl@0
  1313
      dbus_set_error (error, 
sl@0
  1314
		      DBUS_ERROR_NAME_HAS_NO_OWNER,
sl@0
  1315
		      "Could not get security context of name '%s': no such name", service);
sl@0
  1316
      goto failed;
sl@0
  1317
    }
sl@0
  1318
sl@0
  1319
  conn = bus_service_get_primary_owners_connection (serv);
sl@0
  1320
sl@0
  1321
  reply = dbus_message_new_method_return (message);
sl@0
  1322
  if (reply == NULL)
sl@0
  1323
    goto oom;
sl@0
  1324
sl@0
  1325
  context = bus_connection_get_selinux_id (conn);
sl@0
  1326
  if (!context)
sl@0
  1327
    {
sl@0
  1328
      dbus_set_error (error,
sl@0
  1329
                      DBUS_ERROR_SELINUX_SECURITY_CONTEXT_UNKNOWN,
sl@0
  1330
                      "Could not determine security context for '%s'", service);
sl@0
  1331
      goto failed;
sl@0
  1332
    }
sl@0
  1333
sl@0
  1334
  if (! bus_selinux_append_context (reply, context, error))
sl@0
  1335
    goto failed;
sl@0
  1336
sl@0
  1337
  if (! bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  1338
    goto oom;
sl@0
  1339
sl@0
  1340
  dbus_message_unref (reply);
sl@0
  1341
sl@0
  1342
  return TRUE;
sl@0
  1343
sl@0
  1344
 oom:
sl@0
  1345
  BUS_SET_OOM (error);
sl@0
  1346
sl@0
  1347
 failed:
sl@0
  1348
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1349
  if (reply)
sl@0
  1350
    dbus_message_unref (reply);
sl@0
  1351
  return FALSE;
sl@0
  1352
}
sl@0
  1353
sl@0
  1354
static dbus_bool_t
sl@0
  1355
bus_driver_handle_reload_config (DBusConnection *connection,
sl@0
  1356
				 BusTransaction *transaction,
sl@0
  1357
				 DBusMessage    *message,
sl@0
  1358
				 DBusError      *error)
sl@0
  1359
{
sl@0
  1360
  BusContext *context;
sl@0
  1361
  DBusMessage *reply;
sl@0
  1362
sl@0
  1363
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1364
sl@0
  1365
  reply = NULL;
sl@0
  1366
  
sl@0
  1367
  context = bus_connection_get_context (connection);
sl@0
  1368
  if (!bus_context_reload_config (context, error))
sl@0
  1369
    goto failed;
sl@0
  1370
sl@0
  1371
  reply = dbus_message_new_method_return (message);
sl@0
  1372
  if (reply == NULL)
sl@0
  1373
    goto oom;
sl@0
  1374
sl@0
  1375
  if (! bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  1376
    goto oom;
sl@0
  1377
sl@0
  1378
  dbus_message_unref (reply);
sl@0
  1379
  return TRUE;
sl@0
  1380
sl@0
  1381
 oom:
sl@0
  1382
  BUS_SET_OOM (error);
sl@0
  1383
sl@0
  1384
 failed:
sl@0
  1385
  _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1386
  if (reply)
sl@0
  1387
    dbus_message_unref (reply);
sl@0
  1388
  return FALSE;
sl@0
  1389
}
sl@0
  1390
sl@0
  1391
/* For speed it might be useful to sort this in order of
sl@0
  1392
 * frequency of use (but doesn't matter with only a few items
sl@0
  1393
 * anyhow)
sl@0
  1394
 */
sl@0
  1395
struct
sl@0
  1396
{
sl@0
  1397
  const char *name;
sl@0
  1398
  const char *in_args;
sl@0
  1399
  const char *out_args;
sl@0
  1400
  dbus_bool_t (* handler) (DBusConnection *connection,
sl@0
  1401
                           BusTransaction *transaction,
sl@0
  1402
                           DBusMessage    *message,
sl@0
  1403
                           DBusError      *error);
sl@0
  1404
} message_handlers[] = {
sl@0
  1405
  { "RequestName",
sl@0
  1406
    DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_UINT32_AS_STRING,
sl@0
  1407
    DBUS_TYPE_UINT32_AS_STRING,
sl@0
  1408
    bus_driver_handle_acquire_service },
sl@0
  1409
  { "ReleaseName",
sl@0
  1410
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1411
    DBUS_TYPE_UINT32_AS_STRING,
sl@0
  1412
    bus_driver_handle_release_service },
sl@0
  1413
  { "StartServiceByName",
sl@0
  1414
    DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_UINT32_AS_STRING,
sl@0
  1415
    DBUS_TYPE_UINT32_AS_STRING,
sl@0
  1416
    bus_driver_handle_activate_service },
sl@0
  1417
  { "Hello",
sl@0
  1418
    "",
sl@0
  1419
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1420
    bus_driver_handle_hello },
sl@0
  1421
  { "NameHasOwner",
sl@0
  1422
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1423
    DBUS_TYPE_BOOLEAN_AS_STRING,
sl@0
  1424
    bus_driver_handle_service_exists },
sl@0
  1425
  { "ListNames",
sl@0
  1426
    "",
sl@0
  1427
    DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_STRING_AS_STRING,
sl@0
  1428
    bus_driver_handle_list_services },
sl@0
  1429
  { "ListActivatableNames",
sl@0
  1430
    "",
sl@0
  1431
    DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_STRING_AS_STRING,
sl@0
  1432
    bus_driver_handle_list_activatable_services },
sl@0
  1433
  { "AddMatch",
sl@0
  1434
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1435
    "",
sl@0
  1436
    bus_driver_handle_add_match },
sl@0
  1437
  { "RemoveMatch",
sl@0
  1438
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1439
    "",
sl@0
  1440
    bus_driver_handle_remove_match },
sl@0
  1441
  { "GetNameOwner",
sl@0
  1442
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1443
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1444
    bus_driver_handle_get_service_owner },
sl@0
  1445
  { "ListQueuedOwners",
sl@0
  1446
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1447
    DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_STRING_AS_STRING,
sl@0
  1448
    bus_driver_handle_list_queued_owners },
sl@0
  1449
  { "GetConnectionUnixUser",
sl@0
  1450
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1451
    DBUS_TYPE_UINT32_AS_STRING,
sl@0
  1452
    bus_driver_handle_get_connection_unix_user },
sl@0
  1453
  { "GetConnectionUnixProcessID",
sl@0
  1454
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1455
    DBUS_TYPE_UINT32_AS_STRING,
sl@0
  1456
    bus_driver_handle_get_connection_unix_process_id },
sl@0
  1457
  { "GetConnectionSELinuxSecurityContext",
sl@0
  1458
    DBUS_TYPE_STRING_AS_STRING,
sl@0
  1459
    DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_BYTE_AS_STRING,
sl@0
  1460
    bus_driver_handle_get_connection_selinux_security_context },
sl@0
  1461
  { "ReloadConfig",
sl@0
  1462
    "",
sl@0
  1463
    "",
sl@0
  1464
    bus_driver_handle_reload_config }
sl@0
  1465
};
sl@0
  1466
sl@0
  1467
static dbus_bool_t
sl@0
  1468
write_args_for_direction (DBusString *xml,
sl@0
  1469
			  const char *signature,
sl@0
  1470
			  dbus_bool_t in)
sl@0
  1471
{
sl@0
  1472
  DBusTypeReader typereader;
sl@0
  1473
  DBusString sigstr;
sl@0
  1474
  int current_type;
sl@0
  1475
  
sl@0
  1476
  _dbus_string_init_const (&sigstr, signature);
sl@0
  1477
  _dbus_type_reader_init_types_only (&typereader, &sigstr, 0);
sl@0
  1478
      
sl@0
  1479
  while ((current_type = _dbus_type_reader_get_current_type (&typereader)) != DBUS_TYPE_INVALID)
sl@0
  1480
    {
sl@0
  1481
      const DBusString *subsig;
sl@0
  1482
      int start, len;
sl@0
  1483
sl@0
  1484
      _dbus_type_reader_get_signature (&typereader, &subsig, &start, &len);
sl@0
  1485
      if (!_dbus_string_append_printf (xml, "      <arg direction=\"%s\" type=\"",
sl@0
  1486
				       in ? "in" : "out"))
sl@0
  1487
	goto oom;
sl@0
  1488
      if (!_dbus_string_append_len (xml,
sl@0
  1489
				    _dbus_string_get_const_data (subsig) + start,
sl@0
  1490
				    len))
sl@0
  1491
	goto oom;
sl@0
  1492
      if (!_dbus_string_append (xml, "\"/>\n"))
sl@0
  1493
	goto oom;
sl@0
  1494
sl@0
  1495
      _dbus_type_reader_next (&typereader);
sl@0
  1496
    }
sl@0
  1497
  return TRUE;
sl@0
  1498
 oom:
sl@0
  1499
  return FALSE;
sl@0
  1500
}
sl@0
  1501
sl@0
  1502
dbus_bool_t
sl@0
  1503
bus_driver_generate_introspect_string (DBusString *xml)
sl@0
  1504
{
sl@0
  1505
  int i;
sl@0
  1506
sl@0
  1507
  if (!_dbus_string_append (xml, DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE))
sl@0
  1508
    return FALSE;
sl@0
  1509
  if (!_dbus_string_append (xml, "<node>\n"))
sl@0
  1510
    return FALSE;
sl@0
  1511
  if (!_dbus_string_append_printf (xml, "  <interface name=\"%s\">\n", DBUS_INTERFACE_INTROSPECTABLE))
sl@0
  1512
    return FALSE;
sl@0
  1513
  if (!_dbus_string_append (xml, "    <method name=\"Introspect\">\n"))
sl@0
  1514
    return FALSE;
sl@0
  1515
  if (!_dbus_string_append_printf (xml, "      <arg name=\"data\" direction=\"out\" type=\"%s\"/>\n", DBUS_TYPE_STRING_AS_STRING))
sl@0
  1516
    return FALSE;
sl@0
  1517
  if (!_dbus_string_append (xml, "    </method>\n"))
sl@0
  1518
    return FALSE;
sl@0
  1519
  if (!_dbus_string_append (xml, "  </interface>\n"))
sl@0
  1520
    return FALSE;
sl@0
  1521
sl@0
  1522
  if (!_dbus_string_append_printf (xml, "  <interface name=\"%s\">\n",
sl@0
  1523
                                   DBUS_INTERFACE_DBUS))
sl@0
  1524
    return FALSE;
sl@0
  1525
sl@0
  1526
  i = 0;
sl@0
  1527
  while (i < _DBUS_N_ELEMENTS (message_handlers))
sl@0
  1528
    {
sl@0
  1529
	  
sl@0
  1530
      if (!_dbus_string_append_printf (xml, "    <method name=\"%s\">\n",
sl@0
  1531
                                       message_handlers[i].name))
sl@0
  1532
        return FALSE;
sl@0
  1533
sl@0
  1534
      if (!write_args_for_direction (xml, message_handlers[i].in_args, TRUE))
sl@0
  1535
	return FALSE;
sl@0
  1536
sl@0
  1537
      if (!write_args_for_direction (xml, message_handlers[i].out_args, FALSE))
sl@0
  1538
	return FALSE;
sl@0
  1539
sl@0
  1540
      if (!_dbus_string_append (xml, "    </method>\n"))
sl@0
  1541
	return FALSE;
sl@0
  1542
      
sl@0
  1543
      ++i;
sl@0
  1544
    }
sl@0
  1545
sl@0
  1546
  if (!_dbus_string_append_printf (xml, "    <signal name=\"NameOwnerChanged\">\n"))
sl@0
  1547
    return FALSE;
sl@0
  1548
  
sl@0
  1549
  if (!_dbus_string_append_printf (xml, "      <arg type=\"s\"/>\n"))
sl@0
  1550
    return FALSE;
sl@0
  1551
  
sl@0
  1552
  if (!_dbus_string_append_printf (xml, "      <arg type=\"s\"/>\n"))
sl@0
  1553
    return FALSE;
sl@0
  1554
  
sl@0
  1555
  if (!_dbus_string_append_printf (xml, "      <arg type=\"s\"/>\n"))
sl@0
  1556
    return FALSE;
sl@0
  1557
  
sl@0
  1558
  if (!_dbus_string_append_printf (xml, "    </signal>\n"))
sl@0
  1559
    return FALSE;
sl@0
  1560
sl@0
  1561
sl@0
  1562
sl@0
  1563
  if (!_dbus_string_append_printf (xml, "    <signal name=\"NameLost\">\n"))
sl@0
  1564
    return FALSE;
sl@0
  1565
  
sl@0
  1566
  if (!_dbus_string_append_printf (xml, "      <arg type=\"s\"/>\n"))
sl@0
  1567
    return FALSE;
sl@0
  1568
  
sl@0
  1569
  if (!_dbus_string_append_printf (xml, "    </signal>\n"))
sl@0
  1570
    return FALSE;
sl@0
  1571
sl@0
  1572
sl@0
  1573
sl@0
  1574
  if (!_dbus_string_append_printf (xml, "    <signal name=\"NameAcquired\">\n"))
sl@0
  1575
    return FALSE;
sl@0
  1576
  
sl@0
  1577
  if (!_dbus_string_append_printf (xml, "      <arg type=\"s\"/>\n"))
sl@0
  1578
    return FALSE;
sl@0
  1579
  
sl@0
  1580
  if (!_dbus_string_append_printf (xml, "    </signal>\n"))
sl@0
  1581
    return FALSE;
sl@0
  1582
sl@0
  1583
  if (!_dbus_string_append (xml, "  </interface>\n"))
sl@0
  1584
    return FALSE;
sl@0
  1585
  
sl@0
  1586
  if (!_dbus_string_append (xml, "</node>\n"))
sl@0
  1587
    return FALSE;
sl@0
  1588
sl@0
  1589
  return TRUE;
sl@0
  1590
}
sl@0
  1591
sl@0
  1592
static dbus_bool_t
sl@0
  1593
bus_driver_handle_introspect (DBusConnection *connection,
sl@0
  1594
                              BusTransaction *transaction,
sl@0
  1595
                              DBusMessage    *message,
sl@0
  1596
                              DBusError      *error)
sl@0
  1597
{
sl@0
  1598
  DBusString xml;
sl@0
  1599
  DBusMessage *reply;
sl@0
  1600
  const char *v_STRING;
sl@0
  1601
sl@0
  1602
  _dbus_verbose ("Introspect() on bus driver\n");
sl@0
  1603
  
sl@0
  1604
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1605
sl@0
  1606
  reply = NULL;
sl@0
  1607
sl@0
  1608
  if (! dbus_message_get_args (message, error,
sl@0
  1609
			       DBUS_TYPE_INVALID))
sl@0
  1610
    {
sl@0
  1611
      _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1612
      return FALSE;
sl@0
  1613
    }
sl@0
  1614
sl@0
  1615
  if (!_dbus_string_init (&xml))
sl@0
  1616
    {
sl@0
  1617
      BUS_SET_OOM (error);
sl@0
  1618
      return FALSE;
sl@0
  1619
    }
sl@0
  1620
sl@0
  1621
  if (!bus_driver_generate_introspect_string (&xml))
sl@0
  1622
    goto oom;
sl@0
  1623
sl@0
  1624
  v_STRING = _dbus_string_get_const_data (&xml);
sl@0
  1625
sl@0
  1626
  reply = dbus_message_new_method_return (message);
sl@0
  1627
  if (reply == NULL)
sl@0
  1628
    goto oom;
sl@0
  1629
sl@0
  1630
  if (! dbus_message_append_args (reply,
sl@0
  1631
                                  DBUS_TYPE_STRING, &v_STRING,
sl@0
  1632
                                  DBUS_TYPE_INVALID))
sl@0
  1633
    goto oom;
sl@0
  1634
sl@0
  1635
  if (! bus_transaction_send_from_driver (transaction, connection, reply))
sl@0
  1636
    goto oom;
sl@0
  1637
sl@0
  1638
  dbus_message_unref (reply);
sl@0
  1639
  _dbus_string_free (&xml);
sl@0
  1640
sl@0
  1641
  return TRUE;
sl@0
  1642
sl@0
  1643
 oom:
sl@0
  1644
  BUS_SET_OOM (error);
sl@0
  1645
sl@0
  1646
  if (reply)
sl@0
  1647
    dbus_message_unref (reply);
sl@0
  1648
sl@0
  1649
  _dbus_string_free (&xml);
sl@0
  1650
  
sl@0
  1651
  return FALSE;
sl@0
  1652
}
sl@0
  1653
sl@0
  1654
dbus_bool_t
sl@0
  1655
bus_driver_handle_message (DBusConnection *connection,
sl@0
  1656
                           BusTransaction *transaction,
sl@0
  1657
			   DBusMessage    *message,
sl@0
  1658
                           DBusError      *error)
sl@0
  1659
{
sl@0
  1660
  const char *name, *sender, *interface;
sl@0
  1661
  int i;
sl@0
  1662
sl@0
  1663
  _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1664
sl@0
  1665
  if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_METHOD_CALL)
sl@0
  1666
    {
sl@0
  1667
      _dbus_verbose ("Driver got a non-method-call message, ignoring\n");
sl@0
  1668
      return TRUE; /* we just ignore this */
sl@0
  1669
    }
sl@0
  1670
sl@0
  1671
  if (dbus_message_is_method_call (message,
sl@0
  1672
                                   DBUS_INTERFACE_INTROSPECTABLE,
sl@0
  1673
                                   "Introspect"))
sl@0
  1674
    return bus_driver_handle_introspect (connection, transaction, message, error);
sl@0
  1675
  
sl@0
  1676
  interface = dbus_message_get_interface (message);
sl@0
  1677
  if (interface == NULL)
sl@0
  1678
    interface = DBUS_INTERFACE_DBUS;
sl@0
  1679
  
sl@0
  1680
  _dbus_assert (dbus_message_get_member (message) != NULL);
sl@0
  1681
  
sl@0
  1682
  name = dbus_message_get_member (message);
sl@0
  1683
  sender = dbus_message_get_sender (message);
sl@0
  1684
  
sl@0
  1685
  if (strcmp (interface,
sl@0
  1686
              DBUS_INTERFACE_DBUS) != 0)
sl@0
  1687
    {
sl@0
  1688
      _dbus_verbose ("Driver got message to unknown interface \"%s\"\n",
sl@0
  1689
                     interface);
sl@0
  1690
      goto unknown;
sl@0
  1691
    }
sl@0
  1692
  
sl@0
  1693
  _dbus_verbose ("Driver got a method call: %s\n",
sl@0
  1694
		 dbus_message_get_member (message));
sl@0
  1695
  
sl@0
  1696
  /* security checks should have kept this from getting here */
sl@0
  1697
  _dbus_assert (sender != NULL || strcmp (name, "Hello") == 0);
sl@0
  1698
  
sl@0
  1699
  i = 0;
sl@0
  1700
  while (i < _DBUS_N_ELEMENTS (message_handlers))
sl@0
  1701
    {
sl@0
  1702
      if (strcmp (message_handlers[i].name, name) == 0)
sl@0
  1703
        {
sl@0
  1704
          _dbus_verbose ("Found driver handler for %s\n", name);
sl@0
  1705
sl@0
  1706
          if (!dbus_message_has_signature (message, message_handlers[i].in_args))
sl@0
  1707
            {
sl@0
  1708
              _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1709
              _dbus_verbose ("Call to %s has wrong args (%s, expected %s)\n",
sl@0
  1710
                             name, dbus_message_get_signature (message),
sl@0
  1711
                             message_handlers[i].in_args);
sl@0
  1712
              
sl@0
  1713
              dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
sl@0
  1714
                              "Call to %s has wrong args (%s, expected %s)\n",
sl@0
  1715
                              name, dbus_message_get_signature (message),
sl@0
  1716
                              message_handlers[i].in_args);
sl@0
  1717
              _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1718
              return FALSE;
sl@0
  1719
            }
sl@0
  1720
          
sl@0
  1721
          if ((* message_handlers[i].handler) (connection, transaction, message, error))
sl@0
  1722
            {
sl@0
  1723
              _DBUS_ASSERT_ERROR_IS_CLEAR (error);
sl@0
  1724
              _dbus_verbose ("Driver handler succeeded\n");
sl@0
  1725
              return TRUE;
sl@0
  1726
            }
sl@0
  1727
          else
sl@0
  1728
            {
sl@0
  1729
              _DBUS_ASSERT_ERROR_IS_SET (error);
sl@0
  1730
              _dbus_verbose ("Driver handler returned failure\n");
sl@0
  1731
              return FALSE;
sl@0
  1732
            }
sl@0
  1733
        }
sl@0
  1734
      
sl@0
  1735
      ++i;
sl@0
  1736
    }
sl@0
  1737
sl@0
  1738
 unknown:
sl@0
  1739
  _dbus_verbose ("No driver handler for message \"%s\"\n",
sl@0
  1740
                 name);
sl@0
  1741
sl@0
  1742
  dbus_set_error (error, DBUS_ERROR_UNKNOWN_METHOD,
sl@0
  1743
                  "%s does not understand message %s",
sl@0
  1744
                  DBUS_SERVICE_DBUS, name);
sl@0
  1745
  
sl@0
  1746
  return FALSE;
sl@0
  1747
}
sl@0
  1748
sl@0
  1749
void
sl@0
  1750
bus_driver_remove_connection (DBusConnection *connection)
sl@0
  1751
{
sl@0
  1752
  /* FIXME 1.0 Does nothing for now, should unregister the connection
sl@0
  1753
   * with the bus driver.
sl@0
  1754
   */
sl@0
  1755
}