os/ossrv/ofdbus/dbus/bus/test.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
/* test.c  unit test routines
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
sl@0
    24
#ifndef __SYMBIAN32__
sl@0
    25
#include <config.h>
sl@0
    26
#else
sl@0
    27
#include "config.h"
sl@0
    28
#endif //__SYMBIAN32__
sl@0
    29
sl@0
    30
#ifdef DBUS_BUILD_TESTS
sl@0
    31
#include "test.h"
sl@0
    32
#ifndef __SYMBIAN32__
sl@0
    33
#include <dbus/dbus-internals.h>
sl@0
    34
#include <dbus/dbus-list.h>
sl@0
    35
#else
sl@0
    36
#include "dbus-internals.h"
sl@0
    37
#include "dbus-list.h"
sl@0
    38
#endif //__SYMBIAN32__
sl@0
    39
sl@0
    40
/* The "debug client" watch/timeout handlers don't dispatch messages,
sl@0
    41
 * as we manually pull them in order to verify them. This is why they
sl@0
    42
 * are different from the real handlers in connection.c
sl@0
    43
 */
sl@0
    44
static DBusList *clients = NULL;
sl@0
    45
static DBusLoop *client_loop = NULL;
sl@0
    46
sl@0
    47
static dbus_bool_t
sl@0
    48
client_watch_callback (DBusWatch     *watch,
sl@0
    49
                       unsigned int   condition,
sl@0
    50
                       void          *data)
sl@0
    51
{
sl@0
    52
  /* FIXME this can be done in dbus-mainloop.c
sl@0
    53
   * if the code in activation.c for the babysitter
sl@0
    54
   * watch handler is fixed.
sl@0
    55
   */
sl@0
    56
 
sl@0
    57
  return dbus_watch_handle (watch, condition);
sl@0
    58
}
sl@0
    59
sl@0
    60
static dbus_bool_t
sl@0
    61
add_client_watch (DBusWatch      *watch,
sl@0
    62
                  void           *data)
sl@0
    63
{
sl@0
    64
  DBusConnection *connection = data;
sl@0
    65
  
sl@0
    66
  return _dbus_loop_add_watch (client_loop,
sl@0
    67
                               watch, client_watch_callback, connection,
sl@0
    68
                               NULL);
sl@0
    69
}
sl@0
    70
sl@0
    71
static void
sl@0
    72
remove_client_watch (DBusWatch      *watch,
sl@0
    73
                     void           *data)
sl@0
    74
{
sl@0
    75
  DBusConnection *connection = data;
sl@0
    76
  
sl@0
    77
  _dbus_loop_remove_watch (client_loop,
sl@0
    78
                           watch, client_watch_callback, connection);
sl@0
    79
}
sl@0
    80
sl@0
    81
static void
sl@0
    82
client_timeout_callback (DBusTimeout   *timeout,
sl@0
    83
                         void          *data)
sl@0
    84
{
sl@0
    85
  DBusConnection *connection = data;
sl@0
    86
sl@0
    87
  dbus_connection_ref (connection);
sl@0
    88
sl@0
    89
  /* can return FALSE on OOM but we just let it fire again later */
sl@0
    90
  dbus_timeout_handle (timeout);
sl@0
    91
sl@0
    92
  dbus_connection_unref (connection);
sl@0
    93
}
sl@0
    94
sl@0
    95
static dbus_bool_t
sl@0
    96
add_client_timeout (DBusTimeout    *timeout,
sl@0
    97
                    void           *data)
sl@0
    98
{
sl@0
    99
  DBusConnection *connection = data;
sl@0
   100
  
sl@0
   101
  return _dbus_loop_add_timeout (client_loop, timeout, client_timeout_callback, connection, NULL);
sl@0
   102
}
sl@0
   103
sl@0
   104
static void
sl@0
   105
remove_client_timeout (DBusTimeout    *timeout,
sl@0
   106
                       void           *data)
sl@0
   107
{
sl@0
   108
  DBusConnection *connection = data;
sl@0
   109
  
sl@0
   110
  _dbus_loop_remove_timeout (client_loop, timeout, client_timeout_callback, connection);
sl@0
   111
}
sl@0
   112
sl@0
   113
static DBusHandlerResult
sl@0
   114
client_disconnect_filter (DBusConnection     *connection,
sl@0
   115
                          DBusMessage        *message,
sl@0
   116
                          void               *user_data)
sl@0
   117
{
sl@0
   118
  if (!dbus_message_is_signal (message,
sl@0
   119
                               DBUS_INTERFACE_LOCAL,
sl@0
   120
                               "Disconnected"))
sl@0
   121
    return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
sl@0
   122
    
sl@0
   123
  _dbus_verbose ("Removing client %p in disconnect handler\n",
sl@0
   124
                 connection);
sl@0
   125
  
sl@0
   126
  _dbus_list_remove (&clients, connection);
sl@0
   127
sl@0
   128
  dbus_connection_unref (connection);
sl@0
   129
  
sl@0
   130
  if (clients == NULL)
sl@0
   131
    {
sl@0
   132
      _dbus_loop_unref (client_loop);
sl@0
   133
      client_loop = NULL;
sl@0
   134
    }
sl@0
   135
  
sl@0
   136
  return DBUS_HANDLER_RESULT_HANDLED;
sl@0
   137
}
sl@0
   138
