1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/ofdbus/dbus/bus/dispatch.c Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,4196 @@
1.4 +/* -*- mode: C; c-file-style: "gnu" -*- */
1.5 +/* dispatch.c Message dispatcher
1.6 + *
1.7 + * Copyright (C) 2003 CodeFactory AB
1.8 + * Copyright (C) 2003, 2004, 2005 Red Hat, Inc.
1.9 + * Copyright (C) 2004 Imendio HB
1.10 + * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
1.11 + * Licensed under the Academic Free License version 2.1
1.12 + *
1.13 + * This program is free software; you can redistribute it and/or modify
1.14 + * it under the terms of the GNU General Public License as published by
1.15 + * the Free Software Foundation; either version 2 of the License, or
1.16 + * (at your option) any later version.
1.17 + *
1.18 + * This program is distributed in the hope that it will be useful,
1.19 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
1.20 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1.21 + * GNU General Public License for more details.
1.22 + *
1.23 + * You should have received a copy of the GNU General Public License
1.24 + * along with this program; if not, write to the Free Software
1.25 + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
1.26 + *
1.27 + */
1.28 +
1.29 +#include "dispatch.h"
1.30 +#include "connection.h"
1.31 +#include "driver.h"
1.32 +#include "services.h"
1.33 +#include "activation.h"
1.34 +#include "utils.h"
1.35 +#include "bus.h"
1.36 +#include "signals.h"
1.37 +#include "test.h"
1.38 +#ifndef __SYMBIAN32__
1.39 +#include <dbus/dbus-internals.h>
1.40 +#else
1.41 +#include "dbus-internals.h"
1.42 +#endif //__SYMBIAN32__
1.43 +#include <string.h>
1.44 +
1.45 +#ifdef __SYMBIAN32__
1.46 +#include "config.h"
1.47 +#endif //__SYMBIAN32__
1.48 +
1.49 +static dbus_bool_t
1.50 +send_one_message (DBusConnection *connection,
1.51 + BusContext *context,
1.52 + DBusConnection *sender,
1.53 + DBusConnection *addressed_recipient,
1.54 + DBusMessage *message,
1.55 + BusTransaction *transaction,
1.56 + DBusError *error)
1.57 +{
1.58 + if (!bus_context_check_security_policy (context, transaction,
1.59 + sender,
1.60 + addressed_recipient,
1.61 + connection,
1.62 + message,
1.63 + NULL))
1.64 + return TRUE; /* silently don't send it */
1.65 +
1.66 + if (!bus_transaction_send (transaction,
1.67 + connection,
1.68 + message))
1.69 + {
1.70 + BUS_SET_OOM (error);
1.71 + return FALSE;
1.72 + }
1.73 +
1.74 + return TRUE;
1.75 +}
1.76 +
1.77 +dbus_bool_t
1.78 +bus_dispatch_matches (BusTransaction *transaction,
1.79 + DBusConnection *sender,
1.80 + DBusConnection *addressed_recipient,
1.81 + DBusMessage *message,
1.82 + DBusError *error)
1.83 +{
1.84 + DBusError tmp_error;
1.85 + BusConnections *connections;
1.86 + DBusList *recipients;
1.87 + BusMatchmaker *matchmaker;
1.88 + DBusList *link;
1.89 + BusContext *context;
1.90 +
1.91 + _DBUS_ASSERT_ERROR_IS_CLEAR (error);
1.92 +
1.93 + /* sender and recipient can both be NULL for the bus driver,
1.94 + * or for signals with no particular recipient
1.95 + */
1.96 +
1.97 + _dbus_assert (sender == NULL || bus_connection_is_active (sender));
1.98 + _dbus_assert (dbus_message_get_sender (message) != NULL);
1.99 +
1.100 + connections = bus_transaction_get_connections (transaction);
1.101 +
1.102 + dbus_error_init (&tmp_error);
1.103 + context = bus_transaction_get_context (transaction);
1.104 + matchmaker = bus_context_get_matchmaker (context);
1.105 +
1.106 + recipients = NULL;
1.107 + if (!bus_matchmaker_get_recipients (matchmaker, connections,
1.108 + sender, addressed_recipient, message,
1.109 + &recipients))
1.110 + {
1.111 + BUS_SET_OOM (error);
1.112 + return FALSE;
1.113 + }
1.114 +
1.115 + link = _dbus_list_get_first_link (&recipients);
1.116 + while (link != NULL)
1.117 + {
1.118 + DBusConnection *dest;
1.119 +
1.120 + dest = link->data;
1.121 +
1.122 + if (!send_one_message (dest, context, sender, addressed_recipient,
1.123 + message, transaction, &tmp_error))
1.124 + break;
1.125 +
1.126 + link = _dbus_list_get_next_link (&recipients, link);
1.127 + }
1.128 +
1.129 + _dbus_list_clear (&recipients);
1.130 +
1.131 + if (dbus_error_is_set (&tmp_error))
1.132 + {
1.133 + dbus_move_error (&tmp_error, error);
1.134 + return FALSE;
1.135 + }
1.136 + else
1.137 + return TRUE;
1.138 +}
1.139 +
1.140 +static DBusHandlerResult
1.141 +bus_dispatch (DBusConnection *connection,
1.142 + DBusMessage *message)
1.143 +{
1.144 + const char *sender, *service_name;
1.145 + DBusError error;
1.146 + BusTransaction *transaction;
1.147 + BusContext *context;
1.148 + DBusHandlerResult result;
1.149 + DBusConnection *addressed_recipient;
1.150 +
1.151 + result = DBUS_HANDLER_RESULT_HANDLED;
1.152 +
1.153 + transaction = NULL;
1.154 + addressed_recipient = NULL;
1.155 + dbus_error_init (&error);
1.156 +
1.157 + context = bus_connection_get_context (connection);
1.158 + _dbus_assert (context != NULL);
1.159 +
1.160 + /* If we can't even allocate an OOM error, we just go to sleep
1.161 + * until we can.
1.162 + */
1.163 + while (!bus_connection_preallocate_oom_error (connection))
1.164 + _dbus_wait_for_memory ();
1.165 +
1.166 + /* Ref connection in case we disconnect it at some point in here */
1.167 + dbus_connection_ref (connection);
1.168 +
1.169 + service_name = dbus_message_get_destination (message);
1.170 +
1.171 +#ifdef DBUS_ENABLE_VERBOSE_MODE
1.172 + {
1.173 + const char *interface_name, *member_name, *error_name;
1.174 +
1.175 + interface_name = dbus_message_get_interface (message);
1.176 + member_name = dbus_message_get_member (message);
1.177 + error_name = dbus_message_get_error_name (message);
1.178 +
1.179 + _dbus_verbose ("DISPATCH: %s %s %s to %s\n",
1.180 + interface_name ? interface_name : "(no interface)",
1.181 + member_name ? member_name : "(no member)",
1.182 + error_name ? error_name : "(no error name)",
1.183 + service_name ? service_name : "peer");
1.184 + }
1.185 +#endif /* DBUS_ENABLE_VERBOSE_MODE */
1.186 +
1.187 + /* If service_name is NULL, if it's a signal we send it to all
1.188 + * connections with a match rule. If it's not a signal, there
1.189 + * are some special cases here but mostly we just bail out.
1.190 + */
1.191 + if (service_name == NULL)
1.192 + {
1.193 + if (dbus_message_is_signal (message,
1.194 + DBUS_INTERFACE_LOCAL,
1.195 + "Disconnected"))
1.196 + {
1.197 + bus_connection_disconnected (connection);
1.198 + goto out;
1.199 + }
1.200 +
1.201 + if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_SIGNAL)
1.202 + {
1.203 + /* DBusConnection also handles some of these automatically, we leave
1.204 + * it to do so.
1.205 + */
1.206 + result = DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
1.207 + goto out;
1.208 + }
1.209 + }
1.210 +
1.211 + /* Create our transaction */
1.212 + transaction = bus_transaction_new (context);
1.213 + if (transaction == NULL)
1.214 + {
1.215 + BUS_SET_OOM (&error);
1.216 + goto out;
1.217 + }
1.218 +
1.219 + /* Assign a sender to the message */
1.220 + if (bus_connection_is_active (connection))
1.221 + {
1.222 + sender = bus_connection_get_name (connection);
1.223 + _dbus_assert (sender != NULL);
1.224 +
1.225 + if (!dbus_message_set_sender (message, sender))
1.226 + {
1.227 + BUS_SET_OOM (&error);
1.228 + goto out;
1.229 + }
1.230 +
1.231 + /* We need to refetch the service name here, because
1.232 + * dbus_message_set_sender can cause the header to be
1.233 + * reallocated, and thus the service_name pointer will become
1.234 + * invalid.
1.235 + */
1.236 + service_name = dbus_message_get_destination (message);
1.237 + }
1.238 +
1.239 + if (service_name &&
1.240 + strcmp (service_name, DBUS_SERVICE_DBUS) == 0) /* to bus driver */
1.241 + {
1.242 + if (!bus_context_check_security_policy (context, transaction,
1.243 + connection, NULL, NULL, message, &error))
1.244 + {
1.245 + _dbus_verbose ("Security policy rejected message\n");
1.246 + goto out;
1.247 + }
1.248 +
1.249 + _dbus_verbose ("Giving message to %s\n", DBUS_SERVICE_DBUS);
1.250 + if (!bus_driver_handle_message (connection, transaction, message, &error))
1.251 + goto out;
1.252 + }
1.253 + else if (!bus_connection_is_active (connection)) /* clients must talk to bus driver first */
1.254 + {
1.255 + _dbus_verbose ("Received message from non-registered client. Disconnecting.\n");
1.256 + dbus_connection_close (connection);
1.257 + goto out;
1.258 + }
1.259 + else if (service_name != NULL) /* route to named service */
1.260 + {
1.261 + DBusString service_string;
1.262 + BusService *service;
1.263 + BusRegistry *registry;
1.264 +
1.265 + _dbus_assert (service_name != NULL);
1.266 +
1.267 + registry = bus_connection_get_registry (connection);
1.268 +
1.269 + _dbus_string_init_const (&service_string, service_name);
1.270 + service = bus_registry_lookup (registry, &service_string);
1.271 +
1.272 + if (service == NULL && dbus_message_get_auto_start (message))
1.273 + {
1.274 + BusActivation *activation;
1.275 + /* We can't do the security policy check here, since the addressed
1.276 + * recipient service doesn't exist yet. We do it before sending the
1.277 + * message after the service has been created.
1.278 + */
1.279 + activation = bus_connection_get_activation (connection);
1.280 +
1.281 + if (!bus_activation_activate_service (activation, connection, transaction, TRUE,
1.282 + message, service_name, &error))
1.283 + {
1.284 + _DBUS_ASSERT_ERROR_IS_SET (&error);
1.285 + _dbus_verbose ("bus_activation_activate_service() failed: %s\n", error.name);
1.286 + goto out;
1.287 + }
1.288 +
1.289 + goto out;
1.290 + }
1.291 + else if (service == NULL)
1.292 + {
1.293 + dbus_set_error (&error,
1.294 + DBUS_ERROR_NAME_HAS_NO_OWNER,
1.295 + "Name \"%s\" does not exist",
1.296 + service_name);
1.297 + goto out;
1.298 + }
1.299 + else
1.300 + {
1.301 + addressed_recipient = bus_service_get_primary_owners_connection (service);
1.302 + _dbus_assert (addressed_recipient != NULL);
1.303 +
1.304 + if (!bus_context_check_security_policy (context, transaction,
1.305 + connection, addressed_recipient,
1.306 + addressed_recipient,
1.307 + message, &error))
1.308 + goto out;
1.309 +
1.310 + /* Dispatch the message */
1.311 + if (!bus_transaction_send (transaction, addressed_recipient, message))
1.312 + {
1.313 + BUS_SET_OOM (&error);
1.314 + goto out;
1.315 + }
1.316 + }
1.317 + }
1.318 +
1.319 + /* Now match the messages against any match rules, which will send
1.320 + * out signals and such. addressed_recipient may == NULL.
1.321 + */
1.322 + if (!bus_dispatch_matches (transaction, connection, addressed_recipient, message, &error))
1.323 + goto out;
1.324 +
1.325 + out:
1.326 + if (dbus_error_is_set (&error))
1.327 + {
1.328 + if (!dbus_connection_get_is_connected (connection))
1.329 + {
1.330 + /* If we disconnected it, we won't bother to send it any error
1.331 + * messages.
1.332 + */
1.333 + _dbus_verbose ("Not sending error to connection we disconnected\n");
1.334 + }
1.335 + else if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.336 + {
1.337 + bus_connection_send_oom_error (connection, message);
1.338 +
1.339 + /* cancel transaction due to OOM */
1.340 + if (transaction != NULL)
1.341 + {
1.342 + bus_transaction_cancel_and_free (transaction);
1.343 + transaction = NULL;
1.344 + }
1.345 + }
1.346 + else
1.347 + {
1.348 + /* Try to send the real error, if no mem to do that, send
1.349 + * the OOM error
1.350 + */
1.351 + _dbus_assert (transaction != NULL);
1.352 + if (!bus_transaction_send_error_reply (transaction, connection,
1.353 + &error, message))
1.354 + {
1.355 + bus_connection_send_oom_error (connection, message);
1.356 +
1.357 + /* cancel transaction due to OOM */
1.358 + if (transaction != NULL)
1.359 + {
1.360 + bus_transaction_cancel_and_free (transaction);
1.361 + transaction = NULL;
1.362 + }
1.363 + }
1.364 + }
1.365 +
1.366 +
1.367 + dbus_error_free (&error);
1.368 + }
1.369 +
1.370 + if (transaction != NULL)
1.371 + {
1.372 + bus_transaction_execute_and_free (transaction);
1.373 + }
1.374 +
1.375 + dbus_connection_unref (connection);
1.376 +
1.377 + return result;
1.378 +}
1.379 +
1.380 +static DBusHandlerResult
1.381 +bus_dispatch_message_filter (DBusConnection *connection,
1.382 + DBusMessage *message,
1.383 + void *user_data)
1.384 +{
1.385 + return bus_dispatch (connection, message);
1.386 +}
1.387 +
1.388 +dbus_bool_t
1.389 +bus_dispatch_add_connection (DBusConnection *connection)
1.390 +{
1.391 + if (!dbus_connection_add_filter (connection,
1.392 + bus_dispatch_message_filter,
1.393 + NULL, NULL))
1.394 + return FALSE;
1.395 +
1.396 + return TRUE;
1.397 +}
1.398 +
1.399 +void
1.400 +bus_dispatch_remove_connection (DBusConnection *connection)
1.401 +{
1.402 + /* Here we tell the bus driver that we want to get off. */
1.403 + bus_driver_remove_connection (connection);
1.404 +
1.405 + dbus_connection_remove_filter (connection,
1.406 + bus_dispatch_message_filter,
1.407 + NULL);
1.408 +}
1.409 +
1.410 +#ifdef DBUS_BUILD_TESTS
1.411 +
1.412 +#include <stdio.h>
1.413 +
1.414 +/* This is used to know whether we need to block in order to finish
1.415 + * sending a message, or whether the initial dbus_connection_send()
1.416 + * already flushed the queue.
1.417 + */
1.418 +#define SEND_PENDING(connection) (dbus_connection_has_messages_to_send (connection))
1.419 +
1.420 +typedef dbus_bool_t (* Check1Func) (BusContext *context);
1.421 +typedef dbus_bool_t (* Check2Func) (BusContext *context,
1.422 + DBusConnection *connection);
1.423 +
1.424 +static dbus_bool_t check_no_leftovers (BusContext *context);
1.425 +
1.426 +static void
1.427 +block_connection_until_message_from_bus (BusContext *context,
1.428 + DBusConnection *connection,
1.429 + const char *what_is_expected)
1.430 +{
1.431 + _dbus_verbose ("expecting: %s\n", what_is_expected);
1.432 +
1.433 + while (dbus_connection_get_dispatch_status (connection) ==
1.434 + DBUS_DISPATCH_COMPLETE &&
1.435 + dbus_connection_get_is_connected (connection))
1.436 + {
1.437 + #ifndef __SYMBIAN32__
1.438 + bus_test_run_bus_loop (context, TRUE);
1.439 + #else //_dbus_loop_iterate() blocks indefinitly on setting argument to TRUE on Symbian,it must be investigated
1.440 + bus_test_run_bus_loop (context, FALSE);
1.441 + #endif
1.442 + bus_test_run_clients_loop (FALSE);
1.443 + }
1.444 +}
1.445 +
1.446 +static void
1.447 +spin_connection_until_authenticated (BusContext *context,
1.448 + DBusConnection *connection)
1.449 +{
1.450 + _dbus_verbose ("Spinning to auth connection %p\n", connection);
1.451 + while (!dbus_connection_get_is_authenticated (connection) &&
1.452 + dbus_connection_get_is_connected (connection))
1.453 + {
1.454 + bus_test_run_bus_loop (context, FALSE);
1.455 + bus_test_run_clients_loop (FALSE);
1.456 + }
1.457 + _dbus_verbose (" ... done spinning to auth connection %p\n", connection);
1.458 +}
1.459 +
1.460 +/* compensate for fact that pop_message() can return #NULL due to OOM */
1.461 +static DBusMessage*
1.462 +pop_message_waiting_for_memory (DBusConnection *connection)
1.463 +{
1.464 + while (dbus_connection_get_dispatch_status (connection) ==
1.465 + DBUS_DISPATCH_NEED_MEMORY)
1.466 + _dbus_wait_for_memory ();
1.467 +
1.468 + return dbus_connection_pop_message (connection);
1.469 +}
1.470 +
1.471 +static DBusMessage*
1.472 +borrow_message_waiting_for_memory (DBusConnection *connection)
1.473 +{
1.474 + while (dbus_connection_get_dispatch_status (connection) ==
1.475 + DBUS_DISPATCH_NEED_MEMORY)
1.476 + _dbus_wait_for_memory ();
1.477 +
1.478 + return dbus_connection_borrow_message (connection);
1.479 +}
1.480 +
1.481 +static void
1.482 +warn_unexpected_real (DBusConnection *connection,
1.483 + DBusMessage *message,
1.484 + const char *expected,
1.485 + const char *function,
1.486 + int line)
1.487 +{
1.488 + if (message)
1.489 + _dbus_warn ("%s:%d received message interface \"%s\" member \"%s\" error name \"%s\" on %p, expecting %s\n",
1.490 + function, line,
1.491 + dbus_message_get_interface (message) ?
1.492 + dbus_message_get_interface (message) : "(unset)",
1.493 + dbus_message_get_member (message) ?
1.494 + dbus_message_get_member (message) : "(unset)",
1.495 + dbus_message_get_error_name (message) ?
1.496 + dbus_message_get_error_name (message) : "(unset)",
1.497 + connection,
1.498 + expected);
1.499 + else
1.500 + _dbus_warn ("%s:%d received no message on %p, expecting %s\n",
1.501 + function, line, connection, expected);
1.502 +}
1.503 +
1.504 +#define warn_unexpected(connection, message, expected) \
1.505 + warn_unexpected_real (connection, message, expected, _DBUS_FUNCTION_NAME, __LINE__)
1.506 +
1.507 +static void
1.508 +verbose_message_received (DBusConnection *connection,
1.509 + DBusMessage *message)
1.510 +{
1.511 + _dbus_verbose ("Received message interface \"%s\" member \"%s\" error name \"%s\" on %p\n",
1.512 + dbus_message_get_interface (message) ?
1.513 + dbus_message_get_interface (message) : "(unset)",
1.514 + dbus_message_get_member (message) ?
1.515 + dbus_message_get_member (message) : "(unset)",
1.516 + dbus_message_get_error_name (message) ?
1.517 + dbus_message_get_error_name (message) : "(unset)",
1.518 + connection);
1.519 +}
1.520 +
1.521 +typedef enum
1.522 +{
1.523 + SERVICE_CREATED,
1.524 + OWNER_CHANGED,
1.525 + SERVICE_DELETED
1.526 +} ServiceInfoKind;
1.527 +
1.528 +typedef struct
1.529 +{
1.530 + ServiceInfoKind expected_kind;
1.531 + const char *expected_service_name;
1.532 + dbus_bool_t failed;
1.533 + DBusConnection *skip_connection;
1.534 +} CheckServiceOwnerChangedData;
1.535 +
1.536 +static dbus_bool_t
1.537 +check_service_owner_changed_foreach (DBusConnection *connection,
1.538 + void *data)
1.539 +{
1.540 + CheckServiceOwnerChangedData *d = data;
1.541 + DBusMessage *message;
1.542 + DBusError error;
1.543 + const char *service_name, *old_owner, *new_owner;
1.544 +
1.545 + if (d->expected_kind == SERVICE_CREATED
1.546 + && connection == d->skip_connection)
1.547 + return TRUE;
1.548 +
1.549 + dbus_error_init (&error);
1.550 + d->failed = TRUE;
1.551 +
1.552 + message = pop_message_waiting_for_memory (connection);
1.553 + if (message == NULL)
1.554 + {
1.555 + _dbus_warn ("Did not receive a message on %p, expecting %s\n",
1.556 + connection, "NameOwnerChanged");
1.557 + goto out;
1.558 + }
1.559 + else if (!dbus_message_is_signal (message,
1.560 + DBUS_INTERFACE_DBUS,
1.561 + "NameOwnerChanged"))
1.562 + {
1.563 + warn_unexpected (connection, message, "NameOwnerChanged");
1.564 +
1.565 + goto out;
1.566 + }
1.567 + else
1.568 + {
1.569 + reget_service_info_data:
1.570 + service_name = NULL;
1.571 + old_owner = NULL;
1.572 + new_owner = NULL;
1.573 +
1.574 + dbus_message_get_args (message, &error,
1.575 + DBUS_TYPE_STRING, &service_name,
1.576 + DBUS_TYPE_STRING, &old_owner,
1.577 + DBUS_TYPE_STRING, &new_owner,
1.578 + DBUS_TYPE_INVALID);
1.579 +
1.580 + if (dbus_error_is_set (&error))
1.581 + {
1.582 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.583 + {
1.584 + dbus_error_free (&error);
1.585 + _dbus_wait_for_memory ();
1.586 + goto reget_service_info_data;
1.587 + }
1.588 + else
1.589 + {
1.590 + _dbus_warn ("Did not get the expected arguments\n");
1.591 + goto out;
1.592 + }
1.593 + }
1.594 +
1.595 + if ((d->expected_kind == SERVICE_CREATED && ( old_owner[0] || !new_owner[0]))
1.596 + || (d->expected_kind == OWNER_CHANGED && (!old_owner[0] || !new_owner[0]))
1.597 + || (d->expected_kind == SERVICE_DELETED && (!old_owner[0] || new_owner[0])))
1.598 + {
1.599 + _dbus_warn ("inconsistent NameOwnerChanged arguments\n");
1.600 + goto out;
1.601 + }
1.602 +
1.603 + if (strcmp (service_name, d->expected_service_name) != 0)
1.604 + {
1.605 + _dbus_warn ("expected info on service %s, got info on %s\n",
1.606 + d->expected_service_name,
1.607 + service_name);
1.608 + goto out;
1.609 + }
1.610 +
1.611 + if (*service_name == ':' && new_owner[0]
1.612 + && strcmp (service_name, new_owner) != 0)
1.613 + {
1.614 + _dbus_warn ("inconsistent ServiceOwnedChanged message (\"%s\" [ %s -> %s ])\n",
1.615 + service_name, old_owner, new_owner);
1.616 + goto out;
1.617 + }
1.618 + }
1.619 +
1.620 + d->failed = FALSE;
1.621 +
1.622 + out:
1.623 + dbus_error_free (&error);
1.624 +
1.625 + if (message)
1.626 + dbus_message_unref (message);
1.627 +
1.628 + return !d->failed;
1.629 +}
1.630 +
1.631 +
1.632 +static void
1.633 +kill_client_connection (BusContext *context,
1.634 + DBusConnection *connection)
1.635 +{
1.636 + char *base_service;
1.637 + const char *s;
1.638 + CheckServiceOwnerChangedData socd;
1.639 +
1.640 + _dbus_verbose ("killing connection %p\n", connection);
1.641 +
1.642 + s = dbus_bus_get_unique_name (connection);
1.643 + _dbus_assert (s != NULL);
1.644 +
1.645 + while ((base_service = _dbus_strdup (s)) == NULL)
1.646 + _dbus_wait_for_memory ();
1.647 +
1.648 + dbus_connection_ref (connection);
1.649 +
1.650 + /* kick in the disconnect handler that unrefs the connection */
1.651 + dbus_connection_close (connection);
1.652 +
1.653 + bus_test_run_everything (context);
1.654 +
1.655 + _dbus_assert (bus_test_client_listed (connection));
1.656 +
1.657 + /* Run disconnect handler in test.c */
1.658 + if (bus_connection_dispatch_one_message (connection))
1.659 + _dbus_assert_not_reached ("something received on connection being killed other than the disconnect");
1.660 +
1.661 + _dbus_assert (!dbus_connection_get_is_connected (connection));
1.662 + dbus_connection_unref (connection);
1.663 + connection = NULL;
1.664 + _dbus_assert (!bus_test_client_listed (connection));
1.665 +
1.666 + socd.expected_kind = SERVICE_DELETED;
1.667 + socd.expected_service_name = base_service;
1.668 + socd.failed = FALSE;
1.669 + socd.skip_connection = NULL;
1.670 +
1.671 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.672 + &socd);
1.673 +
1.674 + dbus_free (base_service);
1.675 +
1.676 + if (socd.failed)
1.677 + _dbus_assert_not_reached ("didn't get the expected NameOwnerChanged (deletion) messages");
1.678 +
1.679 + if (!check_no_leftovers (context))
1.680 + _dbus_assert_not_reached ("stuff left in message queues after disconnecting a client");
1.681 +}
1.682 +
1.683 +static void
1.684 +kill_client_connection_unchecked (DBusConnection *connection)
1.685 +{
1.686 + /* This kills the connection without expecting it to affect
1.687 + * the rest of the bus.
1.688 + */
1.689 + _dbus_verbose ("Unchecked kill of connection %p\n", connection);
1.690 +
1.691 + dbus_connection_ref (connection);
1.692 + dbus_connection_close (connection);
1.693 + /* dispatching disconnect handler will unref once */
1.694 + if (bus_connection_dispatch_one_message (connection))
1.695 + _dbus_assert_not_reached ("message other than disconnect dispatched after failure to register");
1.696 +
1.697 + _dbus_assert (!bus_test_client_listed (connection));
1.698 + dbus_connection_unref (connection);
1.699 +}
1.700 +
1.701 +typedef struct
1.702 +{
1.703 + dbus_bool_t failed;
1.704 +} CheckNoMessagesData;
1.705 +
1.706 +static dbus_bool_t
1.707 +check_no_messages_foreach (DBusConnection *connection,
1.708 + void *data)
1.709 +{
1.710 + CheckNoMessagesData *d = data;
1.711 + DBusMessage *message;
1.712 +
1.713 + message = pop_message_waiting_for_memory (connection);
1.714 + if (message != NULL)
1.715 + {
1.716 + warn_unexpected (connection, message, "no messages");
1.717 +
1.718 + d->failed = TRUE;
1.719 + }
1.720 +
1.721 + if (message)
1.722 + dbus_message_unref (message);
1.723 + return !d->failed;
1.724 +}
1.725 +
1.726 +static dbus_bool_t
1.727 +check_no_leftovers (BusContext *context)
1.728 +{
1.729 + CheckNoMessagesData nmd;
1.730 +
1.731 + nmd.failed = FALSE;
1.732 + bus_test_clients_foreach (check_no_messages_foreach,
1.733 + &nmd);
1.734 +
1.735 + if (nmd.failed)
1.736 + {
1.737 + _dbus_verbose ("%s: leftover message found\n",
1.738 + _DBUS_FUNCTION_NAME);
1.739 + return FALSE;
1.740 + }
1.741 + else
1.742 + return TRUE;
1.743 +}
1.744 +
1.745 +/* returns TRUE if the correct thing happens,
1.746 + * but the correct thing may include OOM errors.
1.747 + */
1.748 +static dbus_bool_t
1.749 +check_hello_message (BusContext *context,
1.750 + DBusConnection *connection)
1.751 +{
1.752 + DBusMessage *message;
1.753 + DBusMessage *name_message;
1.754 + dbus_uint32_t serial;
1.755 + dbus_bool_t retval;
1.756 + DBusError error;
1.757 + const char *name;
1.758 + const char *acquired;
1.759 +
1.760 + retval = FALSE;
1.761 + dbus_error_init (&error);
1.762 + name = NULL;
1.763 + acquired = NULL;
1.764 + message = NULL;
1.765 + name_message = NULL;
1.766 +
1.767 + _dbus_verbose ("check_hello_message for %p\n", connection);
1.768 +
1.769 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.770 + DBUS_PATH_DBUS,
1.771 + DBUS_INTERFACE_DBUS,
1.772 + "Hello");
1.773 +
1.774 + if (message == NULL)
1.775 + return TRUE;
1.776 +
1.777 + dbus_connection_ref (connection); /* because we may get disconnected */
1.778 +
1.779 + if (!dbus_connection_send (connection, message, &serial))
1.780 + {
1.781 + dbus_message_unref (message);
1.782 + dbus_connection_unref (connection);
1.783 + return TRUE;
1.784 + }
1.785 +
1.786 + _dbus_assert (dbus_message_has_signature (message, ""));
1.787 +
1.788 + dbus_message_unref (message);
1.789 + message = NULL;
1.790 +
1.791 + if (!dbus_connection_get_is_connected (connection))
1.792 + {
1.793 + _dbus_verbose ("connection was disconnected (presumably auth failed)\n");
1.794 +
1.795 + dbus_connection_unref (connection);
1.796 +
1.797 + return TRUE;
1.798 + }
1.799 +
1.800 + /* send our message */
1.801 + bus_test_run_clients_loop (SEND_PENDING (connection));
1.802 +
1.803 + if (!dbus_connection_get_is_connected (connection))
1.804 + {
1.805 + _dbus_verbose ("connection was disconnected (presumably auth failed)\n");
1.806 +
1.807 + dbus_connection_unref (connection);
1.808 +
1.809 + return TRUE;
1.810 + }
1.811 +
1.812 + block_connection_until_message_from_bus (context, connection, "reply to Hello");
1.813 +
1.814 + if (!dbus_connection_get_is_connected (connection))
1.815 + {
1.816 + _dbus_verbose ("connection was disconnected (presumably auth failed)\n");
1.817 +
1.818 + dbus_connection_unref (connection);
1.819 +
1.820 + return TRUE;
1.821 + }
1.822 +
1.823 + dbus_connection_unref (connection);
1.824 +
1.825 + message = pop_message_waiting_for_memory (connection);
1.826 + if (message == NULL)
1.827 + {
1.828 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.829 + "Hello", serial, connection);
1.830 + goto out;
1.831 + }
1.832 +
1.833 + verbose_message_received (connection, message);
1.834 +
1.835 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.836 + {
1.837 + _dbus_warn ("Message has wrong sender %s\n",
1.838 + dbus_message_get_sender (message) ?
1.839 + dbus_message_get_sender (message) : "(none)");
1.840 + goto out;
1.841 + }
1.842 +
1.843 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.844 + {
1.845 + if (dbus_message_is_error (message,
1.846 + DBUS_ERROR_NO_MEMORY))
1.847 + {
1.848 + ; /* good, this is a valid response */
1.849 + }
1.850 + else
1.851 + {
1.852 + warn_unexpected (connection, message, "not this error");
1.853 +
1.854 + goto out;
1.855 + }
1.856 + }
1.857 + else
1.858 + {
1.859 + CheckServiceOwnerChangedData socd;
1.860 +
1.861 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.862 + {
1.863 + ; /* good, expected */
1.864 + }
1.865 + else
1.866 + {
1.867 + warn_unexpected (connection, message, "method return for Hello");
1.868 +
1.869 + goto out;
1.870 + }
1.871 +
1.872 + retry_get_hello_name:
1.873 + if (!dbus_message_get_args (message, &error,
1.874 + DBUS_TYPE_STRING, &name,
1.875 + DBUS_TYPE_INVALID))
1.876 + {
1.877 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.878 + {
1.879 + _dbus_verbose ("no memory to get service name arg from hello\n");
1.880 + dbus_error_free (&error);
1.881 + _dbus_wait_for_memory ();
1.882 + goto retry_get_hello_name;
1.883 + }
1.884 + else
1.885 + {
1.886 + _dbus_assert (dbus_error_is_set (&error));
1.887 + _dbus_warn ("Did not get the expected single string argument to hello\n");
1.888 + goto out;
1.889 + }
1.890 + }
1.891 +
1.892 + _dbus_verbose ("Got hello name: %s\n", name);
1.893 +
1.894 + while (!dbus_bus_set_unique_name (connection, name))
1.895 + _dbus_wait_for_memory ();
1.896 +
1.897 + socd.expected_kind = SERVICE_CREATED;
1.898 + socd.expected_service_name = name;
1.899 + socd.failed = FALSE;
1.900 + socd.skip_connection = connection; /* we haven't done AddMatch so won't get it ourselves */
1.901 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.902 + &socd);
1.903 +
1.904 + if (socd.failed)
1.905 + goto out;
1.906 +
1.907 + name_message = message;
1.908 + /* Client should also have gotten ServiceAcquired */
1.909 +
1.910 + message = pop_message_waiting_for_memory (connection);
1.911 + if (message == NULL)
1.912 + {
1.913 + _dbus_warn ("Expecting %s, got nothing\n",
1.914 + "NameAcquired");
1.915 + goto out;
1.916 + }
1.917 + if (! dbus_message_is_signal (message, DBUS_INTERFACE_DBUS,
1.918 + "NameAcquired"))
1.919 + {
1.920 + _dbus_warn ("Expecting %s, got smthg else\n",
1.921 + "NameAcquired");
1.922 + goto out;
1.923 + }
1.924 +
1.925 + retry_get_acquired_name:
1.926 + if (!dbus_message_get_args (message, &error,
1.927 + DBUS_TYPE_STRING, &acquired,
1.928 + DBUS_TYPE_INVALID))
1.929 + {
1.930 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.931 + {
1.932 + _dbus_verbose ("no memory to get service name arg from acquired\n");
1.933 + dbus_error_free (&error);
1.934 + _dbus_wait_for_memory ();
1.935 + goto retry_get_acquired_name;
1.936 + }
1.937 + else
1.938 + {
1.939 + _dbus_assert (dbus_error_is_set (&error));
1.940 + _dbus_warn ("Did not get the expected single string argument to ServiceAcquired\n");
1.941 + goto out;
1.942 + }
1.943 + }
1.944 +
1.945 + _dbus_verbose ("Got acquired name: %s\n", acquired);
1.946 +
1.947 + if (strcmp (acquired, name) != 0)
1.948 + {
1.949 + _dbus_warn ("Acquired name is %s but expected %s\n",
1.950 + acquired, name);
1.951 + goto out;
1.952 + }
1.953 + acquired = NULL;
1.954 + }
1.955 +
1.956 + if (!check_no_leftovers (context))
1.957 + goto out;
1.958 +
1.959 + retval = TRUE;
1.960 +
1.961 + out:
1.962 + _dbus_verbose ("ending %s retval = %d\n", _DBUS_FUNCTION_NAME, retval);
1.963 +
1.964 + dbus_error_free (&error);
1.965 +
1.966 + if (message)
1.967 + dbus_message_unref (message);
1.968 +
1.969 + if (name_message)
1.970 + dbus_message_unref (name_message);
1.971 +
1.972 + return retval;
1.973 +}
1.974 +
1.975 +/* returns TRUE if the correct thing happens,
1.976 + * but the correct thing may include OOM errors.
1.977 + */
1.978 +static dbus_bool_t
1.979 +check_double_hello_message (BusContext *context,
1.980 + DBusConnection *connection)
1.981 +{
1.982 + DBusMessage *message;
1.983 + dbus_uint32_t serial;
1.984 + dbus_bool_t retval;
1.985 + DBusError error;
1.986 +
1.987 + retval = FALSE;
1.988 + dbus_error_init (&error);
1.989 + message = NULL;
1.990 +
1.991 + _dbus_verbose ("check_double_hello_message for %p\n", connection);
1.992 +
1.993 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.994 + DBUS_PATH_DBUS,
1.995 + DBUS_INTERFACE_DBUS,
1.996 + "Hello");
1.997 +
1.998 + if (message == NULL)
1.999 + return TRUE;
1.1000 +
1.1001 + if (!dbus_connection_send (connection, message, &serial))
1.1002 + {
1.1003 + dbus_message_unref (message);
1.1004 + return TRUE;
1.1005 + }
1.1006 +
1.1007 + dbus_message_unref (message);
1.1008 + message = NULL;
1.1009 +
1.1010 + /* send our message */
1.1011 + bus_test_run_clients_loop (SEND_PENDING (connection));
1.1012 +
1.1013 + dbus_connection_ref (connection); /* because we may get disconnected */
1.1014 + block_connection_until_message_from_bus (context, connection, "reply to Hello");
1.1015 +
1.1016 + if (!dbus_connection_get_is_connected (connection))
1.1017 + {
1.1018 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.1019 +
1.1020 + dbus_connection_unref (connection);
1.1021 +
1.1022 + return TRUE;
1.1023 + }
1.1024 +
1.1025 + dbus_connection_unref (connection);
1.1026 +
1.1027 + message = pop_message_waiting_for_memory (connection);
1.1028 + if (message == NULL)
1.1029 + {
1.1030 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.1031 + "Hello", serial, connection);
1.1032 + goto out;
1.1033 + }
1.1034 +
1.1035 + verbose_message_received (connection, message);
1.1036 +
1.1037 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.1038 + {
1.1039 + _dbus_warn ("Message has wrong sender %s\n",
1.1040 + dbus_message_get_sender (message) ?
1.1041 + dbus_message_get_sender (message) : "(none)");
1.1042 + goto out;
1.1043 + }
1.1044 +
1.1045 + if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_ERROR)
1.1046 + {
1.1047 + warn_unexpected (connection, message, "method return for Hello");
1.1048 + goto out;
1.1049 + }
1.1050 +
1.1051 + if (!check_no_leftovers (context))
1.1052 + goto out;
1.1053 +
1.1054 + retval = TRUE;
1.1055 +
1.1056 + out:
1.1057 + dbus_error_free (&error);
1.1058 +
1.1059 + if (message)
1.1060 + dbus_message_unref (message);
1.1061 +
1.1062 + return retval;
1.1063 +}
1.1064 +
1.1065 +/* returns TRUE if the correct thing happens,
1.1066 + * but the correct thing may include OOM errors.
1.1067 + */
1.1068 +static dbus_bool_t
1.1069 +check_get_connection_unix_user (BusContext *context,
1.1070 + DBusConnection *connection)
1.1071 +{
1.1072 + DBusMessage *message;
1.1073 + dbus_uint32_t serial;
1.1074 + dbus_bool_t retval;
1.1075 + DBusError error;
1.1076 + const char *base_service_name;
1.1077 + dbus_uint32_t uid;
1.1078 +
1.1079 + retval = FALSE;
1.1080 + dbus_error_init (&error);
1.1081 + message = NULL;
1.1082 +
1.1083 + _dbus_verbose ("check_get_connection_unix_user for %p\n", connection);
1.1084 +
1.1085 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.1086 + DBUS_PATH_DBUS,
1.1087 + DBUS_INTERFACE_DBUS,
1.1088 + "GetConnectionUnixUser");
1.1089 +
1.1090 + if (message == NULL)
1.1091 + return TRUE;
1.1092 +
1.1093 + base_service_name = dbus_bus_get_unique_name (connection);
1.1094 +
1.1095 + if (!dbus_message_append_args (message,
1.1096 + DBUS_TYPE_STRING, &base_service_name,
1.1097 + DBUS_TYPE_INVALID))
1.1098 + {
1.1099 + dbus_message_unref (message);
1.1100 + return TRUE;
1.1101 + }
1.1102 +
1.1103 + if (!dbus_connection_send (connection, message, &serial))
1.1104 + {
1.1105 + dbus_message_unref (message);
1.1106 + return TRUE;
1.1107 + }
1.1108 +
1.1109 + /* send our message */
1.1110 + bus_test_run_clients_loop (SEND_PENDING (connection));
1.1111 +
1.1112 + dbus_message_unref (message);
1.1113 + message = NULL;
1.1114 +
1.1115 + dbus_connection_ref (connection); /* because we may get disconnected */
1.1116 + block_connection_until_message_from_bus (context, connection, "reply to GetConnectionUnixUser");
1.1117 +
1.1118 + if (!dbus_connection_get_is_connected (connection))
1.1119 + {
1.1120 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.1121 +
1.1122 + dbus_connection_unref (connection);
1.1123 +
1.1124 + return TRUE;
1.1125 + }
1.1126 +
1.1127 + dbus_connection_unref (connection);
1.1128 +
1.1129 + message = pop_message_waiting_for_memory (connection);
1.1130 + if (message == NULL)
1.1131 + {
1.1132 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.1133 + "GetConnectionUnixUser", serial, connection);
1.1134 + goto out;
1.1135 + }
1.1136 +
1.1137 + verbose_message_received (connection, message);
1.1138 +
1.1139 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.1140 + {
1.1141 + if (dbus_message_is_error (message, DBUS_ERROR_NO_MEMORY))
1.1142 + {
1.1143 + ; /* good, this is a valid response */
1.1144 + }
1.1145 + else
1.1146 + {
1.1147 + warn_unexpected (connection, message, "not this error");
1.1148 +
1.1149 + goto out;
1.1150 + }
1.1151 + }
1.1152 + else
1.1153 + {
1.1154 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.1155 + {
1.1156 + ; /* good, expected */
1.1157 + }
1.1158 + else
1.1159 + {
1.1160 + warn_unexpected (connection, message,
1.1161 + "method_return for GetConnectionUnixUser");
1.1162 +
1.1163 + goto out;
1.1164 + }
1.1165 +
1.1166 + retry_get_property:
1.1167 +
1.1168 + if (!dbus_message_get_args (message, &error,
1.1169 + DBUS_TYPE_UINT32, &uid,
1.1170 + DBUS_TYPE_INVALID))
1.1171 + {
1.1172 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.1173 + {
1.1174 + _dbus_verbose ("no memory to get uid by GetConnectionUnixUser\n");
1.1175 + dbus_error_free (&error);
1.1176 + _dbus_wait_for_memory ();
1.1177 + goto retry_get_property;
1.1178 + }
1.1179 + else
1.1180 + {
1.1181 + _dbus_assert (dbus_error_is_set (&error));
1.1182 + _dbus_warn ("Did not get the expected DBUS_TYPE_UINT32 from GetConnectionUnixUser\n");
1.1183 + goto out;
1.1184 + }
1.1185 + }
1.1186 + }
1.1187 +
1.1188 + if (!check_no_leftovers (context))
1.1189 + goto out;
1.1190 +
1.1191 + retval = TRUE;
1.1192 +
1.1193 + out:
1.1194 + dbus_error_free (&error);
1.1195 +
1.1196 + if (message)
1.1197 + dbus_message_unref (message);
1.1198 +
1.1199 + return retval;
1.1200 +}
1.1201 +
1.1202 +/* returns TRUE if the correct thing happens,
1.1203 + * but the correct thing may include OOM errors.
1.1204 + */
1.1205 +static dbus_bool_t
1.1206 +check_get_connection_unix_process_id (BusContext *context,
1.1207 + DBusConnection *connection)
1.1208 +{
1.1209 + DBusMessage *message;
1.1210 + dbus_uint32_t serial;
1.1211 + dbus_bool_t retval;
1.1212 + DBusError error;
1.1213 + const char *base_service_name;
1.1214 + dbus_uint32_t pid;
1.1215 +
1.1216 + retval = FALSE;
1.1217 + dbus_error_init (&error);
1.1218 + message = NULL;
1.1219 +
1.1220 + _dbus_verbose ("check_get_connection_unix_process_id for %p\n", connection);
1.1221 +
1.1222 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.1223 + DBUS_PATH_DBUS,
1.1224 + DBUS_INTERFACE_DBUS,
1.1225 + "GetConnectionUnixProcessID");
1.1226 +
1.1227 + if (message == NULL)
1.1228 + return TRUE;
1.1229 +
1.1230 + base_service_name = dbus_bus_get_unique_name (connection);
1.1231 +
1.1232 + if (!dbus_message_append_args (message,
1.1233 + DBUS_TYPE_STRING, &base_service_name,
1.1234 + DBUS_TYPE_INVALID))
1.1235 + {
1.1236 + dbus_message_unref (message);
1.1237 + return TRUE;
1.1238 + }
1.1239 +
1.1240 + if (!dbus_connection_send (connection, message, &serial))
1.1241 + {
1.1242 + dbus_message_unref (message);
1.1243 + return TRUE;
1.1244 + }
1.1245 +
1.1246 + /* send our message */
1.1247 + bus_test_run_clients_loop (SEND_PENDING (connection));
1.1248 +
1.1249 + dbus_message_unref (message);
1.1250 + message = NULL;
1.1251 +
1.1252 + dbus_connection_ref (connection); /* because we may get disconnected */
1.1253 + block_connection_until_message_from_bus (context, connection, "reply to GetConnectionUnixProcessID");
1.1254 +
1.1255 + if (!dbus_connection_get_is_connected (connection))
1.1256 + {
1.1257 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.1258 +
1.1259 + dbus_connection_unref (connection);
1.1260 +
1.1261 + return TRUE;
1.1262 + }
1.1263 +
1.1264 + dbus_connection_unref (connection);
1.1265 +
1.1266 + message = pop_message_waiting_for_memory (connection);
1.1267 + if (message == NULL)
1.1268 + {
1.1269 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.1270 + "GetConnectionUnixProcessID", serial, connection);
1.1271 + goto out;
1.1272 + }
1.1273 +
1.1274 + verbose_message_received (connection, message);
1.1275 +
1.1276 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.1277 + {
1.1278 + if (dbus_message_is_error (message, DBUS_ERROR_NO_MEMORY))
1.1279 + {
1.1280 + ; /* good, this is a valid response */
1.1281 + }
1.1282 + else
1.1283 + {
1.1284 + warn_unexpected (connection, message, "not this error");
1.1285 +
1.1286 + goto out;
1.1287 + }
1.1288 + }
1.1289 + else
1.1290 + {
1.1291 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.1292 + {
1.1293 + ; /* good, expected */
1.1294 + }
1.1295 + else
1.1296 + {
1.1297 + warn_unexpected (connection, message,
1.1298 + "method_return for GetConnectionUnixProcessID");
1.1299 +
1.1300 + goto out;
1.1301 + }
1.1302 +
1.1303 + retry_get_property:
1.1304 +
1.1305 + if (!dbus_message_get_args (message, &error,
1.1306 + DBUS_TYPE_UINT32, &pid,
1.1307 + DBUS_TYPE_INVALID))
1.1308 + {
1.1309 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.1310 + {
1.1311 + _dbus_verbose ("no memory to get pid by GetConnectionUnixProcessID\n");
1.1312 + dbus_error_free (&error);
1.1313 + _dbus_wait_for_memory ();
1.1314 + goto retry_get_property;
1.1315 + }
1.1316 + else
1.1317 + {
1.1318 + _dbus_assert (dbus_error_is_set (&error));
1.1319 + _dbus_warn ("Did not get the expected DBUS_TYPE_UINT32 from GetConnectionUnixProcessID\n");
1.1320 + goto out;
1.1321 + }
1.1322 + } else {
1.1323 +
1.1324 + /* test if returned pid is the same as our own pid
1.1325 + *
1.1326 + * @todo It would probably be good to restructure the tests
1.1327 + * in a way so our parent is the bus that we're testing
1.1328 + * cause then we can test that the pid returned matches
1.1329 + * getppid()
1.1330 + */
1.1331 + if (pid != (dbus_uint32_t) _dbus_getpid ())
1.1332 + {
1.1333 + _dbus_assert (dbus_error_is_set (&error));
1.1334 + _dbus_warn ("Result from GetConnectionUnixProcessID is not our own pid\n");
1.1335 + goto out;
1.1336 + }
1.1337 + }
1.1338 + }
1.1339 +
1.1340 + if (!check_no_leftovers (context))
1.1341 + goto out;
1.1342 +
1.1343 + retval = TRUE;
1.1344 +
1.1345 + out:
1.1346 + dbus_error_free (&error);
1.1347 +
1.1348 + if (message)
1.1349 + dbus_message_unref (message);
1.1350 +
1.1351 + return retval;
1.1352 +}
1.1353 +
1.1354 +/* returns TRUE if the correct thing happens,
1.1355 + * but the correct thing may include OOM errors.
1.1356 + */
1.1357 +static dbus_bool_t
1.1358 +check_add_match_all (BusContext *context,
1.1359 + DBusConnection *connection)
1.1360 +{
1.1361 + DBusMessage *message;
1.1362 + dbus_bool_t retval;
1.1363 + dbus_uint32_t serial;
1.1364 + DBusError error;
1.1365 + const char *empty = "";
1.1366 +
1.1367 + retval = FALSE;
1.1368 + dbus_error_init (&error);
1.1369 + message = NULL;
1.1370 +
1.1371 + _dbus_verbose ("check_add_match_all for %p\n", connection);
1.1372 +
1.1373 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.1374 + DBUS_PATH_DBUS,
1.1375 + DBUS_INTERFACE_DBUS,
1.1376 + "AddMatch");
1.1377 +
1.1378 + if (message == NULL)
1.1379 + return TRUE;
1.1380 +
1.1381 + /* empty string match rule matches everything */
1.1382 + if (!dbus_message_append_args (message, DBUS_TYPE_STRING, &empty,
1.1383 + DBUS_TYPE_INVALID))
1.1384 + {
1.1385 + dbus_message_unref (message);
1.1386 + return TRUE;
1.1387 + }
1.1388 +
1.1389 + if (!dbus_connection_send (connection, message, &serial))
1.1390 + {
1.1391 + dbus_message_unref (message);
1.1392 + return TRUE;
1.1393 + }
1.1394 +
1.1395 + dbus_message_unref (message);
1.1396 + message = NULL;
1.1397 +
1.1398 + dbus_connection_ref (connection); /* because we may get disconnected */
1.1399 +
1.1400 + /* send our message */
1.1401 + bus_test_run_clients_loop (SEND_PENDING (connection));
1.1402 +
1.1403 + if (!dbus_connection_get_is_connected (connection))
1.1404 + {
1.1405 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.1406 +
1.1407 + dbus_connection_unref (connection);
1.1408 +
1.1409 + return TRUE;
1.1410 + }
1.1411 +
1.1412 + block_connection_until_message_from_bus (context, connection, "reply to AddMatch");
1.1413 +
1.1414 + if (!dbus_connection_get_is_connected (connection))
1.1415 + {
1.1416 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.1417 +
1.1418 + dbus_connection_unref (connection);
1.1419 +
1.1420 + return TRUE;
1.1421 + }
1.1422 +
1.1423 + dbus_connection_unref (connection);
1.1424 +
1.1425 + message = pop_message_waiting_for_memory (connection);
1.1426 + if (message == NULL)
1.1427 + {
1.1428 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.1429 + "AddMatch", serial, connection);
1.1430 + goto out;
1.1431 + }
1.1432 +
1.1433 + verbose_message_received (connection, message);
1.1434 +
1.1435 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.1436 + {
1.1437 + _dbus_warn ("Message has wrong sender %s\n",
1.1438 + dbus_message_get_sender (message) ?
1.1439 + dbus_message_get_sender (message) : "(none)");
1.1440 + goto out;
1.1441 + }
1.1442 +
1.1443 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.1444 + {
1.1445 + if (dbus_message_is_error (message,
1.1446 + DBUS_ERROR_NO_MEMORY))
1.1447 + {
1.1448 + ; /* good, this is a valid response */
1.1449 + }
1.1450 + else
1.1451 + {
1.1452 + warn_unexpected (connection, message, "not this error");
1.1453 +
1.1454 + goto out;
1.1455 + }
1.1456 + }
1.1457 + else
1.1458 + {
1.1459 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.1460 + {
1.1461 + ; /* good, expected */
1.1462 + _dbus_assert (dbus_message_get_reply_serial (message) == serial);
1.1463 + }
1.1464 + else
1.1465 + {
1.1466 + warn_unexpected (connection, message, "method return for AddMatch");
1.1467 +
1.1468 + goto out;
1.1469 + }
1.1470 + }
1.1471 +
1.1472 + if (!check_no_leftovers (context))
1.1473 + goto out;
1.1474 +
1.1475 + retval = TRUE;
1.1476 +
1.1477 + out:
1.1478 + dbus_error_free (&error);
1.1479 +
1.1480 + if (message)
1.1481 + dbus_message_unref (message);
1.1482 +
1.1483 + return retval;
1.1484 +}
1.1485 +
1.1486 +/* returns TRUE if the correct thing happens,
1.1487 + * but the correct thing may include OOM errors.
1.1488 + */
1.1489 +static dbus_bool_t
1.1490 +check_hello_connection (BusContext *context)
1.1491 +{
1.1492 + DBusConnection *connection;
1.1493 + DBusError error;
1.1494 +
1.1495 + dbus_error_init (&error);
1.1496 +
1.1497 + connection = dbus_connection_open_private ("debug-pipe:name=test-server", &error);
1.1498 + if (connection == NULL)
1.1499 + {
1.1500 + _DBUS_ASSERT_ERROR_IS_SET (&error);
1.1501 + dbus_error_free (&error);
1.1502 + return TRUE;
1.1503 + }
1.1504 +
1.1505 + if (!bus_setup_debug_client (connection))
1.1506 + {
1.1507 + dbus_connection_close (connection);
1.1508 + dbus_connection_unref (connection);
1.1509 + return TRUE;
1.1510 + }
1.1511 +
1.1512 + spin_connection_until_authenticated (context, connection);
1.1513 +
1.1514 + if (!check_hello_message (context, connection))
1.1515 + return FALSE;
1.1516 +
1.1517 + if (dbus_bus_get_unique_name (connection) == NULL)
1.1518 + {
1.1519 + /* We didn't successfully register, so we can't
1.1520 + * do the usual kill_client_connection() checks
1.1521 + */
1.1522 + kill_client_connection_unchecked (connection);
1.1523 + }
1.1524 + else
1.1525 + {
1.1526 + if (!check_add_match_all (context, connection))
1.1527 + return FALSE;
1.1528 +
1.1529 + kill_client_connection (context, connection);
1.1530 + }
1.1531 +
1.1532 + return TRUE;
1.1533 +}
1.1534 +
1.1535 +#define NONEXISTENT_SERVICE_NAME "test.this.service.does.not.exist.ewuoiurjdfxcvn"
1.1536 +
1.1537 +/* returns TRUE if the correct thing happens,
1.1538 + * but the correct thing may include OOM errors.
1.1539 + */
1.1540 +static dbus_bool_t
1.1541 +check_nonexistent_service_no_auto_start (BusContext *context,
1.1542 + DBusConnection *connection)
1.1543 +{
1.1544 + DBusMessage *message;
1.1545 + dbus_uint32_t serial;
1.1546 + dbus_bool_t retval;
1.1547 + const char *nonexistent = NONEXISTENT_SERVICE_NAME;
1.1548 + dbus_uint32_t flags;
1.1549 +
1.1550 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.1551 + DBUS_PATH_DBUS,
1.1552 + DBUS_INTERFACE_DBUS,
1.1553 + "StartServiceByName");
1.1554 +
1.1555 + if (message == NULL)
1.1556 + return TRUE;
1.1557 +
1.1558 + dbus_message_set_auto_start (message, FALSE);
1.1559 +
1.1560 + flags = 0;
1.1561 + if (!dbus_message_append_args (message,
1.1562 + DBUS_TYPE_STRING, &nonexistent,
1.1563 + DBUS_TYPE_UINT32, &flags,
1.1564 + DBUS_TYPE_INVALID))
1.1565 + {
1.1566 + dbus_message_unref (message);
1.1567 + return TRUE;
1.1568 + }
1.1569 +
1.1570 + if (!dbus_connection_send (connection, message, &serial))
1.1571 + {
1.1572 + dbus_message_unref (message);
1.1573 + return TRUE;
1.1574 + }
1.1575 +
1.1576 + dbus_message_unref (message);
1.1577 + message = NULL;
1.1578 +
1.1579 + bus_test_run_everything (context);
1.1580 + block_connection_until_message_from_bus (context, connection, "reply to ActivateService on nonexistent");
1.1581 + bus_test_run_everything (context);
1.1582 +
1.1583 + if (!dbus_connection_get_is_connected (connection))
1.1584 + {
1.1585 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.1586 + return TRUE;
1.1587 + }
1.1588 +
1.1589 + retval = FALSE;
1.1590 +
1.1591 + message = pop_message_waiting_for_memory (connection);
1.1592 + if (message == NULL)
1.1593 + {
1.1594 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.1595 + "StartServiceByName", serial, connection);
1.1596 + goto out;
1.1597 + }
1.1598 +
1.1599 + verbose_message_received (connection, message);
1.1600 +
1.1601 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.1602 + {
1.1603 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.1604 + {
1.1605 + _dbus_warn ("Message has wrong sender %s\n",
1.1606 + dbus_message_get_sender (message) ?
1.1607 + dbus_message_get_sender (message) : "(none)");
1.1608 + goto out;
1.1609 + }
1.1610 +
1.1611 + if (dbus_message_is_error (message,
1.1612 + DBUS_ERROR_NO_MEMORY))
1.1613 + {
1.1614 + ; /* good, this is a valid response */
1.1615 + }
1.1616 + else if (dbus_message_is_error (message,
1.1617 + DBUS_ERROR_SERVICE_UNKNOWN))
1.1618 + {
1.1619 + ; /* good, this is expected also */
1.1620 + }
1.1621 + else
1.1622 + {
1.1623 + warn_unexpected (connection, message, "not this error");
1.1624 + goto out;
1.1625 + }
1.1626 + }
1.1627 + else
1.1628 + {
1.1629 + _dbus_warn ("Did not expect to successfully activate %s\n",
1.1630 + NONEXISTENT_SERVICE_NAME);
1.1631 + goto out;
1.1632 + }
1.1633 +
1.1634 + retval = TRUE;
1.1635 +
1.1636 + out:
1.1637 + if (message)
1.1638 + dbus_message_unref (message);
1.1639 +
1.1640 + return retval;
1.1641 +}
1.1642 +
1.1643 +/* returns TRUE if the correct thing happens,
1.1644 + * but the correct thing may include OOM errors.
1.1645 + */
1.1646 +static dbus_bool_t
1.1647 +check_nonexistent_service_auto_start (BusContext *context,
1.1648 + DBusConnection *connection)
1.1649 +{
1.1650 + DBusMessage *message;
1.1651 + dbus_uint32_t serial;
1.1652 + dbus_bool_t retval;
1.1653 +
1.1654 + message = dbus_message_new_method_call (NONEXISTENT_SERVICE_NAME,
1.1655 + "/org/freedesktop/TestSuite",
1.1656 + "org.freedesktop.TestSuite",
1.1657 + "Echo");
1.1658 +
1.1659 + if (message == NULL)
1.1660 + return TRUE;
1.1661 +
1.1662 + if (!dbus_connection_send (connection, message, &serial))
1.1663 + {
1.1664 + dbus_message_unref (message);
1.1665 + return TRUE;
1.1666 + }
1.1667 +
1.1668 + dbus_message_unref (message);
1.1669 + message = NULL;
1.1670 +
1.1671 + bus_test_run_everything (context);
1.1672 + block_connection_until_message_from_bus (context, connection, "reply to Echo");
1.1673 + bus_test_run_everything (context);
1.1674 +
1.1675 + if (!dbus_connection_get_is_connected (connection))
1.1676 + {
1.1677 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.1678 + return TRUE;
1.1679 + }
1.1680 +
1.1681 + retval = FALSE;
1.1682 +
1.1683 + message = pop_message_waiting_for_memory (connection);
1.1684 +
1.1685 + if (message == NULL)
1.1686 + {
1.1687 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.1688 + "Echo message (auto activation)", serial, connection);
1.1689 + goto out;
1.1690 + }
1.1691 +
1.1692 + verbose_message_received (connection, message);
1.1693 +
1.1694 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.1695 + {
1.1696 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.1697 + {
1.1698 + _dbus_warn ("Message has wrong sender %s\n",
1.1699 + dbus_message_get_sender (message) ?
1.1700 + dbus_message_get_sender (message) : "(none)");
1.1701 + goto out;
1.1702 + }
1.1703 +
1.1704 + if (dbus_message_is_error (message,
1.1705 + DBUS_ERROR_NO_MEMORY))
1.1706 + {
1.1707 + ; /* good, this is a valid response */
1.1708 + }
1.1709 + else if (dbus_message_is_error (message,
1.1710 + DBUS_ERROR_SERVICE_UNKNOWN))
1.1711 + {
1.1712 + ; /* good, this is expected also */
1.1713 + }
1.1714 + else
1.1715 + {
1.1716 + warn_unexpected (connection, message, "not this error");
1.1717 + goto out;
1.1718 + }
1.1719 + }
1.1720 + else
1.1721 + {
1.1722 + _dbus_warn ("Did not expect to successfully activate %s\n",
1.1723 + NONEXISTENT_SERVICE_NAME);
1.1724 + goto out;
1.1725 + }
1.1726 +
1.1727 + retval = TRUE;
1.1728 +
1.1729 + out:
1.1730 + if (message)
1.1731 + dbus_message_unref (message);
1.1732 +
1.1733 + return retval;
1.1734 +}
1.1735 +
1.1736 +static dbus_bool_t
1.1737 +check_base_service_activated (BusContext *context,
1.1738 + DBusConnection *connection,
1.1739 + DBusMessage *initial_message,
1.1740 + const char **base_service_p)
1.1741 +{
1.1742 + DBusMessage *message;
1.1743 + dbus_bool_t retval;
1.1744 + DBusError error;
1.1745 + const char *base_service, *base_service_from_bus, *old_owner;
1.1746 +
1.1747 + retval = FALSE;
1.1748 +
1.1749 + dbus_error_init (&error);
1.1750 + base_service = NULL;
1.1751 + old_owner = NULL;
1.1752 + base_service_from_bus = NULL;
1.1753 +
1.1754 + message = initial_message;
1.1755 + dbus_message_ref (message);
1.1756 +
1.1757 + if (dbus_message_is_signal (message,
1.1758 + DBUS_INTERFACE_DBUS,
1.1759 + "NameOwnerChanged"))
1.1760 + {
1.1761 + CheckServiceOwnerChangedData socd;
1.1762 +
1.1763 + reget_service_name_arg:
1.1764 + base_service = NULL;
1.1765 + old_owner = NULL;
1.1766 + base_service_from_bus = NULL;
1.1767 +
1.1768 + if (!dbus_message_get_args (message, &error,
1.1769 + DBUS_TYPE_STRING, &base_service,
1.1770 + DBUS_TYPE_STRING, &old_owner,
1.1771 + DBUS_TYPE_STRING, &base_service_from_bus,
1.1772 + DBUS_TYPE_INVALID))
1.1773 + {
1.1774 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.1775 + {
1.1776 + dbus_error_free (&error);
1.1777 + _dbus_wait_for_memory ();
1.1778 + goto reget_service_name_arg;
1.1779 + }
1.1780 + else
1.1781 + {
1.1782 + _dbus_warn ("Message %s doesn't have a service name: %s\n",
1.1783 + "NameOwnerChanged (creation)",
1.1784 + error.message);
1.1785 + goto out;
1.1786 + }
1.1787 + }
1.1788 +
1.1789 + if (*base_service != ':')
1.1790 + {
1.1791 + _dbus_warn ("Expected base service activation, got \"%s\" instead\n",
1.1792 + base_service);
1.1793 + goto out;
1.1794 + }
1.1795 +
1.1796 + if (strcmp (base_service, base_service_from_bus) != 0)
1.1797 + {
1.1798 + _dbus_warn ("Expected base service activation, got \"%s\" instead with owner \"%s\"\n",
1.1799 + base_service, base_service_from_bus);
1.1800 + goto out;
1.1801 + }
1.1802 +
1.1803 + if (old_owner[0])
1.1804 + {
1.1805 + _dbus_warn ("Received an old_owner argument during base service activation, \"%s\"\n",
1.1806 + old_owner);
1.1807 + goto out;
1.1808 + }
1.1809 +
1.1810 + socd.expected_kind = SERVICE_CREATED;
1.1811 + socd.expected_service_name = base_service;
1.1812 + socd.failed = FALSE;
1.1813 + socd.skip_connection = connection;
1.1814 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.1815 + &socd);
1.1816 +
1.1817 + if (socd.failed)
1.1818 + goto out;
1.1819 + }
1.1820 + else
1.1821 + {
1.1822 + warn_unexpected (connection, message, "NameOwnerChanged (creation) for base service");
1.1823 +
1.1824 + goto out;
1.1825 + }
1.1826 +
1.1827 + if (base_service_p)
1.1828 + *base_service_p = base_service;
1.1829 +
1.1830 + retval = TRUE;
1.1831 +
1.1832 + out:
1.1833 + if (message)
1.1834 + dbus_message_unref (message);
1.1835 + dbus_error_free (&error);
1.1836 +
1.1837 + return retval;
1.1838 +}
1.1839 +
1.1840 +static dbus_bool_t
1.1841 +check_service_activated (BusContext *context,
1.1842 + DBusConnection *connection,
1.1843 + const char *activated_name,
1.1844 + const char *base_service_name,
1.1845 + DBusMessage *initial_message)
1.1846 +{
1.1847 + DBusMessage *message;
1.1848 + dbus_bool_t retval;
1.1849 + DBusError error;
1.1850 + dbus_uint32_t activation_result;
1.1851 +
1.1852 + retval = FALSE;
1.1853 +
1.1854 + dbus_error_init (&error);
1.1855 +
1.1856 + message = initial_message;
1.1857 + dbus_message_ref (message);
1.1858 +
1.1859 + if (dbus_message_is_signal (message,
1.1860 + DBUS_INTERFACE_DBUS,
1.1861 + "NameOwnerChanged"))
1.1862 + {
1.1863 + CheckServiceOwnerChangedData socd;
1.1864 + const char *service_name, *base_service_from_bus, *old_owner;
1.1865 +
1.1866 + reget_service_name_arg:
1.1867 + service_name = NULL;
1.1868 + old_owner = NULL;
1.1869 + base_service_from_bus = NULL;
1.1870 +
1.1871 + if (!dbus_message_get_args (message, &error,
1.1872 + DBUS_TYPE_STRING, &service_name,
1.1873 + DBUS_TYPE_STRING, &old_owner,
1.1874 + DBUS_TYPE_STRING, &base_service_from_bus,
1.1875 + DBUS_TYPE_INVALID))
1.1876 + {
1.1877 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.1878 + {
1.1879 + dbus_error_free (&error);
1.1880 + _dbus_wait_for_memory ();
1.1881 + goto reget_service_name_arg;
1.1882 + }
1.1883 + else
1.1884 + {
1.1885 + _dbus_warn ("Message %s doesn't have a service name: %s\n",
1.1886 + "NameOwnerChanged (creation)",
1.1887 + error.message);
1.1888 + goto out;
1.1889 + }
1.1890 + }
1.1891 +
1.1892 + if (strcmp (service_name, activated_name) != 0)
1.1893 + {
1.1894 + _dbus_warn ("Expected to see service %s created, saw %s instead\n",
1.1895 + activated_name, service_name);
1.1896 + goto out;
1.1897 + }
1.1898 +
1.1899 + if (strcmp (base_service_name, base_service_from_bus) != 0)
1.1900 + {
1.1901 + _dbus_warn ("NameOwnerChanged reports wrong base service: %s owner, expected %s instead\n",
1.1902 + base_service_from_bus, base_service_name);
1.1903 + goto out;
1.1904 + }
1.1905 +
1.1906 + if (old_owner[0])
1.1907 + {
1.1908 + _dbus_warn ("expected a %s, got a %s\n",
1.1909 + "NameOwnerChanged (creation)",
1.1910 + "NameOwnerChanged (change)");
1.1911 + goto out;
1.1912 + }
1.1913 +
1.1914 + socd.expected_kind = SERVICE_CREATED;
1.1915 + socd.skip_connection = connection;
1.1916 + socd.failed = FALSE;
1.1917 + socd.expected_service_name = service_name;
1.1918 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.1919 + &socd);
1.1920 +
1.1921 + if (socd.failed)
1.1922 + goto out;
1.1923 +
1.1924 + dbus_message_unref (message);
1.1925 + service_name = NULL;
1.1926 + old_owner = NULL;
1.1927 + base_service_from_bus = NULL;
1.1928 +
1.1929 + message = pop_message_waiting_for_memory (connection);
1.1930 + if (message == NULL)
1.1931 + {
1.1932 + _dbus_warn ("Expected a reply to %s, got nothing\n",
1.1933 + "StartServiceByName");
1.1934 + goto out;
1.1935 + }
1.1936 + }
1.1937 + else
1.1938 + {
1.1939 + warn_unexpected (connection, message, "NameOwnerChanged for the activated name");
1.1940 +
1.1941 + goto out;
1.1942 + }
1.1943 +
1.1944 + if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.1945 + {
1.1946 + warn_unexpected (connection, message, "reply to StartServiceByName");
1.1947 +
1.1948 + goto out;
1.1949 + }
1.1950 +
1.1951 + activation_result = 0;
1.1952 + if (!dbus_message_get_args (message, &error,
1.1953 + DBUS_TYPE_UINT32, &activation_result,
1.1954 + DBUS_TYPE_INVALID))
1.1955 + {
1.1956 + if (!dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.1957 + {
1.1958 + _dbus_warn ("Did not have activation result first argument to %s: %s\n",
1.1959 + "StartServiceByName", error.message);
1.1960 + goto out;
1.1961 + }
1.1962 +
1.1963 + dbus_error_free (&error);
1.1964 + }
1.1965 + else
1.1966 + {
1.1967 + if (activation_result == DBUS_START_REPLY_SUCCESS)
1.1968 + ; /* Good */
1.1969 + else if (activation_result == DBUS_START_REPLY_ALREADY_RUNNING)
1.1970 + ; /* Good also */
1.1971 + else
1.1972 + {
1.1973 + _dbus_warn ("Activation result was %u, no good.\n",
1.1974 + activation_result);
1.1975 + goto out;
1.1976 + }
1.1977 + }
1.1978 +
1.1979 + dbus_message_unref (message);
1.1980 + message = NULL;
1.1981 +
1.1982 + if (!check_no_leftovers (context))
1.1983 + {
1.1984 + _dbus_warn ("Messages were left over after verifying existent activation results\n");
1.1985 + goto out;
1.1986 + }
1.1987 +
1.1988 + retval = TRUE;
1.1989 +
1.1990 + out:
1.1991 + if (message)
1.1992 + dbus_message_unref (message);
1.1993 + dbus_error_free (&error);
1.1994 +
1.1995 + return retval;
1.1996 +}
1.1997 +
1.1998 +static dbus_bool_t
1.1999 +check_service_auto_activated (BusContext *context,
1.2000 + DBusConnection *connection,
1.2001 + const char *activated_name,
1.2002 + const char *base_service_name,
1.2003 + DBusMessage *initial_message)
1.2004 +{
1.2005 + DBusMessage *message;
1.2006 + dbus_bool_t retval;
1.2007 + DBusError error;
1.2008 +
1.2009 + retval = FALSE;
1.2010 +
1.2011 + dbus_error_init (&error);
1.2012 +
1.2013 + message = initial_message;
1.2014 + dbus_message_ref (message);
1.2015 +
1.2016 + if (dbus_message_is_signal (message,
1.2017 + DBUS_INTERFACE_DBUS,
1.2018 + "NameOwnerChanged"))
1.2019 + {
1.2020 + const char *service_name;
1.2021 + CheckServiceOwnerChangedData socd;
1.2022 +
1.2023 + reget_service_name_arg:
1.2024 + if (!dbus_message_get_args (message, &error,
1.2025 + DBUS_TYPE_STRING, &service_name,
1.2026 + DBUS_TYPE_INVALID))
1.2027 + {
1.2028 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.2029 + {
1.2030 + dbus_error_free (&error);
1.2031 + _dbus_wait_for_memory ();
1.2032 + goto reget_service_name_arg;
1.2033 + }
1.2034 + else
1.2035 + {
1.2036 + _dbus_warn ("Message %s doesn't have a service name: %s\n",
1.2037 + "NameOwnerChanged",
1.2038 + error.message);
1.2039 + dbus_error_free (&error);
1.2040 + goto out;
1.2041 + }
1.2042 + }
1.2043 +
1.2044 + if (strcmp (service_name, activated_name) != 0)
1.2045 + {
1.2046 + _dbus_warn ("Expected to see service %s created, saw %s instead\n",
1.2047 + activated_name, service_name);
1.2048 + goto out;
1.2049 + }
1.2050 +
1.2051 + socd.expected_kind = SERVICE_CREATED;
1.2052 + socd.expected_service_name = service_name;
1.2053 + socd.failed = FALSE;
1.2054 + socd.skip_connection = connection;
1.2055 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.2056 + &socd);
1.2057 +
1.2058 + if (socd.failed)
1.2059 + goto out;
1.2060 +
1.2061 + /* Note that this differs from regular activation in that we don't get a
1.2062 + * reply to ActivateService here.
1.2063 + */
1.2064 +
1.2065 + dbus_message_unref (message);
1.2066 + message = NULL;
1.2067 + service_name = NULL;
1.2068 + }
1.2069 + else
1.2070 + {
1.2071 + warn_unexpected (connection, message, "NameOwnerChanged for the activated name");
1.2072 +
1.2073 + goto out;
1.2074 + }
1.2075 +
1.2076 + retval = TRUE;
1.2077 +
1.2078 + out:
1.2079 + if (message)
1.2080 + dbus_message_unref (message);
1.2081 +
1.2082 + return retval;
1.2083 +}
1.2084 +
1.2085 +static dbus_bool_t
1.2086 +check_service_deactivated (BusContext *context,
1.2087 + DBusConnection *connection,
1.2088 + const char *activated_name,
1.2089 + const char *base_service)
1.2090 +{
1.2091 + dbus_bool_t retval;
1.2092 + CheckServiceOwnerChangedData socd;
1.2093 +
1.2094 + retval = FALSE;
1.2095 +
1.2096 + /* Now we are expecting ServiceOwnerChanged (deletion) messages for the base
1.2097 + * service and the activated_name. The base service
1.2098 + * notification is required to come last.
1.2099 + */
1.2100 + socd.expected_kind = SERVICE_DELETED;
1.2101 + socd.expected_service_name = activated_name;
1.2102 + socd.failed = FALSE;
1.2103 + socd.skip_connection = NULL;
1.2104 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.2105 + &socd);
1.2106 +
1.2107 + if (socd.failed)
1.2108 + goto out;
1.2109 +
1.2110 + socd.expected_kind = SERVICE_DELETED;
1.2111 + socd.expected_service_name = base_service;
1.2112 + socd.failed = FALSE;
1.2113 + socd.skip_connection = NULL;
1.2114 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.2115 + &socd);
1.2116 +
1.2117 + if (socd.failed)
1.2118 + goto out;
1.2119 +
1.2120 + retval = TRUE;
1.2121 +
1.2122 + out:
1.2123 + return retval;
1.2124 +}
1.2125 +
1.2126 +static dbus_bool_t
1.2127 +check_send_exit_to_service (BusContext *context,
1.2128 + DBusConnection *connection,
1.2129 + const char *service_name,
1.2130 + const char *base_service)
1.2131 +{
1.2132 + dbus_bool_t got_error;
1.2133 + DBusMessage *message;
1.2134 + dbus_uint32_t serial;
1.2135 + dbus_bool_t retval;
1.2136 +
1.2137 + _dbus_verbose ("Sending exit message to the test service\n");
1.2138 +
1.2139 + retval = FALSE;
1.2140 +
1.2141 + /* Kill off the test service by sending it a quit message */
1.2142 + message = dbus_message_new_method_call (service_name,
1.2143 + "/org/freedesktop/TestSuite",
1.2144 + "org.freedesktop.TestSuite",
1.2145 + "Exit");
1.2146 +
1.2147 + if (message == NULL)
1.2148 + {
1.2149 + /* Do this again; we still need the service to exit... */
1.2150 + if (!check_send_exit_to_service (context, connection,
1.2151 + service_name, base_service))
1.2152 + goto out;
1.2153 +
1.2154 + return TRUE;
1.2155 + }
1.2156 +
1.2157 + if (!dbus_connection_send (connection, message, &serial))
1.2158 + {
1.2159 + dbus_message_unref (message);
1.2160 +
1.2161 + /* Do this again; we still need the service to exit... */
1.2162 + if (!check_send_exit_to_service (context, connection,
1.2163 + service_name, base_service))
1.2164 + goto out;
1.2165 +
1.2166 + return TRUE;
1.2167 + }
1.2168 +
1.2169 + dbus_message_unref (message);
1.2170 + message = NULL;
1.2171 +
1.2172 + /* send message */
1.2173 + bus_test_run_clients_loop (SEND_PENDING (connection));
1.2174 +
1.2175 + /* read it in and write it out to test service */
1.2176 + bus_test_run_bus_loop (context, FALSE);
1.2177 +
1.2178 + /* see if we got an error during message bus dispatching */
1.2179 + bus_test_run_clients_loop (FALSE);
1.2180 + message = borrow_message_waiting_for_memory (connection);
1.2181 + got_error = message != NULL && dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR;
1.2182 + if (message)
1.2183 + {
1.2184 + dbus_connection_return_message (connection, message);
1.2185 + message = NULL;
1.2186 + }
1.2187 +
1.2188 + if (!got_error)
1.2189 + {
1.2190 + /* If no error, wait for the test service to exit */
1.2191 + block_connection_until_message_from_bus (context, connection, "test service to exit");
1.2192 +
1.2193 + bus_test_run_everything (context);
1.2194 + }
1.2195 +
1.2196 + if (got_error)
1.2197 + {
1.2198 + message = pop_message_waiting_for_memory (connection);
1.2199 + _dbus_assert (message != NULL);
1.2200 +
1.2201 + if (dbus_message_get_reply_serial (message) != serial)
1.2202 + {
1.2203 + warn_unexpected (connection, message,
1.2204 + "error with the correct reply serial");
1.2205 + goto out;
1.2206 + }
1.2207 +
1.2208 + if (!dbus_message_is_error (message,
1.2209 + DBUS_ERROR_NO_MEMORY))
1.2210 + {
1.2211 + warn_unexpected (connection, message,
1.2212 + "a no memory error from asking test service to exit");
1.2213 + goto out;
1.2214 + }
1.2215 +
1.2216 + _dbus_verbose ("Got error %s when asking test service to exit\n",
1.2217 + dbus_message_get_error_name (message));
1.2218 +
1.2219 + /* Do this again; we still need the service to exit... */
1.2220 + if (!check_send_exit_to_service (context, connection,
1.2221 + service_name, base_service))
1.2222 + goto out;
1.2223 + }
1.2224 + else
1.2225 + {
1.2226 + if (!check_service_deactivated (context, connection,
1.2227 + service_name, base_service))
1.2228 + goto out;
1.2229 +
1.2230 + /* Should now have a NoReply error from the Exit() method
1.2231 + * call; it should have come after all the deactivation
1.2232 + * stuff.
1.2233 + */
1.2234 + message = pop_message_waiting_for_memory (connection);
1.2235 +
1.2236 + if (message == NULL)
1.2237 + {
1.2238 + warn_unexpected (connection, NULL,
1.2239 + "reply to Exit() method call");
1.2240 + goto out;
1.2241 + }
1.2242 + if (!dbus_message_is_error (message,
1.2243 + DBUS_ERROR_NO_REPLY))
1.2244 + {
1.2245 + warn_unexpected (connection, message,
1.2246 + "NoReply error from Exit() method call");
1.2247 + goto out;
1.2248 + }
1.2249 +
1.2250 + if (dbus_message_get_reply_serial (message) != serial)
1.2251 + {
1.2252 + warn_unexpected (connection, message,
1.2253 + "error with the correct reply serial");
1.2254 + goto out;
1.2255 + }
1.2256 +
1.2257 + _dbus_verbose ("Got error %s after test service exited\n",
1.2258 + dbus_message_get_error_name (message));
1.2259 +
1.2260 + if (!check_no_leftovers (context))
1.2261 + {
1.2262 + _dbus_warn ("Messages were left over after %s\n",
1.2263 + _DBUS_FUNCTION_NAME);
1.2264 + goto out;
1.2265 + }
1.2266 + }
1.2267 +
1.2268 + retval = TRUE;
1.2269 +
1.2270 + out:
1.2271 + if (message)
1.2272 + dbus_message_unref (message);
1.2273 +
1.2274 + return retval;
1.2275 +}
1.2276 +
1.2277 +static dbus_bool_t
1.2278 +check_got_error (BusContext *context,
1.2279 + DBusConnection *connection,
1.2280 + const char *first_error_name,
1.2281 + ...)
1.2282 +{
1.2283 + DBusMessage *message;
1.2284 + dbus_bool_t retval;
1.2285 + va_list ap;
1.2286 + dbus_bool_t error_found;
1.2287 + const char *error_name;
1.2288 +
1.2289 + retval = FALSE;
1.2290 +
1.2291 + message = pop_message_waiting_for_memory (connection);
1.2292 + if (message == NULL)
1.2293 + {
1.2294 + _dbus_warn ("Did not get an expected error\n");
1.2295 + goto out;
1.2296 + }
1.2297 +
1.2298 + if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_ERROR)
1.2299 + {
1.2300 + warn_unexpected (connection, message, "an error");
1.2301 +
1.2302 + goto out;
1.2303 + }
1.2304 +
1.2305 + error_found = FALSE;
1.2306 +
1.2307 + va_start (ap, first_error_name);
1.2308 + error_name = first_error_name;
1.2309 + while (error_name != NULL)
1.2310 + {
1.2311 + if (dbus_message_is_error (message, error_name))
1.2312 + {
1.2313 + error_found = TRUE;
1.2314 + break;
1.2315 + }
1.2316 + error_name = va_arg (ap, char*);
1.2317 + }
1.2318 + va_end (ap);
1.2319 +
1.2320 + if (!error_found)
1.2321 + {
1.2322 + _dbus_warn ("Expected error %s or other, got %s instead\n",
1.2323 + first_error_name,
1.2324 + dbus_message_get_error_name (message));
1.2325 + goto out;
1.2326 + }
1.2327 +
1.2328 + retval = TRUE;
1.2329 +
1.2330 + out:
1.2331 + if (message)
1.2332 + dbus_message_unref (message);
1.2333 +
1.2334 + return retval;
1.2335 +}
1.2336 +
1.2337 +typedef enum
1.2338 +{
1.2339 + GOT_SERVICE_CREATED,
1.2340 + GOT_SERVICE_DELETED,
1.2341 + GOT_ERROR,
1.2342 + GOT_SOMETHING_ELSE
1.2343 +} GotServiceInfo;
1.2344 +
1.2345 +static GotServiceInfo
1.2346 +check_got_service_info (DBusMessage *message)
1.2347 +{
1.2348 + GotServiceInfo message_kind;
1.2349 +
1.2350 + if (dbus_message_is_signal (message,
1.2351 + DBUS_INTERFACE_DBUS,
1.2352 + "NameOwnerChanged"))
1.2353 + {
1.2354 + DBusError error;
1.2355 + const char *service_name, *old_owner, *new_owner;
1.2356 + dbus_error_init (&error);
1.2357 +
1.2358 + reget_service_info_data:
1.2359 + service_name = NULL;
1.2360 + old_owner = NULL;
1.2361 + new_owner = NULL;
1.2362 +
1.2363 + dbus_message_get_args (message, &error,
1.2364 + DBUS_TYPE_STRING, &service_name,
1.2365 + DBUS_TYPE_STRING, &old_owner,
1.2366 + DBUS_TYPE_STRING, &new_owner,
1.2367 + DBUS_TYPE_INVALID);
1.2368 + if (dbus_error_is_set (&error))
1.2369 + {
1.2370 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.2371 + {
1.2372 + dbus_error_free (&error);
1.2373 + goto reget_service_info_data;
1.2374 + }
1.2375 + else
1.2376 + {
1.2377 + _dbus_warn ("unexpected arguments for NameOwnerChanged message\n");
1.2378 + message_kind = GOT_SOMETHING_ELSE;
1.2379 + }
1.2380 + }
1.2381 + else if (!old_owner[0])
1.2382 + message_kind = GOT_SERVICE_CREATED;
1.2383 + else if (!new_owner[0])
1.2384 + message_kind = GOT_SERVICE_DELETED;
1.2385 + else
1.2386 + message_kind = GOT_SOMETHING_ELSE;
1.2387 +
1.2388 + dbus_error_free (&error);
1.2389 + }
1.2390 + else if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.2391 + message_kind = GOT_ERROR;
1.2392 + else
1.2393 + message_kind = GOT_SOMETHING_ELSE;
1.2394 +
1.2395 + return message_kind;
1.2396 +}
1.2397 +
1.2398 +#define EXISTENT_SERVICE_NAME "org.freedesktop.DBus.TestSuiteEchoService"
1.2399 +
1.2400 +/* returns TRUE if the correct thing happens,
1.2401 + * but the correct thing may include OOM errors.
1.2402 + */
1.2403 +static dbus_bool_t
1.2404 +check_existent_service_no_auto_start (BusContext *context,
1.2405 + DBusConnection *connection)
1.2406 +{
1.2407 + DBusMessage *message;
1.2408 + DBusMessage *base_service_message;
1.2409 + const char *base_service;
1.2410 + dbus_uint32_t serial;
1.2411 + dbus_bool_t retval;
1.2412 + const char *existent = EXISTENT_SERVICE_NAME;
1.2413 + dbus_uint32_t flags;
1.2414 +
1.2415 + base_service_message = NULL;
1.2416 +
1.2417 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.2418 + DBUS_PATH_DBUS,
1.2419 + DBUS_INTERFACE_DBUS,
1.2420 + "StartServiceByName");
1.2421 +
1.2422 + if (message == NULL)
1.2423 + return TRUE;
1.2424 +
1.2425 + dbus_message_set_auto_start (message, FALSE);
1.2426 +
1.2427 + flags = 0;
1.2428 + if (!dbus_message_append_args (message,
1.2429 + DBUS_TYPE_STRING, &existent,
1.2430 + DBUS_TYPE_UINT32, &flags,
1.2431 + DBUS_TYPE_INVALID))
1.2432 + {
1.2433 + dbus_message_unref (message);
1.2434 + return TRUE;
1.2435 + }
1.2436 +
1.2437 + if (!dbus_connection_send (connection, message, &serial))
1.2438 + {
1.2439 + dbus_message_unref (message);
1.2440 + return TRUE;
1.2441 + }
1.2442 +
1.2443 + dbus_message_unref (message);
1.2444 + message = NULL;
1.2445 +
1.2446 + bus_test_run_everything (context);
1.2447 +
1.2448 + /* now wait for the message bus to hear back from the activated
1.2449 + * service.
1.2450 + */
1.2451 + block_connection_until_message_from_bus (context, connection, "activated service to connect");
1.2452 +
1.2453 + bus_test_run_everything (context);
1.2454 +
1.2455 + if (!dbus_connection_get_is_connected (connection))
1.2456 + {
1.2457 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.2458 + return TRUE;
1.2459 + }
1.2460 +
1.2461 + retval = FALSE;
1.2462 +
1.2463 + message = pop_message_waiting_for_memory (connection);
1.2464 + if (message == NULL)
1.2465 + {
1.2466 + _dbus_warn ("Did not receive any messages after %s %d on %p\n",
1.2467 + "StartServiceByName", serial, connection);
1.2468 + goto out;
1.2469 + }
1.2470 +
1.2471 + verbose_message_received (connection, message);
1.2472 + _dbus_verbose (" (after sending %s)\n", "StartServiceByName");
1.2473 +
1.2474 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.2475 + {
1.2476 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.2477 + {
1.2478 + _dbus_warn ("Message has wrong sender %s\n",
1.2479 + dbus_message_get_sender (message) ?
1.2480 + dbus_message_get_sender (message) : "(none)");
1.2481 + goto out;
1.2482 + }
1.2483 +
1.2484 + if (dbus_message_is_error (message,
1.2485 + DBUS_ERROR_NO_MEMORY))
1.2486 + {
1.2487 + ; /* good, this is a valid response */
1.2488 + }
1.2489 + else if (dbus_message_is_error (message,
1.2490 + DBUS_ERROR_SPAWN_CHILD_EXITED) ||
1.2491 + dbus_message_is_error (message,
1.2492 + DBUS_ERROR_SPAWN_CHILD_SIGNALED) ||
1.2493 + dbus_message_is_error (message,
1.2494 + DBUS_ERROR_SPAWN_EXEC_FAILED))
1.2495 + {
1.2496 + ; /* good, this is expected also */
1.2497 + }
1.2498 + else
1.2499 + {
1.2500 + _dbus_warn ("Did not expect error %s\n",
1.2501 + dbus_message_get_error_name (message));
1.2502 + goto out;
1.2503 + }
1.2504 + }
1.2505 + else
1.2506 + {
1.2507 + GotServiceInfo message_kind;
1.2508 +
1.2509 + if (!check_base_service_activated (context, connection,
1.2510 + message, &base_service))
1.2511 + goto out;
1.2512 +
1.2513 + base_service_message = message;
1.2514 + message = NULL;
1.2515 +
1.2516 + /* We may need to block here for the test service to exit or finish up */
1.2517 + block_connection_until_message_from_bus (context, connection, "test service to exit or finish up");
1.2518 +
1.2519 + message = dbus_connection_borrow_message (connection);
1.2520 + if (message == NULL)
1.2521 + {
1.2522 + _dbus_warn ("Did not receive any messages after base service creation notification\n");
1.2523 + goto out;
1.2524 + }
1.2525 +
1.2526 + message_kind = check_got_service_info (message);
1.2527 +
1.2528 + dbus_connection_return_message (connection, message);
1.2529 + message = NULL;
1.2530 +
1.2531 + switch (message_kind)
1.2532 + {
1.2533 + case GOT_SOMETHING_ELSE:
1.2534 + _dbus_warn ("Unexpected message after ActivateService "
1.2535 + "(should be an error or a service announcement");
1.2536 + goto out;
1.2537 +
1.2538 + case GOT_ERROR:
1.2539 + if (!check_got_error (context, connection,
1.2540 + DBUS_ERROR_SPAWN_CHILD_EXITED,
1.2541 + DBUS_ERROR_NO_MEMORY,
1.2542 + NULL))
1.2543 + goto out;
1.2544 + /* A service deleted should be coming along now after this error.
1.2545 + * We can also get the error *after* the service deleted.
1.2546 + */
1.2547 +
1.2548 + /* fall through */
1.2549 +
1.2550 + case GOT_SERVICE_DELETED:
1.2551 + {
1.2552 + /* The service started up and got a base address, but then
1.2553 + * failed to register under EXISTENT_SERVICE_NAME
1.2554 + */
1.2555 + CheckServiceOwnerChangedData socd;
1.2556 +
1.2557 + socd.expected_kind = SERVICE_DELETED;
1.2558 + socd.expected_service_name = base_service;
1.2559 + socd.failed = FALSE;
1.2560 + socd.skip_connection = NULL;
1.2561 +
1.2562 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.2563 + &socd);
1.2564 +
1.2565 + if (socd.failed)
1.2566 + goto out;
1.2567 +
1.2568 + /* Now we should get an error about the service exiting
1.2569 + * if we didn't get it before.
1.2570 + */
1.2571 + if (message_kind != GOT_ERROR)
1.2572 + {
1.2573 + block_connection_until_message_from_bus (context, connection, "error about service exiting");
1.2574 +
1.2575 + /* and process everything again */
1.2576 + bus_test_run_everything (context);
1.2577 +
1.2578 + if (!check_got_error (context, connection,
1.2579 + DBUS_ERROR_SPAWN_CHILD_EXITED,
1.2580 + DBUS_ERROR_NO_MEMORY,
1.2581 + NULL))
1.2582 + goto out;
1.2583 + }
1.2584 + break;
1.2585 + }
1.2586 +
1.2587 + case GOT_SERVICE_CREATED:
1.2588 + message = pop_message_waiting_for_memory (connection);
1.2589 + if (message == NULL)
1.2590 + {
1.2591 + _dbus_warn ("Failed to pop message we just put back! "
1.2592 + "should have been a NameOwnerChanged (creation)\n");
1.2593 + goto out;
1.2594 + }
1.2595 +
1.2596 + if (!check_service_activated (context, connection, EXISTENT_SERVICE_NAME,
1.2597 + base_service, message))
1.2598 + goto out;
1.2599 +
1.2600 + dbus_message_unref (message);
1.2601 + message = NULL;
1.2602 +
1.2603 + if (!check_no_leftovers (context))
1.2604 + {
1.2605 + _dbus_warn ("Messages were left over after successful activation\n");
1.2606 + goto out;
1.2607 + }
1.2608 +
1.2609 + if (!check_send_exit_to_service (context, connection,
1.2610 + EXISTENT_SERVICE_NAME, base_service))
1.2611 + goto out;
1.2612 +
1.2613 + break;
1.2614 + }
1.2615 + }
1.2616 +
1.2617 + retval = TRUE;
1.2618 +
1.2619 + out:
1.2620 + if (message)
1.2621 + dbus_message_unref (message);
1.2622 +
1.2623 + if (base_service_message)
1.2624 + dbus_message_unref (base_service_message);
1.2625 +
1.2626 + return retval;
1.2627 +}
1.2628 +
1.2629 +/* returns TRUE if the correct thing happens,
1.2630 + * but the correct thing may include OOM errors.
1.2631 + */
1.2632 +static dbus_bool_t
1.2633 +check_segfault_service_no_auto_start (BusContext *context,
1.2634 + DBusConnection *connection)
1.2635 +{
1.2636 + DBusMessage *message;
1.2637 + dbus_uint32_t serial;
1.2638 + dbus_bool_t retval;
1.2639 + const char *segv_service;
1.2640 + dbus_uint32_t flags;
1.2641 +
1.2642 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.2643 + DBUS_PATH_DBUS,
1.2644 + DBUS_INTERFACE_DBUS,
1.2645 + "StartServiceByName");
1.2646 +
1.2647 + if (message == NULL)
1.2648 + return TRUE;
1.2649 +
1.2650 + dbus_message_set_auto_start (message, FALSE);
1.2651 +
1.2652 + segv_service = "org.freedesktop.DBus.TestSuiteSegfaultService";
1.2653 + flags = 0;
1.2654 + if (!dbus_message_append_args (message,
1.2655 + DBUS_TYPE_STRING, &segv_service,
1.2656 + DBUS_TYPE_UINT32, &flags,
1.2657 + DBUS_TYPE_INVALID))
1.2658 + {
1.2659 + dbus_message_unref (message);
1.2660 + return TRUE;
1.2661 + }
1.2662 +
1.2663 + if (!dbus_connection_send (connection, message, &serial))
1.2664 + {
1.2665 + dbus_message_unref (message);
1.2666 + return TRUE;
1.2667 + }
1.2668 +
1.2669 + dbus_message_unref (message);
1.2670 + message = NULL;
1.2671 +
1.2672 + bus_test_run_everything (context);
1.2673 + block_connection_until_message_from_bus (context, connection, "reply to activating segfault service");
1.2674 + bus_test_run_everything (context);
1.2675 +
1.2676 + if (!dbus_connection_get_is_connected (connection))
1.2677 + {
1.2678 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.2679 + return TRUE;
1.2680 + }
1.2681 +
1.2682 + retval = FALSE;
1.2683 +
1.2684 + message = pop_message_waiting_for_memory (connection);
1.2685 + if (message == NULL)
1.2686 + {
1.2687 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.2688 + "StartServiceByName", serial, connection);
1.2689 + goto out;
1.2690 + }
1.2691 +
1.2692 + verbose_message_received (connection, message);
1.2693 +
1.2694 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.2695 + {
1.2696 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.2697 + {
1.2698 + _dbus_warn ("Message has wrong sender %s\n",
1.2699 + dbus_message_get_sender (message) ?
1.2700 + dbus_message_get_sender (message) : "(none)");
1.2701 + goto out;
1.2702 + }
1.2703 +
1.2704 + if (dbus_message_is_error (message,
1.2705 + DBUS_ERROR_NO_MEMORY))
1.2706 + {
1.2707 + ; /* good, this is a valid response */
1.2708 + }
1.2709 + else if (dbus_message_is_error (message,
1.2710 + DBUS_ERROR_SPAWN_CHILD_SIGNALED))
1.2711 + {
1.2712 + ; /* good, this is expected also */
1.2713 + }
1.2714 + else
1.2715 + {
1.2716 + warn_unexpected (connection, message, "not this error");
1.2717 +
1.2718 + goto out;
1.2719 + }
1.2720 + }
1.2721 + else
1.2722 + {
1.2723 + _dbus_warn ("Did not expect to successfully activate segfault service\n");
1.2724 + goto out;
1.2725 + }
1.2726 +
1.2727 + retval = TRUE;
1.2728 +
1.2729 + out:
1.2730 + if (message)
1.2731 + dbus_message_unref (message);
1.2732 +
1.2733 + return retval;
1.2734 +}
1.2735 +
1.2736 +
1.2737 +/* returns TRUE if the correct thing happens,
1.2738 + * but the correct thing may include OOM errors.
1.2739 + */
1.2740 +static dbus_bool_t
1.2741 +check_segfault_service_auto_start (BusContext *context,
1.2742 + DBusConnection *connection)
1.2743 +{
1.2744 + DBusMessage *message;
1.2745 + dbus_uint32_t serial;
1.2746 + dbus_bool_t retval;
1.2747 +
1.2748 + message = dbus_message_new_method_call ("org.freedesktop.DBus.TestSuiteSegfaultService",
1.2749 + "/org/freedesktop/TestSuite",
1.2750 + "org.freedesktop.TestSuite",
1.2751 + "Echo");
1.2752 +
1.2753 + if (message == NULL)
1.2754 + return TRUE;
1.2755 +
1.2756 + if (!dbus_connection_send (connection, message, &serial))
1.2757 + {
1.2758 + dbus_message_unref (message);
1.2759 + return TRUE;
1.2760 + }
1.2761 +
1.2762 + dbus_message_unref (message);
1.2763 + message = NULL;
1.2764 +
1.2765 + bus_test_run_everything (context);
1.2766 + block_connection_until_message_from_bus (context, connection, "reply to Echo on segfault service");
1.2767 + bus_test_run_everything (context);
1.2768 +
1.2769 + if (!dbus_connection_get_is_connected (connection))
1.2770 + {
1.2771 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.2772 + return TRUE;
1.2773 + }
1.2774 +
1.2775 + retval = FALSE;
1.2776 +
1.2777 + message = pop_message_waiting_for_memory (connection);
1.2778 + if (message == NULL)
1.2779 + {
1.2780 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.2781 + "Echo message (auto activation)", serial, connection);
1.2782 + goto out;
1.2783 + }
1.2784 +
1.2785 + verbose_message_received (connection, message);
1.2786 +
1.2787 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.2788 + {
1.2789 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.2790 + {
1.2791 + _dbus_warn ("Message has wrong sender %s\n",
1.2792 + dbus_message_get_sender (message) ?
1.2793 + dbus_message_get_sender (message) : "(none)");
1.2794 + goto out;
1.2795 + }
1.2796 +
1.2797 + if (dbus_message_is_error (message,
1.2798 + DBUS_ERROR_NO_MEMORY))
1.2799 + {
1.2800 + ; /* good, this is a valid response */
1.2801 + }
1.2802 + else if (dbus_message_is_error (message,
1.2803 + DBUS_ERROR_SPAWN_CHILD_SIGNALED))
1.2804 + {
1.2805 + ; /* good, this is expected also */
1.2806 + }
1.2807 + else
1.2808 + {
1.2809 + warn_unexpected (connection, message, "not this error");
1.2810 +
1.2811 + goto out;
1.2812 + }
1.2813 + }
1.2814 + else
1.2815 + {
1.2816 + _dbus_warn ("Did not expect to successfully activate segfault service\n");
1.2817 + goto out;
1.2818 + }
1.2819 +
1.2820 + retval = TRUE;
1.2821 +
1.2822 + out:
1.2823 + if (message)
1.2824 + dbus_message_unref (message);
1.2825 +
1.2826 + return retval;
1.2827 +}
1.2828 +
1.2829 +#define TEST_ECHO_MESSAGE "Test echo message"
1.2830 +#define TEST_RUN_HELLO_FROM_SELF_MESSAGE "Test sending message to self"
1.2831 +
1.2832 +/* returns TRUE if the correct thing happens,
1.2833 + * but the correct thing may include OOM errors.
1.2834 + */
1.2835 +static dbus_bool_t
1.2836 +check_existent_hello_from_self (BusContext *context,
1.2837 + DBusConnection *connection)
1.2838 +{
1.2839 + DBusMessage *message;
1.2840 + dbus_uint32_t serial;
1.2841 + const char *text;
1.2842 +
1.2843 + message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME,
1.2844 + "/org/freedesktop/TestSuite",
1.2845 + "org.freedesktop.TestSuite",
1.2846 + "RunHelloFromSelf");
1.2847 +
1.2848 + if (message == NULL)
1.2849 + return TRUE;
1.2850 +
1.2851 + text = TEST_RUN_HELLO_FROM_SELF_MESSAGE;
1.2852 + if (!dbus_message_append_args (message,
1.2853 + DBUS_TYPE_STRING, &text,
1.2854 + DBUS_TYPE_INVALID))
1.2855 + {
1.2856 + dbus_message_unref (message);
1.2857 + return TRUE;
1.2858 + }
1.2859 +
1.2860 + if (!dbus_connection_send (connection, message, &serial))
1.2861 + {
1.2862 + dbus_message_unref (message);
1.2863 + return TRUE;
1.2864 + }
1.2865 +
1.2866 + dbus_message_unref (message);
1.2867 + message = NULL;
1.2868 +
1.2869 + bus_test_run_everything (context);
1.2870 +
1.2871 + /* Note: if this test is run in OOM mode, it will block when the bus
1.2872 + * doesn't send a reply due to OOM.
1.2873 + */
1.2874 + block_connection_until_message_from_bus (context, connection, "reply from running hello from self");
1.2875 +
1.2876 + message = pop_message_waiting_for_memory (connection);
1.2877 + if (message == NULL)
1.2878 + {
1.2879 + _dbus_warn ("Failed to pop message! Should have been reply from RunHelloFromSelf message\n");
1.2880 + return FALSE;
1.2881 + }
1.2882 +
1.2883 + if (dbus_message_get_reply_serial (message) != serial)
1.2884 + {
1.2885 + _dbus_warn ("Wrong reply serial\n");
1.2886 + dbus_message_unref (message);
1.2887 + return FALSE;
1.2888 + }
1.2889 +
1.2890 + dbus_message_unref (message);
1.2891 + message = NULL;
1.2892 +
1.2893 + return TRUE;
1.2894 +}
1.2895 +
1.2896 +/* returns TRUE if the correct thing happens,
1.2897 + * but the correct thing may include OOM errors.
1.2898 + */
1.2899 +static dbus_bool_t
1.2900 +check_existent_ping (BusContext *context,
1.2901 + DBusConnection *connection)
1.2902 +{
1.2903 + DBusMessage *message;
1.2904 + dbus_uint32_t serial;
1.2905 + message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME,
1.2906 + "/org/freedesktop/TestSuite",
1.2907 + "org.freedesktop.DBus.Peer",
1.2908 + "Ping");
1.2909 +
1.2910 + if (message == NULL)
1.2911 + return TRUE;
1.2912 +
1.2913 + if (!dbus_connection_send (connection, message, &serial))
1.2914 + {
1.2915 + dbus_message_unref (message);
1.2916 + return TRUE;
1.2917 + }
1.2918 +
1.2919 + dbus_message_unref (message);
1.2920 + message = NULL;
1.2921 +
1.2922 + bus_test_run_everything (context);
1.2923 +
1.2924 + /* Note: if this test is run in OOM mode, it will block when the bus
1.2925 + * doesn't send a reply due to OOM.
1.2926 + */
1.2927 + block_connection_until_message_from_bus (context, connection, "reply from running Ping");
1.2928 +
1.2929 + message = pop_message_waiting_for_memory (connection);
1.2930 + if (message == NULL)
1.2931 + {
1.2932 + _dbus_warn ("Failed to pop message! Should have been reply from Ping message\n");
1.2933 + return FALSE;
1.2934 + }
1.2935 +
1.2936 + if (dbus_message_get_reply_serial (message) != serial)
1.2937 + {
1.2938 + _dbus_warn ("Wrong reply serial\n");
1.2939 + dbus_message_unref (message);
1.2940 + return FALSE;
1.2941 + }
1.2942 +
1.2943 + if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.2944 + {
1.2945 + _dbus_warn ("Unexpected message return during Ping\n");
1.2946 + dbus_message_unref (message);
1.2947 + return FALSE;
1.2948 + }
1.2949 +
1.2950 + dbus_message_unref (message);
1.2951 + message = NULL;
1.2952 +
1.2953 + return TRUE;
1.2954 +}
1.2955 +
1.2956 +/* returns TRUE if the correct thing happens,
1.2957 + * but the correct thing may include OOM errors.
1.2958 + */
1.2959 +static dbus_bool_t
1.2960 +check_existent_get_machine_id (BusContext *context,
1.2961 + DBusConnection *connection)
1.2962 +{
1.2963 + DBusMessage *message;
1.2964 + dbus_uint32_t serial;
1.2965 + const char *machine_id;
1.2966 +
1.2967 + message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME,
1.2968 + "/org/freedesktop/TestSuite",
1.2969 + "org.freedesktop.DBus.Peer",
1.2970 + "GetMachineId");
1.2971 +
1.2972 + if (message == NULL)
1.2973 + return TRUE;
1.2974 +
1.2975 + if (!dbus_connection_send (connection, message, &serial))
1.2976 + {
1.2977 + dbus_message_unref (message);
1.2978 + return TRUE;
1.2979 + }
1.2980 +
1.2981 + dbus_message_unref (message);
1.2982 + message = NULL;
1.2983 +
1.2984 + bus_test_run_everything (context);
1.2985 +
1.2986 + /* Note: if this test is run in OOM mode, it will block when the bus
1.2987 + * doesn't send a reply due to OOM.
1.2988 + */
1.2989 + block_connection_until_message_from_bus (context, connection, "reply from running GetMachineId");
1.2990 +
1.2991 + message = pop_message_waiting_for_memory (connection);
1.2992 + if (message == NULL)
1.2993 + {
1.2994 + _dbus_warn ("Failed to pop message! Should have been reply from GetMachineId message\n");
1.2995 + return FALSE;
1.2996 + }
1.2997 +
1.2998 + if (dbus_message_get_reply_serial (message) != serial)
1.2999 + {
1.3000 + _dbus_warn ("Wrong reply serial\n");
1.3001 + dbus_message_unref (message);
1.3002 + return FALSE;
1.3003 + }
1.3004 +
1.3005 + if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.3006 + {
1.3007 + _dbus_warn ("Unexpected message return during GetMachineId\n");
1.3008 + dbus_message_unref (message);
1.3009 + return FALSE;
1.3010 + }
1.3011 +
1.3012 + machine_id = NULL;
1.3013 + if (!dbus_message_get_args (message, NULL, DBUS_TYPE_STRING, &machine_id, DBUS_TYPE_INVALID))
1.3014 + {
1.3015 + _dbus_warn ("Did not get a machine ID in reply to GetMachineId\n");
1.3016 + dbus_message_unref (message);
1.3017 + return FALSE;
1.3018 + }
1.3019 +
1.3020 + if (machine_id == NULL || strlen (machine_id) != 32)
1.3021 + {
1.3022 + _dbus_warn ("Machine id looks bogus: '%s'\n", machine_id ? machine_id : "null");
1.3023 + dbus_message_unref (message);
1.3024 + return FALSE;
1.3025 + }
1.3026 +
1.3027 + /* We can't check that the machine id is correct because during make check it is
1.3028 + * just made up for each process separately
1.3029 + */
1.3030 +
1.3031 + dbus_message_unref (message);
1.3032 + message = NULL;
1.3033 +
1.3034 + return TRUE;
1.3035 +}
1.3036 +
1.3037 +/* returns TRUE if the correct thing happens,
1.3038 + * but the correct thing may include OOM errors.
1.3039 + */
1.3040 +static dbus_bool_t
1.3041 +check_existent_service_auto_start (BusContext *context,
1.3042 + DBusConnection *connection)
1.3043 +{
1.3044 + DBusMessage *message;
1.3045 + DBusMessage *base_service_message;
1.3046 + dbus_uint32_t serial;
1.3047 + dbus_bool_t retval;
1.3048 + const char *base_service;
1.3049 + const char *text;
1.3050 +
1.3051 + base_service_message = NULL;
1.3052 +
1.3053 + message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME,
1.3054 + "/org/freedesktop/TestSuite",
1.3055 + "org.freedesktop.TestSuite",
1.3056 + "Echo");
1.3057 +
1.3058 + if (message == NULL)
1.3059 + return TRUE;
1.3060 +
1.3061 + text = TEST_ECHO_MESSAGE;
1.3062 + if (!dbus_message_append_args (message,
1.3063 + DBUS_TYPE_STRING, &text,
1.3064 + DBUS_TYPE_INVALID))
1.3065 + {
1.3066 + dbus_message_unref (message);
1.3067 + return TRUE;
1.3068 + }
1.3069 +
1.3070 + if (!dbus_connection_send (connection, message, &serial))
1.3071 + {
1.3072 + dbus_message_unref (message);
1.3073 + return TRUE;
1.3074 + }
1.3075 +
1.3076 + dbus_message_unref (message);
1.3077 + message = NULL;
1.3078 +
1.3079 + bus_test_run_everything (context);
1.3080 +
1.3081 + /* now wait for the message bus to hear back from the activated
1.3082 + * service.
1.3083 + */
1.3084 + block_connection_until_message_from_bus (context, connection, "reply to Echo on existent service");
1.3085 + bus_test_run_everything (context);
1.3086 +
1.3087 + if (!dbus_connection_get_is_connected (connection))
1.3088 + {
1.3089 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.3090 + return TRUE;
1.3091 + }
1.3092 +
1.3093 + retval = FALSE;
1.3094 +
1.3095 + message = pop_message_waiting_for_memory (connection);
1.3096 + if (message == NULL)
1.3097 + {
1.3098 + _dbus_warn ("Did not receive any messages after auto start %d on %p\n",
1.3099 + serial, connection);
1.3100 + goto out;
1.3101 + }
1.3102 +
1.3103 + verbose_message_received (connection, message);
1.3104 + _dbus_verbose (" (after sending %s)\n", "auto start");
1.3105 +
1.3106 + /* we should get zero or two ServiceOwnerChanged signals */
1.3107 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_SIGNAL)
1.3108 + {
1.3109 + GotServiceInfo message_kind;
1.3110 +
1.3111 + if (!check_base_service_activated (context, connection,
1.3112 + message, &base_service))
1.3113 + goto out;
1.3114 +
1.3115 + base_service_message = message;
1.3116 + message = NULL;
1.3117 +
1.3118 + /* We may need to block here for the test service to exit or finish up */
1.3119 + block_connection_until_message_from_bus (context, connection, "service to exit");
1.3120 +
1.3121 + /* Should get a service creation notification for the activated
1.3122 + * service name, or a service deletion on the base service name
1.3123 + */
1.3124 + message = dbus_connection_borrow_message (connection);
1.3125 + if (message == NULL)
1.3126 + {
1.3127 + _dbus_warn ("No message after auto activation "
1.3128 + "(should be a service announcement)\n");
1.3129 + dbus_connection_return_message (connection, message);
1.3130 + message = NULL;
1.3131 + goto out;
1.3132 + }
1.3133 +
1.3134 + message_kind = check_got_service_info (message);
1.3135 +
1.3136 + dbus_connection_return_message (connection, message);
1.3137 + message = NULL;
1.3138 +
1.3139 + switch (message_kind)
1.3140 + {
1.3141 + case GOT_SERVICE_CREATED:
1.3142 + message = pop_message_waiting_for_memory (connection);
1.3143 + if (message == NULL)
1.3144 + {
1.3145 + _dbus_warn ("Failed to pop message we just put back! "
1.3146 + "should have been a NameOwnerChanged (creation)\n");
1.3147 + goto out;
1.3148 + }
1.3149 +
1.3150 + /* Check that ServiceOwnerChanged (creation) was correctly received */
1.3151 + if (!check_service_auto_activated (context, connection, EXISTENT_SERVICE_NAME,
1.3152 + base_service, message))
1.3153 + goto out;
1.3154 +
1.3155 + dbus_message_unref (message);
1.3156 + message = NULL;
1.3157 +
1.3158 + break;
1.3159 +
1.3160 + case GOT_SERVICE_DELETED:
1.3161 + {
1.3162 + /* The service started up and got a base address, but then
1.3163 + * failed to register under EXISTENT_SERVICE_NAME
1.3164 + */
1.3165 + CheckServiceOwnerChangedData socd;
1.3166 +
1.3167 + socd.expected_kind = SERVICE_DELETED;
1.3168 + socd.expected_service_name = base_service;
1.3169 + socd.failed = FALSE;
1.3170 + socd.skip_connection = NULL;
1.3171 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.3172 + &socd);
1.3173 +
1.3174 + if (socd.failed)
1.3175 + goto out;
1.3176 +
1.3177 + break;
1.3178 + }
1.3179 +
1.3180 + case GOT_ERROR:
1.3181 + case GOT_SOMETHING_ELSE:
1.3182 + _dbus_warn ("Unexpected message after auto activation\n");
1.3183 + goto out;
1.3184 + }
1.3185 + }
1.3186 +
1.3187 + /* OK, now we've dealt with ServiceOwnerChanged signals, now should
1.3188 + * come the method reply (or error) from the initial method call
1.3189 + */
1.3190 +
1.3191 + /* Note: if this test is run in OOM mode, it will block when the bus
1.3192 + * doesn't send a reply due to OOM.
1.3193 + */
1.3194 + block_connection_until_message_from_bus (context, connection, "reply from echo message after auto-activation");
1.3195 +
1.3196 + message = pop_message_waiting_for_memory (connection);
1.3197 + if (message == NULL)
1.3198 + {
1.3199 + _dbus_warn ("Failed to pop message! Should have been reply from echo message\n");
1.3200 + goto out;
1.3201 + }
1.3202 +
1.3203 + if (dbus_message_get_reply_serial (message) != serial)
1.3204 + {
1.3205 + _dbus_warn ("Wrong reply serial\n");
1.3206 + goto out;
1.3207 + }
1.3208 +
1.3209 + dbus_message_unref (message);
1.3210 + message = NULL;
1.3211 +
1.3212 + if (!check_existent_ping (context, connection))
1.3213 + goto out;
1.3214 +
1.3215 + if (!check_existent_get_machine_id (context, connection))
1.3216 + goto out;
1.3217 +
1.3218 + if (!check_existent_hello_from_self (context, connection))
1.3219 + goto out;
1.3220 +
1.3221 + if (!check_send_exit_to_service (context, connection,
1.3222 + EXISTENT_SERVICE_NAME,
1.3223 + base_service))
1.3224 + goto out;
1.3225 +
1.3226 + retval = TRUE;
1.3227 +
1.3228 + out:
1.3229 + if (message)
1.3230 + dbus_message_unref (message);
1.3231 +
1.3232 + if (base_service_message)
1.3233 + dbus_message_unref (base_service_message);
1.3234 +
1.3235 + return retval;
1.3236 +}
1.3237 +
1.3238 +#define SHELL_FAIL_SERVICE_NAME "org.freedesktop.DBus.TestSuiteShellEchoServiceFail"
1.3239 +
1.3240 +/* returns TRUE if the correct thing happens,
1.3241 + * but the correct thing may include OOM errors.
1.3242 + */
1.3243 +static dbus_bool_t
1.3244 +check_shell_fail_service_auto_start (BusContext *context,
1.3245 + DBusConnection *connection)
1.3246 +{
1.3247 + DBusMessage *message;
1.3248 + dbus_uint32_t serial;
1.3249 + dbus_bool_t retval;
1.3250 +
1.3251 + message = dbus_message_new_method_call (SHELL_FAIL_SERVICE_NAME,
1.3252 + "/org/freedesktop/TestSuite",
1.3253 + "org.freedesktop.TestSuite",
1.3254 + "Echo");
1.3255 +
1.3256 + if (message == NULL)
1.3257 + return TRUE;
1.3258 +
1.3259 + if (!dbus_connection_send (connection, message, &serial))
1.3260 + {
1.3261 + dbus_message_unref (message);
1.3262 + return TRUE;
1.3263 + }
1.3264 +
1.3265 + dbus_message_unref (message);
1.3266 + message = NULL;
1.3267 +
1.3268 + bus_test_run_everything (context);
1.3269 + block_connection_until_message_from_bus (context, connection, "reply to shell Echo on service which should fail to auto-start");
1.3270 + bus_test_run_everything (context);
1.3271 +
1.3272 + if (!dbus_connection_get_is_connected (connection))
1.3273 + {
1.3274 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.3275 + return TRUE;
1.3276 + }
1.3277 +
1.3278 + retval = FALSE;
1.3279 +
1.3280 + message = pop_message_waiting_for_memory (connection);
1.3281 + if (message == NULL)
1.3282 + {
1.3283 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.3284 + "Echo message (auto activation)", serial, connection);
1.3285 + goto out;
1.3286 + }
1.3287 +
1.3288 + verbose_message_received (connection, message);
1.3289 +
1.3290 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.3291 + {
1.3292 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.3293 + {
1.3294 + _dbus_warn ("Message has wrong sender %s\n",
1.3295 + dbus_message_get_sender (message) ?
1.3296 + dbus_message_get_sender (message) : "(none)");
1.3297 + goto out;
1.3298 + }
1.3299 +
1.3300 + if (dbus_message_is_error (message,
1.3301 + DBUS_ERROR_NO_MEMORY))
1.3302 + {
1.3303 + ; /* good, this is a valid response */
1.3304 + }
1.3305 + else if (dbus_message_is_error (message,
1.3306 + DBUS_ERROR_INVALID_ARGS))
1.3307 + {
1.3308 + _dbus_verbose("got invalid args\n");
1.3309 + ; /* good, this is expected also */
1.3310 + }
1.3311 + else
1.3312 + {
1.3313 + warn_unexpected (connection, message, "not this error");
1.3314 +
1.3315 + goto out;
1.3316 + }
1.3317 + }
1.3318 + else
1.3319 + {
1.3320 + _dbus_warn ("Did not expect to successfully auto-start shell fail service\n");
1.3321 + goto out;
1.3322 + }
1.3323 +
1.3324 + retval = TRUE;
1.3325 +
1.3326 + out:
1.3327 + if (message)
1.3328 + dbus_message_unref (message);
1.3329 +
1.3330 + return retval;
1.3331 +}
1.3332 +
1.3333 +#define SHELL_SUCCESS_SERVICE_NAME "org.freedesktop.DBus.TestSuiteShellEchoServiceSuccess"
1.3334 +
1.3335 +/* returns TRUE if the correct thing happens,
1.3336 + * but the correct thing may include OOM errors.
1.3337 + */
1.3338 +static dbus_bool_t
1.3339 +check_shell_service_success_auto_start (BusContext *context,
1.3340 + DBusConnection *connection)
1.3341 +{
1.3342 + DBusMessage *message;
1.3343 + DBusMessage *base_service_message;
1.3344 + dbus_uint32_t serial;
1.3345 + dbus_bool_t retval;
1.3346 + const char *base_service;
1.3347 + const char *argv[7] = {NULL, NULL, NULL, NULL, NULL, NULL, NULL};
1.3348 +
1.3349 + base_service_message = NULL;
1.3350 +
1.3351 + message = dbus_message_new_method_call (SHELL_SUCCESS_SERVICE_NAME,
1.3352 + "/org/freedesktop/TestSuite",
1.3353 + "org.freedesktop.TestSuite",
1.3354 + "Echo");
1.3355 +
1.3356 + if (message == NULL)
1.3357 + return TRUE;
1.3358 +
1.3359 + if (!dbus_connection_send (connection, message, &serial))
1.3360 + {
1.3361 + dbus_message_unref (message);
1.3362 + return TRUE;
1.3363 + }
1.3364 +
1.3365 + dbus_message_unref (message);
1.3366 + message = NULL;
1.3367 +
1.3368 + bus_test_run_everything (context);
1.3369 +
1.3370 + /* now wait for the message bus to hear back from the activated
1.3371 + * service.
1.3372 + */
1.3373 + block_connection_until_message_from_bus (context, connection, "reply to Echo on shell success service");
1.3374 + bus_test_run_everything (context);
1.3375 +
1.3376 + if (!dbus_connection_get_is_connected (connection))
1.3377 + {
1.3378 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.3379 + return TRUE;
1.3380 + }
1.3381 +
1.3382 + retval = FALSE;
1.3383 +
1.3384 + message = pop_message_waiting_for_memory (connection);
1.3385 + if (message == NULL)
1.3386 + {
1.3387 + _dbus_warn ("Did not receive any messages after auto start %d on %p\n",
1.3388 + serial, connection);
1.3389 + goto out;
1.3390 + }
1.3391 +
1.3392 + verbose_message_received (connection, message);
1.3393 + _dbus_verbose (" (after sending %s)\n", "auto start");
1.3394 +
1.3395 + /* we should get zero or two ServiceOwnerChanged signals */
1.3396 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_SIGNAL)
1.3397 + {
1.3398 + GotServiceInfo message_kind;
1.3399 +
1.3400 + if (!check_base_service_activated (context, connection,
1.3401 + message, &base_service))
1.3402 + goto out;
1.3403 +
1.3404 + base_service_message = message;
1.3405 + message = NULL;
1.3406 +
1.3407 + /* We may need to block here for the test service to exit or finish up */
1.3408 + block_connection_until_message_from_bus (context, connection, "service to exit");
1.3409 +
1.3410 + /* Should get a service creation notification for the activated
1.3411 + * service name, or a service deletion on the base service name
1.3412 + */
1.3413 + message = dbus_connection_borrow_message (connection);
1.3414 + if (message == NULL)
1.3415 + {
1.3416 + _dbus_warn ("No message after auto activation "
1.3417 + "(should be a service announcement)\n");
1.3418 + dbus_connection_return_message (connection, message);
1.3419 + message = NULL;
1.3420 + goto out;
1.3421 + }
1.3422 +
1.3423 + message_kind = check_got_service_info (message);
1.3424 +
1.3425 + dbus_connection_return_message (connection, message);
1.3426 + message = NULL;
1.3427 +
1.3428 + switch (message_kind)
1.3429 + {
1.3430 + case GOT_SERVICE_CREATED:
1.3431 + message = pop_message_waiting_for_memory (connection);
1.3432 + if (message == NULL)
1.3433 + {
1.3434 + _dbus_warn ("Failed to pop message we just put back! "
1.3435 + "should have been a NameOwnerChanged (creation)\n");
1.3436 + goto out;
1.3437 + }
1.3438 +
1.3439 + /* Check that ServiceOwnerChanged (creation) was correctly received */
1.3440 + if (!check_service_auto_activated (context, connection, SHELL_SUCCESS_SERVICE_NAME,
1.3441 + base_service, message))
1.3442 + goto out;
1.3443 +
1.3444 + dbus_message_unref (message);
1.3445 + message = NULL;
1.3446 +
1.3447 + break;
1.3448 +
1.3449 + case GOT_SERVICE_DELETED:
1.3450 + {
1.3451 + /* The service started up and got a base address, but then
1.3452 + * failed to register under SHELL_SUCCESS_SERVICE_NAME
1.3453 + */
1.3454 + CheckServiceOwnerChangedData socd;
1.3455 +
1.3456 + socd.expected_kind = SERVICE_DELETED;
1.3457 + socd.expected_service_name = base_service;
1.3458 + socd.failed = FALSE;
1.3459 + socd.skip_connection = NULL;
1.3460 + bus_test_clients_foreach (check_service_owner_changed_foreach,
1.3461 + &socd);
1.3462 +
1.3463 + if (socd.failed)
1.3464 + goto out;
1.3465 +
1.3466 + break;
1.3467 + }
1.3468 +
1.3469 + case GOT_ERROR:
1.3470 + case GOT_SOMETHING_ELSE:
1.3471 + _dbus_warn ("Unexpected message after auto activation\n");
1.3472 + goto out;
1.3473 + }
1.3474 + }
1.3475 +
1.3476 + /* OK, now we've dealt with ServiceOwnerChanged signals, now should
1.3477 + * come the method reply (or error) from the initial method call
1.3478 + */
1.3479 +
1.3480 + /* Note: if this test is run in OOM mode, it will block when the bus
1.3481 + * doesn't send a reply due to OOM.
1.3482 + */
1.3483 + block_connection_until_message_from_bus (context, connection, "reply from echo message after auto-activation");
1.3484 +
1.3485 + message = pop_message_waiting_for_memory (connection);
1.3486 + if (message == NULL)
1.3487 + {
1.3488 + _dbus_warn ("Failed to pop message! Should have been reply from echo message\n");
1.3489 + goto out;
1.3490 + }
1.3491 +
1.3492 + if (dbus_message_get_reply_serial (message) != serial)
1.3493 + {
1.3494 + _dbus_warn ("Wrong reply serial\n");
1.3495 + goto out;
1.3496 + }
1.3497 +
1.3498 + if (!dbus_message_get_args (message, NULL,
1.3499 + DBUS_TYPE_STRING, &argv[0],
1.3500 + DBUS_TYPE_STRING, &argv[1],
1.3501 + DBUS_TYPE_STRING, &argv[2],
1.3502 + DBUS_TYPE_STRING, &argv[3],
1.3503 + DBUS_TYPE_STRING, &argv[4],
1.3504 + DBUS_TYPE_STRING, &argv[5],
1.3505 + DBUS_TYPE_STRING, &argv[6],
1.3506 + DBUS_TYPE_INVALID))
1.3507 + {
1.3508 + _dbus_warn ("Error getting arguments from return\n");
1.3509 + goto out;
1.3510 + }
1.3511 +
1.3512 + /* don't worry about arg[0] as it may be different
1.3513 + depending on the path to the tests
1.3514 + */
1.3515 + if (strcmp("-test", argv[1]) != 0)
1.3516 + {
1.3517 + _dbus_warn ("Unexpected argv[1] in shell success service test (expected: %s, got: %s)\n",
1.3518 + "-test", argv[1]);
1.3519 + goto out;
1.3520 + }
1.3521 +
1.3522 + if (strcmp("that", argv[2]) != 0)
1.3523 + {
1.3524 + _dbus_warn ("Unexpected argv[2] in shell success service test (expected: %s, got: %s)\n",
1.3525 + "that", argv[2]);
1.3526 + goto out;
1.3527 + }
1.3528 +
1.3529 + if (strcmp("we get", argv[3]) != 0)
1.3530 + {
1.3531 + _dbus_warn ("Unexpected argv[3] in shell success service test (expected: %s, got: %s)\n",
1.3532 + "we get", argv[3]);
1.3533 + goto out;
1.3534 + }
1.3535 +
1.3536 + if (strcmp("back", argv[4]) != 0)
1.3537 + {
1.3538 + _dbus_warn ("Unexpected argv[4] in shell success service test (expected: %s, got: %s)\n",
1.3539 + "back", argv[4]);
1.3540 + goto out;
1.3541 + }
1.3542 +
1.3543 + if (strcmp("--what", argv[5]) != 0)
1.3544 + {
1.3545 + _dbus_warn ("Unexpected argv[5] in shell success service test (expected: %s, got: %s)\n",
1.3546 + "--what", argv[5]);
1.3547 + goto out;
1.3548 + }
1.3549 +
1.3550 + if (strcmp("we put in", argv[6]) != 0)
1.3551 + {
1.3552 + _dbus_warn ("Unexpected argv[6] in shell success service test (expected: %s, got: %s)\n",
1.3553 + "we put in", argv[6]);
1.3554 + goto out;
1.3555 + }
1.3556 +
1.3557 + dbus_message_unref (message);
1.3558 + message = NULL;
1.3559 +
1.3560 + if (!check_send_exit_to_service (context, connection,
1.3561 + SHELL_SUCCESS_SERVICE_NAME,
1.3562 + base_service))
1.3563 + goto out;
1.3564 +
1.3565 + retval = TRUE;
1.3566 +
1.3567 + out:
1.3568 + if (message)
1.3569 + dbus_message_unref (message);
1.3570 +
1.3571 + if (base_service_message)
1.3572 + dbus_message_unref (base_service_message);
1.3573 +
1.3574 + return retval;
1.3575 +}
1.3576 +
1.3577 +typedef struct
1.3578 +{
1.3579 + Check1Func func;
1.3580 + BusContext *context;
1.3581 +} Check1Data;
1.3582 +
1.3583 +static dbus_bool_t
1.3584 +check_oom_check1_func (void *data)
1.3585 +{
1.3586 + Check1Data *d = data;
1.3587 +
1.3588 + if (! (* d->func) (d->context))
1.3589 + return FALSE;
1.3590 +
1.3591 + if (!check_no_leftovers (d->context))
1.3592 + {
1.3593 + _dbus_warn ("Messages were left over, should be covered by test suite\n");
1.3594 + return FALSE;
1.3595 + }
1.3596 +
1.3597 + return TRUE;
1.3598 +}
1.3599 +
1.3600 +static void
1.3601 +check1_try_iterations (BusContext *context,
1.3602 + const char *description,
1.3603 + Check1Func func)
1.3604 +{
1.3605 + Check1Data d;
1.3606 +
1.3607 + d.func = func;
1.3608 + d.context = context;
1.3609 +
1.3610 + if (!_dbus_test_oom_handling (description, check_oom_check1_func,
1.3611 + &d))
1.3612 + _dbus_assert_not_reached ("test failed");
1.3613 +}
1.3614 +
1.3615 +static dbus_bool_t
1.3616 +check_get_services (BusContext *context,
1.3617 + DBusConnection *connection,
1.3618 + const char *method,
1.3619 + char ***services,
1.3620 + int *len)
1.3621 +{
1.3622 + DBusMessage *message;
1.3623 + dbus_uint32_t serial;
1.3624 + dbus_bool_t retval;
1.3625 + DBusError error;
1.3626 + char **srvs;
1.3627 + int l;
1.3628 +
1.3629 + retval = FALSE;
1.3630 + dbus_error_init (&error);
1.3631 + message = NULL;
1.3632 +
1.3633 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.3634 + DBUS_PATH_DBUS,
1.3635 + DBUS_INTERFACE_DBUS,
1.3636 + method);
1.3637 +
1.3638 + if (message == NULL)
1.3639 + return TRUE;
1.3640 +
1.3641 + if (!dbus_connection_send (connection, message, &serial))
1.3642 + {
1.3643 + dbus_message_unref (message);
1.3644 + return TRUE;
1.3645 + }
1.3646 +
1.3647 + /* send our message */
1.3648 + bus_test_run_clients_loop (SEND_PENDING (connection));
1.3649 +
1.3650 + dbus_message_unref (message);
1.3651 + message = NULL;
1.3652 +
1.3653 + dbus_connection_ref (connection); /* because we may get disconnected */
1.3654 + block_connection_until_message_from_bus (context, connection, "reply to ListActivatableNames/ListNames");
1.3655 +
1.3656 + if (!dbus_connection_get_is_connected (connection))
1.3657 + {
1.3658 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.3659 +
1.3660 + dbus_connection_unref (connection);
1.3661 +
1.3662 + return TRUE;
1.3663 + }
1.3664 +
1.3665 + dbus_connection_unref (connection);
1.3666 +
1.3667 + message = pop_message_waiting_for_memory (connection);
1.3668 + if (message == NULL)
1.3669 + {
1.3670 + _dbus_warn ("Did not receive a reply to %s %d on %p\n",
1.3671 + method, serial, connection);
1.3672 + goto out;
1.3673 + }
1.3674 +
1.3675 + verbose_message_received (connection, message);
1.3676 +
1.3677 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.3678 + {
1.3679 + if (dbus_message_is_error (message, DBUS_ERROR_NO_MEMORY))
1.3680 + {
1.3681 + ; /* good, this is a valid response */
1.3682 + }
1.3683 + else
1.3684 + {
1.3685 + warn_unexpected (connection, message, "not this error");
1.3686 +
1.3687 + goto out;
1.3688 + }
1.3689 + }
1.3690 + else
1.3691 + {
1.3692 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN)
1.3693 + {
1.3694 + ; /* good, expected */
1.3695 + }
1.3696 + else
1.3697 + {
1.3698 + warn_unexpected (connection, message,
1.3699 + "method_return for ListActivatableNames/ListNames");
1.3700 +
1.3701 + goto out;
1.3702 + }
1.3703 +
1.3704 + retry_get_property:
1.3705 +
1.3706 + if (!dbus_message_get_args (message, &error,
1.3707 + DBUS_TYPE_ARRAY,
1.3708 + DBUS_TYPE_STRING,
1.3709 + &srvs, &l,
1.3710 + DBUS_TYPE_INVALID))
1.3711 + {
1.3712 + if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1.3713 + {
1.3714 + _dbus_verbose ("no memory to list services by %s\n", method);
1.3715 + dbus_error_free (&error);
1.3716 + _dbus_wait_for_memory ();
1.3717 + goto retry_get_property;
1.3718 + }
1.3719 + else
1.3720 + {
1.3721 + _dbus_assert (dbus_error_is_set (&error));
1.3722 + _dbus_warn ("Did not get the expected DBUS_TYPE_ARRAY from %s\n", method);
1.3723 + goto out;
1.3724 + }
1.3725 + } else {
1.3726 + *services = srvs;
1.3727 + *len = l;
1.3728 + }
1.3729 + }
1.3730 +
1.3731 + if (!check_no_leftovers (context))
1.3732 + goto out;
1.3733 +
1.3734 + retval = TRUE;
1.3735 +
1.3736 + out:
1.3737 + dbus_error_free (&error);
1.3738 +
1.3739 + if (message)
1.3740 + dbus_message_unref (message);
1.3741 +
1.3742 + return retval;
1.3743 +}
1.3744 +
1.3745 +/* returns TRUE if the correct thing happens,
1.3746 + * but the correct thing may include OOM errors.
1.3747 + */
1.3748 +static dbus_bool_t
1.3749 +check_list_services (BusContext *context,
1.3750 + DBusConnection *connection)
1.3751 +{
1.3752 + DBusMessage *message;
1.3753 + DBusMessage *base_service_message;
1.3754 + const char *base_service;
1.3755 + dbus_uint32_t serial;
1.3756 + dbus_bool_t retval;
1.3757 + const char *existent = EXISTENT_SERVICE_NAME;
1.3758 + dbus_uint32_t flags;
1.3759 + char **services;
1.3760 + int len;
1.3761 +
1.3762 + _dbus_verbose ("check_list_services for %p\n", connection);
1.3763 +
1.3764 + if (!check_get_services (context, connection, "ListActivatableNames", &services, &len))
1.3765 + {
1.3766 + return TRUE;
1.3767 + }
1.3768 +
1.3769 + if (!_dbus_string_array_contains ((const char **)services, existent))
1.3770 + {
1.3771 + _dbus_warn ("Did not get the expected %s from ListActivatableNames\n", existent);
1.3772 + return FALSE;
1.3773 + }
1.3774 +
1.3775 + dbus_free_string_array (services);
1.3776 +
1.3777 + base_service_message = NULL;
1.3778 +
1.3779 + message = dbus_message_new_method_call (DBUS_SERVICE_DBUS,
1.3780 + DBUS_PATH_DBUS,
1.3781 + DBUS_INTERFACE_DBUS,
1.3782 + "StartServiceByName");
1.3783 +
1.3784 + if (message == NULL)
1.3785 + return TRUE;
1.3786 +
1.3787 + dbus_message_set_auto_start (message, FALSE);
1.3788 +
1.3789 + flags = 0;
1.3790 + if (!dbus_message_append_args (message,
1.3791 + DBUS_TYPE_STRING, &existent,
1.3792 + DBUS_TYPE_UINT32, &flags,
1.3793 + DBUS_TYPE_INVALID))
1.3794 + {
1.3795 + dbus_message_unref (message);
1.3796 + return TRUE;
1.3797 + }
1.3798 +
1.3799 + if (!dbus_connection_send (connection, message, &serial))
1.3800 + {
1.3801 + dbus_message_unref (message);
1.3802 + return TRUE;
1.3803 + }
1.3804 +
1.3805 + dbus_message_unref (message);
1.3806 + message = NULL;
1.3807 +
1.3808 + bus_test_run_everything (context);
1.3809 +
1.3810 + /* now wait for the message bus to hear back from the activated
1.3811 + * service.
1.3812 + */
1.3813 + block_connection_until_message_from_bus (context, connection, "activated service to connect");
1.3814 +
1.3815 + bus_test_run_everything (context);
1.3816 +
1.3817 + if (!dbus_connection_get_is_connected (connection))
1.3818 + {
1.3819 + _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__);
1.3820 + return TRUE;
1.3821 + }
1.3822 +
1.3823 + retval = FALSE;
1.3824 +
1.3825 + message = pop_message_waiting_for_memory (connection);
1.3826 + if (message == NULL)
1.3827 + {
1.3828 + _dbus_warn ("Did not receive any messages after %s %d on %p\n",
1.3829 + "StartServiceByName", serial, connection);
1.3830 + goto out;
1.3831 + }
1.3832 +
1.3833 + verbose_message_received (connection, message);
1.3834 + _dbus_verbose (" (after sending %s)\n", "StartServiceByName");
1.3835 +
1.3836 + if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR)
1.3837 + {
1.3838 + if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS))
1.3839 + {
1.3840 + _dbus_warn ("Message has wrong sender %s\n",
1.3841 + dbus_message_get_sender (message) ?
1.3842 + dbus_message_get_sender (message) : "(none)");
1.3843 + goto out;
1.3844 + }
1.3845 +
1.3846 + if (dbus_message_is_error (message,
1.3847 + DBUS_ERROR_NO_MEMORY))
1.3848 + {
1.3849 + ; /* good, this is a valid response */
1.3850 + }
1.3851 + else if (dbus_message_is_error (message,
1.3852 + DBUS_ERROR_SPAWN_CHILD_EXITED) ||
1.3853 + dbus_message_is_error (message,
1.3854 + DBUS_ERROR_SPAWN_CHILD_SIGNALED) ||
1.3855 + dbus_message_is_error (message,
1.3856 + DBUS_ERROR_SPAWN_EXEC_FAILED))
1.3857 + {
1.3858 + ; /* good, this is expected also */
1.3859 + }
1.3860 + else
1.3861 + {
1.3862 + _dbus_warn ("Did not expect error %s\n",
1.3863 + dbus_message_get_error_name (message));
1.3864 + goto out;
1.3865 + }
1.3866 + }
1.3867 + else
1.3868 + {
1.3869 + GotServiceInfo message_kind;
1.3870 +
1.3871 + if (!check_base_service_activated (context, connection,
1.3872 + message, &base_service))
1.3873 + goto out;
1.3874 +
1.3875 + base_service_message = message;
1.3876 + message = NULL;
1.3877 +
1.3878 + /* We may need to block here for the test service to exit or finish up */
1.3879 + block_connection_until_message_from_bus (context, connection, "test service to exit or finish up");
1.3880 +
1.3881 + message = dbus_connection_borrow_message (connection);
1.3882 + if (message == NULL)
1.3883 + {
1.3884 + _dbus_warn ("Did not receive any messages after base service creation notification\n");
1.3885 + goto out;
1.3886 + }
1.3887 +
1.3888 + message_kind = check_got_service_info (message);
1.3889 +
1.3890 + dbus_connection_return_message (connection, message);
1.3891 + message = NULL;
1.3892 +
1.3893 + switch (message_kind)
1.3894 + {
1.3895 + case GOT_SOMETHING_ELSE:
1.3896 + case GOT_ERROR:
1.3897 + case GOT_SERVICE_DELETED:
1.3898 + _dbus_warn ("Unexpected message after ActivateService "
1.3899 + "(should be an error or a service announcement)\n");
1.3900 + goto out;
1.3901 +
1.3902 + case GOT_SERVICE_CREATED:
1.3903 + message = pop_message_waiting_for_memory (connection);
1.3904 + if (message == NULL)
1.3905 + {
1.3906 + _dbus_warn ("Failed to pop message we just put back! "
1.3907 + "should have been a NameOwnerChanged (creation)\n");
1.3908 + goto out;
1.3909 + }
1.3910 +
1.3911 + if (!check_service_activated (context, connection, EXISTENT_SERVICE_NAME,
1.3912 + base_service, message))
1.3913 + goto out;
1.3914 +
1.3915 + dbus_message_unref (message);
1.3916 + message = NULL;
1.3917 +
1.3918 + if (!check_no_leftovers (context))
1.3919 + {
1.3920 + _dbus_warn ("Messages were left over after successful activation\n");
1.3921 + goto out;
1.3922 + }
1.3923 +
1.3924 + break;
1.3925 + }
1.3926 + }
1.3927 +
1.3928 + if (!check_get_services (context, connection, "ListNames", &services, &len))
1.3929 + {
1.3930 + return TRUE;
1.3931 + }
1.3932 +
1.3933 + if (!_dbus_string_array_contains ((const char **)services, existent))
1.3934 + {
1.3935 + _dbus_warn ("Did not get the expected %s from ListNames\n", existent);
1.3936 + goto out;
1.3937 + }
1.3938 +
1.3939 + dbus_free_string_array (services);
1.3940 +
1.3941 + if (!check_send_exit_to_service (context, connection,
1.3942 + EXISTENT_SERVICE_NAME, base_service))
1.3943 + goto out;
1.3944 +
1.3945 + retval = TRUE;
1.3946 +
1.3947 + out:
1.3948 + if (message)
1.3949 + dbus_message_unref (message);
1.3950 +
1.3951 + if (base_service_message)
1.3952 + dbus_message_unref (base_service_message);
1.3953 +
1.3954 + return retval;
1.3955 +}
1.3956 +
1.3957 +typedef struct
1.3958 +{
1.3959 + Check2Func func;
1.3960 + BusContext *context;
1.3961 + DBusConnection *connection;
1.3962 +} Check2Data;
1.3963 +
1.3964 +static dbus_bool_t
1.3965 +check_oom_check2_func (void *data)
1.3966 +{
1.3967 + Check2Data *d = data;
1.3968 +
1.3969 + if (! (* d->func) (d->context, d->connection))
1.3970 + return FALSE;
1.3971 +
1.3972 + if (!check_no_leftovers (d->context))
1.3973 + {
1.3974 + _dbus_warn ("Messages were left over, should be covered by test suite\n");
1.3975 + return FALSE;
1.3976 + }
1.3977 +
1.3978 + return TRUE;
1.3979 +}
1.3980 +
1.3981 +static void
1.3982 +check2_try_iterations (BusContext *context,
1.3983 + DBusConnection *connection,
1.3984 + const char *description,
1.3985 + Check2Func func)
1.3986 +{
1.3987 + Check2Data d;
1.3988 +
1.3989 + d.func = func;
1.3990 + d.context = context;
1.3991 + d.connection = connection;
1.3992 +
1.3993 + if (!_dbus_test_oom_handling (description, check_oom_check2_func,
1.3994 + &d))
1.3995 + {
1.3996 + _dbus_warn ("%s failed during oom\n", description);
1.3997 + _dbus_assert_not_reached ("test failed");
1.3998 + }
1.3999 +}
1.4000 +
1.4001 +dbus_bool_t
1.4002 +bus_dispatch_test (const DBusString *test_data_dir)
1.4003 +{
1.4004 + BusContext *context;
1.4005 + DBusConnection *foo;
1.4006 + DBusConnection *bar;
1.4007 + DBusConnection *baz;
1.4008 + DBusError error;
1.4009 +
1.4010 + dbus_error_init (&error);
1.4011 +
1.4012 + context = bus_context_new_test (test_data_dir,
1.4013 + "valid-config-files/debug-allow-all.conf");
1.4014 + if (context == NULL)
1.4015 + return FALSE;
1.4016 +#ifndef __SYMBIAN32__
1.4017 + foo = dbus_connection_open_private ("debug-pipe:name=test-server", &error);
1.4018 +#else
1.4019 + foo = dbus_connection_open_private ("tcp:host=localhost,port=12436", &error);
1.4020 +#endif
1.4021 + if (foo == NULL)
1.4022 + _dbus_assert_not_reached ("could not alloc connection");
1.4023 +
1.4024 + if (!bus_setup_debug_client (foo))
1.4025 + _dbus_assert_not_reached ("could not set up connection");
1.4026 +
1.4027 + spin_connection_until_authenticated (context, foo);
1.4028 +
1.4029 + if (!check_hello_message (context, foo))
1.4030 + _dbus_assert_not_reached ("hello message failed");
1.4031 +
1.4032 + if (!check_double_hello_message (context, foo))
1.4033 + _dbus_assert_not_reached ("double hello message failed");
1.4034 +
1.4035 + if (!check_add_match_all (context, foo))
1.4036 + _dbus_assert_not_reached ("AddMatch message failed");
1.4037 + #ifndef __SYMBIAN32__
1.4038 + bar = dbus_connection_open_private ("debug-pipe:name=test-server", &error);
1.4039 + #else
1.4040 + bar = dbus_connection_open_private ("tcp:host=localhost,port=12436", &error);
1.4041 +#endif
1.4042 + if (bar == NULL)
1.4043 + _dbus_assert_not_reached ("could not alloc connection");
1.4044 +
1.4045 + if (!bus_setup_debug_client (bar))
1.4046 + _dbus_assert_not_reached ("could not set up connection");
1.4047 +
1.4048 + spin_connection_until_authenticated (context, bar);
1.4049 +
1.4050 + if (!check_hello_message (context, bar))
1.4051 + _dbus_assert_not_reached ("hello message failed");
1.4052 +
1.4053 + if (!check_add_match_all (context, bar))
1.4054 + _dbus_assert_not_reached ("AddMatch message failed");
1.4055 +#ifndef __SYMBIAN32__
1.4056 + baz = dbus_connection_open_private ("debug-pipe:name=test-server", &error);
1.4057 + #else
1.4058 + baz = dbus_connection_open_private ("tcp:host=localhost,port=12436", &error);
1.4059 +#endif
1.4060 + if (baz == NULL)
1.4061 + _dbus_assert_not_reached ("could not alloc connection");
1.4062 +
1.4063 + if (!bus_setup_debug_client (baz))
1.4064 + _dbus_assert_not_reached ("could not set up connection");
1.4065 +
1.4066 + spin_connection_until_authenticated (context, baz);
1.4067 +
1.4068 + if (!check_hello_message (context, baz))
1.4069 + _dbus_assert_not_reached ("hello message failed");
1.4070 +
1.4071 + if (!check_add_match_all (context, baz))
1.4072 + _dbus_assert_not_reached ("AddMatch message failed");
1.4073 +
1.4074 + #ifndef __SYMBIAN32__
1.4075 + if (!check_get_connection_unix_user (context, baz))
1.4076 + _dbus_assert_not_reached ("GetConnectionUnixUser message failed");
1.4077 +
1.4078 + if (!check_get_connection_unix_process_id (context, baz))
1.4079 + _dbus_assert_not_reached ("GetConnectionUnixProcessID message failed");
1.4080 + #endif
1.4081 +
1.4082 + /* Arun changes : some of the services which are not supported*/
1.4083 + /*
1.4084 + if (!check_list_services (context, baz))
1.4085 + _dbus_assert_not_reached ("ListActivatableNames message failed");
1.4086 + */
1.4087 +
1.4088 + if (!check_no_leftovers (context))
1.4089 + {
1.4090 + _dbus_warn ("Messages were left over after setting up initial connections\n");
1.4091 + _dbus_assert_not_reached ("initial connection setup failed");
1.4092 + }
1.4093 +
1.4094 + check1_try_iterations (context, "create_and_hello",
1.4095 + check_hello_connection);
1.4096 +
1.4097 + /*Arun changes : some of the services which are not supported*/
1.4098 + /*
1.4099 + check2_try_iterations (context, foo, "nonexistent_service_no_auto_start",
1.4100 + check_nonexistent_service_no_auto_start);
1.4101 + */
1.4102 +
1.4103 + check2_try_iterations (context, foo, "segfault_service_no_auto_start",
1.4104 + check_segfault_service_no_auto_start);
1.4105 +
1.4106 +// shilps check2_try_iterations (context, foo, "existent_service_no_auto_start",
1.4107 + // check_existent_service_no_auto_start);
1.4108 +
1.4109 + /*Arun changes : some of the services which are not supported*/
1.4110 + /*
1.4111 + check2_try_iterations (context, foo, "nonexistent_service_auto_start",
1.4112 + check_nonexistent_service_auto_start);
1.4113 +
1.4114 + check2_try_iterations (context, foo, "segfault_service_auto_start",
1.4115 + check_segfault_service_auto_start);
1.4116 + */
1.4117 +
1.4118 + check2_try_iterations (context, foo, "shell_fail_service_auto_start",
1.4119 + check_shell_fail_service_auto_start);
1.4120 +
1.4121 +#if 0
1.4122 + /* Note: need to resolve some issues with the testing code in order to run
1.4123 + * this in oom (handle that we sometimes don't get replies back from the bus
1.4124 + * when oom happens, without blocking the test).
1.4125 + */
1.4126 + check2_try_iterations (context, foo, "existent_service_auto_auto_start",
1.4127 + check_existent_service_auto_start);
1.4128 +#else
1.4129 +// shilps if (!check_existent_service_auto_start (context, foo))
1.4130 + // _dbus_assert_not_reached ("existent service auto start failed");
1.4131 +#endif
1.4132 +
1.4133 +
1.4134 + // shilps if (!check_shell_service_success_auto_start (context, foo))
1.4135 + // _dbus_assert_not_reached ("shell success service auto start failed");
1.4136 +
1.4137 + _dbus_verbose ("Disconnecting foo, bar, and baz\n");
1.4138 +
1.4139 + kill_client_connection_unchecked (foo);
1.4140 + kill_client_connection_unchecked (bar);
1.4141 + kill_client_connection_unchecked (baz);
1.4142 +
1.4143 + bus_context_unref (context);
1.4144 +
1.4145 + return TRUE;
1.4146 +}
1.4147 +
1.4148 +dbus_bool_t
1.4149 +bus_dispatch_sha1_test (const DBusString *test_data_dir)
1.4150 +{
1.4151 + BusContext *context;
1.4152 + DBusConnection *foo;
1.4153 + DBusError error;
1.4154 +
1.4155 + dbus_error_init (&error);
1.4156 +
1.4157 + /* Test SHA1 authentication */
1.4158 + _dbus_verbose ("Testing SHA1 context\n");
1.4159 +
1.4160 + context = bus_context_new_test (test_data_dir,
1.4161 + "valid-config-files/debug-allow-all-sha1.conf");
1.4162 + if (context == NULL)
1.4163 + return FALSE;
1.4164 +#ifndef __SYMBIAN32__
1.4165 + foo = dbus_connection_open_private ("debug-pipe:name=test-server", &error);
1.4166 +#else
1.4167 + foo = dbus_connection_open_private ("tcp:host=localhost,port=12435", &error);
1.4168 +#endif
1.4169 + if (foo == NULL)
1.4170 + _dbus_assert_not_reached ("could not alloc connection");
1.4171 +
1.4172 + if (!bus_setup_debug_client (foo))
1.4173 + _dbus_assert_not_reached ("could not set up connection");
1.4174 +
1.4175 + spin_connection_until_authenticated (context, foo);
1.4176 +
1.4177 + if (!check_hello_message (context, foo))
1.4178 + _dbus_assert_not_reached ("hello message failed");
1.4179 +
1.4180 + if (!check_add_match_all (context, foo))
1.4181 + _dbus_assert_not_reached ("addmatch message failed");
1.4182 +
1.4183 + if (!check_no_leftovers (context))
1.4184 + {
1.4185 + _dbus_warn ("Messages were left over after setting up initial SHA-1 connection\n");
1.4186 + _dbus_assert_not_reached ("initial connection setup failed");
1.4187 + }
1.4188 +
1.4189 + check1_try_iterations (context, "create_and_hello_sha1",
1.4190 + check_hello_connection);
1.4191 +
1.4192 + kill_client_connection_unchecked (foo);
1.4193 +
1.4194 + bus_context_unref (context);
1.4195 +
1.4196 + return TRUE;
1.4197 +}
1.4198 +
1.4199 +#endif /* DBUS_BUILD_TESTS */