1 /* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ 2 /* dbus-message.c DBusMessage object 3 * 4 * Copyright (C) 2002, 2003, 2004, 2005 Red Hat Inc. 5 * Copyright (C) 2002, 2003 CodeFactory AB 6 * 7 * Licensed under the Academic Free License version 2.1 8 * 9 * This program is free software; you can redistribute it and/or modify 10 * it under the terms of the GNU General Public License as published by 11 * the Free Software Foundation; either version 2 of the License, or 12 * (at your option) any later version. 13 * 14 * This program is distributed in the hope that it will be useful, 15 * but WITHOUT ANY WARRANTY; without even the implied warranty of 16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 17 * GNU General Public License for more details. 18 * 19 * You should have received a copy of the GNU General Public License 20 * along with this program; if not, write to the Free Software 21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 22 * 23 */ 24 25 #include <config.h> 26 #include "dbus-internals.h" 27 #include "dbus-marshal-recursive.h" 28 #include "dbus-marshal-validate.h" 29 #include "dbus-marshal-byteswap.h" 30 #include "dbus-marshal-header.h" 31 #include "dbus-signature.h" 32 #include "dbus-message-private.h" 33 #include "dbus-object-tree.h" 34 #include "dbus-memory.h" 35 #include "dbus-list.h" 36 #include "dbus-threads-internal.h" 37 #ifdef HAVE_UNIX_FD_PASSING 38 #include "dbus-sysdeps-unix.h" 39 #endif 40 41 #include <string.h> 42 43 #define _DBUS_TYPE_IS_STRINGLIKE(type) \ 44 (type == DBUS_TYPE_STRING || type == DBUS_TYPE_SIGNATURE || \ 45 type == DBUS_TYPE_OBJECT_PATH) 46 47 static void dbus_message_finalize (DBusMessage *message); 48 49 /** 50 * @defgroup DBusMessageInternals DBusMessage implementation details 51 * @ingroup DBusInternals 52 * @brief DBusMessage private implementation details. 53 * 54 * The guts of DBusMessage and its methods. 55 * 56 * @{ 57 */ 58 59 #ifdef DBUS_BUILD_TESTS 60 static dbus_bool_t 61 _dbus_enable_message_cache (void) 62 { 63 static int enabled = -1; 64 65 if (enabled < 0) 66 { 67 const char *s = _dbus_getenv ("DBUS_MESSAGE_CACHE"); 68 69 enabled = TRUE; 70 71 if (s && *s) 72 { 73 if (*s == '0') 74 enabled = FALSE; 75 else if (*s == '1') 76 enabled = TRUE; 77 else 78 _dbus_warn ("DBUS_MESSAGE_CACHE should be 0 or 1 if set, not '%s'", 79 s); 80 } 81 } 82 83 return enabled; 84 } 85 #else 86 /* constant expression, should be optimized away */ 87 # define _dbus_enable_message_cache() (TRUE) 88 #endif 89 90 #ifndef _dbus_message_trace_ref 91 void 92 _dbus_message_trace_ref (DBusMessage *message, 93 int old_refcount, 94 int new_refcount, 95 const char *why) 96 { 97 static int enabled = -1; 98 99 _dbus_trace_ref ("DBusMessage", message, old_refcount, new_refcount, why, 100 "DBUS_MESSAGE_TRACE", &enabled); 101 } 102 #endif 103 104 /* Not thread locked, but strictly const/read-only so should be OK 105 */ 106 /** An static string representing an empty signature */ 107 _DBUS_STRING_DEFINE_STATIC(_dbus_empty_signature_str, ""); 108 109 /* these have wacky values to help trap uninitialized iterators; 110 * but has to fit in 3 bits 111 */ 112 enum { 113 DBUS_MESSAGE_ITER_TYPE_READER = 3, 114 DBUS_MESSAGE_ITER_TYPE_WRITER = 7 115 }; 116 117 /** typedef for internals of message iterator */ 118 typedef struct DBusMessageRealIter DBusMessageRealIter; 119 120 /** 121 * @brief Internals of DBusMessageIter 122 * 123 * Object representing a position in a message. All fields are internal. 124 */ 125 struct DBusMessageRealIter 126 { 127 DBusMessage *message; /**< Message used */ 128 dbus_uint32_t changed_stamp : CHANGED_STAMP_BITS; /**< stamp to detect invalid iters */ 129 dbus_uint32_t iter_type : 3; /**< whether this is a reader or writer iter */ 130 dbus_uint32_t sig_refcount : 8; /**< depth of open_signature() */ 131 union 132 { 133 DBusTypeWriter writer; /**< writer */ 134 DBusTypeReader reader; /**< reader */ 135 } u; /**< the type writer or reader that does all the work */ 136 }; 137 138 static void 139 get_const_signature (DBusHeader *header, 140 const DBusString **type_str_p, 141 int *type_pos_p) 142 { 143 if (_dbus_header_get_field_raw (header, 144 DBUS_HEADER_FIELD_SIGNATURE, 145 type_str_p, 146 type_pos_p)) 147 { 148 *type_pos_p += 1; /* skip the signature length which is 1 byte */ 149 } 150 else 151 { 152 *type_str_p = &_dbus_empty_signature_str; 153 *type_pos_p = 0; 154 } 155 } 156 157 /** 158 * Swaps the message to compiler byte order if required 159 * 160 * @param message the message 161 */ 162 static void 163 _dbus_message_byteswap (DBusMessage *message) 164 { 165 const DBusString *type_str; 166 int type_pos; 167 char byte_order; 168 169 byte_order = _dbus_header_get_byte_order (&message->header); 170 171 if (byte_order == DBUS_COMPILER_BYTE_ORDER) 172 return; 173 174 _dbus_verbose ("Swapping message into compiler byte order\n"); 175 176 get_const_signature (&message->header, &type_str, &type_pos); 177 178 _dbus_marshal_byteswap (type_str, type_pos, 179 byte_order, 180 DBUS_COMPILER_BYTE_ORDER, 181 &message->body, 0); 182 183 _dbus_header_byteswap (&message->header, DBUS_COMPILER_BYTE_ORDER); 184 _dbus_assert (_dbus_header_get_byte_order (&message->header) == 185 DBUS_COMPILER_BYTE_ORDER); 186 } 187 188 /** byte-swap the message if it doesn't match our byte order. 189 * Called only when we need the message in our own byte order, 190 * normally when reading arrays of integers or doubles. 191 * Otherwise should not be called since it would do needless 192 * work. 193 */ 194 #define ensure_byte_order(message) _dbus_message_byteswap (message) 195 196 /** 197 * Gets the data to be sent over the network for this message. 198 * The header and then the body should be written out. 199 * This function is guaranteed to always return the same 200 * data once a message is locked (with dbus_message_lock()). 201 * 202 * @param message the message. 203 * @param header return location for message header data. 204 * @param body return location for message body data. 205 */ 206 void 207 _dbus_message_get_network_data (DBusMessage *message, 208 const DBusString **header, 209 const DBusString **body) 210 { 211 _dbus_assert (message->locked); 212 213 *header = &message->header.data; 214 *body = &message->body; 215 } 216 217 /** 218 * Gets the unix fds to be sent over the network for this message. 219 * This function is guaranteed to always return the same data once a 220 * message is locked (with dbus_message_lock()). 221 * 222 * @param message the message. 223 * @param fds return location of unix fd array 224 * @param n_fds return number of entries in array 225 */ 226 void _dbus_message_get_unix_fds(DBusMessage *message, 227 const int **fds, 228 unsigned *n_fds) 229 { 230 _dbus_assert (message->locked); 231 232 #ifdef HAVE_UNIX_FD_PASSING 233 *fds = message->unix_fds; 234 *n_fds = message->n_unix_fds; 235 #else 236 *fds = NULL; 237 *n_fds = 0; 238 #endif 239 } 240 241 /** 242 * Sets the serial number of a message. 243 * This can only be done once on a message. 244 * 245 * DBusConnection will automatically set the serial to an appropriate value 246 * when the message is sent; this function is only needed when encapsulating 247 * messages in another protocol, or otherwise bypassing DBusConnection. 248 * 249 * @param message the message 250 * @param serial the serial 251 */ 252 void 253 dbus_message_set_serial (DBusMessage *message, 254 dbus_uint32_t serial) 255 { 256 _dbus_return_if_fail (message != NULL); 257 _dbus_return_if_fail (!message->locked); 258 259 _dbus_header_set_serial (&message->header, serial); 260 } 261 262 /** 263 * Adds a counter to be incremented immediately with the size/unix fds 264 * of this message, and decremented by the size/unix fds of this 265 * message when this message if finalized. The link contains a 266 * counter with its refcount already incremented, but the counter 267 * itself not incremented. Ownership of link and counter refcount is 268 * passed to the message. 269 * 270 * This function may be called with locks held. As a result, the counter's 271 * notify function is not called; the caller is expected to either call 272 * _dbus_counter_notify() on the counter when they are no longer holding 273 * locks, or take the same action that would be taken by the notify function. 274 * 275 * @param message the message 276 * @param link link with counter as data 277 */ 278 void 279 _dbus_message_add_counter_link (DBusMessage *message, 280 DBusList *link) 281 { 282 /* right now we don't recompute the delta when message 283 * size changes, and that's OK for current purposes 284 * I think, but could be important to change later. 285 * Do recompute it whenever there are no outstanding counters, 286 * since it's basically free. 287 */ 288 if (message->counters == NULL) 289 { 290 message->size_counter_delta = 291 _dbus_string_get_length (&message->header.data) + 292 _dbus_string_get_length (&message->body); 293 294 #ifdef HAVE_UNIX_FD_PASSING 295 message->unix_fd_counter_delta = message->n_unix_fds; 296 #endif 297 298 #if 0 299 _dbus_verbose ("message has size %ld\n", 300 message->size_counter_delta); 301 #endif 302 } 303 304 _dbus_list_append_link (&message->counters, link); 305 306 _dbus_counter_adjust_size (link->data, message->size_counter_delta); 307 308 #ifdef HAVE_UNIX_FD_PASSING 309 _dbus_counter_adjust_unix_fd (link->data, message->unix_fd_counter_delta); 310 #endif 311 } 312 313 /** 314 * Adds a counter to be incremented immediately with the size/unix fds 315 * of this message, and decremented by the size/unix fds of this 316 * message when this message if finalized. 317 * 318 * This function may be called with locks held. As a result, the counter's 319 * notify function is not called; the caller is expected to either call 320 * _dbus_counter_notify() on the counter when they are no longer holding 321 * locks, or take the same action that would be taken by the notify function. 322 * 323 * @param message the message 324 * @param counter the counter 325 * @returns #FALSE if no memory 326 */ 327 dbus_bool_t 328 _dbus_message_add_counter (DBusMessage *message, 329 DBusCounter *counter) 330 { 331 DBusList *link; 332 333 link = _dbus_list_alloc_link (counter); 334 if (link == NULL) 335 return FALSE; 336 337 _dbus_counter_ref (counter); 338 _dbus_message_add_counter_link (message, link); 339 340 return TRUE; 341 } 342 343 /** 344 * Removes a counter tracking the size/unix fds of this message, and 345 * decrements the counter by the size/unix fds of this message. 346 * 347 * @param message the message 348 * @param counter the counter 349 */ 350 void 351 _dbus_message_remove_counter (DBusMessage *message, 352 DBusCounter *counter) 353 { 354 DBusList *link; 355 356 link = _dbus_list_find_last (&message->counters, 357 counter); 358 _dbus_assert (link != NULL); 359 360 _dbus_list_remove_link (&message->counters, link); 361 362 _dbus_counter_adjust_size (counter, - message->size_counter_delta); 363 364 #ifdef HAVE_UNIX_FD_PASSING 365 _dbus_counter_adjust_unix_fd (counter, - message->unix_fd_counter_delta); 366 #endif 367 368 _dbus_counter_notify (counter); 369 _dbus_counter_unref (counter); 370 } 371 372 /** 373 * Locks a message. Allows checking that applications don't keep a 374 * reference to a message in the outgoing queue and change it 375 * underneath us. Messages are locked when they enter the outgoing 376 * queue (dbus_connection_send_message()), and the library complains 377 * if the message is modified while locked. This function may also 378 * called externally, for applications wrapping D-Bus in another protocol. 379 * 380 * @param message the message to lock. 381 */ 382 void 383 dbus_message_lock (DBusMessage *message) 384 { 385 if (!message->locked) 386 { 387 _dbus_header_update_lengths (&message->header, 388 _dbus_string_get_length (&message->body)); 389 390 /* must have a signature if you have a body */ 391 _dbus_assert (_dbus_string_get_length (&message->body) == 0 || 392 dbus_message_get_signature (message) != NULL); 393 394 message->locked = TRUE; 395 } 396 } 397 398 static dbus_bool_t 399 set_or_delete_string_field (DBusMessage *message, 400 int field, 401 int typecode, 402 const char *value) 403 { 404 if (value == NULL) 405 return _dbus_header_delete_field (&message->header, field); 406 else 407 return _dbus_header_set_field_basic (&message->header, 408 field, 409 typecode, 410 &value); 411 } 412 413 #if 0 414 /* Probably we don't need to use this */ 415 /** 416 * Sets the signature of the message, i.e. the arguments in the 417 * message payload. The signature includes only "in" arguments for 418 * #DBUS_MESSAGE_TYPE_METHOD_CALL and only "out" arguments for 419 * #DBUS_MESSAGE_TYPE_METHOD_RETURN, so is slightly different from 420 * what you might expect (it does not include the signature of the 421 * entire C++-style method). 422 * 423 * The signature is a string made up of type codes such as 424 * #DBUS_TYPE_INT32. The string is terminated with nul (nul is also 425 * the value of #DBUS_TYPE_INVALID). The macros such as 426 * #DBUS_TYPE_INT32 evaluate to integers; to assemble a signature you 427 * may find it useful to use the string forms, such as 428 * #DBUS_TYPE_INT32_AS_STRING. 429 * 430 * An "unset" or #NULL signature is considered the same as an empty 431 * signature. In fact dbus_message_get_signature() will never return 432 * #NULL. 433 * 434 * @param message the message 435 * @param signature the type signature or #NULL to unset 436 * @returns #FALSE if no memory 437 */ 438 static dbus_bool_t 439 _dbus_message_set_signature (DBusMessage *message, 440 const char *signature) 441 { 442 _dbus_return_val_if_fail (message != NULL, FALSE); 443 _dbus_return_val_if_fail (!message->locked, FALSE); 444 _dbus_return_val_if_fail (signature == NULL || 445 _dbus_check_is_valid_signature (signature)); 446 /* can't delete the signature if you have a message body */ 447 _dbus_return_val_if_fail (_dbus_string_get_length (&message->body) == 0 || 448 signature != NULL); 449 450 return set_or_delete_string_field (message, 451 DBUS_HEADER_FIELD_SIGNATURE, 452 DBUS_TYPE_SIGNATURE, 453 signature); 454 } 455 #endif 456 457 /* Message Cache 458 * 459 * We cache some DBusMessage to reduce the overhead of allocating 460 * them. In my profiling this consistently made about an 8% 461 * difference. It avoids the malloc for the message, the malloc for 462 * the slot list, the malloc for the header string and body string, 463 * and the associated free() calls. It does introduce another global 464 * lock which could be a performance issue in certain cases. 465 * 466 * For the echo client/server the round trip time goes from around 467 * .000077 to .000069 with the message cache on my laptop. The sysprof 468 * change is as follows (numbers are cumulative percentage): 469 * 470 * with message cache implemented as array as it is now (0.000069 per): 471 * new_empty_header 1.46 472 * mutex_lock 0.56 # i.e. _DBUS_LOCK(message_cache) 473 * mutex_unlock 0.25 474 * self 0.41 475 * unref 2.24 476 * self 0.68 477 * list_clear 0.43 478 * mutex_lock 0.33 # i.e. _DBUS_LOCK(message_cache) 479 * mutex_unlock 0.25 480 * 481 * with message cache implemented as list (0.000070 per roundtrip): 482 * new_empty_header 2.72 483 * list_pop_first 1.88 484 * unref 3.3 485 * list_prepend 1.63 486 * 487 * without cache (0.000077 per roundtrip): 488 * new_empty_header 6.7 489 * string_init_preallocated 3.43 490 * dbus_malloc 2.43 491 * dbus_malloc0 2.59 492 * 493 * unref 4.02 494 * string_free 1.82 495 * dbus_free 1.63 496 * dbus_free 0.71 497 * 498 * If you implement the message_cache with a list, the primary reason 499 * it's slower is that you add another thread lock (on the DBusList 500 * mempool). 501 */ 502 503 /** Avoid caching huge messages */ 504 #define MAX_MESSAGE_SIZE_TO_CACHE 10 * _DBUS_ONE_KILOBYTE 505 506 /** Avoid caching too many messages */ 507 #define MAX_MESSAGE_CACHE_SIZE 5 508 509 _DBUS_DEFINE_GLOBAL_LOCK (message_cache); 510 static DBusMessage *message_cache[MAX_MESSAGE_CACHE_SIZE]; 511 static int message_cache_count = 0; 512 static dbus_bool_t message_cache_shutdown_registered = FALSE; 513 514 static void 515 dbus_message_cache_shutdown (void *data) 516 { 517 int i; 518 519 _DBUS_LOCK (message_cache); 520 521 i = 0; 522 while (i < MAX_MESSAGE_CACHE_SIZE) 523 { 524 if (message_cache[i]) 525 dbus_message_finalize (message_cache[i]); 526 527 ++i; 528 } 529 530 message_cache_count = 0; 531 message_cache_shutdown_registered = FALSE; 532 533 _DBUS_UNLOCK (message_cache); 534 } 535 536 /** 537 * Tries to get a message from the message cache. The retrieved 538 * message will have junk in it, so it still needs to be cleared out 539 * in dbus_message_new_empty_header() 540 * 541 * @returns the message, or #NULL if none cached 542 */ 543 static DBusMessage* 544 dbus_message_get_cached (void) 545 { 546 DBusMessage *message; 547 int i; 548 549 message = NULL; 550 551 _DBUS_LOCK (message_cache); 552 553 _dbus_assert (message_cache_count >= 0); 554 555 if (message_cache_count == 0) 556 { 557 _DBUS_UNLOCK (message_cache); 558 return NULL; 559 } 560 561 /* This is not necessarily true unless count > 0, and 562 * message_cache is uninitialized until the shutdown is 563 * registered 564 */ 565 _dbus_assert (message_cache_shutdown_registered); 566 567 i = 0; 568 while (i < MAX_MESSAGE_CACHE_SIZE) 569 { 570 if (message_cache[i]) 571 { 572 message = message_cache[i]; 573 message_cache[i] = NULL; 574 message_cache_count -= 1; 575 break; 576 } 577 ++i; 578 } 579 _dbus_assert (message_cache_count >= 0); 580 _dbus_assert (i < MAX_MESSAGE_CACHE_SIZE); 581 _dbus_assert (message != NULL); 582 583 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0); 584 585 _dbus_assert (message->counters == NULL); 586 587 _DBUS_UNLOCK (message_cache); 588 589 return message; 590 } 591 592 #ifdef HAVE_UNIX_FD_PASSING 593 static void 594 close_unix_fds(int *fds, unsigned *n_fds) 595 { 596 DBusError e; 597 int i; 598 599 if (*n_fds <= 0) 600 return; 601 602 dbus_error_init(&e); 603 604 for (i = 0; i < *n_fds; i++) 605 { 606 if (!_dbus_close(fds[i], &e)) 607 { 608 _dbus_warn("Failed to close file descriptor: %s\n", e.message); 609 dbus_error_free(&e); 610 } 611 } 612 613 *n_fds = 0; 614 615 /* We don't free the array here, in case we can recycle it later */ 616 } 617 #endif 618 619 static void 620 free_counter (void *element, 621 void *data) 622 { 623 DBusCounter *counter = element; 624 DBusMessage *message = data; 625 626 _dbus_counter_adjust_size (counter, - message->size_counter_delta); 627 #ifdef HAVE_UNIX_FD_PASSING 628 _dbus_counter_adjust_unix_fd (counter, - message->unix_fd_counter_delta); 629 #endif 630 631 _dbus_counter_notify (counter); 632 _dbus_counter_unref (counter); 633 } 634 635 /** 636 * Tries to cache a message, otherwise finalize it. 637 * 638 * @param message the message 639 */ 640 static void 641 dbus_message_cache_or_finalize (DBusMessage *message) 642 { 643 dbus_bool_t was_cached; 644 int i; 645 646 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0); 647 648 /* This calls application code and has to be done first thing 649 * without holding the lock 650 */ 651 _dbus_data_slot_list_clear (&message->slot_list); 652 653 _dbus_list_foreach (&message->counters, 654 free_counter, message); 655 _dbus_list_clear (&message->counters); 656 657 #ifdef HAVE_UNIX_FD_PASSING 658 close_unix_fds(message->unix_fds, &message->n_unix_fds); 659 #endif 660 661 was_cached = FALSE; 662 663 _DBUS_LOCK (message_cache); 664 665 if (!message_cache_shutdown_registered) 666 { 667 _dbus_assert (message_cache_count == 0); 668 669 if (!_dbus_register_shutdown_func (dbus_message_cache_shutdown, NULL)) 670 goto out; 671 672 i = 0; 673 while (i < MAX_MESSAGE_CACHE_SIZE) 674 { 675 message_cache[i] = NULL; 676 ++i; 677 } 678 679 message_cache_shutdown_registered = TRUE; 680 } 681 682 _dbus_assert (message_cache_count >= 0); 683 684 if (!_dbus_enable_message_cache ()) 685 goto out; 686 687 if ((_dbus_string_get_length (&message->header.data) + 688 _dbus_string_get_length (&message->body)) > 689 MAX_MESSAGE_SIZE_TO_CACHE) 690 goto out; 691 692 if (message_cache_count >= MAX_MESSAGE_CACHE_SIZE) 693 goto out; 694 695 /* Find empty slot */ 696 i = 0; 697 while (message_cache[i] != NULL) 698 ++i; 699 700 _dbus_assert (i < MAX_MESSAGE_CACHE_SIZE); 701 702 _dbus_assert (message_cache[i] == NULL); 703 message_cache[i] = message; 704 message_cache_count += 1; 705 was_cached = TRUE; 706 #ifndef DBUS_DISABLE_CHECKS 707 message->in_cache = TRUE; 708 #endif 709 710 out: 711 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0); 712 713 _DBUS_UNLOCK (message_cache); 714 715 if (!was_cached) 716 dbus_message_finalize (message); 717 } 718 719 #ifndef DBUS_DISABLE_CHECKS 720 static dbus_bool_t 721 _dbus_message_iter_check (DBusMessageRealIter *iter) 722 { 723 char byte_order; 724 725 if (iter == NULL) 726 { 727 _dbus_warn_check_failed ("dbus message iterator is NULL\n"); 728 return FALSE; 729 } 730 731 byte_order = _dbus_header_get_byte_order (&iter->message->header); 732 733 if (iter->iter_type == DBUS_MESSAGE_ITER_TYPE_READER) 734 { 735 if (iter->u.reader.byte_order != byte_order) 736 { 737 _dbus_warn_check_failed ("dbus message changed byte order since iterator was created\n"); 738 return FALSE; 739 } 740 /* because we swap the message into compiler order when you init an iter */ 741 _dbus_assert (iter->u.reader.byte_order == DBUS_COMPILER_BYTE_ORDER); 742 } 743 else if (iter->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER) 744 { 745 if (iter->u.writer.byte_order != byte_order) 746 { 747 _dbus_warn_check_failed ("dbus message changed byte order since append iterator was created\n"); 748 return FALSE; 749 } 750 /* because we swap the message into compiler order when you init an iter */ 751 _dbus_assert (iter->u.writer.byte_order == DBUS_COMPILER_BYTE_ORDER); 752 } 753 else 754 { 755 _dbus_warn_check_failed ("dbus message iterator looks uninitialized or corrupted\n"); 756 return FALSE; 757 } 758 759 if (iter->changed_stamp != iter->message->changed_stamp) 760 { 761 _dbus_warn_check_failed ("dbus message iterator invalid because the message has been modified (or perhaps the iterator is just uninitialized)\n"); 762 return FALSE; 763 } 764 765 return TRUE; 766 } 767 #endif /* DBUS_DISABLE_CHECKS */ 768 769 /** 770 * Implementation of the varargs arg-getting functions. 771 * dbus_message_get_args() is the place to go for complete 772 * documentation. 773 * 774 * @todo This may leak memory and file descriptors if parsing fails. See #21259 775 * 776 * @see dbus_message_get_args 777 * @param iter the message iter 778 * @param error error to be filled in 779 * @param first_arg_type type of the first argument 780 * @param var_args return location for first argument, followed by list of type/location pairs 781 * @returns #FALSE if error was set 782 */ 783 dbus_bool_t 784 _dbus_message_iter_get_args_valist (DBusMessageIter *iter, 785 DBusError *error, 786 int first_arg_type, 787 va_list var_args) 788 { 789 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 790 int spec_type, msg_type, i; 791 dbus_bool_t retval; 792 793 _dbus_assert (_dbus_message_iter_check (real)); 794 795 retval = FALSE; 796 797 spec_type = first_arg_type; 798 i = 0; 799 800 while (spec_type != DBUS_TYPE_INVALID) 801 { 802 msg_type = dbus_message_iter_get_arg_type (iter); 803 804 if (msg_type != spec_type) 805 { 806 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS, 807 "Argument %d is specified to be of type \"%s\", but " 808 "is actually of type \"%s\"\n", i, 809 _dbus_type_to_string (spec_type), 810 _dbus_type_to_string (msg_type)); 811 812 goto out; 813 } 814 815 if (spec_type == DBUS_TYPE_UNIX_FD) 816 { 817 #ifdef HAVE_UNIX_FD_PASSING 818 DBusBasicValue idx; 819 int *pfd, nfd; 820 821 pfd = va_arg (var_args, int*); 822 _dbus_assert(pfd); 823 824 _dbus_type_reader_read_basic(&real->u.reader, &idx); 825 826 if (idx.u32 >= real->message->n_unix_fds) 827 { 828 dbus_set_error (error, DBUS_ERROR_INCONSISTENT_MESSAGE, 829 "Message refers to file descriptor at index %i," 830 "but has only %i descriptors attached.\n", 831 idx.u32, 832 real->message->n_unix_fds); 833 goto out; 834 } 835 836 if ((nfd = _dbus_dup(real->message->unix_fds[idx.u32], error)) < 0) 837 goto out; 838 839 *pfd = nfd; 840 #else 841 dbus_set_error (error, DBUS_ERROR_NOT_SUPPORTED, 842 "Platform does not support file desciptor passing.\n"); 843 goto out; 844 #endif 845 } 846 else if (dbus_type_is_basic (spec_type)) 847 { 848 DBusBasicValue *ptr; 849 850 ptr = va_arg (var_args, DBusBasicValue*); 851 852 _dbus_assert (ptr != NULL); 853 854 _dbus_type_reader_read_basic (&real->u.reader, 855 ptr); 856 } 857 else if (spec_type == DBUS_TYPE_ARRAY) 858 { 859 int element_type; 860 int spec_element_type; 861 const DBusBasicValue **ptr; 862 int *n_elements_p; 863 DBusTypeReader array; 864 865 spec_element_type = va_arg (var_args, int); 866 element_type = _dbus_type_reader_get_element_type (&real->u.reader); 867 868 if (spec_element_type != element_type) 869 { 870 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS, 871 "Argument %d is specified to be an array of \"%s\", but " 872 "is actually an array of \"%s\"\n", 873 i, 874 _dbus_type_to_string (spec_element_type), 875 _dbus_type_to_string (element_type)); 876 877 goto out; 878 } 879 880 if (dbus_type_is_fixed (spec_element_type) && 881 element_type != DBUS_TYPE_UNIX_FD) 882 { 883 ptr = va_arg (var_args, const DBusBasicValue**); 884 n_elements_p = va_arg (var_args, int*); 885 886 _dbus_assert (ptr != NULL); 887 _dbus_assert (n_elements_p != NULL); 888 889 _dbus_type_reader_recurse (&real->u.reader, &array); 890 891 _dbus_type_reader_read_fixed_multi (&array, 892 (void *) ptr, n_elements_p); 893 } 894 else if (_DBUS_TYPE_IS_STRINGLIKE (spec_element_type)) 895 { 896 char ***str_array_p; 897 int n_elements; 898 char **str_array; 899 900 str_array_p = va_arg (var_args, char***); 901 n_elements_p = va_arg (var_args, int*); 902 903 _dbus_assert (str_array_p != NULL); 904 _dbus_assert (n_elements_p != NULL); 905 906 /* Count elements in the array */ 907 _dbus_type_reader_recurse (&real->u.reader, &array); 908 909 n_elements = 0; 910 while (_dbus_type_reader_get_current_type (&array) != DBUS_TYPE_INVALID) 911 { 912 ++n_elements; 913 _dbus_type_reader_next (&array); 914 } 915 916 str_array = dbus_new0 (char*, n_elements + 1); 917 if (str_array == NULL) 918 { 919 _DBUS_SET_OOM (error); 920 goto out; 921 } 922 923 /* Now go through and dup each string */ 924 _dbus_type_reader_recurse (&real->u.reader, &array); 925 926 i = 0; 927 while (i < n_elements) 928 { 929 const char *s; 930 _dbus_type_reader_read_basic (&array, 931 (void *) &s); 932 933 str_array[i] = _dbus_strdup (s); 934 if (str_array[i] == NULL) 935 { 936 dbus_free_string_array (str_array); 937 _DBUS_SET_OOM (error); 938 goto out; 939 } 940 941 ++i; 942 943 if (!_dbus_type_reader_next (&array)) 944 _dbus_assert (i == n_elements); 945 } 946 947 _dbus_assert (_dbus_type_reader_get_current_type (&array) == DBUS_TYPE_INVALID); 948 _dbus_assert (i == n_elements); 949 _dbus_assert (str_array[i] == NULL); 950 951 *str_array_p = str_array; 952 *n_elements_p = n_elements; 953 } 954 #ifndef DBUS_DISABLE_CHECKS 955 else 956 { 957 _dbus_warn ("you can't read arrays of container types (struct, variant, array) with %s for now\n", 958 _DBUS_FUNCTION_NAME); 959 goto out; 960 } 961 #endif 962 } 963 #ifndef DBUS_DISABLE_CHECKS 964 else 965 { 966 _dbus_warn ("you can only read arrays and basic types with %s for now\n", 967 _DBUS_FUNCTION_NAME); 968 goto out; 969 } 970 #endif 971 972 spec_type = va_arg (var_args, int); 973 if (!_dbus_type_reader_next (&real->u.reader) && spec_type != DBUS_TYPE_INVALID) 974 { 975 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS, 976 "Message has only %d arguments, but more were expected", i); 977 goto out; 978 } 979 980 i++; 981 } 982 983 retval = TRUE; 984 985 out: 986 987 return retval; 988 } 989 990 /** @} */ 991 992 /** 993 * @defgroup DBusMessage DBusMessage 994 * @ingroup DBus 995 * @brief Message to be sent or received over a #DBusConnection. 996 * 997 * A DBusMessage is the most basic unit of communication over a 998 * DBusConnection. A DBusConnection represents a stream of messages 999 * received from a remote application, and a stream of messages 1000 * sent to a remote application. 1001 * 1002 * A message has a message type, returned from 1003 * dbus_message_get_type(). This indicates whether the message is a 1004 * method call, a reply to a method call, a signal, or an error reply. 1005 * 1006 * A message has header fields such as the sender, destination, method 1007 * or signal name, and so forth. DBusMessage has accessor functions for 1008 * these, such as dbus_message_get_member(). 1009 * 1010 * Convenience functions dbus_message_is_method_call(), dbus_message_is_signal(), 1011 * and dbus_message_is_error() check several header fields at once and are 1012 * slightly more efficient than checking the header fields with individual 1013 * accessor functions. 1014 * 1015 * Finally, a message has arguments. The number and types of arguments 1016 * are in the message's signature header field (accessed with 1017 * dbus_message_get_signature()). Simple argument values are usually 1018 * retrieved with dbus_message_get_args() but more complex values such 1019 * as structs may require the use of #DBusMessageIter. 1020 * 1021 * The D-Bus specification goes into some more detail about header fields and 1022 * message types. 1023 * 1024 * @{ 1025 */ 1026 1027 /** 1028 * @typedef DBusMessage 1029 * 1030 * Opaque data type representing a message received from or to be 1031 * sent to another application. 1032 */ 1033 1034 /** 1035 * Returns the serial of a message or 0 if none has been specified. 1036 * The message's serial number is provided by the application sending 1037 * the message and is used to identify replies to this message. 1038 * 1039 * All messages received on a connection will have a serial provided 1040 * by the remote application. 1041 * 1042 * For messages you're sending, dbus_connection_send() will assign a 1043 * serial and return it to you. 1044 * 1045 * @param message the message 1046 * @returns the serial 1047 */ 1048 dbus_uint32_t 1049 dbus_message_get_serial (DBusMessage *message) 1050 { 1051 _dbus_return_val_if_fail (message != NULL, 0); 1052 1053 return _dbus_header_get_serial (&message->header); 1054 } 1055 1056 /** 1057 * Sets the reply serial of a message (the serial of the message this 1058 * is a reply to). 1059 * 1060 * @param message the message 1061 * @param reply_serial the serial we're replying to 1062 * @returns #FALSE if not enough memory 1063 */ 1064 dbus_bool_t 1065 dbus_message_set_reply_serial (DBusMessage *message, 1066 dbus_uint32_t reply_serial) 1067 { 1068 _dbus_return_val_if_fail (message != NULL, FALSE); 1069 _dbus_return_val_if_fail (!message->locked, FALSE); 1070 _dbus_return_val_if_fail (reply_serial != 0, FALSE); /* 0 is invalid */ 1071 1072 return _dbus_header_set_field_basic (&message->header, 1073 DBUS_HEADER_FIELD_REPLY_SERIAL, 1074 DBUS_TYPE_UINT32, 1075 &reply_serial); 1076 } 1077 1078 /** 1079 * Returns the serial that the message is a reply to or 0 if none. 1080 * 1081 * @param message the message 1082 * @returns the reply serial 1083 */ 1084 dbus_uint32_t 1085 dbus_message_get_reply_serial (DBusMessage *message) 1086 { 1087 dbus_uint32_t v_UINT32; 1088 1089 _dbus_return_val_if_fail (message != NULL, 0); 1090 1091 if (_dbus_header_get_field_basic (&message->header, 1092 DBUS_HEADER_FIELD_REPLY_SERIAL, 1093 DBUS_TYPE_UINT32, 1094 &v_UINT32)) 1095 return v_UINT32; 1096 else 1097 return 0; 1098 } 1099 1100 static void 1101 dbus_message_finalize (DBusMessage *message) 1102 { 1103 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0); 1104 1105 /* This calls application callbacks! */ 1106 _dbus_data_slot_list_free (&message->slot_list); 1107 1108 _dbus_list_foreach (&message->counters, 1109 free_counter, message); 1110 _dbus_list_clear (&message->counters); 1111 1112 _dbus_header_free (&message->header); 1113 _dbus_string_free (&message->body); 1114 1115 #ifdef HAVE_UNIX_FD_PASSING 1116 close_unix_fds(message->unix_fds, &message->n_unix_fds); 1117 dbus_free(message->unix_fds); 1118 #endif 1119 1120 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0); 1121 1122 dbus_free (message); 1123 } 1124 1125 static DBusMessage* 1126 dbus_message_new_empty_header (void) 1127 { 1128 DBusMessage *message; 1129 dbus_bool_t from_cache; 1130 1131 message = dbus_message_get_cached (); 1132 1133 if (message != NULL) 1134 { 1135 from_cache = TRUE; 1136 } 1137 else 1138 { 1139 from_cache = FALSE; 1140 message = dbus_new0 (DBusMessage, 1); 1141 if (message == NULL) 1142 return NULL; 1143 #ifndef DBUS_DISABLE_CHECKS 1144 message->generation = _dbus_current_generation; 1145 #endif 1146 1147 #ifdef HAVE_UNIX_FD_PASSING 1148 message->unix_fds = NULL; 1149 message->n_unix_fds_allocated = 0; 1150 #endif 1151 } 1152 1153 _dbus_atomic_inc (&message->refcount); 1154 1155 _dbus_message_trace_ref (message, 0, 1, "new_empty_header"); 1156 1157 message->locked = FALSE; 1158 #ifndef DBUS_DISABLE_CHECKS 1159 message->in_cache = FALSE; 1160 #endif 1161 message->counters = NULL; 1162 message->size_counter_delta = 0; 1163 message->changed_stamp = 0; 1164 1165 #ifdef HAVE_UNIX_FD_PASSING 1166 message->n_unix_fds = 0; 1167 message->n_unix_fds_allocated = 0; 1168 message->unix_fd_counter_delta = 0; 1169 #endif 1170 1171 if (!from_cache) 1172 _dbus_data_slot_list_init (&message->slot_list); 1173 1174 if (from_cache) 1175 { 1176 _dbus_header_reinit (&message->header); 1177 _dbus_string_set_length (&message->body, 0); 1178 } 1179 else 1180 { 1181 if (!_dbus_header_init (&message->header)) 1182 { 1183 dbus_free (message); 1184 return NULL; 1185 } 1186 1187 if (!_dbus_string_init_preallocated (&message->body, 32)) 1188 { 1189 _dbus_header_free (&message->header); 1190 dbus_free (message); 1191 return NULL; 1192 } 1193 } 1194 1195 return message; 1196 } 1197 1198 /** 1199 * Constructs a new message of the given message type. 1200 * Types include #DBUS_MESSAGE_TYPE_METHOD_CALL, 1201 * #DBUS_MESSAGE_TYPE_SIGNAL, and so forth. 1202 * 1203 * Usually you want to use dbus_message_new_method_call(), 1204 * dbus_message_new_method_return(), dbus_message_new_signal(), 1205 * or dbus_message_new_error() instead. 1206 * 1207 * @param message_type type of message 1208 * @returns new message or #NULL if no memory 1209 */ 1210 DBusMessage* 1211 dbus_message_new (int message_type) 1212 { 1213 DBusMessage *message; 1214 1215 _dbus_return_val_if_fail (message_type != DBUS_MESSAGE_TYPE_INVALID, NULL); 1216 1217 message = dbus_message_new_empty_header (); 1218 if (message == NULL) 1219 return NULL; 1220 1221 if (!_dbus_header_create (&message->header, 1222 DBUS_COMPILER_BYTE_ORDER, 1223 message_type, 1224 NULL, NULL, NULL, NULL, NULL)) 1225 { 1226 dbus_message_unref (message); 1227 return NULL; 1228 } 1229 1230 return message; 1231 } 1232 1233 /** 1234 * Constructs a new message to invoke a method on a remote 1235 * object. Returns #NULL if memory can't be allocated for the 1236 * message. The destination may be #NULL in which case no destination 1237 * is set; this is appropriate when using D-Bus in a peer-to-peer 1238 * context (no message bus). The interface may be #NULL, which means 1239 * that if multiple methods with the given name exist it is undefined 1240 * which one will be invoked. 1241 * 1242 * The path and method names may not be #NULL. 1243 * 1244 * Destination, path, interface, and method name can't contain 1245 * any invalid characters (see the D-Bus specification). 1246 * 1247 * @param destination name that the message should be sent to or #NULL 1248 * @param path object path the message should be sent to 1249 * @param interface interface to invoke method on, or #NULL 1250 * @param method method to invoke 1251 * 1252 * @returns a new DBusMessage, free with dbus_message_unref() 1253 */ 1254 DBusMessage* 1255 dbus_message_new_method_call (const char *destination, 1256 const char *path, 1257 const char *interface, 1258 const char *method) 1259 { 1260 DBusMessage *message; 1261 1262 _dbus_return_val_if_fail (path != NULL, NULL); 1263 _dbus_return_val_if_fail (method != NULL, NULL); 1264 _dbus_return_val_if_fail (destination == NULL || 1265 _dbus_check_is_valid_bus_name (destination), NULL); 1266 _dbus_return_val_if_fail (_dbus_check_is_valid_path (path), NULL); 1267 _dbus_return_val_if_fail (interface == NULL || 1268 _dbus_check_is_valid_interface (interface), NULL); 1269 _dbus_return_val_if_fail (_dbus_check_is_valid_member (method), NULL); 1270 1271 message = dbus_message_new_empty_header (); 1272 if (message == NULL) 1273 return NULL; 1274 1275 if (!_dbus_header_create (&message->header, 1276 DBUS_COMPILER_BYTE_ORDER, 1277 DBUS_MESSAGE_TYPE_METHOD_CALL, 1278 destination, path, interface, method, NULL)) 1279 { 1280 dbus_message_unref (message); 1281 return NULL; 1282 } 1283 1284 return message; 1285 } 1286 1287 /** 1288 * Constructs a message that is a reply to a method call. Returns 1289 * #NULL if memory can't be allocated for the message. 1290 * 1291 * @param method_call the message being replied to 1292 * @returns a new DBusMessage, free with dbus_message_unref() 1293 */ 1294 DBusMessage* 1295 dbus_message_new_method_return (DBusMessage *method_call) 1296 { 1297 DBusMessage *message; 1298 const char *sender; 1299 1300 _dbus_return_val_if_fail (method_call != NULL, NULL); 1301 1302 sender = dbus_message_get_sender (method_call); 1303 1304 /* sender is allowed to be null here in peer-to-peer case */ 1305 1306 message = dbus_message_new_empty_header (); 1307 if (message == NULL) 1308 return NULL; 1309 1310 if (!_dbus_header_create (&message->header, 1311 DBUS_COMPILER_BYTE_ORDER, 1312 DBUS_MESSAGE_TYPE_METHOD_RETURN, 1313 sender, NULL, NULL, NULL, NULL)) 1314 { 1315 dbus_message_unref (message); 1316 return NULL; 1317 } 1318 1319 dbus_message_set_no_reply (message, TRUE); 1320 1321 if (!dbus_message_set_reply_serial (message, 1322 dbus_message_get_serial (method_call))) 1323 { 1324 dbus_message_unref (message); 1325 return NULL; 1326 } 1327 1328 return message; 1329 } 1330 1331 /** 1332 * Constructs a new message representing a signal emission. Returns 1333 * #NULL if memory can't be allocated for the message. A signal is 1334 * identified by its originating object path, interface, and the name 1335 * of the signal. 1336 * 1337 * Path, interface, and signal name must all be valid (the D-Bus 1338 * specification defines the syntax of these fields). 1339 * 1340 * @param path the path to the object emitting the signal 1341 * @param interface the interface the signal is emitted from 1342 * @param name name of the signal 1343 * @returns a new DBusMessage, free with dbus_message_unref() 1344 */ 1345 DBusMessage* 1346 dbus_message_new_signal (const char *path, 1347 const char *interface, 1348 const char *name) 1349 { 1350 DBusMessage *message; 1351 1352 _dbus_return_val_if_fail (path != NULL, NULL); 1353 _dbus_return_val_if_fail (interface != NULL, NULL); 1354 _dbus_return_val_if_fail (name != NULL, NULL); 1355 _dbus_return_val_if_fail (_dbus_check_is_valid_path (path), NULL); 1356 _dbus_return_val_if_fail (_dbus_check_is_valid_interface (interface), NULL); 1357 _dbus_return_val_if_fail (_dbus_check_is_valid_member (name), NULL); 1358 1359 message = dbus_message_new_empty_header (); 1360 if (message == NULL) 1361 return NULL; 1362 1363 if (!_dbus_header_create (&message->header, 1364 DBUS_COMPILER_BYTE_ORDER, 1365 DBUS_MESSAGE_TYPE_SIGNAL, 1366 NULL, path, interface, name, NULL)) 1367 { 1368 dbus_message_unref (message); 1369 return NULL; 1370 } 1371 1372 dbus_message_set_no_reply (message, TRUE); 1373 1374 return message; 1375 } 1376 1377 /** 1378 * Creates a new message that is an error reply to another message. 1379 * Error replies are most common in response to method calls, but 1380 * can be returned in reply to any message. 1381 * 1382 * The error name must be a valid error name according to the syntax 1383 * given in the D-Bus specification. If you don't want to make 1384 * up an error name just use #DBUS_ERROR_FAILED. 1385 * 1386 * @param reply_to the message we're replying to 1387 * @param error_name the error name 1388 * @param error_message the error message string (or #NULL for none, but please give a message) 1389 * @returns a new error message object, free with dbus_message_unref() 1390 */ 1391 DBusMessage* 1392 dbus_message_new_error (DBusMessage *reply_to, 1393 const char *error_name, 1394 const char *error_message) 1395 { 1396 DBusMessage *message; 1397 const char *sender; 1398 DBusMessageIter iter; 1399 1400 _dbus_return_val_if_fail (reply_to != NULL, NULL); 1401 _dbus_return_val_if_fail (error_name != NULL, NULL); 1402 _dbus_return_val_if_fail (_dbus_check_is_valid_error_name (error_name), NULL); 1403 1404 sender = dbus_message_get_sender (reply_to); 1405 1406 /* sender may be NULL for non-message-bus case or 1407 * when the message bus is dealing with an unregistered 1408 * connection. 1409 */ 1410 message = dbus_message_new_empty_header (); 1411 if (message == NULL) 1412 return NULL; 1413 1414 if (!_dbus_header_create (&message->header, 1415 DBUS_COMPILER_BYTE_ORDER, 1416 DBUS_MESSAGE_TYPE_ERROR, 1417 sender, NULL, NULL, NULL, error_name)) 1418 { 1419 dbus_message_unref (message); 1420 return NULL; 1421 } 1422 1423 dbus_message_set_no_reply (message, TRUE); 1424 1425 if (!dbus_message_set_reply_serial (message, 1426 dbus_message_get_serial (reply_to))) 1427 { 1428 dbus_message_unref (message); 1429 return NULL; 1430 } 1431 1432 if (error_message != NULL) 1433 { 1434 dbus_message_iter_init_append (message, &iter); 1435 if (!dbus_message_iter_append_basic (&iter, 1436 DBUS_TYPE_STRING, 1437 &error_message)) 1438 { 1439 dbus_message_unref (message); 1440 return NULL; 1441 } 1442 } 1443 1444 return message; 1445 } 1446 1447 /** 1448 * Creates a new message that is an error reply to another message, allowing 1449 * you to use printf formatting. 1450 * 1451 * See dbus_message_new_error() for details - this function is the same 1452 * aside from the printf formatting. 1453 * 1454 * @todo add _DBUS_GNUC_PRINTF to this (requires moving _DBUS_GNUC_PRINTF to 1455 * public header, see DBUS_DEPRECATED for an example) 1456 * 1457 * @param reply_to the original message 1458 * @param error_name the error name 1459 * @param error_format the error message format as with printf 1460 * @param ... format string arguments 1461 * @returns a new error message 1462 */ 1463 DBusMessage* 1464 dbus_message_new_error_printf (DBusMessage *reply_to, 1465 const char *error_name, 1466 const char *error_format, 1467 ...) 1468 { 1469 va_list args; 1470 DBusString str; 1471 DBusMessage *message; 1472 1473 _dbus_return_val_if_fail (reply_to != NULL, NULL); 1474 _dbus_return_val_if_fail (error_name != NULL, NULL); 1475 _dbus_return_val_if_fail (_dbus_check_is_valid_error_name (error_name), NULL); 1476 1477 if (!_dbus_string_init (&str)) 1478 return NULL; 1479 1480 va_start (args, error_format); 1481 1482 if (_dbus_string_append_printf_valist (&str, error_format, args)) 1483 message = dbus_message_new_error (reply_to, error_name, 1484 _dbus_string_get_const_data (&str)); 1485 else 1486 message = NULL; 1487 1488 _dbus_string_free (&str); 1489 1490 va_end (args); 1491 1492 return message; 1493 } 1494 1495 1496 /** 1497 * Creates a new message that is an exact replica of the message 1498 * specified, except that its refcount is set to 1, its message serial 1499 * is reset to 0, and if the original message was "locked" (in the 1500 * outgoing message queue and thus not modifiable) the new message 1501 * will not be locked. 1502 * 1503 * @todo This function can't be used in programs that try to recover from OOM errors. 1504 * 1505 * @param message the message 1506 * @returns the new message.or #NULL if not enough memory or Unix file descriptors (in case the message to copy includes Unix file descriptors) can be allocated. 1507 */ 1508 DBusMessage * 1509 dbus_message_copy (const DBusMessage *message) 1510 { 1511 DBusMessage *retval; 1512 1513 _dbus_return_val_if_fail (message != NULL, NULL); 1514 1515 retval = dbus_new0 (DBusMessage, 1); 1516 if (retval == NULL) 1517 return NULL; 1518 1519 _dbus_atomic_inc (&retval->refcount); 1520 1521 retval->locked = FALSE; 1522 #ifndef DBUS_DISABLE_CHECKS 1523 retval->generation = message->generation; 1524 #endif 1525 1526 if (!_dbus_header_copy (&message->header, &retval->header)) 1527 { 1528 dbus_free (retval); 1529 return NULL; 1530 } 1531 1532 if (!_dbus_string_init_preallocated (&retval->body, 1533 _dbus_string_get_length (&message->body))) 1534 { 1535 _dbus_header_free (&retval->header); 1536 dbus_free (retval); 1537 return NULL; 1538 } 1539 1540 if (!_dbus_string_copy (&message->body, 0, 1541 &retval->body, 0)) 1542 goto failed_copy; 1543 1544 #ifdef HAVE_UNIX_FD_PASSING 1545 retval->unix_fds = dbus_new(int, message->n_unix_fds); 1546 if (retval->unix_fds == NULL && message->n_unix_fds > 0) 1547 goto failed_copy; 1548 1549 retval->n_unix_fds_allocated = message->n_unix_fds; 1550 1551 for (retval->n_unix_fds = 0; 1552 retval->n_unix_fds < message->n_unix_fds; 1553 retval->n_unix_fds++) 1554 { 1555 retval->unix_fds[retval->n_unix_fds] = _dbus_dup(message->unix_fds[retval->n_unix_fds], NULL); 1556 1557 if (retval->unix_fds[retval->n_unix_fds] < 0) 1558 goto failed_copy; 1559 } 1560 1561 #endif 1562 1563 _dbus_message_trace_ref (retval, 0, 1, "copy"); 1564 return retval; 1565 1566 failed_copy: 1567 _dbus_header_free (&retval->header); 1568 _dbus_string_free (&retval->body); 1569 1570 #ifdef HAVE_UNIX_FD_PASSING 1571 close_unix_fds(retval->unix_fds, &retval->n_unix_fds); 1572 dbus_free(retval->unix_fds); 1573 #endif 1574 1575 dbus_free (retval); 1576 1577 return NULL; 1578 } 1579 1580 1581 /** 1582 * Increments the reference count of a DBusMessage. 1583 * 1584 * @param message the message 1585 * @returns the message 1586 * @see dbus_message_unref 1587 */ 1588 DBusMessage * 1589 dbus_message_ref (DBusMessage *message) 1590 { 1591 dbus_int32_t old_refcount; 1592 1593 _dbus_return_val_if_fail (message != NULL, NULL); 1594 _dbus_return_val_if_fail (message->generation == _dbus_current_generation, NULL); 1595 _dbus_return_val_if_fail (!message->in_cache, NULL); 1596 1597 old_refcount = _dbus_atomic_inc (&message->refcount); 1598 _dbus_assert (old_refcount >= 1); 1599 _dbus_message_trace_ref (message, old_refcount, old_refcount + 1, "ref"); 1600 1601 return message; 1602 } 1603 1604 /** 1605 * Decrements the reference count of a DBusMessage, freeing the 1606 * message if the count reaches 0. 1607 * 1608 * @param message the message 1609 * @see dbus_message_ref 1610 */ 1611 void 1612 dbus_message_unref (DBusMessage *message) 1613 { 1614 dbus_int32_t old_refcount; 1615 1616 _dbus_return_if_fail (message != NULL); 1617 _dbus_return_if_fail (message->generation == _dbus_current_generation); 1618 _dbus_return_if_fail (!message->in_cache); 1619 1620 old_refcount = _dbus_atomic_dec (&message->refcount); 1621 1622 _dbus_assert (old_refcount >= 1); 1623 1624 _dbus_message_trace_ref (message, old_refcount, old_refcount - 1, "unref"); 1625 1626 if (old_refcount == 1) 1627 { 1628 /* Calls application callbacks! */ 1629 dbus_message_cache_or_finalize (message); 1630 } 1631 } 1632 1633 /** 1634 * Gets the type of a message. Types include 1635 * #DBUS_MESSAGE_TYPE_METHOD_CALL, #DBUS_MESSAGE_TYPE_METHOD_RETURN, 1636 * #DBUS_MESSAGE_TYPE_ERROR, #DBUS_MESSAGE_TYPE_SIGNAL, but other 1637 * types are allowed and all code must silently ignore messages of 1638 * unknown type. #DBUS_MESSAGE_TYPE_INVALID will never be returned. 1639 * 1640 * @param message the message 1641 * @returns the type of the message 1642 */ 1643 int 1644 dbus_message_get_type (DBusMessage *message) 1645 { 1646 _dbus_return_val_if_fail (message != NULL, DBUS_MESSAGE_TYPE_INVALID); 1647 1648 return _dbus_header_get_message_type (&message->header); 1649 } 1650 1651 /** 1652 * Appends fields to a message given a variable argument list. The 1653 * variable argument list should contain the type of each argument 1654 * followed by the value to append. Appendable types are basic types, 1655 * and arrays of fixed-length basic types (except arrays of Unix file 1656 * descriptors). To append variable-length basic types, or any more 1657 * complex value, you have to use an iterator rather than this 1658 * function. 1659 * 1660 * To append a basic type, specify its type code followed by the 1661 * address of the value. For example: 1662 * 1663 * @code 1664 * 1665 * dbus_int32_t v_INT32 = 42; 1666 * const char *v_STRING = "Hello World"; 1667 * dbus_message_append_args (message, 1668 * DBUS_TYPE_INT32, &v_INT32, 1669 * DBUS_TYPE_STRING, &v_STRING, 1670 * DBUS_TYPE_INVALID); 1671 * @endcode 1672 * 1673 * To append an array of fixed-length basic types (except Unix file 1674 * descriptors), pass in the DBUS_TYPE_ARRAY typecode, the element 1675 * typecode, the address of the array pointer, and a 32-bit integer 1676 * giving the number of elements in the array. So for example: @code 1677 * const dbus_int32_t array[] = { 1, 2, 3 }; const dbus_int32_t 1678 * *v_ARRAY = array; dbus_message_append_args (message, 1679 * DBUS_TYPE_ARRAY, DBUS_TYPE_INT32, &v_ARRAY, 3, DBUS_TYPE_INVALID); 1680 * @endcode 1681 * 1682 * This function does not support arrays of Unix file descriptors. If 1683 * you need those you need to manually recurse into the array. 1684 * 1685 * For Unix file descriptors this function will internally duplicate 1686 * the descriptor you passed in. Hence you may close the descriptor 1687 * immediately after this call. 1688 * 1689 * @warning in C, given "int array[]", "&array == array" (the 1690 * comp.lang.c FAQ says otherwise, but gcc and the FAQ don't agree). 1691 * So if you're using an array instead of a pointer you have to create 1692 * a pointer variable, assign the array to it, then take the address 1693 * of the pointer variable. For strings it works to write 1694 * const char *array = "Hello" and then use &array though. 1695 * 1696 * The last argument to this function must be #DBUS_TYPE_INVALID, 1697 * marking the end of the argument list. If you don't do this 1698 * then libdbus won't know to stop and will read invalid memory. 1699 * 1700 * String/signature/path arrays should be passed in as "const char*** 1701 * address_of_array" and "int n_elements" 1702 * 1703 * @todo support DBUS_TYPE_STRUCT and DBUS_TYPE_VARIANT and complex arrays 1704 * 1705 * @todo If this fails due to lack of memory, the message is hosed and 1706 * you have to start over building the whole message. 1707 * 1708 * @param message the message 1709 * @param first_arg_type type of the first argument 1710 * @param ... value of first argument, list of additional type-value pairs 1711 * @returns #TRUE on success 1712 */ 1713 dbus_bool_t 1714 dbus_message_append_args (DBusMessage *message, 1715 int first_arg_type, 1716 ...) 1717 { 1718 dbus_bool_t retval; 1719 va_list var_args; 1720 1721 _dbus_return_val_if_fail (message != NULL, FALSE); 1722 1723 va_start (var_args, first_arg_type); 1724 retval = dbus_message_append_args_valist (message, 1725 first_arg_type, 1726 var_args); 1727 va_end (var_args); 1728 1729 return retval; 1730 } 1731 1732 /** 1733 * Like dbus_message_append_args() but takes a va_list for use by language bindings. 1734 * 1735 * @todo for now, if this function fails due to OOM it will leave 1736 * the message half-written and you have to discard the message 1737 * and start over. 1738 * 1739 * @see dbus_message_append_args. 1740 * @param message the message 1741 * @param first_arg_type type of first argument 1742 * @param var_args value of first argument, then list of type/value pairs 1743 * @returns #TRUE on success 1744 */ 1745 dbus_bool_t 1746 dbus_message_append_args_valist (DBusMessage *message, 1747 int first_arg_type, 1748 va_list var_args) 1749 { 1750 int type; 1751 DBusMessageIter iter; 1752 1753 _dbus_return_val_if_fail (message != NULL, FALSE); 1754 1755 type = first_arg_type; 1756 1757 dbus_message_iter_init_append (message, &iter); 1758 1759 while (type != DBUS_TYPE_INVALID) 1760 { 1761 if (dbus_type_is_basic (type)) 1762 { 1763 const DBusBasicValue *value; 1764 value = va_arg (var_args, const DBusBasicValue*); 1765 1766 if (!dbus_message_iter_append_basic (&iter, 1767 type, 1768 value)) 1769 goto failed; 1770 } 1771 else if (type == DBUS_TYPE_ARRAY) 1772 { 1773 int element_type; 1774 DBusMessageIter array; 1775 char buf[2]; 1776 1777 element_type = va_arg (var_args, int); 1778 1779 buf[0] = element_type; 1780 buf[1] = '\0'; 1781 if (!dbus_message_iter_open_container (&iter, 1782 DBUS_TYPE_ARRAY, 1783 buf, 1784 &array)) 1785 goto failed; 1786 1787 if (dbus_type_is_fixed (element_type) && 1788 element_type != DBUS_TYPE_UNIX_FD) 1789 { 1790 const DBusBasicValue **value; 1791 int n_elements; 1792 1793 value = va_arg (var_args, const DBusBasicValue**); 1794 n_elements = va_arg (var_args, int); 1795 1796 if (!dbus_message_iter_append_fixed_array (&array, 1797 element_type, 1798 value, 1799 n_elements)) { 1800 dbus_message_iter_abandon_container (&iter, &array); 1801 goto failed; 1802 } 1803 } 1804 else if (_DBUS_TYPE_IS_STRINGLIKE (element_type)) 1805 { 1806 const char ***value_p; 1807 const char **value; 1808 int n_elements; 1809 int i; 1810 1811 value_p = va_arg (var_args, const char***); 1812 n_elements = va_arg (var_args, int); 1813 1814 value = *value_p; 1815 1816 i = 0; 1817 while (i < n_elements) 1818 { 1819 if (!dbus_message_iter_append_basic (&array, 1820 element_type, 1821 &value[i])) { 1822 dbus_message_iter_abandon_container (&iter, &array); 1823 goto failed; 1824 } 1825 ++i; 1826 } 1827 } 1828 else 1829 { 1830 _dbus_warn ("arrays of %s can't be appended with %s for now\n", 1831 _dbus_type_to_string (element_type), 1832 _DBUS_FUNCTION_NAME); 1833 goto failed; 1834 } 1835 1836 if (!dbus_message_iter_close_container (&iter, &array)) 1837 goto failed; 1838 } 1839 #ifndef DBUS_DISABLE_CHECKS 1840 else 1841 { 1842 _dbus_warn ("type %s isn't supported yet in %s\n", 1843 _dbus_type_to_string (type), _DBUS_FUNCTION_NAME); 1844 goto failed; 1845 } 1846 #endif 1847 1848 type = va_arg (var_args, int); 1849 } 1850 1851 return TRUE; 1852 1853 failed: 1854 return FALSE; 1855 } 1856 1857 /** 1858 * Gets arguments from a message given a variable argument list. The 1859 * supported types include those supported by 1860 * dbus_message_append_args(); that is, basic types and arrays of 1861 * fixed-length basic types. The arguments are the same as they would 1862 * be for dbus_message_iter_get_basic() or 1863 * dbus_message_iter_get_fixed_array(). 1864 * 1865 * In addition to those types, arrays of string, object path, and 1866 * signature are supported; but these are returned as allocated memory 1867 * and must be freed with dbus_free_string_array(), while the other 1868 * types are returned as const references. To get a string array 1869 * pass in "char ***array_location" and "int *n_elements". 1870 * 1871 * Similar to dbus_message_get_fixed_array() this function does not 1872 * support arrays of type DBUS_TYPE_UNIX_FD. If you need to parse 1873 * messages with arrays of Unix file descriptors you need to recurse 1874 * into the array manually. 1875 * 1876 * Unix file descriptors that are read with this function will have 1877 * the FD_CLOEXEC flag set. If you need them without this flag set, 1878 * make sure to unset it with fcntl(). 1879 * 1880 * The variable argument list should contain the type of the argument 1881 * followed by a pointer to where the value should be stored. The list 1882 * is terminated with #DBUS_TYPE_INVALID. 1883 * 1884 * Except for string arrays, the returned values are constant; do not 1885 * free them. They point into the #DBusMessage. 1886 * 1887 * If the requested arguments are not present, or do not have the 1888 * requested types, then an error will be set. 1889 * 1890 * If more arguments than requested are present, the requested 1891 * arguments are returned and the extra arguments are ignored. 1892 * 1893 * @todo support DBUS_TYPE_STRUCT and DBUS_TYPE_VARIANT and complex arrays 1894 * 1895 * @param message the message 1896 * @param error error to be filled in on failure 1897 * @param first_arg_type the first argument type 1898 * @param ... location for first argument value, then list of type-location pairs 1899 * @returns #FALSE if the error was set 1900 */ 1901 dbus_bool_t 1902 dbus_message_get_args (DBusMessage *message, 1903 DBusError *error, 1904 int first_arg_type, 1905 ...) 1906 { 1907 dbus_bool_t retval; 1908 va_list var_args; 1909 1910 _dbus_return_val_if_fail (message != NULL, FALSE); 1911 _dbus_return_val_if_error_is_set (error, FALSE); 1912 1913 va_start (var_args, first_arg_type); 1914 retval = dbus_message_get_args_valist (message, error, first_arg_type, var_args); 1915 va_end (var_args); 1916 1917 return retval; 1918 } 1919 1920 /** 1921 * Like dbus_message_get_args but takes a va_list for use by language bindings. 1922 * 1923 * @see dbus_message_get_args 1924 * @param message the message 1925 * @param error error to be filled in 1926 * @param first_arg_type type of the first argument 1927 * @param var_args return location for first argument, followed by list of type/location pairs 1928 * @returns #FALSE if error was set 1929 */ 1930 dbus_bool_t 1931 dbus_message_get_args_valist (DBusMessage *message, 1932 DBusError *error, 1933 int first_arg_type, 1934 va_list var_args) 1935 { 1936 DBusMessageIter iter; 1937 1938 _dbus_return_val_if_fail (message != NULL, FALSE); 1939 _dbus_return_val_if_error_is_set (error, FALSE); 1940 1941 dbus_message_iter_init (message, &iter); 1942 return _dbus_message_iter_get_args_valist (&iter, error, first_arg_type, var_args); 1943 } 1944 1945 static void 1946 _dbus_message_iter_init_common (DBusMessage *message, 1947 DBusMessageRealIter *real, 1948 int iter_type) 1949 { 1950 _dbus_assert (sizeof (DBusMessageRealIter) <= sizeof (DBusMessageIter)); 1951 1952 /* Since the iterator will read or write who-knows-what from the 1953 * message, we need to get in the right byte order 1954 */ 1955 ensure_byte_order (message); 1956 1957 real->message = message; 1958 real->changed_stamp = message->changed_stamp; 1959 real->iter_type = iter_type; 1960 real->sig_refcount = 0; 1961 } 1962 1963 /** 1964 * Initializes a #DBusMessageIter for reading the arguments of the 1965 * message passed in. 1966 * 1967 * When possible, dbus_message_get_args() is much more convenient. 1968 * Some types of argument can only be read with #DBusMessageIter 1969 * however. 1970 * 1971 * The easiest way to iterate is like this: 1972 * @code 1973 * dbus_message_iter_init (message, &iter); 1974 * while ((current_type = dbus_message_iter_get_arg_type (&iter)) != DBUS_TYPE_INVALID) 1975 * dbus_message_iter_next (&iter); 1976 * @endcode 1977 * 1978 * #DBusMessageIter contains no allocated memory; it need not be 1979 * freed, and can be copied by assignment or memcpy(). 1980 * 1981 * @param message the message 1982 * @param iter pointer to an iterator to initialize 1983 * @returns #FALSE if the message has no arguments 1984 */ 1985 dbus_bool_t 1986 dbus_message_iter_init (DBusMessage *message, 1987 DBusMessageIter *iter) 1988 { 1989 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 1990 const DBusString *type_str; 1991 int type_pos; 1992 1993 _dbus_return_val_if_fail (message != NULL, FALSE); 1994 _dbus_return_val_if_fail (iter != NULL, FALSE); 1995 1996 get_const_signature (&message->header, &type_str, &type_pos); 1997 1998 _dbus_message_iter_init_common (message, real, 1999 DBUS_MESSAGE_ITER_TYPE_READER); 2000 2001 _dbus_type_reader_init (&real->u.reader, 2002 _dbus_header_get_byte_order (&message->header), 2003 type_str, type_pos, 2004 &message->body, 2005 0); 2006 2007 return _dbus_type_reader_get_current_type (&real->u.reader) != DBUS_TYPE_INVALID; 2008 } 2009 2010 /** 2011 * Checks if an iterator has any more fields. 2012 * 2013 * @param iter the message iter 2014 * @returns #TRUE if there are more fields following 2015 */ 2016 dbus_bool_t 2017 dbus_message_iter_has_next (DBusMessageIter *iter) 2018 { 2019 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2020 2021 _dbus_return_val_if_fail (_dbus_message_iter_check (real), FALSE); 2022 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, FALSE); 2023 2024 return _dbus_type_reader_has_next (&real->u.reader); 2025 } 2026 2027 /** 2028 * Moves the iterator to the next field, if any. If there's no next 2029 * field, returns #FALSE. If the iterator moves forward, returns 2030 * #TRUE. 2031 * 2032 * @param iter the message iter 2033 * @returns #TRUE if the iterator was moved to the next field 2034 */ 2035 dbus_bool_t 2036 dbus_message_iter_next (DBusMessageIter *iter) 2037 { 2038 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2039 2040 _dbus_return_val_if_fail (_dbus_message_iter_check (real), FALSE); 2041 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, FALSE); 2042 2043 return _dbus_type_reader_next (&real->u.reader); 2044 } 2045 2046 /** 2047 * Returns the argument type of the argument that the message iterator 2048 * points to. If the iterator is at the end of the message, returns 2049 * #DBUS_TYPE_INVALID. You can thus write a loop as follows: 2050 * 2051 * @code 2052 * dbus_message_iter_init (message, &iter); 2053 * while ((current_type = dbus_message_iter_get_arg_type (&iter)) != DBUS_TYPE_INVALID) 2054 * dbus_message_iter_next (&iter); 2055 * @endcode 2056 * 2057 * @param iter the message iter 2058 * @returns the argument type 2059 */ 2060 int 2061 dbus_message_iter_get_arg_type (DBusMessageIter *iter) 2062 { 2063 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2064 2065 _dbus_return_val_if_fail (_dbus_message_iter_check (real), DBUS_TYPE_INVALID); 2066 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, FALSE); 2067 2068 return _dbus_type_reader_get_current_type (&real->u.reader); 2069 } 2070 2071 /** 2072 * Returns the element type of the array that the message iterator 2073 * points to. Note that you need to check that the iterator points to 2074 * an array prior to using this function. 2075 * 2076 * @param iter the message iter 2077 * @returns the array element type 2078 */ 2079 int 2080 dbus_message_iter_get_element_type (DBusMessageIter *iter) 2081 { 2082 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2083 2084 _dbus_return_val_if_fail (_dbus_message_iter_check (real), DBUS_TYPE_INVALID); 2085 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, DBUS_TYPE_INVALID); 2086 _dbus_return_val_if_fail (dbus_message_iter_get_arg_type (iter) == DBUS_TYPE_ARRAY, DBUS_TYPE_INVALID); 2087 2088 return _dbus_type_reader_get_element_type (&real->u.reader); 2089 } 2090 2091 /** 2092 * Recurses into a container value when reading values from a message, 2093 * initializing a sub-iterator to use for traversing the child values 2094 * of the container. 2095 * 2096 * Note that this recurses into a value, not a type, so you can only 2097 * recurse if the value exists. The main implication of this is that 2098 * if you have for example an empty array of array of int32, you can 2099 * recurse into the outermost array, but it will have no values, so 2100 * you won't be able to recurse further. There's no array of int32 to 2101 * recurse into. 2102 * 2103 * If a container is an array of fixed-length types (except Unix file 2104 * descriptors), it is much more efficient to use 2105 * dbus_message_iter_get_fixed_array() to get the whole array in one 2106 * shot, rather than individually walking over the array elements. 2107 * 2108 * Be sure you have somehow checked that 2109 * dbus_message_iter_get_arg_type() matches the type you are expecting 2110 * to recurse into. Results of this function are undefined if there is 2111 * no container to recurse into at the current iterator position. 2112 * 2113 * @param iter the message iterator 2114 * @param sub the sub-iterator to initialize 2115 */ 2116 void 2117 dbus_message_iter_recurse (DBusMessageIter *iter, 2118 DBusMessageIter *sub) 2119 { 2120 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2121 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub; 2122 2123 _dbus_return_if_fail (_dbus_message_iter_check (real)); 2124 _dbus_return_if_fail (sub != NULL); 2125 2126 *real_sub = *real; 2127 _dbus_type_reader_recurse (&real->u.reader, &real_sub->u.reader); 2128 } 2129 2130 /** 2131 * Returns the current signature of a message iterator. This 2132 * is useful primarily for dealing with variants; one can 2133 * recurse into a variant and determine the signature of 2134 * the variant's value. 2135 * 2136 * The returned string must be freed with dbus_free(). 2137 * 2138 * @param iter the message iterator 2139 * @returns the contained signature, or NULL if out of memory 2140 */ 2141 char * 2142 dbus_message_iter_get_signature (DBusMessageIter *iter) 2143 { 2144 const DBusString *sig; 2145 DBusString retstr; 2146 char *ret; 2147 int start, len; 2148 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2149 2150 _dbus_return_val_if_fail (_dbus_message_iter_check (real), NULL); 2151 2152 if (!_dbus_string_init (&retstr)) 2153 return NULL; 2154 2155 _dbus_type_reader_get_signature (&real->u.reader, &sig, 2156 &start, &len); 2157 if (!_dbus_string_append_len (&retstr, 2158 _dbus_string_get_const_data (sig) + start, 2159 len)) 2160 return NULL; 2161 if (!_dbus_string_steal_data (&retstr, &ret)) 2162 return NULL; 2163 _dbus_string_free (&retstr); 2164 return ret; 2165 } 2166 2167 /** 2168 * Reads a basic-typed value from the message iterator. 2169 * Basic types are the non-containers such as integer and string. 2170 * 2171 * The value argument should be the address of a location to store 2172 * the returned value. So for int32 it should be a "dbus_int32_t*" 2173 * and for string a "const char**". The returned value is 2174 * by reference and should not be freed. 2175 * 2176 * This call duplicates Unix file descriptors when reading them. It is 2177 * your job to close them when you don't need them anymore. 2178 * 2179 * Unix file descriptors that are read with this function will have 2180 * the FD_CLOEXEC flag set. If you need them without this flag set, 2181 * make sure to unset it with fcntl(). 2182 * 2183 * Be sure you have somehow checked that 2184 * dbus_message_iter_get_arg_type() matches the type you are 2185 * expecting, or you'll crash when you try to use an integer as a 2186 * string or something. 2187 * 2188 * To read any container type (array, struct, dict) you will need to 2189 * recurse into the container with dbus_message_iter_recurse(). If 2190 * the container is an array of fixed-length values (except Unix file 2191 * descriptors), you can get all the array elements at once with 2192 * dbus_message_iter_get_fixed_array(). Otherwise, you have to iterate 2193 * over the container's contents one value at a time. 2194 * 2195 * All basic-typed values are guaranteed to fit in a #DBusBasicValue, 2196 * so in versions of libdbus that have that type, you can write code like this: 2197 * 2198 * @code 2199 * DBusBasicValue value; 2200 * int type; 2201 * dbus_message_iter_get_basic (&read_iter, &value); 2202 * type = dbus_message_iter_get_arg_type (&read_iter); 2203 * dbus_message_iter_append_basic (&write_iter, type, &value); 2204 * @endcode 2205 * 2206 * (All D-Bus basic types are either numeric and 8 bytes or smaller, or 2207 * behave like a string; so in older versions of libdbus, DBusBasicValue 2208 * can be replaced with union { char *string; unsigned char bytes[8]; }, 2209 * for instance.) 2210 * 2211 * @param iter the iterator 2212 * @param value location to store the value 2213 */ 2214 void 2215 dbus_message_iter_get_basic (DBusMessageIter *iter, 2216 void *value) 2217 { 2218 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2219 2220 _dbus_return_if_fail (_dbus_message_iter_check (real)); 2221 _dbus_return_if_fail (value != NULL); 2222 2223 if (dbus_message_iter_get_arg_type (iter) == DBUS_TYPE_UNIX_FD) 2224 { 2225 #ifdef HAVE_UNIX_FD_PASSING 2226 DBusBasicValue idx; 2227 2228 _dbus_type_reader_read_basic(&real->u.reader, &idx); 2229 2230 if (idx.u32 >= real->message->n_unix_fds) { 2231 /* Hmm, we cannot really signal an error here, so let's make 2232 sure to return an invalid fd. */ 2233 *((int*) value) = -1; 2234 return; 2235 } 2236 2237 *((int*) value) = _dbus_dup(real->message->unix_fds[idx.u32], NULL); 2238 #else 2239 *((int*) value) = -1; 2240 #endif 2241 } 2242 else 2243 { 2244 _dbus_type_reader_read_basic (&real->u.reader, 2245 value); 2246 } 2247 } 2248 2249 /** 2250 * Returns the number of bytes in the array as marshaled in the wire 2251 * protocol. The iterator must currently be inside an array-typed 2252 * value. 2253 * 2254 * This function is deprecated on the grounds that it is stupid. Why 2255 * would you want to know how many bytes are in the array as marshaled 2256 * in the wire protocol? For now, use the n_elements returned from 2257 * dbus_message_iter_get_fixed_array() instead, or iterate over the 2258 * array values and count them. 2259 * 2260 * @todo introduce a variant of this get_n_elements that returns 2261 * the number of elements, though with a non-fixed array it will not 2262 * be very efficient, so maybe it's not good. 2263 * 2264 * @param iter the iterator 2265 * @returns the number of bytes in the array 2266 */ 2267 int 2268 dbus_message_iter_get_array_len (DBusMessageIter *iter) 2269 { 2270 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2271 2272 _dbus_return_val_if_fail (_dbus_message_iter_check (real), 0); 2273 2274 return _dbus_type_reader_get_array_length (&real->u.reader); 2275 } 2276 2277 /** 2278 * Reads a block of fixed-length values from the message iterator. 2279 * Fixed-length values are those basic types that are not string-like, 2280 * such as integers, bool, double. The returned block will be from the 2281 * current position in the array until the end of the array. 2282 * 2283 * There is one exception here: although DBUS_TYPE_UNIX_FD is 2284 * considered a 'fixed' type arrays of this type may not be read with 2285 * this function. 2286 * 2287 * The message iter should be "in" the array (that is, you recurse into the 2288 * array, and then you call dbus_message_iter_get_fixed_array() on the 2289 * "sub-iterator" created by dbus_message_iter_recurse()). 2290 * 2291 * The value argument should be the address of a location to store the 2292 * returned array. So for int32 it should be a "const dbus_int32_t**" 2293 * The returned value is by reference and should not be freed. 2294 * 2295 * This function should only be used if dbus_type_is_fixed() returns 2296 * #TRUE for the element type. 2297 * 2298 * If an array's elements are not fixed in size, you have to recurse 2299 * into the array with dbus_message_iter_recurse() and read the 2300 * elements one by one. 2301 * 2302 * Because the array is not copied, this function runs in constant 2303 * time and is fast; it's much preferred over walking the entire array 2304 * with an iterator. (However, you can always use 2305 * dbus_message_iter_recurse(), even for fixed-length types; 2306 * dbus_message_iter_get_fixed_array() is just an optimization.) 2307 * 2308 * @param iter the iterator 2309 * @param value location to store the block 2310 * @param n_elements number of elements in the block 2311 */ 2312 void 2313 dbus_message_iter_get_fixed_array (DBusMessageIter *iter, 2314 void *value, 2315 int *n_elements) 2316 { 2317 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2318 #ifndef DBUS_DISABLE_CHECKS 2319 int subtype = _dbus_type_reader_get_current_type(&real->u.reader); 2320 2321 _dbus_return_if_fail (_dbus_message_iter_check (real)); 2322 _dbus_return_if_fail (value != NULL); 2323 _dbus_return_if_fail ((subtype == DBUS_TYPE_INVALID) || 2324 (dbus_type_is_fixed (subtype) && subtype != DBUS_TYPE_UNIX_FD)); 2325 #endif 2326 2327 _dbus_type_reader_read_fixed_multi (&real->u.reader, 2328 value, n_elements); 2329 } 2330 2331 /** 2332 * Initializes a #DBusMessageIter for appending arguments to the end 2333 * of a message. 2334 * 2335 * @todo If appending any of the arguments fails due to lack of 2336 * memory, the message is hosed and you have to start over building 2337 * the whole message. 2338 * 2339 * @param message the message 2340 * @param iter pointer to an iterator to initialize 2341 */ 2342 void 2343 dbus_message_iter_init_append (DBusMessage *message, 2344 DBusMessageIter *iter) 2345 { 2346 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2347 2348 _dbus_return_if_fail (message != NULL); 2349 _dbus_return_if_fail (iter != NULL); 2350 2351 _dbus_message_iter_init_common (message, real, 2352 DBUS_MESSAGE_ITER_TYPE_WRITER); 2353 2354 /* We create the signature string and point iterators at it "on demand" 2355 * when a value is actually appended. That means that init() never fails 2356 * due to OOM. 2357 */ 2358 _dbus_type_writer_init_types_delayed (&real->u.writer, 2359 _dbus_header_get_byte_order (&message->header), 2360 &message->body, 2361 _dbus_string_get_length (&message->body)); 2362 } 2363 2364 /** 2365 * Creates a temporary signature string containing the current 2366 * signature, stores it in the iterator, and points the iterator to 2367 * the end of it. Used any time we write to the message. 2368 * 2369 * @param real an iterator without a type_str 2370 * @returns #FALSE if no memory 2371 */ 2372 static dbus_bool_t 2373 _dbus_message_iter_open_signature (DBusMessageRealIter *real) 2374 { 2375 DBusString *str; 2376 const DBusString *current_sig; 2377 int current_sig_pos; 2378 2379 _dbus_assert (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER); 2380 2381 if (real->u.writer.type_str != NULL) 2382 { 2383 _dbus_assert (real->sig_refcount > 0); 2384 real->sig_refcount += 1; 2385 return TRUE; 2386 } 2387 2388 str = dbus_new (DBusString, 1); 2389 if (str == NULL) 2390 return FALSE; 2391 2392 if (!_dbus_header_get_field_raw (&real->message->header, 2393 DBUS_HEADER_FIELD_SIGNATURE, 2394 ¤t_sig, ¤t_sig_pos)) 2395 current_sig = NULL; 2396 2397 if (current_sig) 2398 { 2399 int current_len; 2400 2401 current_len = _dbus_string_get_byte (current_sig, current_sig_pos); 2402 current_sig_pos += 1; /* move on to sig data */ 2403 2404 if (!_dbus_string_init_preallocated (str, current_len + 4)) 2405 { 2406 dbus_free (str); 2407 return FALSE; 2408 } 2409 2410 if (!_dbus_string_copy_len (current_sig, current_sig_pos, current_len, 2411 str, 0)) 2412 { 2413 _dbus_string_free (str); 2414 dbus_free (str); 2415 return FALSE; 2416 } 2417 } 2418 else 2419 { 2420 if (!_dbus_string_init_preallocated (str, 4)) 2421 { 2422 dbus_free (str); 2423 return FALSE; 2424 } 2425 } 2426 2427 real->sig_refcount = 1; 2428 2429 _dbus_type_writer_add_types (&real->u.writer, 2430 str, _dbus_string_get_length (str)); 2431 return TRUE; 2432 } 2433 2434 /** 2435 * Sets the new signature as the message signature, frees the 2436 * signature string, and marks the iterator as not having a type_str 2437 * anymore. Frees the signature even if it fails, so you can't 2438 * really recover from failure. Kinda busted. 2439 * 2440 * @param real an iterator without a type_str 2441 * @returns #FALSE if no memory 2442 */ 2443 static dbus_bool_t 2444 _dbus_message_iter_close_signature (DBusMessageRealIter *real) 2445 { 2446 DBusString *str; 2447 const char *v_STRING; 2448 dbus_bool_t retval; 2449 2450 _dbus_assert (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER); 2451 _dbus_assert (real->u.writer.type_str != NULL); 2452 _dbus_assert (real->sig_refcount > 0); 2453 2454 real->sig_refcount -= 1; 2455 2456 if (real->sig_refcount > 0) 2457 return TRUE; 2458 _dbus_assert (real->sig_refcount == 0); 2459 2460 retval = TRUE; 2461 2462 str = real->u.writer.type_str; 2463 2464 v_STRING = _dbus_string_get_const_data (str); 2465 if (!_dbus_header_set_field_basic (&real->message->header, 2466 DBUS_HEADER_FIELD_SIGNATURE, 2467 DBUS_TYPE_SIGNATURE, 2468 &v_STRING)) 2469 retval = FALSE; 2470 2471 _dbus_type_writer_remove_types (&real->u.writer); 2472 _dbus_string_free (str); 2473 dbus_free (str); 2474 2475 return retval; 2476 } 2477 2478 /** 2479 * Frees the signature string and marks the iterator as not having a 2480 * type_str anymore. Since the new signature is not set, the message 2481 * will generally be hosed after this is called. 2482 * 2483 * @param real an iterator without a type_str 2484 */ 2485 static void 2486 _dbus_message_iter_abandon_signature (DBusMessageRealIter *real) 2487 { 2488 DBusString *str; 2489 2490 _dbus_assert (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER); 2491 _dbus_assert (real->u.writer.type_str != NULL); 2492 _dbus_assert (real->sig_refcount > 0); 2493 2494 real->sig_refcount -= 1; 2495 2496 if (real->sig_refcount > 0) 2497 return; 2498 _dbus_assert (real->sig_refcount == 0); 2499 2500 str = real->u.writer.type_str; 2501 2502 _dbus_type_writer_remove_types (&real->u.writer); 2503 _dbus_string_free (str); 2504 dbus_free (str); 2505 } 2506 2507 #ifndef DBUS_DISABLE_CHECKS 2508 static dbus_bool_t 2509 _dbus_message_iter_append_check (DBusMessageRealIter *iter) 2510 { 2511 if (!_dbus_message_iter_check (iter)) 2512 return FALSE; 2513 2514 if (iter->message->locked) 2515 { 2516 _dbus_warn_check_failed ("dbus append iterator can't be used: message is locked (has already been sent)\n"); 2517 return FALSE; 2518 } 2519 2520 return TRUE; 2521 } 2522 #endif /* DBUS_DISABLE_CHECKS */ 2523 2524 #ifdef HAVE_UNIX_FD_PASSING 2525 static int * 2526 expand_fd_array(DBusMessage *m, 2527 unsigned n) 2528 { 2529 _dbus_assert(m); 2530 2531 /* This makes space for adding n new fds to the array and returns a 2532 pointer to the place were the first fd should be put. */ 2533 2534 if (m->n_unix_fds + n > m->n_unix_fds_allocated) 2535 { 2536 unsigned k; 2537 int *p; 2538 2539 /* Make twice as much space as necessary */ 2540 k = (m->n_unix_fds + n) * 2; 2541 2542 /* Allocate at least four */ 2543 if (k < 4) 2544 k = 4; 2545 2546 p = dbus_realloc(m->unix_fds, k * sizeof(int)); 2547 if (p == NULL) 2548 return NULL; 2549 2550 m->unix_fds = p; 2551 m->n_unix_fds_allocated = k; 2552 } 2553 2554 return m->unix_fds + m->n_unix_fds; 2555 } 2556 #endif 2557 2558 /** 2559 * Appends a basic-typed value to the message. The basic types are the 2560 * non-container types such as integer and string. 2561 * 2562 * The "value" argument should be the address of a basic-typed value. 2563 * So for string, const char**. For integer, dbus_int32_t*. 2564 * 2565 * For Unix file descriptors this function will internally duplicate 2566 * the descriptor you passed in. Hence you may close the descriptor 2567 * immediately after this call. 2568 * 2569 * @todo If this fails due to lack of memory, the message is hosed and 2570 * you have to start over building the whole message. 2571 * 2572 * @param iter the append iterator 2573 * @param type the type of the value 2574 * @param value the address of the value 2575 * @returns #FALSE if not enough memory 2576 */ 2577 dbus_bool_t 2578 dbus_message_iter_append_basic (DBusMessageIter *iter, 2579 int type, 2580 const void *value) 2581 { 2582 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2583 dbus_bool_t ret; 2584 2585 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE); 2586 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE); 2587 _dbus_return_val_if_fail (dbus_type_is_basic (type), FALSE); 2588 _dbus_return_val_if_fail (value != NULL, FALSE); 2589 2590 #ifndef DBUS_DISABLE_CHECKS 2591 switch (type) 2592 { 2593 const char * const *string_p; 2594 const dbus_bool_t *bool_p; 2595 2596 case DBUS_TYPE_STRING: 2597 string_p = value; 2598 _dbus_return_val_if_fail (_dbus_check_is_valid_utf8 (*string_p), FALSE); 2599 break; 2600 2601 case DBUS_TYPE_OBJECT_PATH: 2602 string_p = value; 2603 _dbus_return_val_if_fail (_dbus_check_is_valid_path (*string_p), FALSE); 2604 break; 2605 2606 case DBUS_TYPE_SIGNATURE: 2607 string_p = value; 2608 _dbus_return_val_if_fail (_dbus_check_is_valid_signature (*string_p), FALSE); 2609 break; 2610 2611 case DBUS_TYPE_BOOLEAN: 2612 bool_p = value; 2613 _dbus_return_val_if_fail (*bool_p == 0 || *bool_p == 1, FALSE); 2614 break; 2615 2616 default: 2617 { 2618 /* nothing to check, all possible values are allowed */ 2619 } 2620 } 2621 #endif 2622 2623 if (!_dbus_message_iter_open_signature (real)) 2624 return FALSE; 2625 2626 if (type == DBUS_TYPE_UNIX_FD) 2627 { 2628 #ifdef HAVE_UNIX_FD_PASSING 2629 int *fds; 2630 dbus_uint32_t u; 2631 2632 /* First step, include the fd in the fd list of this message */ 2633 if (!(fds = expand_fd_array(real->message, 1))) 2634 return FALSE; 2635 2636 *fds = _dbus_dup(*(int*) value, NULL); 2637 if (*fds < 0) 2638 return FALSE; 2639 2640 u = real->message->n_unix_fds; 2641 2642 /* Second step, write the index to the fd */ 2643 if (!(ret = _dbus_type_writer_write_basic (&real->u.writer, DBUS_TYPE_UNIX_FD, &u))) { 2644 _dbus_close(*fds, NULL); 2645 return FALSE; 2646 } 2647 2648 real->message->n_unix_fds += 1; 2649 u += 1; 2650 2651 /* Final step, update the header accordingly */ 2652 ret = _dbus_header_set_field_basic (&real->message->header, 2653 DBUS_HEADER_FIELD_UNIX_FDS, 2654 DBUS_TYPE_UINT32, 2655 &u); 2656 2657 /* If any of these operations fail the message is 2658 hosed. However, no memory or fds should be leaked since what 2659 has been added to message has been added to the message, and 2660 can hence be accounted for when the message is being 2661 freed. */ 2662 #else 2663 ret = FALSE; 2664 #endif 2665 } 2666 else 2667 { 2668 ret = _dbus_type_writer_write_basic (&real->u.writer, type, value); 2669 } 2670 2671 if (!_dbus_message_iter_close_signature (real)) 2672 ret = FALSE; 2673 2674 return ret; 2675 } 2676 2677 /** 2678 * Appends a block of fixed-length values to an array. The 2679 * fixed-length types are all basic types that are not string-like. So 2680 * int32, double, bool, etc. (Unix file descriptors however are not 2681 * supported.) You must call dbus_message_iter_open_container() to 2682 * open an array of values before calling this function. You may call 2683 * this function multiple times (and intermixed with calls to 2684 * dbus_message_iter_append_basic()) for the same array. 2685 * 2686 * The "value" argument should be the address of the array. So for 2687 * integer, "dbus_int32_t**" is expected for example. 2688 * 2689 * @warning in C, given "int array[]", "&array == array" (the 2690 * comp.lang.c FAQ says otherwise, but gcc and the FAQ don't agree). 2691 * So if you're using an array instead of a pointer you have to create 2692 * a pointer variable, assign the array to it, then take the address 2693 * of the pointer variable. 2694 * @code 2695 * const dbus_int32_t array[] = { 1, 2, 3 }; 2696 * const dbus_int32_t *v_ARRAY = array; 2697 * if (!dbus_message_iter_append_fixed_array (&iter, DBUS_TYPE_INT32, &v_ARRAY, 3)) 2698 * fprintf (stderr, "No memory!\n"); 2699 * @endcode 2700 * For strings it works to write const char *array = "Hello" and then 2701 * use &array though. 2702 * 2703 * @todo If this fails due to lack of memory, the message is hosed and 2704 * you have to start over building the whole message. 2705 * 2706 * @param iter the append iterator 2707 * @param element_type the type of the array elements 2708 * @param value the address of the array 2709 * @param n_elements the number of elements to append 2710 * @returns #FALSE if not enough memory 2711 */ 2712 dbus_bool_t 2713 dbus_message_iter_append_fixed_array (DBusMessageIter *iter, 2714 int element_type, 2715 const void *value, 2716 int n_elements) 2717 { 2718 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2719 dbus_bool_t ret; 2720 2721 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE); 2722 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE); 2723 _dbus_return_val_if_fail (dbus_type_is_fixed (element_type) && element_type != DBUS_TYPE_UNIX_FD, FALSE); 2724 _dbus_return_val_if_fail (real->u.writer.container_type == DBUS_TYPE_ARRAY, FALSE); 2725 _dbus_return_val_if_fail (value != NULL, FALSE); 2726 _dbus_return_val_if_fail (n_elements >= 0, FALSE); 2727 _dbus_return_val_if_fail (n_elements <= 2728 DBUS_MAXIMUM_ARRAY_LENGTH / _dbus_type_get_alignment (element_type), 2729 FALSE); 2730 2731 #ifndef DBUS_DISABLE_CHECKS 2732 if (element_type == DBUS_TYPE_BOOLEAN) 2733 { 2734 const dbus_bool_t * const *bools = value; 2735 int i; 2736 2737 for (i = 0; i < n_elements; i++) 2738 { 2739 _dbus_return_val_if_fail ((*bools)[i] == 0 || (*bools)[i] == 1, FALSE); 2740 } 2741 } 2742 #endif 2743 2744 ret = _dbus_type_writer_write_fixed_multi (&real->u.writer, element_type, value, n_elements); 2745 2746 return ret; 2747 } 2748 2749 /** 2750 * Appends a container-typed value to the message; you are required to 2751 * append the contents of the container using the returned 2752 * sub-iterator, and then call 2753 * dbus_message_iter_close_container(). Container types are for 2754 * example struct, variant, and array. For variants, the 2755 * contained_signature should be the type of the single value inside 2756 * the variant. For structs and dict entries, contained_signature 2757 * should be #NULL; it will be set to whatever types you write into 2758 * the struct. For arrays, contained_signature should be the type of 2759 * the array elements. 2760 * 2761 * @todo If this fails due to lack of memory, the message is hosed and 2762 * you have to start over building the whole message. 2763 * 2764 * @param iter the append iterator 2765 * @param type the type of the value 2766 * @param contained_signature the type of container contents 2767 * @param sub sub-iterator to initialize 2768 * @returns #FALSE if not enough memory 2769 */ 2770 dbus_bool_t 2771 dbus_message_iter_open_container (DBusMessageIter *iter, 2772 int type, 2773 const char *contained_signature, 2774 DBusMessageIter *sub) 2775 { 2776 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2777 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub; 2778 DBusString contained_str; 2779 2780 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE); 2781 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE); 2782 _dbus_return_val_if_fail (dbus_type_is_container (type), FALSE); 2783 _dbus_return_val_if_fail (sub != NULL, FALSE); 2784 _dbus_return_val_if_fail ((type == DBUS_TYPE_STRUCT && 2785 contained_signature == NULL) || 2786 (type == DBUS_TYPE_DICT_ENTRY && 2787 contained_signature == NULL) || 2788 (type == DBUS_TYPE_VARIANT && 2789 contained_signature != NULL) || 2790 (type == DBUS_TYPE_ARRAY && 2791 contained_signature != NULL), FALSE); 2792 2793 /* this would fail if the contained_signature is a dict entry, since 2794 * dict entries are invalid signatures standalone (they must be in 2795 * an array) 2796 */ 2797 _dbus_return_val_if_fail ((type == DBUS_TYPE_ARRAY && contained_signature && *contained_signature == DBUS_DICT_ENTRY_BEGIN_CHAR) || 2798 (contained_signature == NULL || 2799 _dbus_check_is_valid_signature (contained_signature)), 2800 FALSE); 2801 2802 if (!_dbus_message_iter_open_signature (real)) 2803 return FALSE; 2804 2805 *real_sub = *real; 2806 2807 if (contained_signature != NULL) 2808 { 2809 _dbus_string_init_const (&contained_str, contained_signature); 2810 2811 return _dbus_type_writer_recurse (&real->u.writer, 2812 type, 2813 &contained_str, 0, 2814 &real_sub->u.writer); 2815 } 2816 else 2817 { 2818 return _dbus_type_writer_recurse (&real->u.writer, 2819 type, 2820 NULL, 0, 2821 &real_sub->u.writer); 2822 } 2823 } 2824 2825 2826 /** 2827 * Closes a container-typed value appended to the message; may write 2828 * out more information to the message known only after the entire 2829 * container is written, and may free resources created by 2830 * dbus_message_iter_open_container(). 2831 * 2832 * @todo If this fails due to lack of memory, the message is hosed and 2833 * you have to start over building the whole message. 2834 * 2835 * @param iter the append iterator 2836 * @param sub sub-iterator to close 2837 * @returns #FALSE if not enough memory 2838 */ 2839 dbus_bool_t 2840 dbus_message_iter_close_container (DBusMessageIter *iter, 2841 DBusMessageIter *sub) 2842 { 2843 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2844 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub; 2845 dbus_bool_t ret; 2846 2847 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE); 2848 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE); 2849 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real_sub), FALSE); 2850 _dbus_return_val_if_fail (real_sub->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE); 2851 2852 ret = _dbus_type_writer_unrecurse (&real->u.writer, 2853 &real_sub->u.writer); 2854 2855 if (!_dbus_message_iter_close_signature (real)) 2856 ret = FALSE; 2857 2858 return ret; 2859 } 2860 2861 /** 2862 * Abandons creation of a contained-typed value and frees resources created 2863 * by dbus_message_iter_open_container(). Once this returns, the message 2864 * is hosed and you have to start over building the whole message. 2865 * 2866 * This should only be used to abandon creation of a message when you have 2867 * open containers. 2868 * 2869 * @param iter the append iterator 2870 * @param sub sub-iterator to close 2871 */ 2872 void 2873 dbus_message_iter_abandon_container (DBusMessageIter *iter, 2874 DBusMessageIter *sub) 2875 { 2876 DBusMessageRealIter *real = (DBusMessageRealIter *)iter; 2877 #ifndef DBUS_DISABLE_CHECKS 2878 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub; 2879 2880 _dbus_return_if_fail (_dbus_message_iter_append_check (real)); 2881 _dbus_return_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER); 2882 _dbus_return_if_fail (_dbus_message_iter_append_check (real_sub)); 2883 _dbus_return_if_fail (real_sub->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER); 2884 #endif 2885 2886 _dbus_message_iter_abandon_signature (real); 2887 } 2888 2889 /** 2890 * Sets a flag indicating that the message does not want a reply; if 2891 * this flag is set, the other end of the connection may (but is not 2892 * required to) optimize by not sending method return or error 2893 * replies. If this flag is set, there is no way to know whether the 2894 * message successfully arrived at the remote end. Normally you know a 2895 * message was received when you receive the reply to it. 2896 * 2897 * The flag is #FALSE by default, that is by default the other end is 2898 * required to reply. 2899 * 2900 * On the protocol level this toggles #DBUS_HEADER_FLAG_NO_REPLY_EXPECTED 2901 * 2902 * @param message the message 2903 * @param no_reply #TRUE if no reply is desired 2904 */ 2905 void 2906 dbus_message_set_no_reply (DBusMessage *message, 2907 dbus_bool_t no_reply) 2908 { 2909 _dbus_return_if_fail (message != NULL); 2910 _dbus_return_if_fail (!message->locked); 2911 2912 _dbus_header_toggle_flag (&message->header, 2913 DBUS_HEADER_FLAG_NO_REPLY_EXPECTED, 2914 no_reply); 2915 } 2916 2917 /** 2918 * Returns #TRUE if the message does not expect 2919 * a reply. 2920 * 2921 * @param message the message 2922 * @returns #TRUE if the message sender isn't waiting for a reply 2923 */ 2924 dbus_bool_t 2925 dbus_message_get_no_reply (DBusMessage *message) 2926 { 2927 _dbus_return_val_if_fail (message != NULL, FALSE); 2928 2929 return _dbus_header_get_flag (&message->header, 2930 DBUS_HEADER_FLAG_NO_REPLY_EXPECTED); 2931 } 2932 2933 /** 2934 * Sets a flag indicating that an owner for the destination name will 2935 * be automatically started before the message is delivered. When this 2936 * flag is set, the message is held until a name owner finishes 2937 * starting up, or fails to start up. In case of failure, the reply 2938 * will be an error. 2939 * 2940 * The flag is set to #TRUE by default, i.e. auto starting is the default. 2941 * 2942 * On the protocol level this toggles #DBUS_HEADER_FLAG_NO_AUTO_START 2943 * 2944 * @param message the message 2945 * @param auto_start #TRUE if auto-starting is desired 2946 */ 2947 void 2948 dbus_message_set_auto_start (DBusMessage *message, 2949 dbus_bool_t auto_start) 2950 { 2951 _dbus_return_if_fail (message != NULL); 2952 _dbus_return_if_fail (!message->locked); 2953 2954 _dbus_header_toggle_flag (&message->header, 2955 DBUS_HEADER_FLAG_NO_AUTO_START, 2956 !auto_start); 2957 } 2958 2959 /** 2960 * Returns #TRUE if the message will cause an owner for 2961 * destination name to be auto-started. 2962 * 2963 * @param message the message 2964 * @returns #TRUE if the message will use auto-start 2965 */ 2966 dbus_bool_t 2967 dbus_message_get_auto_start (DBusMessage *message) 2968 { 2969 _dbus_return_val_if_fail (message != NULL, FALSE); 2970 2971 return !_dbus_header_get_flag (&message->header, 2972 DBUS_HEADER_FLAG_NO_AUTO_START); 2973 } 2974 2975 2976 /** 2977 * Sets the object path this message is being sent to (for 2978 * DBUS_MESSAGE_TYPE_METHOD_CALL) or the one a signal is being 2979 * emitted from (for DBUS_MESSAGE_TYPE_SIGNAL). 2980 * 2981 * The path must contain only valid characters as defined 2982 * in the D-Bus specification. 2983 * 2984 * @param message the message 2985 * @param object_path the path or #NULL to unset 2986 * @returns #FALSE if not enough memory 2987 */ 2988 dbus_bool_t 2989 dbus_message_set_path (DBusMessage *message, 2990 const char *object_path) 2991 { 2992 _dbus_return_val_if_fail (message != NULL, FALSE); 2993 _dbus_return_val_if_fail (!message->locked, FALSE); 2994 _dbus_return_val_if_fail (object_path == NULL || 2995 _dbus_check_is_valid_path (object_path), 2996 FALSE); 2997 2998 return set_or_delete_string_field (message, 2999 DBUS_HEADER_FIELD_PATH, 3000 DBUS_TYPE_OBJECT_PATH, 3001 object_path); 3002 } 3003 3004 /** 3005 * Gets the object path this message is being sent to (for 3006 * DBUS_MESSAGE_TYPE_METHOD_CALL) or being emitted from (for 3007 * DBUS_MESSAGE_TYPE_SIGNAL). Returns #NULL if none. 3008 * 3009 * See also dbus_message_get_path_decomposed(). 3010 * 3011 * The returned string becomes invalid if the message is 3012 * modified, since it points into the wire-marshaled message data. 3013 * 3014 * @param message the message 3015 * @returns the path (should not be freed) or #NULL 3016 */ 3017 const char* 3018 dbus_message_get_path (DBusMessage *message) 3019 { 3020 const char *v; 3021 3022 _dbus_return_val_if_fail (message != NULL, NULL); 3023 3024 v = NULL; /* in case field doesn't exist */ 3025 _dbus_header_get_field_basic (&message->header, 3026 DBUS_HEADER_FIELD_PATH, 3027 DBUS_TYPE_OBJECT_PATH, 3028 (void *) &v); 3029 return v; 3030 } 3031 3032 /** 3033 * Checks if the message has a particular object path. The object 3034 * path is the destination object for a method call or the emitting 3035 * object for a signal. 3036 * 3037 * @param message the message 3038 * @param path the path name 3039 * @returns #TRUE if there is a path field in the header 3040 */ 3041 dbus_bool_t 3042 dbus_message_has_path (DBusMessage *message, 3043 const char *path) 3044 { 3045 const char *msg_path; 3046 msg_path = dbus_message_get_path (message); 3047 3048 if (msg_path == NULL) 3049 { 3050 if (path == NULL) 3051 return TRUE; 3052 else 3053 return FALSE; 3054 } 3055 3056 if (path == NULL) 3057 return FALSE; 3058 3059 if (strcmp (msg_path, path) == 0) 3060 return TRUE; 3061 3062 return FALSE; 3063 } 3064 3065 /** 3066 * Gets the object path this message is being sent to 3067 * (for DBUS_MESSAGE_TYPE_METHOD_CALL) or being emitted 3068 * from (for DBUS_MESSAGE_TYPE_SIGNAL) in a decomposed 3069 * format (one array element per path component). 3070 * Free the returned array with dbus_free_string_array(). 3071 * 3072 * An empty but non-NULL path array means the path "/". 3073 * So the path "/foo/bar" becomes { "foo", "bar", NULL } 3074 * and the path "/" becomes { NULL }. 3075 * 3076 * See also dbus_message_get_path(). 3077 * 3078 * @todo this could be optimized by using the len from the message 3079 * instead of calling strlen() again 3080 * 3081 * @param message the message 3082 * @param path place to store allocated array of path components; #NULL set here if no path field exists 3083 * @returns #FALSE if no memory to allocate the array 3084 */ 3085 dbus_bool_t 3086 dbus_message_get_path_decomposed (DBusMessage *message, 3087 char ***path) 3088 { 3089 const char *v; 3090 3091 _dbus_return_val_if_fail (message != NULL, FALSE); 3092 _dbus_return_val_if_fail (path != NULL, FALSE); 3093 3094 *path = NULL; 3095 3096 v = dbus_message_get_path (message); 3097 if (v != NULL) 3098 { 3099 if (!_dbus_decompose_path (v, strlen (v), 3100 path, NULL)) 3101 return FALSE; 3102 } 3103 return TRUE; 3104 } 3105 3106 /** 3107 * Sets the interface this message is being sent to 3108 * (for DBUS_MESSAGE_TYPE_METHOD_CALL) or 3109 * the interface a signal is being emitted from 3110 * (for DBUS_MESSAGE_TYPE_SIGNAL). 3111 * 3112 * The interface name must contain only valid characters as defined 3113 * in the D-Bus specification. 3114 * 3115 * @param message the message 3116 * @param interface the interface or #NULL to unset 3117 * @returns #FALSE if not enough memory 3118 */ 3119 dbus_bool_t 3120 dbus_message_set_interface (DBusMessage *message, 3121 const char *interface) 3122 { 3123 _dbus_return_val_if_fail (message != NULL, FALSE); 3124 _dbus_return_val_if_fail (!message->locked, FALSE); 3125 _dbus_return_val_if_fail (interface == NULL || 3126 _dbus_check_is_valid_interface (interface), 3127 FALSE); 3128 3129 return set_or_delete_string_field (message, 3130 DBUS_HEADER_FIELD_INTERFACE, 3131 DBUS_TYPE_STRING, 3132 interface); 3133 } 3134 3135 /** 3136 * Gets the interface this message is being sent to 3137 * (for DBUS_MESSAGE_TYPE_METHOD_CALL) or being emitted 3138 * from (for DBUS_MESSAGE_TYPE_SIGNAL). 3139 * The interface name is fully-qualified (namespaced). 3140 * Returns #NULL if none. 3141 * 3142 * The returned string becomes invalid if the message is 3143 * modified, since it points into the wire-marshaled message data. 3144 * 3145 * @param message the message 3146 * @returns the message interface (should not be freed) or #NULL 3147 */ 3148 const char* 3149 dbus_message_get_interface (DBusMessage *message) 3150 { 3151 const char *v; 3152 3153 _dbus_return_val_if_fail (message != NULL, NULL); 3154 3155 v = NULL; /* in case field doesn't exist */ 3156 _dbus_header_get_field_basic (&message->header, 3157 DBUS_HEADER_FIELD_INTERFACE, 3158 DBUS_TYPE_STRING, 3159 (void *) &v); 3160 return v; 3161 } 3162 3163 /** 3164 * Checks if the message has an interface 3165 * 3166 * @param message the message 3167 * @param interface the interface name 3168 * @returns #TRUE if the interface field in the header matches 3169 */ 3170 dbus_bool_t 3171 dbus_message_has_interface (DBusMessage *message, 3172 const char *interface) 3173 { 3174 const char *msg_interface; 3175 msg_interface = dbus_message_get_interface (message); 3176 3177 if (msg_interface == NULL) 3178 { 3179 if (interface == NULL) 3180 return TRUE; 3181 else 3182 return FALSE; 3183 } 3184 3185 if (interface == NULL) 3186 return FALSE; 3187 3188 if (strcmp (msg_interface, interface) == 0) 3189 return TRUE; 3190 3191 return FALSE; 3192 3193 } 3194 3195 /** 3196 * Sets the interface member being invoked 3197 * (DBUS_MESSAGE_TYPE_METHOD_CALL) or emitted 3198 * (DBUS_MESSAGE_TYPE_SIGNAL). 3199 * 3200 * The member name must contain only valid characters as defined 3201 * in the D-Bus specification. 3202 * 3203 * @param message the message 3204 * @param member the member or #NULL to unset 3205 * @returns #FALSE if not enough memory 3206 */ 3207 dbus_bool_t 3208 dbus_message_set_member (DBusMessage *message, 3209 const char *member) 3210 { 3211 _dbus_return_val_if_fail (message != NULL, FALSE); 3212 _dbus_return_val_if_fail (!message->locked, FALSE); 3213 _dbus_return_val_if_fail (member == NULL || 3214 _dbus_check_is_valid_member (member), 3215 FALSE); 3216 3217 return set_or_delete_string_field (message, 3218 DBUS_HEADER_FIELD_MEMBER, 3219 DBUS_TYPE_STRING, 3220 member); 3221 } 3222 3223 /** 3224 * Gets the interface member being invoked 3225 * (DBUS_MESSAGE_TYPE_METHOD_CALL) or emitted 3226 * (DBUS_MESSAGE_TYPE_SIGNAL). Returns #NULL if none. 3227 * 3228 * The returned string becomes invalid if the message is 3229 * modified, since it points into the wire-marshaled message data. 3230 * 3231 * @param message the message 3232 * @returns the member name (should not be freed) or #NULL 3233 */ 3234 const char* 3235 dbus_message_get_member (DBusMessage *message) 3236 { 3237 const char *v; 3238 3239 _dbus_return_val_if_fail (message != NULL, NULL); 3240 3241 v = NULL; /* in case field doesn't exist */ 3242 _dbus_header_get_field_basic (&message->header, 3243 DBUS_HEADER_FIELD_MEMBER, 3244 DBUS_TYPE_STRING, 3245 (void *) &v); 3246 return v; 3247 } 3248 3249 /** 3250 * Checks if the message has an interface member 3251 * 3252 * @param message the message 3253 * @param member the member name 3254 * @returns #TRUE if there is a member field in the header 3255 */ 3256 dbus_bool_t 3257 dbus_message_has_member (DBusMessage *message, 3258 const char *member) 3259 { 3260 const char *msg_member; 3261 msg_member = dbus_message_get_member (message); 3262 3263 if (msg_member == NULL) 3264 { 3265 if (member == NULL) 3266 return TRUE; 3267 else 3268 return FALSE; 3269 } 3270 3271 if (member == NULL) 3272 return FALSE; 3273 3274 if (strcmp (msg_member, member) == 0) 3275 return TRUE; 3276 3277 return FALSE; 3278 3279 } 3280 3281 /** 3282 * Sets the name of the error (DBUS_MESSAGE_TYPE_ERROR). 3283 * The name is fully-qualified (namespaced). 3284 * 3285 * The error name must contain only valid characters as defined 3286 * in the D-Bus specification. 3287 * 3288 * @param message the message 3289 * @param error_name the name or #NULL to unset 3290 * @returns #FALSE if not enough memory 3291 */ 3292 dbus_bool_t 3293 dbus_message_set_error_name (DBusMessage *message, 3294 const char *error_name) 3295 { 3296 _dbus_return_val_if_fail (message != NULL, FALSE); 3297 _dbus_return_val_if_fail (!message->locked, FALSE); 3298 _dbus_return_val_if_fail (error_name == NULL || 3299 _dbus_check_is_valid_error_name (error_name), 3300 FALSE); 3301 3302 return set_or_delete_string_field (message, 3303 DBUS_HEADER_FIELD_ERROR_NAME, 3304 DBUS_TYPE_STRING, 3305 error_name); 3306 } 3307 3308 /** 3309 * Gets the error name (DBUS_MESSAGE_TYPE_ERROR only) 3310 * or #NULL if none. 3311 * 3312 * The returned string becomes invalid if the message is 3313 * modified, since it points into the wire-marshaled message data. 3314 * 3315 * @param message the message 3316 * @returns the error name (should not be freed) or #NULL 3317 */ 3318 const char* 3319 dbus_message_get_error_name (DBusMessage *message) 3320 { 3321 const char *v; 3322 3323 _dbus_return_val_if_fail (message != NULL, NULL); 3324 3325 v = NULL; /* in case field doesn't exist */ 3326 _dbus_header_get_field_basic (&message->header, 3327 DBUS_HEADER_FIELD_ERROR_NAME, 3328 DBUS_TYPE_STRING, 3329 (void *) &v); 3330 return v; 3331 } 3332 3333 /** 3334 * Sets the message's destination. The destination is the name of 3335 * another connection on the bus and may be either the unique name 3336 * assigned by the bus to each connection, or a well-known name 3337 * specified in advance. 3338 * 3339 * The destination name must contain only valid characters as defined 3340 * in the D-Bus specification. 3341 * 3342 * @param message the message 3343 * @param destination the destination name or #NULL to unset 3344 * @returns #FALSE if not enough memory 3345 */ 3346 dbus_bool_t 3347 dbus_message_set_destination (DBusMessage *message, 3348 const char *destination) 3349 { 3350 _dbus_return_val_if_fail (message != NULL, FALSE); 3351 _dbus_return_val_if_fail (!message->locked, FALSE); 3352 _dbus_return_val_if_fail (destination == NULL || 3353 _dbus_check_is_valid_bus_name (destination), 3354 FALSE); 3355 3356 return set_or_delete_string_field (message, 3357 DBUS_HEADER_FIELD_DESTINATION, 3358 DBUS_TYPE_STRING, 3359 destination); 3360 } 3361 3362 /** 3363 * Gets the destination of a message or #NULL if there is none set. 3364 * 3365 * The returned string becomes invalid if the message is 3366 * modified, since it points into the wire-marshaled message data. 3367 * 3368 * @param message the message 3369 * @returns the message destination (should not be freed) or #NULL 3370 */ 3371 const char* 3372 dbus_message_get_destination (DBusMessage *message) 3373 { 3374 const char *v; 3375 3376 _dbus_return_val_if_fail (message != NULL, NULL); 3377 3378 v = NULL; /* in case field doesn't exist */ 3379 _dbus_header_get_field_basic (&message->header, 3380 DBUS_HEADER_FIELD_DESTINATION, 3381 DBUS_TYPE_STRING, 3382 (void *) &v); 3383 return v; 3384 } 3385 3386 /** 3387 * Sets the message sender. 3388 * 3389 * The sender must be a valid bus name as defined in the D-Bus 3390 * specification. 3391 * 3392 * Usually you don't want to call this. The message bus daemon will 3393 * call it to set the origin of each message. If you aren't implementing 3394 * a message bus daemon you shouldn't need to set the sender. 3395 * 3396 * @param message the message 3397 * @param sender the sender or #NULL to unset 3398 * @returns #FALSE if not enough memory 3399 */ 3400 dbus_bool_t 3401 dbus_message_set_sender (DBusMessage *message, 3402 const char *sender) 3403 { 3404 _dbus_return_val_if_fail (message != NULL, FALSE); 3405 _dbus_return_val_if_fail (!message->locked, FALSE); 3406 _dbus_return_val_if_fail (sender == NULL || 3407 _dbus_check_is_valid_bus_name (sender), 3408 FALSE); 3409 3410 return set_or_delete_string_field (message, 3411 DBUS_HEADER_FIELD_SENDER, 3412 DBUS_TYPE_STRING, 3413 sender); 3414 } 3415 3416 /** 3417 * Gets the unique name of the connection which originated this 3418 * message, or #NULL if unknown or inapplicable. The sender is filled 3419 * in by the message bus. 3420 * 3421 * Note, the returned sender is always the unique bus name. 3422 * Connections may own multiple other bus names, but those 3423 * are not found in the sender field. 3424 * 3425 * The returned string becomes invalid if the message is 3426 * modified, since it points into the wire-marshaled message data. 3427 * 3428 * @param message the message 3429 * @returns the unique name of the sender or #NULL 3430 */ 3431 const char* 3432 dbus_message_get_sender (DBusMessage *message) 3433 { 3434 const char *v; 3435 3436 _dbus_return_val_if_fail (message != NULL, NULL); 3437 3438 v = NULL; /* in case field doesn't exist */ 3439 _dbus_header_get_field_basic (&message->header, 3440 DBUS_HEADER_FIELD_SENDER, 3441 DBUS_TYPE_STRING, 3442 (void *) &v); 3443 return v; 3444 } 3445 3446 /** 3447 * Gets the type signature of the message, i.e. the arguments in the 3448 * message payload. The signature includes only "in" arguments for 3449 * #DBUS_MESSAGE_TYPE_METHOD_CALL and only "out" arguments for 3450 * #DBUS_MESSAGE_TYPE_METHOD_RETURN, so is slightly different from 3451 * what you might expect (that is, it does not include the signature of the 3452 * entire C++-style method). 3453 * 3454 * The signature is a string made up of type codes such as 3455 * #DBUS_TYPE_INT32. The string is terminated with nul (nul is also 3456 * the value of #DBUS_TYPE_INVALID). 3457 * 3458 * The returned string becomes invalid if the message is 3459 * modified, since it points into the wire-marshaled message data. 3460 * 3461 * @param message the message 3462 * @returns the type signature 3463 */ 3464 const char* 3465 dbus_message_get_signature (DBusMessage *message) 3466 { 3467 const DBusString *type_str; 3468 int type_pos; 3469 3470 _dbus_return_val_if_fail (message != NULL, NULL); 3471 3472 get_const_signature (&message->header, &type_str, &type_pos); 3473 3474 return _dbus_string_get_const_data_len (type_str, type_pos, 0); 3475 } 3476 3477 static dbus_bool_t 3478 _dbus_message_has_type_interface_member (DBusMessage *message, 3479 int type, 3480 const char *interface, 3481 const char *member) 3482 { 3483 const char *n; 3484 3485 _dbus_assert (message != NULL); 3486 _dbus_assert (interface != NULL); 3487 _dbus_assert (member != NULL); 3488 3489 if (dbus_message_get_type (message) != type) 3490 return FALSE; 3491 3492 /* Optimize by checking the short member name first 3493 * instead of the longer interface name 3494 */ 3495 3496 n = dbus_message_get_member (message); 3497 3498 if (n && strcmp (n, member) == 0) 3499 { 3500 n = dbus_message_get_interface (message); 3501 3502 if (n == NULL || strcmp (n, interface) == 0) 3503 return TRUE; 3504 } 3505 3506 return FALSE; 3507 } 3508 3509 /** 3510 * Checks whether the message is a method call with the given 3511 * interface and member fields. If the message is not 3512 * #DBUS_MESSAGE_TYPE_METHOD_CALL, or has a different interface or 3513 * member field, returns #FALSE. If the interface field is missing, 3514 * then it will be assumed equal to the provided interface. The D-Bus 3515 * protocol allows method callers to leave out the interface name. 3516 * 3517 * @param message the message 3518 * @param interface the name to check (must not be #NULL) 3519 * @param method the name to check (must not be #NULL) 3520 * 3521 * @returns #TRUE if the message is the specified method call 3522 */ 3523 dbus_bool_t 3524 dbus_message_is_method_call (DBusMessage *message, 3525 const char *interface, 3526 const char *method) 3527 { 3528 _dbus_return_val_if_fail (message != NULL, FALSE); 3529 _dbus_return_val_if_fail (interface != NULL, FALSE); 3530 _dbus_return_val_if_fail (method != NULL, FALSE); 3531 /* don't check that interface/method are valid since it would be 3532 * expensive, and not catch many common errors 3533 */ 3534 3535 return _dbus_message_has_type_interface_member (message, 3536 DBUS_MESSAGE_TYPE_METHOD_CALL, 3537 interface, method); 3538 } 3539 3540 /** 3541 * Checks whether the message is a signal with the given interface and 3542 * member fields. If the message is not #DBUS_MESSAGE_TYPE_SIGNAL, or 3543 * has a different interface or member field, returns #FALSE. 3544 * 3545 * @param message the message 3546 * @param interface the name to check (must not be #NULL) 3547 * @param signal_name the name to check (must not be #NULL) 3548 * 3549 * @returns #TRUE if the message is the specified signal 3550 */ 3551 dbus_bool_t 3552 dbus_message_is_signal (DBusMessage *message, 3553 const char *interface, 3554 const char *signal_name) 3555 { 3556 _dbus_return_val_if_fail (message != NULL, FALSE); 3557 _dbus_return_val_if_fail (interface != NULL, FALSE); 3558 _dbus_return_val_if_fail (signal_name != NULL, FALSE); 3559 /* don't check that interface/name are valid since it would be 3560 * expensive, and not catch many common errors 3561 */ 3562 3563 return _dbus_message_has_type_interface_member (message, 3564 DBUS_MESSAGE_TYPE_SIGNAL, 3565 interface, signal_name); 3566 } 3567 3568 /** 3569 * Checks whether the message is an error reply with the given error 3570 * name. If the message is not #DBUS_MESSAGE_TYPE_ERROR, or has a 3571 * different name, returns #FALSE. 3572 * 3573 * @param message the message 3574 * @param error_name the name to check (must not be #NULL) 3575 * 3576 * @returns #TRUE if the message is the specified error 3577 */ 3578 dbus_bool_t 3579 dbus_message_is_error (DBusMessage *message, 3580 const char *error_name) 3581 { 3582 const char *n; 3583 3584 _dbus_return_val_if_fail (message != NULL, FALSE); 3585 _dbus_return_val_if_fail (error_name != NULL, FALSE); 3586 /* don't check that error_name is valid since it would be expensive, 3587 * and not catch many common errors 3588 */ 3589 3590 if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_ERROR) 3591 return FALSE; 3592 3593 n = dbus_message_get_error_name (message); 3594 3595 if (n && strcmp (n, error_name) == 0) 3596 return TRUE; 3597 else 3598 return FALSE; 3599 } 3600 3601 /** 3602 * Checks whether the message was sent to the given name. If the 3603 * message has no destination specified or has a different 3604 * destination, returns #FALSE. 3605 * 3606 * @param message the message 3607 * @param name the name to check (must not be #NULL) 3608 * 3609 * @returns #TRUE if the message has the given destination name 3610 */ 3611 dbus_bool_t 3612 dbus_message_has_destination (DBusMessage *message, 3613 const char *name) 3614 { 3615 const char *s; 3616 3617 _dbus_return_val_if_fail (message != NULL, FALSE); 3618 _dbus_return_val_if_fail (name != NULL, FALSE); 3619 /* don't check that name is valid since it would be expensive, and 3620 * not catch many common errors 3621 */ 3622 3623 s = dbus_message_get_destination (message); 3624 3625 if (s && strcmp (s, name) == 0) 3626 return TRUE; 3627 else 3628 return FALSE; 3629 } 3630 3631 /** 3632 * Checks whether the message has the given unique name as its sender. 3633 * If the message has no sender specified or has a different sender, 3634 * returns #FALSE. Note that a peer application will always have the 3635 * unique name of the connection as the sender. So you can't use this 3636 * function to see whether a sender owned a well-known name. 3637 * 3638 * Messages from the bus itself will have #DBUS_SERVICE_DBUS 3639 * as the sender. 3640 * 3641 * @param message the message 3642 * @param name the name to check (must not be #NULL) 3643 * 3644 * @returns #TRUE if the message has the given sender 3645 */ 3646 dbus_bool_t 3647 dbus_message_has_sender (DBusMessage *message, 3648 const char *name) 3649 { 3650 const char *s; 3651 3652 _dbus_return_val_if_fail (message != NULL, FALSE); 3653 _dbus_return_val_if_fail (name != NULL, FALSE); 3654 /* don't check that name is valid since it would be expensive, and 3655 * not catch many common errors 3656 */ 3657 3658 s = dbus_message_get_sender (message); 3659 3660 if (s && strcmp (s, name) == 0) 3661 return TRUE; 3662 else 3663 return FALSE; 3664 } 3665 3666 /** 3667 * Checks whether the message has the given signature; see 3668 * dbus_message_get_signature() for more details on what the signature 3669 * looks like. 3670 * 3671 * @param message the message 3672 * @param signature typecode array 3673 * @returns #TRUE if message has the given signature 3674 */ 3675 dbus_bool_t 3676 dbus_message_has_signature (DBusMessage *message, 3677 const char *signature) 3678 { 3679 const char *s; 3680 3681 _dbus_return_val_if_fail (message != NULL, FALSE); 3682 _dbus_return_val_if_fail (signature != NULL, FALSE); 3683 /* don't check that signature is valid since it would be expensive, 3684 * and not catch many common errors 3685 */ 3686 3687 s = dbus_message_get_signature (message); 3688 3689 if (s && strcmp (s, signature) == 0) 3690 return TRUE; 3691 else 3692 return FALSE; 3693 } 3694 3695 /** 3696 * Sets a #DBusError based on the contents of the given 3697 * message. The error is only set if the message 3698 * is an error message, as in #DBUS_MESSAGE_TYPE_ERROR. 3699 * The name of the error is set to the name of the message, 3700 * and the error message is set to the first argument 3701 * if the argument exists and is a string. 3702 * 3703 * The return value indicates whether the error was set (the error is 3704 * set if and only if the message is an error message). So you can 3705 * check for an error reply and convert it to DBusError in one go: 3706 * @code 3707 * if (dbus_set_error_from_message (error, reply)) 3708 * return error; 3709 * else 3710 * process reply; 3711 * @endcode 3712 * 3713 * @param error the error to set 3714 * @param message the message to set it from 3715 * @returns #TRUE if the message had type #DBUS_MESSAGE_TYPE_ERROR 3716 */ 3717 dbus_bool_t 3718 dbus_set_error_from_message (DBusError *error, 3719 DBusMessage *message) 3720 { 3721 const char *str; 3722 3723 _dbus_return_val_if_fail (message != NULL, FALSE); 3724 _dbus_return_val_if_error_is_set (error, FALSE); 3725 3726 if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_ERROR) 3727 return FALSE; 3728 3729 str = NULL; 3730 dbus_message_get_args (message, NULL, 3731 DBUS_TYPE_STRING, &str, 3732 DBUS_TYPE_INVALID); 3733 3734 dbus_set_error (error, dbus_message_get_error_name (message), 3735 str ? "%s" : NULL, str); 3736 3737 return TRUE; 3738 } 3739 3740 /** 3741 * Checks whether a message contains unix fds 3742 * 3743 * @param message the message 3744 * @returns #TRUE if the message contains unix fds 3745 */ 3746 dbus_bool_t 3747 dbus_message_contains_unix_fds(DBusMessage *message) 3748 { 3749 #ifdef HAVE_UNIX_FD_PASSING 3750 _dbus_assert(message); 3751 3752 return message->n_unix_fds > 0; 3753 #else 3754 return FALSE; 3755 #endif 3756 } 3757 3758 /** @} */ 3759 3760 /** 3761 * @addtogroup DBusMessageInternals 3762 * 3763 * @{ 3764 */ 3765 3766 /** 3767 * The initial buffer size of the message loader. 3768 * 3769 * @todo this should be based on min header size plus some average 3770 * body size, or something. Or rather, the min header size only, if we 3771 * want to try to read only the header, store that in a DBusMessage, 3772 * then read only the body and store that, etc., depends on 3773 * how we optimize _dbus_message_loader_get_buffer() and what 3774 * the exact message format is. 3775 */ 3776 #define INITIAL_LOADER_DATA_LEN 32 3777 3778 /** 3779 * Creates a new message loader. Returns #NULL if memory can't 3780 * be allocated. 3781 * 3782 * @returns new loader, or #NULL. 3783 */ 3784 DBusMessageLoader* 3785 _dbus_message_loader_new (void) 3786 { 3787 DBusMessageLoader *loader; 3788 3789 loader = dbus_new0 (DBusMessageLoader, 1); 3790 if (loader == NULL) 3791 return NULL; 3792 3793 loader->refcount = 1; 3794 3795 loader->corrupted = FALSE; 3796 loader->corruption_reason = DBUS_VALID; 3797 3798 /* this can be configured by the app, but defaults to the protocol max */ 3799 loader->max_message_size = DBUS_MAXIMUM_MESSAGE_LENGTH; 3800 3801 /* We set a very relatively conservative default here since due to how 3802 SCM_RIGHTS works we need to preallocate an fd array of the maximum 3803 number of unix fds we want to receive in advance. A 3804 try-and-reallocate loop is not possible. */ 3805 loader->max_message_unix_fds = 1024; 3806 3807 if (!_dbus_string_init (&loader->data)) 3808 { 3809 dbus_free (loader); 3810 return NULL; 3811 } 3812 3813 /* preallocate the buffer for speed, ignore failure */ 3814 _dbus_string_set_length (&loader->data, INITIAL_LOADER_DATA_LEN); 3815 _dbus_string_set_length (&loader->data, 0); 3816 3817 #ifdef HAVE_UNIX_FD_PASSING 3818 loader->unix_fds = NULL; 3819 loader->n_unix_fds = loader->n_unix_fds_allocated = 0; 3820 loader->unix_fds_outstanding = FALSE; 3821 #endif 3822 3823 return loader; 3824 } 3825 3826 /** 3827 * Increments the reference count of the loader. 3828 * 3829 * @param loader the loader. 3830 * @returns the loader 3831 */ 3832 DBusMessageLoader * 3833 _dbus_message_loader_ref (DBusMessageLoader *loader) 3834 { 3835 loader->refcount += 1; 3836 3837 return loader; 3838 } 3839 3840 /** 3841 * Decrements the reference count of the loader and finalizes the 3842 * loader when the count reaches zero. 3843 * 3844 * @param loader the loader. 3845 */ 3846 void 3847 _dbus_message_loader_unref (DBusMessageLoader *loader) 3848 { 3849 loader->refcount -= 1; 3850 if (loader->refcount == 0) 3851 { 3852 #ifdef HAVE_UNIX_FD_PASSING 3853 close_unix_fds(loader->unix_fds, &loader->n_unix_fds); 3854 dbus_free(loader->unix_fds); 3855 #endif 3856 _dbus_list_foreach (&loader->messages, 3857 (DBusForeachFunction) dbus_message_unref, 3858 NULL); 3859 _dbus_list_clear (&loader->messages); 3860 _dbus_string_free (&loader->data); 3861 dbus_free (loader); 3862 } 3863 } 3864 3865 /** 3866 * Gets the buffer to use for reading data from the network. Network 3867 * data is read directly into an allocated buffer, which is then used 3868 * in the DBusMessage, to avoid as many extra memcpy's as possible. 3869 * The buffer must always be returned immediately using 3870 * _dbus_message_loader_return_buffer(), even if no bytes are 3871 * successfully read. 3872 * 3873 * @todo this function can be a lot more clever. For example 3874 * it can probably always return a buffer size to read exactly 3875 * the body of the next message, thus avoiding any memory wastage 3876 * or reallocs. 3877 * 3878 * @todo we need to enforce a max length on strings in header fields. 3879 * 3880 * @param loader the message loader. 3881 * @param buffer the buffer 3882 */ 3883 void 3884 _dbus_message_loader_get_buffer (DBusMessageLoader *loader, 3885 DBusString **buffer) 3886 { 3887 _dbus_assert (!loader->buffer_outstanding); 3888 3889 *buffer = &loader->data; 3890 3891 loader->buffer_outstanding = TRUE; 3892 } 3893 3894 /** 3895 * Returns a buffer obtained from _dbus_message_loader_get_buffer(), 3896 * indicating to the loader how many bytes of the buffer were filled 3897 * in. This function must always be called, even if no bytes were 3898 * successfully read. 3899 * 3900 * @param loader the loader. 3901 * @param buffer the buffer. 3902 * @param bytes_read number of bytes that were read into the buffer. 3903 */ 3904 void 3905 _dbus_message_loader_return_buffer (DBusMessageLoader *loader, 3906 DBusString *buffer, 3907 int bytes_read) 3908 { 3909 _dbus_assert (loader->buffer_outstanding); 3910 _dbus_assert (buffer == &loader->data); 3911 3912 loader->buffer_outstanding = FALSE; 3913 } 3914 3915 /** 3916 * Gets the buffer to use for reading unix fds from the network. 3917 * 3918 * This works similar to _dbus_message_loader_get_buffer() 3919 * 3920 * @param loader the message loader. 3921 * @param fds the array to read fds into 3922 * @param max_n_fds how many fds to read at most 3923 * @return TRUE on success, FALSE on OOM 3924 */ 3925 dbus_bool_t 3926 _dbus_message_loader_get_unix_fds(DBusMessageLoader *loader, 3927 int **fds, 3928 unsigned *max_n_fds) 3929 { 3930 #ifdef HAVE_UNIX_FD_PASSING 3931 _dbus_assert (!loader->unix_fds_outstanding); 3932 3933 /* Allocate space where we can put the fds we read. We allocate 3934 space for max_message_unix_fds since this is an 3935 upper limit how many fds can be received within a single 3936 message. Since SCM_RIGHTS doesn't allow a reallocate+retry logic 3937 we are allocating the maximum possible array size right from the 3938 beginning. This sucks a bit, however unless SCM_RIGHTS is fixed 3939 there is no better way. */ 3940 3941 if (loader->n_unix_fds_allocated < loader->max_message_unix_fds) 3942 { 3943 int *a = dbus_realloc(loader->unix_fds, 3944 loader->max_message_unix_fds * sizeof(loader->unix_fds[0])); 3945 3946 if (!a) 3947 return FALSE; 3948 3949 loader->unix_fds = a; 3950 loader->n_unix_fds_allocated = loader->max_message_unix_fds; 3951 } 3952 3953 *fds = loader->unix_fds + loader->n_unix_fds; 3954 *max_n_fds = loader->n_unix_fds_allocated - loader->n_unix_fds; 3955 3956 loader->unix_fds_outstanding = TRUE; 3957 return TRUE; 3958 #else 3959 _dbus_assert_not_reached("Platform doesn't support unix fd passing"); 3960 return FALSE; 3961 #endif 3962 } 3963 3964 /** 3965 * Returns a buffer obtained from _dbus_message_loader_get_unix_fds(). 3966 * 3967 * This works similar to _dbus_message_loader_return_buffer() 3968 * 3969 * @param loader the message loader. 3970 * @param fds the array fds were read into 3971 * @param max_n_fds how many fds were read 3972 */ 3973 3974 void 3975 _dbus_message_loader_return_unix_fds(DBusMessageLoader *loader, 3976 int *fds, 3977 unsigned n_fds) 3978 { 3979 #ifdef HAVE_UNIX_FD_PASSING 3980 _dbus_assert(loader->unix_fds_outstanding); 3981 _dbus_assert(loader->unix_fds + loader->n_unix_fds == fds); 3982 _dbus_assert(loader->n_unix_fds + n_fds <= loader->n_unix_fds_allocated); 3983 3984 loader->n_unix_fds += n_fds; 3985 loader->unix_fds_outstanding = FALSE; 3986 #else 3987 _dbus_assert_not_reached("Platform doesn't support unix fd passing"); 3988 #endif 3989 } 3990 3991 /* 3992 * FIXME when we move the header out of the buffer, that memmoves all 3993 * buffered messages. Kind of crappy. 3994 * 3995 * Also we copy the header and body, which is kind of crappy. To 3996 * avoid this, we have to allow header and body to be in a single 3997 * memory block, which is good for messages we read and bad for 3998 * messages we are creating. But we could move_len() the buffer into 3999 * this single memory block, and move_len() will just swap the buffers 4000 * if you're moving the entire buffer replacing the dest string. 4001 * 4002 * We could also have the message loader tell the transport how many 4003 * bytes to read; so it would first ask for some arbitrary number like 4004 * 256, then if the message was incomplete it would use the 4005 * header/body len to ask for exactly the size of the message (or 4006 * blocks the size of a typical kernel buffer for the socket). That 4007 * way we don't get trailing bytes in the buffer that have to be 4008 * memmoved. Though I suppose we also don't have a chance of reading a 4009 * bunch of small messages at once, so the optimization may be stupid. 4010 * 4011 * Another approach would be to keep a "start" index into 4012 * loader->data and only delete it occasionally, instead of after 4013 * each message is loaded. 4014 * 4015 * load_message() returns FALSE if not enough memory OR the loader was corrupted 4016 */ 4017 static dbus_bool_t 4018 load_message (DBusMessageLoader *loader, 4019 DBusMessage *message, 4020 int byte_order, 4021 int fields_array_len, 4022 int header_len, 4023 int body_len) 4024 { 4025 dbus_bool_t oom; 4026 DBusValidity validity; 4027 const DBusString *type_str; 4028 int type_pos; 4029 DBusValidationMode mode; 4030 dbus_uint32_t n_unix_fds = 0; 4031 4032 mode = DBUS_VALIDATION_MODE_DATA_IS_UNTRUSTED; 4033 4034 oom = FALSE; 4035 4036 #if 0 4037 _dbus_verbose_bytes_of_string (&loader->data, 0, header_len /* + body_len */); 4038 #endif 4039 4040 /* 1. VALIDATE AND COPY OVER HEADER */ 4041 _dbus_assert (_dbus_string_get_length (&message->header.data) == 0); 4042 _dbus_assert ((header_len + body_len) <= _dbus_string_get_length (&loader->data)); 4043 4044 if (!_dbus_header_load (&message->header, 4045 mode, 4046 &validity, 4047 byte_order, 4048 fields_array_len, 4049 header_len, 4050 body_len, 4051 &loader->data, 0, 4052 _dbus_string_get_length (&loader->data))) 4053 { 4054 _dbus_verbose ("Failed to load header for new message code %d\n", validity); 4055 4056 /* assert here so we can catch any code that still uses DBUS_VALID to indicate 4057 oom errors. They should use DBUS_VALIDITY_UNKNOWN_OOM_ERROR instead */ 4058 _dbus_assert (validity != DBUS_VALID); 4059 4060 if (validity == DBUS_VALIDITY_UNKNOWN_OOM_ERROR) 4061 oom = TRUE; 4062 else 4063 { 4064 loader->corrupted = TRUE; 4065 loader->corruption_reason = validity; 4066 } 4067 goto failed; 4068 } 4069 4070 _dbus_assert (validity == DBUS_VALID); 4071 4072 /* 2. VALIDATE BODY */ 4073 if (mode != DBUS_VALIDATION_MODE_WE_TRUST_THIS_DATA_ABSOLUTELY) 4074 { 4075 get_const_signature (&message->header, &type_str, &type_pos); 4076 4077 /* Because the bytes_remaining arg is NULL, this validates that the 4078 * body is the right length 4079 */ 4080 validity = _dbus_validate_body_with_reason (type_str, 4081 type_pos, 4082 byte_order, 4083 NULL, 4084 &loader->data, 4085 header_len, 4086 body_len); 4087 if (validity != DBUS_VALID) 4088 { 4089 _dbus_verbose ("Failed to validate message body code %d\n", validity); 4090 4091 loader->corrupted = TRUE; 4092 loader->corruption_reason = validity; 4093 4094 goto failed; 4095 } 4096 } 4097 4098 /* 3. COPY OVER UNIX FDS */ 4099 _dbus_header_get_field_basic(&message->header, 4100 DBUS_HEADER_FIELD_UNIX_FDS, 4101 DBUS_TYPE_UINT32, 4102 &n_unix_fds); 4103 4104 #ifdef HAVE_UNIX_FD_PASSING 4105 4106 if (n_unix_fds > loader->n_unix_fds) 4107 { 4108 _dbus_verbose("Message contains references to more unix fds than were sent %u != %u\n", 4109 n_unix_fds, loader->n_unix_fds); 4110 4111 loader->corrupted = TRUE; 4112 loader->corruption_reason = DBUS_INVALID_MISSING_UNIX_FDS; 4113 goto failed; 4114 } 4115 4116 /* If this was a recycled message there might still be 4117 some memory allocated for the fds */ 4118 dbus_free(message->unix_fds); 4119 4120 if (n_unix_fds > 0) 4121 { 4122 message->unix_fds = _dbus_memdup(loader->unix_fds, n_unix_fds * sizeof(message->unix_fds[0])); 4123 if (message->unix_fds == NULL) 4124 { 4125 _dbus_verbose ("Failed to allocate file descriptor array\n"); 4126 oom = TRUE; 4127 goto failed; 4128 } 4129 4130 message->n_unix_fds_allocated = message->n_unix_fds = n_unix_fds; 4131 loader->n_unix_fds -= n_unix_fds; 4132 memmove(loader->unix_fds + n_unix_fds, loader->unix_fds, loader->n_unix_fds); 4133 } 4134 else 4135 message->unix_fds = NULL; 4136 4137 #else 4138 4139 if (n_unix_fds > 0) 4140 { 4141 _dbus_verbose ("Hmm, message claims to come with file descriptors " 4142 "but that's not supported on our platform, disconnecting.\n"); 4143 4144 loader->corrupted = TRUE; 4145 loader->corruption_reason = DBUS_INVALID_MISSING_UNIX_FDS; 4146 goto failed; 4147 } 4148 4149 #endif 4150 4151 /* 3. COPY OVER BODY AND QUEUE MESSAGE */ 4152 4153 if (!_dbus_list_append (&loader->messages, message)) 4154 { 4155 _dbus_verbose ("Failed to append new message to loader queue\n"); 4156 oom = TRUE; 4157 goto failed; 4158 } 4159 4160 _dbus_assert (_dbus_string_get_length (&message->body) == 0); 4161 _dbus_assert (_dbus_string_get_length (&loader->data) >= 4162 (header_len + body_len)); 4163 4164 if (!_dbus_string_copy_len (&loader->data, header_len, body_len, &message->body, 0)) 4165 { 4166 _dbus_verbose ("Failed to move body into new message\n"); 4167 oom = TRUE; 4168 goto failed; 4169 } 4170 4171 _dbus_string_delete (&loader->data, 0, header_len + body_len); 4172 4173 /* don't waste more than 2k of memory */ 4174 _dbus_string_compact (&loader->data, 2048); 4175 4176 _dbus_assert (_dbus_string_get_length (&message->header.data) == header_len); 4177 _dbus_assert (_dbus_string_get_length (&message->body) == body_len); 4178 4179 _dbus_verbose ("Loaded message %p\n", message); 4180 4181 _dbus_assert (!oom); 4182 _dbus_assert (!loader->corrupted); 4183 _dbus_assert (loader->messages != NULL); 4184 _dbus_assert (_dbus_list_find_last (&loader->messages, message) != NULL); 4185 4186 return TRUE; 4187 4188 failed: 4189 4190 /* Clean up */ 4191 4192 /* does nothing if the message isn't in the list */ 4193 _dbus_list_remove_last (&loader->messages, message); 4194 4195 if (oom) 4196 _dbus_assert (!loader->corrupted); 4197 else 4198 _dbus_assert (loader->corrupted); 4199 4200 _dbus_verbose_bytes_of_string (&loader->data, 0, _dbus_string_get_length (&loader->data)); 4201 4202 return FALSE; 4203 } 4204 4205 /** 4206 * Converts buffered data into messages, if we have enough data. If 4207 * we don't have enough data, does nothing. 4208 * 4209 * @todo we need to check that the proper named header fields exist 4210 * for each message type. 4211 * 4212 * @todo If a message has unknown type, we should probably eat it 4213 * right here rather than passing it out to applications. However 4214 * it's not an error to see messages of unknown type. 4215 * 4216 * @param loader the loader. 4217 * @returns #TRUE if we had enough memory to finish. 4218 */ 4219 dbus_bool_t 4220 _dbus_message_loader_queue_messages (DBusMessageLoader *loader) 4221 { 4222 while (!loader->corrupted && 4223 _dbus_string_get_length (&loader->data) >= DBUS_MINIMUM_HEADER_SIZE) 4224 { 4225 DBusValidity validity; 4226 int byte_order, fields_array_len, header_len, body_len; 4227 4228 if (_dbus_header_have_message_untrusted (loader->max_message_size, 4229 &validity, 4230 &byte_order, 4231 &fields_array_len, 4232 &header_len, 4233 &body_len, 4234 &loader->data, 0, 4235 _dbus_string_get_length (&loader->data))) 4236 { 4237 DBusMessage *message; 4238 4239 _dbus_assert (validity == DBUS_VALID); 4240 4241 message = dbus_message_new_empty_header (); 4242 if (message == NULL) 4243 return FALSE; 4244 4245 if (!load_message (loader, message, 4246 byte_order, fields_array_len, 4247 header_len, body_len)) 4248 { 4249 dbus_message_unref (message); 4250 /* load_message() returns false if corrupted or OOM; if 4251 * corrupted then return TRUE for not OOM 4252 */ 4253 return loader->corrupted; 4254 } 4255 4256 _dbus_assert (loader->messages != NULL); 4257 _dbus_assert (_dbus_list_find_last (&loader->messages, message) != NULL); 4258 } 4259 else 4260 { 4261 _dbus_verbose ("Initial peek at header says we don't have a whole message yet, or data broken with invalid code %d\n", 4262 validity); 4263 if (validity != DBUS_VALID) 4264 { 4265 loader->corrupted = TRUE; 4266 loader->corruption_reason = validity; 4267 } 4268 return TRUE; 4269 } 4270 } 4271 4272 return TRUE; 4273 } 4274 4275 /** 4276 * Peeks at first loaded message, returns #NULL if no messages have 4277 * been queued. 4278 * 4279 * @param loader the loader. 4280 * @returns the next message, or #NULL if none. 4281 */ 4282 DBusMessage* 4283 _dbus_message_loader_peek_message (DBusMessageLoader *loader) 4284 { 4285 if (loader->messages) 4286 return loader->messages->data; 4287 else 4288 return NULL; 4289 } 4290 4291 /** 4292 * Pops a loaded message (passing ownership of the message 4293 * to the caller). Returns #NULL if no messages have been 4294 * queued. 4295 * 4296 * @param loader the loader. 4297 * @returns the next message, or #NULL if none. 4298 */ 4299 DBusMessage* 4300 _dbus_message_loader_pop_message (DBusMessageLoader *loader) 4301 { 4302 return _dbus_list_pop_first (&loader->messages); 4303 } 4304 4305 /** 4306 * Pops a loaded message inside a list link (passing ownership of the 4307 * message and link to the caller). Returns #NULL if no messages have 4308 * been loaded. 4309 * 4310 * @param loader the loader. 4311 * @returns the next message link, or #NULL if none. 4312 */ 4313 DBusList* 4314 _dbus_message_loader_pop_message_link (DBusMessageLoader *loader) 4315 { 4316 return _dbus_list_pop_first_link (&loader->messages); 4317 } 4318 4319 /** 4320 * Returns a popped message link, used to undo a pop. 4321 * 4322 * @param loader the loader 4323 * @param link the link with a message in it 4324 */ 4325 void 4326 _dbus_message_loader_putback_message_link (DBusMessageLoader *loader, 4327 DBusList *link) 4328 { 4329 _dbus_list_prepend_link (&loader->messages, link); 4330 } 4331 4332 /** 4333 * Checks whether the loader is confused due to bad data. 4334 * If messages are received that are invalid, the 4335 * loader gets confused and gives up permanently. 4336 * This state is called "corrupted." 4337 * 4338 * @param loader the loader 4339 * @returns #TRUE if the loader is hosed. 4340 */ 4341 dbus_bool_t 4342 _dbus_message_loader_get_is_corrupted (DBusMessageLoader *loader) 4343 { 4344 _dbus_assert ((loader->corrupted && loader->corruption_reason != DBUS_VALID) || 4345 (!loader->corrupted && loader->corruption_reason == DBUS_VALID)); 4346 return loader->corrupted; 4347 } 4348 4349 /** 4350 * Checks what kind of bad data confused the loader. 4351 * 4352 * @param loader the loader 4353 * @returns why the loader is hosed, or DBUS_VALID if it isn't. 4354 */ 4355 DBusValidity 4356 _dbus_message_loader_get_corruption_reason (DBusMessageLoader *loader) 4357 { 4358 _dbus_assert ((loader->corrupted && loader->corruption_reason != DBUS_VALID) || 4359 (!loader->corrupted && loader->corruption_reason == DBUS_VALID)); 4360 4361 return loader->corruption_reason; 4362 } 4363 4364 /** 4365 * Sets the maximum size message we allow. 4366 * 4367 * @param loader the loader 4368 * @param size the max message size in bytes 4369 */ 4370 void 4371 _dbus_message_loader_set_max_message_size (DBusMessageLoader *loader, 4372 long size) 4373 { 4374 if (size > DBUS_MAXIMUM_MESSAGE_LENGTH) 4375 { 4376 _dbus_verbose ("clamping requested max message size %ld to %d\n", 4377 size, DBUS_MAXIMUM_MESSAGE_LENGTH); 4378 size = DBUS_MAXIMUM_MESSAGE_LENGTH; 4379 } 4380 loader->max_message_size = size; 4381 } 4382 4383 /** 4384 * Gets the maximum allowed message size in bytes. 4385 * 4386 * @param loader the loader 4387 * @returns max size in bytes 4388 */ 4389 long 4390 _dbus_message_loader_get_max_message_size (DBusMessageLoader *loader) 4391 { 4392 return loader->max_message_size; 4393 } 4394 4395 /** 4396 * Sets the maximum unix fds per message we allow. 4397 * 4398 * @param loader the loader 4399 * @param size the max number of unix fds in a message 4400 */ 4401 void 4402 _dbus_message_loader_set_max_message_unix_fds (DBusMessageLoader *loader, 4403 long n) 4404 { 4405 if (n > DBUS_MAXIMUM_MESSAGE_UNIX_FDS) 4406 { 4407 _dbus_verbose ("clamping requested max message unix_fds %ld to %d\n", 4408 n, DBUS_MAXIMUM_MESSAGE_UNIX_FDS); 4409 n = DBUS_MAXIMUM_MESSAGE_UNIX_FDS; 4410 } 4411 loader->max_message_unix_fds = n; 4412 } 4413 4414 /** 4415 * Gets the maximum allowed number of unix fds per message 4416 * 4417 * @param loader the loader 4418 * @returns max unix fds 4419 */ 4420 long 4421 _dbus_message_loader_get_max_message_unix_fds (DBusMessageLoader *loader) 4422 { 4423 return loader->max_message_unix_fds; 4424 } 4425 4426 static DBusDataSlotAllocator slot_allocator; 4427 _DBUS_DEFINE_GLOBAL_LOCK (message_slots); 4428 4429 /** 4430 * Allocates an integer ID to be used for storing application-specific 4431 * data on any DBusMessage. The allocated ID may then be used 4432 * with dbus_message_set_data() and dbus_message_get_data(). 4433 * The passed-in slot must be initialized to -1, and is filled in 4434 * with the slot ID. If the passed-in slot is not -1, it's assumed 4435 * to be already allocated, and its refcount is incremented. 4436 * 4437 * The allocated slot is global, i.e. all DBusMessage objects will 4438 * have a slot with the given integer ID reserved. 4439 * 4440 * @param slot_p address of a global variable storing the slot 4441 * @returns #FALSE on failure (no memory) 4442 */ 4443 dbus_bool_t 4444 dbus_message_allocate_data_slot (dbus_int32_t *slot_p) 4445 { 4446 return _dbus_data_slot_allocator_alloc (&slot_allocator, 4447 &_DBUS_LOCK_NAME (message_slots), 4448 slot_p); 4449 } 4450 4451 /** 4452 * Deallocates a global ID for message data slots. 4453 * dbus_message_get_data() and dbus_message_set_data() may no 4454 * longer be used with this slot. Existing data stored on existing 4455 * DBusMessage objects will be freed when the message is 4456 * finalized, but may not be retrieved (and may only be replaced if 4457 * someone else reallocates the slot). When the refcount on the 4458 * passed-in slot reaches 0, it is set to -1. 4459 * 4460 * @param slot_p address storing the slot to deallocate 4461 */ 4462 void 4463 dbus_message_free_data_slot (dbus_int32_t *slot_p) 4464 { 4465 _dbus_return_if_fail (*slot_p >= 0); 4466 4467 _dbus_data_slot_allocator_free (&slot_allocator, slot_p); 4468 } 4469 4470 /** 4471 * Stores a pointer on a DBusMessage, along 4472 * with an optional function to be used for freeing 4473 * the data when the data is set again, or when 4474 * the message is finalized. The slot number 4475 * must have been allocated with dbus_message_allocate_data_slot(). 4476 * 4477 * @param message the message 4478 * @param slot the slot number 4479 * @param data the data to store 4480 * @param free_data_func finalizer function for the data 4481 * @returns #TRUE if there was enough memory to store the data 4482 */ 4483 dbus_bool_t 4484 dbus_message_set_data (DBusMessage *message, 4485 dbus_int32_t slot, 4486 void *data, 4487 DBusFreeFunction free_data_func) 4488 { 4489 DBusFreeFunction old_free_func; 4490 void *old_data; 4491 dbus_bool_t retval; 4492 4493 _dbus_return_val_if_fail (message != NULL, FALSE); 4494 _dbus_return_val_if_fail (slot >= 0, FALSE); 4495 4496 retval = _dbus_data_slot_list_set (&slot_allocator, 4497 &message->slot_list, 4498 slot, data, free_data_func, 4499 &old_free_func, &old_data); 4500 4501 if (retval) 4502 { 4503 /* Do the actual free outside the message lock */ 4504 if (old_free_func) 4505 (* old_free_func) (old_data); 4506 } 4507 4508 return retval; 4509 } 4510 4511 /** 4512 * Retrieves data previously set with dbus_message_set_data(). 4513 * The slot must still be allocated (must not have been freed). 4514 * 4515 * @param message the message 4516 * @param slot the slot to get data from 4517 * @returns the data, or #NULL if not found 4518 */ 4519 void* 4520 dbus_message_get_data (DBusMessage *message, 4521 dbus_int32_t slot) 4522 { 4523 void *res; 4524 4525 _dbus_return_val_if_fail (message != NULL, NULL); 4526 4527 res = _dbus_data_slot_list_get (&slot_allocator, 4528 &message->slot_list, 4529 slot); 4530 4531 return res; 4532 } 4533 4534 /** 4535 * Utility function to convert a machine-readable (not translated) 4536 * string into a D-Bus message type. 4537 * 4538 * @code 4539 * "method_call" -> DBUS_MESSAGE_TYPE_METHOD_CALL 4540 * "method_return" -> DBUS_MESSAGE_TYPE_METHOD_RETURN 4541 * "signal" -> DBUS_MESSAGE_TYPE_SIGNAL 4542 * "error" -> DBUS_MESSAGE_TYPE_ERROR 4543 * anything else -> DBUS_MESSAGE_TYPE_INVALID 4544 * @endcode 4545 * 4546 */ 4547 int 4548 dbus_message_type_from_string (const char *type_str) 4549 { 4550 if (strcmp (type_str, "method_call") == 0) 4551 return DBUS_MESSAGE_TYPE_METHOD_CALL; 4552 if (strcmp (type_str, "method_return") == 0) 4553 return DBUS_MESSAGE_TYPE_METHOD_RETURN; 4554 else if (strcmp (type_str, "signal") == 0) 4555 return DBUS_MESSAGE_TYPE_SIGNAL; 4556 else if (strcmp (type_str, "error") == 0) 4557 return DBUS_MESSAGE_TYPE_ERROR; 4558 else 4559 return DBUS_MESSAGE_TYPE_INVALID; 4560 } 4561 4562 /** 4563 * Utility function to convert a D-Bus message type into a 4564 * machine-readable string (not translated). 4565 * 4566 * @code 4567 * DBUS_MESSAGE_TYPE_METHOD_CALL -> "method_call" 4568 * DBUS_MESSAGE_TYPE_METHOD_RETURN -> "method_return" 4569 * DBUS_MESSAGE_TYPE_SIGNAL -> "signal" 4570 * DBUS_MESSAGE_TYPE_ERROR -> "error" 4571 * DBUS_MESSAGE_TYPE_INVALID -> "invalid" 4572 * @endcode 4573 * 4574 */ 4575 const char * 4576 dbus_message_type_to_string (int type) 4577 { 4578 switch (type) 4579 { 4580 case DBUS_MESSAGE_TYPE_METHOD_CALL: 4581 return "method_call"; 4582 case DBUS_MESSAGE_TYPE_METHOD_RETURN: 4583 return "method_return"; 4584 case DBUS_MESSAGE_TYPE_SIGNAL: 4585 return "signal"; 4586 case DBUS_MESSAGE_TYPE_ERROR: 4587 return "error"; 4588 default: 4589 return "invalid"; 4590 } 4591 } 4592 4593 /** 4594 * Turn a DBusMessage into the marshalled form as described in the D-Bus 4595 * specification. 4596 * 4597 * Generally, this function is only useful for encapsulating D-Bus messages in 4598 * a different protocol. 4599 * 4600 * @param msg the DBusMessage 4601 * @param marshalled_data_p the location to save the marshalled form to 4602 * @param len_p the location to save the length of the marshalled form to 4603 * @returns #FALSE if there was not enough memory 4604 */ 4605 dbus_bool_t 4606 dbus_message_marshal (DBusMessage *msg, 4607 char **marshalled_data_p, 4608 int *len_p) 4609 { 4610 DBusString tmp; 4611 dbus_bool_t was_locked; 4612 4613 _dbus_return_val_if_fail (msg != NULL, FALSE); 4614 _dbus_return_val_if_fail (marshalled_data_p != NULL, FALSE); 4615 _dbus_return_val_if_fail (len_p != NULL, FALSE); 4616 4617 if (!_dbus_string_init (&tmp)) 4618 return FALSE; 4619 4620 /* Ensure the message is locked, to ensure the length header is filled in. */ 4621 was_locked = msg->locked; 4622 4623 if (!was_locked) 4624 dbus_message_lock (msg); 4625 4626 if (!_dbus_string_copy (&(msg->header.data), 0, &tmp, 0)) 4627 goto fail; 4628 4629 *len_p = _dbus_string_get_length (&tmp); 4630 4631 if (!_dbus_string_copy (&(msg->body), 0, &tmp, *len_p)) 4632 goto fail; 4633 4634 *len_p = _dbus_string_get_length (&tmp); 4635 4636 if (!_dbus_string_steal_data (&tmp, marshalled_data_p)) 4637 goto fail; 4638 4639 _dbus_string_free (&tmp); 4640 4641 if (!was_locked) 4642 msg->locked = FALSE; 4643 4644 return TRUE; 4645 4646 fail: 4647 _dbus_string_free (&tmp); 4648 4649 if (!was_locked) 4650 msg->locked = FALSE; 4651 4652 return FALSE; 4653 } 4654 4655 /** 4656 * Demarshal a D-Bus message from the format described in the D-Bus 4657 * specification. 4658 * 4659 * Generally, this function is only useful for encapsulating D-Bus messages in 4660 * a different protocol. 4661 * 4662 * @param str the marshalled DBusMessage 4663 * @param len the length of str 4664 * @param error the location to save errors to 4665 * @returns #NULL if there was an error 4666 */ 4667 DBusMessage * 4668 dbus_message_demarshal (const char *str, 4669 int len, 4670 DBusError *error) 4671 { 4672 DBusMessageLoader *loader; 4673 DBusString *buffer; 4674 DBusMessage *msg; 4675 4676 _dbus_return_val_if_fail (str != NULL, NULL); 4677 4678 loader = _dbus_message_loader_new (); 4679 4680 if (loader == NULL) 4681 return NULL; 4682 4683 _dbus_message_loader_get_buffer (loader, &buffer); 4684 _dbus_string_append_len (buffer, str, len); 4685 _dbus_message_loader_return_buffer (loader, buffer, len); 4686 4687 if (!_dbus_message_loader_queue_messages (loader)) 4688 goto fail_oom; 4689 4690 if (_dbus_message_loader_get_is_corrupted (loader)) 4691 goto fail_corrupt; 4692 4693 msg = _dbus_message_loader_pop_message (loader); 4694 4695 if (!msg) 4696 goto fail_oom; 4697 4698 _dbus_message_loader_unref (loader); 4699 return msg; 4700 4701 fail_corrupt: 4702 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS, "Message is corrupted (%s)", 4703 _dbus_validity_to_error_message (loader->corruption_reason)); 4704 _dbus_message_loader_unref (loader); 4705 return NULL; 4706 4707 fail_oom: 4708 _DBUS_SET_OOM (error); 4709 _dbus_message_loader_unref (loader); 4710 return NULL; 4711 } 4712 4713 /** 4714 * Returns the number of bytes required to be in the buffer to demarshal a 4715 * D-Bus message. 4716 * 4717 * Generally, this function is only useful for encapsulating D-Bus messages in 4718 * a different protocol. 4719 * 4720 * @param str data to be marshalled 4721 * @param len the length of str 4722 * @param error the location to save errors to 4723 * @returns -1 if there was no valid data to be demarshalled, 0 if there wasn't enough data to determine how much should be demarshalled. Otherwise returns the number of bytes to be demarshalled 4724 * 4725 */ 4726 int 4727 dbus_message_demarshal_bytes_needed(const char *buf, 4728 int len) 4729 { 4730 DBusString str; 4731 int byte_order, fields_array_len, header_len, body_len; 4732 DBusValidity validity = DBUS_VALID; 4733 int have_message; 4734 4735 if (!buf || len < DBUS_MINIMUM_HEADER_SIZE) 4736 return 0; 4737 4738 if (len > DBUS_MAXIMUM_MESSAGE_LENGTH) 4739 len = DBUS_MAXIMUM_MESSAGE_LENGTH; 4740 _dbus_string_init_const_len (&str, buf, len); 4741 4742 validity = DBUS_VALID; 4743 have_message 4744 = _dbus_header_have_message_untrusted(DBUS_MAXIMUM_MESSAGE_LENGTH, 4745 &validity, &byte_order, 4746 &fields_array_len, 4747 &header_len, 4748 &body_len, 4749 &str, 0, 4750 len); 4751 _dbus_string_free (&str); 4752 4753 if (validity == DBUS_VALID) 4754 { 4755 _dbus_assert (have_message || (header_len + body_len) > len); 4756 (void) have_message; /* unused unless asserting */ 4757 return header_len + body_len; 4758 } 4759 else 4760 { 4761 return -1; /* broken! */ 4762 } 4763 } 4764 4765 /** @} */ 4766 4767 /* tests in dbus-message-util.c */ 4768