sl@0
   139
dbus_bool_t
sl@0
   140
bus_setup_debug_client (DBusConnection *connection)
sl@0
   141
{
sl@0
   142
  dbus_bool_t retval;  
sl@0
   143
sl@0
   144
  if (!dbus_connection_add_filter (connection,
sl@0
   145
                                   client_disconnect_filter,
sl@0
   146
                                   NULL, NULL))
sl@0
   147
    return FALSE;
sl@0
   148
sl@0
   149
  retval = FALSE;
sl@0
   150
sl@0
   151
  if (client_loop == NULL)
sl@0
   152
    {
sl@0
   153
      client_loop = _dbus_loop_new ();
sl@0
   154
      if (client_loop == NULL)
sl@0
   155
        goto out;
sl@0
   156
    }
sl@0
   157
  
sl@0
   158
  if (!dbus_connection_set_watch_functions (connection,
sl@0
   159
                                            add_client_watch,
sl@0
   160
                                            remove_client_watch,
sl@0
   161
                                            NULL,
sl@0
   162
                                            connection,
sl@0
   163
                                            NULL))
sl@0
   164
    goto out;
sl@0
   165
      
sl@0
   166
  if (!dbus_connection_set_timeout_functions (connection,
sl@0
   167
                                              add_client_timeout,
sl@0
   168
                                              remove_client_timeout,
sl@0
   169
                                              NULL,
sl@0
   170
                                              connection, NULL))
sl@0
   171
    goto out;
sl@0
   172
sl@0
   173
  if (!_dbus_list_append (&clients, connection))
sl@0
   174
    goto out;
sl@0
   175
  
sl@0
   176
  retval = TRUE;
sl@0
   177
  
sl@0
   178
 out:
sl@0
   179
  if (!retval)
sl@0
   180
    {
sl@0
   181
      dbus_connection_remove_filter (connection,
sl@0
   182
                                     client_disconnect_filter,
sl@0
   183
                                     NULL);
sl@0
   184
      
sl@0
   185
      dbus_connection_set_watch_functions (connection,
sl@0
   186
                                           NULL, NULL, NULL, NULL, NULL);
sl@0
   187
      dbus_connection_set_timeout_functions (connection,
sl@0
   188
                                             NULL, NULL, NULL, NULL, NULL);
sl@0
   189
sl@0
   190
      _dbus_list_remove_last (&clients, connection);
sl@0
   191
sl@0
   192
      if (clients == NULL)
sl@0
   193
        {
sl@0
   194
          _dbus_loop_unref (client_loop);
sl@0
   195
          client_loop = NULL;
sl@0
   196
        }
sl@0
   197
    }
sl@0
   198
      
sl@0
   199
  return retval;
sl@0
   200
}
sl@0
   201
sl@0
   202
void
sl@0
   203
bus_test_clients_foreach (BusConnectionForeachFunction  function,
sl@0
   204
                          void                         *data)
sl@0
   205
{
sl@0
   206
  DBusList *link;
sl@0
   207
  
sl@0
   208
  link = _dbus_list_get_first_link (&clients);
sl@0
   209
  while (link != NULL)
sl@0
   210
    {
sl@0
   211
      DBusConnection *connection = link->data;
sl@0
   212
      DBusList *next = _dbus_list_get_next_link (&clients, link);
sl@0
   213
sl@0
   214
      if (!(* function) (connection, data))
sl@0
   215
        break;
sl@0
   216
      
sl@0
   217
      link = next;
sl@0
   218
    }
sl@0
   219
}
sl@0
   220
sl@0
   221
dbus_bool_t
sl@0
   222
bus_test_client_listed (DBusConnection *connection)
sl@0
   223
{
sl@0
   224
  DBusList *link;
sl@0
   225
  
sl@0
   226
  link = _dbus_list_get_first_link (&clients);
sl@0
   227
  while (link != NULL)
sl@0
   228
    {
sl@0
   229
      DBusConnection *c = link->data;
sl@0
   230
      DBusList *next = _dbus_list_get_next_link (&clients, link);
sl@0
   231
sl@0
   232
      if (c == connection)
sl@0
   233
        return TRUE;
sl@0
   234
      
sl@0
   235
      link = next;
sl@0
   236
    }
sl@0
   237
sl@0
   238
  return FALSE;
sl@0
   239
}
sl@0
   240
sl@0
   241
void
sl@0
   242
