sl@0: /* -*- mode: C; c-file-style: "gnu" -*- */ sl@0: /* dispatch.c Message dispatcher sl@0: * sl@0: * Copyright (C) 2003 CodeFactory AB sl@0: * Copyright (C) 2003, 2004, 2005 Red Hat, Inc. sl@0: * Copyright (C) 2004 Imendio HB sl@0: * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved. sl@0: * Licensed under the Academic Free License version 2.1 sl@0: * sl@0: * This program is free software; you can redistribute it and/or modify sl@0: * it under the terms of the GNU General Public License as published by sl@0: * the Free Software Foundation; either version 2 of the License, or sl@0: * (at your option) any later version. sl@0: * sl@0: * This program is distributed in the hope that it will be useful, sl@0: * but WITHOUT ANY WARRANTY; without even the implied warranty of sl@0: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the sl@0: * GNU General Public License for more details. sl@0: * sl@0: * You should have received a copy of the GNU General Public License sl@0: * along with this program; if not, write to the Free Software sl@0: * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA sl@0: * sl@0: */ sl@0: sl@0: #include "dispatch.h" sl@0: #include "connection.h" sl@0: #include "driver.h" sl@0: #include "services.h" sl@0: #include "activation.h" sl@0: #include "utils.h" sl@0: #include "bus.h" sl@0: #include "signals.h" sl@0: #include "test.h" sl@0: #ifndef __SYMBIAN32__ sl@0: #include sl@0: #else sl@0: #include "dbus-internals.h" sl@0: #endif //__SYMBIAN32__ sl@0: #include sl@0: sl@0: #ifdef __SYMBIAN32__ sl@0: #include "config.h" sl@0: #endif //__SYMBIAN32__ sl@0: sl@0: static dbus_bool_t sl@0: send_one_message (DBusConnection *connection, sl@0: BusContext *context, sl@0: DBusConnection *sender, sl@0: DBusConnection *addressed_recipient, sl@0: DBusMessage *message, sl@0: BusTransaction *transaction, sl@0: DBusError *error) sl@0: { sl@0: if (!bus_context_check_security_policy (context, transaction, sl@0: sender, sl@0: addressed_recipient, sl@0: connection, sl@0: message, sl@0: NULL)) sl@0: return TRUE; /* silently don't send it */ sl@0: sl@0: if (!bus_transaction_send (transaction, sl@0: connection, sl@0: message)) sl@0: { sl@0: BUS_SET_OOM (error); sl@0: return FALSE; sl@0: } sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_bool_t sl@0: bus_dispatch_matches (BusTransaction *transaction, sl@0: DBusConnection *sender, sl@0: DBusConnection *addressed_recipient, sl@0: DBusMessage *message, sl@0: DBusError *error) sl@0: { sl@0: DBusError tmp_error; sl@0: BusConnections *connections; sl@0: DBusList *recipients; sl@0: BusMatchmaker *matchmaker; sl@0: DBusList *link; sl@0: BusContext *context; sl@0: sl@0: _DBUS_ASSERT_ERROR_IS_CLEAR (error); sl@0: sl@0: /* sender and recipient can both be NULL for the bus driver, sl@0: * or for signals with no particular recipient sl@0: */ sl@0: sl@0: _dbus_assert (sender == NULL || bus_connection_is_active (sender)); sl@0: _dbus_assert (dbus_message_get_sender (message) != NULL); sl@0: sl@0: connections = bus_transaction_get_connections (transaction); sl@0: sl@0: dbus_error_init (&tmp_error); sl@0: context = bus_transaction_get_context (transaction); sl@0: matchmaker = bus_context_get_matchmaker (context); sl@0: sl@0: recipients = NULL; sl@0: if (!bus_matchmaker_get_recipients (matchmaker, connections, sl@0: sender, addressed_recipient, message, sl@0: &recipients)) sl@0: { sl@0: BUS_SET_OOM (error); sl@0: return FALSE; sl@0: } sl@0: sl@0: link = _dbus_list_get_first_link (&recipients); sl@0: while (link != NULL) sl@0: { sl@0: DBusConnection *dest; sl@0: sl@0: dest = link->data; sl@0: sl@0: if (!send_one_message (dest, context, sender, addressed_recipient, sl@0: message, transaction, &tmp_error)) sl@0: break; sl@0: sl@0: link = _dbus_list_get_next_link (&recipients, link); sl@0: } sl@0: sl@0: _dbus_list_clear (&recipients); sl@0: sl@0: if (dbus_error_is_set (&tmp_error)) sl@0: { sl@0: dbus_move_error (&tmp_error, error); sl@0: return FALSE; sl@0: } sl@0: else sl@0: return TRUE; sl@0: } sl@0: sl@0: static DBusHandlerResult sl@0: bus_dispatch (DBusConnection *connection, sl@0: DBusMessage *message) sl@0: { sl@0: const char *sender, *service_name; sl@0: DBusError error; sl@0: BusTransaction *transaction; sl@0: BusContext *context; sl@0: DBusHandlerResult result; sl@0: DBusConnection *addressed_recipient; sl@0: sl@0: result = DBUS_HANDLER_RESULT_HANDLED; sl@0: sl@0: transaction = NULL; sl@0: addressed_recipient = NULL; sl@0: dbus_error_init (&error); sl@0: sl@0: context = bus_connection_get_context (connection); sl@0: _dbus_assert (context != NULL); sl@0: sl@0: /* If we can't even allocate an OOM error, we just go to sleep sl@0: * until we can. sl@0: */ sl@0: while (!bus_connection_preallocate_oom_error (connection)) sl@0: _dbus_wait_for_memory (); sl@0: sl@0: /* Ref connection in case we disconnect it at some point in here */ sl@0: dbus_connection_ref (connection); sl@0: sl@0: service_name = dbus_message_get_destination (message); sl@0: sl@0: #ifdef DBUS_ENABLE_VERBOSE_MODE sl@0: { sl@0: const char *interface_name, *member_name, *error_name; sl@0: sl@0: interface_name = dbus_message_get_interface (message); sl@0: member_name = dbus_message_get_member (message); sl@0: error_name = dbus_message_get_error_name (message); sl@0: sl@0: _dbus_verbose ("DISPATCH: %s %s %s to %s\n", sl@0: interface_name ? interface_name : "(no interface)", sl@0: member_name ? member_name : "(no member)", sl@0: error_name ? error_name : "(no error name)", sl@0: service_name ? service_name : "peer"); sl@0: } sl@0: #endif /* DBUS_ENABLE_VERBOSE_MODE */ sl@0: sl@0: /* If service_name is NULL, if it's a signal we send it to all sl@0: * connections with a match rule. If it's not a signal, there sl@0: * are some special cases here but mostly we just bail out. sl@0: */ sl@0: if (service_name == NULL) sl@0: { sl@0: if (dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_LOCAL, sl@0: "Disconnected")) sl@0: { sl@0: bus_connection_disconnected (connection); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_SIGNAL) sl@0: { sl@0: /* DBusConnection also handles some of these automatically, we leave sl@0: * it to do so. sl@0: */ sl@0: result = DBUS_HANDLER_RESULT_NOT_YET_HANDLED; sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: /* Create our transaction */ sl@0: transaction = bus_transaction_new (context); sl@0: if (transaction == NULL) sl@0: { sl@0: BUS_SET_OOM (&error); sl@0: goto out; sl@0: } sl@0: sl@0: /* Assign a sender to the message */ sl@0: if (bus_connection_is_active (connection)) sl@0: { sl@0: sender = bus_connection_get_name (connection); sl@0: _dbus_assert (sender != NULL); sl@0: sl@0: if (!dbus_message_set_sender (message, sender)) sl@0: { sl@0: BUS_SET_OOM (&error); sl@0: goto out; sl@0: } sl@0: sl@0: /* We need to refetch the service name here, because sl@0: * dbus_message_set_sender can cause the header to be sl@0: * reallocated, and thus the service_name pointer will become sl@0: * invalid. sl@0: */ sl@0: service_name = dbus_message_get_destination (message); sl@0: } sl@0: sl@0: if (service_name && sl@0: strcmp (service_name, DBUS_SERVICE_DBUS) == 0) /* to bus driver */ sl@0: { sl@0: if (!bus_context_check_security_policy (context, transaction, sl@0: connection, NULL, NULL, message, &error)) sl@0: { sl@0: _dbus_verbose ("Security policy rejected message\n"); sl@0: goto out; sl@0: } sl@0: sl@0: _dbus_verbose ("Giving message to %s\n", DBUS_SERVICE_DBUS); sl@0: if (!bus_driver_handle_message (connection, transaction, message, &error)) sl@0: goto out; sl@0: } sl@0: else if (!bus_connection_is_active (connection)) /* clients must talk to bus driver first */ sl@0: { sl@0: _dbus_verbose ("Received message from non-registered client. Disconnecting.\n"); sl@0: dbus_connection_close (connection); sl@0: goto out; sl@0: } sl@0: else if (service_name != NULL) /* route to named service */ sl@0: { sl@0: DBusString service_string; sl@0: BusService *service; sl@0: BusRegistry *registry; sl@0: sl@0: _dbus_assert (service_name != NULL); sl@0: sl@0: registry = bus_connection_get_registry (connection); sl@0: sl@0: _dbus_string_init_const (&service_string, service_name); sl@0: service = bus_registry_lookup (registry, &service_string); sl@0: sl@0: if (service == NULL && dbus_message_get_auto_start (message)) sl@0: { sl@0: BusActivation *activation; sl@0: /* We can't do the security policy check here, since the addressed sl@0: * recipient service doesn't exist yet. We do it before sending the sl@0: * message after the service has been created. sl@0: */ sl@0: activation = bus_connection_get_activation (connection); sl@0: sl@0: if (!bus_activation_activate_service (activation, connection, transaction, TRUE, sl@0: message, service_name, &error)) sl@0: { sl@0: _DBUS_ASSERT_ERROR_IS_SET (&error); sl@0: _dbus_verbose ("bus_activation_activate_service() failed: %s\n", error.name); sl@0: goto out; sl@0: } sl@0: sl@0: goto out; sl@0: } sl@0: else if (service == NULL) sl@0: { sl@0: dbus_set_error (&error, sl@0: DBUS_ERROR_NAME_HAS_NO_OWNER, sl@0: "Name \"%s\" does not exist", sl@0: service_name); sl@0: goto out; sl@0: } sl@0: else sl@0: { sl@0: addressed_recipient = bus_service_get_primary_owners_connection (service); sl@0: _dbus_assert (addressed_recipient != NULL); sl@0: sl@0: if (!bus_context_check_security_policy (context, transaction, sl@0: connection, addressed_recipient, sl@0: addressed_recipient, sl@0: message, &error)) sl@0: goto out; sl@0: sl@0: /* Dispatch the message */ sl@0: if (!bus_transaction_send (transaction, addressed_recipient, message)) sl@0: { sl@0: BUS_SET_OOM (&error); sl@0: goto out; sl@0: } sl@0: } sl@0: } sl@0: sl@0: /* Now match the messages against any match rules, which will send sl@0: * out signals and such. addressed_recipient may == NULL. sl@0: */ sl@0: if (!bus_dispatch_matches (transaction, connection, addressed_recipient, message, &error)) sl@0: goto out; sl@0: sl@0: out: sl@0: if (dbus_error_is_set (&error)) sl@0: { sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: /* If we disconnected it, we won't bother to send it any error sl@0: * messages. sl@0: */ sl@0: _dbus_verbose ("Not sending error to connection we disconnected\n"); sl@0: } sl@0: else if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: bus_connection_send_oom_error (connection, message); sl@0: sl@0: /* cancel transaction due to OOM */ sl@0: if (transaction != NULL) sl@0: { sl@0: bus_transaction_cancel_and_free (transaction); sl@0: transaction = NULL; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: /* Try to send the real error, if no mem to do that, send sl@0: * the OOM error sl@0: */ sl@0: _dbus_assert (transaction != NULL); sl@0: if (!bus_transaction_send_error_reply (transaction, connection, sl@0: &error, message)) sl@0: { sl@0: bus_connection_send_oom_error (connection, message); sl@0: sl@0: /* cancel transaction due to OOM */ sl@0: if (transaction != NULL) sl@0: { sl@0: bus_transaction_cancel_and_free (transaction); sl@0: transaction = NULL; sl@0: } sl@0: } sl@0: } sl@0: sl@0: sl@0: dbus_error_free (&error); sl@0: } sl@0: sl@0: if (transaction != NULL) sl@0: { sl@0: bus_transaction_execute_and_free (transaction); sl@0: } sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return result; sl@0: } sl@0: sl@0: static DBusHandlerResult sl@0: bus_dispatch_message_filter (DBusConnection *connection, sl@0: DBusMessage *message, sl@0: void *user_data) sl@0: { sl@0: return bus_dispatch (connection, message); sl@0: } sl@0: sl@0: dbus_bool_t sl@0: bus_dispatch_add_connection (DBusConnection *connection) sl@0: { sl@0: if (!dbus_connection_add_filter (connection, sl@0: bus_dispatch_message_filter, sl@0: NULL, NULL)) sl@0: return FALSE; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: void sl@0: bus_dispatch_remove_connection (DBusConnection *connection) sl@0: { sl@0: /* Here we tell the bus driver that we want to get off. */ sl@0: bus_driver_remove_connection (connection); sl@0: sl@0: dbus_connection_remove_filter (connection, sl@0: bus_dispatch_message_filter, sl@0: NULL); sl@0: } sl@0: sl@0: #ifdef DBUS_BUILD_TESTS sl@0: sl@0: #include sl@0: sl@0: /* This is used to know whether we need to block in order to finish sl@0: * sending a message, or whether the initial dbus_connection_send() sl@0: * already flushed the queue. sl@0: */ sl@0: #define SEND_PENDING(connection) (dbus_connection_has_messages_to_send (connection)) sl@0: sl@0: typedef dbus_bool_t (* Check1Func) (BusContext *context); sl@0: typedef dbus_bool_t (* Check2Func) (BusContext *context, sl@0: DBusConnection *connection); sl@0: sl@0: static dbus_bool_t check_no_leftovers (BusContext *context); sl@0: sl@0: static void sl@0: block_connection_until_message_from_bus (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *what_is_expected) sl@0: { sl@0: _dbus_verbose ("expecting: %s\n", what_is_expected); sl@0: sl@0: while (dbus_connection_get_dispatch_status (connection) == sl@0: DBUS_DISPATCH_COMPLETE && sl@0: dbus_connection_get_is_connected (connection)) sl@0: { sl@0: #ifndef __SYMBIAN32__ sl@0: bus_test_run_bus_loop (context, TRUE); sl@0: #else //_dbus_loop_iterate() blocks indefinitly on setting argument to TRUE on Symbian,it must be investigated sl@0: bus_test_run_bus_loop (context, FALSE); sl@0: #endif sl@0: bus_test_run_clients_loop (FALSE); sl@0: } sl@0: } sl@0: sl@0: static void sl@0: spin_connection_until_authenticated (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: _dbus_verbose ("Spinning to auth connection %p\n", connection); sl@0: while (!dbus_connection_get_is_authenticated (connection) && sl@0: dbus_connection_get_is_connected (connection)) sl@0: { sl@0: bus_test_run_bus_loop (context, FALSE); sl@0: bus_test_run_clients_loop (FALSE); sl@0: } sl@0: _dbus_verbose (" ... done spinning to auth connection %p\n", connection); sl@0: } sl@0: sl@0: /* compensate for fact that pop_message() can return #NULL due to OOM */ sl@0: static DBusMessage* sl@0: pop_message_waiting_for_memory (DBusConnection *connection) sl@0: { sl@0: while (dbus_connection_get_dispatch_status (connection) == sl@0: DBUS_DISPATCH_NEED_MEMORY) sl@0: _dbus_wait_for_memory (); sl@0: sl@0: return dbus_connection_pop_message (connection); sl@0: } sl@0: sl@0: static DBusMessage* sl@0: borrow_message_waiting_for_memory (DBusConnection *connection) sl@0: { sl@0: while (dbus_connection_get_dispatch_status (connection) == sl@0: DBUS_DISPATCH_NEED_MEMORY) sl@0: _dbus_wait_for_memory (); sl@0: sl@0: return dbus_connection_borrow_message (connection); sl@0: } sl@0: sl@0: static void sl@0: warn_unexpected_real (DBusConnection *connection, sl@0: DBusMessage *message, sl@0: const char *expected, sl@0: const char *function, sl@0: int line) sl@0: { sl@0: if (message) sl@0: _dbus_warn ("%s:%d received message interface \"%s\" member \"%s\" error name \"%s\" on %p, expecting %s\n", sl@0: function, line, sl@0: dbus_message_get_interface (message) ? sl@0: dbus_message_get_interface (message) : "(unset)", sl@0: dbus_message_get_member (message) ? sl@0: dbus_message_get_member (message) : "(unset)", sl@0: dbus_message_get_error_name (message) ? sl@0: dbus_message_get_error_name (message) : "(unset)", sl@0: connection, sl@0: expected); sl@0: else sl@0: _dbus_warn ("%s:%d received no message on %p, expecting %s\n", sl@0: function, line, connection, expected); sl@0: } sl@0: sl@0: #define warn_unexpected(connection, message, expected) \ sl@0: warn_unexpected_real (connection, message, expected, _DBUS_FUNCTION_NAME, __LINE__) sl@0: sl@0: static void sl@0: verbose_message_received (DBusConnection *connection, sl@0: DBusMessage *message) sl@0: { sl@0: _dbus_verbose ("Received message interface \"%s\" member \"%s\" error name \"%s\" on %p\n", sl@0: dbus_message_get_interface (message) ? sl@0: dbus_message_get_interface (message) : "(unset)", sl@0: dbus_message_get_member (message) ? sl@0: dbus_message_get_member (message) : "(unset)", sl@0: dbus_message_get_error_name (message) ? sl@0: dbus_message_get_error_name (message) : "(unset)", sl@0: connection); sl@0: } sl@0: sl@0: typedef enum sl@0: { sl@0: SERVICE_CREATED, sl@0: OWNER_CHANGED, sl@0: SERVICE_DELETED sl@0: } ServiceInfoKind; sl@0: sl@0: typedef struct sl@0: { sl@0: ServiceInfoKind expected_kind; sl@0: const char *expected_service_name; sl@0: dbus_bool_t failed; sl@0: DBusConnection *skip_connection; sl@0: } CheckServiceOwnerChangedData; sl@0: sl@0: static dbus_bool_t sl@0: check_service_owner_changed_foreach (DBusConnection *connection, sl@0: void *data) sl@0: { sl@0: CheckServiceOwnerChangedData *d = data; sl@0: DBusMessage *message; sl@0: DBusError error; sl@0: const char *service_name, *old_owner, *new_owner; sl@0: sl@0: if (d->expected_kind == SERVICE_CREATED sl@0: && connection == d->skip_connection) sl@0: return TRUE; sl@0: sl@0: dbus_error_init (&error); sl@0: d->failed = TRUE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a message on %p, expecting %s\n", sl@0: connection, "NameOwnerChanged"); sl@0: goto out; sl@0: } sl@0: else if (!dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_DBUS, sl@0: "NameOwnerChanged")) sl@0: { sl@0: warn_unexpected (connection, message, "NameOwnerChanged"); sl@0: sl@0: goto out; sl@0: } sl@0: else sl@0: { sl@0: reget_service_info_data: sl@0: service_name = NULL; sl@0: old_owner = NULL; sl@0: new_owner = NULL; sl@0: sl@0: dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_STRING, &service_name, sl@0: DBUS_TYPE_STRING, &old_owner, sl@0: DBUS_TYPE_STRING, &new_owner, sl@0: DBUS_TYPE_INVALID); sl@0: sl@0: if (dbus_error_is_set (&error)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto reget_service_info_data; sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not get the expected arguments\n"); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: if ((d->expected_kind == SERVICE_CREATED && ( old_owner[0] || !new_owner[0])) sl@0: || (d->expected_kind == OWNER_CHANGED && (!old_owner[0] || !new_owner[0])) sl@0: || (d->expected_kind == SERVICE_DELETED && (!old_owner[0] || new_owner[0]))) sl@0: { sl@0: _dbus_warn ("inconsistent NameOwnerChanged arguments\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp (service_name, d->expected_service_name) != 0) sl@0: { sl@0: _dbus_warn ("expected info on service %s, got info on %s\n", sl@0: d->expected_service_name, sl@0: service_name); sl@0: goto out; sl@0: } sl@0: sl@0: if (*service_name == ':' && new_owner[0] sl@0: && strcmp (service_name, new_owner) != 0) sl@0: { sl@0: _dbus_warn ("inconsistent ServiceOwnedChanged message (\"%s\" [ %s -> %s ])\n", sl@0: service_name, old_owner, new_owner); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: d->failed = FALSE; sl@0: sl@0: out: sl@0: dbus_error_free (&error); sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return !d->failed; sl@0: } sl@0: sl@0: sl@0: static void sl@0: kill_client_connection (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: char *base_service; sl@0: const char *s; sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: _dbus_verbose ("killing connection %p\n", connection); sl@0: sl@0: s = dbus_bus_get_unique_name (connection); sl@0: _dbus_assert (s != NULL); sl@0: sl@0: while ((base_service = _dbus_strdup (s)) == NULL) sl@0: _dbus_wait_for_memory (); sl@0: sl@0: dbus_connection_ref (connection); sl@0: sl@0: /* kick in the disconnect handler that unrefs the connection */ sl@0: dbus_connection_close (connection); sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: _dbus_assert (bus_test_client_listed (connection)); sl@0: sl@0: /* Run disconnect handler in test.c */ sl@0: if (bus_connection_dispatch_one_message (connection)) sl@0: _dbus_assert_not_reached ("something received on connection being killed other than the disconnect"); sl@0: sl@0: _dbus_assert (!dbus_connection_get_is_connected (connection)); sl@0: dbus_connection_unref (connection); sl@0: connection = NULL; sl@0: _dbus_assert (!bus_test_client_listed (connection)); sl@0: sl@0: socd.expected_kind = SERVICE_DELETED; sl@0: socd.expected_service_name = base_service; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = NULL; sl@0: sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: dbus_free (base_service); sl@0: sl@0: if (socd.failed) sl@0: _dbus_assert_not_reached ("didn't get the expected NameOwnerChanged (deletion) messages"); sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: _dbus_assert_not_reached ("stuff left in message queues after disconnecting a client"); sl@0: } sl@0: sl@0: static void sl@0: kill_client_connection_unchecked (DBusConnection *connection) sl@0: { sl@0: /* This kills the connection without expecting it to affect sl@0: * the rest of the bus. sl@0: */ sl@0: _dbus_verbose ("Unchecked kill of connection %p\n", connection); sl@0: sl@0: dbus_connection_ref (connection); sl@0: dbus_connection_close (connection); sl@0: /* dispatching disconnect handler will unref once */ sl@0: if (bus_connection_dispatch_one_message (connection)) sl@0: _dbus_assert_not_reached ("message other than disconnect dispatched after failure to register"); sl@0: sl@0: _dbus_assert (!bus_test_client_listed (connection)); sl@0: dbus_connection_unref (connection); sl@0: } sl@0: sl@0: typedef struct sl@0: { sl@0: dbus_bool_t failed; sl@0: } CheckNoMessagesData; sl@0: sl@0: static dbus_bool_t sl@0: check_no_messages_foreach (DBusConnection *connection, sl@0: void *data) sl@0: { sl@0: CheckNoMessagesData *d = data; sl@0: DBusMessage *message; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message != NULL) sl@0: { sl@0: warn_unexpected (connection, message, "no messages"); sl@0: sl@0: d->failed = TRUE; sl@0: } sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: return !d->failed; sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_no_leftovers (BusContext *context) sl@0: { sl@0: CheckNoMessagesData nmd; sl@0: sl@0: nmd.failed = FALSE; sl@0: bus_test_clients_foreach (check_no_messages_foreach, sl@0: &nmd); sl@0: sl@0: if (nmd.failed) sl@0: { sl@0: _dbus_verbose ("%s: leftover message found\n", sl@0: _DBUS_FUNCTION_NAME); sl@0: return FALSE; sl@0: } sl@0: else sl@0: return TRUE; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_hello_message (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: DBusMessage *name_message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: const char *name; sl@0: const char *acquired; sl@0: sl@0: retval = FALSE; sl@0: dbus_error_init (&error); sl@0: name = NULL; sl@0: acquired = NULL; sl@0: message = NULL; sl@0: name_message = NULL; sl@0: sl@0: _dbus_verbose ("check_hello_message for %p\n", connection); sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "Hello"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: dbus_connection_ref (connection); /* because we may get disconnected */ sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: dbus_connection_unref (connection); sl@0: return TRUE; sl@0: } sl@0: sl@0: _dbus_assert (dbus_message_has_signature (message, "")); sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected (presumably auth failed)\n"); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: /* send our message */ sl@0: bus_test_run_clients_loop (SEND_PENDING (connection)); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected (presumably auth failed)\n"); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: block_connection_until_message_from_bus (context, connection, "reply to Hello"); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected (presumably auth failed)\n"); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "Hello", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: ; /* good, expected */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "method return for Hello"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: retry_get_hello_name: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_STRING, &name, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: _dbus_verbose ("no memory to get service name arg from hello\n"); sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto retry_get_hello_name; sl@0: } sl@0: else sl@0: { sl@0: _dbus_assert (dbus_error_is_set (&error)); sl@0: _dbus_warn ("Did not get the expected single string argument to hello\n"); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: _dbus_verbose ("Got hello name: %s\n", name); sl@0: sl@0: while (!dbus_bus_set_unique_name (connection, name)) sl@0: _dbus_wait_for_memory (); sl@0: sl@0: socd.expected_kind = SERVICE_CREATED; sl@0: socd.expected_service_name = name; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = connection; /* we haven't done AddMatch so won't get it ourselves */ sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: name_message = message; sl@0: /* Client should also have gotten ServiceAcquired */ sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Expecting %s, got nothing\n", sl@0: "NameAcquired"); sl@0: goto out; sl@0: } sl@0: if (! dbus_message_is_signal (message, DBUS_INTERFACE_DBUS, sl@0: "NameAcquired")) sl@0: { sl@0: _dbus_warn ("Expecting %s, got smthg else\n", sl@0: "NameAcquired"); sl@0: goto out; sl@0: } sl@0: sl@0: retry_get_acquired_name: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_STRING, &acquired, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: _dbus_verbose ("no memory to get service name arg from acquired\n"); sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto retry_get_acquired_name; sl@0: } sl@0: else sl@0: { sl@0: _dbus_assert (dbus_error_is_set (&error)); sl@0: _dbus_warn ("Did not get the expected single string argument to ServiceAcquired\n"); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: _dbus_verbose ("Got acquired name: %s\n", acquired); sl@0: sl@0: if (strcmp (acquired, name) != 0) sl@0: { sl@0: _dbus_warn ("Acquired name is %s but expected %s\n", sl@0: acquired, name); sl@0: goto out; sl@0: } sl@0: acquired = NULL; sl@0: } sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: _dbus_verbose ("ending %s retval = %d\n", _DBUS_FUNCTION_NAME, retval); sl@0: sl@0: dbus_error_free (&error); sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: if (name_message) sl@0: dbus_message_unref (name_message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_double_hello_message (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: sl@0: retval = FALSE; sl@0: dbus_error_init (&error); sl@0: message = NULL; sl@0: sl@0: _dbus_verbose ("check_double_hello_message for %p\n", connection); sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "Hello"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: /* send our message */ sl@0: bus_test_run_clients_loop (SEND_PENDING (connection)); sl@0: sl@0: dbus_connection_ref (connection); /* because we may get disconnected */ sl@0: block_connection_until_message_from_bus (context, connection, "reply to Hello"); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "Hello", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: warn_unexpected (connection, message, "method return for Hello"); sl@0: goto out; sl@0: } sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: dbus_error_free (&error); sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_get_connection_unix_user (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: const char *base_service_name; sl@0: dbus_uint32_t uid; sl@0: sl@0: retval = FALSE; sl@0: dbus_error_init (&error); sl@0: message = NULL; sl@0: sl@0: _dbus_verbose ("check_get_connection_unix_user for %p\n", connection); sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "GetConnectionUnixUser"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: base_service_name = dbus_bus_get_unique_name (connection); sl@0: sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &base_service_name, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: /* send our message */ sl@0: bus_test_run_clients_loop (SEND_PENDING (connection)); sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: dbus_connection_ref (connection); /* because we may get disconnected */ sl@0: block_connection_until_message_from_bus (context, connection, "reply to GetConnectionUnixUser"); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "GetConnectionUnixUser", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (dbus_message_is_error (message, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: ; /* good, expected */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, sl@0: "method_return for GetConnectionUnixUser"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: retry_get_property: sl@0: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_UINT32, &uid, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: _dbus_verbose ("no memory to get uid by GetConnectionUnixUser\n"); sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto retry_get_property; sl@0: } sl@0: else sl@0: { sl@0: _dbus_assert (dbus_error_is_set (&error)); sl@0: _dbus_warn ("Did not get the expected DBUS_TYPE_UINT32 from GetConnectionUnixUser\n"); sl@0: goto out; sl@0: } sl@0: } sl@0: } sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: dbus_error_free (&error); sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_get_connection_unix_process_id (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: const char *base_service_name; sl@0: dbus_uint32_t pid; sl@0: sl@0: retval = FALSE; sl@0: dbus_error_init (&error); sl@0: message = NULL; sl@0: sl@0: _dbus_verbose ("check_get_connection_unix_process_id for %p\n", connection); sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "GetConnectionUnixProcessID"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: base_service_name = dbus_bus_get_unique_name (connection); sl@0: sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &base_service_name, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: /* send our message */ sl@0: bus_test_run_clients_loop (SEND_PENDING (connection)); sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: dbus_connection_ref (connection); /* because we may get disconnected */ sl@0: block_connection_until_message_from_bus (context, connection, "reply to GetConnectionUnixProcessID"); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "GetConnectionUnixProcessID", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (dbus_message_is_error (message, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: ; /* good, expected */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, sl@0: "method_return for GetConnectionUnixProcessID"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: retry_get_property: sl@0: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_UINT32, &pid, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: _dbus_verbose ("no memory to get pid by GetConnectionUnixProcessID\n"); sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto retry_get_property; sl@0: } sl@0: else sl@0: { sl@0: _dbus_assert (dbus_error_is_set (&error)); sl@0: _dbus_warn ("Did not get the expected DBUS_TYPE_UINT32 from GetConnectionUnixProcessID\n"); sl@0: goto out; sl@0: } sl@0: } else { sl@0: sl@0: /* test if returned pid is the same as our own pid sl@0: * sl@0: * @todo It would probably be good to restructure the tests sl@0: * in a way so our parent is the bus that we're testing sl@0: * cause then we can test that the pid returned matches sl@0: * getppid() sl@0: */ sl@0: if (pid != (dbus_uint32_t) _dbus_getpid ()) sl@0: { sl@0: _dbus_assert (dbus_error_is_set (&error)); sl@0: _dbus_warn ("Result from GetConnectionUnixProcessID is not our own pid\n"); sl@0: goto out; sl@0: } sl@0: } sl@0: } sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: dbus_error_free (&error); sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_add_match_all (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_bool_t retval; sl@0: dbus_uint32_t serial; sl@0: DBusError error; sl@0: const char *empty = ""; sl@0: sl@0: retval = FALSE; sl@0: dbus_error_init (&error); sl@0: message = NULL; sl@0: sl@0: _dbus_verbose ("check_add_match_all for %p\n", connection); sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "AddMatch"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: /* empty string match rule matches everything */ sl@0: if (!dbus_message_append_args (message, DBUS_TYPE_STRING, &empty, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: dbus_connection_ref (connection); /* because we may get disconnected */ sl@0: sl@0: /* send our message */ sl@0: bus_test_run_clients_loop (SEND_PENDING (connection)); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: block_connection_until_message_from_bus (context, connection, "reply to AddMatch"); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "AddMatch", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: ; /* good, expected */ sl@0: _dbus_assert (dbus_message_get_reply_serial (message) == serial); sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "method return for AddMatch"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: dbus_error_free (&error); sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_hello_connection (BusContext *context) sl@0: { sl@0: DBusConnection *connection; sl@0: DBusError error; sl@0: sl@0: dbus_error_init (&error); sl@0: sl@0: connection = dbus_connection_open_private ("debug-pipe:name=test-server", &error); sl@0: if (connection == NULL) sl@0: { sl@0: _DBUS_ASSERT_ERROR_IS_SET (&error); sl@0: dbus_error_free (&error); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!bus_setup_debug_client (connection)) sl@0: { sl@0: dbus_connection_close (connection); sl@0: dbus_connection_unref (connection); sl@0: return TRUE; sl@0: } sl@0: sl@0: spin_connection_until_authenticated (context, connection); sl@0: sl@0: if (!check_hello_message (context, connection)) sl@0: return FALSE; sl@0: sl@0: if (dbus_bus_get_unique_name (connection) == NULL) sl@0: { sl@0: /* We didn't successfully register, so we can't sl@0: * do the usual kill_client_connection() checks sl@0: */ sl@0: kill_client_connection_unchecked (connection); sl@0: } sl@0: else sl@0: { sl@0: if (!check_add_match_all (context, connection)) sl@0: return FALSE; sl@0: sl@0: kill_client_connection (context, connection); sl@0: } sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: #define NONEXISTENT_SERVICE_NAME "test.this.service.does.not.exist.ewuoiurjdfxcvn" sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_nonexistent_service_no_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: const char *nonexistent = NONEXISTENT_SERVICE_NAME; sl@0: dbus_uint32_t flags; sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "StartServiceByName"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: dbus_message_set_auto_start (message, FALSE); sl@0: sl@0: flags = 0; sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &nonexistent, sl@0: DBUS_TYPE_UINT32, &flags, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: block_connection_until_message_from_bus (context, connection, "reply to ActivateService on nonexistent"); sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "StartServiceByName", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else if (dbus_message_is_error (message, sl@0: DBUS_ERROR_SERVICE_UNKNOWN)) sl@0: { sl@0: ; /* good, this is expected also */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not expect to successfully activate %s\n", sl@0: NONEXISTENT_SERVICE_NAME); sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_nonexistent_service_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: sl@0: message = dbus_message_new_method_call (NONEXISTENT_SERVICE_NAME, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.TestSuite", sl@0: "Echo"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: block_connection_until_message_from_bus (context, connection, "reply to Echo"); sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "Echo message (auto activation)", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else if (dbus_message_is_error (message, sl@0: DBUS_ERROR_SERVICE_UNKNOWN)) sl@0: { sl@0: ; /* good, this is expected also */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not expect to successfully activate %s\n", sl@0: NONEXISTENT_SERVICE_NAME); sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_base_service_activated (BusContext *context, sl@0: DBusConnection *connection, sl@0: DBusMessage *initial_message, sl@0: const char **base_service_p) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: const char *base_service, *base_service_from_bus, *old_owner; sl@0: sl@0: retval = FALSE; sl@0: sl@0: dbus_error_init (&error); sl@0: base_service = NULL; sl@0: old_owner = NULL; sl@0: base_service_from_bus = NULL; sl@0: sl@0: message = initial_message; sl@0: dbus_message_ref (message); sl@0: sl@0: if (dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_DBUS, sl@0: "NameOwnerChanged")) sl@0: { sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: reget_service_name_arg: sl@0: base_service = NULL; sl@0: old_owner = NULL; sl@0: base_service_from_bus = NULL; sl@0: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_STRING, &base_service, sl@0: DBUS_TYPE_STRING, &old_owner, sl@0: DBUS_TYPE_STRING, &base_service_from_bus, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto reget_service_name_arg; sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Message %s doesn't have a service name: %s\n", sl@0: "NameOwnerChanged (creation)", sl@0: error.message); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: if (*base_service != ':') sl@0: { sl@0: _dbus_warn ("Expected base service activation, got \"%s\" instead\n", sl@0: base_service); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp (base_service, base_service_from_bus) != 0) sl@0: { sl@0: _dbus_warn ("Expected base service activation, got \"%s\" instead with owner \"%s\"\n", sl@0: base_service, base_service_from_bus); sl@0: goto out; sl@0: } sl@0: sl@0: if (old_owner[0]) sl@0: { sl@0: _dbus_warn ("Received an old_owner argument during base service activation, \"%s\"\n", sl@0: old_owner); sl@0: goto out; sl@0: } sl@0: sl@0: socd.expected_kind = SERVICE_CREATED; sl@0: socd.expected_service_name = base_service; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = connection; sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "NameOwnerChanged (creation) for base service"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: if (base_service_p) sl@0: *base_service_p = base_service; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: dbus_error_free (&error); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_service_activated (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *activated_name, sl@0: const char *base_service_name, sl@0: DBusMessage *initial_message) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: dbus_uint32_t activation_result; sl@0: sl@0: retval = FALSE; sl@0: sl@0: dbus_error_init (&error); sl@0: sl@0: message = initial_message; sl@0: dbus_message_ref (message); sl@0: sl@0: if (dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_DBUS, sl@0: "NameOwnerChanged")) sl@0: { sl@0: CheckServiceOwnerChangedData socd; sl@0: const char *service_name, *base_service_from_bus, *old_owner; sl@0: sl@0: reget_service_name_arg: sl@0: service_name = NULL; sl@0: old_owner = NULL; sl@0: base_service_from_bus = NULL; sl@0: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_STRING, &service_name, sl@0: DBUS_TYPE_STRING, &old_owner, sl@0: DBUS_TYPE_STRING, &base_service_from_bus, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto reget_service_name_arg; sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Message %s doesn't have a service name: %s\n", sl@0: "NameOwnerChanged (creation)", sl@0: error.message); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: if (strcmp (service_name, activated_name) != 0) sl@0: { sl@0: _dbus_warn ("Expected to see service %s created, saw %s instead\n", sl@0: activated_name, service_name); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp (base_service_name, base_service_from_bus) != 0) sl@0: { sl@0: _dbus_warn ("NameOwnerChanged reports wrong base service: %s owner, expected %s instead\n", sl@0: base_service_from_bus, base_service_name); sl@0: goto out; sl@0: } sl@0: sl@0: if (old_owner[0]) sl@0: { sl@0: _dbus_warn ("expected a %s, got a %s\n", sl@0: "NameOwnerChanged (creation)", sl@0: "NameOwnerChanged (change)"); sl@0: goto out; sl@0: } sl@0: sl@0: socd.expected_kind = SERVICE_CREATED; sl@0: socd.skip_connection = connection; sl@0: socd.failed = FALSE; sl@0: socd.expected_service_name = service_name; sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: dbus_message_unref (message); sl@0: service_name = NULL; sl@0: old_owner = NULL; sl@0: base_service_from_bus = NULL; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Expected a reply to %s, got nothing\n", sl@0: "StartServiceByName"); sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "NameOwnerChanged for the activated name"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: warn_unexpected (connection, message, "reply to StartServiceByName"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: activation_result = 0; sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_UINT32, &activation_result, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (!dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: _dbus_warn ("Did not have activation result first argument to %s: %s\n", sl@0: "StartServiceByName", error.message); sl@0: goto out; sl@0: } sl@0: sl@0: dbus_error_free (&error); sl@0: } sl@0: else sl@0: { sl@0: if (activation_result == DBUS_START_REPLY_SUCCESS) sl@0: ; /* Good */ sl@0: else if (activation_result == DBUS_START_REPLY_ALREADY_RUNNING) sl@0: ; /* Good also */ sl@0: else sl@0: { sl@0: _dbus_warn ("Activation result was %u, no good.\n", sl@0: activation_result); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: { sl@0: _dbus_warn ("Messages were left over after verifying existent activation results\n"); sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: dbus_error_free (&error); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_service_auto_activated (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *activated_name, sl@0: const char *base_service_name, sl@0: DBusMessage *initial_message) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: sl@0: retval = FALSE; sl@0: sl@0: dbus_error_init (&error); sl@0: sl@0: message = initial_message; sl@0: dbus_message_ref (message); sl@0: sl@0: if (dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_DBUS, sl@0: "NameOwnerChanged")) sl@0: { sl@0: const char *service_name; sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: reget_service_name_arg: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_STRING, &service_name, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto reget_service_name_arg; sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Message %s doesn't have a service name: %s\n", sl@0: "NameOwnerChanged", sl@0: error.message); sl@0: dbus_error_free (&error); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: if (strcmp (service_name, activated_name) != 0) sl@0: { sl@0: _dbus_warn ("Expected to see service %s created, saw %s instead\n", sl@0: activated_name, service_name); sl@0: goto out; sl@0: } sl@0: sl@0: socd.expected_kind = SERVICE_CREATED; sl@0: socd.expected_service_name = service_name; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = connection; sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: /* Note that this differs from regular activation in that we don't get a sl@0: * reply to ActivateService here. sl@0: */ sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: service_name = NULL; sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "NameOwnerChanged for the activated name"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_service_deactivated (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *activated_name, sl@0: const char *base_service) sl@0: { sl@0: dbus_bool_t retval; sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: retval = FALSE; sl@0: sl@0: /* Now we are expecting ServiceOwnerChanged (deletion) messages for the base sl@0: * service and the activated_name. The base service sl@0: * notification is required to come last. sl@0: */ sl@0: socd.expected_kind = SERVICE_DELETED; sl@0: socd.expected_service_name = activated_name; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = NULL; sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: socd.expected_kind = SERVICE_DELETED; sl@0: socd.expected_service_name = base_service; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = NULL; sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: return retval; sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_send_exit_to_service (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *service_name, sl@0: const char *base_service) sl@0: { sl@0: dbus_bool_t got_error; sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: sl@0: _dbus_verbose ("Sending exit message to the test service\n"); sl@0: sl@0: retval = FALSE; sl@0: sl@0: /* Kill off the test service by sending it a quit message */ sl@0: message = dbus_message_new_method_call (service_name, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.TestSuite", sl@0: "Exit"); sl@0: sl@0: if (message == NULL) sl@0: { sl@0: /* Do this again; we still need the service to exit... */ sl@0: if (!check_send_exit_to_service (context, connection, sl@0: service_name, base_service)) sl@0: goto out; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: sl@0: /* Do this again; we still need the service to exit... */ sl@0: if (!check_send_exit_to_service (context, connection, sl@0: service_name, base_service)) sl@0: goto out; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: /* send message */ sl@0: bus_test_run_clients_loop (SEND_PENDING (connection)); sl@0: sl@0: /* read it in and write it out to test service */ sl@0: bus_test_run_bus_loop (context, FALSE); sl@0: sl@0: /* see if we got an error during message bus dispatching */ sl@0: bus_test_run_clients_loop (FALSE); sl@0: message = borrow_message_waiting_for_memory (connection); sl@0: got_error = message != NULL && dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR; sl@0: if (message) sl@0: { sl@0: dbus_connection_return_message (connection, message); sl@0: message = NULL; sl@0: } sl@0: sl@0: if (!got_error) sl@0: { sl@0: /* If no error, wait for the test service to exit */ sl@0: block_connection_until_message_from_bus (context, connection, "test service to exit"); sl@0: sl@0: bus_test_run_everything (context); sl@0: } sl@0: sl@0: if (got_error) sl@0: { sl@0: message = pop_message_waiting_for_memory (connection); sl@0: _dbus_assert (message != NULL); sl@0: sl@0: if (dbus_message_get_reply_serial (message) != serial) sl@0: { sl@0: warn_unexpected (connection, message, sl@0: "error with the correct reply serial"); sl@0: goto out; sl@0: } sl@0: sl@0: if (!dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: warn_unexpected (connection, message, sl@0: "a no memory error from asking test service to exit"); sl@0: goto out; sl@0: } sl@0: sl@0: _dbus_verbose ("Got error %s when asking test service to exit\n", sl@0: dbus_message_get_error_name (message)); sl@0: sl@0: /* Do this again; we still need the service to exit... */ sl@0: if (!check_send_exit_to_service (context, connection, sl@0: service_name, base_service)) sl@0: goto out; sl@0: } sl@0: else sl@0: { sl@0: if (!check_service_deactivated (context, connection, sl@0: service_name, base_service)) sl@0: goto out; sl@0: sl@0: /* Should now have a NoReply error from the Exit() method sl@0: * call; it should have come after all the deactivation sl@0: * stuff. sl@0: */ sl@0: message = pop_message_waiting_for_memory (connection); sl@0: sl@0: if (message == NULL) sl@0: { sl@0: warn_unexpected (connection, NULL, sl@0: "reply to Exit() method call"); sl@0: goto out; sl@0: } sl@0: if (!dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_REPLY)) sl@0: { sl@0: warn_unexpected (connection, message, sl@0: "NoReply error from Exit() method call"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_reply_serial (message) != serial) sl@0: { sl@0: warn_unexpected (connection, message, sl@0: "error with the correct reply serial"); sl@0: goto out; sl@0: } sl@0: sl@0: _dbus_verbose ("Got error %s after test service exited\n", sl@0: dbus_message_get_error_name (message)); sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: { sl@0: _dbus_warn ("Messages were left over after %s\n", sl@0: _DBUS_FUNCTION_NAME); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_got_error (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *first_error_name, sl@0: ...) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_bool_t retval; sl@0: va_list ap; sl@0: dbus_bool_t error_found; sl@0: const char *error_name; sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not get an expected error\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: warn_unexpected (connection, message, "an error"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: error_found = FALSE; sl@0: sl@0: va_start (ap, first_error_name); sl@0: error_name = first_error_name; sl@0: while (error_name != NULL) sl@0: { sl@0: if (dbus_message_is_error (message, error_name)) sl@0: { sl@0: error_found = TRUE; sl@0: break; sl@0: } sl@0: error_name = va_arg (ap, char*); sl@0: } sl@0: va_end (ap); sl@0: sl@0: if (!error_found) sl@0: { sl@0: _dbus_warn ("Expected error %s or other, got %s instead\n", sl@0: first_error_name, sl@0: dbus_message_get_error_name (message)); sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: typedef enum sl@0: { sl@0: GOT_SERVICE_CREATED, sl@0: GOT_SERVICE_DELETED, sl@0: GOT_ERROR, sl@0: GOT_SOMETHING_ELSE sl@0: } GotServiceInfo; sl@0: sl@0: static GotServiceInfo sl@0: check_got_service_info (DBusMessage *message) sl@0: { sl@0: GotServiceInfo message_kind; sl@0: sl@0: if (dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_DBUS, sl@0: "NameOwnerChanged")) sl@0: { sl@0: DBusError error; sl@0: const char *service_name, *old_owner, *new_owner; sl@0: dbus_error_init (&error); sl@0: sl@0: reget_service_info_data: sl@0: service_name = NULL; sl@0: old_owner = NULL; sl@0: new_owner = NULL; sl@0: sl@0: dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_STRING, &service_name, sl@0: DBUS_TYPE_STRING, &old_owner, sl@0: DBUS_TYPE_STRING, &new_owner, sl@0: DBUS_TYPE_INVALID); sl@0: if (dbus_error_is_set (&error)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: dbus_error_free (&error); sl@0: goto reget_service_info_data; sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("unexpected arguments for NameOwnerChanged message\n"); sl@0: message_kind = GOT_SOMETHING_ELSE; sl@0: } sl@0: } sl@0: else if (!old_owner[0]) sl@0: message_kind = GOT_SERVICE_CREATED; sl@0: else if (!new_owner[0]) sl@0: message_kind = GOT_SERVICE_DELETED; sl@0: else sl@0: message_kind = GOT_SOMETHING_ELSE; sl@0: sl@0: dbus_error_free (&error); sl@0: } sl@0: else if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: message_kind = GOT_ERROR; sl@0: else sl@0: message_kind = GOT_SOMETHING_ELSE; sl@0: sl@0: return message_kind; sl@0: } sl@0: sl@0: #define EXISTENT_SERVICE_NAME "org.freedesktop.DBus.TestSuiteEchoService" sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_existent_service_no_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: DBusMessage *base_service_message; sl@0: const char *base_service; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: const char *existent = EXISTENT_SERVICE_NAME; sl@0: dbus_uint32_t flags; sl@0: sl@0: base_service_message = NULL; sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "StartServiceByName"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: dbus_message_set_auto_start (message, FALSE); sl@0: sl@0: flags = 0; sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &existent, sl@0: DBUS_TYPE_UINT32, &flags, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: /* now wait for the message bus to hear back from the activated sl@0: * service. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "activated service to connect"); sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive any messages after %s %d on %p\n", sl@0: "StartServiceByName", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: _dbus_verbose (" (after sending %s)\n", "StartServiceByName"); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else if (dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_CHILD_EXITED) || sl@0: dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_CHILD_SIGNALED) || sl@0: dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_EXEC_FAILED)) sl@0: { sl@0: ; /* good, this is expected also */ sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not expect error %s\n", sl@0: dbus_message_get_error_name (message)); sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: GotServiceInfo message_kind; sl@0: sl@0: if (!check_base_service_activated (context, connection, sl@0: message, &base_service)) sl@0: goto out; sl@0: sl@0: base_service_message = message; sl@0: message = NULL; sl@0: sl@0: /* We may need to block here for the test service to exit or finish up */ sl@0: block_connection_until_message_from_bus (context, connection, "test service to exit or finish up"); sl@0: sl@0: message = dbus_connection_borrow_message (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive any messages after base service creation notification\n"); sl@0: goto out; sl@0: } sl@0: sl@0: message_kind = check_got_service_info (message); sl@0: sl@0: dbus_connection_return_message (connection, message); sl@0: message = NULL; sl@0: sl@0: switch (message_kind) sl@0: { sl@0: case GOT_SOMETHING_ELSE: sl@0: _dbus_warn ("Unexpected message after ActivateService " sl@0: "(should be an error or a service announcement"); sl@0: goto out; sl@0: sl@0: case GOT_ERROR: sl@0: if (!check_got_error (context, connection, sl@0: DBUS_ERROR_SPAWN_CHILD_EXITED, sl@0: DBUS_ERROR_NO_MEMORY, sl@0: NULL)) sl@0: goto out; sl@0: /* A service deleted should be coming along now after this error. sl@0: * We can also get the error *after* the service deleted. sl@0: */ sl@0: sl@0: /* fall through */ sl@0: sl@0: case GOT_SERVICE_DELETED: sl@0: { sl@0: /* The service started up and got a base address, but then sl@0: * failed to register under EXISTENT_SERVICE_NAME sl@0: */ sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: socd.expected_kind = SERVICE_DELETED; sl@0: socd.expected_service_name = base_service; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = NULL; sl@0: sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: /* Now we should get an error about the service exiting sl@0: * if we didn't get it before. sl@0: */ sl@0: if (message_kind != GOT_ERROR) sl@0: { sl@0: block_connection_until_message_from_bus (context, connection, "error about service exiting"); sl@0: sl@0: /* and process everything again */ sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!check_got_error (context, connection, sl@0: DBUS_ERROR_SPAWN_CHILD_EXITED, sl@0: DBUS_ERROR_NO_MEMORY, sl@0: NULL)) sl@0: goto out; sl@0: } sl@0: break; sl@0: } sl@0: sl@0: case GOT_SERVICE_CREATED: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message we just put back! " sl@0: "should have been a NameOwnerChanged (creation)\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (!check_service_activated (context, connection, EXISTENT_SERVICE_NAME, sl@0: base_service, message)) sl@0: goto out; sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: { sl@0: _dbus_warn ("Messages were left over after successful activation\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (!check_send_exit_to_service (context, connection, sl@0: EXISTENT_SERVICE_NAME, base_service)) sl@0: goto out; sl@0: sl@0: break; sl@0: } sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: if (base_service_message) sl@0: dbus_message_unref (base_service_message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_segfault_service_no_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: const char *segv_service; sl@0: dbus_uint32_t flags; sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "StartServiceByName"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: dbus_message_set_auto_start (message, FALSE); sl@0: sl@0: segv_service = "org.freedesktop.DBus.TestSuiteSegfaultService"; sl@0: flags = 0; sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &segv_service, sl@0: DBUS_TYPE_UINT32, &flags, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: block_connection_until_message_from_bus (context, connection, "reply to activating segfault service"); sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "StartServiceByName", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else if (dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_CHILD_SIGNALED)) sl@0: { sl@0: ; /* good, this is expected also */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not expect to successfully activate segfault service\n"); sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_segfault_service_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: sl@0: message = dbus_message_new_method_call ("org.freedesktop.DBus.TestSuiteSegfaultService", sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.TestSuite", sl@0: "Echo"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: block_connection_until_message_from_bus (context, connection, "reply to Echo on segfault service"); sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "Echo message (auto activation)", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else if (dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_CHILD_SIGNALED)) sl@0: { sl@0: ; /* good, this is expected also */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not expect to successfully activate segfault service\n"); sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: #define TEST_ECHO_MESSAGE "Test echo message" sl@0: #define TEST_RUN_HELLO_FROM_SELF_MESSAGE "Test sending message to self" sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_existent_hello_from_self (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: const char *text; sl@0: sl@0: message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.TestSuite", sl@0: "RunHelloFromSelf"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: text = TEST_RUN_HELLO_FROM_SELF_MESSAGE; sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &text, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: /* Note: if this test is run in OOM mode, it will block when the bus sl@0: * doesn't send a reply due to OOM. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "reply from running hello from self"); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message! Should have been reply from RunHelloFromSelf message\n"); sl@0: return FALSE; sl@0: } sl@0: sl@0: if (dbus_message_get_reply_serial (message) != serial) sl@0: { sl@0: _dbus_warn ("Wrong reply serial\n"); sl@0: dbus_message_unref (message); sl@0: return FALSE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_existent_ping (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.DBus.Peer", sl@0: "Ping"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: /* Note: if this test is run in OOM mode, it will block when the bus sl@0: * doesn't send a reply due to OOM. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "reply from running Ping"); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message! Should have been reply from Ping message\n"); sl@0: return FALSE; sl@0: } sl@0: sl@0: if (dbus_message_get_reply_serial (message) != serial) sl@0: { sl@0: _dbus_warn ("Wrong reply serial\n"); sl@0: dbus_message_unref (message); sl@0: return FALSE; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: _dbus_warn ("Unexpected message return during Ping\n"); sl@0: dbus_message_unref (message); sl@0: return FALSE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_existent_get_machine_id (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: const char *machine_id; sl@0: sl@0: message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.DBus.Peer", sl@0: "GetMachineId"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: /* Note: if this test is run in OOM mode, it will block when the bus sl@0: * doesn't send a reply due to OOM. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "reply from running GetMachineId"); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message! Should have been reply from GetMachineId message\n"); sl@0: return FALSE; sl@0: } sl@0: sl@0: if (dbus_message_get_reply_serial (message) != serial) sl@0: { sl@0: _dbus_warn ("Wrong reply serial\n"); sl@0: dbus_message_unref (message); sl@0: return FALSE; sl@0: } sl@0: sl@0: if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: _dbus_warn ("Unexpected message return during GetMachineId\n"); sl@0: dbus_message_unref (message); sl@0: return FALSE; sl@0: } sl@0: sl@0: machine_id = NULL; sl@0: if (!dbus_message_get_args (message, NULL, DBUS_TYPE_STRING, &machine_id, DBUS_TYPE_INVALID)) sl@0: { sl@0: _dbus_warn ("Did not get a machine ID in reply to GetMachineId\n"); sl@0: dbus_message_unref (message); sl@0: return FALSE; sl@0: } sl@0: sl@0: if (machine_id == NULL || strlen (machine_id) != 32) sl@0: { sl@0: _dbus_warn ("Machine id looks bogus: '%s'\n", machine_id ? machine_id : "null"); sl@0: dbus_message_unref (message); sl@0: return FALSE; sl@0: } sl@0: sl@0: /* We can't check that the machine id is correct because during make check it is sl@0: * just made up for each process separately sl@0: */ sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_existent_service_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: DBusMessage *base_service_message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: const char *base_service; sl@0: const char *text; sl@0: sl@0: base_service_message = NULL; sl@0: sl@0: message = dbus_message_new_method_call (EXISTENT_SERVICE_NAME, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.TestSuite", sl@0: "Echo"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: text = TEST_ECHO_MESSAGE; sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &text, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: /* now wait for the message bus to hear back from the activated sl@0: * service. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "reply to Echo on existent service"); sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive any messages after auto start %d on %p\n", sl@0: serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: _dbus_verbose (" (after sending %s)\n", "auto start"); sl@0: sl@0: /* we should get zero or two ServiceOwnerChanged signals */ sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_SIGNAL) sl@0: { sl@0: GotServiceInfo message_kind; sl@0: sl@0: if (!check_base_service_activated (context, connection, sl@0: message, &base_service)) sl@0: goto out; sl@0: sl@0: base_service_message = message; sl@0: message = NULL; sl@0: sl@0: /* We may need to block here for the test service to exit or finish up */ sl@0: block_connection_until_message_from_bus (context, connection, "service to exit"); sl@0: sl@0: /* Should get a service creation notification for the activated sl@0: * service name, or a service deletion on the base service name sl@0: */ sl@0: message = dbus_connection_borrow_message (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("No message after auto activation " sl@0: "(should be a service announcement)\n"); sl@0: dbus_connection_return_message (connection, message); sl@0: message = NULL; sl@0: goto out; sl@0: } sl@0: sl@0: message_kind = check_got_service_info (message); sl@0: sl@0: dbus_connection_return_message (connection, message); sl@0: message = NULL; sl@0: sl@0: switch (message_kind) sl@0: { sl@0: case GOT_SERVICE_CREATED: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message we just put back! " sl@0: "should have been a NameOwnerChanged (creation)\n"); sl@0: goto out; sl@0: } sl@0: sl@0: /* Check that ServiceOwnerChanged (creation) was correctly received */ sl@0: if (!check_service_auto_activated (context, connection, EXISTENT_SERVICE_NAME, sl@0: base_service, message)) sl@0: goto out; sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: break; sl@0: sl@0: case GOT_SERVICE_DELETED: sl@0: { sl@0: /* The service started up and got a base address, but then sl@0: * failed to register under EXISTENT_SERVICE_NAME sl@0: */ sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: socd.expected_kind = SERVICE_DELETED; sl@0: socd.expected_service_name = base_service; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = NULL; sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: break; sl@0: } sl@0: sl@0: case GOT_ERROR: sl@0: case GOT_SOMETHING_ELSE: sl@0: _dbus_warn ("Unexpected message after auto activation\n"); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: /* OK, now we've dealt with ServiceOwnerChanged signals, now should sl@0: * come the method reply (or error) from the initial method call sl@0: */ sl@0: sl@0: /* Note: if this test is run in OOM mode, it will block when the bus sl@0: * doesn't send a reply due to OOM. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "reply from echo message after auto-activation"); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message! Should have been reply from echo message\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_reply_serial (message) != serial) sl@0: { sl@0: _dbus_warn ("Wrong reply serial\n"); sl@0: goto out; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: if (!check_existent_ping (context, connection)) sl@0: goto out; sl@0: sl@0: if (!check_existent_get_machine_id (context, connection)) sl@0: goto out; sl@0: sl@0: if (!check_existent_hello_from_self (context, connection)) sl@0: goto out; sl@0: sl@0: if (!check_send_exit_to_service (context, connection, sl@0: EXISTENT_SERVICE_NAME, sl@0: base_service)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: if (base_service_message) sl@0: dbus_message_unref (base_service_message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: #define SHELL_FAIL_SERVICE_NAME "org.freedesktop.DBus.TestSuiteShellEchoServiceFail" sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_shell_fail_service_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: sl@0: message = dbus_message_new_method_call (SHELL_FAIL_SERVICE_NAME, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.TestSuite", sl@0: "Echo"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: block_connection_until_message_from_bus (context, connection, "reply to shell Echo on service which should fail to auto-start"); sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: "Echo message (auto activation)", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else if (dbus_message_is_error (message, sl@0: DBUS_ERROR_INVALID_ARGS)) sl@0: { sl@0: _dbus_verbose("got invalid args\n"); sl@0: ; /* good, this is expected also */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not expect to successfully auto-start shell fail service\n"); sl@0: goto out; sl@0: } sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: #define SHELL_SUCCESS_SERVICE_NAME "org.freedesktop.DBus.TestSuiteShellEchoServiceSuccess" sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_shell_service_success_auto_start (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: DBusMessage *base_service_message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: const char *base_service; sl@0: const char *argv[7] = {NULL, NULL, NULL, NULL, NULL, NULL, NULL}; sl@0: sl@0: base_service_message = NULL; sl@0: sl@0: message = dbus_message_new_method_call (SHELL_SUCCESS_SERVICE_NAME, sl@0: "/org/freedesktop/TestSuite", sl@0: "org.freedesktop.TestSuite", sl@0: "Echo"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: /* now wait for the message bus to hear back from the activated sl@0: * service. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "reply to Echo on shell success service"); sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive any messages after auto start %d on %p\n", sl@0: serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: _dbus_verbose (" (after sending %s)\n", "auto start"); sl@0: sl@0: /* we should get zero or two ServiceOwnerChanged signals */ sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_SIGNAL) sl@0: { sl@0: GotServiceInfo message_kind; sl@0: sl@0: if (!check_base_service_activated (context, connection, sl@0: message, &base_service)) sl@0: goto out; sl@0: sl@0: base_service_message = message; sl@0: message = NULL; sl@0: sl@0: /* We may need to block here for the test service to exit or finish up */ sl@0: block_connection_until_message_from_bus (context, connection, "service to exit"); sl@0: sl@0: /* Should get a service creation notification for the activated sl@0: * service name, or a service deletion on the base service name sl@0: */ sl@0: message = dbus_connection_borrow_message (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("No message after auto activation " sl@0: "(should be a service announcement)\n"); sl@0: dbus_connection_return_message (connection, message); sl@0: message = NULL; sl@0: goto out; sl@0: } sl@0: sl@0: message_kind = check_got_service_info (message); sl@0: sl@0: dbus_connection_return_message (connection, message); sl@0: message = NULL; sl@0: sl@0: switch (message_kind) sl@0: { sl@0: case GOT_SERVICE_CREATED: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message we just put back! " sl@0: "should have been a NameOwnerChanged (creation)\n"); sl@0: goto out; sl@0: } sl@0: sl@0: /* Check that ServiceOwnerChanged (creation) was correctly received */ sl@0: if (!check_service_auto_activated (context, connection, SHELL_SUCCESS_SERVICE_NAME, sl@0: base_service, message)) sl@0: goto out; sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: break; sl@0: sl@0: case GOT_SERVICE_DELETED: sl@0: { sl@0: /* The service started up and got a base address, but then sl@0: * failed to register under SHELL_SUCCESS_SERVICE_NAME sl@0: */ sl@0: CheckServiceOwnerChangedData socd; sl@0: sl@0: socd.expected_kind = SERVICE_DELETED; sl@0: socd.expected_service_name = base_service; sl@0: socd.failed = FALSE; sl@0: socd.skip_connection = NULL; sl@0: bus_test_clients_foreach (check_service_owner_changed_foreach, sl@0: &socd); sl@0: sl@0: if (socd.failed) sl@0: goto out; sl@0: sl@0: break; sl@0: } sl@0: sl@0: case GOT_ERROR: sl@0: case GOT_SOMETHING_ELSE: sl@0: _dbus_warn ("Unexpected message after auto activation\n"); sl@0: goto out; sl@0: } sl@0: } sl@0: sl@0: /* OK, now we've dealt with ServiceOwnerChanged signals, now should sl@0: * come the method reply (or error) from the initial method call sl@0: */ sl@0: sl@0: /* Note: if this test is run in OOM mode, it will block when the bus sl@0: * doesn't send a reply due to OOM. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "reply from echo message after auto-activation"); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message! Should have been reply from echo message\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_get_reply_serial (message) != serial) sl@0: { sl@0: _dbus_warn ("Wrong reply serial\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (!dbus_message_get_args (message, NULL, sl@0: DBUS_TYPE_STRING, &argv[0], sl@0: DBUS_TYPE_STRING, &argv[1], sl@0: DBUS_TYPE_STRING, &argv[2], sl@0: DBUS_TYPE_STRING, &argv[3], sl@0: DBUS_TYPE_STRING, &argv[4], sl@0: DBUS_TYPE_STRING, &argv[5], sl@0: DBUS_TYPE_STRING, &argv[6], sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: _dbus_warn ("Error getting arguments from return\n"); sl@0: goto out; sl@0: } sl@0: sl@0: /* don't worry about arg[0] as it may be different sl@0: depending on the path to the tests sl@0: */ sl@0: if (strcmp("-test", argv[1]) != 0) sl@0: { sl@0: _dbus_warn ("Unexpected argv[1] in shell success service test (expected: %s, got: %s)\n", sl@0: "-test", argv[1]); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp("that", argv[2]) != 0) sl@0: { sl@0: _dbus_warn ("Unexpected argv[2] in shell success service test (expected: %s, got: %s)\n", sl@0: "that", argv[2]); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp("we get", argv[3]) != 0) sl@0: { sl@0: _dbus_warn ("Unexpected argv[3] in shell success service test (expected: %s, got: %s)\n", sl@0: "we get", argv[3]); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp("back", argv[4]) != 0) sl@0: { sl@0: _dbus_warn ("Unexpected argv[4] in shell success service test (expected: %s, got: %s)\n", sl@0: "back", argv[4]); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp("--what", argv[5]) != 0) sl@0: { sl@0: _dbus_warn ("Unexpected argv[5] in shell success service test (expected: %s, got: %s)\n", sl@0: "--what", argv[5]); sl@0: goto out; sl@0: } sl@0: sl@0: if (strcmp("we put in", argv[6]) != 0) sl@0: { sl@0: _dbus_warn ("Unexpected argv[6] in shell success service test (expected: %s, got: %s)\n", sl@0: "we put in", argv[6]); sl@0: goto out; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: if (!check_send_exit_to_service (context, connection, sl@0: SHELL_SUCCESS_SERVICE_NAME, sl@0: base_service)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: if (base_service_message) sl@0: dbus_message_unref (base_service_message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: typedef struct sl@0: { sl@0: Check1Func func; sl@0: BusContext *context; sl@0: } Check1Data; sl@0: sl@0: static dbus_bool_t sl@0: check_oom_check1_func (void *data) sl@0: { sl@0: Check1Data *d = data; sl@0: sl@0: if (! (* d->func) (d->context)) sl@0: return FALSE; sl@0: sl@0: if (!check_no_leftovers (d->context)) sl@0: { sl@0: _dbus_warn ("Messages were left over, should be covered by test suite\n"); sl@0: return FALSE; sl@0: } sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: static void sl@0: check1_try_iterations (BusContext *context, sl@0: const char *description, sl@0: Check1Func func) sl@0: { sl@0: Check1Data d; sl@0: sl@0: d.func = func; sl@0: d.context = context; sl@0: sl@0: if (!_dbus_test_oom_handling (description, check_oom_check1_func, sl@0: &d)) sl@0: _dbus_assert_not_reached ("test failed"); sl@0: } sl@0: sl@0: static dbus_bool_t sl@0: check_get_services (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *method, sl@0: char ***services, sl@0: int *len) sl@0: { sl@0: DBusMessage *message; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: DBusError error; sl@0: char **srvs; sl@0: int l; sl@0: sl@0: retval = FALSE; sl@0: dbus_error_init (&error); sl@0: message = NULL; sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: method); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: /* send our message */ sl@0: bus_test_run_clients_loop (SEND_PENDING (connection)); sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: dbus_connection_ref (connection); /* because we may get disconnected */ sl@0: block_connection_until_message_from_bus (context, connection, "reply to ListActivatableNames/ListNames"); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_connection_unref (connection); sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive a reply to %s %d on %p\n", sl@0: method, serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (dbus_message_is_error (message, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, "not this error"); sl@0: sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN) sl@0: { sl@0: ; /* good, expected */ sl@0: } sl@0: else sl@0: { sl@0: warn_unexpected (connection, message, sl@0: "method_return for ListActivatableNames/ListNames"); sl@0: sl@0: goto out; sl@0: } sl@0: sl@0: retry_get_property: sl@0: sl@0: if (!dbus_message_get_args (message, &error, sl@0: DBUS_TYPE_ARRAY, sl@0: DBUS_TYPE_STRING, sl@0: &srvs, &l, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: _dbus_verbose ("no memory to list services by %s\n", method); sl@0: dbus_error_free (&error); sl@0: _dbus_wait_for_memory (); sl@0: goto retry_get_property; sl@0: } sl@0: else sl@0: { sl@0: _dbus_assert (dbus_error_is_set (&error)); sl@0: _dbus_warn ("Did not get the expected DBUS_TYPE_ARRAY from %s\n", method); sl@0: goto out; sl@0: } sl@0: } else { sl@0: *services = srvs; sl@0: *len = l; sl@0: } sl@0: } sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: dbus_error_free (&error); sl@0: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: /* returns TRUE if the correct thing happens, sl@0: * but the correct thing may include OOM errors. sl@0: */ sl@0: static dbus_bool_t sl@0: check_list_services (BusContext *context, sl@0: DBusConnection *connection) sl@0: { sl@0: DBusMessage *message; sl@0: DBusMessage *base_service_message; sl@0: const char *base_service; sl@0: dbus_uint32_t serial; sl@0: dbus_bool_t retval; sl@0: const char *existent = EXISTENT_SERVICE_NAME; sl@0: dbus_uint32_t flags; sl@0: char **services; sl@0: int len; sl@0: sl@0: _dbus_verbose ("check_list_services for %p\n", connection); sl@0: sl@0: if (!check_get_services (context, connection, "ListActivatableNames", &services, &len)) sl@0: { sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!_dbus_string_array_contains ((const char **)services, existent)) sl@0: { sl@0: _dbus_warn ("Did not get the expected %s from ListActivatableNames\n", existent); sl@0: return FALSE; sl@0: } sl@0: sl@0: dbus_free_string_array (services); sl@0: sl@0: base_service_message = NULL; sl@0: sl@0: message = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "StartServiceByName"); sl@0: sl@0: if (message == NULL) sl@0: return TRUE; sl@0: sl@0: dbus_message_set_auto_start (message, FALSE); sl@0: sl@0: flags = 0; sl@0: if (!dbus_message_append_args (message, sl@0: DBUS_TYPE_STRING, &existent, sl@0: DBUS_TYPE_UINT32, &flags, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!dbus_connection_send (connection, message, &serial)) sl@0: { sl@0: dbus_message_unref (message); sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: /* now wait for the message bus to hear back from the activated sl@0: * service. sl@0: */ sl@0: block_connection_until_message_from_bus (context, connection, "activated service to connect"); sl@0: sl@0: bus_test_run_everything (context); sl@0: sl@0: if (!dbus_connection_get_is_connected (connection)) sl@0: { sl@0: _dbus_verbose ("connection was disconnected: %s %d\n", _DBUS_FUNCTION_NAME, __LINE__); sl@0: return TRUE; sl@0: } sl@0: sl@0: retval = FALSE; sl@0: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive any messages after %s %d on %p\n", sl@0: "StartServiceByName", serial, connection); sl@0: goto out; sl@0: } sl@0: sl@0: verbose_message_received (connection, message); sl@0: _dbus_verbose (" (after sending %s)\n", "StartServiceByName"); sl@0: sl@0: if (dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR) sl@0: { sl@0: if (!dbus_message_has_sender (message, DBUS_SERVICE_DBUS)) sl@0: { sl@0: _dbus_warn ("Message has wrong sender %s\n", sl@0: dbus_message_get_sender (message) ? sl@0: dbus_message_get_sender (message) : "(none)"); sl@0: goto out; sl@0: } sl@0: sl@0: if (dbus_message_is_error (message, sl@0: DBUS_ERROR_NO_MEMORY)) sl@0: { sl@0: ; /* good, this is a valid response */ sl@0: } sl@0: else if (dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_CHILD_EXITED) || sl@0: dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_CHILD_SIGNALED) || sl@0: dbus_message_is_error (message, sl@0: DBUS_ERROR_SPAWN_EXEC_FAILED)) sl@0: { sl@0: ; /* good, this is expected also */ sl@0: } sl@0: else sl@0: { sl@0: _dbus_warn ("Did not expect error %s\n", sl@0: dbus_message_get_error_name (message)); sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: GotServiceInfo message_kind; sl@0: sl@0: if (!check_base_service_activated (context, connection, sl@0: message, &base_service)) sl@0: goto out; sl@0: sl@0: base_service_message = message; sl@0: message = NULL; sl@0: sl@0: /* We may need to block here for the test service to exit or finish up */ sl@0: block_connection_until_message_from_bus (context, connection, "test service to exit or finish up"); sl@0: sl@0: message = dbus_connection_borrow_message (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Did not receive any messages after base service creation notification\n"); sl@0: goto out; sl@0: } sl@0: sl@0: message_kind = check_got_service_info (message); sl@0: sl@0: dbus_connection_return_message (connection, message); sl@0: message = NULL; sl@0: sl@0: switch (message_kind) sl@0: { sl@0: case GOT_SOMETHING_ELSE: sl@0: case GOT_ERROR: sl@0: case GOT_SERVICE_DELETED: sl@0: _dbus_warn ("Unexpected message after ActivateService " sl@0: "(should be an error or a service announcement)\n"); sl@0: goto out; sl@0: sl@0: case GOT_SERVICE_CREATED: sl@0: message = pop_message_waiting_for_memory (connection); sl@0: if (message == NULL) sl@0: { sl@0: _dbus_warn ("Failed to pop message we just put back! " sl@0: "should have been a NameOwnerChanged (creation)\n"); sl@0: goto out; sl@0: } sl@0: sl@0: if (!check_service_activated (context, connection, EXISTENT_SERVICE_NAME, sl@0: base_service, message)) sl@0: goto out; sl@0: sl@0: dbus_message_unref (message); sl@0: message = NULL; sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: { sl@0: _dbus_warn ("Messages were left over after successful activation\n"); sl@0: goto out; sl@0: } sl@0: sl@0: break; sl@0: } sl@0: } sl@0: sl@0: if (!check_get_services (context, connection, "ListNames", &services, &len)) sl@0: { sl@0: return TRUE; sl@0: } sl@0: sl@0: if (!_dbus_string_array_contains ((const char **)services, existent)) sl@0: { sl@0: _dbus_warn ("Did not get the expected %s from ListNames\n", existent); sl@0: goto out; sl@0: } sl@0: sl@0: dbus_free_string_array (services); sl@0: sl@0: if (!check_send_exit_to_service (context, connection, sl@0: EXISTENT_SERVICE_NAME, base_service)) sl@0: goto out; sl@0: sl@0: retval = TRUE; sl@0: sl@0: out: sl@0: if (message) sl@0: dbus_message_unref (message); sl@0: sl@0: if (base_service_message) sl@0: dbus_message_unref (base_service_message); sl@0: sl@0: return retval; sl@0: } sl@0: sl@0: typedef struct sl@0: { sl@0: Check2Func func; sl@0: BusContext *context; sl@0: DBusConnection *connection; sl@0: } Check2Data; sl@0: sl@0: static dbus_bool_t sl@0: check_oom_check2_func (void *data) sl@0: { sl@0: Check2Data *d = data; sl@0: sl@0: if (! (* d->func) (d->context, d->connection)) sl@0: return FALSE; sl@0: sl@0: if (!check_no_leftovers (d->context)) sl@0: { sl@0: _dbus_warn ("Messages were left over, should be covered by test suite\n"); sl@0: return FALSE; sl@0: } sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: static void sl@0: check2_try_iterations (BusContext *context, sl@0: DBusConnection *connection, sl@0: const char *description, sl@0: Check2Func func) sl@0: { sl@0: Check2Data d; sl@0: sl@0: d.func = func; sl@0: d.context = context; sl@0: d.connection = connection; sl@0: sl@0: if (!_dbus_test_oom_handling (description, check_oom_check2_func, sl@0: &d)) sl@0: { sl@0: _dbus_warn ("%s failed during oom\n", description); sl@0: _dbus_assert_not_reached ("test failed"); sl@0: } sl@0: } sl@0: sl@0: dbus_bool_t sl@0: bus_dispatch_test (const DBusString *test_data_dir) sl@0: { sl@0: BusContext *context; sl@0: DBusConnection *foo; sl@0: DBusConnection *bar; sl@0: DBusConnection *baz; sl@0: DBusError error; sl@0: sl@0: dbus_error_init (&error); sl@0: sl@0: context = bus_context_new_test (test_data_dir, sl@0: "valid-config-files/debug-allow-all.conf"); sl@0: if (context == NULL) sl@0: return FALSE; sl@0: #ifndef __SYMBIAN32__ sl@0: foo = dbus_connection_open_private ("debug-pipe:name=test-server", &error); sl@0: #else sl@0: foo = dbus_connection_open_private ("tcp:host=localhost,port=12436", &error); sl@0: #endif sl@0: if (foo == NULL) sl@0: _dbus_assert_not_reached ("could not alloc connection"); sl@0: sl@0: if (!bus_setup_debug_client (foo)) sl@0: _dbus_assert_not_reached ("could not set up connection"); sl@0: sl@0: spin_connection_until_authenticated (context, foo); sl@0: sl@0: if (!check_hello_message (context, foo)) sl@0: _dbus_assert_not_reached ("hello message failed"); sl@0: sl@0: if (!check_double_hello_message (context, foo)) sl@0: _dbus_assert_not_reached ("double hello message failed"); sl@0: sl@0: if (!check_add_match_all (context, foo)) sl@0: _dbus_assert_not_reached ("AddMatch message failed"); sl@0: #ifndef __SYMBIAN32__ sl@0: bar = dbus_connection_open_private ("debug-pipe:name=test-server", &error); sl@0: #else sl@0: bar = dbus_connection_open_private ("tcp:host=localhost,port=12436", &error); sl@0: #endif sl@0: if (bar == NULL) sl@0: _dbus_assert_not_reached ("could not alloc connection"); sl@0: sl@0: if (!bus_setup_debug_client (bar)) sl@0: _dbus_assert_not_reached ("could not set up connection"); sl@0: sl@0: spin_connection_until_authenticated (context, bar); sl@0: sl@0: if (!check_hello_message (context, bar)) sl@0: _dbus_assert_not_reached ("hello message failed"); sl@0: sl@0: if (!check_add_match_all (context, bar)) sl@0: _dbus_assert_not_reached ("AddMatch message failed"); sl@0: #ifndef __SYMBIAN32__ sl@0: baz = dbus_connection_open_private ("debug-pipe:name=test-server", &error); sl@0: #else sl@0: baz = dbus_connection_open_private ("tcp:host=localhost,port=12436", &error); sl@0: #endif sl@0: if (baz == NULL) sl@0: _dbus_assert_not_reached ("could not alloc connection"); sl@0: sl@0: if (!bus_setup_debug_client (baz)) sl@0: _dbus_assert_not_reached ("could not set up connection"); sl@0: sl@0: spin_connection_until_authenticated (context, baz); sl@0: sl@0: if (!check_hello_message (context, baz)) sl@0: _dbus_assert_not_reached ("hello message failed"); sl@0: sl@0: if (!check_add_match_all (context, baz)) sl@0: _dbus_assert_not_reached ("AddMatch message failed"); sl@0: sl@0: #ifndef __SYMBIAN32__ sl@0: if (!check_get_connection_unix_user (context, baz)) sl@0: _dbus_assert_not_reached ("GetConnectionUnixUser message failed"); sl@0: sl@0: if (!check_get_connection_unix_process_id (context, baz)) sl@0: _dbus_assert_not_reached ("GetConnectionUnixProcessID message failed"); sl@0: #endif sl@0: sl@0: /* Arun changes : some of the services which are not supported*/ sl@0: /* sl@0: if (!check_list_services (context, baz)) sl@0: _dbus_assert_not_reached ("ListActivatableNames message failed"); sl@0: */ sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: { sl@0: _dbus_warn ("Messages were left over after setting up initial connections\n"); sl@0: _dbus_assert_not_reached ("initial connection setup failed"); sl@0: } sl@0: sl@0: check1_try_iterations (context, "create_and_hello", sl@0: check_hello_connection); sl@0: sl@0: /*Arun changes : some of the services which are not supported*/ sl@0: /* sl@0: check2_try_iterations (context, foo, "nonexistent_service_no_auto_start", sl@0: check_nonexistent_service_no_auto_start); sl@0: */ sl@0: sl@0: check2_try_iterations (context, foo, "segfault_service_no_auto_start", sl@0: check_segfault_service_no_auto_start); sl@0: sl@0: // shilps check2_try_iterations (context, foo, "existent_service_no_auto_start", sl@0: // check_existent_service_no_auto_start); sl@0: sl@0: /*Arun changes : some of the services which are not supported*/ sl@0: /* sl@0: check2_try_iterations (context, foo, "nonexistent_service_auto_start", sl@0: check_nonexistent_service_auto_start); sl@0: sl@0: check2_try_iterations (context, foo, "segfault_service_auto_start", sl@0: check_segfault_service_auto_start); sl@0: */ sl@0: sl@0: check2_try_iterations (context, foo, "shell_fail_service_auto_start", sl@0: check_shell_fail_service_auto_start); sl@0: sl@0: #if 0 sl@0: /* Note: need to resolve some issues with the testing code in order to run sl@0: * this in oom (handle that we sometimes don't get replies back from the bus sl@0: * when oom happens, without blocking the test). sl@0: */ sl@0: check2_try_iterations (context, foo, "existent_service_auto_auto_start", sl@0: check_existent_service_auto_start); sl@0: #else sl@0: // shilps if (!check_existent_service_auto_start (context, foo)) sl@0: // _dbus_assert_not_reached ("existent service auto start failed"); sl@0: #endif sl@0: sl@0: sl@0: // shilps if (!check_shell_service_success_auto_start (context, foo)) sl@0: // _dbus_assert_not_reached ("shell success service auto start failed"); sl@0: sl@0: _dbus_verbose ("Disconnecting foo, bar, and baz\n"); sl@0: sl@0: kill_client_connection_unchecked (foo); sl@0: kill_client_connection_unchecked (bar); sl@0: kill_client_connection_unchecked (baz); sl@0: sl@0: bus_context_unref (context); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: dbus_bool_t sl@0: bus_dispatch_sha1_test (const DBusString *test_data_dir) sl@0: { sl@0: BusContext *context; sl@0: DBusConnection *foo; sl@0: DBusError error; sl@0: sl@0: dbus_error_init (&error); sl@0: sl@0: /* Test SHA1 authentication */ sl@0: _dbus_verbose ("Testing SHA1 context\n"); sl@0: sl@0: context = bus_context_new_test (test_data_dir, sl@0: "valid-config-files/debug-allow-all-sha1.conf"); sl@0: if (context == NULL) sl@0: return FALSE; sl@0: #ifndef __SYMBIAN32__ sl@0: foo = dbus_connection_open_private ("debug-pipe:name=test-server", &error); sl@0: #else sl@0: foo = dbus_connection_open_private ("tcp:host=localhost,port=12435", &error); sl@0: #endif sl@0: if (foo == NULL) sl@0: _dbus_assert_not_reached ("could not alloc connection"); sl@0: sl@0: if (!bus_setup_debug_client (foo)) sl@0: _dbus_assert_not_reached ("could not set up connection"); sl@0: sl@0: spin_connection_until_authenticated (context, foo); sl@0: sl@0: if (!check_hello_message (context, foo)) sl@0: _dbus_assert_not_reached ("hello message failed"); sl@0: sl@0: if (!check_add_match_all (context, foo)) sl@0: _dbus_assert_not_reached ("addmatch message failed"); sl@0: sl@0: if (!check_no_leftovers (context)) sl@0: { sl@0: _dbus_warn ("Messages were left over after setting up initial SHA-1 connection\n"); sl@0: _dbus_assert_not_reached ("initial connection setup failed"); sl@0: } sl@0: sl@0: check1_try_iterations (context, "create_and_hello_sha1", sl@0: check_hello_connection); sl@0: sl@0: kill_client_connection_unchecked (foo); sl@0: sl@0: bus_context_unref (context); sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: #endif /* DBUS_BUILD_TESTS */