* dbus-connection.c (dbus_connection_send_with_reply): return TRUE
[platform/upstream/dbus.git] / dbus / dbus-pending-call.c
1 /* -*- mode: C; c-file-style: "gnu" -*- */
2 /* dbus-pending-call.c Object representing a call in progress.
3  *
4  * Copyright (C) 2002, 2003 Red Hat Inc.
5  *
6  * Licensed under the Academic Free License version 2.1
7  *
8  * This program is free software; you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation; either version 2 of the License, or
11  * (at your option) any later version.
12  *
13  * This program is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program; if not, write to the Free Software
20  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
21  *
22  */
23
24 #include "dbus-internals.h"
25 #include "dbus-connection-internal.h"
26 #include "dbus-pending-call-internal.h"
27 #include "dbus-pending-call.h"
28 #include "dbus-list.h"
29 #include "dbus-threads.h"
30 #include "dbus-test.h"
31
32 /**
33  * @defgroup DBusPendingCallInternals DBusPendingCall implementation details
34  * @ingroup DBusInternals
35  * @brief DBusPendingCall private implementation details.
36  *
37  * The guts of DBusPendingCall and its methods.
38  *
39  * @{
40  */
41
42 /**
43  * @brief Internals of DBusPendingCall
44  *
45  * Opaque object representing a reply message that we're waiting for.
46  */
47 struct DBusPendingCall
48 {
49   DBusAtomic refcount;                            /**< reference count */
50
51   DBusDataSlotList slot_list;                     /**< Data stored by allocated integer ID */
52   
53   DBusPendingCallNotifyFunction function;         /**< Notifier when reply arrives. */
54
55   DBusConnection *connection;                     /**< Connections we're associated with */
56   DBusMessage *reply;                             /**< Reply (after we've received it) */
57   DBusTimeout *timeout;                           /**< Timeout */
58
59   DBusList *timeout_link;                         /**< Preallocated timeout response */
60   
61   dbus_uint32_t reply_serial;                     /**< Expected serial of reply */
62
63   unsigned int completed : 1;                     /**< TRUE if completed */
64   unsigned int timeout_added : 1;                 /**< Have added the timeout */
65 };
66
67 static dbus_int32_t notify_user_data_slot = -1;
68
69 /**
70  * Creates a new pending reply object.
71  *
72  * @param connection connection where reply will arrive
73  * @param timeout_milliseconds length of timeout, -1 for default
74  * @param timeout_handler timeout handler, takes pending call as data
75  * @returns a new #DBusPendingCall or #NULL if no memory.
76  */
77 DBusPendingCall*
78 _dbus_pending_call_new (DBusConnection    *connection,
79                         int                timeout_milliseconds,
80                         DBusTimeoutHandler timeout_handler)
81 {
82   DBusPendingCall *pending;
83   DBusTimeout *timeout;
84
85   _dbus_assert (timeout_milliseconds >= 0 || timeout_milliseconds == -1);
86  
87   if (timeout_milliseconds == -1)
88     timeout_milliseconds = _DBUS_DEFAULT_TIMEOUT_VALUE;
89
90   /* it would probably seem logical to pass in _DBUS_INT_MAX for
91    * infinite timeout, but then math in
92    * _dbus_connection_block_for_reply would get all overflow-prone, so
93    * smack that down.
94    */
95   if (timeout_milliseconds > _DBUS_ONE_HOUR_IN_MILLISECONDS * 6)
96     timeout_milliseconds = _DBUS_ONE_HOUR_IN_MILLISECONDS * 6;
97   
98   if (!dbus_pending_call_allocate_data_slot (&notify_user_data_slot))
99     return NULL;
100   
101   pending = dbus_new0 (DBusPendingCall, 1);
102   
103   if (pending == NULL)
104     {
105       dbus_pending_call_free_data_slot (&notify_user_data_slot);
106       return NULL;
107     }
108
109   timeout = _dbus_timeout_new (timeout_milliseconds,
110                                timeout_handler,
111                                pending, NULL);  
112
113   if (timeout == NULL)
114     {
115       dbus_pending_call_free_data_slot (&notify_user_data_slot);
116       dbus_free (pending);
117       return NULL;
118     }
119   
120   pending->refcount.value = 1;
121   pending->connection = connection;
122   dbus_connection_ref (pending->connection);
123
124   pending->timeout = timeout;
125
126
127   _dbus_data_slot_list_init (&pending->slot_list);
128   
129   return pending;
130 }
131
132 /**
133  * Sets the reply of a pending call with the given message,
134  * or if the message is #NULL, by timing out the pending call.
135  * 
136  * @param pending the pending call
137  * @param message the message to complete the call with, or #NULL
138  *  to time out the call
139  */
140 void
141 _dbus_pending_call_set_reply (DBusPendingCall *pending,
142                               DBusMessage     *message)
143 {
144   if (message == NULL)
145     {
146       message = pending->timeout_link->data;
147       _dbus_list_clear (&pending->timeout_link);
148     }
149   else
150     dbus_message_ref (message);
151
152   _dbus_verbose ("  handing message %p (%s) to pending call serial %u\n",
153                  message,
154                  dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_METHOD_RETURN ?
155                  "method return" :
156                  dbus_message_get_type (message) == DBUS_MESSAGE_TYPE_ERROR ?
157                  "error" : "other type",
158                  pending->reply_serial);
159   
160   _dbus_assert (pending->reply == NULL);
161   _dbus_assert (pending->reply_serial == dbus_message_get_reply_serial (message));
162   pending->reply = message;
163 }
164
165 /**
166  * Calls notifier function for the pending call
167  * and sets the call to completed.
168  *
169  * @param pending the pending call
170  * 
171  */
172 void
173 _dbus_pending_call_complete (DBusPendingCall *pending)
174 {
175   _dbus_assert (!pending->completed);
176   
177   pending->completed = TRUE;
178
179   if (pending->function)
180     {
181       void *user_data;
182       user_data = dbus_pending_call_get_data (pending,
183                                               notify_user_data_slot);
184       
185       (* pending->function) (pending, user_data);
186     }
187 }
188
189 void
190 _dbus_pending_call_queue_timeout_error (DBusPendingCall *pending, 
191                                         DBusConnection *connection)
192 {
193   if (pending->timeout_link)
194     {
195       _dbus_connection_queue_synthesized_message_link (connection,
196                                                        pending->timeout_link);
197       pending->timeout_link = NULL;
198     }
199 }
200
201 /**
202  * Checks to see if a timeout has been added
203  *
204  * @param pending the pending_call
205  * @returns #TRUE if there is a timeout or #FALSE if not
206  */
207 dbus_bool_t 
208 _dbus_pending_call_is_timeout_added (DBusPendingCall  *pending)
209 {
210   _dbus_assert (pending != NULL);
211
212   return pending->timeout_added;
213 }
214
215
216 /**
217  * Sets wether the timeout has been added
218  *
219  * @param pending the pending_call
220  * @param is_added whether or not a timeout is added
221  */
222 void
223 _dbus_pending_call_set_timeout_added (DBusPendingCall  *pending,
224                                       dbus_bool_t       is_added)
225 {
226   _dbus_assert (pending != NULL);
227
228   pending->timeout_added = is_added;
229 }
230
231
232 /**
233  * Retrives the timeout
234  *
235  * @param pending the pending_call
236  * @returns a timeout object 
237  */
238 DBusTimeout *
239 _dbus_pending_call_get_timeout (DBusPendingCall  *pending)
240 {
241   _dbus_assert (pending != NULL);
242
243   return pending->timeout;
244 }
245
246 /**
247  * Gets the reply's serial number
248  *
249  * @param pending the pending_call
250  * @returns a serial number for the reply or 0 
251  */
252 dbus_uint32_t 
253 _dbus_pending_call_get_reply_serial (DBusPendingCall  *pending)
254 {
255   _dbus_assert (pending != NULL);
256
257   return pending->reply_serial;
258 }
259
260 /**
261  * Sets the reply's serial number
262  *
263  * @param pending the pending_call
264  * @param serial the serial number 
265  */
266 void
267 _dbus_pending_call_set_reply_serial  (DBusPendingCall *pending,
268                                       dbus_uint32_t serial)
269 {
270   _dbus_assert (pending != NULL);
271   _dbus_assert (pending->reply_serial == 0);
272
273   pending->reply_serial = serial;
274 }
275
276 /**
277  * Gets the connection associated with this pending call
278  *
279  * @param pending the pending_call
280  * @returns the connection associated with the pending call
281  */
282 DBusConnection *
283 _dbus_pending_call_get_connection (DBusPendingCall *pending)
284 {
285   _dbus_assert (pending != NULL);
286  
287   return pending->connection; 
288 }
289
290 /**
291  * Sets the reply message associated with the pending call to a timeout error
292  *
293  * @param pending the pending_call
294  * @param message the message we are sending the error reply to 
295  * @param serial serial number for the reply
296  * @return #FALSE on OOM
297  */
298 dbus_bool_t
299 _dbus_pending_call_set_timeout_error (DBusPendingCall *pending,
300                                       DBusMessage *message,
301                                       dbus_uint32_t serial)
302
303   DBusList *reply_link;
304   DBusMessage *reply;
305
306   reply = dbus_message_new_error (message, DBUS_ERROR_NO_REPLY,
307                                   "No reply within specified time");
308   if (reply == NULL)
309     return FALSE;
310
311   reply_link = _dbus_list_alloc_link (reply);
312   if (reply_link == NULL)
313     {
314       dbus_message_unref (reply);
315       return FALSE;
316     }
317
318   pending->timeout_link = reply_link;
319
320   _dbus_pending_call_set_reply_serial (pending, serial);
321   
322   return TRUE;
323 }
324
325 /** @} */
326
327 /**
328  * @defgroup DBusPendingCall DBusPendingCall
329  * @ingroup  DBus
330  * @brief Pending reply to a method call message
331  *
332  * A DBusPendingCall is an object representing an
333  * expected reply. A #DBusPendingCall can be created
334  * when you send a message that should have a reply.
335  *
336  * @{
337  */
338
339 /**
340  * @typedef DBusPendingCall
341  *
342  * Opaque data type representing a message pending.
343  */
344
345 /**
346  * Increments the reference count on a pending call.
347  *
348  * @param pending the pending call object
349  * @returns the pending call object
350  */
351 DBusPendingCall *
352 dbus_pending_call_ref (DBusPendingCall *pending)
353 {
354   _dbus_return_val_if_fail (pending != NULL, NULL);
355
356   _dbus_atomic_inc (&pending->refcount);
357
358   return pending;
359 }
360
361 /**
362  * Decrements the reference count on a pending call,
363  * freeing it if the count reaches 0.
364  *
365  * @param pending the pending call object
366  */
367 void
368 dbus_pending_call_unref (DBusPendingCall *pending)
369 {
370   dbus_bool_t last_unref;
371
372   _dbus_return_if_fail (pending != NULL);
373
374   last_unref = (_dbus_atomic_dec (&pending->refcount) == 1);
375
376   if (last_unref)
377     {
378       /* If we get here, we should be already detached
379        * from the connection, or never attached.
380        */
381       _dbus_assert (!pending->timeout_added);  
382      
383       dbus_connection_unref (pending->connection);
384
385       /* this assumes we aren't holding connection lock... */
386       _dbus_data_slot_list_free (&pending->slot_list);
387
388       if (pending->timeout != NULL)
389         _dbus_timeout_unref (pending->timeout);
390       
391       if (pending->timeout_link)
392         {
393           dbus_message_unref ((DBusMessage *)pending->timeout_link->data);
394           _dbus_list_free_link (pending->timeout_link);
395           pending->timeout_link = NULL;
396         }
397
398       if (pending->reply)
399         {
400           dbus_message_unref (pending->reply);
401           pending->reply = NULL;
402         }
403       
404       dbus_free (pending);
405
406       dbus_pending_call_free_data_slot (&notify_user_data_slot);
407     }
408 }
409
410 /**
411  * Sets a notification function to be called when the reply is
412  * received or the pending call times out.
413  *
414  * @param pending the pending call
415  * @param function notifier function
416  * @param user_data data to pass to notifier function
417  * @param free_user_data function to free the user data
418  * @returns #FALSE if not enough memory
419  */
420 dbus_bool_t
421 dbus_pending_call_set_notify (DBusPendingCall              *pending,
422                               DBusPendingCallNotifyFunction function,
423                               void                         *user_data,
424                               DBusFreeFunction              free_user_data)
425 {
426   _dbus_return_val_if_fail (pending != NULL, FALSE);
427
428   /* could invoke application code! */
429   if (!dbus_pending_call_set_data (pending, notify_user_data_slot,
430                                    user_data, free_user_data))
431     return FALSE;
432   
433   pending->function = function;
434
435   return TRUE;
436 }
437
438 /**
439  * Cancels the pending call, such that any reply or error received
440  * will just be ignored.  Drops the dbus library's internal reference
441  * to the #DBusPendingCall so will free the call if nobody else is
442  * holding a reference. However you usually get a reference
443  * from dbus_connection_send() so probably your app owns a ref also.
444  * 
445  * @param pending the pending call
446  */
447 void
448 dbus_pending_call_cancel (DBusPendingCall *pending)
449 {
450   if (pending->connection)
451     _dbus_connection_remove_pending_call (pending->connection,
452                                           pending);
453 }
454
455 /**
456  * Checks whether the pending call has received a reply
457  * yet, or not.
458  *
459  * @todo not thread safe? I guess it has to lock though it sucks
460  *
461  * @param pending the pending call
462  * @returns #TRUE if a reply has been received */
463 dbus_bool_t
464 dbus_pending_call_get_completed (DBusPendingCall *pending)
465 {
466   return pending->completed;
467 }
468
469 /**
470  * Gets the reply, or returns #NULL if none has been received
471  * yet. Ownership of the reply message passes to the caller. This
472  * function can only be called once per pending call, since the reply
473  * message is tranferred to the caller.
474  * 
475  * @param pending the pending call
476  * @returns the reply message or #NULL.
477  */
478 DBusMessage*
479 dbus_pending_call_steal_reply (DBusPendingCall *pending)
480 {
481   DBusMessage *message;
482   
483   _dbus_return_val_if_fail (pending->completed, NULL);
484   _dbus_return_val_if_fail (pending->reply != NULL, NULL);
485   
486   message = pending->reply;
487   pending->reply = NULL;
488
489   return message;
490 }
491
492 /**
493  * Block until the pending call is completed.  The blocking is as with
494  * dbus_connection_send_with_reply_and_block(); it does not enter the
495  * main loop or process other messages, it simply waits for the reply
496  * in question.
497  *
498  * If the pending call is already completed, this function returns
499  * immediately.
500  *
501  * @todo when you start blocking, the timeout is reset, but it should
502  * really only use time remaining since the pending call was created.
503  *
504  * @param pending the pending call
505  */
506 void
507 dbus_pending_call_block (DBusPendingCall *pending)
508 {
509   _dbus_connection_block_pending_call (pending);
510 }
511
512 static DBusDataSlotAllocator slot_allocator;
513 _DBUS_DEFINE_GLOBAL_LOCK (pending_call_slots);
514
515 /**
516  * Allocates an integer ID to be used for storing application-specific
517  * data on any DBusPendingCall. The allocated ID may then be used
518  * with dbus_pending_call_set_data() and dbus_pending_call_get_data().
519  * The passed-in slot must be initialized to -1, and is filled in
520  * with the slot ID. If the passed-in slot is not -1, it's assumed
521  * to be already allocated, and its refcount is incremented.
522  * 
523  * The allocated slot is global, i.e. all DBusPendingCall objects will
524  * have a slot with the given integer ID reserved.
525  *
526  * @param slot_p address of a global variable storing the slot
527  * @returns #FALSE on failure (no memory)
528  */
529 dbus_bool_t
530 dbus_pending_call_allocate_data_slot (dbus_int32_t *slot_p)
531 {
532   return _dbus_data_slot_allocator_alloc (&slot_allocator,
533                                           _DBUS_LOCK_NAME (pending_call_slots),
534                                           slot_p);
535 }
536
537 /**
538  * Deallocates a global ID for #DBusPendingCall data slots.
539  * dbus_pending_call_get_data() and dbus_pending_call_set_data() may
540  * no longer be used with this slot.  Existing data stored on existing
541  * DBusPendingCall objects will be freed when the #DBusPendingCall is
542  * finalized, but may not be retrieved (and may only be replaced if
543  * someone else reallocates the slot).  When the refcount on the
544  * passed-in slot reaches 0, it is set to -1.
545  *
546  * @param slot_p address storing the slot to deallocate
547  */
548 void
549 dbus_pending_call_free_data_slot (dbus_int32_t *slot_p)
550 {
551   _dbus_return_if_fail (*slot_p >= 0);
552   
553   _dbus_data_slot_allocator_free (&slot_allocator, slot_p);
554 }
555
556 /**
557  * Stores a pointer on a #DBusPendingCall, along
558  * with an optional function to be used for freeing
559  * the data when the data is set again, or when
560  * the pending call is finalized. The slot number
561  * must have been allocated with dbus_pending_call_allocate_data_slot().
562  *
563  * @param pending the pending_call
564  * @param slot the slot number
565  * @param data the data to store
566  * @param free_data_func finalizer function for the data
567  * @returns #TRUE if there was enough memory to store the data
568  */
569 dbus_bool_t
570 dbus_pending_call_set_data (DBusPendingCall  *pending,
571                             dbus_int32_t      slot,
572                             void             *data,
573                             DBusFreeFunction  free_data_func)
574 {
575   DBusFreeFunction old_free_func;
576   void *old_data;
577   dbus_bool_t retval;
578
579   _dbus_return_val_if_fail (pending != NULL, FALSE);
580   _dbus_return_val_if_fail (slot >= 0, FALSE);
581
582   retval = _dbus_data_slot_list_set (&slot_allocator,
583                                      &pending->slot_list,
584                                      slot, data, free_data_func,
585                                      &old_free_func, &old_data);
586
587   if (retval)
588     {
589       if (old_free_func)
590         (* old_free_func) (old_data);
591     }
592
593   return retval;
594 }
595
596 /**
597  * Retrieves data previously set with dbus_pending_call_set_data().
598  * The slot must still be allocated (must not have been freed).
599  *
600  * @param pending the pending_call
601  * @param slot the slot to get data from
602  * @returns the data, or #NULL if not found
603  */
604 void*
605 dbus_pending_call_get_data (DBusPendingCall   *pending,
606                             dbus_int32_t       slot)
607 {
608   void *res;
609
610   _dbus_return_val_if_fail (pending != NULL, NULL);
611
612   res = _dbus_data_slot_list_get (&slot_allocator,
613                                   &pending->slot_list,
614                                   slot);
615
616   return res;
617 }
618
619 /** @} */
620
621 #ifdef DBUS_BUILD_TESTS
622
623 /**
624  * @ingroup DBusPendingCallInternals
625  * Unit test for DBusPendingCall.
626  *
627  * @returns #TRUE on success.
628  */
629 dbus_bool_t
630 _dbus_pending_call_test (const char *test_data_dir)
631 {  
632
633   return TRUE;
634 }
635 #endif /* DBUS_BUILD_TESTS */