bus_test_run_clients_loop (dbus_bool_t block_once)
sl@0
   243
{  
sl@0
   244
  if (client_loop == NULL)
sl@0
   245
    return;
sl@0
   246
sl@0
   247
  _dbus_verbose ("---> Dispatching on \"client side\"\n");
sl@0
   248
  
sl@0
   249
  /* dispatch before we block so pending dispatches
sl@0
   250
   * won't make our block return early
sl@0
   251
   */
sl@0
   252
  _dbus_loop_dispatch (client_loop);
sl@0
   253
  
sl@0
   254
  /* Do one blocking wait, since we're expecting data */
sl@0
   255
  if (block_once)
sl@0
   256
    {
sl@0
   257
      _dbus_verbose ("---> blocking on \"client side\"\n");
sl@0
   258
      _dbus_loop_iterate (client_loop, TRUE);
sl@0
   259
    }
sl@0
   260
sl@0
   261
  /* Then mop everything up */
sl@0
   262
  while (_dbus_loop_iterate (client_loop, FALSE))
sl@0
   263
    ;
sl@0
   264
sl@0
   265
  _dbus_verbose ("---> Done dispatching on \"client side\"\n");
sl@0
   266
}
sl@0
   267
sl@0
   268
void
sl@0
   269
bus_test_run_bus_loop (BusContext *context,
sl@0
   270
                       dbus_bool_t block_once)
sl@0
   271
{
sl@0
   272
  _dbus_verbose ("---> Dispatching on \"server side\"\n");
sl@0
   273
  
sl@0
   274
  /* dispatch before we block so pending dispatches
sl@0
   275
   * won't make our block return early
sl@0
   276
   */
sl@0
   277
  _dbus_loop_dispatch (bus_context_get_loop (context));
sl@0
   278
  
sl@0
   279
  /* Do one blocking wait, since we're expecting data */
sl@0
   280
  if (block_once)
sl@0
   281
    {
sl@0
   282
      _dbus_verbose ("---> blocking on \"server side\"\n");
sl@0
   283
      _dbus_loop_iterate (bus_context_get_loop (context), TRUE);
sl@0
   284
    }
sl@0
   285
sl@0
   286
  /* Then mop everything up */
sl@0
   287
  while (_dbus_loop_iterate (bus_context_get_loop (context), FALSE))
sl@0
   288
    ;
sl@0
   289
sl@0
   290
  _dbus_verbose ("---> Done dispatching on \"server side\"\n");
sl@0
   291
}
sl@0
   292
sl@0
   293
void
sl@0
   294
bus_test_run_everything (BusContext *context)
sl@0
   295
{
sl@0
   296
  while (_dbus_loop_iterate (bus_context_get_loop (context), FALSE) ||
sl@0
   297
         (client_loop == NULL || _dbus_loop_iterate (client_loop, FALSE)))
sl@0
   298
    ;
sl@0
   299
}
sl@0
   300
sl@0
   301
BusContext*
sl@0
   302
bus_context_new_test (const DBusString *test_data_dir,
sl@0
   303
                      const char       *filename)
sl@0
   304
{
sl@0
   305
  DBusError error;
sl@0
   306
  DBusString config_file;
sl@0
   307
  DBusString relative;
sl@0
   308
  BusContext *context;
sl@0
   309
  
sl@0
   310
  if (!_dbus_string_init (&config_file))
sl@0
   311
    {
sl@0
   312
      _dbus_warn ("No memory\n");
sl@0
   313
      return NULL;
sl@0
   314
    }
sl@0
   315
sl@0
   316
  if (!_dbus_string_copy (test_data_dir, 0,
sl@0
   317
                          &config_file, 0))
sl@0
   318
    {
sl@0
   319
      _dbus_warn ("No memory\n");
sl@0
   320
      _dbus_string_free (&config_file);
sl@0
   321
      return NULL;
sl@0
   322
    }
sl@0
   323
sl@0
   324
  _dbus_string_init_const (&relative, filename);
sl@0
   325
sl@0
   326
  if (!_dbus_concat_dir_and_file (&config_file, &relative))
sl@0
   327
    {
sl@0
   328
      _dbus_warn ("No memory\n");
sl@0
   329
      _dbus_string_free (&config_file);
sl@0
   330
      return NULL;
sl@0
   331
    }
sl@0
   332
  
sl@0
   333
  dbus_error_init (&error);
sl@0
   334
  context = bus_context_new (&config_file, FALSE, -1, -1, &error);
sl@0
   335
  if (context == NULL)
sl@0
   336
    {
sl@0
   337
      _DBUS_ASSERT_ERROR_IS_SET (&error);
sl@0
   338
      
sl@0
   339
      _dbus_warn ("Failed to create debug bus context from configuration file %s: %s\n",
sl@0
   340
                  filename, error.message);
sl@0
   341
sl@0
   342
      dbus_error_free (&error);
sl@0
   343
      
sl@0
   344
      _dbus_string_free (&config_file);
sl@0
   345
      
sl@0
   346
      return NULL;
sl@0
   347
    }
sl@0
   348
sl@0
   349
  _dbus_string_free (&config_file);
sl@0
   350
  
sl@0
   351
  return context;
sl@0
   352
}
sl@0
   353
sl@0
   354
#endif